package config import ( "net/url" "path/filepath" "regexp" "strings" "github.com/photoprism/photoprism/internal/config/ttl" "github.com/photoprism/photoprism/internal/server/limiter" "github.com/photoprism/photoprism/pkg/fs" "github.com/photoprism/photoprism/pkg/service/http/header" "github.com/photoprism/photoprism/pkg/service/http/scheme" ) const ( HttpModeProd = "release" HttpModeDebug = "debug" ) // DetachServer checks if server should detach from console (daemon mode). func (c *Config) DetachServer() bool { return c.options.DetachServer } // TrustedPlatform returns the trusted platform client IP address header name. func (c *Config) TrustedPlatform() string { return c.options.TrustedPlatform } // TrustedProxy returns the ranges from which reverse proxy headers can be trusted as comma-separated list. func (c *Config) TrustedProxy() string { return strings.Join(c.options.TrustedProxies, ", ") } // TrustedProxies returns proxy server ranges from which reverse proxy headers can be trusted. func (c *Config) TrustedProxies() []string { return c.options.TrustedProxies } // ProxyClientHeader returns the proxy client IP address header names as comma-separated list. func (c *Config) ProxyClientHeader() string { return strings.Join(c.options.ProxyClientHeaders, ", ") } // ProxyClientHeaders returns the proxy client IP address header names, if any. func (c *Config) ProxyClientHeaders() []string { return c.options.ProxyClientHeaders } // ProxyProtoHeader returns the proxy protocol header names. func (c *Config) ProxyProtoHeader() []string { return c.options.ProxyProtoHeaders } // ProxyProtoHttps returns the proxy protocol header HTTPS values. func (c *Config) ProxyProtoHttps() []string { return c.options.ProxyProtoHttps } // ProxyProtoHeaders returns a map with the proxy https protocol headers. func (c *Config) ProxyProtoHeaders() map[string]string { p := len(c.options.ProxyProtoHeaders) h := make(map[string]string, p+1) if p == 0 { h[header.XForwardedProto] = scheme.Https return h } for k, v := range c.options.ProxyProtoHeaders { if l := len(c.options.ProxyProtoHttps); l == 0 { h[v] = scheme.Https } else if l > k { h[v] = c.options.ProxyProtoHttps[k] } else { h[v] = c.options.ProxyProtoHttps[0] } } return h } // HttpMode returns the server mode. func (c *Config) HttpMode() string { if c.Prod() { return HttpModeProd } else if c.options.HttpMode == "" { if c.Debug() { return HttpModeDebug } return HttpModeProd } return c.options.HttpMode } // HttpCompression returns the http compression method (gzip, or none). func (c *Config) HttpCompression() string { return strings.ToLower(strings.TrimSpace(c.options.HttpCompression)) } // HttpCachePublic checks whether static content may be cached by a CDN or caching proxy. func (c *Config) HttpCachePublic() bool { if c.options.HttpCachePublic { return true } return c.options.CdnUrl != "" } // HttpCacheMaxAge returns the time in seconds until cached content expires. func (c *Config) HttpCacheMaxAge() ttl.Duration { // Return default cache maxage? if c.options.HttpCacheMaxAge < 1 { return ttl.CacheDefault } else if c.options.HttpCacheMaxAge > 31536000 { return ttl.Duration(31536000) } // Return the configured cache expiration time. return ttl.Duration(c.options.HttpCacheMaxAge) } // HttpVideoMaxAge returns the time in seconds until cached videos expire. func (c *Config) HttpVideoMaxAge() ttl.Duration { // Return default video maxage? if c.options.HttpVideoMaxAge < 1 { return ttl.CacheVideo } else if c.options.HttpVideoMaxAge > 31536000 { return ttl.Duration(31536000) } // Return the configured cache expiration time. return ttl.Duration(c.options.HttpVideoMaxAge) } // HttpHost returns the built-in HTTP server host name or IP address (empty for all interfaces). func (c *Config) HttpHost() string { // Set http host to "0.0.0.0" if unix socket is used to serve requests. if c.options.HttpHost == "" { return limiter.DefaultIP } return c.options.HttpHost } // HttpPort returns the HTTP server port number. func (c *Config) HttpPort() int { if c.options.HttpPort == 0 { return 2342 } return c.options.HttpPort } // HttpSocket tries to parse the HttpHost as a Unix socket URL and returns it, or nil if it fails. func (c *Config) HttpSocket() *url.URL { if c.options.HttpSocket != nil { // Return cached resource URI. return c.options.HttpSocket } else if host := c.options.HttpHost; !strings.HasPrefix(host, "unix:") { return nil } // Parse socket resource URI. socket, err := url.Parse(c.options.HttpHost) // Return nil if parsing failed, or it's not a Unix domain socket URI. if err != nil { return nil } if socket.Scheme == scheme.HttpUnix { socket.Scheme = scheme.Unix } if socket.Scheme != scheme.Unix || socket.Host == "" && socket.Path == "" { return nil } else if socket.Host != "" && socket.Path == "" { // Create a path from the host if an absolute socket path is not specified, socket.Path = fs.Abs(socket.Host) socket.Host = "" } // Should never happen. if socket.Path == "" { return nil } // Cache parsed resource URI. c.options.HttpSocket = socket // Return parsed resource URI. return c.options.HttpSocket } // TemplatesPath returns the server templates path. func (c *Config) TemplatesPath() string { return filepath.Join(c.AssetsPath(), fs.TemplatesDir) } // CustomTemplatesPath returns the path to custom templates. func (c *Config) CustomTemplatesPath() string { if dir := c.CustomAssetsPath(); dir == "" { return "" } else if dir = filepath.Join(dir, fs.TemplatesDir); fs.PathExists(dir) { return dir } return "" } // TemplateFiles returns the file paths of all templates found. func (c *Config) TemplateFiles() []string { results := make([]string, 0, 32) var tmplPaths []string // Path set for custom templates? if cDir := c.CustomTemplatesPath(); cDir != "" { tmplPaths = []string{c.TemplatesPath(), cDir} } else { tmplPaths = []string{c.TemplatesPath()} } // Find template files. for _, dir := range tmplPaths { if dir == "" { continue } matches, err := filepath.Glob(regexp.QuoteMeta(dir) + "/[A-Za-z0-9]*.*") if err != nil { continue } for _, tmplName := range matches { results = append(results, tmplName) } } return results } // TemplateExists checks if a template with the given name exists (e.g. index.gohtml). func (c *Config) TemplateExists(name string) bool { if found := fs.FileExists(filepath.Join(c.TemplatesPath(), name)); found { return true } else if dir := c.CustomTemplatesPath(); dir != "" { return fs.FileExists(filepath.Join(dir, name)) } else { return false } } // TemplateName returns the name of the user interface bootstrap template. func (c *Config) TemplateName() string { if s := c.Settings(); s != nil { if c.TemplateExists(s.Templates.Default) { return s.Templates.Default } } return "index.gohtml" } // StaticPath returns the static assets' path. func (c *Config) StaticPath() string { return filepath.Join(c.AssetsPath(), fs.StaticDir) } // StaticFile returns the path to a static file. func (c *Config) StaticFile(fileName string) string { return filepath.Join(c.AssetsPath(), fs.StaticDir, fileName) } // BuildPath returns the static build path. func (c *Config) BuildPath() string { return filepath.Join(c.StaticPath(), fs.BuildDir) } // ImgPath returns the path to static image files. func (c *Config) ImgPath() string { return filepath.Join(c.StaticPath(), fs.ImgDir) } // ThemePath returns the path to static theme files. func (c *Config) ThemePath() string { if c.options.CustomThemePath != "" { return c.options.CustomThemePath } return filepath.Join(c.ConfigPath(), fs.ThemeDir) } // SetThemePath sets a custom theme files path. func (c *Config) SetThemePath(dir string) *Config { if dir != "" { dir = fs.Abs(dir) } c.options.CustomThemePath = dir return c }