在中间件中无法追踪 HTTP 响应代码
本篇文章主要是结合我之前面试的各种经历和实战开发中遇到的问题解决经验整理的,希望这篇《在中间件中无法追踪 HTTP 响应代码》对你有很大帮助!欢迎收藏,分享给更多的需要的朋友学习~
我正在使用 go-openapi 从 swagger 配置生成一个 http 服务器并处理所有处理程序。
我的中间件的架构是 request -> override func -> do http stuff -> 日志响应代码 -> response。
这是我的中间件:
func (pm *prometheusmetrics) httpmiddleware(h http.handler) http.handler {
return http.handlerfunc(func(w http.responsewriter, r *http.request) {
start := time.now()
statusrecoder := &utils.statusrecorder{responsewriter: w, statuscode: 200}
logrus.infof("--> %v %v", r.method, r.url.path)
// statusrecoder := negroni.newresponsewriter(w) <-- tried this solution, didn't worked
h.servehttp(statusrecoder, r)
logrus.infof("<-- %v", statusrecoder.status()) // should display status code
duration := time.since(start)
statuscode := strconv.itoa(statusrecoder.status())
pm.pushhttpmetrics(r.url.path, statuscode, duration.seconds())
})
}
这是我传递中间件的方式:
func setupglobalmiddleware(handler http.handler, prommetrics *apihandlers.prometheusmetrics) http.handler {
middle := interpose.new()
middle.usehandler(handler)
recoverymiddleware := recovr.new()
middle.use(recoverymiddleware)
logrusmiddleware := interposemiddleware.negronilogrus()
middle.use(logrusmiddleware)
middle.use(func(h http.handler) http.handler {
return prommetrics.httpmiddleware(h)
})
corsmiddleware := cors.new(cors.options{
allowedheaders: []string{"*"},
allowedorigins: []string{"*"},
allowedmethods: []string{"get", "post", "put", "patch", "delete", "options"},
maxage: 1000,
optionspassthrough: false,
})
if log.getlevel() >= log.debuglevel {
corsmiddleware.log = log.standardlogger()
}
return corsmiddleware.handler(middle)
}
最后,我调用 setupglobalmiddleware 函数:
func configureapi(api *operations.kubeesapi) http.handler { // operations is package generated by go-openapi
// some setup...
return setupglobalmiddleware(api.serve(setupmiddlewares), prommetrics)
}
statusrecoder 的重写:
type StatusRecorder struct {
http.ResponseWriter
StatusCode int
}
// WriteHeader is the fake function to record status code
func (rec *StatusRecorder) WriteHeader(statusCode int) {
logrus.Infof("hello there: %v", statusCode)
rec.StatusCode = statusCode
rec.ResponseWriter.WriteHeader(statusCode)
}
如您所见,我重写了 http.handler 中的 writeheader 函数,以仅存储状态代码并在外部访问它,然后调用原始 writeheader 函数。这里的问题是我的函数 writeheader 从未被调用。并且无法弄清楚为什么不是。
解决方案
如果您没有完整的软件包列表,那么答案会更难理解。
问题在于 go-openapi 和 go-swagger 在代码中实现中间件并处理它们的方式。许多其他已安装的软件包也存在此问题。
为了确保正确检索状态代码,我将中间件放在中间件声明的右底部。
代码:
func setupglobalmiddleware(handler http.handler, prommetrics *apihandlers.prometheusmetrics) http.handler {
middle := interpose.new()
middle.usehandler(handler)
recoverymiddleware := recovr.new()
middle.use(recoverymiddleware)
logrusmiddleware := interposemiddleware.negronilogrus()
middle.use(logrusmiddleware)
// the old declaration was here
corsmiddleware := cors.new(cors.options{
allowedheaders: []string{"*"},
allowedorigins: []string{"*"},
allowedmethods: []string{"get", "post", "put", "patch", "delete", "options"},
maxage: 1000,
optionspassthrough: false,
})
if log.getlevel() >= log.debuglevel {
corsmiddleware.log = log.standardlogger()
}
// notice the changement where the prommetrics.httpmiddleware is now in the bottom
return corsmiddleware.handler(prommetrics.httpmiddleware(middle))
// return corsmiddleware.handler(middle) <-- old line
}
因为go使用组合,而不是继承。 writeheader 最常在第一次调用 write() 时自动调用。来自http.ResponseWriter documentation:
但由于您尚未定义自己的 write 方法,因此将调用嵌入式方法(由 http.responsewriter 的嵌入式实例提供)。当调用 write() 方法,然后调用 writeheader 时,它会调用原始 writeheader 方法,而不是您的版本。
解决方案是还提供您自己的 write 方法,该方法包装原始方法,并在需要时调用您的 writeheader 版本。
type StatusRecorder struct {
http.ResponseWriter
StatusCode int
written bool
}
// WriteHeader is the fake function to record status code
func (rec *StatusRecorder) WriteHeader(statusCode int) {
rec.written = true
logrus.Infof("hello there: %v", statusCode)
rec.StatusCode = statusCode
rec.ResponseWriter.WriteHeader(statusCode)
}
func (rec *StatusRecorder) Write(p []byte) (int, error) {method
if !rec.written {
rec.WriteHeader(http.StatusOK)
}
return rec.Write(p)
}
以上就是《在中间件中无法追踪 HTTP 响应代码》的详细内容,更多关于的资料请关注golang学习网公众号!
golang中的嵌套结构不会被编码为http响应
- 上一篇
- golang中的嵌套结构不会被编码为http响应
- 下一篇
- 在Go中使用递归和迭代处理结构体数组
-
- Golang · Go问答 | 3个月前 | go atomic原理 Go并发安全
- Go语言中atomic包如何保证并发安全?
- 109浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ljg-skills
- ljg-skills 是李继刚开源的 AI 技能与提示词集合,面向大模型使用者整理了一批可复用的 prompt、角色设定和任务技能模板,适合用于学习提示词设计、搭建个人 AI 工作流和沉淀团队常用智能体能力。
- 1460次使用
-
- MELO音乐
- MELO音乐是一站式AI视频与音乐制作助手,对标suno, udio的高品质体验。提供伴奏生成、原创写词、无损导出、哼唱识曲、混音变声等全套音频与短视频编辑工具。无论是流行Kpop、电音说唱、民谣古风、摇滚儿歌还是商用轻音乐,MELO为你免费谱曲,轻松做同款!
- 1398次使用
-
- UniScribe
- UniScribe 是一款 AI 音视频转文字与内容整理工具,支持上传音频、视频文件或粘贴 YouTube 链接,自动生成转写文本、摘要、思维导图和关键问题,并支持多格式导出,适合会议记录、课程学习、访谈整理和内容创作复盘。
- 1355次使用
-
- 剧云
- 剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
- 1537次使用
-
- 万象有声
- 万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
- 1517次使用
-
- 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浏览

