mirror of
https://github.com/postmannen/ctrl.git
synced 2024-12-14 12:37:31 +00:00
257 lines
10 KiB
Go
257 lines
10 KiB
Go
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
|
|
StartPubSayHello 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.
|
|
StartSubErrorLog flagNodeSlice
|
|
// Subscriber for hello messages
|
|
StartSubSayHello flagNodeSlice
|
|
// Subscriber for CLI Commands
|
|
StartSubCLICommand flagNodeSlice
|
|
// Subscriber for text logging
|
|
StartSubTextLogging flagNodeSlice
|
|
// Subscriber for Echo Request
|
|
StartSubEchoRequest flagNodeSlice
|
|
// Subscriber for Echo Reply
|
|
StartSubEchoReply flagNodeSlice
|
|
// Subscriber for CLICommandRequest
|
|
StartSubCLICommandRequest flagNodeSlice
|
|
// Subscriber for CLICommandRequestNOSEQ
|
|
StartSubCLICommandRequestNOSEQ flagNodeSlice
|
|
// Subscriber for CLICommandReply
|
|
StartSubCLICommandReply flagNodeSlice
|
|
// Subscriber for OpCommand
|
|
StartSubOpCommand flagNodeSlice
|
|
}
|
|
|
|
func NewConfiguration() *Configuration {
|
|
c := Configuration{}
|
|
return &c
|
|
}
|
|
|
|
// Default configuration
|
|
func newConfigurationDefaults() Configuration {
|
|
c := Configuration{
|
|
ConfigFolder: "./etc",
|
|
BrokerAddress: "127.0.0.1:4222",
|
|
ProfilingPort: "",
|
|
PromHostAndPort: "",
|
|
DefaultMessageTimeout: 10,
|
|
DefaultMessageRetries: 1,
|
|
StartPubSayHello: 30,
|
|
SubscribersDataFolder: "./data",
|
|
CentralNodeName: "",
|
|
StartSubErrorLog: flagNodeSlice{Values: []node{}},
|
|
StartSubSayHello: flagNodeSlice{OK: true, Values: []node{"*"}},
|
|
StartSubCLICommand: flagNodeSlice{OK: true, Values: []node{"*"}},
|
|
StartSubTextLogging: flagNodeSlice{OK: true, Values: []node{"*"}},
|
|
StartSubEchoRequest: flagNodeSlice{OK: true, Values: []node{"*"}},
|
|
StartSubEchoReply: flagNodeSlice{OK: true, Values: []node{"*"}},
|
|
StartSubCLICommandRequest: flagNodeSlice{OK: true, Values: []node{"*"}},
|
|
StartSubCLICommandRequestNOSEQ: flagNodeSlice{OK: true, Values: []node{"*"}},
|
|
StartSubCLICommandReply: flagNodeSlice{OK: true, Values: []node{"*"}},
|
|
StartSubOpCommand: flagNodeSlice{OK: true, Values: []node{"*"}},
|
|
}
|
|
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.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.IntVar(&c.StartPubSayHello, "startPubSayHello", fc.StartPubSayHello, "Make the current node send hello messages to central at given interval in seconds")
|
|
|
|
flag.Var(&c.StartSubErrorLog, "startSubErrorLog", "Specify comma separated list for nodes to allow messages from. Use \"*\" for from all. Value RST will turn off subscriber.")
|
|
flag.Var(&c.StartSubSayHello, "startSubSayHello", "Specify comma separated list for nodes to allow messages from. Use \"*\" for from all. Value RST will turn off subscriber.")
|
|
flag.Var(&c.StartSubCLICommand, "startSubCLICommand", "Specify comma separated list for nodes to allow messages from. Use \"*\" for from all. Value RST will turn off subscriber.")
|
|
flag.Var(&c.StartSubTextLogging, "startSubTextLogging", "Specify comma separated list for nodes to allow messages from. Use \"*\" for from all. Value RST will turn off subscriber.")
|
|
flag.Var(&c.StartSubEchoRequest, "startSubEchoRequest", "Specify comma separated list for nodes to allow messages from. Use \"*\" for from all. Value RST will turn off subscriber.")
|
|
flag.Var(&c.StartSubEchoReply, "startSubEchoReply", "Specify comma separated list for nodes to allow messages from. Use \"*\" for from all. Value RST will turn off subscriber.")
|
|
flag.Var(&c.StartSubCLICommandRequest, "startSubCLICommandRequest", "Specify comma separated list for nodes to allow messages from. Use \"*\" for from all. Value RST will turn off subscriber.")
|
|
flag.Var(&c.StartSubCLICommandRequestNOSEQ, "startSubCLICommandRequestNOSEQ", "Specify comma separated list for nodes to allow messages from. Use \"*\" for from all. Value RST will turn off subscriber.")
|
|
flag.Var(&c.StartSubCLICommandReply, "startSubCLICommandReply", "Specify comma separated list for nodes to allow messages from. Use \"*\" for from all. Value RST will turn off subscriber.")
|
|
flag.Var(&c.StartSubOpCommand, "startSubOpCommand", "Specify comma separated list for nodes to allow messages from. Use \"*\" for from all. Value RST will turn off subscriber.")
|
|
|
|
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
|
|
}
|