/* ---------------------------------------------------------------------- * ______ ___ __ * / ____/___ / | ____ __ ___ __/ /_ ___ ________ * / / __/ __ \/ /| | / __ \/ / / / | /| / / __ \/ _ \/ ___/ _ \ * / /_/ / /_/ / ___ |/ / / / /_/ /| |/ |/ / / / / __/ / / __/ * \____/\____/_/ |_/_/ /_/\__. / |__/|__/_/ /_/\___/_/ \___/ * /____/ * * (C) Copyright 2015 GoAnywhere (http://goanywhere.io). * ---------------------------------------------------------------------- * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * ----------------------------------------------------------------------*/ package rex import ( "fmt" "net/http" "path/filepath" "reflect" "runtime" "time" "github.com/goanywhere/rex/internal" "github.com/gorilla/mux" ) type Router struct { mod *internal.Module mux *mux.Router ready bool subrouters []*Router } func New() *Router { return &Router{ mod: new(internal.Module), mux: mux.NewRouter().StrictSlash(true), } } // build constructs all router/subrouters along with their middleware modules chain. func (self *Router) build() http.Handler { if !self.ready { self.ready = true // * activate router's middleware modules. self.mod.Use(self.mux) // * activate subrouters's middleware modules. for index := 0; index < len(self.subrouters); index++ { sr := self.subrouters[index] sr.mod.Use(sr.mux) } } return self.mod } // register adds the http.Handler/http.HandleFunc into Gorilla mux. func (self *Router) register(method string, pattern string, handler interface{}) { // 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(method).Name(name) case func(http.ResponseWriter, *http.Request): self.mux.HandleFunc(pattern, H).Methods(method).Name(name) default: Fatalf("Unsupported handler (%s) passed in.", name) } } // 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 *Router) Get(pattern string, handler interface{}) { self.register("GET", pattern, handler) } // 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 *Router) Head(pattern string, handler interface{}) { self.register("HEAD", pattern, handler) } // 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 *Router) Options(pattern string, handler interface{}) { self.register("OPTIONS", pattern, handler) } // 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 *Router) Post(pattern string, handler interface{}) { self.register("POST", pattern, handler) } // 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 *Router) Put(pattern string, handler interface{}) { self.register("PUT", pattern, handler) } // 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 *Router) Delete(pattern string, handler interface{}) { self.register("Delete", pattern, handler) } // Group creates a new application group under the given path prefix. func (self *Router) Group(prefix string) *Router { var mod = new(internal.Module) self.mux.PathPrefix(prefix).Handler(mod) var mux = self.mux.PathPrefix(prefix).Subrouter() router := &Router{mod: mod, mux: mux} self.subrouters = append(self.subrouters, router) return router } // FileServer registers a handler to serve HTTP requests // with the contents of the file system rooted at root. func (self *Router) FileServer(prefix, dir string) { if abs, err := filepath.Abs(dir); err == nil { Infof("FS: %s", abs) fs := http.StripPrefix(prefix, http.FileServer(http.Dir(abs))) self.mux.PathPrefix(prefix).Handler(fs) } else { Fatalf("Failed to setup file server: %v", err) } } // Name returns route name for the given request, if any. func (self *Router) Name(r *http.Request) (name string) { var match mux.RouteMatch if self.mux.Match(r, &match) { name = match.Route.GetName() } return name } func (self *Router) Use(module interface{}) { self.mod.Use(module) } // ServeHTTP dispatches the request to the handler whose // pattern most closely matches the request URL. func (self *Router) ServeHTTP(w http.ResponseWriter, r *http.Request) { self.build().ServeHTTP(w, r) } // Run starts the application server to serve incoming requests at the given address. func (self *Router) Run() { runtime.GOMAXPROCS(config.maxprocs) go func() { time.Sleep(500 * time.Millisecond) Infof("Application server is listening at %d", config.port) }() if err := http.ListenAndServe(fmt.Sprintf(":%d", config.port), self); err != nil { Fatalf("Failed to start the server: %v", err) } }