manager.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. package router
  2. import (
  3. "bp.jydev.jianyu360.cn/BaseService/gateway/common/db"
  4. "bp.jydev.jianyu360.cn/BaseService/gateway/common/enum"
  5. . "bp.jydev.jianyu360.cn/BaseService/gateway/common/gatecode"
  6. "fmt"
  7. "github.com/gogf/gf/v2/frame/g"
  8. "github.com/gogf/gf/v2/net/ghttp"
  9. "github.com/gogf/gf/v2/os/gctx"
  10. "github.com/gogf/gf/v2/util/gconv"
  11. "log"
  12. "regexp"
  13. )
  14. type Manager struct {
  15. eqRouters map[string]*Router
  16. regexRouter map[*regexp.Regexp]*Router
  17. }
  18. // InitRouterManager 初始化系统代理路由
  19. // 支持完全匹配和正则匹配
  20. func InitRouterManager() (*Manager, error) {
  21. //加载规则
  22. res := db.GateWatMySql.Query(fmt.Sprintf("SELECT status,middleground_code,url,function_code,check_sess,check_power,check_auth,check_status,check_blacklist,ploy_code,timeout,remark,appid,deduct_source,power_type FROM %s",
  23. g.Cfg().MustGet(gctx.New(), "system.routerTable", "front_proxy").String()))
  24. if res == nil || len(*res) == 0 {
  25. return nil, fmt.Errorf("未发现可用路由")
  26. }
  27. // 初始化 routerManager
  28. routerManager := &Manager{
  29. eqRouters: make(map[string]*Router),
  30. regexRouter: make(map[*regexp.Regexp]*Router),
  31. }
  32. for _, row := range *res {
  33. // 获取路由信息
  34. router := gconv.String(row["url"])
  35. routerRule := &Router{
  36. Status: gconv.Int(row["status"]),
  37. PowerCheck: gconv.Int(row["check_power"]),
  38. SessCheck: enum.NewSessCheck(gconv.Int64(row["check_sess"])),
  39. AccountCheck: enum.NewAccountCheck(gconv.Int64(row["check_status"])),
  40. AuthCheck: enum.NewAuthCheck(gconv.Int64(row["check_auth"])),
  41. BlackCheck: gconv.Int(row["check_blacklist"]) == 1,
  42. LimitPloy: gconv.String(row["ploy_code"]), //策略
  43. Deduct: gconv.Int(row["deduct_source"]),
  44. FuncCode: gconv.String(row["function_code"]),
  45. MiddleCode: gconv.String(row["middleground_code"]),
  46. AppId: gconv.String(row["appid"]),
  47. ReqUrl: router,
  48. TimeOut: gconv.Int64(row["timeout"]),
  49. Remark: gconv.String(row["remark"]),
  50. PowerType: gconv.Int(row["power_type"]),
  51. }
  52. // 判断路由匹配方式是完全匹配还是正则匹配 (此处逻辑参考x-web框架)
  53. if regexp.QuoteMeta(router) == router {
  54. routerManager.eqRouters[router] = routerRule
  55. } else {
  56. reg, err := regexp.Compile(router)
  57. if err != nil {
  58. log.Printf("路由%s装载异常 %v\n", router, err)
  59. }
  60. routerManager.regexRouter[reg] = routerRule
  61. }
  62. }
  63. return routerManager, nil
  64. }
  65. // GetRouterRule 获取路由规则
  66. // 根据用户请求地址匹配路由规则,优先绝对匹配后用正则匹配。
  67. func (m *Manager) GetRouterRule(url string) (*Router, error) {
  68. routerRule, exists := m.eqRouters[url]
  69. if !exists {
  70. for reg, thisRouterRule := range m.regexRouter {
  71. if reg.MatchString(url) {
  72. routerRule = thisRouterRule
  73. }
  74. }
  75. }
  76. if routerRule == nil {
  77. return nil, NewErrorWithCode(GATEWAY_ROUTER_NOTFIND, fmt.Sprintf("未找到请求地址%s,请检查是否注册到数据库\n", url))
  78. }
  79. // 路由状态判断
  80. if routerRule.Status != 1 {
  81. return nil, NewErrorWithCode(GATEWAY_ROUTER_UPHOLD, fmt.Sprintf("接口状态:%d\n", routerRule.Status))
  82. }
  83. return routerRule, nil
  84. }
  85. // InfusionContext 注入通用结构体gContext
  86. func (m *Manager) InfusionContext(r *ghttp.Request) (err error) {
  87. var router *Router
  88. var GCtx = &GContext{
  89. Sess: &JySession{},
  90. RouterRule: &Router{},
  91. }
  92. router, err = m.GetRouterRule(r.RequestURI)
  93. if err != nil {
  94. r.SetCtxVar(GContextKey, GCtx)
  95. return
  96. }
  97. GCtx.RouterRule = router
  98. //获取session
  99. jySess, _ := InitJySessionContext(r)
  100. GCtx.Sess = jySess
  101. r.SetCtxVar(GContextKey, GCtx)
  102. return
  103. }