mirror of
https://github.com/btcsuite/btcd.git
synced 2025-01-19 05:33:36 +01:00
ba5407615d
The doc formatting changes introduced in the recent go version is increasing the diff for all of the new commits. Formatting it all in this commit will help the readability of future PRs by reducing the diff.
374 lines
12 KiB
Go
374 lines
12 KiB
Go
// 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"
|
|
"io/ioutil"
|
|
"net"
|
|
"os"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"github.com/btcsuite/btcd/btcjson"
|
|
"github.com/btcsuite/btcd/btcutil"
|
|
"github.com/btcsuite/btcd/chaincfg"
|
|
flags "github.com/jessevdk/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 {
|
|
ConfigFile string `short:"C" long:"configfile" description:"Path to configuration file"`
|
|
ListCommands bool `short:"l" long:"listcommands" description:"List all of the supported commands and exit"`
|
|
NoTLS bool `long:"notls" description:"Disable TLS"`
|
|
Proxy string `long:"proxy" description:"Connect via SOCKS5 proxy (eg. 127.0.0.1:9050)"`
|
|
ProxyPass string `long:"proxypass" default-mask:"-" description:"Password for proxy server"`
|
|
ProxyUser string `long:"proxyuser" description:"Username for proxy server"`
|
|
RegressionTest bool `long:"regtest" description:"Connect to the regression test network"`
|
|
RPCCert string `short:"c" long:"rpccert" description:"RPC server certificate chain for validation"`
|
|
RPCPassword string `short:"P" long:"rpcpass" default-mask:"-" description:"RPC password"`
|
|
RPCServer string `short:"s" long:"rpcserver" description:"RPC server to connect to"`
|
|
RPCUser string `short:"u" long:"rpcuser" description:"RPC username"`
|
|
SimNet bool `long:"simnet" description:"Connect to the simulation test network"`
|
|
TLSSkipVerify bool `long:"skipverify" description:"Do not verify tls certificates (not recommended!)"`
|
|
TestNet3 bool `long:"testnet" description:"Connect to testnet"`
|
|
SigNet bool `long:"signet" description:"Connect to signet"`
|
|
ShowVersion bool `short:"V" long:"version" description:"Display version information and exit"`
|
|
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, chain *chaincfg.Params, useWallet bool) (string, error) {
|
|
_, _, err := net.SplitHostPort(addr)
|
|
if err != nil {
|
|
var defaultPort string
|
|
switch chain {
|
|
case &chaincfg.TestNet3Params:
|
|
if useWallet {
|
|
defaultPort = "18332"
|
|
} else {
|
|
defaultPort = "18334"
|
|
}
|
|
case &chaincfg.SimNetParams:
|
|
if useWallet {
|
|
defaultPort = "18554"
|
|
} else {
|
|
defaultPort = "18556"
|
|
}
|
|
case &chaincfg.RegressionNetParams:
|
|
if useWallet {
|
|
// TODO: add port once regtest is supported in btcwallet
|
|
paramErr := fmt.Errorf("cannot use -wallet with -regtest, btcwallet not yet compatible with regtest")
|
|
return "", paramErr
|
|
} else {
|
|
defaultPort = "18334"
|
|
}
|
|
case &chaincfg.SigNetParams:
|
|
if useWallet {
|
|
defaultPort = "38332"
|
|
} else {
|
|
defaultPort = "38332"
|
|
}
|
|
default:
|
|
if useWallet {
|
|
defaultPort = "8332"
|
|
} else {
|
|
defaultPort = "8334"
|
|
}
|
|
}
|
|
|
|
return net.JoinHostPort(addr, defaultPort), nil
|
|
}
|
|
return addr, nil
|
|
}
|
|
|
|
// 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,
|
|
}
|
|
|
|
// 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)
|
|
}
|
|
|
|
if _, err := os.Stat(preCfg.ConfigFile); os.IsNotExist(err) {
|
|
// Use config file for RPC server to create default btcctl config
|
|
var serverConfigPath string
|
|
if preCfg.Wallet {
|
|
serverConfigPath = filepath.Join(btcwalletHomeDir, "btcwallet.conf")
|
|
} else {
|
|
serverConfigPath = filepath.Join(btcdHomeDir, "btcd.conf")
|
|
}
|
|
|
|
err := createDefaultConfigFile(preCfg.ConfigFile, serverConfigPath)
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "Error creating a default config file: %v\n", err)
|
|
}
|
|
}
|
|
|
|
// 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
|
|
}
|
|
|
|
// default network is mainnet
|
|
network := &chaincfg.MainNetParams
|
|
|
|
// Multiple networks can't be selected simultaneously.
|
|
numNets := 0
|
|
if cfg.TestNet3 {
|
|
numNets++
|
|
network = &chaincfg.TestNet3Params
|
|
}
|
|
if cfg.SimNet {
|
|
numNets++
|
|
network = &chaincfg.SimNetParams
|
|
}
|
|
if cfg.RegressionTest {
|
|
numNets++
|
|
network = &chaincfg.RegressionNetParams
|
|
}
|
|
if cfg.SigNet {
|
|
numNets++
|
|
network = &chaincfg.SigNetParams
|
|
}
|
|
|
|
if numNets > 1 {
|
|
str := "%s: Multiple network params can't be used " +
|
|
"together -- choose one"
|
|
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, err = normalizeAddress(cfg.RPCServer, network, cfg.Wallet)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
return &cfg, remainingArgs, nil
|
|
}
|
|
|
|
// createDefaultConfig creates a basic config file at the given destination path.
|
|
// For this it tries to read the config file for the RPC server (either btcd or
|
|
// btcwallet), and extract the RPC user and password from it.
|
|
func createDefaultConfigFile(destinationPath, serverConfigPath string) error {
|
|
// Read the RPC server config
|
|
serverConfigFile, err := os.Open(serverConfigPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer serverConfigFile.Close()
|
|
content, err := ioutil.ReadAll(serverConfigFile)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Extract the rpcuser
|
|
rpcUserRegexp := regexp.MustCompile(`(?m)^\s*rpcuser=([^\s]+)`)
|
|
userSubmatches := rpcUserRegexp.FindSubmatch(content)
|
|
if userSubmatches == nil {
|
|
// No user found, nothing to do
|
|
return nil
|
|
}
|
|
|
|
// Extract the rpcpass
|
|
rpcPassRegexp := regexp.MustCompile(`(?m)^\s*rpcpass=([^\s]+)`)
|
|
passSubmatches := rpcPassRegexp.FindSubmatch(content)
|
|
if passSubmatches == nil {
|
|
// No password found, nothing to do
|
|
return nil
|
|
}
|
|
|
|
// Extract the notls
|
|
noTLSRegexp := regexp.MustCompile(`(?m)^\s*notls=(0|1)(?:\s|$)`)
|
|
noTLSSubmatches := noTLSRegexp.FindSubmatch(content)
|
|
|
|
// Create the destination directory if it does not exists
|
|
err = os.MkdirAll(filepath.Dir(destinationPath), 0700)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Create the destination file and write the rpcuser and rpcpass to it
|
|
dest, err := os.OpenFile(destinationPath,
|
|
os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0600)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer dest.Close()
|
|
|
|
destString := fmt.Sprintf("rpcuser=%s\nrpcpass=%s\n",
|
|
string(userSubmatches[1]), string(passSubmatches[1]))
|
|
if noTLSSubmatches != nil {
|
|
destString += fmt.Sprintf("notls=%s\n", noTLSSubmatches[1])
|
|
}
|
|
|
|
dest.WriteString(destString)
|
|
|
|
return nil
|
|
}
|