mirror of
https://github.com/photoprism/photoprism.git
synced 2025-09-26 21:01:58 +08:00
211 lines
5.9 KiB
Go
211 lines
5.9 KiB
Go
package config
|
|
|
|
import (
|
|
_ "embed"
|
|
"fmt"
|
|
"net/url"
|
|
"os"
|
|
"path"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/photoprism/photoprism/pkg/clean"
|
|
"github.com/photoprism/photoprism/pkg/fs"
|
|
)
|
|
|
|
//go:embed robots.txt
|
|
var robotsTxt []byte
|
|
|
|
const localhost = "localhost"
|
|
|
|
// BaseUri returns the site base URI for a given resource.
|
|
func (c *Config) BaseUri(res string) string {
|
|
if c.SiteUrl() == "" {
|
|
return res
|
|
}
|
|
|
|
u, err := url.Parse(c.SiteUrl())
|
|
|
|
if err != nil {
|
|
return res
|
|
}
|
|
|
|
return strings.TrimRight(u.EscapedPath(), "/") + res
|
|
}
|
|
|
|
// ApiUri returns the api URI.
|
|
func (c *Config) ApiUri() string {
|
|
return c.BaseUri(ApiUri)
|
|
}
|
|
|
|
// LibraryUri returns the user interface URI for the given resource.
|
|
func (c *Config) LibraryUri(res string) string {
|
|
return c.BaseUri(LibraryUri + res)
|
|
}
|
|
|
|
// ContentUri returns the content delivery URI based on the CdnUrl and the ApiUri.
|
|
func (c *Config) ContentUri() string {
|
|
return c.CdnUrl(c.ApiUri())
|
|
}
|
|
|
|
// DownloadUrl returns the download URL based on the SiteUrl and the DownloadUri.
|
|
func (c *Config) DownloadUrl() string {
|
|
return strings.TrimRight(c.options.SiteUrl, "/") + DownloadUri
|
|
}
|
|
|
|
// VideoUri returns the video streaming URI.
|
|
func (c *Config) VideoUri() string {
|
|
if c.CdnVideo() {
|
|
return c.ContentUri()
|
|
}
|
|
|
|
return c.ApiUri()
|
|
}
|
|
|
|
// StaticUri returns the static content URI.
|
|
func (c *Config) StaticUri() string {
|
|
return c.CdnUrl(c.BaseUri(StaticUri))
|
|
}
|
|
|
|
// StaticAssetUri returns the resource URI of the static file asset.
|
|
func (c *Config) StaticAssetUri(res string) string {
|
|
return c.StaticUri() + "/" + res
|
|
}
|
|
|
|
// SiteUrl returns the public server URL (default is "http://localhost:2342/").
|
|
func (c *Config) SiteUrl() string {
|
|
if c.options.SiteUrl == "" {
|
|
return "http://localhost:2342/"
|
|
}
|
|
|
|
return strings.TrimRight(c.options.SiteUrl, "/") + "/"
|
|
}
|
|
|
|
// SiteHttps checks if the site URL uses HTTPS.
|
|
func (c *Config) SiteHttps() bool {
|
|
if c.options.SiteUrl == "" {
|
|
return false
|
|
}
|
|
|
|
return strings.HasPrefix(c.options.SiteUrl, "https://")
|
|
}
|
|
|
|
// SiteDomain returns the public hostname without protocol or post.
|
|
func (c *Config) SiteDomain() string {
|
|
if u, err := url.Parse(c.SiteUrl()); err != nil {
|
|
return localhost
|
|
} else {
|
|
return u.Hostname()
|
|
}
|
|
}
|
|
|
|
// SiteHost returns the public hostname and port number in the format "domain:port".
|
|
func (c *Config) SiteHost() string {
|
|
if u, err := url.Parse(c.SiteUrl()); err != nil {
|
|
return localhost
|
|
} else if hostname := u.Hostname(); hostname == "" {
|
|
return localhost
|
|
} else if port := u.Port(); port != "" {
|
|
return fmt.Sprintf("%s:%s", hostname, port)
|
|
} else {
|
|
return hostname
|
|
}
|
|
}
|
|
|
|
// SiteAuthor returns the site author / copyright.
|
|
func (c *Config) SiteAuthor() string {
|
|
return c.options.SiteAuthor
|
|
}
|
|
|
|
// SiteTitle returns the main site title (default is application name).
|
|
func (c *Config) SiteTitle() string {
|
|
if c.options.SiteTitle == "" {
|
|
return c.Name()
|
|
}
|
|
|
|
return c.options.SiteTitle
|
|
}
|
|
|
|
// SiteCaption returns a short site caption.
|
|
func (c *Config) SiteCaption() string {
|
|
return c.options.SiteCaption
|
|
}
|
|
|
|
// SiteDescription returns a long site description.
|
|
func (c *Config) SiteDescription() string {
|
|
return c.options.SiteDescription
|
|
}
|
|
|
|
// SiteFavicon returns the site favicon image name.
|
|
func (c *Config) SiteFavicon() string {
|
|
if c.options.SiteFavicon != "" {
|
|
if fs.FileExistsNotEmpty(c.options.SiteFavicon) {
|
|
return c.options.SiteFavicon
|
|
} else if fileName := filepath.Join(c.ThemePath(), strings.TrimPrefix(c.options.SiteFavicon, ThemeUri)); fs.FileExistsNotEmpty(fileName) {
|
|
return fileName
|
|
} else if fileName = filepath.Join(c.ImgPath(), c.options.SiteFavicon); fs.FileExistsNotEmpty(fileName) {
|
|
return fileName
|
|
}
|
|
}
|
|
|
|
return filepath.Join(c.ImgPath(), "favicon.ico")
|
|
}
|
|
|
|
// SitePreview returns the site preview image URL for sharing.
|
|
func (c *Config) SitePreview() string {
|
|
if c.options.SitePreview != "" {
|
|
if strings.HasPrefix(c.options.SitePreview, "http") {
|
|
return c.options.SitePreview
|
|
|
|
} else if fileName := filepath.Join(c.ThemePath(), c.options.SitePreview); fs.FileExistsNotEmpty(fileName) {
|
|
return strings.TrimRight(c.options.SiteUrl, "/") + path.Join(ThemeUri, c.options.SitePreview)
|
|
}
|
|
|
|
return c.SiteUrl() + strings.TrimPrefix(c.options.SitePreview, "/")
|
|
}
|
|
|
|
return fmt.Sprintf("https://i.photoprism.app/prism?cover=64&style=centered%%20dark&caption=none&title=%s", url.QueryEscape(c.AppName()))
|
|
}
|
|
|
|
// LegalInfo returns the legal info text for the page footer.
|
|
func (c *Config) LegalInfo() string {
|
|
if s := c.CliContextString("imprint"); s != "" {
|
|
log.Warnf("config: option 'imprint' is deprecated, please use 'legal-info'")
|
|
return s
|
|
}
|
|
|
|
return c.options.LegalInfo
|
|
}
|
|
|
|
// LegalUrl returns the legal info url.
|
|
func (c *Config) LegalUrl() string {
|
|
if s := c.CliContextString("imprint-url"); s != "" {
|
|
log.Warnf("config: option 'imprint-url' is deprecated, please use 'legal-url'")
|
|
return s
|
|
}
|
|
|
|
return c.options.LegalUrl
|
|
}
|
|
|
|
// RobotsTxt returns the content of the robots.txt file to be used for this site:
|
|
// https://developers.google.com/search/docs/crawling-indexing/robots/create-robots-txt
|
|
func (c *Config) RobotsTxt() ([]byte, error) {
|
|
if c.Demo() && c.Public() {
|
|
// Allow public demo instances to be indexed.
|
|
return []byte(fmt.Sprintf("User-agent: *\nDisallow: /\nAllow: %s/\nAllow: %s/\nAllow: .js\nAllow: .css", LibraryUri, StaticUri)), nil
|
|
} else if c.Public() {
|
|
// Do not allow other instances to be indexed when public mode is enabled.
|
|
return robotsTxt, nil
|
|
} else if fileName := filepath.Join(c.ConfigPath(), "robots.txt"); !fs.FileExists(fileName) {
|
|
// Do not allow indexing if config/robots.txt does not exist.
|
|
return robotsTxt, nil
|
|
} else if robots, robotsErr := os.ReadFile(fileName); robotsErr != nil {
|
|
// Log error and do not allow indexing if config/robots.txt cannot be read.
|
|
log.Debugf("config: failed to read robots.txt file (%s)", clean.Error(robotsErr))
|
|
return robotsTxt, robotsErr
|
|
} else {
|
|
// Return content of the config/robots.txt file.
|
|
return robots, nil
|
|
}
|
|
}
|