1
0
mirror of https://github.com/pocketbase/pocketbase.git synced 2025-01-10 00:43:36 +02:00
pocketbase/apis/admin.go

356 lines
10 KiB
Go
Raw Normal View History

2022-07-06 23:19:05 +02:00
package apis
import (
"log"
"net/http"
"github.com/labstack/echo/v5"
"github.com/pocketbase/pocketbase/core"
"github.com/pocketbase/pocketbase/forms"
"github.com/pocketbase/pocketbase/models"
"github.com/pocketbase/pocketbase/tokens"
"github.com/pocketbase/pocketbase/tools/routine"
"github.com/pocketbase/pocketbase/tools/search"
)
2022-10-30 10:28:14 +02:00
// bindAdminApi registers the admin api endpoints and the corresponding handlers.
func bindAdminApi(app core.App, rg *echo.Group) {
2022-07-06 23:19:05 +02:00
api := adminApi{app: app}
subGroup := rg.Group("/admins", ActivityLogger(app))
2023-01-07 22:25:56 +02:00
subGroup.POST("/auth-with-password", api.authWithPassword)
2022-07-06 23:19:05 +02:00
subGroup.POST("/request-password-reset", api.requestPasswordReset)
subGroup.POST("/confirm-password-reset", api.confirmPasswordReset)
2022-10-30 10:28:14 +02:00
subGroup.POST("/auth-refresh", api.authRefresh, RequireAdminAuth())
2022-07-06 23:19:05 +02:00
subGroup.GET("", api.list, RequireAdminAuth())
subGroup.POST("", api.create, RequireAdminAuthOnlyIfAny(app))
2022-07-06 23:19:05 +02:00
subGroup.GET("/:id", api.view, RequireAdminAuth())
subGroup.PATCH("/:id", api.update, RequireAdminAuth())
subGroup.DELETE("/:id", api.delete, RequireAdminAuth())
}
type adminApi struct {
app core.App
}
2023-05-29 20:50:07 +02:00
func (api *adminApi) authResponse(c echo.Context, admin *models.Admin, finalizers ...func(token string) error) error {
2022-07-06 23:19:05 +02:00
token, tokenErr := tokens.NewAdminAuthToken(api.app, admin)
if tokenErr != nil {
2022-10-30 10:28:14 +02:00
return NewBadRequestError("Failed to create auth token.", tokenErr)
2022-07-06 23:19:05 +02:00
}
2023-05-29 20:50:07 +02:00
for _, f := range finalizers {
if err := f(token); err != nil {
return err
}
}
event := new(core.AdminAuthEvent)
event.HttpContext = c
event.Admin = admin
event.Token = token
2022-07-06 23:19:05 +02:00
return api.app.OnAdminAuthRequest().Trigger(event, func(e *core.AdminAuthEvent) error {
2023-07-20 09:40:03 +02:00
if e.HttpContext.Response().Committed {
return nil
}
2022-07-06 23:19:05 +02:00
return e.HttpContext.JSON(200, map[string]any{
"token": e.Token,
"admin": e.Admin,
})
})
}
2022-10-30 10:28:14 +02:00
func (api *adminApi) authRefresh(c echo.Context) error {
2022-07-06 23:19:05 +02:00
admin, _ := c.Get(ContextAdminKey).(*models.Admin)
if admin == nil {
2022-10-30 10:28:14 +02:00
return NewNotFoundError("Missing auth admin context.", nil)
2022-07-06 23:19:05 +02:00
}
event := new(core.AdminAuthRefreshEvent)
event.HttpContext = c
event.Admin = admin
2023-05-29 20:50:07 +02:00
return api.app.OnAdminBeforeAuthRefreshRequest().Trigger(event, func(e *core.AdminAuthRefreshEvent) error {
2023-07-18 14:31:36 +02:00
return api.app.OnAdminAfterAuthRefreshRequest().Trigger(event, func(e *core.AdminAuthRefreshEvent) error {
return api.authResponse(e.HttpContext, e.Admin)
})
2023-05-29 20:50:07 +02:00
})
2022-07-06 23:19:05 +02:00
}
2022-10-30 10:28:14 +02:00
func (api *adminApi) authWithPassword(c echo.Context) error {
2022-07-06 23:19:05 +02:00
form := forms.NewAdminLogin(api.app)
if err := c.Bind(form); err != nil {
return NewBadRequestError("An error occurred while loading the submitted data.", err)
2022-07-06 23:19:05 +02:00
}
event := new(core.AdminAuthWithPasswordEvent)
event.HttpContext = c
event.Password = form.Password
event.Identity = form.Identity
2022-07-06 23:19:05 +02:00
_, submitErr := form.Submit(func(next forms.InterceptorNextFunc[*models.Admin]) forms.InterceptorNextFunc[*models.Admin] {
return func(admin *models.Admin) error {
event.Admin = admin
return api.app.OnAdminBeforeAuthWithPasswordRequest().Trigger(event, func(e *core.AdminAuthWithPasswordEvent) error {
if err := next(e.Admin); err != nil {
return NewBadRequestError("Failed to authenticate.", err)
}
2023-07-18 14:31:36 +02:00
return api.app.OnAdminAfterAuthWithPasswordRequest().Trigger(event, func(e *core.AdminAuthWithPasswordEvent) error {
return api.authResponse(e.HttpContext, e.Admin)
})
})
}
})
return submitErr
2022-07-06 23:19:05 +02:00
}
func (api *adminApi) requestPasswordReset(c echo.Context) error {
form := forms.NewAdminPasswordResetRequest(api.app)
if err := c.Bind(form); err != nil {
2022-10-30 10:28:14 +02:00
return NewBadRequestError("An error occurred while loading the submitted data.", err)
2022-07-06 23:19:05 +02:00
}
if err := form.Validate(); err != nil {
2022-10-30 10:28:14 +02:00
return NewBadRequestError("An error occurred while validating the form.", err)
2022-07-06 23:19:05 +02:00
}
event := new(core.AdminRequestPasswordResetEvent)
event.HttpContext = c
submitErr := form.Submit(func(next forms.InterceptorNextFunc[*models.Admin]) forms.InterceptorNextFunc[*models.Admin] {
return func(Admin *models.Admin) error {
event.Admin = Admin
return api.app.OnAdminBeforeRequestPasswordResetRequest().Trigger(event, func(e *core.AdminRequestPasswordResetEvent) error {
// run in background because we don't need to show the result to the client
routine.FireAndForget(func() {
if err := next(e.Admin); err != nil && api.app.IsDebug() {
2023-05-29 20:50:07 +02:00
// @todo replace after logs generalization
log.Println(err)
}
})
2023-07-18 14:31:36 +02:00
return api.app.OnAdminAfterRequestPasswordResetRequest().Trigger(event, func(e *core.AdminRequestPasswordResetEvent) error {
2023-07-20 09:40:03 +02:00
if e.HttpContext.Response().Committed {
return nil
}
2023-07-18 14:31:36 +02:00
return e.HttpContext.NoContent(http.StatusNoContent)
})
})
2022-07-06 23:19:05 +02:00
}
})
2023-05-29 20:50:07 +02:00
// eagerly write 204 response and skip submit errors
// as a measure against admins enumeration
if !c.Response().Committed {
c.NoContent(http.StatusNoContent)
}
2023-05-29 20:50:07 +02:00
return submitErr
2022-07-06 23:19:05 +02:00
}
func (api *adminApi) confirmPasswordReset(c echo.Context) error {
form := forms.NewAdminPasswordResetConfirm(api.app)
if readErr := c.Bind(form); readErr != nil {
2022-10-30 10:28:14 +02:00
return NewBadRequestError("An error occurred while loading the submitted data.", readErr)
2022-07-06 23:19:05 +02:00
}
event := new(core.AdminConfirmPasswordResetEvent)
event.HttpContext = c
2022-07-06 23:19:05 +02:00
_, submitErr := form.Submit(func(next forms.InterceptorNextFunc[*models.Admin]) forms.InterceptorNextFunc[*models.Admin] {
return func(admin *models.Admin) error {
event.Admin = admin
return api.app.OnAdminBeforeConfirmPasswordResetRequest().Trigger(event, func(e *core.AdminConfirmPasswordResetEvent) error {
if err := next(e.Admin); err != nil {
return NewBadRequestError("Failed to set new password.", err)
}
2023-07-18 14:31:36 +02:00
return api.app.OnAdminAfterConfirmPasswordResetRequest().Trigger(event, func(e *core.AdminConfirmPasswordResetEvent) error {
2023-07-20 09:40:03 +02:00
if e.HttpContext.Response().Committed {
return nil
}
2023-07-18 14:31:36 +02:00
return e.HttpContext.NoContent(http.StatusNoContent)
})
})
}
})
return submitErr
2022-07-06 23:19:05 +02:00
}
func (api *adminApi) list(c echo.Context) error {
fieldResolver := search.NewSimpleFieldResolver(
"id", "created", "updated", "name", "email",
)
admins := []*models.Admin{}
result, err := search.NewProvider(fieldResolver).
Query(api.app.Dao().AdminQuery()).
ParseAndExec(c.QueryParams().Encode(), &admins)
2022-07-06 23:19:05 +02:00
if err != nil {
2022-10-30 10:28:14 +02:00
return NewBadRequestError("", err)
2022-07-06 23:19:05 +02:00
}
event := new(core.AdminsListEvent)
event.HttpContext = c
event.Admins = admins
event.Result = result
2022-07-06 23:19:05 +02:00
return api.app.OnAdminsListRequest().Trigger(event, func(e *core.AdminsListEvent) error {
2023-07-20 09:40:03 +02:00
if e.HttpContext.Response().Committed {
return nil
}
2022-07-06 23:19:05 +02:00
return e.HttpContext.JSON(http.StatusOK, e.Result)
})
}
func (api *adminApi) view(c echo.Context) error {
id := c.PathParam("id")
if id == "" {
2022-10-30 10:28:14 +02:00
return NewNotFoundError("", nil)
2022-07-06 23:19:05 +02:00
}
admin, err := api.app.Dao().FindAdminById(id)
if err != nil || admin == nil {
2022-10-30 10:28:14 +02:00
return NewNotFoundError("", err)
2022-07-06 23:19:05 +02:00
}
event := new(core.AdminViewEvent)
event.HttpContext = c
event.Admin = admin
2022-07-06 23:19:05 +02:00
return api.app.OnAdminViewRequest().Trigger(event, func(e *core.AdminViewEvent) error {
2023-07-20 09:40:03 +02:00
if e.HttpContext.Response().Committed {
return nil
}
2022-07-06 23:19:05 +02:00
return e.HttpContext.JSON(http.StatusOK, e.Admin)
})
}
func (api *adminApi) create(c echo.Context) error {
admin := &models.Admin{}
form := forms.NewAdminUpsert(api.app, admin)
// load request
if err := c.Bind(form); err != nil {
2022-10-30 10:28:14 +02:00
return NewBadRequestError("Failed to load the submitted data due to invalid formatting.", err)
2022-07-06 23:19:05 +02:00
}
event := new(core.AdminCreateEvent)
event.HttpContext = c
event.Admin = admin
2022-07-06 23:19:05 +02:00
// create the admin
submitErr := form.Submit(func(next forms.InterceptorNextFunc[*models.Admin]) forms.InterceptorNextFunc[*models.Admin] {
return func(m *models.Admin) error {
event.Admin = m
return api.app.OnAdminBeforeCreateRequest().Trigger(event, func(e *core.AdminCreateEvent) error {
if err := next(e.Admin); err != nil {
2022-10-30 10:28:14 +02:00
return NewBadRequestError("Failed to create admin.", err)
}
2022-07-06 23:19:05 +02:00
2023-07-18 14:31:36 +02:00
return api.app.OnAdminAfterCreateRequest().Trigger(event, func(e *core.AdminCreateEvent) error {
2023-07-20 09:40:03 +02:00
if e.HttpContext.Response().Committed {
return nil
}
2023-07-18 14:31:36 +02:00
return e.HttpContext.JSON(http.StatusOK, e.Admin)
})
})
}
2022-07-06 23:19:05 +02:00
})
return submitErr
2022-07-06 23:19:05 +02:00
}
func (api *adminApi) update(c echo.Context) error {
id := c.PathParam("id")
if id == "" {
2022-10-30 10:28:14 +02:00
return NewNotFoundError("", nil)
2022-07-06 23:19:05 +02:00
}
admin, err := api.app.Dao().FindAdminById(id)
if err != nil || admin == nil {
2022-10-30 10:28:14 +02:00
return NewNotFoundError("", err)
2022-07-06 23:19:05 +02:00
}
form := forms.NewAdminUpsert(api.app, admin)
// load request
if err := c.Bind(form); err != nil {
2022-10-30 10:28:14 +02:00
return NewBadRequestError("Failed to load the submitted data due to invalid formatting.", err)
2022-07-06 23:19:05 +02:00
}
event := new(core.AdminUpdateEvent)
event.HttpContext = c
event.Admin = admin
2022-07-06 23:19:05 +02:00
// update the admin
submitErr := form.Submit(func(next forms.InterceptorNextFunc[*models.Admin]) forms.InterceptorNextFunc[*models.Admin] {
return func(m *models.Admin) error {
event.Admin = m
return api.app.OnAdminBeforeUpdateRequest().Trigger(event, func(e *core.AdminUpdateEvent) error {
if err := next(e.Admin); err != nil {
2022-10-30 10:28:14 +02:00
return NewBadRequestError("Failed to update admin.", err)
}
2022-07-06 23:19:05 +02:00
2023-07-18 14:31:36 +02:00
return api.app.OnAdminAfterUpdateRequest().Trigger(event, func(e *core.AdminUpdateEvent) error {
2023-07-20 09:40:03 +02:00
if e.HttpContext.Response().Committed {
return nil
}
2023-07-18 14:31:36 +02:00
return e.HttpContext.JSON(http.StatusOK, e.Admin)
})
})
}
2022-07-06 23:19:05 +02:00
})
return submitErr
2022-07-06 23:19:05 +02:00
}
func (api *adminApi) delete(c echo.Context) error {
id := c.PathParam("id")
if id == "" {
2022-10-30 10:28:14 +02:00
return NewNotFoundError("", nil)
2022-07-06 23:19:05 +02:00
}
admin, err := api.app.Dao().FindAdminById(id)
if err != nil || admin == nil {
2022-10-30 10:28:14 +02:00
return NewNotFoundError("", err)
2022-07-06 23:19:05 +02:00
}
event := new(core.AdminDeleteEvent)
event.HttpContext = c
event.Admin = admin
2022-07-06 23:19:05 +02:00
2023-05-29 20:50:07 +02:00
return api.app.OnAdminBeforeDeleteRequest().Trigger(event, func(e *core.AdminDeleteEvent) error {
2022-07-06 23:19:05 +02:00
if err := api.app.Dao().DeleteAdmin(e.Admin); err != nil {
2022-10-30 10:28:14 +02:00
return NewBadRequestError("Failed to delete admin.", err)
2022-07-06 23:19:05 +02:00
}
2023-07-18 14:31:36 +02:00
return api.app.OnAdminAfterDeleteRequest().Trigger(event, func(e *core.AdminDeleteEvent) error {
2023-07-20 09:40:03 +02:00
if e.HttpContext.Response().Committed {
return nil
}
2023-07-18 14:31:36 +02:00
return e.HttpContext.NoContent(http.StatusNoContent)
})
2023-05-29 20:50:07 +02:00
})
2022-07-06 23:19:05 +02:00
}