4f425790ab
Signed-off-by: Michael Mayer <michael@photoprism.app>
226 lines
5.7 KiB
Go
226 lines
5.7 KiB
Go
package server
|
|
|
|
import (
|
|
"net/http"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
"github.com/photoprism/photoprism/internal/api"
|
|
"github.com/photoprism/photoprism/internal/config"
|
|
"github.com/photoprism/photoprism/pkg/clean"
|
|
)
|
|
|
|
func registerRoutes(router *gin.Engine, conf *config.Config) {
|
|
// Enables automatic redirection if the current route cannot be matched but a
|
|
// handler for the path with (without) the trailing slash exists.
|
|
router.RedirectTrailingSlash = true
|
|
|
|
// Static favicon image.
|
|
router.StaticFile(conf.BaseUri("/favicon.ico"), filepath.Join(conf.ImgPath(), "favicon.ico"))
|
|
|
|
// Static assets like js, css and font files.
|
|
router.Static(conf.BaseUri(config.StaticUri), conf.StaticPath())
|
|
|
|
// Serve custom static assets if dir exists.
|
|
if dir := conf.CustomStaticPath(); dir != "" {
|
|
router.Static(conf.BaseUri(config.CustomStaticUri), dir)
|
|
}
|
|
|
|
// PWA Manifest.
|
|
router.GET(conf.BaseUri("/manifest.json"), func(c *gin.Context) {
|
|
c.Header("Cache-Control", "no-store")
|
|
c.Header("Content-Type", "application/json")
|
|
|
|
clientConfig := conf.ClientPublic()
|
|
c.HTML(http.StatusOK, "manifest.json", gin.H{"config": clientConfig})
|
|
})
|
|
|
|
// PWA Service Worker.
|
|
router.GET(conf.BaseUri("/sw.js"), func(c *gin.Context) {
|
|
c.Header("Cache-Control", "no-store")
|
|
c.File(filepath.Join(conf.BuildPath(), "sw.js"))
|
|
})
|
|
|
|
// Rainbow Page.
|
|
router.GET(conf.BaseUri("/_rainbow"), func(c *gin.Context) {
|
|
clientConfig := conf.ClientPublic()
|
|
c.HTML(http.StatusOK, "rainbow.tmpl", gin.H{"config": clientConfig})
|
|
})
|
|
|
|
// Splash Screen.
|
|
router.GET(conf.BaseUri("/_splash"), func(c *gin.Context) {
|
|
clientConfig := conf.ClientPublic()
|
|
c.HTML(http.StatusOK, "splash.tmpl", gin.H{"config": clientConfig})
|
|
})
|
|
|
|
// JSON-REST API Version 1
|
|
v1 := router.Group(conf.BaseUri(config.ApiUri))
|
|
{
|
|
// Global App Config.
|
|
api.GetConfigOptions(v1)
|
|
api.SaveConfigOptions(v1)
|
|
|
|
// Custom Settings.
|
|
api.GetClientConfig(v1)
|
|
api.GetSettings(v1)
|
|
api.SaveSettings(v1)
|
|
|
|
// User Authentication.
|
|
api.ChangePassword(v1)
|
|
api.CreateSession(v1)
|
|
api.GetSession(v1)
|
|
api.DeleteSession(v1)
|
|
|
|
// External Account Management.
|
|
api.SearchAccounts(v1)
|
|
api.GetAccount(v1)
|
|
api.GetAccountFolders(v1)
|
|
api.ShareWithAccount(v1)
|
|
api.CreateAccount(v1)
|
|
api.DeleteAccount(v1)
|
|
api.UpdateAccount(v1)
|
|
|
|
// Thumbs, Downloads, and Videos.
|
|
api.GetThumb(v1)
|
|
api.GetDownload(v1)
|
|
api.GetVideo(v1)
|
|
api.ZipCreate(v1)
|
|
api.ZipDownload(v1)
|
|
|
|
// Index, Upload, and Import.
|
|
api.Upload(v1)
|
|
api.StartImport(v1)
|
|
api.CancelImport(v1)
|
|
api.StartIndexing(v1)
|
|
api.CancelIndexing(v1)
|
|
|
|
// Search & Organization.
|
|
api.SearchPhotos(v1)
|
|
api.SearchGeo(v1)
|
|
api.GetPhoto(v1)
|
|
api.GetPhotoYaml(v1)
|
|
api.UpdatePhoto(v1)
|
|
api.GetPhotoDownload(v1)
|
|
// api.GetPhotoLinks(v1)
|
|
// api.CreatePhotoLink(v1)
|
|
// api.UpdatePhotoLink(v1)
|
|
// api.DeletePhotoLink(v1)
|
|
api.ApprovePhoto(v1)
|
|
api.LikePhoto(v1)
|
|
api.DislikePhoto(v1)
|
|
api.AddPhotoLabel(v1)
|
|
api.RemovePhotoLabel(v1)
|
|
api.UpdatePhotoLabel(v1)
|
|
api.GetMomentsTime(v1)
|
|
api.GetFile(v1)
|
|
api.DeleteFile(v1)
|
|
api.UpdateMarker(v1)
|
|
api.ClearMarkerSubject(v1)
|
|
api.PhotoPrimary(v1)
|
|
api.PhotoUnstack(v1)
|
|
|
|
// Photo Albums.
|
|
api.SearchAlbums(v1)
|
|
api.GetAlbum(v1)
|
|
api.AlbumCover(v1)
|
|
api.CreateAlbum(v1)
|
|
api.UpdateAlbum(v1)
|
|
api.DeleteAlbum(v1)
|
|
api.DownloadAlbum(v1)
|
|
api.GetAlbumLinks(v1)
|
|
api.CreateAlbumLink(v1)
|
|
api.UpdateAlbumLink(v1)
|
|
api.DeleteAlbumLink(v1)
|
|
api.LikeAlbum(v1)
|
|
api.DislikeAlbum(v1)
|
|
api.CloneAlbums(v1)
|
|
api.AddPhotosToAlbum(v1)
|
|
api.RemovePhotosFromAlbum(v1)
|
|
|
|
// Photo Labels.
|
|
api.SearchLabels(v1)
|
|
api.LabelCover(v1)
|
|
api.UpdateLabel(v1)
|
|
// api.GetLabelLinks(v1)
|
|
// api.CreateLabelLink(v1)
|
|
// api.UpdateLabelLink(v1)
|
|
// api.DeleteLabelLink(v1)
|
|
api.LikeLabel(v1)
|
|
api.DislikeLabel(v1)
|
|
|
|
// Files and Folders.
|
|
api.SearchFoldersOriginals(v1)
|
|
api.SearchFoldersImport(v1)
|
|
api.FolderCover(v1)
|
|
|
|
// People.
|
|
api.SearchSubjects(v1)
|
|
api.GetSubject(v1)
|
|
api.UpdateSubject(v1)
|
|
api.LikeSubject(v1)
|
|
api.DislikeSubject(v1)
|
|
|
|
// Faces.
|
|
api.SearchFaces(v1)
|
|
api.GetFace(v1)
|
|
api.UpdateFace(v1)
|
|
|
|
// Batch Operations.
|
|
api.BatchPhotosApprove(v1)
|
|
api.BatchPhotosArchive(v1)
|
|
api.BatchPhotosRestore(v1)
|
|
api.BatchPhotosPrivate(v1)
|
|
api.BatchPhotosDelete(v1)
|
|
api.BatchAlbumsDelete(v1)
|
|
api.BatchLabelsDelete(v1)
|
|
|
|
// Technical Endpoints.
|
|
api.GetSvg(v1)
|
|
api.GetStatus(v1)
|
|
api.GetErrors(v1)
|
|
api.DeleteErrors(v1)
|
|
api.SendFeedback(v1)
|
|
api.Connect(v1)
|
|
api.WebSocket(v1)
|
|
}
|
|
|
|
// Configure link sharing.
|
|
s := router.Group(conf.BaseUri("/s"))
|
|
{
|
|
api.Shares(s)
|
|
api.SharePreview(s)
|
|
}
|
|
|
|
// WebDAV server for file management, sync and sharing.
|
|
if conf.DisableWebDAV() {
|
|
log.Info("webdav: server disabled")
|
|
} else {
|
|
WebDAV(conf.OriginalsPath(), router.Group(conf.BaseUri(WebDAVOriginals), BasicAuth()), conf)
|
|
log.Infof("webdav: %s/ enabled, waiting for requests", conf.BaseUri(WebDAVOriginals))
|
|
|
|
if conf.ImportPath() != "" {
|
|
WebDAV(conf.ImportPath(), router.Group(conf.BaseUri(WebDAVImport), BasicAuth()), conf)
|
|
log.Infof("webdav: %s/ enabled, waiting for requests", conf.BaseUri(WebDAVImport))
|
|
}
|
|
}
|
|
|
|
// Initialize package extensions.
|
|
for _, ext := range Extensions() {
|
|
start := time.Now()
|
|
|
|
if err := ext.init(router, conf); err != nil {
|
|
log.Warnf("server: %s in %s extension[%s]", err, clean.Log(ext.name), time.Since(start))
|
|
} else {
|
|
log.Debugf("server: %s extension loaded [%s]", clean.Log(ext.name), time.Since(start))
|
|
}
|
|
}
|
|
|
|
// Default HTML page for client-side rendering and routing via VueJS.
|
|
router.NoRoute(func(c *gin.Context) {
|
|
signUp := gin.H{"message": config.MsgSponsor, "url": config.SignUpURL}
|
|
values := gin.H{"signUp": signUp, "config": conf.ClientPublic()}
|
|
c.HTML(http.StatusOK, conf.TemplateName(), values)
|
|
})
|
|
}
|