2021-03-01 19:49:43 +00:00
package steward
import (
"flag"
"fmt"
"log"
"os"
"path/filepath"
2021-03-24 06:05:41 +00:00
"strings"
2021-03-01 19:49:43 +00:00
toml "github.com/pelletier/go-toml"
)
2021-03-24 06:05:41 +00:00
// --- 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 flagStringSlice struct {
value string
ok bool
values [ ] string
}
func ( f * flagStringSlice ) 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 * flagStringSlice ) Set ( s string ) error {
f . value = s
f . Parse ( )
return nil
}
func ( f * flagStringSlice ) Parse ( ) error {
if len ( f . value ) == 0 {
return nil
}
fv := f . value
sp := strings . Split ( fv , "," )
f . ok = true
f . values = sp
return nil
}
// --- Configuration
2021-03-01 19:49:43 +00:00
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
2021-03-02 12:46:02 +00:00
// Publisher data folder
SubscribersDataFolder string
2021-03-12 09:41:03 +00:00
// central node to receive messages published from nodes
2021-03-12 10:13:42 +00:00
CentralNodeName string
2021-03-24 06:05:41 +00:00
// HERE: SHOULD TAKE STRING
// ALSO RENAME ALL STARTING FLAGS TO BEGIN WITH START
StartCentralErrorLogger bool
// default message timeout in seconds. This can be overridden on the message level")
StartSayHelloSubscriber bool
2021-03-01 19:49:43 +00:00
}
func NewConfiguration ( ) * Configuration {
c := Configuration { }
return & c
}
2021-03-12 10:13:42 +00:00
// Default configuration
2021-03-02 05:51:08 +00:00
func newConfigurationDefaults ( ) Configuration {
c := Configuration {
ConfigFolder : "./etc" ,
BrokerAddress : "127.0.0.1:4222" ,
ProfilingPort : "" ,
PromHostAndPort : "" ,
2021-03-24 06:05:41 +00:00
StartCentralErrorLogger : false ,
2021-03-02 05:51:08 +00:00
DefaultMessageTimeout : 10 ,
DefaultMessageRetries : 1 ,
PublisherServiceSayhello : 30 ,
2021-03-02 12:46:02 +00:00
SubscribersDataFolder : "./data" ,
2021-03-12 10:13:42 +00:00
CentralNodeName : "" ,
2021-03-02 05:51:08 +00:00
}
return c
}
2021-03-01 19:49:43 +00:00
func ( c * Configuration ) CheckFlags ( ) {
2021-03-02 05:51:08 +00:00
// Create an empty default config
var fc Configuration
// Read file config. Set system default if it can't find config file.
2021-03-01 19:49:43 +00:00
fc , err := c . ReadConfigFile ( )
if err != nil {
log . Printf ( "%v\n" , err )
2021-03-02 05:51:08 +00:00
fc = newConfigurationDefaults ( )
2021-03-01 19:49:43 +00:00
}
2021-03-12 05:14:54 +00:00
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." )
2021-03-01 19:49:43 +00:00
flag . StringVar ( & c . NodeName , "node" , 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" )
2021-03-24 06:05:41 +00:00
flag . BoolVar ( & c . StartCentralErrorLogger , "startCentralErrorLogger" , fc . StartCentralErrorLogger , "set to true if this is the node that should receive the error log's from other nodes" )
2021-03-01 19:49:43 +00:00
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" )
2021-03-02 12:46:02 +00:00
flag . StringVar ( & c . SubscribersDataFolder , "subscribersDataFolder" , fc . SubscribersDataFolder , "The data folder where subscribers are allowed to write their data if needed" )
2021-03-12 10:13:42 +00:00
flag . StringVar ( & c . CentralNodeName , "centralNodeName" , fc . CentralNodeName , "The name of the central node to receive messages published by this node" )
2021-03-02 12:46:02 +00:00
2021-03-01 19:49:43 +00:00
flag . Parse ( )
if err := c . WriteConfigFile ( ) ; err != nil {
log . Printf ( "error: checkFlags: failed writing config file: %v\n" , err )
os . Exit ( 1 )
}
}
2021-03-24 06:05:41 +00:00
// Reads the current config file from disk.
2021-03-01 19:49:43 +00:00
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 {
2021-03-02 12:46:02 +00:00
return Configuration { } , fmt . Errorf ( "error: ReadConfigFile: failed to open file: %v" , err )
2021-03-01 19:49:43 +00:00
}
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
}
2021-03-02 05:51:08 +00:00
// 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.
2021-03-01 19:49:43 +00:00
func ( c * Configuration ) WriteConfigFile ( ) error {
if _ , err := os . Stat ( c . ConfigFolder ) ; os . IsNotExist ( err ) {
2021-03-02 05:51:08 +00:00
err := os . Mkdir ( c . ConfigFolder , 0700 )
2021-03-01 19:49:43 +00:00
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 {
2021-03-02 12:46:02 +00:00
return fmt . Errorf ( "error: WriteConfigFile: failed to open file: %v" , err )
2021-03-01 19:49:43 +00:00
}
defer f . Close ( )
enc := toml . NewEncoder ( f )
enc . Encode ( c )
return nil
}