大家好,我是peachesTao,今天给大家推荐一个go的支持多语言的error自动生成的插件,插件主页可以访问下方链接。
在一个多语言国际化的项目中,后端接口返回给前端的错误描述也需要国际化,我们来看一下后端给前端返回多语言错误描述的实现方式有哪些。
服务端将错误码和不同语言的错误描述硬编码在代码中,通过前端从http head中传过来的language来决定是返回中文还是英文。
该结构体实现标准库的error接口,实现自定义error
type Error struct {
Code int
Msg string
}
func (e *Error) Error() string {
return fmt.Sprintf("%d,%s", e.Code, e.Msg)
}
const (
Err_Code_Success = 0
Err_Code_UnKnown = -1
Err_Code_InValid_Phone = 10001
)
const (
Language_Chinese = 0 //中文
Language_Enligh = 1 //英文
)
//不同语言对应的错误描述
var errMap = map[int]map[int]string{
Language_Chinese: {
Err_Code_Success: "成功",
Err_Code_InValid_Phone: "手机号格式不正确",
Err_Code_UnKnown: "未知错误",
},
Language_Enligh: {
Err_Code_Success: "success",
Err_Code_InValid_Phone: "invalid phone no",
Err_Code_UnKnown: "unknown err",
},
}
根据客户端传过来的http header中的language的值决定返回中文还是英文的错误描述
func main() {
http.HandleFunc("/user/register", func(w http.ResponseWriter, r *http.Request) {
languageStr := r.Header.Get("language")
language, _ := strconv.Atoi(languageStr)
values, _ := url.ParseQuery(r.URL.RawQuery)
phone := values["phone"][0]
err := checkPhone(phone)
response(w, language, err)
})
http.ListenAndServe(":8080", nil)
}
func response(w http.ResponseWriter, language int, err error) {
e := &Error{Code: Err_Code_Success}
if err != nil {
var ok bool
if e, ok = err.(*Error); !ok {
e = &Error{Code: Err_Code_UnKnown}
}
}
msg := errMap[language][e.Code]
res := make(map[string]interface{})
res["code"] = e.Code
res["msg"] = msg
json, _ := json.Marshal(res)
w.WriteHeader(200)
w.Write(json)
}
func checkPhone(phoneNo string) error {
if len(phoneNo) != 11 {
return &Error{Code: Err_Code_InValid_Phone}
}
return nil
}
我们通过curl命令来看看效果
语言设置为中文时:
curl -H "language:0" "http://127.0.0.1:8080/user/register?phone=187111111112"
{"code":10001,"msg":"手机号格式不正确"}
语言设置为英文时:
curl -H "language:1" "http://127.0.0.1:8080/user/register?phone=187111111112"
{"code":10001,"msg":"invalid phone no"}
这种实现方式确实能满足业务需求,但是有下面几个缺点:
一旦涉及到修改代码就存在出现bug的风险,有没有一种更优雅的方案,尽量减少修改代码?
有人会想到将错误描述放在json文件中维护,这种方案只是在修改错误描述时比较便利,不需要改动业务代码,但在新增错误和新语言时存在同样的问题。
下面我们来看看通过go-error-generator插件的方法来实现
go-error-generator是一个通过protobuf文件的Enum对象自动生成Error的插件,通过在扩展的EnumValueOptions中定义多个option轻松实现error的多语言。
它包含如下功能:
关于插件的原理和其他细节可以访问github主页了解。
我们回到刚才那个需求,用插件的方式怎么实现错误多语言
删除代码中的的Error结构体,取代的是在protobuf中定义,新建一个protobuf文件,取名为error.proto,在这里自定义error结构体和语言标识。
其中:
syntax = "proto3";
package errors;
option go_package = "github.com/classtorch/go-error-generator-examples/internal/errors";
import "google/protobuf/descriptor.proto";
message Error {
int32 code = 1;
string msg = 2;
};
extend google.protobuf.EnumValueOptions {
string msg = 1108;
string msg_english = 1109;
}
新建一个protobuf文件,取名为account.proto
导入上面定义好的error.proto,自定义msg和msg_english对应的错误描述
syntax = "proto3";
package uclass.service.account;
option go_package = "/golang/account";
import "errors/errors.proto";
enum ErrorCode {
SUCCESS = 0 [(errors.msg) = "成功", (errors.msg_english) = "success"]; // 成功
UnKnown = -1 [(errors.msg) = "未知错误", (errors.msg_english) = "unknown err"]; // 账号不存在
InValid_Phone = 10001 [(errors.msg) = "手机号格式不正确", (errors.msg_english) = "invalid phone no"]; // 登录失效,请重新登录
}
该插件需要安装go和protobuf运行环境
安装好运行环境后再安装go-error-generator插件
go install github.com/classtorch/go-error-generator/protoc-gen-go-error-generator
安装好后执行下面脚本生成代码
protoc --go-error-generator_out=:. \
--go-error-generator_opt descriptor_file=errors/errors.proto \
--go-error-generator_opt merge_error=false \
--go-error-generator_opt merge_error_path=golang/errors \
--go_out=. -I . account.proto
插件自动生成的代码如下,包含error对象和error map
var (
SUCCESS = &errors.Error{Code: 0, Msg: "成功"} //成功
UnKnown = &errors.Error{Code: -1, Msg: "未知错误"} //未知错误
InValid_Phone = &errors.Error{Code: 10001, Msg: "手机号格式不正确"} //手机号格式不正确
)
var (
Msg = map[int32]*errors.Error{
0: &errors.Error{Code: 0, Msg: "成功"},
-1: &errors.Error{Code: -1, Msg: "未知错误"},
10001: &errors.Error{Code: 10001, Msg: "手机号格式不正确"},
}
Msg_English = map[int32]*errors.Error{
0: &errors.Error{Code: 0, Msg: "success"},
-1: &errors.Error{Code: -1, Msg: "unknown err"},
10001: &errors.Error{Code: 10001, Msg: "invalid phone no"},
}
)
使用生成的error对象和error map改写response和checkPhone方法
func response(w http.ResponseWriter, language int, err error) {
e := account.SUCCESS
var ok bool
if err != nil {
if e, ok = err.(*errors.Error); !ok {
e = account.UnKnown
}
}
if language == Language_Chinese {
if e, ok = account.Msg[e.Code]; !ok {
e = account.UnKnown
}
} else if language == Language_Enligh {
if e, ok = account.Msg_English[e.Code]; !ok {
e = account.UnKnown
}
}
res := make(map[string]interface{})
res["code"] = e.Code
res["msg"] = e.Msg
json, _ := json.Marshal(res)
w.WriteHeader(200)
w.Write(json)
}
func checkPhone(phoneNo string) error {
if len(phoneNo) != 11 {
return account.InValid_Phone
}
return nil
}
完整的代码可以访问go-error-generator-examples项目进行了解
我们来看下这是实现方式的优点
可以看出对于第一个和和第三个需求来说只需要修改protobuf文件,重新生成代码就可以,无须修改业务代码。第二个需求也只是简单的引入新的错误对象。
由于该插件是基于protobuf实现的,如果项目中没有使用prorobuf技术栈的话会带来一些引入成本。不过这点成本相对于频繁修改业务代码还是值得的。
go-error-generator
go-error-generator-examples