2018-11-17 06:21:39 +01:00
|
|
|
package context
|
|
|
|
|
|
|
|
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"
|
|
|
|
"github.com/photoprism/photoprism/internal/fsutil"
|
2019-05-04 05:25:00 +02:00
|
|
|
log "github.com/sirupsen/logrus"
|
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
|
|
|
)
|
|
|
|
|
2018-12-18 18:38:30 +01:00
|
|
|
const (
|
2018-12-21 08:44:13 +01:00
|
|
|
DbTiDB = "internal"
|
2018-12-18 18:38:30 +01:00
|
|
|
DbMySQL = "mysql"
|
|
|
|
)
|
|
|
|
|
2018-11-17 06:21:39 +01:00
|
|
|
// Config provides a struct in which application configuration is stored.
|
|
|
|
// 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
|
|
|
|
type Config struct {
|
2019-05-03 18:57:28 +02:00
|
|
|
Name string
|
|
|
|
Version string
|
|
|
|
Copyright string
|
2019-05-04 05:25:00 +02:00
|
|
|
Debug bool `yaml:"debug" flag:"debug"`
|
|
|
|
LogLevel string `yaml:"log-level" flag:"log-level"`
|
2019-05-03 18:57:28 +02:00
|
|
|
ConfigFile string
|
2019-05-04 05:25:00 +02:00
|
|
|
AssetsPath string `yaml:"assets-path" flag:"assets-path"`
|
|
|
|
CachePath string `yaml:"cache-path" flag:"cache-path"`
|
|
|
|
OriginalsPath string `yaml:"originals-path" flag:"originals-path"`
|
|
|
|
ImportPath string `yaml:"import-path" flag:"import-path"`
|
|
|
|
ExportPath string `yaml:"export-path" flag:"export-path"`
|
|
|
|
SqlServerHost string `yaml:"sql-host" flag:"sql-host"`
|
|
|
|
SqlServerPort uint `yaml:"sql-port" flag:"sql-port"`
|
|
|
|
SqlServerPath string `yaml:"sql-path" flag:"sql-path"`
|
|
|
|
SqlServerPassword string `yaml:"sql-password" flag:"sql-password"`
|
|
|
|
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"`
|
|
|
|
DarktableCli string `yaml:"darktable-cli" flag:"darktable-cli"`
|
|
|
|
DatabaseDriver string `yaml:"database-driver" flag:"database-driver"`
|
|
|
|
DatabaseDsn string `yaml:"database-dsn" flag:"database-dsn"`
|
2018-11-17 06:21:39 +01:00
|
|
|
}
|
|
|
|
|
2019-05-04 05:25:00 +02:00
|
|
|
// NewConfig() creates a new configuration entity by using two methods:
|
|
|
|
//
|
|
|
|
// 1. SetValuesFromFile: This will initialize values from a yaml config file.
|
|
|
|
//
|
|
|
|
// 2. SetValuesFromCliContext: Which comes after SetValuesFromFile and overrides
|
|
|
|
// any previous values giving an option two override file configs through the CLI.
|
|
|
|
func NewConfig(ctx *cli.Context) *Config {
|
|
|
|
c := &Config{}
|
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
|
2018-11-17 06:21:39 +01:00
|
|
|
|
2019-05-04 05:25:00 +02:00
|
|
|
if err := c.SetValuesFromFile(fsutil.ExpandedFilename(ctx.GlobalString("config-file"))); err != nil {
|
|
|
|
log.Debug(err)
|
2018-11-17 06:21:39 +01:00
|
|
|
}
|
|
|
|
|
2019-05-04 05:25:00 +02:00
|
|
|
if err := c.SetValuesFromCliContext(ctx); err != nil {
|
|
|
|
log.Error(err)
|
2018-11-17 06:21:39 +01:00
|
|
|
}
|
|
|
|
|
2019-05-04 05:25:00 +02:00
|
|
|
c.expandFilenames()
|
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
|
|
|
|
2019-05-04 05:25:00 +02:00
|
|
|
func (c *Config) expandFilenames() {
|
|
|
|
c.AssetsPath = fsutil.ExpandedFilename(c.AssetsPath)
|
|
|
|
c.CachePath = fsutil.ExpandedFilename(c.CachePath)
|
|
|
|
c.OriginalsPath = fsutil.ExpandedFilename(c.OriginalsPath)
|
|
|
|
c.ImportPath = fsutil.ExpandedFilename(c.ImportPath)
|
|
|
|
c.ExportPath = fsutil.ExpandedFilename(c.ExportPath)
|
|
|
|
c.DarktableCli = fsutil.ExpandedFilename(c.DarktableCli)
|
|
|
|
c.SqlServerPath = fsutil.ExpandedFilename(c.SqlServerPath)
|
|
|
|
}
|
2018-11-17 06:21:39 +01:00
|
|
|
|
2019-05-04 05:25:00 +02:00
|
|
|
// SetValuesFromFile uses a yaml config file to initiate the configuration entity.
|
|
|
|
func (c *Config) SetValuesFromFile(fileName string) error {
|
|
|
|
if !fsutil.Exists(fileName) {
|
|
|
|
return errors.New(fmt.Sprintf("config file not found: \"%s\"", 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
|
|
|
}
|
|
|
|
|
|
|
|
// SetValuesFromCliContext uses values from the CLI to setup configuration overrides
|
|
|
|
// for the entity.
|
|
|
|
func (c *Config) SetValuesFromCliContext(ctx *cli.Context) error {
|
2019-05-04 05:25:00 +02:00
|
|
|
v := reflect.ValueOf(c).Elem()
|
|
|
|
|
|
|
|
// Iterate through all config fields
|
|
|
|
for i := 0; i < v.NumField(); i++ {
|
|
|
|
fieldValue := v.Field(i)
|
|
|
|
|
|
|
|
tagValue := v.Type().Field(i).Tag.Get("flag")
|
|
|
|
|
|
|
|
// Automatically assign values to fields with "flag" tag
|
|
|
|
if tagValue != "" {
|
|
|
|
switch t := fieldValue.Interface().(type) {
|
|
|
|
case int, int64:
|
|
|
|
// Only if explicitly set or current value is empty (use default)
|
|
|
|
if ctx.GlobalIsSet(tagValue) || fieldValue.Int() == 0 {
|
|
|
|
f := ctx.GlobalInt64(tagValue)
|
|
|
|
fieldValue.SetInt(f)
|
|
|
|
}
|
|
|
|
case uint, uint64:
|
|
|
|
// Only if explicitly set or current value is empty (use default)
|
|
|
|
if ctx.GlobalIsSet(tagValue) || fieldValue.Uint() == 0 {
|
|
|
|
f := ctx.GlobalUint64(tagValue)
|
|
|
|
fieldValue.SetUint(f)
|
|
|
|
}
|
|
|
|
case string:
|
|
|
|
// Only if explicitly set or current value is empty (use default)
|
|
|
|
if ctx.GlobalIsSet(tagValue) || fieldValue.String() == "" {
|
|
|
|
f := ctx.GlobalString(tagValue)
|
|
|
|
fieldValue.SetString(f)
|
|
|
|
}
|
|
|
|
case bool:
|
|
|
|
if ctx.GlobalIsSet(tagValue) {
|
|
|
|
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
|
|
|
|
}
|