photoprism/internal/server/routes.go
Michael Mayer 86086753c2 Server: Improve initialization and logging of extensions
Signed-off-by: Michael Mayer <michael@photoprism.app>
2022-09-02 19:09:25 +02:00

223 lines
5.6 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.PublicConfig()
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.PublicConfig()
c.HTML(http.StatusOK, "rainbow.tmpl", gin.H{"config": clientConfig})
})
// Splash Screen.
router.GET(conf.BaseUri("/_splash"), func(c *gin.Context) {
clientConfig := conf.PublicConfig()
c.HTML(http.StatusOK, "splash.tmpl", gin.H{"config": clientConfig})
})
// JSON-REST API Version 1
v1 := router.Group(conf.BaseUri(config.ApiUri))
{
// Config options.
api.GetConfig(v1)
api.GetConfigOptions(v1)
api.SaveConfigOptions(v1)
// User profile and settings.
api.GetSettings(v1)
api.SaveSettings(v1)
api.ChangePassword(v1)
api.CreateSession(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)
// Thumbnails and downloads.
api.GetThumb(v1)
api.GetDownload(v1)
api.GetVideo(v1)
api.ZipCreate(v1)
api.ZipDownload(v1)
// Photos.
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)
// 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)
// 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)
// Folders.
api.SearchFoldersOriginals(v1)
api.SearchFoldersImport(v1)
api.FolderCover(v1)
// People and other subjects.
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)
// Indexing and importing.
api.Upload(v1)
api.StartImport(v1)
api.CancelImport(v1)
api.StartIndexing(v1)
api.CancelIndexing(v1)
// Batch operations.
api.BatchPhotosApprove(v1)
api.BatchPhotosArchive(v1)
api.BatchPhotosRestore(v1)
api.BatchPhotosPrivate(v1)
api.BatchPhotosDelete(v1)
api.BatchAlbumsDelete(v1)
api.BatchLabelsDelete(v1)
// Other.
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.PublicConfig()}
c.HTML(http.StatusOK, conf.TemplateName(), values)
})
}