engine.go 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  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 (ng *engine) AddRoutes(r featuredRoutes) {
  43. ng.routes = append(ng.routes, r)
  44. }
  45. func (ng *engine) SetUnauthorizedCallback(callback handler.UnauthorizedCallback) {
  46. ng.unauthorizedCallback = callback
  47. }
  48. func (ng *engine) SetUnsignedCallback(callback handler.UnsignedCallback) {
  49. ng.unsignedCallback = callback
  50. }
  51. func (ng *engine) Start() error {
  52. return ng.StartWithRouter(router.NewRouter())
  53. }
  54. func (ng *engine) StartWithRouter(router httpx.Router) error {
  55. if err := ng.bindRoutes(router); err != nil {
  56. return err
  57. }
  58. if len(ng.conf.CertFile) == 0 && len(ng.conf.KeyFile) == 0 {
  59. return internal.StartHttp(ng.conf.Host, ng.conf.Port, router)
  60. }
  61. return internal.StartHttps(ng.conf.Host, ng.conf.Port, ng.conf.CertFile,
  62. ng.conf.KeyFile, router, func(srv *http.Server) {
  63. if ng.tlsConfig != nil {
  64. srv.TLSConfig = ng.tlsConfig
  65. }
  66. })
  67. }
  68. func (ng *engine) appendAuthHandler(fr featuredRoutes, chain alice.Chain,
  69. verifier func(alice.Chain) alice.Chain) alice.Chain {
  70. if fr.jwt.enabled {
  71. if len(fr.jwt.prevSecret) == 0 {
  72. chain = chain.Append(handler.Authorize(fr.jwt.secret,
  73. handler.WithUnauthorizedCallback(ng.unauthorizedCallback)))
  74. } else {
  75. chain = chain.Append(handler.Authorize(fr.jwt.secret,
  76. handler.WithPrevSecret(fr.jwt.prevSecret),
  77. handler.WithUnauthorizedCallback(ng.unauthorizedCallback)))
  78. }
  79. }
  80. return verifier(chain)
  81. }
  82. func (ng *engine) bindFeaturedRoutes(router httpx.Router, fr featuredRoutes, metrics *stat.Metrics) error {
  83. verifier, err := ng.signatureVerifier(fr.signature)
  84. if err != nil {
  85. return err
  86. }
  87. for _, route := range fr.routes {
  88. if err := ng.bindRoute(fr, router, metrics, route, verifier); err != nil {
  89. return err
  90. }
  91. }
  92. return nil
  93. }
  94. func (ng *engine) bindRoute(fr featuredRoutes, router httpx.Router, metrics *stat.Metrics,
  95. route Route, verifier func(chain alice.Chain) alice.Chain) error {
  96. chain := alice.New(
  97. handler.TracingHandler(ng.conf.Name, route.Path),
  98. ng.getLogHandler(),
  99. handler.PrometheusHandler(route.Path),
  100. handler.MaxConns(ng.conf.MaxConns),
  101. handler.BreakerHandler(route.Method, route.Path, metrics),
  102. handler.SheddingHandler(ng.getShedder(fr.priority), metrics),
  103. handler.TimeoutHandler(time.Duration(ng.conf.Timeout)*time.Millisecond),
  104. handler.RecoverHandler,
  105. handler.MetricHandler(metrics),
  106. handler.MaxBytesHandler(ng.conf.MaxBytes),
  107. handler.GunzipHandler,
  108. )
  109. chain = ng.appendAuthHandler(fr, chain, verifier)
  110. for _, middleware := range ng.middlewares {
  111. chain = chain.Append(convertMiddleware(middleware))
  112. }
  113. handle := chain.ThenFunc(route.Handler)
  114. return router.Handle(route.Method, route.Path, handle)
  115. }
  116. func (ng *engine) bindRoutes(router httpx.Router) error {
  117. metrics := ng.createMetrics()
  118. for _, fr := range ng.routes {
  119. if err := ng.bindFeaturedRoutes(router, fr, metrics); err != nil {
  120. return err
  121. }
  122. }
  123. return nil
  124. }
  125. func (ng *engine) createMetrics() *stat.Metrics {
  126. var metrics *stat.Metrics
  127. if len(ng.conf.Name) > 0 {
  128. metrics = stat.NewMetrics(ng.conf.Name)
  129. } else {
  130. metrics = stat.NewMetrics(fmt.Sprintf("%s:%d", ng.conf.Host, ng.conf.Port))
  131. }
  132. return metrics
  133. }
  134. func (ng *engine) getLogHandler() func(http.Handler) http.Handler {
  135. if ng.conf.Verbose {
  136. return handler.DetailedLogHandler
  137. }
  138. return handler.LogHandler
  139. }
  140. func (ng *engine) getShedder(priority bool) load.Shedder {
  141. if priority && ng.priorityShedder != nil {
  142. return ng.priorityShedder
  143. }
  144. return ng.shedder
  145. }
  146. func (ng *engine) setTlsConfig(cfg *tls.Config) {
  147. ng.tlsConfig = cfg
  148. }
  149. func (ng *engine) signatureVerifier(signature signatureSetting) (func(chain alice.Chain) alice.Chain, error) {
  150. if !signature.enabled {
  151. return func(chain alice.Chain) alice.Chain {
  152. return chain
  153. }, nil
  154. }
  155. if len(signature.PrivateKeys) == 0 {
  156. if signature.Strict {
  157. return nil, ErrSignatureConfig
  158. }
  159. return func(chain alice.Chain) alice.Chain {
  160. return chain
  161. }, nil
  162. }
  163. decrypters := make(map[string]codec.RsaDecrypter)
  164. for _, key := range signature.PrivateKeys {
  165. fingerprint := key.Fingerprint
  166. file := key.KeyFile
  167. decrypter, err := codec.NewRsaDecrypter(file)
  168. if err != nil {
  169. return nil, err
  170. }
  171. decrypters[fingerprint] = decrypter
  172. }
  173. return func(chain alice.Chain) alice.Chain {
  174. if ng.unsignedCallback != nil {
  175. return chain.Append(handler.ContentSecurityHandler(
  176. decrypters, signature.Expiry, signature.Strict, ng.unsignedCallback))
  177. }
  178. return chain.Append(handler.ContentSecurityHandler(
  179. decrypters, signature.Expiry, signature.Strict))
  180. }, nil
  181. }
  182. func (ng *engine) use(middleware Middleware) {
  183. ng.middlewares = append(ng.middlewares, middleware)
  184. }
  185. func convertMiddleware(ware Middleware) func(http.Handler) http.Handler {
  186. return func(next http.Handler) http.Handler {
  187. return ware(next.ServeHTTP)
  188. }
  189. }