dave/app/config.go

130 lines
2.9 KiB
Go
Raw Normal View History

2018-04-09 14:12:54 +02:00
package app
import (
"fmt"
"github.com/fsnotify/fsnotify"
2018-04-09 15:27:21 +02:00
"github.com/spf13/viper"
"log"
2018-04-09 15:27:21 +02:00
"os"
"path/filepath"
2018-04-09 14:12:54 +02:00
)
// Config represents the configuration of the server application.
type Config struct {
Address string
Port string
Prefix string
Dir string
TLS *TLS
Users map[string]*UserInfo
}
// TLS allows specification of a certificate and private key file.
type TLS struct {
CertFile string
KeyFile string
2018-04-09 14:12:54 +02:00
}
// UserInfo allows storing of a password and user directory.
type UserInfo struct {
Password string
Subdir *string
}
2018-04-09 14:12:54 +02:00
// ParseConfig parses the application configuration an sets defaults.
func ParseConfig() *Config {
var cfg = &Config{}
2018-04-09 14:12:54 +02:00
2018-04-09 15:27:21 +02:00
setDefaults()
2018-04-09 14:12:54 +02:00
viper.SetConfigName("config")
viper.AddConfigPath("./config")
viper.AddConfigPath("$HOME/.swd")
viper.AddConfigPath(".")
err := viper.ReadInConfig()
if err != nil {
log.Fatal(fmt.Errorf("Fatal error config file: %s", err))
2018-04-09 14:12:54 +02:00
}
err = viper.Unmarshal(&cfg)
if err != nil {
log.Fatal(fmt.Errorf("Fatal error parsing config file: %s", err))
}
if cfg.TLS != nil {
if _, err := os.Stat(cfg.TLS.KeyFile); err != nil {
log.Fatal(fmt.Errorf("TLS keyFile doesn't exist: %s", err))
}
if _, err := os.Stat(cfg.TLS.CertFile); err != nil {
log.Fatal(fmt.Errorf("TLS certFile doesn't exist: %s", err))
}
2018-04-09 14:12:54 +02:00
}
viper.WatchConfig()
viper.OnConfigChange(cfg.updateConfig)
cfg.ensureUserDirs()
2018-04-09 14:12:54 +02:00
return cfg
}
// setDefaults adds some default values for the configuration
func setDefaults() {
viper.SetDefault("Address", "127.0.0.1")
viper.SetDefault("Port", "8000")
viper.SetDefault("Prefix", "")
viper.SetDefault("Dir", "/tmp")
viper.SetDefault("TLS", nil)
2018-04-09 14:12:54 +02:00
}
func (cfg *Config) updateConfig(e fsnotify.Event) {
fmt.Println("Config file changed:", e.Name)
file, err := os.Open(e.Name)
if err != nil {
fmt.Println("Error reloading config", e.Name)
}
var updatedCfg = &Config{}
viper.ReadConfig(file)
viper.Unmarshal(&updatedCfg)
for username := range cfg.Users {
if updatedCfg.Users[username] == nil {
fmt.Printf("Removed User from configuration: %s\n", username)
cfg.Users[username] = nil
}
}
for username, v := range updatedCfg.Users {
if cfg.Users[username] == nil {
fmt.Printf("Added User to configuration: %s\n", username)
cfg.Users[username] = v
} else {
if cfg.Users[username].Password != v.Password {
fmt.Printf("Updated password of user: %s\n", username)
cfg.Users[username].Password = v.Password
}
}
}
cfg.ensureUserDirs()
}
func (cfg *Config) ensureUserDirs() {
2018-04-11 15:58:21 +02:00
if _, err := os.Stat(cfg.Dir); os.IsNotExist(err) {
os.Mkdir(cfg.Dir, os.ModePerm)
fmt.Printf("Created base dir: %s\n", cfg.Dir)
}
for _, user := range cfg.Users {
if user.Subdir != nil {
path := filepath.Join(cfg.Dir, *user.Subdir)
if _, err := os.Stat(path); os.IsNotExist(err) {
os.Mkdir(path, os.ModePerm)
fmt.Printf("Created user dir: %s\n", path)
}
}
}
}