server.go 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. package http
  2. import (
  3. "context"
  4. "git.nspix.com/golang/micro/gateway/http/router"
  5. "golang.org/x/net/websocket"
  6. "net"
  7. "net/http"
  8. "sync"
  9. )
  10. var (
  11. contextPool sync.Pool
  12. )
  13. type contextRouteKey struct{}
  14. type HandleFunc func(ctx *Context) (err error)
  15. type Middleware func(next HandleFunc) HandleFunc
  16. type Node struct {
  17. Method string
  18. Path string
  19. Handle HandleFunc
  20. }
  21. type Server struct {
  22. svr *http.Server
  23. middleware []Middleware
  24. keepAlive bool
  25. isSetKeepAlive bool
  26. router *router.Router
  27. }
  28. func getContext() *Context {
  29. if v := contextPool.Get(); v != nil {
  30. return v.(*Context)
  31. }
  32. return &Context{}
  33. }
  34. func releaseContext(c *Context) {
  35. contextPool.Put(c)
  36. }
  37. func (r *Server) Group(prefix string, nodes []Node, middleware ...Middleware) {
  38. for _, node := range nodes {
  39. r.Handle(node.Method, prefix+node.Path, node.Handle, middleware...)
  40. }
  41. }
  42. func (r *Server) Handler(method, path string, h http.Handler) {
  43. r.router.Handler(method, path, h)
  44. }
  45. func (r *Server) createWebsocket(h websocket.Handler) *websocket.Server {
  46. return &websocket.Server{
  47. Handshake: func(cfg *websocket.Config, req *http.Request) (err error) {
  48. cfg.Origin, err = websocket.Origin(cfg, req)
  49. if err == nil && cfg != nil {
  50. if cfg.Header == nil {
  51. cfg.Header = http.Header(make(map[string][]string))
  52. }
  53. if cfg.Origin != nil {
  54. cfg.Header.Set("Access-Control-Allow-Origin", cfg.Origin.Host)
  55. }
  56. cfg.Header.Set("Access-Control-Request-Headers", "GET,POST,PUT,OPTIONS")
  57. cfg.Header.Set("Access-Control-Allow-Credentials", "true")
  58. }
  59. return
  60. },
  61. Handler: h,
  62. }
  63. }
  64. func (r *Server) Websocket(path string, h websocket.Handler) {
  65. r.Handler("GET", path, r.createWebsocket(h))
  66. }
  67. func (r *Server) Handle(method string, path string, h HandleFunc, middleware ...Middleware) {
  68. r.router.Handle(method, path, func(writer http.ResponseWriter, request *http.Request, params router.Params) {
  69. ctx := getContext()
  70. ps := make(map[string]string)
  71. for _, v := range params {
  72. ps[v.Key] = v.Value
  73. }
  74. ctx.Reset(request, writer, ps)
  75. for i := len(r.middleware) - 1; i >= 0; i-- {
  76. h = r.middleware[i](h)
  77. }
  78. for i := len(middleware) - 1; i >= 0; i-- {
  79. h = middleware[i](h)
  80. }
  81. if err := h(ctx); err != nil {
  82. writer.WriteHeader(http.StatusInternalServerError)
  83. }
  84. releaseContext(ctx)
  85. })
  86. }
  87. func (r *Server) Use(ms ...Middleware) {
  88. r.middleware = append(r.middleware, ms...)
  89. }
  90. func (r *Server) ServeHTTP(res http.ResponseWriter, req *http.Request) {
  91. if req.Method == http.MethodOptions {
  92. res.Header().Add("Vary", "Origin")
  93. res.Header().Add("Vary", "Access-Control-Request-Method")
  94. res.Header().Add("Vary", "Access-Control-Request-Headers")
  95. res.Header().Set("Access-Control-Allow-Origin", "*")
  96. res.Header().Set("Access-Control-Allow-Credentials", "true")
  97. res.Header().Set("Access-Control-Allow-Methods", "GET,HEAD,PUT,PATCH,POST,DELETE")
  98. h := req.Header.Get("Access-Control-Request-Headers")
  99. if h != "" {
  100. res.Header().Set("Access-Control-Allow-Headers", h)
  101. }
  102. res.WriteHeader(http.StatusNoContent)
  103. return
  104. } else {
  105. res.Header().Add("Vary", "Origin")
  106. res.Header().Set("Access-Control-Allow-Origin", "*")
  107. res.Header().Set("Access-Control-Allow-Credentials", "true")
  108. h := req.Header.Get("Access-Control-Request-Headers")
  109. if h != "" {
  110. res.Header().Set("Access-Control-Allow-Headers", h)
  111. }
  112. r.router.ServeHTTP(res, req)
  113. }
  114. }
  115. func (r *Server) Server() *http.Server {
  116. return r.svr
  117. }
  118. func (r *Server) SetKeepAlivesEnabled(enable bool) {
  119. r.isSetKeepAlive = true
  120. r.keepAlive = enable
  121. }
  122. func (r *Server) ListenAndServe(addr string) (err error) {
  123. r.svr = &http.Server{
  124. Addr: addr,
  125. Handler: r,
  126. }
  127. if r.isSetKeepAlive {
  128. r.svr.SetKeepAlivesEnabled(r.keepAlive)
  129. }
  130. return r.svr.ListenAndServe()
  131. }
  132. func (r *Server) Serve(l net.Listener) (err error) {
  133. r.svr = &http.Server{
  134. Handler: r,
  135. }
  136. if r.isSetKeepAlive {
  137. r.svr.SetKeepAlivesEnabled(r.keepAlive)
  138. }
  139. return r.svr.Serve(l)
  140. }
  141. func (r *Server) Shutdown(ctx context.Context) (err error) {
  142. if r.svr != nil {
  143. return r.svr.Shutdown(ctx)
  144. }
  145. return
  146. }
  147. func New() *Server {
  148. return &Server{
  149. router: router.New(),
  150. }
  151. }
  152. func WithContext(ctx context.Context, route *Server) context.Context {
  153. return context.WithValue(ctx, contextRouteKey{}, route)
  154. }
  155. func FromContext(ctx context.Context) *Server {
  156. if v := ctx.Value(contextRouteKey{}); v != nil {
  157. return v.(*Server)
  158. }
  159. return nil
  160. }