通用包
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
utils/transport/v1/http/response.go

208 lines
5.4 KiB

package http
import (
"encoding/hex"
"fmt"
"gitea.drugeyes.vip/pharnexbase/utils/enum"
transPbV1 "gitea.drugeyes.vip/pharnexbase/utils/transport/v1"
"github.com/go-kratos/kratos/v2/errors"
"github.com/go-kratos/kratos/v2/transport/http"
"github.com/tjfoc/gmsm/sm4"
stdhttp "net/http"
"strings"
)
const (
baseContentType = "application"
)
var ErrInternalServer = errors.InternalServer("InternalServerError", "服务错误")
type Response struct {
Code int `json:"code" form:"code"`
Data interface{} `json:"data" form:"data"`
Reason string `json:"reason" form:"reason"`
Message string `json:"message" form:"message"`
}
func (t *Transport) ResponseEncoder() http.EncodeResponseFunc {
return func(w stdhttp.ResponseWriter, r *stdhttp.Request, v interface{}) error {
if v == nil {
return nil
}
// 重定向
if rd, ok := v.(http.Redirector); ok {
url, code := rd.Redirect()
stdhttp.Redirect(w, r, url, code)
return nil
}
// 文件下载
if data, ok := v.(*transPbV1.FileResponse); ok {
w.Header().Set("Content-Disposition", fmt.Sprintf("attachment; filename=\"%s\"", data.Filename))
if data.Filetype != "" {
w.Header().Set("Content-Type", data.Filetype)
}
w.Header().Set("Access-Control-Expose-Headers", "*")
w.WriteHeader(stdhttp.StatusOK)
w.Write(data.Data)
return nil
}
codec, _ := http.CodecForRequest(r, "Accept")
data, err := codec.Marshal(v)
if err != nil {
return err
}
w.Header().Set("Content-Type", ContentType(codec.Name()))
w.Header().Set("Access-Control-Expose-Headers", "Traceid, En ,Content-Disposition, Authorization")
d := fmt.Sprintf(`{"code": 200,"data": %s,"reason": "","message": ""}`, string(data))
_, err = w.Write([]byte(d))
if err != nil {
return err
}
return nil
}
}
func (t *Transport) ResponseEncoderWithEncrypt() http.EncodeResponseFunc {
return func(w stdhttp.ResponseWriter, r *stdhttp.Request, v interface{}) error {
if v == nil {
return nil
}
// 重定向
if rd, ok := v.(http.Redirector); ok {
url, code := rd.Redirect()
stdhttp.Redirect(w, r, url, code)
return nil
}
// 文件下载
if data, ok := v.(*transPbV1.FileResponse); ok {
w.Header().Set("Content-Disposition", fmt.Sprintf("attachment; filename=\"%s\"", data.Filename))
if data.Filetype != "" {
w.Header().Set("Content-Type", data.Filetype)
}
w.Header().Set("Access-Control-Expose-Headers", "*")
w.WriteHeader(stdhttp.StatusOK)
w.Write(data.Data)
return nil
}
codec, _ := http.CodecForRequest(r, "Accept")
data, err := codec.Marshal(v)
if err != nil {
return err
}
dataRes := string(data)
// 正式服加密
if t.env == enum.Env_production {
ecbMsg, err := sm4.Sm4Ecb([]byte(t.sm4Key), data, true)
if err == nil {
dataRes = fmt.Sprintf("\"%s\"", hex.EncodeToString(ecbMsg))
w.Header().Set("en", "4")
}
}
w.Header().Set("Content-Type", ContentType(codec.Name()))
w.Header().Set("Access-Control-Expose-Headers", "Traceid, En ,Content-Disposition, Authorization")
d := fmt.Sprintf(`{"code": 200,"data": %s,"reason": "","message": ""}`, dataRes)
_, err = w.Write([]byte(d))
if err != nil {
return err
}
return nil
}
}
func (t *Transport) ResponseEncoderWithEncryptFilter(noEncrypt map[string]bool) http.EncodeResponseFunc {
return func(w stdhttp.ResponseWriter, r *stdhttp.Request, v interface{}) error {
if v == nil {
return nil
}
// 重定向
if rd, ok := v.(http.Redirector); ok {
url, code := rd.Redirect()
stdhttp.Redirect(w, r, url, code)
return nil
}
// 文件下载
if data, ok := v.(*transPbV1.FileResponse); ok {
w.Header().Set("Content-Disposition", fmt.Sprintf("attachment; filename=\"%s\"", data.Filename))
if data.Filetype != "" {
w.Header().Set("Content-Type", data.Filetype)
}
w.Header().Set("Access-Control-Expose-Headers", "*")
w.WriteHeader(stdhttp.StatusOK)
w.Write(data.Data)
return nil
}
codec, _ := http.CodecForRequest(r, "Accept")
data, err := codec.Marshal(v)
if err != nil {
return err
}
dataRes := string(data)
// 正式服加密
if t.env == enum.Env_production && !noEncrypt[r.URL.Path] {
ecbMsg, err := sm4.Sm4Ecb([]byte(t.sm4Key), data, true)
if err == nil {
dataRes = fmt.Sprintf("\"%s\"", hex.EncodeToString(ecbMsg))
w.Header().Set("en", "4")
}
}
w.Header().Set("Content-Type", ContentType(codec.Name()))
w.Header().Set("Access-Control-Expose-Headers", "Traceid, En ,Content-Disposition, Authorization")
d := fmt.Sprintf(`{"code": 200,"data": %s,"reason": "","message": ""}`, dataRes)
_, err = w.Write([]byte(d))
if err != nil {
return err
}
return nil
}
}
func (t *Transport) ErrorEncoder() http.EncodeErrorFunc {
return func(w stdhttp.ResponseWriter, r *stdhttp.Request, err error) {
se := errors.FromError(err)
if t.env == enum.Env_production {
if _, ok := err.(*errors.Error); !ok {
se = ErrInternalServer
}
}
codec, _ := http.CodecForRequest(r, "Accept")
body, err := codec.Marshal(se)
if err != nil {
w.WriteHeader(stdhttp.StatusInternalServerError)
return
}
w.Header().Set("Content-Type", ContentType(codec.Name()))
w.Header().Set("Access-Control-Expose-Headers", "TraceID, En ,Content-Disposition, Authorization")
w.WriteHeader(int(se.Code))
_, _ = w.Write(body)
}
}
// ContentType returns the content-type with base prefix.
func ContentType(subtype string) string {
return strings.Join([]string{baseContentType, subtype}, "/")
}