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