|
- package rex
-
- import (
- "bytes"
- "encoding/json"
- "flag"
- "fmt"
- "net/http"
- "path/filepath"
- "runtime"
- "strings"
- "sync"
- "time"
-
- log "github.com/Sirupsen/logrus"
- "github.com/goanywhere/env"
- "github.com/gorilla/mux"
- )
-
- var (
- debug bool
- port int
- maxprocs int
-
- once sync.Once
- )
-
- type server struct {
- middleware *middleware
- mux *mux.Router
- ready bool
- subservers []*server
- }
-
- func New() *server {
- self := &server{
- middleware: new(middleware),
- mux: mux.NewRouter().StrictSlash(true),
- }
- self.configure()
- return self
- }
-
- func (self *server) configure() {
- once.Do(func() {
- flag.BoolVar(&debug, "debug", env.Bool("DEBUG", true), "flag to toggle debug mode")
- flag.IntVar(&port, "port", env.Int("PORT", 5000), "port to run the application server")
- flag.IntVar(&maxprocs, "maxprocs", env.Int("MAXPROCS", runtime.NumCPU()), "maximum cpu processes to run the server")
- flag.Parse()
- })
- }
-
- // build constructs all server/subservers along with their middleware modules chain.
- func (self *server) build() http.Handler {
- if !self.ready {
- // * add server mux into middlware stack to serve as final http.Handler.
- self.Use(func(http.Handler) http.Handler {
- return self.mux
- })
- // * add subservers into middlware stack to serve as final http.Handler.
- for index := 0; index < len(self.subservers); index++ {
- server := self.subservers[index]
- server.Use(func(http.Handler) http.Handler {
- return server.mux
- })
- }
- self.ready = true
- }
- return self.middleware
- }
-
- // register adds the http.Handler/http.HandleFunc into Gorilla mux.
- func (self *server) register(pattern string, handler interface{}, methods ...string) {
- var name = strings.Join(methods, "|") + ":" + pattern
- // finds the full function name (with package) as its mappings.
- //var name = runtime.FuncForPC(reflect.ValueOf(handler).Pointer()).Name()
-
- switch H := handler.(type) {
- case http.Handler:
- self.mux.Handle(pattern, H).Methods(methods...).Name(name)
-
- case func(http.ResponseWriter, *http.Request):
- self.mux.HandleFunc(pattern, H).Methods(methods...).Name(name)
-
- default:
- panic("Unsupported handler: " + name)
- }
- }
-
- // Any maps most common HTTP methods request to the given `http.Handler`.
- // Supports: GET | POST | PUT | DELETE | OPTIONS | HEAD
- func (self *server) Any(pattern string, handler interface{}) {
- self.register(pattern, handler, "GET", "POST", "PUT", "DELETE", "OPTIONS", "HEAD")
- }
-
- // Group creates a new application group under the given path prefix.
- func (self *server) Group(prefix string) *server {
- var middleware = new(middleware)
- self.mux.PathPrefix(prefix).Handler(middleware)
- var mux = self.mux.PathPrefix(prefix).Subrouter()
-
- server := &server{middleware: middleware, mux: mux}
- self.subservers = append(self.subservers, server)
- return server
- }
-
- // Name returns route name for the given request, if any.
- func (self *server) Name(r *http.Request) (name string) {
- var match mux.RouteMatch
- if self.mux.Match(r, &match) {
- name = match.Route.GetName()
- }
- return name
- }
-
- // FileServer registers a handler to serve HTTP (GET|HEAD) requests
- // with the contents of file system under the given directory.
- func (self *server) FileServer(prefix, dir string) {
- if abs, err := filepath.Abs(dir); err == nil {
- fs := http.StripPrefix(prefix, http.FileServer(http.Dir(abs)))
- self.mux.PathPrefix(prefix).Handler(fs)
- } else {
- panic("Failed to setup file server: " + err.Error())
- }
- }
-
- // Use add the middleware module into the stack chain.
- func (self *server) Use(modules ...func(http.Handler) http.Handler) {
- self.middleware.stack = append(self.middleware.stack, modules...)
- }
-
- // Get is a shortcut for mux.HandleFunc(pattern, handler).Methods("GET"),
- // it also fetch the full function name of the handler (with package) to name the route.
- func (self *server) Get(pattern string, handler interface{}) {
- self.register(pattern, handler, "GET")
- }
-
- // Head is a shortcut for mux.HandleFunc(pattern, handler).Methods("HEAD")
- // it also fetch the full function name of the handler (with package) to name the route.
- func (self *server) Head(pattern string, handler interface{}) {
- self.register(pattern, handler, "HEAD")
- }
-
- // Options is a shortcut for mux.HandleFunc(pattern, handler).Methods("OPTIONS")
- // it also fetch the full function name of the handler (with package) to name the route.
- // NOTE method OPTIONS is **NOT** cachable, beware of what you are going to do.
- func (self *server) Options(pattern string, handler interface{}) {
- self.register(pattern, handler, "OPTIONS")
- }
-
- // POST is a shortcut for mux.HandleFunc(pattern, handler).Methods("POST")
- // it also fetch the full function name of the handler (with package) to name the route.
- func (self *server) Post(pattern string, handler interface{}) {
- self.register(pattern, handler, "POST")
- }
-
- // Put is a shortcut for mux.HandleFunc(pattern, handler).Methods("PUT")
- // it also fetch the full function name of the handler (with package) to name the route.
- func (self *server) Put(pattern string, handler interface{}) {
- self.register(pattern, handler, "PUT")
- }
-
- // Delete is a shortcut for mux.HandleFunc(pattern, handler).Methods("DELETE")
- // it also fetch the full function name of the handler (with package) to name the route.
- func (self *server) Delete(pattern string, handler interface{}) {
- self.register(pattern, handler, "DELETE")
- }
-
- // Trace is a shortcut for mux.HandleFunc(pattern, handler).Methods("TRACE")
- // it also fetch the full function name of the handler (with package) to name the route.
- func (self *server) Trace(pattern string, handler interface{}) {
- self.register(pattern, handler, "TRACE")
- }
-
- // Connect is a shortcut for mux.HandleFunc(pattern, handler).Methods("CONNECT")
- // it also fetch the full function name of the handler (with package) to name the route.
- func (self *server) Connect(pattern string, handler interface{}) {
- self.register(pattern, handler, "CONNECT")
- }
-
- // ServeHTTP dispatches the request to the handler whose
- // pattern most closely matches the request URL.
- func (self *server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
- self.build().ServeHTTP(w, r)
- }
-
- // Sends the HTTP response in JSON.
- func (self *server) Send(w http.ResponseWriter, v interface{}) {
- var buffer = new(bytes.Buffer)
-
- w.Header().Set("Content-type", "application/json; charset=utf-8")
- if err := json.NewEncoder(buffer).Encode(v); err == nil {
- w.Write(buffer.Bytes())
- } else {
- http.Error(w, err.Error(), http.StatusInternalServerError)
- }
- }
-
- // Renders a view (Pongo) and sends the rendered HTML string to the client.
- // Optional parameters: value, local variables for the view.
- // func (self *server) Render(filename string, v ...interface{}) {}
-
- // Run starts the application server to serve incoming requests at the given address.
- func (self *server) Run() {
- runtime.GOMAXPROCS(maxprocs)
-
- go func() {
- time.Sleep(500 * time.Millisecond)
- log.Infof("Application server is listening at %d", port)
- }()
-
- if err := http.ListenAndServe(fmt.Sprintf(":%d", port), self); err != nil {
- log.Fatalf("Failed to start the server: %v", err)
- }
- }
-
- // Vars returns the route variables for the current request, if any.
- func (self *server) Vars(r *http.Request) map[string]string {
- return mux.Vars(r)
- }
|