package steward import ( "flag" "fmt" "log" "os" "path/filepath" "strings" toml "github.com/pelletier/go-toml" ) // --- flag string slice // flagStringSlice is a type used when a flag value contains // comma separated values like `-myflag="somevalue,anothervalue`. // If a flag of this type is used, and it contains a value, the // Set(string) method will call the Parse() method. // The comma separated content will then be split, and put into // the []values field, and the `ok` field will be set to true, so // it can be used to check if the flag was used and contained any // values. type flagNodeSlice struct { value string OK bool Values []node } func (f *flagNodeSlice) String() string { return "" } // Set will be called when this flag type is used as a flag.Var. // It will put the comma separated string value given as input into // the `value`field, then call the Parse function to split those // comma separated values into []values. func (f *flagNodeSlice) Set(s string) error { f.value = s f.Parse() return nil } // If the flag value "RST" is given, set the default values // for each of the flag var's fields, and return back. // Since we reset the actual flag values, it is also these // blank values that will be written to the config file, and // making the change persistent in the system. This will also // be reflected in values stored in the config file, since the // config file is written after the flags have been parsed. func (f *flagNodeSlice) Parse() error { if len(f.value) == 0 { return nil } split := strings.Split(f.value, ",") // Reset values if RST was the flag value. if split[0] == "RST" { f.OK = false f.value = "" f.Values = []node{} return nil } fv := f.value sp := strings.Split(fv, ",") f.OK = true f.Values = []node{} for _, v := range sp { f.Values = append(f.Values, node(v)) } return nil } // --- Configuration type Configuration struct { // The configuration folder on disk ConfigFolder string // some unique string to identify this Edge unit NodeName string // the address of the message broker BrokerAddress string // The number of the profiling port ProfilingPort string // host and port for prometheus listener, e.g. localhost:2112 PromHostAndPort string // set to true if this is the node that should receive the error log's from other nodes DefaultMessageTimeout int // default amount of retries that will be done before a message is thrown away, and out of the system DefaultMessageRetries int // Make the current node send hello messages to central at given interval in seconds PublisherServiceSayhello int // Publisher data folder SubscribersDataFolder string // central node to receive messages published from nodes CentralNodeName string // Start the central error logger. // Takes a comma separated string of nodes to receive from or "*" for all nodes. StartCentralErrorLogger flagNodeSlice // default message timeout in seconds. This can be overridden on the message level") StartSayHelloSubscriber bool } func NewConfiguration() *Configuration { c := Configuration{} return &c } // Default configuration func newConfigurationDefaults() Configuration { c := Configuration{ ConfigFolder: "./etc", BrokerAddress: "127.0.0.1:4222", ProfilingPort: "", PromHostAndPort: "", StartCentralErrorLogger: flagNodeSlice{Values: []node{}}, DefaultMessageTimeout: 10, DefaultMessageRetries: 1, PublisherServiceSayhello: 30, SubscribersDataFolder: "./data", CentralNodeName: "", } return c } func (c *Configuration) CheckFlags() error { // Create an empty default config var fc Configuration // NB: Disabling the config file options for now. // Read file config. Set system default if it can't find config file. fc, err := c.ReadConfigFile() if err != nil { log.Printf("%v\n", err) fc = newConfigurationDefaults() } *c = fc flag.StringVar(&c.ConfigFolder, "configFolder", fc.ConfigFolder, "folder who contains the config file. Defaults to ./etc/. If other folder is used this flag must be specified at startup.") flag.StringVar(&c.NodeName, "nodeName", fc.NodeName, "some unique string to identify this Edge unit") flag.StringVar(&c.BrokerAddress, "brokerAddress", fc.BrokerAddress, "the address of the message broker") flag.StringVar(&c.ProfilingPort, "profilingPort", fc.ProfilingPort, "The number of the profiling port") flag.StringVar(&c.PromHostAndPort, "promHostAndPort", fc.PromHostAndPort, "host and port for prometheus listener, e.g. localhost:2112") flag.IntVar(&c.DefaultMessageTimeout, "defaultMessageTimeout", fc.DefaultMessageTimeout, "default message timeout in seconds. This can be overridden on the message level") flag.IntVar(&c.DefaultMessageRetries, "defaultMessageRetries", fc.DefaultMessageRetries, "default amount of retries that will be done before a message is thrown away, and out of the system") flag.IntVar(&c.PublisherServiceSayhello, "publisherServiceSayhello", fc.PublisherServiceSayhello, "Make the current node send hello messages to central at given interval in seconds") flag.StringVar(&c.SubscribersDataFolder, "subscribersDataFolder", fc.SubscribersDataFolder, "The data folder where subscribers are allowed to write their data if needed") flag.StringVar(&c.CentralNodeName, "centralNodeName", fc.CentralNodeName, "The name of the central node to receive messages published by this node") flag.Var(&c.StartCentralErrorLogger, "startCentralErrorLogger", "When value are given this node will become central error logger. Value can be \"*\" to receive from all hosts, or a comma separated list of hosts to allow processing messages from can be specified, like \"node1,node2\". Use the value RST to reset the value set in the config file to turn off the process.") flag.Parse() // Check that mandatory flag values have been set. switch { case c.NodeName == "": return fmt.Errorf("error: the nodeName config option or flag cannot be empty, check -help") case c.CentralNodeName == "": return fmt.Errorf("error: the centralNodeName config option or flag cannot be empty, check -help") } // NB: Disabling the config file options for now. if err := c.WriteConfigFile(); err != nil { log.Printf("error: checkFlags: failed writing config file: %v\n", err) os.Exit(1) } return nil } // Reads the current config file from disk. func (c *Configuration) ReadConfigFile() (Configuration, error) { fp := filepath.Join("./etc/", "config.toml") if _, err := os.Stat(fp); os.IsNotExist(err) { return Configuration{}, fmt.Errorf("error: no config file found %v: %v", fp, err) } f, err := os.OpenFile(fp, os.O_RDONLY, 0600) if err != nil { return Configuration{}, fmt.Errorf("error: ReadConfigFile: failed to open file: %v", err) } defer f.Close() var conf Configuration dec := toml.NewDecoder(f) err = dec.Decode(&conf) if err != nil { return Configuration{}, fmt.Errorf("error: decode toml file %v: %v", fp, err) } fmt.Printf("%+v\n", c) return conf, nil } // WriteConfigFile will write the current config to file. If the file or the // directory for the config file does not exist it will be created. func (c *Configuration) WriteConfigFile() error { if _, err := os.Stat(c.ConfigFolder); os.IsNotExist(err) { err := os.Mkdir(c.ConfigFolder, 0700) if err != nil { return fmt.Errorf("error: failed to create directory %v: %v", c.ConfigFolder, err) } } fp := filepath.Join(c.ConfigFolder, "config.toml") f, err := os.OpenFile(fp, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0600) if err != nil { return fmt.Errorf("error: WriteConfigFile: failed to open file: %v", err) } defer f.Close() enc := toml.NewEncoder(f) enc.Encode(c) return nil }