2013-08-06 16:55:22 -05:00
// Copyright (c) 2013 Conformal Systems LLC.
// Use of this source code is governed by an ISC
// license that can be found in the LICENSE file.
package main
import (
"fmt"
2013-09-15 15:21:46 -05:00
"github.com/conformal/btcdb"
2013-09-13 14:52:34 -05:00
_ "github.com/conformal/btcdb/ldb"
_ "github.com/conformal/btcdb/sqlite3"
2013-08-06 16:55:22 -05:00
"github.com/conformal/btcwire"
2013-08-07 00:33:46 -05:00
"github.com/conformal/go-flags"
2013-08-06 16:55:22 -05:00
"net"
"os"
"path/filepath"
2013-09-17 17:40:27 -04:00
"strconv"
2013-08-07 18:53:01 -05:00
"strings"
2013-08-06 16:55:22 -05:00
"time"
)
const (
defaultConfigFilename = "btcd.conf"
defaultLogLevel = "info"
defaultBtcnet = btcwire . MainNet
2013-10-14 21:56:39 +01:00
defaultMaxPeers = 125
2013-08-06 16:55:22 -05:00
defaultBanDuration = time . Hour * 24
defaultVerifyEnabled = false
2013-10-08 15:00:34 -05:00
defaultDbType = "leveldb"
2013-08-06 16:55:22 -05:00
)
var (
defaultConfigFile = filepath . Join ( btcdHomeDir ( ) , defaultConfigFilename )
2013-09-15 12:08:42 -05:00
defaultDataDir = filepath . Join ( btcdHomeDir ( ) , "data" )
2013-09-15 15:21:46 -05:00
knownDbTypes = btcdb . SupportedDBs ( )
2013-08-06 16:55:22 -05:00
)
// config defines the configuration options for btcd.
//
// See loadConfig for details on the configuration load process.
type config struct {
2013-10-09 19:34:02 -05:00
ShowVersion bool ` short:"V" long:"version" description:"Display version information and exit" `
ConfigFile string ` short:"C" long:"configfile" description:"Path to configuration file" `
DataDir string ` short:"b" long:"datadir" description:"Directory to store data" `
AddPeers [ ] string ` short:"a" long:"addpeer" description:"Add a peer to connect with at startup" `
ConnectPeers [ ] string ` long:"connect" description:"Connect only to the specified peers at startup" `
DisableListen bool ` long:"nolisten" description:"Disable listening for incoming connections -- NOTE: Listening is automatically disabled if the --connect option is used or if the --proxy option is used without the --tor option" `
Port string ` short:"p" long:"port" description:"Listen for connections on this port (default: 8333, testnet: 18333)" `
MaxPeers int ` long:"maxpeers" description:"Max number of inbound and outbound peers" `
BanDuration time . Duration ` long:"banduration" description:"How long to ban misbehaving peers. Valid time units are { s, m, h}. Minimum 1 second" `
RPCUser string ` short:"u" long:"rpcuser" description:"Username for RPC connections" `
RPCPass string ` short:"P" long:"rpcpass" description:"Password for RPC connections" `
RPCPort string ` short:"r" long:"rpcport" description:"Listen for JSON/RPC messages on this port" `
DisableRPC bool ` long:"norpc" description:"Disable built-in RPC server -- NOTE: The RPC server is disabled by default if no rpcuser/rpcpass is specified" `
DisableDNSSeed bool ` long:"nodnsseed" description:"Disable DNS seeding for peers" `
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" description:"Password for proxy server" `
UseTor bool ` long:"tor" description:"Specifies the proxy server used is a Tor node" `
TestNet3 bool ` long:"testnet" description:"Use the test network" `
RegressionTest bool ` long:"regtest" description:"Use the regression test network" `
DisableCheckpoints bool ` long:"nocheckpoints" description:"Disable built-in checkpoints. Don't do this unless you know what you're doing." `
DbType string ` long:"dbtype" description:"Database backend to use for the Block Chain" `
Profile string ` long:"profile" description:"Enable HTTP profiling on given port -- NOTE port must be between 1024 and 65536" `
DebugLevel string ` short:"d" long:"debuglevel" description:"Logging level { trace, debug, info, warn, error, critical}" `
2013-08-06 16:55:22 -05:00
}
// btcdHomeDir returns an OS appropriate home directory for btcd.
func btcdHomeDir ( ) string {
// Search for Windows APPDATA first. This won't exist on POSIX OSes.
appData := os . Getenv ( "APPDATA" )
if appData != "" {
return filepath . Join ( appData , "btcd" )
}
// Fall back to standard HOME directory that works for most POSIX OSes.
home := os . Getenv ( "HOME" )
if home != "" {
return filepath . Join ( home , ".btcd" )
}
// In the worst case, use the current directory.
return "."
}
2013-09-18 00:13:36 -05:00
// 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 ( btcdHomeDir ( ) )
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 ) )
}
2013-08-06 16:55:22 -05:00
// validLogLevel returns whether or not logLevel is a valid debug log level.
func validLogLevel ( logLevel string ) bool {
switch logLevel {
case "trace" :
fallthrough
case "debug" :
fallthrough
case "info" :
fallthrough
case "warn" :
fallthrough
case "error" :
fallthrough
case "critical" :
return true
}
return false
}
2013-09-15 12:08:42 -05:00
// validDbType returns whether or not dbType is a supported database type.
func validDbType ( dbType string ) bool {
for _ , knownType := range knownDbTypes {
if dbType == knownType {
return true
}
}
return false
}
2013-08-06 16:55:22 -05:00
// normalizePeerAddress returns addr with the default peer port appended if
// there is not already a port specified.
func normalizePeerAddress ( addr string ) string {
_ , _ , err := net . SplitHostPort ( addr )
if err != nil {
return net . JoinHostPort ( addr , activeNetParams . peerPort )
}
return addr
}
// removeDuplicateAddresses returns a new slice with all duplicate entries in
// addrs removed.
func removeDuplicateAddresses ( addrs [ ] string ) [ ] string {
result := make ( [ ] string , 0 )
seen := map [ string ] bool { }
for _ , val := range addrs {
if _ , ok := seen [ val ] ; ! ok {
result = append ( result , val )
seen [ val ] = true
}
}
return result
}
2013-08-08 12:45:20 -05:00
// normalizeAndRemoveDuplicateAddresses return a new slice with all the passed
// addresses normalized and duplicates removed.
2013-08-06 16:55:22 -05:00
func normalizeAndRemoveDuplicateAddresses ( addrs [ ] string ) [ ] string {
for i , addr := range addrs {
addrs [ i ] = normalizePeerAddress ( addr )
}
addrs = removeDuplicateAddresses ( addrs )
return addrs
}
// updateConfigWithActiveParams update the passed config with parameters
// from the active net params if the relevant options in the passed config
// object are the default so options specified by the user on the command line
// are not overridden.
func updateConfigWithActiveParams ( cfg * config ) {
if cfg . Port == netParams ( defaultBtcnet ) . listenPort {
cfg . Port = activeNetParams . listenPort
}
2013-09-18 00:36:40 -05:00
if cfg . RPCPort == netParams ( defaultBtcnet ) . rpcPort {
cfg . RPCPort = activeNetParams . rpcPort
2013-08-06 16:55:22 -05:00
}
}
// filesExists reports whether the named file or directory exists.
func fileExists ( name string ) bool {
if _ , err := os . Stat ( name ) ; err != nil {
if os . IsNotExist ( err ) {
return false
}
}
return true
}
// 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 btcd 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 {
DebugLevel : defaultLogLevel ,
Port : netParams ( defaultBtcnet ) . listenPort ,
2013-09-18 00:36:40 -05:00
RPCPort : netParams ( defaultBtcnet ) . rpcPort ,
2013-08-06 16:55:22 -05:00
MaxPeers : defaultMaxPeers ,
BanDuration : defaultBanDuration ,
ConfigFile : defaultConfigFile ,
2013-09-13 18:02:10 -05:00
DataDir : defaultDataDir ,
2013-09-15 12:08:42 -05:00
DbType : defaultDbType ,
2013-08-06 16:55:22 -05:00
}
// Pre-parse the command line options to see if an alternative config
2013-08-07 18:53:01 -05:00
// file or the version flag was specified.
2013-08-06 16:55:22 -05:00
preCfg := cfg
preParser := flags . NewParser ( & preCfg , flags . Default )
_ , err := preParser . Parse ( )
if err != nil {
if e , ok := err . ( * flags . Error ) ; ! ok || e . Type != flags . ErrHelp {
preParser . WriteHelp ( os . Stderr )
}
return nil , nil , err
}
2013-08-07 18:53:01 -05:00
// Show the version and exit if the version flag was specified.
if preCfg . ShowVersion {
appName := filepath . Base ( os . Args [ 0 ] )
appName = strings . TrimSuffix ( appName , filepath . Ext ( appName ) )
fmt . Println ( appName , "version" , version ( ) )
os . Exit ( 0 )
}
2013-08-06 16:55:22 -05:00
// Load additional config from file.
parser := flags . NewParser ( & cfg , flags . Default )
err = parser . ParseIniFile ( preCfg . ConfigFile )
if err != nil {
if _ , ok := err . ( * os . PathError ) ; ! ok {
fmt . Fprintln ( os . Stderr , err )
parser . WriteHelp ( os . Stderr )
return nil , nil , err
}
log . Warnf ( "%v" , err )
}
2013-08-17 14:44:43 -05:00
// Don't add peers from the config file when in regression test mode.
if preCfg . RegressionTest && len ( cfg . AddPeers ) > 0 {
cfg . AddPeers = nil
}
2013-08-06 16:55:22 -05:00
// 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 {
parser . WriteHelp ( os . Stderr )
}
return nil , nil , err
}
// The two test networks can't be selected simultaneously.
if cfg . TestNet3 && cfg . RegressionTest {
str := "%s: The testnet and regtest params can't be used " +
"together -- choose one of the two"
2013-09-18 00:36:40 -05:00
err := fmt . Errorf ( str , "loadConfig" )
2013-08-06 16:55:22 -05:00
fmt . Fprintln ( os . Stderr , err )
parser . WriteHelp ( os . Stderr )
return nil , nil , err
}
// Choose the active network params based on the testnet and regression
// test net flags.
if cfg . TestNet3 {
activeNetParams = netParams ( btcwire . TestNet3 )
} else if cfg . RegressionTest {
activeNetParams = netParams ( btcwire . TestNet )
}
updateConfigWithActiveParams ( & cfg )
// Validate debug log level.
if ! validLogLevel ( cfg . DebugLevel ) {
2013-09-15 12:08:42 -05:00
str := "%s: The specified debug level [%v] is invalid"
2013-09-18 00:36:40 -05:00
err := fmt . Errorf ( str , "loadConfig" , cfg . DebugLevel )
2013-08-06 16:55:22 -05:00
fmt . Fprintln ( os . Stderr , err )
parser . WriteHelp ( os . Stderr )
return nil , nil , err
}
2013-09-15 12:08:42 -05:00
// Validate database type.
if ! validDbType ( cfg . DbType ) {
str := "%s: The specified database type [%v] is invalid -- " +
"supported types %v"
2013-09-18 00:36:40 -05:00
err := fmt . Errorf ( str , "loadConfig" , cfg . DbType , knownDbTypes )
2013-09-15 12:08:42 -05:00
fmt . Fprintln ( os . Stderr , err )
parser . WriteHelp ( os . Stderr )
return nil , nil , err
}
2013-09-17 17:40:27 -04:00
// Validate profile port number
2013-09-17 17:46:03 -04:00
if cfg . Profile != "" {
2013-09-17 17:04:40 -05:00
profilePort , err := strconv . Atoi ( cfg . Profile )
if err != nil || profilePort < 1024 || profilePort > 65535 {
str := "%s: The profile port must be between 1024 and 65535"
2013-09-18 00:36:40 -05:00
err := fmt . Errorf ( str , "loadConfig" )
2013-09-17 17:04:40 -05:00
fmt . Fprintln ( os . Stderr , err )
parser . WriteHelp ( os . Stderr )
return nil , nil , err
}
2013-09-17 17:40:27 -04:00
}
2013-09-15 12:08:42 -05:00
// Append the network type to the data directory so it is "namespaced"
// per network. In addition to the block database, there are other
// pieces of data that are saved to disk such as address manager state.
// All data is specific to a network, so namespacing the data directory
// means each individual piece of serialized data does not have to
// worry about changing names per network and such.
2013-09-18 00:13:36 -05:00
cfg . DataDir = cleanAndExpandPath ( cfg . DataDir )
2013-09-15 12:08:42 -05:00
cfg . DataDir = filepath . Join ( cfg . DataDir , activeNetParams . netName )
2013-08-06 16:55:22 -05:00
// Don't allow ban durations that are too short.
if cfg . BanDuration < time . Duration ( time . Second ) {
str := "%s: The banduration option may not be less than 1s -- parsed [%v]"
2013-09-18 00:36:40 -05:00
err := fmt . Errorf ( str , "loadConfig" , cfg . BanDuration )
2013-08-06 16:55:22 -05:00
fmt . Fprintln ( os . Stderr , err )
parser . WriteHelp ( os . Stderr )
return nil , nil , err
}
// --addPeer and --connect do not mix.
if len ( cfg . AddPeers ) > 0 && len ( cfg . ConnectPeers ) > 0 {
str := "%s: the --addpeer and --connect options can not be " +
"mixed"
2013-09-18 00:36:40 -05:00
err := fmt . Errorf ( str , "loadConfig" )
2013-08-06 16:55:22 -05:00
fmt . Fprintln ( os . Stderr , err )
parser . WriteHelp ( os . Stderr )
return nil , nil , err
}
2013-08-08 12:11:39 -04:00
// --tor requires --proxy to be set.
if cfg . UseTor && cfg . Proxy == "" {
str := "%s: the --tor option requires --proxy to be set"
2013-09-18 00:36:40 -05:00
err := fmt . Errorf ( str , "loadConfig" )
2013-08-08 12:11:39 -04:00
fmt . Fprintln ( os . Stderr , err )
parser . WriteHelp ( os . Stderr )
return nil , nil , err
}
// --proxy without --tor means no listening.
if cfg . Proxy != "" && ! cfg . UseTor {
cfg . DisableListen = true
}
2013-08-06 16:55:22 -05:00
// Connect means no seeding or listening.
if len ( cfg . ConnectPeers ) > 0 {
cfg . DisableDNSSeed = true
2013-08-07 12:35:01 -05:00
cfg . DisableListen = true
2013-08-06 16:55:22 -05:00
}
2013-08-07 11:02:31 -05:00
// The RPC server is disabled if no username or password is provided.
2013-09-18 00:36:40 -05:00
if cfg . RPCUser == "" || cfg . RPCPass == "" {
cfg . DisableRPC = true
2013-08-07 11:02:31 -05:00
}
2013-08-06 16:55:22 -05:00
// Add default port to all added peer addresses if needed and remove
// duplicate addresses.
cfg . AddPeers = normalizeAndRemoveDuplicateAddresses ( cfg . AddPeers )
2013-09-15 12:08:42 -05:00
cfg . ConnectPeers = normalizeAndRemoveDuplicateAddresses ( cfg . ConnectPeers )
2013-08-06 16:55:22 -05:00
return & cfg , remainingArgs , nil
}