当前位置: 首页 > 工具软件 > Kratos > 使用案例 >

kratos的返回值问题与错误返回问题

萧晔
2023-12-01

习惯了这种格式的返回值:

{
	"code":"0",
	"message":"成功"
    "captchaId": "0TFqCFHqeGuvYFy3EajN",
    ...
}

今天按照kratos写api发现返回的是这种

{
    "captchaId": "0TFqCFHqeGuvYFy3EajN",
    "captcha": ""
}

可以在http中添加方法,统一添加业务成功代码

package server

import (
	"github.com/go-kratos/kratos/v2/encoding"
	"github.com/go-kratos/kratos/v2/log"
	"github.com/go-kratos/kratos/v2/middleware/logging"
	mmd "github.com/go-kratos/kratos/v2/middleware/metadata"
	"github.com/go-kratos/kratos/v2/middleware/metrics"
	"github.com/go-kratos/kratos/v2/middleware/recovery"
	"github.com/go-kratos/kratos/v2/middleware/validate"
	"github.com/go-kratos/kratos/v2/transport/http"
	"github.com/go-kratos/swagger-api/openapiv2"
	baseHttp "kratosadmin/api/base/v1"
	userHttp "kratosadmin/api/user/v1"
	"kratosadmin/app/user/service/internal/conf"
	"kratosadmin/app/user/service/internal/service"
	"kratosadmin/app/user/service/middleware"
	httpNet "net/http"
	"time"
)

// NewHTTPServer new a HTTP server.
func NewHTTPServer(c *conf.Server, userService *service.UserService, baseService *service.BaseService, logger log.Logger, middleware2 middleware.Middleware) *http.Server {
	var opts = []http.ServerOption{
		http.Middleware(
			recovery.Recovery(),
			logging.Server(logger),
			metrics.Server(),
			validate.Validator(),
			mmd.Server(),
			middleware2.Auth(),
		),
		http.Filter(middleware2.Cors()),
	}
	if c.Http.Network != "" {
		opts = append(opts, http.Network(c.Http.Network))
	}
	if c.Http.Addr != "" {
		opts = append(opts, http.Address(c.Http.Addr))
	}

	if c.Http.Timeout != nil {
		opts = append(opts, http.Timeout(c.Http.Timeout.AsDuration()))
	}
	opts = append(opts, http.ResponseEncoder(func(
		w httpNet.ResponseWriter,
		r *httpNet.Request,
		i interface{},
	) error {
		type response struct {
			Code int
			Data interface{}
			Ts   string
		}
		reply := &response{
			Code: 200,
			Data: i,
			Ts:   time.Now().String(),
		}
		codec := encoding.GetCodec("json")
		data, err := codec.Marshal(reply)
		if err != nil {
			return err
		}
		w.Header().Set("Content-Type", "application/json")
		w.Write(data)
		return nil
	}))

	srv := http.NewServer(opts...)

	//user服务
	userHttp.RegisterUserHTTPServer(srv, userService)
	//基础服务
	baseHttp.RegisterBaseHTTPServer(srv, baseService)
	openAPIhandler := openapiv2.NewHandler()
	srv.HandlePrefix("/q/", openAPIhandler)
	return srv
}

返回值就变成了下面的了,复合要求
资料来源:https://mp.weixin.qq.com/s/4ocdoAVXXKTvJ3U65YXltw

{
    "Code": 200,
    "Data": {
        "captchaId": "NYAa3oSuojuvzBFvCX7u",
        "captcha": ""
    },
    "Ts": "2021-09-22 14:48:21.9387671 +0800 CST m=+0.239897501"
}

然后,貌似还有一种哲学,既然正常返回http状态吗为200,那么返回体内容就应该是业务数据不应该再加载业务code/message等,所以kratos默认返回的符合这个逻辑,那么错误消息怎么返回呢?比如400错误、500错误呢,为什么这么设计,业务code与http code一致呢?可以方便运维抓取接口状态数据吧
链接地址:https://go-kratos.dev/docs/component/errors

可以这样自定义错误

// 通过 errors.New() 响应错误
	err = errors.New(500, "USER_NAME_EMPTY", "user name is empty")

也可以这样:

// 通过 proto 生成的代码响应错误,并且包名应替换为自己生成代码后的 package name
	err = v1.ErrorUsernameConflict("user %s not found", "kratos")

覆盖返回错误码即可,比如这样返回结果

{
    "code": 409,
    "reason": "USERNAME_CONFLICT",
    "message": "user kratos not found",
    "metadata": {}
}
 类似资料: