mirror of
https://github.com/pocketbase/pocketbase.git
synced 2025-01-10 00:43:36 +02:00
161 lines
4.5 KiB
Go
161 lines
4.5 KiB
Go
package apis
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"net/http"
|
|
|
|
validation "github.com/go-ozzo/ozzo-validation/v4"
|
|
"github.com/labstack/echo/v5"
|
|
"github.com/pocketbase/pocketbase/core"
|
|
"github.com/pocketbase/pocketbase/forms"
|
|
"github.com/pocketbase/pocketbase/models/settings"
|
|
"github.com/pocketbase/pocketbase/tools/security"
|
|
)
|
|
|
|
// bindSettingsApi registers the settings api endpoints.
|
|
func bindSettingsApi(app core.App, rg *echo.Group) {
|
|
api := settingsApi{app: app}
|
|
|
|
subGroup := rg.Group("/settings", ActivityLogger(app), RequireAdminAuth())
|
|
subGroup.GET("", api.list)
|
|
subGroup.PATCH("", api.set)
|
|
subGroup.POST("/test/s3", api.testS3)
|
|
subGroup.POST("/test/email", api.testEmail)
|
|
subGroup.POST("/apple/generate-client-secret", api.generateAppleClientSecret)
|
|
}
|
|
|
|
type settingsApi struct {
|
|
app core.App
|
|
}
|
|
|
|
func (api *settingsApi) list(c echo.Context) error {
|
|
settings, err := api.app.Settings().RedactClone()
|
|
if err != nil {
|
|
return NewBadRequestError("", err)
|
|
}
|
|
|
|
event := new(core.SettingsListEvent)
|
|
event.HttpContext = c
|
|
event.RedactedSettings = settings
|
|
|
|
return api.app.OnSettingsListRequest().Trigger(event, func(e *core.SettingsListEvent) error {
|
|
return e.HttpContext.JSON(http.StatusOK, e.RedactedSettings)
|
|
})
|
|
}
|
|
|
|
func (api *settingsApi) set(c echo.Context) error {
|
|
form := forms.NewSettingsUpsert(api.app)
|
|
|
|
// load request
|
|
if err := c.Bind(form); err != nil {
|
|
return NewBadRequestError("An error occurred while loading the submitted data.", err)
|
|
}
|
|
|
|
event := new(core.SettingsUpdateEvent)
|
|
event.HttpContext = c
|
|
event.OldSettings = api.app.Settings()
|
|
|
|
// update the settings
|
|
submitErr := form.Submit(func(next forms.InterceptorNextFunc[*settings.Settings]) forms.InterceptorNextFunc[*settings.Settings] {
|
|
return func(s *settings.Settings) error {
|
|
event.NewSettings = s
|
|
|
|
return api.app.OnSettingsBeforeUpdateRequest().Trigger(event, func(e *core.SettingsUpdateEvent) error {
|
|
if err := next(e.NewSettings); err != nil {
|
|
return NewBadRequestError("An error occurred while submitting the form.", err)
|
|
}
|
|
|
|
redactedSettings, err := api.app.Settings().RedactClone()
|
|
if err != nil {
|
|
return NewBadRequestError("", err)
|
|
}
|
|
|
|
return e.HttpContext.JSON(http.StatusOK, redactedSettings)
|
|
})
|
|
}
|
|
})
|
|
|
|
if submitErr == nil {
|
|
if err := api.app.OnSettingsAfterUpdateRequest().Trigger(event); err != nil && api.app.IsDebug() {
|
|
log.Println(err)
|
|
}
|
|
}
|
|
|
|
return submitErr
|
|
}
|
|
|
|
func (api *settingsApi) testS3(c echo.Context) error {
|
|
if !api.app.Settings().S3.Enabled {
|
|
return NewBadRequestError("S3 storage is not enabled.", nil)
|
|
}
|
|
|
|
fs, err := api.app.NewFilesystem()
|
|
if err != nil {
|
|
return NewBadRequestError("Failed to initialize the S3 storage. Raw error: \n"+err.Error(), nil)
|
|
}
|
|
defer fs.Close()
|
|
|
|
testPrefix := "pb_settings_test_" + security.PseudorandomString(5)
|
|
testFileKey := testPrefix + "/test.txt"
|
|
|
|
// try to upload a test file
|
|
if err := fs.Upload([]byte("test"), testFileKey); err != nil {
|
|
return NewBadRequestError("Failed to upload a test file. Raw error: \n"+err.Error(), nil)
|
|
}
|
|
|
|
// test prefix deletion (ensures that both bucket list and delete works)
|
|
if errs := fs.DeletePrefix(testPrefix); len(errs) > 0 {
|
|
return NewBadRequestError(fmt.Sprintf("Failed to delete a test file. Raw error: %v", errs), nil)
|
|
}
|
|
|
|
return c.NoContent(http.StatusNoContent)
|
|
}
|
|
|
|
func (api *settingsApi) testEmail(c echo.Context) error {
|
|
form := forms.NewTestEmailSend(api.app)
|
|
|
|
// load request
|
|
if err := c.Bind(form); err != nil {
|
|
return NewBadRequestError("An error occurred while loading the submitted data.", err)
|
|
}
|
|
|
|
// send
|
|
if err := form.Submit(); err != nil {
|
|
// form error
|
|
if fErr, ok := err.(validation.Errors); ok {
|
|
return NewBadRequestError("Failed to send the test email.", fErr)
|
|
}
|
|
|
|
// mailer error
|
|
return NewBadRequestError("Failed to send the test email. Raw error: \n"+err.Error(), nil)
|
|
}
|
|
|
|
return c.NoContent(http.StatusNoContent)
|
|
}
|
|
|
|
func (api *settingsApi) generateAppleClientSecret(c echo.Context) error {
|
|
form := forms.NewAppleClientSecretCreate(api.app)
|
|
|
|
// load request
|
|
if err := c.Bind(form); err != nil {
|
|
return NewBadRequestError("An error occurred while loading the submitted data.", err)
|
|
}
|
|
|
|
// generate
|
|
secret, err := form.Submit()
|
|
if err != nil {
|
|
// form error
|
|
if fErr, ok := err.(validation.Errors); ok {
|
|
return NewBadRequestError("Invalid client secret data.", fErr)
|
|
}
|
|
|
|
// secret generation error
|
|
return NewBadRequestError("Failed to generate client secret. Raw error: \n"+err.Error(), nil)
|
|
}
|
|
|
|
return c.JSON(http.StatusOK, map[string]any{
|
|
"secret": secret,
|
|
})
|
|
}
|