在所有应用程序日志中启用 Golang Logrus Opentelemetry Trace ID 和 Span ID
来源:stackoverflow
时间:2024-02-15 21:18:24 244浏览 收藏
本篇文章向大家介绍《在所有应用程序日志中启用 Golang Logrus Opentelemetry Trace ID 和 Span ID》,主要包括,具有一定的参考价值,需要的朋友可以参考一下。
问题内容
我正在尝试为每个应用程序日志启用跟踪 id 和跨度 id。我正在使用 opentelemtry api 进行注入,并使用 logrus labrary 进行日志记录。下面是一个示例代码,我可以在其中启用所有日志中输入的跟踪/跨度,但跟踪和跨度 id 值始终为 0。
#server.go#
package main
import (
log "github.com/sirupsen/logrus"
"fmt"
"context"
"os"
"net/http"
"go.opentelemetry.io/otel/trace"
stdout "go.opentelemetry.io/otel/exporters/stdout/stdouttrace"
sdktrace "go.opentelemetry.io/otel/sdk/trace"
"go.opentelemetry.io/otel/sdk/resource"
semconv "go.opentelemetry.io/otel/semconv/v1.10.0"
"go.opentelemetry.io/otel"
"go.opentelemetry.io/contrib/propagators/b3"
"go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp"
)
func init() {
log.setformatter(customlogger{
formatter: log.jsonformatter{fieldmap: log.fieldmap{
"msg": "message",
}},
})
log.setoutput(os.stdout)
log.setlevel(log.infolevel)
}
func inittracer() (*sdktrace.tracerprovider, error) {
exporter, err := stdout.new(stdout.withprettyprint())
if err != nil {
return nil, err
}
tp := sdktrace.newtracerprovider(
sdktrace.withsampler(sdktrace.alwayssample()),
sdktrace.withbatcher(exporter),
sdktrace.withresource(resource.newwithattributes(semconv.schemaurl, semconv.servicenamekey.string("preferencesservice"))),
)
otel.settracerprovider(tp)
b3 := b3.new()
otel.settextmappropagator(b3)
return tp, err
}
func main() {
ctx := context.background()
tp, err := inittracer()
if err != nil {
log.error(err)
}
defer func() {
if err := tp.shutdown(ctx); err != nil {
log.error("error shutting down tracer provider: %v", err)
}
}()
otelhandler := otelhttp.newhandler(http.handlerfunc(func(w http.responsewriter, _ *http.request) {
log.info("about end point printning...")
fmt.fprintln(w, "about page")
}),"about")
http.handle("/about", otelhandler)
log.info("listening...")
log.fatal(http.listenandserve(":8080", nil))
}
type customlogger struct {
formatter log.jsonformatter
}
func (l customlogger) format(entry *log.entry) ([]byte, error) {
span := trace.spanfromcontext(entry.context)
entry.data["trace_id"] = span.spancontext().traceid().string()
entry.data["span_id"] = span.spancontext().spanid().string()
//below injection is just to understand what context has
entry.data["context"] = span.spancontext()
return l.formatter.format(entry)
}
#client.go#
package main
import (
"fmt"
"io/ioutil"
"log"
"net/http"
)
func main() {
resp, err := http.get("http://localhost:8080/about")
if err != nil {
log.fatal(err)
}
defer resp.body.close()
body, err := ioutil.readall(resp.body)
if err != nil {
log.fatal(err)
}
fmt.println(string(body))
}
#输出:#
{"Context":{"TraceID":"00000000000000000000000000000000","SpanID":"0000000000000000","TraceFlags":"00","TraceState":"","Remote":false},"level":"info","message":"Listening...","span_id":"0000000000000000","time":"2022-06-30T12:34:21+05:30","trace_id":"00000000000000000000000000000000"}
{"Context":
**Below Line Should have Proper Trace ID and Span ID**
{"TraceID":"00000000000000000000000000000000","SpanID":"0000000000000000","TraceFlags":"00","TraceState":"","Remote":false},"level":"info","message":"About End Point Printning...","span_id":"0000000000000000","time":"2022-06-30T12:34:28+05:30","trace_id":"00000000000000000000000000000000"}
{
"Name": "About",
"SpanContext": {
"TraceID": "cb4dc4953a633fbe9b41c19a0870e9d6",
"SpanID": "b7346b7aba15e074",
"TraceFlags": "01",
"TraceState": "",
"Remote": false
},
"Parent": {
"TraceID": "00000000000000000000000000000000",
"SpanID": "0000000000000000",
"TraceFlags": "00",
"TraceState": "",
"Remote": false
},
"SpanKind": 2,
"StartTime": "2022-06-30T12:34:28.838523497+05:30",
"EndTime": "2022-06-30T12:34:28.838858989+05:30",
"Attributes": [
{
"Key": "net.transport",
"Value": {
"Type": "STRING",
"Value": "ip_tcp"
}
},
{
"Key": "net.peer.ip",
"Value": {
"Type": "STRING",
"Value": "127.0.0.1"
}
},
{
"Key": "net.peer.port",
"Value": {
"Type": "INT64",
"Value": 41900
}
},
{
"Key": "net.host.name",
"Value": {
"Type": "STRING",
"Value": "localhost"
}
},
{
"Key": "net.host.port",
"Value": {
"Type": "INT64",
"Value": 8080
}
},
{
"Key": "http.method",
"Value": {
"Type": "STRING",
"Value": "GET"
}
},
{
"Key": "http.target",
"Value": {
"Type": "STRING",
"Value": "/about"
}
},
{
"Key": "http.server_name",
"Value": {
"Type": "STRING",
"Value": "About"
}
},
{
"Key": "http.user_agent",
"Value": {
"Type": "STRING",
"Value": "Go-http-client/1.1"
}
},
{
"Key": "http.scheme",
"Value": {
"Type": "STRING",
"Value": "http"
}
},
{
"Key": "http.host",
"Value": {
"Type": "STRING",
"Value": "localhost:8080"
}
},
{
"Key": "http.flavor",
"Value": {
"Type": "STRING",
"Value": "1.1"
}
},
{
"Key": "http.wrote_bytes",
"Value": {
"Type": "INT64",
"Value": 11
}
},
{
"Key": "http.status_code",
"Value": {
"Type": "INT64",
"Value": 200
}
}
],
"Events": null,
"Links": null,
"Status": {
"Code": "Unset",
"Description": ""
},
"DroppedAttributes": 0,
"DroppedEvents": 0,
"DroppedLinks": 0,
"ChildSpanCount": 0,
"Resource": [
{
"Key": "service.name",
"Value": {
"Type": "STRING",
"Value": "PreferencesService"
}
}
],
"InstrumentationLibrary": {
"Name": "go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp",
"Version": "semver:0.32.0",
"SchemaURL": ""
}
}
不确定错误在哪里/什么,但如果有人将我引导到正确的方向,我如何将跟踪 id 和跨度 id 注入到所有日志中,这将会很有帮助。
注意:我不想将请求上下文传递给所有函数。寻找其他选择
正确答案
问题是 span := trace.spanfromcontext(entry.context) 无法获取 span 上下文。
也许你应该将 req.context() 注入 logrus 然后打印日志
otelhandler := otelhttp.newhandler(http.handlerfunc(func(w http.responsewriter, req *http.request) {
//log.info("about end point printning...")
// inject req.context
log.withcontext(req.context()).info("about end point printning...")
fmt.fprintln(w, "about page")
}), "about")
运行server.go,然后运行client.go,输出为
{"Context":{"TraceID":"00000000000000000000000000000000","SpanID":"0000000000000000","TraceFlags":"00","TraceState":"","Remote":false},"level":"info","message":"Listening...","span_id":"0000000000000000","time":"2022-07-02T21:39:49+08:00","trace_id":"00000000000000000000000000000000"}
{"Context":{"TraceID":"7cab2739ea2c5b2d3e6a08cfb57858a9","SpanID":"ec5ba151b6823658","TraceFlags":"01","TraceState":"","Remote":false},"level":"info","message":"About End Point Printning...","span_id":"ec5ba151b6823658","time":"2022-07-02T21:39:54+08:00","trace_id":"7cab2739ea2c5b2d3e6a08cfb57858a9"}
注意:第一行不打印traceid是因为没有请求
希望对您有帮助。
终于介绍完啦!小伙伴们,这篇关于《在所有应用程序日志中启用 Golang Logrus Opentelemetry Trace ID 和 Span ID》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!
声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
最新阅读
更多>
-
139 收藏
-
204 收藏
-
325 收藏
-
478 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
课程推荐
更多>
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习