package main import ( "errors" "fmt" "io/ioutil" "os" "strings" "time" "github.com/harmony-one/harmony/internal/cli" harmonyconfig "github.com/harmony-one/harmony/internal/configs/harmony" nodeconfig "github.com/harmony-one/harmony/internal/configs/node" "github.com/pelletier/go-toml" "github.com/spf13/cobra" ) // TODO: use specific type wise validation instead of general string types assertion. func validateHarmonyConfig(config harmonyconfig.HarmonyConfig) error { var accepts []string nodeType := config.General.NodeType accepts = []string{nodeTypeValidator, nodeTypeExplorer} if err := checkStringAccepted("--run", nodeType, accepts); err != nil { return err } netType := config.Network.NetworkType parsed := parseNetworkType(netType) if len(parsed) == 0 { return fmt.Errorf("unknown network type: %v", netType) } passType := config.BLSKeys.PassSrcType accepts = []string{blsPassTypeAuto, blsPassTypeFile, blsPassTypePrompt} if err := checkStringAccepted("--bls.pass.src", passType, accepts); err != nil { return err } kmsType := config.BLSKeys.KMSConfigSrcType accepts = []string{kmsConfigTypeShared, kmsConfigTypePrompt, kmsConfigTypeFile} if err := checkStringAccepted("--bls.kms.src", kmsType, accepts); err != nil { return err } if config.General.NodeType == nodeTypeExplorer && config.General.ShardID < 0 { return errors.New("flag --run.shard must be specified for explorer node") } if config.General.IsOffline && config.P2P.IP != nodeconfig.DefaultLocalListenIP { return fmt.Errorf("flag --run.offline must have p2p IP be %v", nodeconfig.DefaultLocalListenIP) } if !config.Sync.Downloader && !config.DNSSync.Client { // There is no module up for sync return errors.New("either --sync.downloader or --sync.legacy.client shall be enabled") } return nil } func sanityFixHarmonyConfig(hc *harmonyconfig.HarmonyConfig) { // When running sync downloader, set sync.Enabled to true if hc.Sync.Downloader && !hc.Sync.Enabled { fmt.Println("Set Sync.Enabled to true when running stream downloader") hc.Sync.Enabled = true } } func checkStringAccepted(flag string, val string, accepts []string) error { for _, accept := range accepts { if val == accept { return nil } } acceptsStr := strings.Join(accepts, ", ") return fmt.Errorf("unknown arg for %s: %s (%v)", flag, val, acceptsStr) } func getDefaultDNSSyncConfig(nt nodeconfig.NetworkType) harmonyconfig.DnsSync { zone := nodeconfig.GetDefaultDNSZone(nt) port := nodeconfig.GetDefaultDNSPort(nt) dnsSync := harmonyconfig.DnsSync{ Port: port, Zone: zone, LegacySyncing: false, ServerPort: nodeconfig.DefaultDNSPort, } switch nt { case nodeconfig.Mainnet: dnsSync.Server = true dnsSync.Client = true case nodeconfig.Testnet: dnsSync.Server = true dnsSync.Client = true case nodeconfig.Localnet: dnsSync.Server = true dnsSync.Client = false default: dnsSync.Server = true dnsSync.Client = false } return dnsSync } func getDefaultNetworkConfig(nt nodeconfig.NetworkType) harmonyconfig.NetworkConfig { bn := nodeconfig.GetDefaultBootNodes(nt) return harmonyconfig.NetworkConfig{ NetworkType: string(nt), BootNodes: bn, } } func parseNetworkType(nt string) nodeconfig.NetworkType { switch nt { case "mainnet": return nodeconfig.Mainnet case "testnet": return nodeconfig.Testnet case "pangaea", "staking", "stk": return nodeconfig.Pangaea case "partner": return nodeconfig.Partner case "stressnet", "stress", "stn": return nodeconfig.Stressnet case "localnet": return nodeconfig.Localnet case "devnet", "dev": return nodeconfig.Devnet default: return "" } } func getDefaultSyncConfig(nt nodeconfig.NetworkType) harmonyconfig.SyncConfig { switch nt { case nodeconfig.Mainnet: return defaultMainnetSyncConfig case nodeconfig.Testnet: return defaultTestNetSyncConfig case nodeconfig.Localnet: return defaultLocalNetSyncConfig default: return defaultElseSyncConfig } } var configCmd = &cobra.Command{ Use: "config", Short: "dump or update config", Long: "", } var updateConfigCmd = &cobra.Command{ Use: "update [config_file]", Short: "update config to latest version", Long: "updates config file to latest version, preserving values", Args: cobra.MinimumNArgs(1), Run: func(cmd *cobra.Command, args []string) { err := updateConfigFile(args[0]) if err != nil { fmt.Println(err) os.Exit(128) } }, } func dumpConfig(cmd *cobra.Command, args []string) { nt := getNetworkType(cmd) config := getDefaultHmyConfigCopy(nt) if err := writeHarmonyConfigToFile(config, args[0]); err != nil { fmt.Println(err) os.Exit(128) } } var dumpConfigCmd = &cobra.Command{ Use: "dump [config_file]", Short: "dump default config file for harmony binary configurations", Long: "dump default config file for harmony binary configurations", Args: cobra.MinimumNArgs(1), Run: dumpConfig, } var dumpConfigLegacyCmd = &cobra.Command{ Use: "dumpconfig [config_file]", Short: "depricated - use config dump instead", Long: "depricated - use config dump instead", Args: cobra.MinimumNArgs(1), Hidden: true, Run: dumpConfig, } func registerDumpConfigFlags() error { return cli.RegisterFlags(dumpConfigCmd, []cli.Flag{networkTypeFlag}) return cli.RegisterFlags(dumpConfigLegacyCmd, []cli.Flag{networkTypeFlag}) } func promptConfigUpdate() bool { var readStr string fmt.Println("Do you want to update config to the latest version: [y/N]") timeoutTimer := time.NewTimer(time.Second * 15) read := make(chan string) go func() { fmt.Scanf("%s", &readStr) read <- readStr }() select { case <-timeoutTimer.C: fmt.Println("Timed out - update manually with ./harmony config update [config_file]") return false case <-read: readStr = strings.TrimSpace(readStr) if len(readStr) > 1 { readStr = readStr[0:1] } readStr = strings.ToLower(readStr) return readStr == "y" } } func loadHarmonyConfig(file string) (harmonyconfig.HarmonyConfig, string, error) { b, err := ioutil.ReadFile(file) if err != nil { return harmonyconfig.HarmonyConfig{}, "", err } config, migratedVer, err := migrateConf(b) if err != nil { return harmonyconfig.HarmonyConfig{}, "", err } return config, migratedVer, nil } func updateConfigFile(file string) error { configBytes, err := ioutil.ReadFile(file) if err != nil { return err } backup := fmt.Sprintf("%s.backup", file) if err := ioutil.WriteFile(backup, configBytes, 0664); err != nil { return err } fmt.Printf("Original config backed up to %s\n", fmt.Sprintf("%s.backup", file)) config, migratedFromVer, err := migrateConf(configBytes) if err != nil { return err } if err := writeHarmonyConfigToFile(config, file); err != nil { return err } fmt.Printf("Successfully migrated %s from %s to %s \n", file, migratedFromVer, config.Version) return nil } func writeHarmonyConfigToFile(config harmonyconfig.HarmonyConfig, file string) error { b, err := toml.Marshal(config) if err != nil { return err } return ioutil.WriteFile(file, b, 0644) }