2020-12-18 20:42:12 +01:00
|
|
|
package config
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
2021-10-06 07:10:50 +02:00
|
|
|
"os"
|
2020-12-18 20:42:12 +01:00
|
|
|
"reflect"
|
2022-04-03 14:51:58 +02:00
|
|
|
"time"
|
|
|
|
|
2020-12-18 20:42:12 +01:00
|
|
|
"github.com/urfave/cli"
|
|
|
|
"gopkg.in/yaml.v2"
|
2021-12-09 07:47:23 +01:00
|
|
|
|
2022-04-15 09:42:07 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/clean"
|
2021-12-09 07:47:23 +01:00
|
|
|
"github.com/photoprism/photoprism/pkg/fs"
|
2022-04-08 14:22:07 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/txt"
|
2020-12-18 20:42:12 +01:00
|
|
|
)
|
|
|
|
|
2022-04-08 14:22:07 +02:00
|
|
|
// Options hold the global configuration values without further validation or processing.
|
|
|
|
// Application code should retrieve option values via getter functions since they provide
|
|
|
|
// validation and return defaults if a value is empty.
|
2020-12-18 20:42:12 +01:00
|
|
|
type Options struct {
|
2022-04-03 14:51:58 +02:00
|
|
|
Name string `json:"-"`
|
2022-05-20 19:27:33 +02:00
|
|
|
Edition string `json:"-"`
|
2022-04-03 14:51:58 +02:00
|
|
|
Version string `json:"-"`
|
|
|
|
Copyright string `json:"-"`
|
|
|
|
PartnerID string `yaml:"-" json:"-" flag:"partner-id"`
|
2022-04-12 20:00:06 +02:00
|
|
|
AdminPassword string `yaml:"AdminPassword" json:"-" flag:"admin-password"`
|
2022-04-03 14:51:58 +02:00
|
|
|
LogLevel string `yaml:"LogLevel" json:"-" flag:"log-level"`
|
|
|
|
Debug bool `yaml:"Debug" json:"Debug" flag:"debug"`
|
2022-04-12 19:14:21 +02:00
|
|
|
Trace bool `yaml:"Trace" json:"Trace" flag:"Trace"`
|
2022-07-05 23:13:34 +02:00
|
|
|
AuthMode string `yaml:"AuthMode" json:"-" flag:"auth-mode"`
|
2022-04-12 19:14:21 +02:00
|
|
|
Public bool `yaml:"Public" json:"-" flag:"public"`
|
2022-04-03 14:51:58 +02:00
|
|
|
Test bool `yaml:"-" json:"Test,omitempty" flag:"test"`
|
|
|
|
Unsafe bool `yaml:"-" json:"-" flag:"unsafe"`
|
|
|
|
Demo bool `yaml:"Demo" json:"-" flag:"demo"`
|
|
|
|
Sponsor bool `yaml:"-" json:"-" flag:"sponsor"`
|
|
|
|
ReadOnly bool `yaml:"ReadOnly" json:"ReadOnly" flag:"read-only"`
|
|
|
|
Experimental bool `yaml:"Experimental" json:"Experimental" flag:"experimental"`
|
|
|
|
ConfigPath string `yaml:"ConfigPath" json:"-" flag:"config-path"`
|
2022-04-13 22:43:49 +02:00
|
|
|
DefaultsYaml string `json:"-" yaml:"-" flag:"defaults-yaml"`
|
2022-04-03 14:51:58 +02:00
|
|
|
OriginalsPath string `yaml:"OriginalsPath" json:"-" flag:"originals-path"`
|
|
|
|
OriginalsLimit int `yaml:"OriginalsLimit" json:"OriginalsLimit" flag:"originals-limit"`
|
|
|
|
ResolutionLimit int `yaml:"ResolutionLimit" json:"ResolutionLimit" flag:"resolution-limit"`
|
|
|
|
StoragePath string `yaml:"StoragePath" json:"-" flag:"storage-path"`
|
|
|
|
SidecarPath string `yaml:"SidecarPath" json:"-" flag:"sidecar-path"`
|
|
|
|
BackupPath string `yaml:"BackupPath" json:"-" flag:"backup-path"`
|
2022-04-18 13:55:17 +02:00
|
|
|
CachePath string `yaml:"CachePath" json:"-" flag:"cache-path"`
|
2022-04-07 11:56:44 +02:00
|
|
|
ImportPath string `yaml:"ImportPath" json:"-" flag:"import-path"`
|
2022-04-03 14:51:58 +02:00
|
|
|
AssetsPath string `yaml:"AssetsPath" json:"-" flag:"assets-path"`
|
2022-07-05 23:13:34 +02:00
|
|
|
CustomAssetsPath string `yaml:"-" json:"-" flag:"custom-assets-path"`
|
2022-04-07 11:56:44 +02:00
|
|
|
TempPath string `yaml:"TempPath" json:"-" flag:"temp-path"`
|
2022-04-03 14:51:58 +02:00
|
|
|
Workers int `yaml:"Workers" json:"Workers" flag:"workers"`
|
|
|
|
WakeupInterval time.Duration `yaml:"WakeupInterval" json:"WakeupInterval" flag:"wakeup-interval"`
|
|
|
|
AutoIndex int `yaml:"AutoIndex" json:"AutoIndex" flag:"auto-index"`
|
|
|
|
AutoImport int `yaml:"AutoImport" json:"AutoImport" flag:"auto-import"`
|
|
|
|
DisableWebDAV bool `yaml:"DisableWebDAV" json:"DisableWebDAV" flag:"disable-webdav"`
|
|
|
|
DisableBackups bool `yaml:"DisableBackups" json:"DisableBackups" flag:"disable-backups"`
|
|
|
|
DisableSettings bool `yaml:"DisableSettings" json:"-" flag:"disable-settings"`
|
|
|
|
DisablePlaces bool `yaml:"DisablePlaces" json:"DisablePlaces" flag:"disable-places"`
|
|
|
|
DisableTensorFlow bool `yaml:"DisableTensorFlow" json:"DisableTensorFlow" flag:"disable-tensorflow"`
|
|
|
|
DisableFaces bool `yaml:"DisableFaces" json:"DisableFaces" flag:"disable-faces"`
|
|
|
|
DisableClassification bool `yaml:"DisableClassification" json:"DisableClassification" flag:"disable-classification"`
|
|
|
|
DisableFFmpeg bool `yaml:"DisableFFmpeg" json:"DisableFFmpeg" flag:"disable-ffmpeg"`
|
|
|
|
DisableExifTool bool `yaml:"DisableExifTool" json:"DisableExifTool" flag:"disable-exiftool"`
|
2022-04-06 17:46:41 +02:00
|
|
|
DisableHeifConvert bool `yaml:"DisableHeifConvert" json:"DisableHeifConvert" flag:"disable-heifconvert"`
|
|
|
|
DisableDarktable bool `yaml:"DisableDarktable" json:"DisableDarktable" flag:"disable-darktable"`
|
|
|
|
DisableRawtherapee bool `yaml:"DisableRawtherapee" json:"DisableRawtherapee" flag:"disable-rawtherapee"`
|
|
|
|
DisableSips bool `yaml:"DisableSips" json:"DisableSips" flag:"disable-sips"`
|
|
|
|
DisableRaw bool `yaml:"DisableRaw" json:"DisableRaw" flag:"disable-raw"`
|
2022-04-03 14:51:58 +02:00
|
|
|
RawPresets bool `yaml:"RawPresets" json:"RawPresets" flag:"raw-presets"`
|
2022-04-06 17:46:41 +02:00
|
|
|
ExifBruteForce bool `yaml:"ExifBruteForce" json:"ExifBruteForce" flag:"exif-bruteforce"`
|
2022-04-03 14:51:58 +02:00
|
|
|
DetectNSFW bool `yaml:"DetectNSFW" json:"DetectNSFW" flag:"detect-nsfw"`
|
|
|
|
UploadNSFW bool `yaml:"UploadNSFW" json:"-" flag:"upload-nsfw"`
|
|
|
|
DefaultTheme string `yaml:"DefaultTheme" json:"DefaultTheme" flag:"default-theme"`
|
|
|
|
DefaultLocale string `yaml:"DefaultLocale" json:"DefaultLocale" flag:"default-locale"`
|
|
|
|
AppIcon string `yaml:"AppIcon" json:"AppIcon" flag:"app-icon"`
|
|
|
|
AppName string `yaml:"AppName" json:"AppName" flag:"app-name"`
|
|
|
|
AppMode string `yaml:"AppMode" json:"AppMode" flag:"app-mode"`
|
2022-04-22 17:38:40 +02:00
|
|
|
WallpaperUri string `yaml:"WallpaperUri" json:"WallpaperUri" flag:"wallpaper-uri"`
|
2022-04-03 14:51:58 +02:00
|
|
|
CdnUrl string `yaml:"CdnUrl" json:"CdnUrl" flag:"cdn-url"`
|
|
|
|
SiteUrl string `yaml:"SiteUrl" json:"SiteUrl" flag:"site-url"`
|
|
|
|
SiteAuthor string `yaml:"SiteAuthor" json:"SiteAuthor" flag:"site-author"`
|
|
|
|
SiteTitle string `yaml:"SiteTitle" json:"SiteTitle" flag:"site-title"`
|
|
|
|
SiteCaption string `yaml:"SiteCaption" json:"SiteCaption" flag:"site-caption"`
|
|
|
|
SiteDescription string `yaml:"SiteDescription" json:"SiteDescription" flag:"site-description"`
|
|
|
|
SitePreview string `yaml:"SitePreview" json:"SitePreview" flag:"site-preview"`
|
|
|
|
Imprint string `yaml:"Imprint" json:"Imprint" flag:"imprint"`
|
|
|
|
ImprintUrl string `yaml:"ImprintUrl" json:"ImprintUrl" flag:"imprint-url"`
|
|
|
|
DatabaseDriver string `yaml:"DatabaseDriver" json:"-" flag:"database-driver"`
|
|
|
|
DatabaseDsn string `yaml:"DatabaseDsn" json:"-" flag:"database-dsn"`
|
|
|
|
DatabaseServer string `yaml:"DatabaseServer" json:"-" flag:"database-server"`
|
|
|
|
DatabaseName string `yaml:"DatabaseName" json:"-" flag:"database-name"`
|
|
|
|
DatabaseUser string `yaml:"DatabaseUser" json:"-" flag:"database-user"`
|
|
|
|
DatabasePassword string `yaml:"DatabasePassword" json:"-" flag:"database-password"`
|
|
|
|
DatabaseConns int `yaml:"DatabaseConns" json:"-" flag:"database-conns"`
|
|
|
|
DatabaseConnsIdle int `yaml:"DatabaseConnsIdle" json:"-" flag:"database-conns-idle"`
|
|
|
|
HttpHost string `yaml:"HttpHost" json:"-" flag:"http-host"`
|
|
|
|
HttpPort int `yaml:"HttpPort" json:"-" flag:"http-port"`
|
|
|
|
HttpMode string `yaml:"HttpMode" json:"-" flag:"http-mode"`
|
|
|
|
HttpCompression string `yaml:"HttpCompression" json:"-" flag:"http-compression"`
|
|
|
|
DarktableBin string `yaml:"DarktableBin" json:"-" flag:"darktable-bin"`
|
2022-04-06 17:46:41 +02:00
|
|
|
DarktableCachePath string `yaml:"DarktableCachePath" json:"-" flag:"darktable-cache-path"`
|
|
|
|
DarktableConfigPath string `yaml:"DarktableConfigPath" json:"-" flag:"darktable-config-path"`
|
2022-04-03 14:51:58 +02:00
|
|
|
DarktableBlacklist string `yaml:"DarktableBlacklist" json:"-" flag:"darktable-blacklist"`
|
|
|
|
RawtherapeeBin string `yaml:"RawtherapeeBin" json:"-" flag:"rawtherapee-bin"`
|
|
|
|
RawtherapeeBlacklist string `yaml:"RawtherapeeBlacklist" json:"-" flag:"rawtherapee-blacklist"`
|
|
|
|
SipsBin string `yaml:"SipsBin" json:"-" flag:"sips-bin"`
|
|
|
|
HeifConvertBin string `yaml:"HeifConvertBin" json:"-" flag:"heifconvert-bin"`
|
|
|
|
FFmpegBin string `yaml:"FFmpegBin" json:"-" flag:"ffmpeg-bin"`
|
|
|
|
FFmpegEncoder string `yaml:"FFmpegEncoder" json:"FFmpegEncoder" flag:"ffmpeg-encoder"`
|
|
|
|
FFmpegBitrate int `yaml:"FFmpegBitrate" json:"FFmpegBitrate" flag:"ffmpeg-bitrate"`
|
|
|
|
ExifToolBin string `yaml:"ExifToolBin" json:"-" flag:"exiftool-bin"`
|
|
|
|
DetachServer bool `yaml:"DetachServer" json:"-" flag:"detach-server"`
|
|
|
|
DownloadToken string `yaml:"DownloadToken" json:"-" flag:"download-token"`
|
|
|
|
PreviewToken string `yaml:"PreviewToken" json:"-" flag:"preview-token"`
|
2022-04-06 18:41:15 +02:00
|
|
|
ThumbColor string `yaml:"ThumbColor" json:"ThumbColor" flag:"thumb-color"`
|
2022-04-07 11:56:44 +02:00
|
|
|
ThumbFilter string `yaml:"ThumbFilter" json:"ThumbFilter" flag:"thumb-filter"`
|
2022-04-03 14:51:58 +02:00
|
|
|
ThumbSize int `yaml:"ThumbSize" json:"ThumbSize" flag:"thumb-size"`
|
|
|
|
ThumbSizeUncached int `yaml:"ThumbSizeUncached" json:"ThumbSizeUncached" flag:"thumb-size-uncached"`
|
2022-04-08 14:45:03 +02:00
|
|
|
ThumbUncached bool `yaml:"ThumbUncached" json:"ThumbUncached" flag:"thumb-uncached"`
|
2022-04-03 14:51:58 +02:00
|
|
|
JpegQuality string `yaml:"JpegQuality" json:"JpegQuality" flag:"jpeg-quality"`
|
2022-04-08 14:45:03 +02:00
|
|
|
JpegSize int `yaml:"JpegSize" json:"JpegSize" flag:"jpeg-size"`
|
2022-04-03 14:51:58 +02:00
|
|
|
FaceSize int `yaml:"-" json:"-" flag:"face-size"`
|
|
|
|
FaceScore float64 `yaml:"-" json:"-" flag:"face-score"`
|
|
|
|
FaceOverlap int `yaml:"-" json:"-" flag:"face-overlap"`
|
|
|
|
FaceClusterSize int `yaml:"-" json:"-" flag:"face-cluster-size"`
|
|
|
|
FaceClusterScore int `yaml:"-" json:"-" flag:"face-cluster-score"`
|
|
|
|
FaceClusterCore int `yaml:"-" json:"-" flag:"face-cluster-core"`
|
|
|
|
FaceClusterDist float64 `yaml:"-" json:"-" flag:"face-cluster-dist"`
|
|
|
|
FaceMatchDist float64 `yaml:"-" json:"-" flag:"face-match-dist"`
|
|
|
|
PIDFilename string `yaml:"PIDFilename" json:"-" flag:"pid-filename"`
|
|
|
|
LogFilename string `yaml:"LogFilename" json:"-" flag:"log-filename"`
|
2020-12-18 20:42:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewOptions creates a new configuration entity by using two methods:
|
|
|
|
//
|
|
|
|
// 1. Load: This will initialize options from a yaml config file.
|
|
|
|
//
|
|
|
|
// 2. SetContext: Which comes after Load and overrides
|
|
|
|
// any previous options giving an option two override file configs through the CLI.
|
|
|
|
func NewOptions(ctx *cli.Context) *Options {
|
|
|
|
c := &Options{}
|
|
|
|
|
2022-05-20 19:27:33 +02:00
|
|
|
// Has context?
|
2020-12-18 20:42:12 +01:00
|
|
|
if ctx == nil {
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
2022-05-20 15:41:18 +02:00
|
|
|
// Set app name from metadata if possible.
|
|
|
|
if s, ok := ctx.App.Metadata["Name"]; ok {
|
2022-05-20 19:27:33 +02:00
|
|
|
c.Name = fmt.Sprintf("%s", s)
|
2022-05-20 15:41:18 +02:00
|
|
|
}
|
|
|
|
|
2022-05-20 19:27:33 +02:00
|
|
|
// Set app edition from metadata if possible.
|
|
|
|
if s, ok := ctx.App.Metadata["Edition"]; ok {
|
|
|
|
c.Edition = fmt.Sprintf("%s", s)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set copyright and version information.
|
2020-12-18 20:42:12 +01:00
|
|
|
c.Copyright = ctx.App.Copyright
|
|
|
|
c.Version = ctx.App.Version
|
|
|
|
|
2022-04-13 22:43:49 +02:00
|
|
|
// Load defaults from YAML file?
|
|
|
|
if defaultsYaml := ctx.GlobalString("defaults-yaml"); defaultsYaml == "" {
|
|
|
|
log.Tracef("config: defaults yaml file not specified")
|
|
|
|
} else if c.DefaultsYaml = fs.Abs(defaultsYaml); !fs.FileExists(c.DefaultsYaml) {
|
2022-04-15 09:42:07 +02:00
|
|
|
log.Tracef("config: defaults file %s does not exist", clean.Log(c.DefaultsYaml))
|
2022-04-13 22:43:49 +02:00
|
|
|
} else if err := c.Load(c.DefaultsYaml); err != nil {
|
2022-04-15 09:42:07 +02:00
|
|
|
log.Warnf("config: failed loading defaults from %s (%s)", clean.Log(c.DefaultsYaml), err)
|
2020-12-18 20:42:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if err := c.SetContext(ctx); err != nil {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
// expandFilenames converts path in config to absolute path
|
|
|
|
func (c *Options) expandFilenames() {
|
|
|
|
c.ConfigPath = fs.Abs(c.ConfigPath)
|
|
|
|
c.StoragePath = fs.Abs(c.StoragePath)
|
|
|
|
c.BackupPath = fs.Abs(c.BackupPath)
|
|
|
|
c.AssetsPath = fs.Abs(c.AssetsPath)
|
|
|
|
c.CachePath = fs.Abs(c.CachePath)
|
|
|
|
c.OriginalsPath = fs.Abs(c.OriginalsPath)
|
|
|
|
c.ImportPath = fs.Abs(c.ImportPath)
|
|
|
|
c.TempPath = fs.Abs(c.TempPath)
|
|
|
|
c.PIDFilename = fs.Abs(c.PIDFilename)
|
|
|
|
c.LogFilename = fs.Abs(c.LogFilename)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load uses a yaml config file to initiate the configuration entity.
|
|
|
|
func (c *Options) Load(fileName string) error {
|
|
|
|
if fileName == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if !fs.FileExists(fileName) {
|
2022-04-13 22:43:49 +02:00
|
|
|
return errors.New(fmt.Sprintf("%s not found", fileName))
|
2020-12-18 20:42:12 +01:00
|
|
|
}
|
|
|
|
|
2021-10-06 07:10:50 +02:00
|
|
|
yamlConfig, err := os.ReadFile(fileName)
|
2020-12-18 20:42:12 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return yaml.Unmarshal(yamlConfig, c)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetContext uses options from the CLI to setup configuration overrides
|
|
|
|
// for the entity.
|
|
|
|
func (c *Options) SetContext(ctx *cli.Context) error {
|
|
|
|
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")
|
|
|
|
|
2021-09-23 13:16:05 +02:00
|
|
|
// Assign value to field with "flag" tag.
|
2020-12-18 20:42:12 +01:00
|
|
|
if tagValue != "" {
|
|
|
|
switch t := fieldValue.Interface().(type) {
|
2022-04-03 14:51:58 +02:00
|
|
|
case time.Duration:
|
|
|
|
var s string
|
|
|
|
|
|
|
|
// Get duration string.
|
|
|
|
if ctx.IsSet(tagValue) {
|
|
|
|
s = ctx.String(tagValue)
|
|
|
|
} else if ctx.GlobalIsSet(tagValue) || fieldValue.Interface().(time.Duration) == 0 {
|
|
|
|
s = ctx.GlobalString(tagValue)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse duration string.
|
|
|
|
if s == "" {
|
|
|
|
// Omit.
|
|
|
|
} else if sec := txt.UInt(s); sec > 0 {
|
|
|
|
fieldValue.Set(reflect.ValueOf(time.Duration(sec) * time.Second))
|
|
|
|
} else if d, err := time.ParseDuration(s); err == nil {
|
|
|
|
fieldValue.Set(reflect.ValueOf(d))
|
|
|
|
}
|
2021-09-23 13:16:05 +02:00
|
|
|
case float64:
|
|
|
|
// Only if explicitly set or current value is empty (use default).
|
|
|
|
if ctx.IsSet(tagValue) {
|
|
|
|
f := ctx.Float64(tagValue)
|
|
|
|
fieldValue.SetFloat(f)
|
|
|
|
} else if ctx.GlobalIsSet(tagValue) || fieldValue.Float() == 0 {
|
|
|
|
f := ctx.GlobalFloat64(tagValue)
|
|
|
|
fieldValue.SetFloat(f)
|
|
|
|
}
|
2020-12-18 20:42:12 +01:00
|
|
|
case int, int64:
|
|
|
|
// Only if explicitly set or current value is empty (use default).
|
|
|
|
if ctx.IsSet(tagValue) {
|
|
|
|
f := ctx.Int64(tagValue)
|
|
|
|
fieldValue.SetInt(f)
|
|
|
|
} else 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.IsSet(tagValue) {
|
|
|
|
f := ctx.Uint64(tagValue)
|
|
|
|
fieldValue.SetUint(f)
|
|
|
|
} else 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.IsSet(tagValue) {
|
|
|
|
f := ctx.String(tagValue)
|
|
|
|
fieldValue.SetString(f)
|
|
|
|
} else if ctx.GlobalIsSet(tagValue) || fieldValue.String() == "" {
|
|
|
|
f := ctx.GlobalString(tagValue)
|
|
|
|
fieldValue.SetString(f)
|
|
|
|
}
|
|
|
|
case bool:
|
|
|
|
if ctx.IsSet(tagValue) {
|
|
|
|
f := ctx.Bool(tagValue)
|
|
|
|
fieldValue.SetBool(f)
|
|
|
|
} else if ctx.GlobalIsSet(tagValue) {
|
|
|
|
f := ctx.GlobalBool(tagValue)
|
|
|
|
fieldValue.SetBool(f)
|
|
|
|
}
|
|
|
|
default:
|
2022-01-05 11:40:44 +01:00
|
|
|
log.Warnf("cannot assign value of type %s from cli flag %s", t, tagValue)
|
2020-12-18 20:42:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|