mirror of
https://github.com/kubeshark/kubeshark.git
synced 2025-10-21 07:21:49 +00:00
223 lines
5.8 KiB
Go
223 lines
5.8 KiB
Go
package mizu
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"path"
|
|
"reflect"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/creasty/defaults"
|
|
"github.com/spf13/cobra"
|
|
"github.com/spf13/pflag"
|
|
"github.com/up9inc/mizu/cli/mizu/configStructs"
|
|
"github.com/up9inc/mizu/cli/uiUtils"
|
|
"gopkg.in/yaml.v3"
|
|
)
|
|
|
|
const (
|
|
Separator = "="
|
|
SetCommandName = "set"
|
|
)
|
|
|
|
var allowedSetFlags = []string{
|
|
AgentImageConfigName,
|
|
MizuResourcesNamespaceConfigName,
|
|
TelemetryConfigName,
|
|
DumpLogsConfigName,
|
|
configStructs.AnalysisDestinationTapName,
|
|
configStructs.SleepIntervalSecTapName,
|
|
}
|
|
|
|
var Config = ConfigStruct{}
|
|
|
|
func (config *ConfigStruct) Validate() error {
|
|
if config.IsNsRestrictedMode() {
|
|
if config.Tap.AllNamespaces || len(config.Tap.Namespaces) != 1 || config.Tap.Namespaces[0] != config.MizuResourcesNamespace {
|
|
return fmt.Errorf("Not supported mode. Mizu can't resolve IPs in other namespaces when running in namespace restricted mode.\n"+
|
|
"You can use the same namespace for --%s and --%s", configStructs.NamespacesTapName, MizuResourcesNamespaceConfigName)
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func InitConfig(cmd *cobra.Command) error {
|
|
if err := defaults.Set(&Config); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := mergeConfigFile(); err != nil {
|
|
Log.Errorf("Could not load config file, error %v", err)
|
|
Log.Fatalf("You can regenerate the file using `mizu config -r` or just remove it %v", GetConfigFilePath())
|
|
}
|
|
|
|
cmd.Flags().Visit(initFlag)
|
|
|
|
finalConfigPrettified, _ := uiUtils.PrettyJson(Config)
|
|
Log.Debugf("Init config finished\n Final config: %v", finalConfigPrettified)
|
|
|
|
return nil
|
|
}
|
|
|
|
func GetConfigWithDefaults() (string, error) {
|
|
defaultConf := ConfigStruct{}
|
|
if err := defaults.Set(&defaultConf); err != nil {
|
|
return "", err
|
|
}
|
|
return uiUtils.PrettyYaml(defaultConf)
|
|
}
|
|
|
|
func GetConfigFilePath() string {
|
|
return path.Join(GetMizuFolderPath(), "config.yaml")
|
|
}
|
|
|
|
func mergeConfigFile() error {
|
|
reader, openErr := os.Open(GetConfigFilePath())
|
|
if openErr != nil {
|
|
return nil
|
|
}
|
|
|
|
buf, readErr := ioutil.ReadAll(reader)
|
|
if readErr != nil {
|
|
return readErr
|
|
}
|
|
|
|
if err := yaml.Unmarshal(buf, &Config); err != nil {
|
|
return err
|
|
}
|
|
Log.Debugf("Found config file, merged to default options")
|
|
|
|
return nil
|
|
}
|
|
|
|
func initFlag(f *pflag.Flag) {
|
|
configElem := reflect.ValueOf(&Config).Elem()
|
|
|
|
sliceValue, isSliceValue := f.Value.(pflag.SliceValue)
|
|
if !isSliceValue {
|
|
mergeFlagValue(configElem, f.Name, f.Value.String())
|
|
return
|
|
}
|
|
|
|
if f.Name == SetCommandName {
|
|
mergeSetFlag(sliceValue.GetSlice())
|
|
return
|
|
}
|
|
|
|
mergeFlagValues(configElem, f.Name, sliceValue.GetSlice())
|
|
}
|
|
|
|
func mergeSetFlag(setValues []string) {
|
|
configElem := reflect.ValueOf(&Config).Elem()
|
|
|
|
for _, setValue := range setValues {
|
|
if !strings.Contains(setValue, Separator) {
|
|
Log.Warningf(uiUtils.Warning, fmt.Sprintf("Ignoring set argument %s (set argument format: <flag name>=<flag value>)", setValue))
|
|
}
|
|
|
|
split := strings.SplitN(setValue, Separator, 2)
|
|
if len(split) != 2 {
|
|
Log.Warningf(uiUtils.Warning, fmt.Sprintf("Ignoring set argument %s (set argument format: <flag name>=<flag value>)", setValue))
|
|
}
|
|
|
|
argumentKey, argumentValue := split[0], split[1]
|
|
|
|
if !Contains(allowedSetFlags, argumentKey) {
|
|
Log.Warningf(uiUtils.Warning, fmt.Sprintf("Ignoring set argument %s, flag name must be one of the following: \"%s\"", setValue, strings.Join(allowedSetFlags, "\", \"")))
|
|
}
|
|
|
|
mergeFlagValue(configElem, argumentKey, argumentValue)
|
|
}
|
|
}
|
|
|
|
func mergeFlagValue(currentElem reflect.Value, flagKey string, flagValue string) {
|
|
for i := 0; i < currentElem.NumField(); i++ {
|
|
currentField := currentElem.Type().Field(i)
|
|
currentFieldByName := currentElem.FieldByName(currentField.Name)
|
|
|
|
if currentField.Type.Kind() == reflect.Struct {
|
|
mergeFlagValue(currentFieldByName, flagKey, flagValue)
|
|
continue
|
|
}
|
|
|
|
if currentField.Tag.Get("yaml") != flagKey {
|
|
continue
|
|
}
|
|
|
|
flagValueKind := currentField.Type.Kind()
|
|
|
|
parsedValue, err := getParsedValue(flagValueKind, flagValue)
|
|
if err != nil {
|
|
Log.Warningf(uiUtils.Red, fmt.Sprintf("Invalid value %v for flag name %s, expected %s", flagValue, flagKey, flagValueKind))
|
|
return
|
|
}
|
|
|
|
currentFieldByName.Set(parsedValue)
|
|
}
|
|
}
|
|
|
|
func mergeFlagValues(currentElem reflect.Value, flagKey string, flagValues []string) {
|
|
for i := 0; i < currentElem.NumField(); i++ {
|
|
currentField := currentElem.Type().Field(i)
|
|
currentFieldByName := currentElem.FieldByName(currentField.Name)
|
|
|
|
if currentField.Type.Kind() == reflect.Struct {
|
|
mergeFlagValues(currentFieldByName, flagKey, flagValues)
|
|
continue
|
|
}
|
|
|
|
if currentField.Tag.Get("yaml") != flagKey {
|
|
continue
|
|
}
|
|
|
|
flagValueKind := currentField.Type.Elem().Kind()
|
|
|
|
parsedValues := reflect.MakeSlice(reflect.SliceOf(currentField.Type.Elem()), 0, 0)
|
|
for _, flagValue := range flagValues {
|
|
parsedValue, err := getParsedValue(flagValueKind, flagValue)
|
|
if err != nil {
|
|
Log.Warningf(uiUtils.Red, fmt.Sprintf("Invalid value %v for flag name %s, expected %s", flagValue, flagKey, flagValueKind))
|
|
return
|
|
}
|
|
|
|
parsedValues = reflect.Append(parsedValues, parsedValue)
|
|
}
|
|
|
|
currentFieldByName.Set(parsedValues)
|
|
}
|
|
}
|
|
|
|
func getParsedValue(kind reflect.Kind, value string) (reflect.Value, error) {
|
|
switch kind {
|
|
case reflect.String:
|
|
return reflect.ValueOf(value), nil
|
|
case reflect.Bool:
|
|
boolArgumentValue, err := strconv.ParseBool(value)
|
|
if err != nil {
|
|
break
|
|
}
|
|
|
|
return reflect.ValueOf(boolArgumentValue), nil
|
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
|
intArgumentValue, err := strconv.ParseInt(value, 10, 64)
|
|
if err != nil {
|
|
break
|
|
}
|
|
|
|
return reflect.ValueOf(intArgumentValue), nil
|
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
|
uintArgumentValue, err := strconv.ParseUint(value, 10, 64)
|
|
if err != nil {
|
|
break
|
|
}
|
|
|
|
return reflect.ValueOf(uintArgumentValue), nil
|
|
}
|
|
|
|
return reflect.ValueOf(nil), errors.New("value to parse does not match type")
|
|
}
|