proxyServer.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. package proxy
  2. import (
  3. "fmt"
  4. "net/http"
  5. "net/url"
  6. log "app.yhyue.com/moapp/jylog"
  7. . "bp.jydev.jianyu360.cn/BaseService/gateway/common/gatecode"
  8. "bp.jydev.jianyu360.cn/BaseService/gateway/core/node"
  9. "bp.jydev.jianyu360.cn/BaseService/gateway/core/proxy/broker"
  10. "bp.jydev.jianyu360.cn/BaseService/gateway/core/proxy/broker/outServer"
  11. "bp.jydev.jianyu360.cn/BaseService/gateway/core/proxy/middleware"
  12. "bp.jydev.jianyu360.cn/BaseService/gateway/core/proxy/proxyClient"
  13. "bp.jydev.jianyu360.cn/BaseService/gateway/core/proxy/vars"
  14. "bp.jydev.jianyu360.cn/BaseService/gateway/core/router"
  15. "github.com/gogf/gf/v2/frame/g"
  16. "github.com/gogf/gf/v2/net/ghttp"
  17. "github.com/gogf/gf/v2/net/gtrace"
  18. "github.com/gogf/gf/v2/os/gcfg"
  19. "github.com/gogf/gf/v2/os/gctx"
  20. )
  21. var routerManager *router.Manager
  22. const errTryTime = 10 //错误尝试
  23. // InitGateWayServer 初始化网关服务
  24. func InitGateWayServer(initCtx gctx.Ctx) *ghttp.Server {
  25. initCtx, span := gtrace.NewSpan(initCtx, "InitGateWayServer")
  26. defer span.End()
  27. //创建节点,并持续观察节点变化
  28. watchNode := node.NewNode(gcfg.Instance().MustGet(initCtx, "system.etcdListen", nil).Strings()...)
  29. go watchNode.NewWatcher(initCtx, vars.BManager)
  30. //初始化可访问路由
  31. var err error
  32. routerManager, err = router.InitRouterManager()
  33. if err != nil {
  34. log.WithContext(initCtx).Errorln(err)
  35. }
  36. //初始化外部服务
  37. sussBiServer, err := outServer.InitSussBi(gcfg.Instance().MustGet(initCtx, "outServer.sussbi", nil).Map())
  38. if err != nil {
  39. log.WithContext(initCtx).Infoln(err)
  40. } else {
  41. vars.BManager.RegisterOutServer(sussBiServer.Url, sussBiServer)
  42. }
  43. gateWayServer := g.Server()
  44. //关闭系统自带请求日志
  45. gateWayServer.SetLogger(g.Log())
  46. gateWayServer.SetErrorLogEnabled(false)
  47. //注册中间件
  48. gateWayServer.Use(middleware.ErrorHandler) //错误拦截
  49. gateWayServer.Use(func(r *ghttp.Request) {
  50. r.SetError(routerManager.InfusionContext(r)) //context注入全局信息
  51. r.Middleware.Next()
  52. })
  53. gateWayServer.Use(middleware.FilterPolyHandler) //反爬虫策略
  54. gateWayServer.Use(middleware.FilterHandler) //权限过滤
  55. //加载代理客户端
  56. proxyClient.ReLoadClient()
  57. //注册代理
  58. gateWayServer.BindHandler("POST:/*", proxyHandler) //接口代理
  59. gateWayServer.BindHandler("GET:/*", proxyHandler) //页面代理
  60. gateWayServer.AddStaticPath("/antiRes", "./resources/antiRes") //静态资源
  61. return gateWayServer
  62. }
  63. // proxyHandler 网关代理Handler处理
  64. // 完成所有前置校验后,请求代理服务逻辑
  65. var proxyHandler = func(r *ghttp.Request) {
  66. _, span := gtrace.NewSpan(r.Context(), "proxyHandler")
  67. defer span.End()
  68. if r.GetError() != nil {
  69. return
  70. }
  71. // 获取请求上下文内容
  72. gCtx := router.GetGContext(r.GetCtx())
  73. // 请求重试,防止某个服务中断不可用,导致接口不可用。
  74. for i := 0; i < errTryTime; i++ {
  75. var proxyAddr *url.URL
  76. var err error
  77. var changeFunc func(resp *http.Response) error
  78. if gCtx.RouterRule.IsOutServer {
  79. //外部服务直接获取
  80. proxyAddr, err = vars.BManager.GetOutSeverAutoLogin(gCtx.RouterRule.MiddleCode, r)
  81. changeFunc = broker.UnLoginSetErr //当未登录时,通过处罚异常尝试重新加载
  82. } else {
  83. // 从etcd注册的地址,根据负载规则获取服务地址
  84. proxyAddr, err = vars.BManager.GetServerAddr(gCtx.RouterRule.MiddleCode, r.GetClientIp())
  85. }
  86. if err != nil {
  87. r.SetError(err)
  88. return
  89. }
  90. // 代理地址存入上下文ctx中
  91. gCtx.ServerAddr = proxyAddr.String()
  92. router.UpdateGContext(r, gCtx)
  93. // 捕获异常,若代理出错,则进行重试
  94. var hasErr bool
  95. errHandel := func(hw http.ResponseWriter, hr *http.Request, err error) {
  96. hasErr = true
  97. if err.Error() == "context canceled" {
  98. return
  99. }
  100. if i == (errTryTime - 1) {
  101. r.SetError(NewErrorWithCode(GATEWAY_PROXY_ERR, fmt.Sprintf("代理异常:%s err:%v \n", gCtx.ServerAddr, err.Error())))
  102. }
  103. log.WithContext(r.Context()).Error("ErrorHandler ", err)
  104. }
  105. // 代理请求
  106. proxyClient.CreateCustomProxyClient(proxyAddr, errHandel, changeFunc).ServeHTTP(r.Response.ResponseWriter, r.Request)
  107. if gCtx.RouterRule.IsOutServer { //检测外部服务登录状态是否过期
  108. if vars.BManager.CheckOutSeverLoginOut(gCtx.RouterRule.MiddleCode, r) || hasErr {
  109. if err := vars.BManager.OutSeverLoginIn(gCtx.RouterRule.MiddleCode); err != nil {
  110. log.WithContext(r.GetCtx()).Errorf("%s 外部服务自动登录异常 %v \n", gCtx.RouterRule.MiddleCode, err)
  111. }
  112. }
  113. }
  114. // 未捕获到请求,标识请求成功
  115. if !hasErr {
  116. return
  117. }
  118. }
  119. }