2019-05-06 23:18:10 +02:00
|
|
|
package config
|
2018-11-17 06:21:39 +01:00
|
|
|
|
|
|
|
import (
|
2019-05-04 05:25:00 +02:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"reflect"
|
|
|
|
|
2018-11-17 06:21:39 +01:00
|
|
|
_ "github.com/jinzhu/gorm/dialects/mysql"
|
|
|
|
_ "github.com/jinzhu/gorm/dialects/sqlite"
|
2020-01-12 14:00:56 +01:00
|
|
|
"github.com/photoprism/photoprism/pkg/fs"
|
2018-11-17 06:21:39 +01:00
|
|
|
"github.com/urfave/cli"
|
2019-05-04 05:25:00 +02:00
|
|
|
"gopkg.in/yaml.v2"
|
2018-11-17 06:21:39 +01:00
|
|
|
)
|
|
|
|
|
2020-05-30 14:52:47 +02:00
|
|
|
// Database drivers (sql dialects).
|
2018-12-18 18:38:30 +01:00
|
|
|
const (
|
2020-05-30 14:52:47 +02:00
|
|
|
MySQL = "mysql"
|
|
|
|
SQLite = "sqlite3"
|
2018-12-18 18:38:30 +01:00
|
|
|
)
|
|
|
|
|
2019-05-06 23:18:10 +02:00
|
|
|
// Params provides a struct in which application configuration is stored.
|
2018-11-17 06:21:39 +01:00
|
|
|
// Application code must use functions to get config values, for two reasons:
|
|
|
|
//
|
|
|
|
// 1. Some values are computed and we don't want to leak implementation details (aims at reducing refactoring overhead).
|
|
|
|
//
|
|
|
|
// 2. Paths might actually be dynamic later (if we build a multi-user version).
|
|
|
|
//
|
|
|
|
// See https://github.com/photoprism/photoprism/issues/50#issuecomment-433856358
|
2019-05-06 23:18:10 +02:00
|
|
|
type Params struct {
|
2019-05-03 18:57:28 +02:00
|
|
|
Name string
|
|
|
|
Version string
|
|
|
|
Copyright string
|
2020-05-31 02:09:52 +02:00
|
|
|
SiteUrl string `yaml:"site-url" flag:"site-url"`
|
2020-06-26 14:26:36 +02:00
|
|
|
SitePreview string `yaml:"site-preview" flag:"site-preview"`
|
2020-05-31 02:09:52 +02:00
|
|
|
SiteTitle string `yaml:"site-title" flag:"site-title"`
|
|
|
|
SiteCaption string `yaml:"site-caption" flag:"site-caption"`
|
|
|
|
SiteDescription string `yaml:"site-description" flag:"site-description"`
|
|
|
|
SiteAuthor string `yaml:"site-author" flag:"site-author"`
|
2020-05-05 18:26:44 +02:00
|
|
|
Public bool `yaml:"public" flag:"public"`
|
2019-05-04 05:25:00 +02:00
|
|
|
Debug bool `yaml:"debug" flag:"debug"`
|
2019-05-04 09:11:33 +02:00
|
|
|
ReadOnly bool `yaml:"read-only" flag:"read-only"`
|
2020-01-06 23:43:19 +01:00
|
|
|
Experimental bool `yaml:"experimental" flag:"experimental"`
|
2020-05-31 02:09:52 +02:00
|
|
|
TensorFlowOff bool `yaml:"tf-off" flag:"tf-off"`
|
2020-01-06 23:43:19 +01:00
|
|
|
Workers int `yaml:"workers" flag:"workers"`
|
2020-04-06 22:09:45 +02:00
|
|
|
WakeupInterval int `yaml:"wakeup-interval" flag:"wakeup-interval"`
|
2020-05-05 18:26:44 +02:00
|
|
|
AdminPassword string `yaml:"admin-password" flag:"admin-password"`
|
2019-05-04 05:25:00 +02:00
|
|
|
LogLevel string `yaml:"log-level" flag:"log-level"`
|
2020-05-31 02:09:52 +02:00
|
|
|
AssetsPath string `yaml:"assets-path" flag:"assets-path"`
|
|
|
|
StoragePath string `yaml:"storage-path" flag:"storage-path"`
|
|
|
|
ImportPath string `yaml:"import-path" flag:"import-path"`
|
|
|
|
OriginalsPath string `yaml:"originals-path" flag:"originals-path"`
|
|
|
|
OriginalsLimit int64 `yaml:"originals-limit" flag:"originals-limit"`
|
2019-05-03 18:57:28 +02:00
|
|
|
ConfigFile string
|
2020-05-31 02:09:52 +02:00
|
|
|
SettingsPath string `yaml:"settings-path" flag:"settings-path"`
|
|
|
|
SettingsHidden bool `yaml:"settings-hidden" flag:"settings-hidden"`
|
2020-04-06 16:34:29 +02:00
|
|
|
TempPath string `yaml:"temp-path" flag:"temp-path"`
|
2019-05-04 05:25:00 +02:00
|
|
|
CachePath string `yaml:"cache-path" flag:"cache-path"`
|
2020-03-23 20:29:31 +01:00
|
|
|
DatabaseDriver string `yaml:"database-driver" flag:"database-driver"`
|
|
|
|
DatabaseDsn string `yaml:"database-dsn" flag:"database-dsn"`
|
2020-05-28 15:37:08 +02:00
|
|
|
DatabaseConns int `yaml:"database-conns" flag:"database-conns"`
|
2019-05-04 05:25:00 +02:00
|
|
|
HttpServerHost string `yaml:"http-host" flag:"http-host"`
|
|
|
|
HttpServerPort int `yaml:"http-port" flag:"http-port"`
|
|
|
|
HttpServerMode string `yaml:"http-mode" flag:"http-mode"`
|
|
|
|
HttpServerPassword string `yaml:"http-password" flag:"http-password"`
|
2020-05-05 17:04:13 +02:00
|
|
|
SipsBin string `yaml:"sips-bin" flag:"sips-bin"`
|
|
|
|
DarktableBin string `yaml:"darktable-bin" flag:"darktable-bin"`
|
|
|
|
HeifConvertBin string `yaml:"heifconvert-bin" flag:"heifconvert-bin"`
|
2020-05-11 18:29:17 +02:00
|
|
|
FFmpegBin string `yaml:"ffmpeg-bin" flag:"ffmpeg-bin"`
|
|
|
|
ExifToolBin string `yaml:"exiftool-bin" flag:"exiftool-bin"`
|
2020-05-13 20:53:15 +02:00
|
|
|
SidecarJson bool `yaml:"sidecar-json" flag:"sidecar-json"`
|
2020-05-18 22:18:58 +02:00
|
|
|
SidecarYaml bool `yaml:"sidecar-yaml" flag:"sidecar-yaml"`
|
2020-06-07 10:09:35 +02:00
|
|
|
SidecarPath string `yaml:"sidecar-path" flag:"sidecar-path"`
|
2020-05-05 17:04:13 +02:00
|
|
|
PIDFilename string `yaml:"pid-filename" flag:"pid-filename"`
|
|
|
|
LogFilename string `yaml:"log-filename" flag:"log-filename"`
|
|
|
|
DetachServer bool `yaml:"detach-server" flag:"detach-server"`
|
|
|
|
DetectNSFW bool `yaml:"detect-nsfw" flag:"detect-nsfw"`
|
|
|
|
UploadNSFW bool `yaml:"upload-nsfw" flag:"upload-nsfw"`
|
|
|
|
GeoCodingApi string `yaml:"geocoding-api" flag:"geocoding-api"`
|
2020-05-27 19:38:40 +02:00
|
|
|
DownloadToken string `yaml:"download-token" flag:"download-token"`
|
2020-05-27 19:56:56 +02:00
|
|
|
PreviewToken string `yaml:"preview-token" flag:"preview-token"`
|
2020-05-05 17:17:19 +02:00
|
|
|
ThumbFilter string `yaml:"thumb-filter" flag:"thumb-filter"`
|
2020-05-05 17:04:13 +02:00
|
|
|
ThumbUncached bool `yaml:"thumb-uncached" flag:"thumb-uncached"`
|
|
|
|
ThumbSize int `yaml:"thumb-size" flag:"thumb-size"`
|
|
|
|
ThumbLimit int `yaml:"thumb-limit" flag:"thumb-limit"`
|
2020-05-27 19:38:40 +02:00
|
|
|
JpegQuality int `yaml:"jpeg-quality" flag:"jpeg-quality"`
|
2018-11-17 06:21:39 +01:00
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// NewParams creates a new configuration entity by using two methods:
|
2019-05-04 05:25:00 +02:00
|
|
|
//
|
2020-03-28 21:44:30 +01:00
|
|
|
// 1. Load: This will initialize values from a yaml config file.
|
2019-05-04 05:25:00 +02:00
|
|
|
//
|
2020-03-28 21:44:30 +01:00
|
|
|
// 2. SetContext: Which comes after Load and overrides
|
2019-05-04 05:25:00 +02:00
|
|
|
// any previous values giving an option two override file configs through the CLI.
|
2019-05-06 23:18:10 +02:00
|
|
|
func NewParams(ctx *cli.Context) *Params {
|
|
|
|
c := &Params{}
|
2018-12-26 11:40:01 +01:00
|
|
|
|
2019-05-04 05:25:00 +02:00
|
|
|
c.Name = ctx.App.Name
|
|
|
|
c.Copyright = ctx.App.Copyright
|
|
|
|
c.Version = ctx.App.Version
|
2020-01-31 15:29:06 +01:00
|
|
|
c.ConfigFile = fs.Abs(ctx.GlobalString("config-file"))
|
2018-11-17 06:21:39 +01:00
|
|
|
|
2020-03-28 21:44:30 +01:00
|
|
|
if err := c.Load(c.ConfigFile); err != nil {
|
2019-05-04 05:25:00 +02:00
|
|
|
log.Debug(err)
|
2018-11-17 06:21:39 +01:00
|
|
|
}
|
|
|
|
|
2020-03-28 21:44:30 +01:00
|
|
|
if err := c.SetContext(ctx); err != nil {
|
2019-05-04 05:25:00 +02:00
|
|
|
log.Error(err)
|
2018-11-17 06:21:39 +01:00
|
|
|
}
|
|
|
|
|
2019-05-04 05:25:00 +02:00
|
|
|
return c
|
|
|
|
}
|
2018-11-17 06:21:39 +01:00
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// expandFilenames converts path in config to absolute path
|
2019-05-06 23:18:10 +02:00
|
|
|
func (c *Params) expandFilenames() {
|
2020-05-31 02:09:52 +02:00
|
|
|
c.SettingsPath = fs.Abs(c.SettingsPath)
|
|
|
|
c.StoragePath = fs.Abs(c.StoragePath)
|
2020-01-31 15:29:06 +01:00
|
|
|
c.AssetsPath = fs.Abs(c.AssetsPath)
|
|
|
|
c.CachePath = fs.Abs(c.CachePath)
|
|
|
|
c.OriginalsPath = fs.Abs(c.OriginalsPath)
|
|
|
|
c.ImportPath = fs.Abs(c.ImportPath)
|
2020-04-06 23:04:52 +02:00
|
|
|
c.TempPath = fs.Abs(c.TempPath)
|
2020-01-31 15:29:06 +01:00
|
|
|
c.PIDFilename = fs.Abs(c.PIDFilename)
|
|
|
|
c.LogFilename = fs.Abs(c.LogFilename)
|
2019-05-04 05:25:00 +02:00
|
|
|
}
|
2018-11-17 06:21:39 +01:00
|
|
|
|
2020-03-28 21:44:30 +01:00
|
|
|
// Load uses a yaml config file to initiate the configuration entity.
|
|
|
|
func (c *Params) Load(fileName string) error {
|
2020-04-30 15:41:47 +02:00
|
|
|
if fileName == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-12 14:00:56 +01:00
|
|
|
if !fs.FileExists(fileName) {
|
2020-04-30 15:41:47 +02:00
|
|
|
return errors.New(fmt.Sprintf("config: %s not found", fileName))
|
2018-11-17 06:21:39 +01:00
|
|
|
}
|
|
|
|
|
2019-05-04 05:25:00 +02:00
|
|
|
yamlConfig, err := ioutil.ReadFile(fileName)
|
2018-11-17 06:21:39 +01:00
|
|
|
|
2019-05-04 05:25:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2018-11-17 06:21:39 +01:00
|
|
|
}
|
|
|
|
|
2019-05-04 05:25:00 +02:00
|
|
|
return yaml.Unmarshal(yamlConfig, c)
|
2018-11-17 06:21:39 +01:00
|
|
|
}
|
|
|
|
|
2020-03-28 21:44:30 +01:00
|
|
|
// SetContext uses values from the CLI to setup configuration overrides
|
2018-11-17 06:21:39 +01:00
|
|
|
// for the entity.
|
2020-03-28 21:44:30 +01:00
|
|
|
func (c *Params) SetContext(ctx *cli.Context) error {
|
2019-05-04 05:25:00 +02:00
|
|
|
v := reflect.ValueOf(c).Elem()
|
|
|
|
|
2020-05-23 20:58:58 +02:00
|
|
|
// Iterate through all config fields.
|
2019-05-04 05:25:00 +02:00
|
|
|
for i := 0; i < v.NumField(); i++ {
|
|
|
|
fieldValue := v.Field(i)
|
|
|
|
|
|
|
|
tagValue := v.Type().Field(i).Tag.Get("flag")
|
|
|
|
|
2020-05-23 20:58:58 +02:00
|
|
|
// Automatically assign values to fields with "flag" tag.
|
2019-05-04 05:25:00 +02:00
|
|
|
if tagValue != "" {
|
|
|
|
switch t := fieldValue.Interface().(type) {
|
|
|
|
case int, int64:
|
2020-05-23 20:58:58 +02:00
|
|
|
// Only if explicitly set or current value is empty (use default).
|
2019-06-30 05:38:39 +02:00
|
|
|
if ctx.IsSet(tagValue) {
|
|
|
|
f := ctx.Int64(tagValue)
|
|
|
|
fieldValue.SetInt(f)
|
|
|
|
} else if ctx.GlobalIsSet(tagValue) || fieldValue.Int() == 0 {
|
2019-05-04 05:25:00 +02:00
|
|
|
f := ctx.GlobalInt64(tagValue)
|
|
|
|
fieldValue.SetInt(f)
|
|
|
|
}
|
|
|
|
case uint, uint64:
|
2020-05-23 20:58:58 +02:00
|
|
|
// Only if explicitly set or current value is empty (use default).
|
2019-06-30 05:38:39 +02:00
|
|
|
if ctx.IsSet(tagValue) {
|
|
|
|
f := ctx.Uint64(tagValue)
|
|
|
|
fieldValue.SetUint(f)
|
|
|
|
} else if ctx.GlobalIsSet(tagValue) || fieldValue.Uint() == 0 {
|
2019-05-04 05:25:00 +02:00
|
|
|
f := ctx.GlobalUint64(tagValue)
|
|
|
|
fieldValue.SetUint(f)
|
|
|
|
}
|
|
|
|
case string:
|
|
|
|
// Only if explicitly set or current value is empty (use default)
|
2019-06-30 05:38:39 +02:00
|
|
|
if ctx.IsSet(tagValue) {
|
|
|
|
f := ctx.String(tagValue)
|
|
|
|
fieldValue.SetString(f)
|
|
|
|
} else if ctx.GlobalIsSet(tagValue) || fieldValue.String() == "" {
|
2019-05-04 05:25:00 +02:00
|
|
|
f := ctx.GlobalString(tagValue)
|
|
|
|
fieldValue.SetString(f)
|
|
|
|
}
|
|
|
|
case bool:
|
2019-06-30 05:38:39 +02:00
|
|
|
if ctx.IsSet(tagValue) {
|
|
|
|
f := ctx.Bool(tagValue)
|
|
|
|
fieldValue.SetBool(f)
|
|
|
|
} else if ctx.GlobalIsSet(tagValue) {
|
2019-05-04 05:25:00 +02:00
|
|
|
f := ctx.GlobalBool(tagValue)
|
|
|
|
fieldValue.SetBool(f)
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
log.Warnf("can't assign value of type %s from cli flag %s", t, tagValue)
|
|
|
|
}
|
|
|
|
}
|
2018-11-17 06:21:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|