溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

企業(yè)項目遷移go-zero的方法教程

發(fā)布時間:2021-10-19 17:26:42 來源:億速云 閱讀:113 作者:iii 欄目:編程語言

這篇文章主要介紹“企業(yè)項目遷移go-zero的方法教程”,在日常操作中,相信很多人在企業(yè)項目遷移go-zero的方法教程問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”企業(yè)項目遷移go-zero的方法教程”的疑惑有所幫助!接下來,請跟著小編一起來學(xué)習(xí)吧!

gateway service

gateway 中我做了一些自定義,在端請求我們后臺接口情況下,雖然多數(shù)情況是不需要關(guān)心錯誤碼的,但是避免不了要某些場景還是需要根據(jù)固定錯誤碼去做特殊處理,我自己定義了一個錯誤類,這個錯誤類只在 gateway 中使用:

err.go:

package xerr

import "fmt"

type CodeError struct {
   errCode int
   errMsg  string
}

// 屬性
func (e *CodeError) GetErrCode() int {
   return e.errCode
}

func (e *CodeError) GetErrMsg() string {
   return e.errMsg
}

func (e *CodeError) Error() string {
   return fmt.Sprintf("ErrCode:%d,ErrMsg:%s", e.errCode, e.errMsg)
}

func New(errCode int, errMsg string) *CodeError {
   return &CodeError{errCode: errCode, errMsg: errMsg}
}

func NewErrCode(errCode int) *CodeError {
   return &CodeError{errCode: errCode, errMsg: MapErrMsg(errCode)}
}

func NewErrMsg(errMsg string) *CodeError {
   return &CodeError{errCode: BAD_REUQEST_ERROR, errMsg: errMsg}
}

errmsg.go

package xerr

var message map[int]string

func init()  {
   message = make(map[int]string)
   message[OK] = "SUCCESS"
   message[BAD_REUQEST_ERROR] = "服務(wù)器繁忙,請稍后再試"
   message[REUQES_PARAM_ERROR] = "參數(shù)錯誤"
   message[USER_NOT_FOUND] = "用戶不存在"
}

func MapErrMsg(errcode int) string {
   if msg, ok := message[errcode]; ok {
      return msg
   } else {
      return "服務(wù)器繁忙,請稍后再試"
   }
}

errcode.go

package xerr

// 成功返回
const OK = 200

// 全局錯誤碼
// 前3位代表業(yè)務(wù),后三位代表具體功能
const BAD_REUQEST_ERROR = 100001
const REUQES_PARAM_ERROR = 100002

// 用戶模塊
const USER_NOT_FOUND = 200001

我將三個文件統(tǒng)一放在 lib/xerr 目錄

企業(yè)項目遷移go-zero的方法教程

有了錯誤碼還不行,還要定義統(tǒng)一返回http的結(jié)果,goctl 生成的默認(rèn)的是挺好的,但是沒法符合我這種返回自定義錯誤碼需求,于是我自己有寫了一個統(tǒng)一返回結(jié)果的文件:

httpresult:

package xhttp

import (
   "fishtwo/lib/xerr"
   "fmt"
   "github.com/tal-tech/go-zero/core/logx"
   "github.com/tal-tech/go-zero/rest/httpx"
   "google.golang.org/grpc/status"
   "net/http"
   "github.com/pkg/errors"
)

// http方法
func HttpResult(r *http.Request,w http.ResponseWriter,resp interface{},err error)  {
   if err == nil {
      // 成功返回
      r:= Success(resp)
      httpx.WriteJson(w, http.StatusOK, r)
   } else {
      // 錯誤返回
      errcode := xerr.BAD_REUQEST_ERROR
      errmsg := "服務(wù)器繁忙,請稍后再試"
      if e,ok := err.(*xerr.CodeError);ok{
         // 自定義CodeError
         errcode = e.GetErrCode()
         errmsg = e.GetErrMsg()
      } else {
         originErr := errors.Cause(err) // err類型
         if gstatus, ok := status.FromError(originErr);ok{
            // grpc err錯誤
            errmsg = gstatus.Message()
         }
      }
      logx.WithContext(r.Context()).Error("【GATEWAY-SRV-ERR】 : %+v ",err)

      httpx.WriteJson(w, http.StatusBadRequest, Error(errcode,errmsg))
   }
}

// http參數(shù)錯誤返回
func ParamErrorResult(r *http.Request,w http.ResponseWriter,err error)  {
   errMsg := fmt.Sprintf("%s ,%s", xerr.MapErrMsg(xerr.REUQES_PARAM_ERROR), err.Error())
   httpx.WriteJson(w, http.StatusBadRequest, Error(xerr.REUQES_PARAM_ERROR,errMsg))
}

responsebean

package xhttp

type (
   NullJson struct {}

   ResponseSuccessBean struct {
      Code int         `json:"code"`
      Msg  string      `json:"msg"`
      Data interface{} `json:"data"`
   }
)

func Success(data interface{}) *ResponseSuccessBean {
   return &ResponseSuccessBean{200, "OK", data}
}


type ResponseErrorBean struct {
   Code int         `json:"code"`
   Msg  string      `json:"msg"`
}

func Error(errCode int,errMsg string) *ResponseErrorBean {
   return &ResponseErrorBean{errCode, errMsg}
}

放在 lib/xhttp下

企業(yè)項目遷移go-zero的方法教程

然后改造了internal/handler/下通過goctl生成的代碼:

企業(yè)項目遷移go-zero的方法教程

當(dāng)然你會說,每次生成完都要手動去改,好麻煩!

當(dāng)當(dāng)當(dāng)當(dāng)~~~ goctltemplate 來咯 https://www.yuque.com/tal-tech/go-zero/mkpuit

然后修改 ~/.goctl/api/handler.tpl:

package handler

import (
	 "net/http"

	 {{.ImportPackages}}
)

func {{.HandlerName}}(ctx *svc.ServiceContext) http.HandlerFunc {
   return func(w http.ResponseWriter, r *http.Request) {
      {{if .HasRequest}}var req types.{{.RequestType}}
      if err := httpx.Parse(r, &req); err != nil {
         xhttp.ParamErrorResult(r,w,err)
         return
      }{{end}}

      l := logic.New{{.LogicType}}(r.Context(), ctx)
      resp, err := l.Login(req)
      xhttp.HttpResult(r,w,resp,err)
   }
}

在重新生成看看,是不是就 beautiful 了,哈哈

然后在說我們的 gateway log,如果眼神好的用戶,在上面的 httpresult.go 中已經(jīng)看到了 log 的身影:

企業(yè)項目遷移go-zero的方法教程

是的是的,這樣處理就可以啦,這樣只要有錯誤就會打印日志了,go-zero 已經(jīng)把 trace-id 帶進(jìn)去了,啥?trace-id 不知道是啥?嗯,其實就是把一次請求通過此 id 串聯(lián)起來,比如你 user-api 調(diào)用 user->srv 或者其他 srv,那要把他們這一次請求都串聯(lián)起來,需要一個唯一標(biāo)識別,這個 id 就是做這個,做鏈路追蹤有很多,比如 jaeger、zipkin。

RPC service

企業(yè)項目遷移go-zero的方法教程

modelrpc 服務(wù)中,官方文檔推薦是將 model 放在 services 目錄下,與每個 rpc 服務(wù)一層,但是個人感覺每個 model 對應(yīng)一張表,一張表只能由一個服務(wù)去控制,哪個服務(wù)控制這張表就哪個服務(wù)擁有控制這個 model 權(quán)利,其他服務(wù)想訪問就要通過 grpc,這是個人的想法,所以我把每個服務(wù)自己管控的 model 放在了 internal

enum:另外我在服務(wù)下加了 enum 枚舉目錄,因為其他 rpc 服務(wù)或者 api 服務(wù)會調(diào)用這個枚舉去比對,我就放在 internal 外部

企業(yè)項目遷移go-zero的方法教程

到此,關(guān)于“企業(yè)項目遷移go-zero的方法教程”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識,請繼續(xù)關(guān)注億速云網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI