mirror of
https://github.com/crazy-max/diun.git
synced 2025-12-24 14:31:47 +01:00
180 lines
4.1 KiB
Go
180 lines
4.1 KiB
Go
package config
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"net/mail"
|
|
"os"
|
|
"path"
|
|
"regexp"
|
|
|
|
"github.com/crazy-max/diun/internal/model"
|
|
"github.com/imdario/mergo"
|
|
"github.com/rs/zerolog/log"
|
|
"gopkg.in/yaml.v2"
|
|
)
|
|
|
|
// Config holds configuration details
|
|
type Config struct {
|
|
Flags model.Flags
|
|
App model.App
|
|
Db model.Db `yaml:"db,omitempty"`
|
|
Watch model.Watch `yaml:"watch,omitempty"`
|
|
Notif model.Notif `yaml:"notif,omitempty"`
|
|
RegOpts map[string]model.RegOpts `yaml:"regopts,omitempty"`
|
|
Image []model.Image `yaml:"image,omitempty"`
|
|
}
|
|
|
|
// Load returns Configuration struct
|
|
func Load(flags model.Flags, version string) (*Config, error) {
|
|
var err error
|
|
var cfg = Config{
|
|
Flags: flags,
|
|
App: model.App{
|
|
ID: "diun",
|
|
Name: "Diun",
|
|
Desc: "Docker image update notifier",
|
|
URL: "https://github.com/crazy-max/diun",
|
|
Author: "CrazyMax",
|
|
Version: version,
|
|
},
|
|
Db: model.Db{
|
|
Path: "diun.db",
|
|
},
|
|
Watch: model.Watch{
|
|
Workers: 10,
|
|
Schedule: "0 0 * * * *",
|
|
},
|
|
Notif: model.Notif{
|
|
Mail: model.Mail{
|
|
Enable: false,
|
|
Host: "localhost",
|
|
Port: 25,
|
|
SSL: false,
|
|
InsecureSkipVerify: false,
|
|
},
|
|
Webhook: model.Webhook{
|
|
Enable: false,
|
|
Method: "GET",
|
|
Timeout: 10,
|
|
},
|
|
},
|
|
}
|
|
|
|
if _, err = os.Lstat(flags.Cfgfile); err != nil {
|
|
return nil, fmt.Errorf("unable to open config file, %s", err)
|
|
}
|
|
|
|
bytes, err := ioutil.ReadFile(flags.Cfgfile)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to read config file, %s", err)
|
|
}
|
|
|
|
if err := yaml.UnmarshalStrict(bytes, &cfg); err != nil {
|
|
return nil, fmt.Errorf("unable to decode into struct, %v", err)
|
|
}
|
|
|
|
if err := cfg.validate(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &cfg, nil
|
|
}
|
|
|
|
func (cfg *Config) validate() error {
|
|
if cfg.Flags.Docker {
|
|
cfg.Db.Path = "/data/diun.db"
|
|
}
|
|
|
|
if cfg.Db.Path == "" {
|
|
return errors.New("database path is required")
|
|
}
|
|
cfg.Db.Path = path.Clean(cfg.Db.Path)
|
|
|
|
for id, regopts := range cfg.RegOpts {
|
|
if err := cfg.validateRegOpts(id, regopts); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
for key, img := range cfg.Image {
|
|
if err := cfg.validateImage(key, img); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if cfg.Notif.Mail.Enable {
|
|
if _, err := mail.ParseAddress(cfg.Notif.Mail.From); err != nil {
|
|
return fmt.Errorf("cannot parse sender mail address, %v", err)
|
|
}
|
|
if _, err := mail.ParseAddress(cfg.Notif.Mail.To); err != nil {
|
|
return fmt.Errorf("cannot parse recipient mail address, %v", err)
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (cfg *Config) validateRegOpts(id string, regopts model.RegOpts) error {
|
|
defTimeout := 10
|
|
if regopts.Timeout <= 0 {
|
|
defTimeout = 0
|
|
}
|
|
|
|
if err := mergo.Merge(®opts, model.RegOpts{
|
|
InsecureTLS: false,
|
|
Timeout: defTimeout,
|
|
}); err != nil {
|
|
return fmt.Errorf("cannot set default registry options values for %s: %v", id, err)
|
|
}
|
|
|
|
cfg.RegOpts[id] = regopts
|
|
return nil
|
|
}
|
|
|
|
func (cfg *Config) validateImage(key int, img model.Image) error {
|
|
if img.Name == "" {
|
|
return fmt.Errorf("name is required for image %d", key)
|
|
}
|
|
|
|
if err := mergo.Merge(&img, model.Image{
|
|
Os: "linux",
|
|
Arch: "amd64",
|
|
WatchRepo: false,
|
|
MaxTags: 0,
|
|
}); err != nil {
|
|
return fmt.Errorf("cannot set default image values for %s: %v", img.Name, err)
|
|
}
|
|
|
|
if img.RegOptsID != "" {
|
|
regopts, found := cfg.RegOpts[img.RegOptsID]
|
|
if !found {
|
|
return fmt.Errorf("registry options %s not found for %s", img.RegOptsID, img.Name)
|
|
}
|
|
img.RegOpts = regopts
|
|
}
|
|
|
|
for _, includeTag := range img.IncludeTags {
|
|
if _, err := regexp.Compile(includeTag); err != nil {
|
|
return fmt.Errorf("include tag regex '%s' for %s cannot compile, %v", includeTag, img.Name, err)
|
|
}
|
|
}
|
|
|
|
for _, excludeTag := range img.ExcludeTags {
|
|
if _, err := regexp.Compile(excludeTag); err != nil {
|
|
return fmt.Errorf("exclude tag regex '%s' for '%s' image cannot compile, %v", img.Name, excludeTag, err)
|
|
}
|
|
}
|
|
|
|
cfg.Image[key] = img
|
|
return nil
|
|
}
|
|
|
|
// Display logs configuration in a pretty JSON format
|
|
func (cfg *Config) Display() {
|
|
b, _ := json.MarshalIndent(cfg, "", " ")
|
|
log.Debug().Msg(string(b))
|
|
}
|