engine.go 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. package rest
  2. import (
  3. "crypto/tls"
  4. "errors"
  5. "fmt"
  6. "net/http"
  7. "time"
  8. "github.com/justinas/alice"
  9. "github.com/tal-tech/go-zero/core/codec"
  10. "github.com/tal-tech/go-zero/core/load"
  11. "github.com/tal-tech/go-zero/core/stat"
  12. "github.com/tal-tech/go-zero/rest/handler"
  13. "github.com/tal-tech/go-zero/rest/httpx"
  14. "github.com/tal-tech/go-zero/rest/internal"
  15. "github.com/tal-tech/go-zero/rest/router"
  16. )
  17. // use 1000m to represent 100%
  18. const topCpuUsage = 1000
  19. // ErrSignatureConfig is an error that indicates bad config for signature.
  20. var ErrSignatureConfig = errors.New("bad config for Signature")
  21. type engine struct {
  22. conf RestConf
  23. routes []featuredRoutes
  24. unauthorizedCallback handler.UnauthorizedCallback
  25. unsignedCallback handler.UnsignedCallback
  26. middlewares []Middleware
  27. shedder load.Shedder
  28. priorityShedder load.Shedder
  29. tlsConfig *tls.Config
  30. }
  31. func newEngine(c RestConf) *engine {
  32. srv := &engine{
  33. conf: c,
  34. }
  35. if c.CpuThreshold > 0 {
  36. srv.shedder = load.NewAdaptiveShedder(load.WithCpuThreshold(c.CpuThreshold))
  37. srv.priorityShedder = load.NewAdaptiveShedder(load.WithCpuThreshold(
  38. (c.CpuThreshold + topCpuUsage) >> 1))
  39. }
  40. return srv
  41. }
  42. func (s *engine) AddRoutes(r featuredRoutes) {
  43. s.routes = append(s.routes, r)
  44. }
  45. func (s *engine) SetUnauthorizedCallback(callback handler.UnauthorizedCallback) {
  46. s.unauthorizedCallback = callback
  47. }
  48. func (s *engine) SetUnsignedCallback(callback handler.UnsignedCallback) {
  49. s.unsignedCallback = callback
  50. }
  51. func (s *engine) Start() error {
  52. return s.StartWithRouter(router.NewRouter())
  53. }
  54. func (s *engine) StartWithRouter(router httpx.Router) error {
  55. if err := s.bindRoutes(router); err != nil {
  56. return err
  57. }
  58. if len(s.conf.CertFile) == 0 && len(s.conf.KeyFile) == 0 {
  59. return internal.StartHttp(s.conf.Host, s.conf.Port, router)
  60. }
  61. return internal.StartHttps(s.conf.Host, s.conf.Port, s.conf.CertFile, s.conf.KeyFile, s.tlsConfig, router)
  62. }
  63. func (s *engine) appendAuthHandler(fr featuredRoutes, chain alice.Chain,
  64. verifier func(alice.Chain) alice.Chain) alice.Chain {
  65. if fr.jwt.enabled {
  66. if len(fr.jwt.prevSecret) == 0 {
  67. chain = chain.Append(handler.Authorize(fr.jwt.secret,
  68. handler.WithUnauthorizedCallback(s.unauthorizedCallback)))
  69. } else {
  70. chain = chain.Append(handler.Authorize(fr.jwt.secret,
  71. handler.WithPrevSecret(fr.jwt.prevSecret),
  72. handler.WithUnauthorizedCallback(s.unauthorizedCallback)))
  73. }
  74. }
  75. return verifier(chain)
  76. }
  77. func (s *engine) bindFeaturedRoutes(router httpx.Router, fr featuredRoutes, metrics *stat.Metrics) error {
  78. verifier, err := s.signatureVerifier(fr.signature)
  79. if err != nil {
  80. return err
  81. }
  82. for _, route := range fr.routes {
  83. if err := s.bindRoute(fr, router, metrics, route, verifier); err != nil {
  84. return err
  85. }
  86. }
  87. return nil
  88. }
  89. func (s *engine) bindRoute(fr featuredRoutes, router httpx.Router, metrics *stat.Metrics,
  90. route Route, verifier func(chain alice.Chain) alice.Chain) error {
  91. chain := alice.New(
  92. handler.TracingHandler(s.conf.Name, route.Path),
  93. s.getLogHandler(),
  94. handler.PrometheusHandler(route.Path),
  95. handler.MaxConns(s.conf.MaxConns),
  96. handler.BreakerHandler(route.Method, route.Path, metrics),
  97. handler.SheddingHandler(s.getShedder(fr.priority), metrics),
  98. handler.TimeoutHandler(time.Duration(s.conf.Timeout)*time.Millisecond),
  99. handler.RecoverHandler,
  100. handler.MetricHandler(metrics),
  101. handler.MaxBytesHandler(s.conf.MaxBytes),
  102. handler.GunzipHandler,
  103. )
  104. chain = s.appendAuthHandler(fr, chain, verifier)
  105. for _, middleware := range s.middlewares {
  106. chain = chain.Append(convertMiddleware(middleware))
  107. }
  108. handle := chain.ThenFunc(route.Handler)
  109. return router.Handle(route.Method, route.Path, handle)
  110. }
  111. func (s *engine) bindRoutes(router httpx.Router) error {
  112. metrics := s.createMetrics()
  113. for _, fr := range s.routes {
  114. if err := s.bindFeaturedRoutes(router, fr, metrics); err != nil {
  115. return err
  116. }
  117. }
  118. return nil
  119. }
  120. func (s *engine) createMetrics() *stat.Metrics {
  121. var metrics *stat.Metrics
  122. if len(s.conf.Name) > 0 {
  123. metrics = stat.NewMetrics(s.conf.Name)
  124. } else {
  125. metrics = stat.NewMetrics(fmt.Sprintf("%s:%d", s.conf.Host, s.conf.Port))
  126. }
  127. return metrics
  128. }
  129. func (s *engine) getLogHandler() func(http.Handler) http.Handler {
  130. if s.conf.Verbose {
  131. return handler.DetailedLogHandler
  132. }
  133. return handler.LogHandler
  134. }
  135. func (s *engine) getShedder(priority bool) load.Shedder {
  136. if priority && s.priorityShedder != nil {
  137. return s.priorityShedder
  138. }
  139. return s.shedder
  140. }
  141. func (s *engine) signatureVerifier(signature signatureSetting) (func(chain alice.Chain) alice.Chain, error) {
  142. if !signature.enabled {
  143. return func(chain alice.Chain) alice.Chain {
  144. return chain
  145. }, nil
  146. }
  147. if len(signature.PrivateKeys) == 0 {
  148. if signature.Strict {
  149. return nil, ErrSignatureConfig
  150. }
  151. return func(chain alice.Chain) alice.Chain {
  152. return chain
  153. }, nil
  154. }
  155. decrypters := make(map[string]codec.RsaDecrypter)
  156. for _, key := range signature.PrivateKeys {
  157. fingerprint := key.Fingerprint
  158. file := key.KeyFile
  159. decrypter, err := codec.NewRsaDecrypter(file)
  160. if err != nil {
  161. return nil, err
  162. }
  163. decrypters[fingerprint] = decrypter
  164. }
  165. return func(chain alice.Chain) alice.Chain {
  166. if s.unsignedCallback != nil {
  167. return chain.Append(handler.ContentSecurityHandler(
  168. decrypters, signature.Expiry, signature.Strict, s.unsignedCallback))
  169. }
  170. return chain.Append(handler.ContentSecurityHandler(
  171. decrypters, signature.Expiry, signature.Strict))
  172. }, nil
  173. }
  174. func (s *engine) use(middleware Middleware) {
  175. s.middlewares = append(s.middlewares, middleware)
  176. }
  177. func convertMiddleware(ware Middleware) func(http.Handler) http.Handler {
  178. return func(next http.Handler) http.Handler {
  179. return ware(next.ServeHTTP)
  180. }
  181. }