server.go 4.5 KB

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