2019-11-08 06:53:40 +01:00
|
|
|
/*
|
2022-04-25 09:53:55 +02:00
|
|
|
Package api provides REST API request handlers.
|
2020-06-23 13:44:14 +02:00
|
|
|
|
2022-04-13 22:17:59 +02:00
|
|
|
Copyright (c) 2018 - 2022 PhotoPrism UG. All rights reserved.
|
2020-06-23 13:44:14 +02:00
|
|
|
|
2022-08-10 16:09:21 +02:00
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under Version 3 of the GNU Affero General Public License (the "AGPL"):
|
|
|
|
<https://docs.photoprism.app/license/agpl>
|
2020-06-23 13:44:14 +02:00
|
|
|
|
2022-08-10 16:09:21 +02:00
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU Affero General Public License for more details.
|
2020-06-23 13:44:14 +02:00
|
|
|
|
2022-08-10 16:09:21 +02:00
|
|
|
The AGPL is supplemented by our Trademark and Brand Guidelines,
|
|
|
|
which describe how our Brand Assets may be used:
|
|
|
|
<https://photoprism.app/trademark>
|
2020-06-23 13:44:14 +02:00
|
|
|
|
2022-04-13 22:17:59 +02:00
|
|
|
Feel free to send an email to hello@photoprism.app if you have questions,
|
2020-06-23 13:44:14 +02:00
|
|
|
want to support our work, or just want to say hello.
|
2019-11-08 06:53:40 +01:00
|
|
|
|
|
|
|
Additional information can be found in our Developer Guide:
|
2022-02-27 17:32:54 +01:00
|
|
|
<https://docs.photoprism.app/developer-guide/>
|
2019-11-08 06:53:40 +01:00
|
|
|
*/
|
|
|
|
package api
|
|
|
|
|
2019-12-02 00:30:58 +01:00
|
|
|
import (
|
2020-07-04 12:54:35 +02:00
|
|
|
"net/http"
|
2022-07-19 16:58:43 +02:00
|
|
|
"strings"
|
2020-07-04 12:54:35 +02:00
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
2021-12-14 18:34:52 +01:00
|
|
|
|
2019-12-02 00:30:58 +01:00
|
|
|
"github.com/photoprism/photoprism/internal/event"
|
2020-07-04 12:54:35 +02:00
|
|
|
"github.com/photoprism/photoprism/internal/i18n"
|
2020-06-25 14:54:04 +02:00
|
|
|
"github.com/photoprism/photoprism/internal/service"
|
2022-04-15 09:42:07 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/clean"
|
2019-12-02 00:30:58 +01:00
|
|
|
)
|
2019-11-08 06:53:40 +01:00
|
|
|
|
2019-12-02 00:30:58 +01:00
|
|
|
var log = event.Log
|
2020-05-25 19:10:44 +02:00
|
|
|
|
2020-05-28 16:26:22 +02:00
|
|
|
func logError(prefix string, err error) {
|
2020-05-25 19:10:44 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Errorf("%s: %s", prefix, err.Error())
|
|
|
|
}
|
|
|
|
}
|
2020-05-27 19:38:40 +02:00
|
|
|
|
2021-10-02 14:24:44 +02:00
|
|
|
func logWarn(prefix string, err error) {
|
|
|
|
if err != nil {
|
|
|
|
log.Warnf("%s: %s", prefix, err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-25 14:54:04 +02:00
|
|
|
func UpdateClientConfig() {
|
|
|
|
conf := service.Config()
|
|
|
|
|
2020-06-25 01:20:58 +02:00
|
|
|
event.Publish("config.updated", event.Data{"config": conf.UserConfig()})
|
2020-05-27 19:38:40 +02:00
|
|
|
}
|
2020-07-04 12:54:35 +02:00
|
|
|
|
|
|
|
func Abort(c *gin.Context, code int, id i18n.Message, params ...interface{}) {
|
|
|
|
resp := i18n.NewResponse(code, id, params...)
|
2020-07-07 10:51:55 +02:00
|
|
|
|
2022-07-19 16:58:43 +02:00
|
|
|
log.Debugf("api-v1: abort %s with code %d (%s)", clean.Log(c.FullPath()), code, strings.ToLower(resp.String()))
|
2020-07-07 10:51:55 +02:00
|
|
|
|
|
|
|
c.AbortWithStatusJSON(code, resp)
|
|
|
|
}
|
|
|
|
|
|
|
|
func Error(c *gin.Context, code int, err error, id i18n.Message, params ...interface{}) {
|
|
|
|
resp := i18n.NewResponse(code, id, params...)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
resp.Details = err.Error()
|
2022-07-19 16:58:43 +02:00
|
|
|
log.Errorf("api-v1: error %s with code %d in %s (%s)", clean.Log(err.Error()), code, clean.Log(c.FullPath()), strings.ToLower(resp.String()))
|
2020-07-07 10:51:55 +02:00
|
|
|
}
|
|
|
|
|
2020-07-04 12:54:35 +02:00
|
|
|
c.AbortWithStatusJSON(code, resp)
|
|
|
|
}
|
|
|
|
|
|
|
|
func AbortUnauthorized(c *gin.Context) {
|
|
|
|
Abort(c, http.StatusUnauthorized, i18n.ErrUnauthorized)
|
|
|
|
}
|
|
|
|
|
|
|
|
func AbortEntityNotFound(c *gin.Context) {
|
|
|
|
Abort(c, http.StatusNotFound, i18n.ErrEntityNotFound)
|
|
|
|
}
|
|
|
|
|
2022-08-01 15:57:19 +02:00
|
|
|
func AbortAlbumNotFound(c *gin.Context) {
|
|
|
|
Abort(c, http.StatusNotFound, i18n.ErrAlbumNotFound)
|
|
|
|
}
|
|
|
|
|
2020-07-04 12:54:35 +02:00
|
|
|
func AbortSaveFailed(c *gin.Context) {
|
|
|
|
Abort(c, http.StatusInternalServerError, i18n.ErrSaveFailed)
|
|
|
|
}
|
|
|
|
|
2020-10-04 14:21:40 +02:00
|
|
|
func AbortDeleteFailed(c *gin.Context) {
|
|
|
|
Abort(c, http.StatusInternalServerError, i18n.ErrDeleteFailed)
|
|
|
|
}
|
|
|
|
|
2020-07-04 12:54:35 +02:00
|
|
|
func AbortUnexpected(c *gin.Context) {
|
|
|
|
Abort(c, http.StatusInternalServerError, i18n.ErrUnexpected)
|
|
|
|
}
|
|
|
|
|
|
|
|
func AbortBadRequest(c *gin.Context) {
|
|
|
|
Abort(c, http.StatusBadRequest, i18n.ErrBadRequest)
|
|
|
|
}
|
|
|
|
|
|
|
|
func AbortAlreadyExists(c *gin.Context, s string) {
|
|
|
|
Abort(c, http.StatusConflict, i18n.ErrAlreadyExists, s)
|
|
|
|
}
|
2020-07-07 10:51:55 +02:00
|
|
|
|
|
|
|
func AbortFeatureDisabled(c *gin.Context) {
|
|
|
|
Abort(c, http.StatusForbidden, i18n.ErrFeatureDisabled)
|
|
|
|
}
|
2021-12-09 02:33:41 +01:00
|
|
|
|
|
|
|
func AbortBusy(c *gin.Context) {
|
|
|
|
Abort(c, http.StatusTooManyRequests, i18n.ErrBusy)
|
|
|
|
}
|