Remove reflect; Improve defaults
This commit is contained in:
		
							parent
							
								
									283c75b32e
								
							
						
					
					
						commit
						b6045aa53d
					
				
					 6 changed files with 62 additions and 80 deletions
				
			
		
							
								
								
									
										11
									
								
								context.go
									
										
									
									
									
								
							
							
						
						
									
										11
									
								
								context.go
									
										
									
									
									
								
							|  | @ -25,6 +25,13 @@ func NewContext(router *Router, res http.ResponseWriter, req *http.Request, para | |||
| 	return &Context{router, req, res, param.ByName, make(map[string]interface{})} | ||||
| } | ||||
| 
 | ||||
| func (c *Context) Validate(v interface{}) bool { | ||||
| 	if c.router.Validator == nil { | ||||
| 		return true | ||||
| 	} | ||||
| 	return c.router.Validator(c, v) | ||||
| } | ||||
| 
 | ||||
| // QueryParam returns the specified parameter from the query string. | ||||
| // Returns an empty string if it doesn't exist. Returns the first parameter if multiple instances exist | ||||
| func (c *Context) QueryParam(param string) string { | ||||
|  | @ -57,6 +64,10 @@ func (c *Context) String(code int, s string) error { | |||
| 	return err | ||||
| } | ||||
| 
 | ||||
| func (c *Context) BadRequest(reason string) error { | ||||
| 	return c.router.BadRequestFormatter(c, reason) | ||||
| } | ||||
| 
 | ||||
| // StatusText returns the given status code with the matching status text | ||||
| func (c *Context) StatusText(code int) error { | ||||
| 	return c.String(code, http.StatusText(code)) | ||||
|  |  | |||
							
								
								
									
										22
									
								
								default.go
									
										
									
									
									
								
							
							
						
						
									
										22
									
								
								default.go
									
										
									
									
									
								
							|  | @ -2,6 +2,7 @@ package router | |||
| 
 | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| ) | ||||
| 
 | ||||
|  | @ -14,14 +15,27 @@ func defaultMethodNotAllowedHandler(c *Context) error { | |||
| } | ||||
| 
 | ||||
| func defaultErrorHandler(c *Context, err interface{}) { | ||||
| 	fmt.Printf("%s '%s': %s\n", c.Request.Method, c.Request.URL, err) | ||||
| 
 | ||||
| 	_ = c.StatusText(http.StatusInternalServerError) | ||||
| } | ||||
| 
 | ||||
| func defaultReader(c *Context, dst interface{}) (bool, error) { | ||||
| 	err := json.NewDecoder(c.Request.Body).Decode(dst) | ||||
| func defaultBadRequestFormatter(c *Context, reason string) error { | ||||
| 	return c.String(http.StatusBadRequest, reason) | ||||
| } | ||||
| 
 | ||||
| func BodyJSON[T any](handle func(*Context, T) error) Handle { | ||||
| 	return func(c *Context) error { | ||||
| 		var dst T | ||||
| 		err := json.NewDecoder(c.Request.Body).Decode(&dst) | ||||
| 		if err != nil { | ||||
| 		return false, c.StatusText(http.StatusBadRequest) | ||||
| 			return c.BadRequest(`Invalid JSON`) | ||||
| 		} | ||||
| 
 | ||||
| 	return true, nil | ||||
| 		if !c.Validate(dst) { | ||||
| 			return nil | ||||
| 		} | ||||
| 
 | ||||
| 		return handle(c, dst) | ||||
| 	} | ||||
| } | ||||
|  |  | |||
							
								
								
									
										5
									
								
								go.mod
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								go.mod
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,5 @@ | |||
| module git.fuyu.moe/Fuyu/router | ||||
| 
 | ||||
| go 1.19 | ||||
| 
 | ||||
| require github.com/julienschmidt/httprouter v1.3.0 | ||||
							
								
								
									
										2
									
								
								go.sum
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								go.sum
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | |||
| github.com/julienschmidt/httprouter v1.3.0 h1:U0609e9tgbseu3rBINet9P48AI/D3oJs4dN7jwJOQ1U= | ||||
| github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM= | ||||
							
								
								
									
										6
									
								
								group.go
									
										
									
									
									
								
							
							
						
						
									
										6
									
								
								group.go
									
										
									
									
									
								
							|  | @ -24,7 +24,7 @@ func (g *Group) GET(path string, handle Handle, middleware ...Middleware) { | |||
| } | ||||
| 
 | ||||
| // POST adds a POST route | ||||
| func (g *Group) POST(path string, handle interface{}, middleware ...Middleware) { | ||||
| func (g *Group) POST(path string, handle Handle, middleware ...Middleware) { | ||||
| 	g.router.POST(join(g.prefix, path), handle, append(g.middleware, middleware...)...) | ||||
| } | ||||
| 
 | ||||
|  | @ -34,12 +34,12 @@ func (g *Group) DELETE(path string, handle Handle, middleware ...Middleware) { | |||
| } | ||||
| 
 | ||||
| // PUT adds a PUT route | ||||
| func (g *Group) PUT(path string, handle interface{}, middleware ...Middleware) { | ||||
| func (g *Group) PUT(path string, handle Handle, middleware ...Middleware) { | ||||
| 	g.router.PUT(join(g.prefix, path), handle, append(g.middleware, middleware...)...) | ||||
| } | ||||
| 
 | ||||
| // PATCH adds a PATCH route | ||||
| func (g *Group) PATCH(path string, handle interface{}, middleware ...Middleware) { | ||||
| func (g *Group) PATCH(path string, handle Handle, middleware ...Middleware) { | ||||
| 	g.router.PATCH(join(g.prefix, path), handle, append(g.middleware, middleware...)...) | ||||
| } | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										90
									
								
								router.go
									
										
									
									
									
								
							
							
						
						
									
										90
									
								
								router.go
									
										
									
									
									
								
							|  | @ -4,7 +4,6 @@ import ( | |||
| 	"context" | ||||
| 	"crypto/tls" | ||||
| 	"net/http" | ||||
| 	"reflect" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/julienschmidt/httprouter" | ||||
|  | @ -13,7 +12,7 @@ import ( | |||
| type route struct { | ||||
| 	Method     string | ||||
| 	Path       string | ||||
| 	Handle     interface{} | ||||
| 	Handle     Handle | ||||
| 	Middleware []Middleware | ||||
| } | ||||
| 
 | ||||
|  | @ -23,28 +22,38 @@ type Handle = func(*Context) error | |||
| // ErrorHandle handles a request | ||||
| type ErrorHandle func(*Context, interface{}) | ||||
| 
 | ||||
| // BadRequestFormatter formats a bad request | ||||
| type BadRequestFormatter func(*Context, string) error | ||||
| 
 | ||||
| // Validator validates data. | ||||
| // If validation fails it should write a response and return false | ||||
| type Validator func(*Context, interface{}) bool | ||||
| 
 | ||||
| // Middleware is a function that runs before your route, it gets the next handler as a parameter | ||||
| type Middleware func(Handle) Handle | ||||
| 
 | ||||
| // Reader reads input to dst, returns true if successful | ||||
| type Reader func(c *Context, dst interface{}) (bool, error) | ||||
| 
 | ||||
| // Router is the router itself | ||||
| type Router struct { | ||||
| 	routes                  []route | ||||
| 	Reader                  Reader | ||||
| 	Renderer                Renderer | ||||
| 	middleware              []Middleware | ||||
| 	NotFoundHandler         Handle | ||||
| 	MethodNotAllowedHandler Handle | ||||
| 	ErrorHandler            ErrorHandle | ||||
| 	BadRequestFormatter     BadRequestFormatter | ||||
| 	Validator               Validator | ||||
| 	TrimTrailingSlashes     bool | ||||
| 	server                  *http.Server | ||||
| } | ||||
| 
 | ||||
| // New returns a new Router | ||||
| func New() *Router { | ||||
| 	return &Router{Reader: defaultReader, NotFoundHandler: defaultNotFoundHandler, MethodNotAllowedHandler: defaultMethodNotAllowedHandler, ErrorHandler: defaultErrorHandler} | ||||
| 	return &Router{ | ||||
| 		NotFoundHandler:         defaultNotFoundHandler, | ||||
| 		MethodNotAllowedHandler: defaultMethodNotAllowedHandler, | ||||
| 		ErrorHandler:            defaultErrorHandler, | ||||
| 		BadRequestFormatter:     defaultBadRequestFormatter, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // Use adds a global middleware | ||||
|  | @ -63,8 +72,7 @@ func (r *Router) GET(path string, handle Handle, middleware ...Middleware) { | |||
| } | ||||
| 
 | ||||
| // POST adds a POST route | ||||
| func (r *Router) POST(path string, handle interface{}, middleware ...Middleware) { | ||||
| 	checkInterfaceHandle(handle) | ||||
| func (r *Router) POST(path string, handle Handle, middleware ...Middleware) { | ||||
| 	r.routes = append(r.routes, route{`POST`, path, handle, middleware}) | ||||
| } | ||||
| 
 | ||||
|  | @ -74,14 +82,12 @@ func (r *Router) DELETE(path string, handle Handle, middleware ...Middleware) { | |||
| } | ||||
| 
 | ||||
| // PUT adds a PUT route | ||||
| func (r *Router) PUT(path string, handle interface{}, middleware ...Middleware) { | ||||
| 	checkInterfaceHandle(handle) | ||||
| func (r *Router) PUT(path string, handle Handle, middleware ...Middleware) { | ||||
| 	r.routes = append(r.routes, route{`PUT`, path, handle, middleware}) | ||||
| } | ||||
| 
 | ||||
| // PATCH adds a PATCH route | ||||
| func (r *Router) PATCH(path string, handle interface{}, middleware ...Middleware) { | ||||
| 	checkInterfaceHandle(handle) | ||||
| func (r *Router) PATCH(path string, handle Handle, middleware ...Middleware) { | ||||
| 	r.routes = append(r.routes, route{`PATCH`, path, handle, middleware}) | ||||
| } | ||||
| 
 | ||||
|  | @ -132,16 +138,11 @@ func (r *Router) getHttpr() http.Handler { | |||
| 	httpr := httprouter.New() | ||||
| 
 | ||||
| 	for _, v := range r.routes { | ||||
| 		handle, ok := v.Handle.(Handle) | ||||
| 		if !ok { | ||||
| 			handle = handlePOST(r, v.Handle) | ||||
| 		} | ||||
| 
 | ||||
| 		middleware := make([]Middleware, len(r.middleware)+len(v.Middleware)) | ||||
| 		copy(middleware, r.middleware) | ||||
| 		copy(middleware[len(r.middleware):], v.Middleware) | ||||
| 
 | ||||
| 		httpr.Handle(v.Method, v.Path, handleReq(r, handle, middleware)) | ||||
| 		httpr.Handle(v.Method, v.Path, handleReq(r, v.Handle, middleware)) | ||||
| 	} | ||||
| 
 | ||||
| 	httpr.NotFound = http.HandlerFunc(func(res http.ResponseWriter, req *http.Request) { | ||||
|  | @ -172,56 +173,6 @@ func (r *Router) getHttpr() http.Handler { | |||
| 	return httpr | ||||
| } | ||||
| 
 | ||||
| func checkInterfaceHandle(f interface{}) { | ||||
| 	if _, ok := f.(Handle); ok { | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	rt := reflect.TypeOf(f) | ||||
| 
 | ||||
| 	if rt.Kind() != reflect.Func { | ||||
| 		panic(`non-func handle`) | ||||
| 	} | ||||
| 
 | ||||
| 	if rt.NumIn() != 2 { | ||||
| 		panic(`handle should take 2 arguments`) | ||||
| 	} | ||||
| 
 | ||||
| 	if rt.NumOut() != 1 || rt.Out(0).Name() != `error` { | ||||
| 		panic(`handle should return only error`) | ||||
| 	} | ||||
| 
 | ||||
| 	if rt.In(0) != reflect.TypeOf(&Context{}) { | ||||
| 		panic(`handle should accept Context as first argument`) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func handlePOST(r *Router, f interface{}) Handle { | ||||
| 	funcRv, inputRt := reflect.ValueOf(f), reflect.TypeOf(f).In(1) | ||||
| 
 | ||||
| 	return func(c *Context) error { | ||||
| 		data := reflect.New(inputRt) | ||||
| 
 | ||||
| 		if r.Reader != nil { | ||||
| 			ok, err := r.Reader(c, data.Interface()) | ||||
| 			_ = c.Request.Body.Close() | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 			if !ok { | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		out := funcRv.Call([]reflect.Value{reflect.ValueOf(c), data.Elem()}) | ||||
| 
 | ||||
| 		if out[0].IsNil() { | ||||
| 			return nil | ||||
| 		} | ||||
| 		return out[0].Interface().(error) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func handleReq(r *Router, handle Handle, m []Middleware) httprouter.Handle { | ||||
| 	return func(res http.ResponseWriter, req *http.Request, param httprouter.Params) { | ||||
| 		c := NewContext(r, res, req, param) | ||||
|  | @ -232,7 +183,6 @@ func handleReq(r *Router, handle Handle, m []Middleware) httprouter.Handle { | |||
| 		} | ||||
| 
 | ||||
| 		err := f(c) | ||||
| 
 | ||||
| 		if err != nil { | ||||
| 			r.ErrorHandler(c, err) | ||||
| 		} | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue