在所有应用程序日志中启用 Golang Logrus Opentelemetry Trace ID 和 Span ID
来源:stackoverflow
2024-02-15 21:18:24
0浏览
收藏
本篇文章向大家介绍《在所有应用程序日志中启用 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删除

- 上一篇
- 使用Golang的time.Ticker实现滴答时钟

- 下一篇
- win11电脑如何启用安全启动?win11安全启动的设置指南
查看更多
最新文章
-
- Golang · Go问答 | 1年前 |
- 在读取缓冲通道中的内容之前退出
- 139浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 戈兰岛的全球 GOPRIVATE 设置
- 204浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何将结构作为参数传递给 xml-rpc
- 325浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何用golang获得小数点以下两位长度?
- 477浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何通过 client-go 和 golang 检索 Kubernetes 指标
- 486浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 将多个“参数”映射到单个可变参数的习惯用法
- 439浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 将 HTTP 响应正文写入文件后出现 EOF 错误
- 357浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 结构中映射的匿名列表的“复合文字中缺少类型”
- 352浏览 收藏
-
- Golang · Go问答 | 1年前 |
- NATS Jetstream 的性能
- 101浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何将复杂的字符串输入转换为mapstring?
- 440浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 相当于GoLang中Java将Object作为方法参数传递
- 212浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何确保所有 goroutine 在没有 time.Sleep 的情况下终止?
- 143浏览 收藏
查看更多
课程推荐
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
查看更多
AI推荐
-
- 美图AI抠图
- 美图AI抠图,依托CVPR 2024竞赛亚军技术,提供顶尖的图像处理解决方案。适用于证件照、商品、毛发等多场景,支持批量处理,3秒出图,零PS基础也能轻松操作,满足个人与商业需求。
- 13次使用
-
- PetGPT
- SEO摘要PetGPT 是一款基于 Python 和 PyQt 开发的智能桌面宠物程序,集成了 OpenAI 的 GPT 模型,提供上下文感知对话和主动聊天功能。用户可高度自定义宠物的外观和行为,支持插件热更新和二次开发。适用于需要陪伴和效率辅助的办公族、学生及 AI 技术爱好者。
- 14次使用
-
- 可图AI图片生成
- 探索快手旗下可灵AI2.0发布的可图AI2.0图像生成大模型,体验从文本生成图像、图像编辑到风格转绘的全链路创作。了解其技术突破、功能创新及在广告、影视、非遗等领域的应用,领先于Midjourney、DALL-E等竞品。
- 42次使用
-
- MeowTalk喵说
- MeowTalk喵说是一款由Akvelon公司开发的AI应用,通过分析猫咪的叫声,帮助主人理解猫咪的需求和情感。支持iOS和Android平台,提供个性化翻译、情感互动、趣味对话等功能,增进人猫之间的情感联系。
- 39次使用
-
- Traini
- SEO摘要Traini是一家专注于宠物健康教育的创新科技公司,利用先进的人工智能技术,提供宠物行为解读、个性化训练计划、在线课程、医疗辅助和个性化服务推荐等多功能服务。通过PEBI系统,Traini能够精准识别宠物狗的12种情绪状态,推动宠物与人类的智能互动,提升宠物生活质量。
- 36次使用
查看更多
相关文章
-
- GoLand调式动态执行代码
- 2023-01-13 502浏览
-
- 用Nginx反向代理部署go写的网站。
- 2023-01-17 502浏览
-
- Golang取得代码运行时间的问题
- 2023-02-24 501浏览
-
- 请问 go 代码如何实现在代码改动后不需要Ctrl+c,然后重新 go run *.go 文件?
- 2023-01-08 501浏览
-
- 如何从同一个 io.Reader 读取多次
- 2023-04-11 501浏览