You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
270 lines
8.7 KiB
270 lines
8.7 KiB
// Copyright (c) 2013-2015 The btcsuite developers |
|
// Use of this source code is governed by an ISC |
|
// license that can be found in the LICENSE file. |
|
|
|
package main |
|
|
|
import ( |
|
"fmt" |
|
"net" |
|
"os" |
|
"path/filepath" |
|
"strings" |
|
|
|
"github.com/btcsuite/btcd/btcjson" |
|
"github.com/btcsuite/btcutil" |
|
flags "github.com/btcsuite/go-flags" |
|
) |
|
|
|
const ( |
|
// unusableFlags are the command usage flags which this utility are not |
|
// able to use. In particular it doesn't support websockets and |
|
// consequently notifications. |
|
unusableFlags = btcjson.UFWebsocketOnly | btcjson.UFNotification |
|
) |
|
|
|
var ( |
|
btcdHomeDir = btcutil.AppDataDir("btcd", false) |
|
btcctlHomeDir = btcutil.AppDataDir("btcctl", false) |
|
btcwalletHomeDir = btcutil.AppDataDir("btcwallet", false) |
|
defaultConfigFile = filepath.Join(btcctlHomeDir, "btcctl.conf") |
|
defaultRPCServer = "localhost" |
|
defaultRPCCertFile = filepath.Join(btcdHomeDir, "rpc.cert") |
|
defaultWalletCertFile = filepath.Join(btcwalletHomeDir, "rpc.cert") |
|
) |
|
|
|
// listCommands categorizes and lists all of the usable commands along with |
|
// their one-line usage. |
|
func listCommands() { |
|
const ( |
|
categoryChain uint8 = iota |
|
categoryWallet |
|
numCategories |
|
) |
|
|
|
// Get a list of registered commands and categorize and filter them. |
|
cmdMethods := btcjson.RegisteredCmdMethods() |
|
categorized := make([][]string, numCategories) |
|
for _, method := range cmdMethods { |
|
flags, err := btcjson.MethodUsageFlags(method) |
|
if err != nil { |
|
// This should never happen since the method was just |
|
// returned from the package, but be safe. |
|
continue |
|
} |
|
|
|
// Skip the commands that aren't usable from this utility. |
|
if flags&unusableFlags != 0 { |
|
continue |
|
} |
|
|
|
usage, err := btcjson.MethodUsageText(method) |
|
if err != nil { |
|
// This should never happen since the method was just |
|
// returned from the package, but be safe. |
|
continue |
|
} |
|
|
|
// Categorize the command based on the usage flags. |
|
category := categoryChain |
|
if flags&btcjson.UFWalletOnly != 0 { |
|
category = categoryWallet |
|
} |
|
categorized[category] = append(categorized[category], usage) |
|
} |
|
|
|
// Display the command according to their categories. |
|
categoryTitles := make([]string, numCategories) |
|
categoryTitles[categoryChain] = "Chain Server Commands:" |
|
categoryTitles[categoryWallet] = "Wallet Server Commands (--wallet):" |
|
for category := uint8(0); category < numCategories; category++ { |
|
fmt.Println(categoryTitles[category]) |
|
for _, usage := range categorized[category] { |
|
fmt.Println(usage) |
|
} |
|
fmt.Println() |
|
} |
|
} |
|
|
|
// config defines the configuration options for btcctl. |
|
// |
|
// See loadConfig for details on the configuration load process. |
|
type config struct { |
|
ShowVersion bool `short:"V" long:"version" description:"Display version information and exit"` |
|
ListCommands bool `short:"l" long:"listcommands" description:"List all of the supported commands and exit"` |
|
ConfigFile string `short:"C" long:"configfile" description:"Path to configuration file"` |
|
RPCUser string `short:"u" long:"rpcuser" description:"RPC username"` |
|
RPCPassword string `short:"P" long:"rpcpass" default-mask:"-" description:"RPC password"` |
|
RPCServer string `short:"s" long:"rpcserver" description:"RPC server to connect to"` |
|
RPCCert string `short:"c" long:"rpccert" description:"RPC server certificate chain for validation"` |
|
NoTLS bool `long:"notls" description:"Disable TLS"` |
|
Proxy string `long:"proxy" description:"Connect via SOCKS5 proxy (eg. 127.0.0.1:9050)"` |
|
ProxyUser string `long:"proxyuser" description:"Username for proxy server"` |
|
ProxyPass string `long:"proxypass" default-mask:"-" description:"Password for proxy server"` |
|
TestNet3 bool `long:"testnet" description:"Connect to testnet"` |
|
SimNet bool `long:"simnet" description:"Connect to the simulation test network"` |
|
TLSSkipVerify bool `long:"skipverify" description:"Do not verify tls certificates (not recommended!)"` |
|
Wallet bool `long:"wallet" description:"Connect to wallet"` |
|
} |
|
|
|
// normalizeAddress returns addr with the passed default port appended if |
|
// there is not already a port specified. |
|
func normalizeAddress(addr string, useTestNet3, useSimNet, useWallet bool) string { |
|
_, _, err := net.SplitHostPort(addr) |
|
if err != nil { |
|
var defaultPort string |
|
switch { |
|
case useTestNet3: |
|
if useWallet { |
|
defaultPort = "18332" |
|
} else { |
|
defaultPort = "18334" |
|
} |
|
case useSimNet: |
|
if useWallet { |
|
defaultPort = "18554" |
|
} else { |
|
defaultPort = "18556" |
|
} |
|
default: |
|
if useWallet { |
|
defaultPort = "8332" |
|
} else { |
|
defaultPort = "8334" |
|
} |
|
} |
|
|
|
return net.JoinHostPort(addr, defaultPort) |
|
} |
|
return addr |
|
} |
|
|
|
// cleanAndExpandPath expands environement variables and leading ~ in the |
|
// passed path, cleans the result, and returns it. |
|
func cleanAndExpandPath(path string) string { |
|
// Expand initial ~ to OS specific home directory. |
|
if strings.HasPrefix(path, "~") { |
|
homeDir := filepath.Dir(btcctlHomeDir) |
|
path = strings.Replace(path, "~", homeDir, 1) |
|
} |
|
|
|
// NOTE: The os.ExpandEnv doesn't work with Windows-style %VARIABLE%, |
|
// but they variables can still be expanded via POSIX-style $VARIABLE. |
|
return filepath.Clean(os.ExpandEnv(path)) |
|
} |
|
|
|
// loadConfig initializes and parses the config using a config file and command |
|
// line options. |
|
// |
|
// The configuration proceeds as follows: |
|
// 1) Start with a default config with sane settings |
|
// 2) Pre-parse the command line to check for an alternative config file |
|
// 3) Load configuration file overwriting defaults with any specified options |
|
// 4) Parse CLI options and overwrite/add any specified options |
|
// |
|
// The above results in functioning properly without any config settings |
|
// while still allowing the user to override settings with config files and |
|
// command line options. Command line options always take precedence. |
|
func loadConfig() (*config, []string, error) { |
|
// Default config. |
|
cfg := config{ |
|
ConfigFile: defaultConfigFile, |
|
RPCServer: defaultRPCServer, |
|
RPCCert: defaultRPCCertFile, |
|
} |
|
|
|
// Create the home directory if it doesn't already exist. |
|
err := os.MkdirAll(btcdHomeDir, 0700) |
|
if err != nil { |
|
fmt.Fprintf(os.Stderr, "%v\n", err) |
|
os.Exit(-1) |
|
} |
|
|
|
// Pre-parse the command line options to see if an alternative config |
|
// file, the version flag, or the list commands flag was specified. Any |
|
// errors aside from the help message error can be ignored here since |
|
// they will be caught by the final parse below. |
|
preCfg := cfg |
|
preParser := flags.NewParser(&preCfg, flags.HelpFlag) |
|
_, err = preParser.Parse() |
|
if err != nil { |
|
if e, ok := err.(*flags.Error); ok && e.Type == flags.ErrHelp { |
|
fmt.Fprintln(os.Stderr, err) |
|
fmt.Fprintln(os.Stderr, "") |
|
fmt.Fprintln(os.Stderr, "The special parameter `-` "+ |
|
"indicates that a parameter should be read "+ |
|
"from the\nnext unread line from standard "+ |
|
"input.") |
|
return nil, nil, err |
|
} |
|
} |
|
|
|
// Show the version and exit if the version flag was specified. |
|
appName := filepath.Base(os.Args[0]) |
|
appName = strings.TrimSuffix(appName, filepath.Ext(appName)) |
|
usageMessage := fmt.Sprintf("Use %s -h to show options", appName) |
|
if preCfg.ShowVersion { |
|
fmt.Println(appName, "version", version()) |
|
os.Exit(0) |
|
} |
|
|
|
// Show the available commands and exit if the associated flag was |
|
// specified. |
|
if preCfg.ListCommands { |
|
listCommands() |
|
os.Exit(0) |
|
} |
|
|
|
// Load additional config from file. |
|
parser := flags.NewParser(&cfg, flags.Default) |
|
err = flags.NewIniParser(parser).ParseFile(preCfg.ConfigFile) |
|
if err != nil { |
|
if _, ok := err.(*os.PathError); !ok { |
|
fmt.Fprintf(os.Stderr, "Error parsing config file: %v\n", |
|
err) |
|
fmt.Fprintln(os.Stderr, usageMessage) |
|
return nil, nil, err |
|
} |
|
} |
|
|
|
// Parse command line options again to ensure they take precedence. |
|
remainingArgs, err := parser.Parse() |
|
if err != nil { |
|
if e, ok := err.(*flags.Error); !ok || e.Type != flags.ErrHelp { |
|
fmt.Fprintln(os.Stderr, usageMessage) |
|
} |
|
return nil, nil, err |
|
} |
|
|
|
// Multiple networks can't be selected simultaneously. |
|
numNets := 0 |
|
if cfg.TestNet3 { |
|
numNets++ |
|
} |
|
if cfg.SimNet { |
|
numNets++ |
|
} |
|
if numNets > 1 { |
|
str := "%s: The testnet and simnet params can't be used " + |
|
"together -- choose one of the two" |
|
err := fmt.Errorf(str, "loadConfig") |
|
fmt.Fprintln(os.Stderr, err) |
|
return nil, nil, err |
|
} |
|
|
|
// Override the RPC certificate if the --wallet flag was specified and |
|
// the user did not specify one. |
|
if cfg.Wallet && cfg.RPCCert == defaultRPCCertFile { |
|
cfg.RPCCert = defaultWalletCertFile |
|
} |
|
|
|
// Handle environment variable expansion in the RPC certificate path. |
|
cfg.RPCCert = cleanAndExpandPath(cfg.RPCCert) |
|
|
|
// Add default port to RPC server based on --testnet and --wallet flags |
|
// if needed. |
|
cfg.RPCServer = normalizeAddress(cfg.RPCServer, cfg.TestNet3, |
|
cfg.SimNet, cfg.Wallet) |
|
|
|
return &cfg, remainingArgs, nil |
|
}
|
|
|