我正在使用Github的grpc-gateway项目从gRPC规范自动生成REST API。在这个REST API中,我希望支持ETag头和304未修改的响应。
据我所知,通常您会在gRPC服务器中创建一个带有特定状态代码的响应,然后该状态代码将被grpc-gateway转换为一个HTTP状态代码。但是,由于标准gRPC并不真正支持缓存概念,因此没有映射到HTTP304状态代码的gRPC状态代码。
使用grpc-gateway,似乎可以在gRPC状态码为en错误码(覆盖runtime.HTTPError函数)时自定义HTTP状态码。然而,当gRPC响应代码正常时,我还没有找到任何定制HTTP响应代码的方法。
那么,有什么推荐的方法来实现这一点吗?
发布于 2018-02-07 12:21:01
下面是一个使用自定义转发器实现基本etag和304响应的示例。
您可以参考这些directions来进行设置,然后按照以下方式实现该方法:
func forwardGetPost(ctx context.Context, mux *runtime.ServeMux, marshaler runtime.Marshaler, w http.ResponseWriter, req *http.Request, resp proto.Message, opts ...func(context.Context, http.ResponseWriter, proto.Message) error) {
// add cache-control rules for this proxy endpoint
w.Header().Add("Cache-Control", "max-age=60")
// create an etag
// (when the response represents some entity from the db it may have a last edited timestamp)
p := resp.(*Post)
etag := fmt.Sprintf("W/%q", p.GetLastEdited())
w.Header().Add("ETag", etag)
// check whether the request provides an etag
inm := req.Header.Get("If-None-Match")
if inm != "" {
if inm == etag {
w.WriteHeader(http.StatusNotModified)
}
}
runtime.ForwardResponseMessage(ctx, mux, marshaler, w, req, resp, opts...)
}这不会阻止代理向grpc服务器发出请求,但当etags匹配时,它会阻止将这些字节发送回客户端。
https://stackoverflow.com/questions/46608167
复制相似问题