我正在编写自己的logginMiddleware。基本上,我需要记录请求和响应的正文。我面临的问题是,当我阅读正文时,它变成空的,无法两次阅读。我知道发生这种情况是因为它属于ReadCloser类型。有没有一种方法可以将身体重绕到开始?
初次阅读主体时,必须对其进行存储,以便在处理完该主体之后,可以将新io.ReadCloser
的主体设置为根据原始数据构造的请求主体。因此,当您在链中前进时,下一个处理程序可以读取同一正文。
一种选择是使用读取整个正文ioutil.ReadAll()
,这会将您作为正文作为字节片。
您可以用来从字节片中bytes.NewBuffer()
获取一个io.Reader
。
最后缺少的是使io.Reader
an
io.ReadCloser
,因为bytes.Buffer
没有Close()
方法。为此,您可以使用ioutil.NopCloser()
包裹io.Reader
,然后返回io.ReadCloser
,其添加的Close()
方法将是no-
op(不执行任何操作)。
请注意,您甚至可以修改用于创建“新”正文的字节片的内容。您完全可以控制它。
但是,必须小心,因为如果仅修改数据,可能还有其他HTTP字段,例如content-
length和checksum,它们可能变得无效。如果后续处理程序检查了这些内容,那么您也需要对其进行修改!
如果您还想读取响应主体,则必须包装http.ResponseWriter
您获得的包装,并将包装器传递给链。该包装器可能会缓存发送出的数据,您可以在运行中即时检查(随后的处理程序将其写入)。
这是一个简单的ResponseWriter
包装程序,它仅缓存数据,因此在后续处理程序返回后将可用:
type MyResponseWriter struct {
http.ResponseWriter
buf *bytes.Buffer
}
func (mrw *MyResponseWriter) Write(p []byte) (int, error) {
return mrw.buf.Write(p)
}
请注意,MyResponseWriter.Write()
只是将数据写入缓冲区。您也可以选择动态检查(在Write()
方法中)并将其立即写入包装/嵌入的数据中ResponseWriter
。您甚至可以修改数据。您拥有完全的控制权。
但是,必须再次小心,因为后续处理程序可能还会发送与响应数据相关的HTTP响应标头(例如长度或校验和),如果您更改响应数据,它们也可能变得无效。
将各个部分放在一起,这是一个完整的工作示例:
func loginmw(handler http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
body, err := ioutil.ReadAll(r.Body)
if err != nil {
log.Printf("Error reading body: %v", err)
http.Error(w, "can't read body", http.StatusBadRequest)
return
}
// Work / inspect body. You may even modify it!
// And now set a new body, which will simulate the same data we read:
r.Body = ioutil.NopCloser(bytes.NewBuffer(body))
// Create a response wrapper:
mrw := &MyResponseWriter{
ResponseWriter: w,
buf: &bytes.Buffer{},
}
// Call next handler, passing the response wrapper:
handler.ServeHTTP(mrw, r)
// Now inspect response, and finally send it out:
// (You can also modify it before sending it out!)
if _, err := io.Copy(w, mrw.buf); err != nil {
log.Printf("Failed to send out response: %v", err)
}
})
}
问题内容: type ValidationModel struct { Name string Email string Password string } 首先,我使用govalidator验证请求正文。 在验证了请求之后,我再次将请求主体解码为用户结构,但已使用validationModel读取了请求主体一次,因此当我尝试再次将其解码为用户时,它没有提供任何值。 我在这里可以想到两种解决方案:
Go Env: goarch=“AMD64” goroot=“/usr/local/go” goTooldir=“/usr/local/go/pkg/tool/linux_amd64” go15vendorexperiment=“1” go版本GO1.6.3 Linux/AMD64 这个问题发生在高负载的“性能测试环境”库贝-APIServer上。库贝-Apiserver恐慌和退出: 相应源代码:
问题内容: 我正在构建Blob存储系统,我选择了Go作为编程语言。我创建了一个流,以执行从客户端到Blob服务器的分段文件上传。 流工作正常,但我想从请求正文中进行sha1哈希处理。我需要io。复制身体两次。创建了sha1,但是多部分流之后0字节。 用于创建哈希 用于将身体分为多个部分 任何想法我该怎么做? 客户上传 sha1函数 } 上传句柄 } 问题答案: 您不能直接执行此操作,但可以编写在i
问题内容: 我正在尝试将JSON参数发送到我的服务器,并使用json.Decoder解析它们。我读到您应该能够从request.Body属性获取查询参数。以下是我的服务器代码: 每次,我都会看到(当然是不同的时间戳记)。我的客户端AJAX调用如下: 发送内容的示例URL: 更好看一下参数: 我已经尝试了GET和POST请求。 为什么我的req.Body从不解码?如果我尝试单独打印req.Body,
我有一个静态网站托管在AWS CloudFront上。在一条路由上,我需要接受POST方法,因为它是OAuth服务器的重定向,所以我决定开发一个lambda@edge。 我的想法是在'Viewer Request'上注册lambda并截取POST方法,读取正文并复制标头上的值,以使它们在我的静态网站上可读(我知道我可以用javascript访问Referrer标头)。 我设置了Lambda,我可以
PS Vita的搭载机能可能因机型而异。详细请参阅随附的使用说明书。 使用PS Vita及其功能时,需适用各种使用承诺条款。这些使用承诺条款可能随时更改。请阅览各区域的官方网站,确认最新的条款内容以及使用说明书、用户指南等的规定使用条件之改定内容。 若要使用PlayStation®Network的功能,系统软件需随时升级为最新版本。 PlayStation®Network和PlayStation®