mirror of
https://github.com/fnproject/fn.git
synced 2022-10-28 21:29:17 +03:00
142 lines
3.9 KiB
Go
142 lines
3.9 KiB
Go
package server
|
|
|
|
import (
|
|
"context"
|
|
"net/http"
|
|
"path"
|
|
"strings"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/fnproject/fn/api"
|
|
"github.com/fnproject/fn/api/models"
|
|
)
|
|
|
|
/* handleRouteCreateOrUpdate is used to handle POST PUT and PATCH for routes.
|
|
Post will only create route if its not there and create app if its not.
|
|
create only
|
|
Post does not skip validation of zero values
|
|
Put will create app if its not there and if route is there update if not it will create new route.
|
|
update if exists or create if not exists
|
|
Put does not skip validation of zero values
|
|
Patch will not create app if it does not exist since the route needs to exist as well...
|
|
update only
|
|
Patch accepts partial updates / skips validation of zero values.
|
|
*/
|
|
func (s *Server) handleRouteCreateOrUpdate(c *gin.Context) {
|
|
ctx := c.Request.Context()
|
|
method := strings.ToUpper(c.Request.Method)
|
|
|
|
var wroute models.RouteWrapper
|
|
|
|
err := s.bindAndValidate(c, method, &wroute)
|
|
if err != nil {
|
|
handleErrorResponse(c, err)
|
|
return
|
|
}
|
|
|
|
// Create the app if it does not exist.
|
|
err = s.ensureApp(ctx, &wroute, method)
|
|
if err != nil {
|
|
handleErrorResponse(c, err)
|
|
return
|
|
}
|
|
|
|
resp, err := s.updateOrInsertRoute(ctx, method, wroute)
|
|
if err != nil {
|
|
handleErrorResponse(c, err)
|
|
return
|
|
}
|
|
|
|
s.cachedelete(resp.Route.AppName, resp.Route.Path)
|
|
|
|
c.JSON(http.StatusOK, resp)
|
|
}
|
|
|
|
// ensureApp will only execute if it is on post or put. Patch is not allowed to create apps.
|
|
func (s *Server) ensureApp(ctx context.Context, wroute *models.RouteWrapper, method string) error {
|
|
if !(method == http.MethodPost || method == http.MethodPut) {
|
|
return nil
|
|
}
|
|
app, err := s.Datastore.GetApp(ctx, wroute.Route.AppName)
|
|
if err != nil && err != models.ErrAppsNotFound {
|
|
return err
|
|
} else if app == nil {
|
|
// Create a new application
|
|
newapp := &models.App{Name: wroute.Route.AppName}
|
|
if err = newapp.Validate(); err != nil {
|
|
return err
|
|
}
|
|
|
|
err = s.FireBeforeAppCreate(ctx, newapp)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = s.Datastore.InsertApp(ctx, newapp)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = s.FireAfterAppCreate(ctx, newapp)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
}
|
|
return nil
|
|
}
|
|
|
|
/* bindAndValidate binds the RouteWrapper to the json from the request and validates that it is correct.
|
|
If it is a put or patch it makes sure that the path in the url matches the provideed one in the body.
|
|
Defaults are set and if patch skipZero is true for validating the RouteWrapper
|
|
*/
|
|
func (s *Server) bindAndValidate(c *gin.Context, method string, wroute *models.RouteWrapper) error {
|
|
err := c.BindJSON(wroute)
|
|
if err != nil {
|
|
return models.ErrInvalidJSON
|
|
}
|
|
|
|
if wroute.Route == nil {
|
|
return models.ErrRoutesMissingNew
|
|
}
|
|
wroute.Route.AppName = c.MustGet(api.AppName).(string)
|
|
|
|
if method == http.MethodPut || method == http.MethodPatch {
|
|
p := path.Clean(c.MustGet(api.Path).(string))
|
|
|
|
if wroute.Route.Path != "" && wroute.Route.Path != p {
|
|
return models.ErrRoutesPathImmutable
|
|
}
|
|
wroute.Route.Path = p
|
|
}
|
|
|
|
wroute.Route.SetDefaults()
|
|
|
|
return wroute.Validate(method == http.MethodPatch)
|
|
}
|
|
|
|
// updateOrInsertRoute will either update or insert the route respective the method.
|
|
func (s *Server) updateOrInsertRoute(ctx context.Context, method string, wroute models.RouteWrapper) (routeResponse, error) {
|
|
var route *models.Route
|
|
var err error
|
|
resp := routeResponse{"Route successfully created", nil}
|
|
up := routeResponse{"Route successfully updated", nil}
|
|
|
|
switch method {
|
|
case http.MethodPost:
|
|
route, err = s.Datastore.InsertRoute(ctx, wroute.Route)
|
|
case http.MethodPut:
|
|
route, err = s.Datastore.UpdateRoute(ctx, wroute.Route)
|
|
if err == models.ErrRoutesNotFound {
|
|
// try insert then
|
|
route, err = s.Datastore.InsertRoute(ctx, wroute.Route)
|
|
}
|
|
case http.MethodPatch:
|
|
// When patching if there is an error around the app we will return one and the update fails.
|
|
route, err = s.Datastore.UpdateRoute(ctx, wroute.Route)
|
|
resp = up
|
|
}
|
|
resp.Route = route
|
|
return resp, err
|
|
}
|