mirror of
https://github.com/fnproject/fn.git
synced 2022-10-28 21:29:17 +03:00
* Extend extension mechanism to support per-route API extensions * Tidy up comment * Remove print statement * Minor improvement to README * Avoid calling c.Request.Context() twice
143 lines
4.2 KiB
Go
143 lines
4.2 KiB
Go
// TODO: it would be nice to move these into the top level folder so people can use these with the "functions" package, eg: functions.ApiHandler
|
|
package server
|
|
|
|
import (
|
|
"net/http"
|
|
|
|
"github.com/fnproject/fn/api"
|
|
"github.com/fnproject/fn/api/models"
|
|
"github.com/gin-gonic/gin"
|
|
)
|
|
|
|
type ApiHandlerFunc func(w http.ResponseWriter, r *http.Request)
|
|
|
|
// ServeHTTP calls f(w, r).
|
|
func (f ApiHandlerFunc) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
f(w, r)
|
|
}
|
|
|
|
type ApiHandler interface {
|
|
// Handle(ctx context.Context)
|
|
ServeHTTP(w http.ResponseWriter, r *http.Request)
|
|
}
|
|
|
|
type ApiAppHandler interface {
|
|
// Handle(ctx context.Context)
|
|
ServeHTTP(w http.ResponseWriter, r *http.Request, app *models.App)
|
|
}
|
|
|
|
type ApiAppHandlerFunc func(w http.ResponseWriter, r *http.Request, app *models.App)
|
|
|
|
// ServeHTTP calls f(w, r).
|
|
func (f ApiAppHandlerFunc) ServeHTTP(w http.ResponseWriter, r *http.Request, app *models.App) {
|
|
f(w, r, app)
|
|
}
|
|
|
|
func (s *Server) apiHandlerWrapperFunc(apiHandler ApiHandler) gin.HandlerFunc {
|
|
return func(c *gin.Context) {
|
|
apiHandler.ServeHTTP(c.Writer, c.Request)
|
|
}
|
|
}
|
|
|
|
func (s *Server) apiAppHandlerWrapperFunc(apiHandler ApiAppHandler) gin.HandlerFunc {
|
|
return func(c *gin.Context) {
|
|
// get the app
|
|
appName := c.Param(api.CApp)
|
|
app, err := s.Datastore.GetApp(c.Request.Context(), appName)
|
|
if err != nil {
|
|
handleErrorResponse(c, err)
|
|
c.Abort()
|
|
return
|
|
}
|
|
if app == nil {
|
|
handleErrorResponse(c, models.ErrAppsNotFound)
|
|
c.Abort()
|
|
return
|
|
}
|
|
|
|
apiHandler.ServeHTTP(c.Writer, c.Request, app)
|
|
}
|
|
}
|
|
|
|
// per Route
|
|
|
|
type ApiRouteHandler interface {
|
|
// Handle(ctx context.Context)
|
|
ServeHTTP(w http.ResponseWriter, r *http.Request, app *models.App, route *models.Route)
|
|
}
|
|
|
|
type ApiRouteHandlerFunc func(w http.ResponseWriter, r *http.Request, app *models.App, route *models.Route)
|
|
|
|
// ServeHTTP calls f(w, r).
|
|
func (f ApiRouteHandlerFunc) ServeHTTP(w http.ResponseWriter, r *http.Request, app *models.App, route *models.Route) {
|
|
f(w, r, app, route)
|
|
}
|
|
|
|
func (s *Server) apiRouteHandlerWrapperFunc(apiHandler ApiRouteHandler) gin.HandlerFunc {
|
|
return func(c *gin.Context) {
|
|
context := c.Request.Context()
|
|
// get the app
|
|
appName := c.Param(api.CApp)
|
|
app, err := s.Datastore.GetApp(context, appName)
|
|
if err != nil {
|
|
handleErrorResponse(c, err)
|
|
c.Abort()
|
|
return
|
|
}
|
|
if app == nil {
|
|
handleErrorResponse(c, models.ErrAppsNotFound)
|
|
c.Abort()
|
|
return
|
|
}
|
|
// get the route TODO
|
|
routePath := "/" + c.Param(api.CRoute)
|
|
route, err := s.Datastore.GetRoute(context, appName, routePath)
|
|
if err != nil {
|
|
handleErrorResponse(c, err)
|
|
c.Abort()
|
|
return
|
|
}
|
|
if route == nil {
|
|
handleErrorResponse(c, models.ErrRoutesNotFound)
|
|
c.Abort()
|
|
return
|
|
}
|
|
|
|
apiHandler.ServeHTTP(c.Writer, c.Request, app, route)
|
|
}
|
|
}
|
|
|
|
// AddEndpoint adds an endpoint to /v1/x
|
|
func (s *Server) AddEndpoint(method, path string, handler ApiHandler) {
|
|
v1 := s.Router.Group("/v1")
|
|
// v1.GET("/apps/:app/log", logHandler(cfg))
|
|
v1.Handle(method, path, s.apiHandlerWrapperFunc(handler))
|
|
}
|
|
|
|
// AddEndpoint adds an endpoint to /v1/x
|
|
func (s *Server) AddEndpointFunc(method, path string, handler func(w http.ResponseWriter, r *http.Request)) {
|
|
s.AddEndpoint(method, path, ApiHandlerFunc(handler))
|
|
}
|
|
|
|
// AddAppEndpoint adds an endpoints to /v1/apps/:app/x
|
|
func (s *Server) AddAppEndpoint(method, path string, handler ApiAppHandler) {
|
|
v1 := s.Router.Group("/v1")
|
|
v1.Handle(method, "/apps/:app"+path, s.apiAppHandlerWrapperFunc(handler))
|
|
}
|
|
|
|
// AddAppEndpoint adds an endpoints to /v1/apps/:app/x
|
|
func (s *Server) AddAppEndpointFunc(method, path string, handler func(w http.ResponseWriter, r *http.Request, app *models.App)) {
|
|
s.AddAppEndpoint(method, path, ApiAppHandlerFunc(handler))
|
|
}
|
|
|
|
// AddRouteEndpoint adds an endpoints to /v1/apps/:app/routes/:route/x
|
|
func (s *Server) AddRouteEndpoint(method, path string, handler ApiRouteHandler) {
|
|
v1 := s.Router.Group("/v1")
|
|
v1.Handle(method, "/apps/:app/routes/:route"+path, s.apiRouteHandlerWrapperFunc(handler)) // conflicts with existing wildcard
|
|
}
|
|
|
|
// AddRouteEndpoint adds an endpoints to /v1/apps/:app/routes/:route/x
|
|
func (s *Server) AddRouteEndpointFunc(method, path string, handler func(w http.ResponseWriter, r *http.Request, app *models.App, route *models.Route)) {
|
|
s.AddRouteEndpoint(method, path, ApiRouteHandlerFunc(handler))
|
|
}
|