flags.go 68.9 KB
Newer Older
F
Felix Lange 已提交
1 2 3 4 5 6 7 8 9 10
// Copyright 2015 The go-ethereum Authors
// This file is part of go-ethereum.
//
// go-ethereum is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// go-ethereum is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
11
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
F
Felix Lange 已提交
12 13 14
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
15
// along with go-ethereum. If not, see <http://www.gnu.org/licenses/>.
F
Felix Lange 已提交
16

17
// Package utils contains internal helper functions for go-ethereum commands.
18 19 20 21
package utils

import (
	"crypto/ecdsa"
22
	"fmt"
23
	"io"
24
	"io/ioutil"
25
	"math/big"
26
	"os"
27
	"path/filepath"
28
	"strconv"
29
	"strings"
30 31
	"text/tabwriter"
	"text/template"
32
	"time"
33

F
Felix Lange 已提交
34
	"github.com/ethereum/go-ethereum/accounts"
35
	"github.com/ethereum/go-ethereum/accounts/keystore"
Z
zelig 已提交
36
	"github.com/ethereum/go-ethereum/common"
37
	"github.com/ethereum/go-ethereum/common/fdlimit"
38 39
	"github.com/ethereum/go-ethereum/consensus"
	"github.com/ethereum/go-ethereum/consensus/clique"
40
	"github.com/ethereum/go-ethereum/consensus/ethash"
41
	"github.com/ethereum/go-ethereum/core"
42
	"github.com/ethereum/go-ethereum/core/rawdb"
43
	"github.com/ethereum/go-ethereum/core/vm"
44 45
	"github.com/ethereum/go-ethereum/crypto"
	"github.com/ethereum/go-ethereum/eth"
46
	"github.com/ethereum/go-ethereum/eth/downloader"
47
	"github.com/ethereum/go-ethereum/eth/ethconfig"
48
	"github.com/ethereum/go-ethereum/eth/gasprice"
49
	"github.com/ethereum/go-ethereum/eth/tracers"
50
	"github.com/ethereum/go-ethereum/ethdb"
51
	"github.com/ethereum/go-ethereum/ethstats"
52
	"github.com/ethereum/go-ethereum/graphql"
R
rene 已提交
53
	"github.com/ethereum/go-ethereum/internal/ethapi"
54
	"github.com/ethereum/go-ethereum/internal/flags"
55
	"github.com/ethereum/go-ethereum/les"
56
	"github.com/ethereum/go-ethereum/log"
B
Bas van Kervel 已提交
57
	"github.com/ethereum/go-ethereum/metrics"
58
	"github.com/ethereum/go-ethereum/metrics/exp"
59
	"github.com/ethereum/go-ethereum/metrics/influxdb"
60
	"github.com/ethereum/go-ethereum/miner"
61
	"github.com/ethereum/go-ethereum/node"
62
	"github.com/ethereum/go-ethereum/p2p"
63
	"github.com/ethereum/go-ethereum/p2p/enode"
64
	"github.com/ethereum/go-ethereum/p2p/nat"
65
	"github.com/ethereum/go-ethereum/p2p/netutil"
66
	"github.com/ethereum/go-ethereum/params"
67
	pcsclite "github.com/gballet/go-libpcsclite"
68
	"gopkg.in/urfave/cli.v1"
69 70
)

71 72 73 74 75 76 77 78 79 80 81 82 83
func init() {
	cli.AppHelpTemplate = `{{.Name}} {{if .Flags}}[global options] {{end}}command{{if .Flags}} [command options]{{end}} [arguments...]

VERSION:
   {{.Version}}

COMMANDS:
   {{range .Commands}}{{.Name}}{{with .ShortName}}, {{.}}{{end}}{{ "\t" }}{{.Usage}}
   {{end}}{{if .Flags}}
GLOBAL OPTIONS:
   {{range .Flags}}{{.}}
   {{end}}{{end}}
`
84
	cli.CommandHelpTemplate = flags.CommandHelpTemplate
85
	cli.HelpPrinter = printHelp
86 87
}

88 89 90 91 92 93 94 95 96 97 98
func printHelp(out io.Writer, templ string, data interface{}) {
	funcMap := template.FuncMap{"join": strings.Join}
	t := template.Must(template.New("help").Funcs(funcMap).Parse(templ))
	w := tabwriter.NewWriter(out, 38, 8, 2, ' ', 0)
	err := t.Execute(w, data)
	if err != nil {
		panic(err)
	}
	w.Flush()
}

99 100 101 102 103 104 105 106 107
// These are all the command line flags we support.
// If you add to this list, please remember to include the
// flag in the appropriate command definition.
//
// The flags are defined here so their names and help texts
// are the same for all commands.

var (
	// General settings
108
	DataDirFlag = DirectoryFlag{
109
		Name:  "datadir",
110
		Usage: "Data directory for the databases and keystore",
111
		Value: DirectoryString(node.DefaultDataDir()),
112
	}
113 114 115 116
	AncientFlag = DirectoryFlag{
		Name:  "datadir.ancient",
		Usage: "Data directory for ancient chain segments (default = inside chaindata)",
	}
117 118 119 120
	MinFreeDiskSpaceFlag = DirectoryFlag{
		Name:  "datadir.minfreedisk",
		Usage: "Minimum free disk space in MB, once reached triggers auto shut down (default = --cache.gc converted to MB, 0 = disabled)",
	}
K
Kobi Gurkan 已提交
121 122 123 124
	KeyStoreDirFlag = DirectoryFlag{
		Name:  "keystore",
		Usage: "Directory for the keystore (default = inside the datadir)",
	}
125 126
	NoUSBFlag = cli.BoolFlag{
		Name:  "nousb",
127 128 129 130 131
		Usage: "Disables monitoring for and managing USB hardware wallets (deprecated)",
	}
	USBFlag = cli.BoolFlag{
		Name:  "usb",
		Usage: "Enable monitoring and management of USB hardware wallets",
132
	}
133 134 135 136 137
	SmartCardDaemonPathFlag = cli.StringFlag{
		Name:  "pcscdpath",
		Usage: "Path to the smartcard daemon (pcscd) socket file",
		Value: pcsclite.PCSCDSockName,
	}
138
	NetworkIdFlag = cli.Uint64Flag{
Z
zelig 已提交
139
		Name:  "networkid",
140
		Usage: "Explicitly set network id (integer)(For testnets: use --ropsten, --rinkeby, --goerli instead)",
141
		Value: ethconfig.Defaults.NetworkId,
Z
zelig 已提交
142
	}
143 144 145 146
	MainnetFlag = cli.BoolFlag{
		Name:  "mainnet",
		Usage: "Ethereum mainnet",
	}
147 148 149
	GoerliFlag = cli.BoolFlag{
		Name:  "goerli",
		Usage: "Görli network: pre-configured proof-of-authority test network",
150
	}
151 152 153
	YoloV3Flag = cli.BoolFlag{
		Name:  "yolov3",
		Usage: "YOLOv3 network: pre-configured proof-of-authority shortlived test network.",
154
	}
155 156 157 158
	RinkebyFlag = cli.BoolFlag{
		Name:  "rinkeby",
		Usage: "Rinkeby network: pre-configured proof-of-authority test network",
	}
159 160 161
	RopstenFlag = cli.BoolFlag{
		Name:  "ropsten",
		Usage: "Ropsten network: pre-configured proof-of-work test network",
162
	}
163
	DeveloperFlag = cli.BoolFlag{
164
		Name:  "dev",
165 166 167 168 169
		Usage: "Ephemeral proof-of-authority network with a pre-funded developer account, mining enabled",
	}
	DeveloperPeriodFlag = cli.IntFlag{
		Name:  "dev.period",
		Usage: "Block period to use in developer mode (0 = mine only if transaction pending)",
170
	}
171 172
	IdentityFlag = cli.StringFlag{
		Name:  "identity",
173
		Usage: "Custom node name",
174
	}
Z
zelig 已提交
175 176 177
	DocRootFlag = DirectoryFlag{
		Name:  "docroot",
		Usage: "Document Root for HTTPClient file scheme",
178
		Value: DirectoryString(HomeDir()),
Z
zelig 已提交
179
	}
180 181
	ExitWhenSyncedFlag = cli.BoolFlag{
		Name:  "exitwhensynced",
182
		Usage: "Exits after block synchronisation completes",
183
	}
184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199
	IterativeOutputFlag = cli.BoolFlag{
		Name:  "iterative",
		Usage: "Print streaming JSON iteratively, delimited by newlines",
	}
	ExcludeStorageFlag = cli.BoolFlag{
		Name:  "nostorage",
		Usage: "Exclude storage entries (save db lookups)",
	}
	IncludeIncompletesFlag = cli.BoolFlag{
		Name:  "incompletes",
		Usage: "Include accounts for which we don't have the address (missing preimage)",
	}
	ExcludeCodeFlag = cli.BoolFlag{
		Name:  "nocode",
		Usage: "Exclude contract code (save db lookups)",
	}
200
	defaultSyncMode = ethconfig.Defaults.SyncMode
201 202
	SyncModeFlag    = TextMarshalerFlag{
		Name:  "syncmode",
203
		Usage: `Blockchain sync mode ("fast", "full", "snap" or "light")`,
204 205
		Value: &defaultSyncMode,
	}
206 207 208 209 210
	GCModeFlag = cli.StringFlag{
		Name:  "gcmode",
		Usage: `Blockchain garbage collection mode ("full", "archive")`,
		Value: "full",
	}
211 212 213 214
	SnapshotFlag = cli.BoolFlag{
		Name:  "snapshot",
		Usage: `Enables snapshot-database mode -- experimental work in progress feature`,
	}
215 216 217 218 219
	TxLookupLimitFlag = cli.Int64Flag{
		Name:  "txlookuplimit",
		Usage: "Number of recent blocks to maintain transactions index by-hash for (default = index all blocks)",
		Value: 0,
	}
220 221 222 223 224 225 226 227
	LightKDFFlag = cli.BoolFlag{
		Name:  "lightkdf",
		Usage: "Reduce key-derivation RAM & CPU usage at some expense of KDF strength",
	}
	WhitelistFlag = cli.StringFlag{
		Name:  "whitelist",
		Usage: "Comma separated block number-to-hash mappings to enforce (<number>=<hash>)",
	}
228 229 230
	// Light server and client settings
	LightServeFlag = cli.IntFlag{
		Name:  "light.serve",
231
		Usage: "Maximum percentage of time allowed for serving LES requests (multi-threaded processing allows values over 100)",
232
		Value: ethconfig.Defaults.LightServ,
233
	}
234 235 236
	LightIngressFlag = cli.IntFlag{
		Name:  "light.ingress",
		Usage: "Incoming bandwidth limit for serving light clients (kilobytes/sec, 0 = unlimited)",
237
		Value: ethconfig.Defaults.LightIngress,
238
	}
239 240 241
	LightEgressFlag = cli.IntFlag{
		Name:  "light.egress",
		Usage: "Outgoing bandwidth limit for serving light clients (kilobytes/sec, 0 = unlimited)",
242
		Value: ethconfig.Defaults.LightEgress,
243
	}
244 245 246
	LightMaxPeersFlag = cli.IntFlag{
		Name:  "light.maxpeers",
		Usage: "Maximum number of light clients to serve, or light servers to attach to",
247
		Value: ethconfig.Defaults.LightPeers,
248
	}
249 250 251
	UltraLightServersFlag = cli.StringFlag{
		Name:  "ulc.servers",
		Usage: "List of trusted ultra-light servers",
252
		Value: strings.Join(ethconfig.Defaults.UltraLightServers, ","),
253
	}
254 255 256
	UltraLightFractionFlag = cli.IntFlag{
		Name:  "ulc.fraction",
		Usage: "Minimum % of trusted ultra-light servers required to announce a new head",
257
		Value: ethconfig.Defaults.UltraLightFraction,
258 259 260 261
	}
	UltraLightOnlyAnnounceFlag = cli.BoolFlag{
		Name:  "ulc.onlyannounce",
		Usage: "Ultra light server sends announcements only",
262
	}
263 264 265 266
	LightNoPruneFlag = cli.BoolFlag{
		Name:  "light.nopruning",
		Usage: "Disable ancient light chain data pruning",
	}
267 268 269 270 271 272 273 274
	// Ethash settings
	EthashCacheDirFlag = DirectoryFlag{
		Name:  "ethash.cachedir",
		Usage: "Directory to store the ethash verification caches (default = inside the datadir)",
	}
	EthashCachesInMemoryFlag = cli.IntFlag{
		Name:  "ethash.cachesinmem",
		Usage: "Number of recent ethash caches to keep in memory (16MB each)",
275
		Value: ethconfig.Defaults.Ethash.CachesInMem,
276 277 278 279
	}
	EthashCachesOnDiskFlag = cli.IntFlag{
		Name:  "ethash.cachesondisk",
		Usage: "Number of recent ethash caches to keep on disk (16MB each)",
280
		Value: ethconfig.Defaults.Ethash.CachesOnDisk,
281
	}
282 283 284 285
	EthashCachesLockMmapFlag = cli.BoolFlag{
		Name:  "ethash.cacheslockmmap",
		Usage: "Lock memory maps of recent ethash caches",
	}
286 287
	EthashDatasetDirFlag = DirectoryFlag{
		Name:  "ethash.dagdir",
288
		Usage: "Directory to store the ethash mining DAGs",
289
		Value: DirectoryString(ethconfig.Defaults.Ethash.DatasetDir),
290 291 292 293
	}
	EthashDatasetsInMemoryFlag = cli.IntFlag{
		Name:  "ethash.dagsinmem",
		Usage: "Number of recent ethash mining DAGs to keep in memory (1+GB each)",
294
		Value: ethconfig.Defaults.Ethash.DatasetsInMem,
295 296 297 298
	}
	EthashDatasetsOnDiskFlag = cli.IntFlag{
		Name:  "ethash.dagsondisk",
		Usage: "Number of recent ethash mining DAGs to keep on disk (1+GB each)",
299
		Value: ethconfig.Defaults.Ethash.DatasetsOnDisk,
300
	}
301 302 303 304
	EthashDatasetsLockMmapFlag = cli.BoolFlag{
		Name:  "ethash.dagslockmmap",
		Usage: "Lock memory maps for recent ethash mining DAGs",
	}
305
	// Transaction pool settings
306 307 308 309
	TxPoolLocalsFlag = cli.StringFlag{
		Name:  "txpool.locals",
		Usage: "Comma separated accounts to treat as locals (no flush, priority inclusion)",
	}
310 311 312 313
	TxPoolNoLocalsFlag = cli.BoolFlag{
		Name:  "txpool.nolocals",
		Usage: "Disables price exemptions for locally submitted transactions",
	}
314 315 316 317 318 319 320 321 322 323
	TxPoolJournalFlag = cli.StringFlag{
		Name:  "txpool.journal",
		Usage: "Disk journal for local transaction to survive node restarts",
		Value: core.DefaultTxPoolConfig.Journal,
	}
	TxPoolRejournalFlag = cli.DurationFlag{
		Name:  "txpool.rejournal",
		Usage: "Time interval to regenerate the local transaction journal",
		Value: core.DefaultTxPoolConfig.Rejournal,
	}
324 325 326
	TxPoolPriceLimitFlag = cli.Uint64Flag{
		Name:  "txpool.pricelimit",
		Usage: "Minimum gas price limit to enforce for acceptance into the pool",
327
		Value: ethconfig.Defaults.TxPool.PriceLimit,
328 329 330 331
	}
	TxPoolPriceBumpFlag = cli.Uint64Flag{
		Name:  "txpool.pricebump",
		Usage: "Price bump percentage to replace an already existing transaction",
332
		Value: ethconfig.Defaults.TxPool.PriceBump,
333 334 335 336
	}
	TxPoolAccountSlotsFlag = cli.Uint64Flag{
		Name:  "txpool.accountslots",
		Usage: "Minimum number of executable transaction slots guaranteed per account",
337
		Value: ethconfig.Defaults.TxPool.AccountSlots,
338 339 340 341
	}
	TxPoolGlobalSlotsFlag = cli.Uint64Flag{
		Name:  "txpool.globalslots",
		Usage: "Maximum number of executable transaction slots for all accounts",
342
		Value: ethconfig.Defaults.TxPool.GlobalSlots,
343 344 345 346
	}
	TxPoolAccountQueueFlag = cli.Uint64Flag{
		Name:  "txpool.accountqueue",
		Usage: "Maximum number of non-executable transaction slots permitted per account",
347
		Value: ethconfig.Defaults.TxPool.AccountQueue,
348 349 350 351
	}
	TxPoolGlobalQueueFlag = cli.Uint64Flag{
		Name:  "txpool.globalqueue",
		Usage: "Maximum number of non-executable transaction slots for all accounts",
352
		Value: ethconfig.Defaults.TxPool.GlobalQueue,
353 354 355 356
	}
	TxPoolLifetimeFlag = cli.DurationFlag{
		Name:  "txpool.lifetime",
		Usage: "Maximum amount of time non-executable transaction are queued",
357
		Value: ethconfig.Defaults.TxPool.Lifetime,
358
	}
359 360 361
	// Performance tuning settings
	CacheFlag = cli.IntFlag{
		Name:  "cache",
362
		Usage: "Megabytes of memory allocated to internal caching (default = 4096 mainnet full node, 128 light mode)",
363 364 365 366 367
		Value: 1024,
	}
	CacheDatabaseFlag = cli.IntFlag{
		Name:  "cache.database",
		Usage: "Percentage of cache memory allowance to use for database io",
368 369 370 371
		Value: 50,
	}
	CacheTrieFlag = cli.IntFlag{
		Name:  "cache.trie",
372 373
		Usage: "Percentage of cache memory allowance to use for trie caching (default = 15% full mode, 30% archive mode)",
		Value: 15,
374
	}
375 376 377
	CacheTrieJournalFlag = cli.StringFlag{
		Name:  "cache.trie.journal",
		Usage: "Disk journal directory for trie cache to survive node restarts",
378
		Value: ethconfig.Defaults.TrieCleanCacheJournal,
379 380 381 382
	}
	CacheTrieRejournalFlag = cli.DurationFlag{
		Name:  "cache.trie.rejournal",
		Usage: "Time interval to regenerate the trie cache journal",
383
		Value: ethconfig.Defaults.TrieCleanCacheRejournal,
384
	}
385 386
	CacheGCFlag = cli.IntFlag{
		Name:  "cache.gc",
387
		Usage: "Percentage of cache memory allowance to use for trie pruning (default = 25% full mode, 0% archive mode)",
388
		Value: 25,
389
	}
390 391 392 393 394
	CacheSnapshotFlag = cli.IntFlag{
		Name:  "cache.snapshot",
		Usage: "Percentage of cache memory allowance to use for snapshot caching (default = 10% full mode, 20% archive mode)",
		Value: 10,
	}
395 396 397 398
	CacheNoPrefetchFlag = cli.BoolFlag{
		Name:  "cache.noprefetch",
		Usage: "Disable heuristic state prefetch during block import (less CPU and disk IO, more time waiting for data)",
	}
399 400 401 402
	CachePreimagesFlag = cli.BoolTFlag{
		Name:  "cache.preimages",
		Usage: "Enable recording the SHA3/keccak preimages of trie keys (default: true)",
	}
403 404 405 406
	// Miner settings
	MiningEnabledFlag = cli.BoolFlag{
		Name:  "mine",
		Usage: "Enable mining",
407
	}
408
	MinerThreadsFlag = cli.IntFlag{
409
		Name:  "miner.threads",
410
		Usage: "Number of CPU threads to use for mining",
411 412 413 414 415
		Value: 0,
	}
	MinerNotifyFlag = cli.StringFlag{
		Name:  "miner.notify",
		Usage: "Comma separated HTTP URL list to notify of new work packages",
416
	}
417 418 419
	MinerGasTargetFlag = cli.Uint64Flag{
		Name:  "miner.gastarget",
		Usage: "Target gas floor for mined blocks",
420
		Value: ethconfig.Defaults.Miner.GasFloor,
421
	}
422 423 424
	MinerGasLimitFlag = cli.Uint64Flag{
		Name:  "miner.gaslimit",
		Usage: "Target gas ceiling for mined blocks",
425
		Value: ethconfig.Defaults.Miner.GasCeil,
426
	}
427 428
	MinerGasPriceFlag = BigFlag{
		Name:  "miner.gasprice",
429
		Usage: "Minimum gas price for mining a transaction",
430
		Value: ethconfig.Defaults.Miner.GasPrice,
Z
zelig 已提交
431
	}
432 433 434 435 436 437 438
	MinerEtherbaseFlag = cli.StringFlag{
		Name:  "miner.etherbase",
		Usage: "Public address for block mining rewards (default = first account)",
		Value: "0",
	}
	MinerExtraDataFlag = cli.StringFlag{
		Name:  "miner.extradata",
439
		Usage: "Block extra data set by the miner (default = client version)",
Z
zelig 已提交
440
	}
441 442
	MinerRecommitIntervalFlag = cli.DurationFlag{
		Name:  "miner.recommit",
443
		Usage: "Time interval to recreate the block being mined",
444
		Value: ethconfig.Defaults.Miner.Recommit,
445
	}
446 447 448 449
	MinerNoVerfiyFlag = cli.BoolFlag{
		Name:  "miner.noverify",
		Usage: "Disable remote sealing verification",
	}
450
	// Account settings
Z
zelig 已提交
451 452
	UnlockedAccountFlag = cli.StringFlag{
		Name:  "unlock",
453
		Usage: "Comma separated list of accounts to unlock",
Z
zelig 已提交
454 455 456 457
		Value: "",
	}
	PasswordFileFlag = cli.StringFlag{
		Name:  "password",
A
ayeowch 已提交
458
		Usage: "Password file to use for non-interactive password input",
Z
zelig 已提交
459
		Value: "",
Z
zelig 已提交
460
	}
461 462 463 464 465
	ExternalSignerFlag = cli.StringFlag{
		Name:  "signer",
		Usage: "External signer (url or path to ipc file)",
		Value: "",
	}
466 467 468 469
	VMEnableDebugFlag = cli.BoolFlag{
		Name:  "vmdebug",
		Usage: "Record information useful for VM and contract debugging",
	}
470 471 472 473
	InsecureUnlockAllowedFlag = cli.BoolFlag{
		Name:  "allow-insecure-unlock",
		Usage: "Allow insecure account unlocking when account-related RPCs are exposed by http",
	}
474
	RPCGlobalGasCapFlag = cli.Uint64Flag{
475
		Name:  "rpc.gascap",
476
		Usage: "Sets a cap on gas that can be used in eth_call/estimateGas (0=infinite)",
477
		Value: ethconfig.Defaults.RPCGasCap,
478
	}
479
	RPCGlobalTxFeeCapFlag = cli.Float64Flag{
480 481
		Name:  "rpc.txfeecap",
		Usage: "Sets a cap on transaction fee (in ether) that can be sent via the RPC APIs (0 = no cap)",
482
		Value: ethconfig.Defaults.RPCTxFeeCap,
483
	}
484 485 486 487 488
	// Logging and debug settings
	EthStatsURLFlag = cli.StringFlag{
		Name:  "ethstats",
		Usage: "Reporting URL of a ethstats service (nodename:secret@host:port)",
	}
489 490 491 492
	FakePoWFlag = cli.BoolFlag{
		Name:  "fakepow",
		Usage: "Disables proof-of-work verification",
	}
493 494 495 496
	NoCompactionFlag = cli.BoolFlag{
		Name:  "nocompaction",
		Usage: "Disables db compaction after import",
	}
497
	// RPC settings
498 499 500 501 502 503 504 505
	IPCDisabledFlag = cli.BoolFlag{
		Name:  "ipcdisable",
		Usage: "Disable the IPC-RPC server",
	}
	IPCPathFlag = DirectoryFlag{
		Name:  "ipcpath",
		Usage: "Filename for IPC socket/pipe within the datadir (explicit paths escape it)",
	}
506 507
	HTTPEnabledFlag = cli.BoolFlag{
		Name:  "http",
508
		Usage: "Enable the HTTP-RPC server",
509
	}
510 511
	HTTPListenAddrFlag = cli.StringFlag{
		Name:  "http.addr",
512
		Usage: "HTTP-RPC server listening interface",
513
		Value: node.DefaultHTTPHost,
514
	}
515 516
	HTTPPortFlag = cli.IntFlag{
		Name:  "http.port",
517
		Usage: "HTTP-RPC server listening port",
518
		Value: node.DefaultHTTPPort,
519
	}
520 521
	HTTPCORSDomainFlag = cli.StringFlag{
		Name:  "http.corsdomain",
522
		Usage: "Comma separated list of domains from which to accept cross origin requests (browser enforced)",
523 524
		Value: "",
	}
525 526
	HTTPVirtualHostsFlag = cli.StringFlag{
		Name:  "http.vhosts",
527
		Usage: "Comma separated list of virtual hostnames from which to accept requests (server enforced). Accepts '*' wildcard.",
528
		Value: strings.Join(node.DefaultConfig.HTTPVirtualHosts, ","),
529
	}
530 531
	HTTPApiFlag = cli.StringFlag{
		Name:  "http.api",
532
		Usage: "API's offered over the HTTP-RPC interface",
533
		Value: "",
534
	}
535 536 537 538 539
	HTTPPathPrefixFlag = cli.StringFlag{
		Name:  "http.rpcprefix",
		Usage: "HTTP path path prefix on which JSON-RPC is served. Use '/' to serve on all paths.",
		Value: "",
	}
R
rene 已提交
540 541 542 543 544 545 546 547 548 549 550 551 552 553
	GraphQLEnabledFlag = cli.BoolFlag{
		Name:  "graphql",
		Usage: "Enable GraphQL on the HTTP-RPC server. Note that GraphQL can only be started if an HTTP server is started as well.",
	}
	GraphQLCORSDomainFlag = cli.StringFlag{
		Name:  "graphql.corsdomain",
		Usage: "Comma separated list of domains from which to accept cross origin requests (browser enforced)",
		Value: "",
	}
	GraphQLVirtualHostsFlag = cli.StringFlag{
		Name:  "graphql.vhosts",
		Usage: "Comma separated list of virtual hostnames from which to accept requests (server enforced). Accepts '*' wildcard.",
		Value: strings.Join(node.DefaultConfig.GraphQLVirtualHosts, ","),
	}
554
	WSEnabledFlag = cli.BoolFlag{
555
		Name:  "ws",
556 557 558
		Usage: "Enable the WS-RPC server",
	}
	WSListenAddrFlag = cli.StringFlag{
559
		Name:  "ws.addr",
560
		Usage: "WS-RPC server listening interface",
561
		Value: node.DefaultWSHost,
562 563
	}
	WSPortFlag = cli.IntFlag{
564
		Name:  "ws.port",
565
		Usage: "WS-RPC server listening port",
566
		Value: node.DefaultWSPort,
567 568
	}
	WSApiFlag = cli.StringFlag{
569
		Name:  "ws.api",
570
		Usage: "API's offered over the WS-RPC interface",
571
		Value: "",
572
	}
B
Bas van Kervel 已提交
573
	WSAllowedOriginsFlag = cli.StringFlag{
574
		Name:  "ws.origins",
B
Bas van Kervel 已提交
575
		Usage: "Origins from which to accept websockets requests",
576
		Value: "",
577
	}
578 579 580 581 582
	WSPathPrefixFlag = cli.StringFlag{
		Name:  "ws.rpcprefix",
		Usage: "HTTP path prefix on which JSON-RPC is served. Use '/' to serve on all paths.",
		Value: "",
	}
583 584
	ExecFlag = cli.StringFlag{
		Name:  "exec",
585
		Usage: "Execute JavaScript statement",
586
	}
587
	PreloadJSFlag = cli.StringFlag{
588 589 590
		Name:  "preload",
		Usage: "Comma separated list of JavaScript files to preload into the console",
	}
591

592 593 594
	// Network Settings
	MaxPeersFlag = cli.IntFlag{
		Name:  "maxpeers",
595
		Usage: "Maximum number of network peers (network disabled if set to 0)",
596
		Value: node.DefaultConfig.P2P.MaxPeers,
597
	}
598 599 600
	MaxPendingPeersFlag = cli.IntFlag{
		Name:  "maxpendpeers",
		Usage: "Maximum number of pending connection attempts (defaults used if set to 0)",
601
		Value: node.DefaultConfig.P2P.MaxPendingPeers,
602
	}
603 604 605 606 607
	ListenPortFlag = cli.IntFlag{
		Name:  "port",
		Usage: "Network listening port",
		Value: 30303,
	}
608
	BootnodesFlag = cli.StringFlag{
609
		Name:  "bootnodes",
610
		Usage: "Comma separated enode URLs for P2P discovery bootstrap",
611
		Value: "",
612 613 614 615 616 617 618 619 620 621 622
	}
	NodeKeyFileFlag = cli.StringFlag{
		Name:  "nodekey",
		Usage: "P2P node key file",
	}
	NodeKeyHexFlag = cli.StringFlag{
		Name:  "nodekeyhex",
		Usage: "P2P node key as hex (for testing)",
	}
	NATFlag = cli.StringFlag{
		Name:  "nat",
623
		Usage: "NAT port mapping mechanism (any|none|upnp|pmp|extip:<IP>)",
624 625
		Value: "any",
	}
626 627 628 629
	NoDiscoverFlag = cli.BoolFlag{
		Name:  "nodiscover",
		Usage: "Disables the peer discovery mechanism (manual peer addition)",
	}
630 631 632 633
	DiscoveryV5Flag = cli.BoolFlag{
		Name:  "v5disc",
		Usage: "Enables the experimental RLPx V5 (Topic Discovery) mechanism",
	}
634 635 636 637
	NetrestrictFlag = cli.StringFlag{
		Name:  "netrestrict",
		Usage: "Restricts network communication to the given IP networks (CIDR masks)",
	}
638 639 640 641
	DNSDiscoveryFlag = cli.StringFlag{
		Name:  "discovery.dns",
		Usage: "Sets DNS discovery entry points (use \"\" to disable DNS)",
	}
642

643
	// ATM the url is left to the user and deployment to
Z
CLI:  
zelig 已提交
644 645
	JSpathFlag = cli.StringFlag{
		Name:  "jspath",
646
		Usage: "JavaScript root path for `loadScript`",
Z
CLI:  
zelig 已提交
647 648
		Value: ".",
	}
649 650

	// Gas price oracle settings
651
	GpoBlocksFlag = cli.IntFlag{
652
		Name:  "gpo.blocks",
653
		Usage: "Number of recent blocks to check for gas prices",
654
		Value: ethconfig.Defaults.GPO.Blocks,
Z
zsfelfoldi 已提交
655
	}
656
	GpoPercentileFlag = cli.IntFlag{
657
		Name:  "gpo.percentile",
658
		Usage: "Suggested gas price is the given percentile of a set of recent transaction gas prices",
659
		Value: ethconfig.Defaults.GPO.Percentile,
Z
zsfelfoldi 已提交
660
	}
661 662 663
	GpoMaxGasPriceFlag = cli.Int64Flag{
		Name:  "gpo.maxprice",
		Usage: "Maximum gas price will be recommended by gpo",
664
		Value: ethconfig.Defaults.GPO.MaxPrice.Int64(),
665
	}
666 667 668 669 670 671 672
	WhisperEnabledFlag = cli.BoolFlag{
		Name:  "shh",
		Usage: "Enable Whisper",
	}
	WhisperMaxMessageSizeFlag = cli.IntFlag{
		Name:  "shh.maxmessagesize",
		Usage: "Max message size accepted",
673
		Value: 1024 * 1024,
674 675 676 677
	}
	WhisperMinPOWFlag = cli.Float64Flag{
		Name:  "shh.pow",
		Usage: "Minimum POW accepted",
678
		Value: 0.2,
679
	}
680 681 682 683
	WhisperRestrictConnectionBetweenLightClientsFlag = cli.BoolFlag{
		Name:  "shh.restrict-light",
		Usage: "Restrict connection between two whisper light clients",
	}
684 685 686

	// Metrics flags
	MetricsEnabledFlag = cli.BoolFlag{
687
		Name:  "metrics",
688 689
		Usage: "Enable metrics collection and reporting",
	}
690 691 692 693
	MetricsEnabledExpensiveFlag = cli.BoolFlag{
		Name:  "metrics.expensive",
		Usage: "Enable expensive metrics collection and reporting",
	}
694 695 696 697 698 699 700 701

	// MetricsHTTPFlag defines the endpoint for a stand-alone metrics HTTP endpoint.
	// Since the pprof service enables sensitive/vulnerable behavior, this allows a user
	// to enable a public-OK metrics endpoint without having to worry about ALSO exposing
	// other profiling behavior or information.
	MetricsHTTPFlag = cli.StringFlag{
		Name:  "metrics.addr",
		Usage: "Enable stand-alone metrics HTTP server listening interface",
702
		Value: metrics.DefaultConfig.HTTP,
703 704 705 706
	}
	MetricsPortFlag = cli.IntFlag{
		Name:  "metrics.port",
		Usage: "Metrics HTTP server listening port",
707
		Value: metrics.DefaultConfig.Port,
708
	}
709 710 711 712 713 714 715
	MetricsEnableInfluxDBFlag = cli.BoolFlag{
		Name:  "metrics.influxdb",
		Usage: "Enable metrics export/push to an external InfluxDB database",
	}
	MetricsInfluxDBEndpointFlag = cli.StringFlag{
		Name:  "metrics.influxdb.endpoint",
		Usage: "InfluxDB API endpoint to report metrics to",
716
		Value: metrics.DefaultConfig.InfluxDBEndpoint,
717 718 719 720
	}
	MetricsInfluxDBDatabaseFlag = cli.StringFlag{
		Name:  "metrics.influxdb.database",
		Usage: "InfluxDB database name to push reported metrics to",
721
		Value: metrics.DefaultConfig.InfluxDBDatabase,
722 723 724 725
	}
	MetricsInfluxDBUsernameFlag = cli.StringFlag{
		Name:  "metrics.influxdb.username",
		Usage: "Username to authorize access to the database",
726
		Value: metrics.DefaultConfig.InfluxDBUsername,
727 728 729 730
	}
	MetricsInfluxDBPasswordFlag = cli.StringFlag{
		Name:  "metrics.influxdb.password",
		Usage: "Password to authorize access to the database",
731
		Value: metrics.DefaultConfig.InfluxDBPassword,
732
	}
733 734 735
	// Tags are part of every measurement sent to InfluxDB. Queries on tags are faster in InfluxDB.
	// For example `host` tag could be used so that we can group all nodes and average a measurement
	// across all of them, but also so that we can select a specific node and inspect its measurements.
736
	// https://docs.influxdata.com/influxdb/v1.4/concepts/key_concepts/#tag-key
737 738 739
	MetricsInfluxDBTagsFlag = cli.StringFlag{
		Name:  "metrics.influxdb.tags",
		Usage: "Comma-separated InfluxDB tags (key/values) attached to all measurements",
740
		Value: metrics.DefaultConfig.InfluxDBTags,
741
	}
742 743 744 745 746 747 748 749 750 751
	EWASMInterpreterFlag = cli.StringFlag{
		Name:  "vm.ewasm",
		Usage: "External ewasm configuration (default = built-in interpreter)",
		Value: "",
	}
	EVMInterpreterFlag = cli.StringFlag{
		Name:  "vm.evm",
		Usage: "External EVM configuration (default = built-in interpreter)",
		Value: "",
	}
752 753
)

754
// MakeDataDir retrieves the currently requested data directory, terminating
755
// if none (or the empty string) is specified. If the node is starting a testnet,
756
// then a subdirectory of the specified datadir will be used.
757
func MakeDataDir(ctx *cli.Context) string {
758
	if path := ctx.GlobalString(DataDirFlag.Name); path != "" {
759 760 761 762 763 764 765 766
		if ctx.GlobalBool(LegacyTestnetFlag.Name) || ctx.GlobalBool(RopstenFlag.Name) {
			// Maintain compatibility with older Geth configurations storing the
			// Ropsten database in `testnet` instead of `ropsten`.
			legacyPath := filepath.Join(path, "testnet")
			if _, err := os.Stat(legacyPath); !os.IsNotExist(err) {
				return legacyPath
			}
			return filepath.Join(path, "ropsten")
767
		}
768 769 770
		if ctx.GlobalBool(RinkebyFlag.Name) {
			return filepath.Join(path, "rinkeby")
		}
771 772 773
		if ctx.GlobalBool(GoerliFlag.Name) {
			return filepath.Join(path, "goerli")
		}
774 775
		if ctx.GlobalBool(YoloV3Flag.Name) {
			return filepath.Join(path, "yolo-v3")
776
		}
777
		return path
778
	}
779
	Fatalf("Cannot determine default data directory, please set manually (--datadir)")
780
	return ""
781 782
}

783
// setNodeKey creates a node key from set command line flags, either loading it
784 785
// from a file or as a specified hex value. If neither flags were provided, this
// method returns nil and an emphemeral key is to be generated.
786
func setNodeKey(ctx *cli.Context, cfg *p2p.Config) {
787 788 789
	var (
		hex  = ctx.GlobalString(NodeKeyHexFlag.Name)
		file = ctx.GlobalString(NodeKeyFileFlag.Name)
790 791
		key  *ecdsa.PrivateKey
		err  error
792
	)
793 794
	switch {
	case file != "" && hex != "":
795
		Fatalf("Options %q and %q are mutually exclusive", NodeKeyFileFlag.Name, NodeKeyHexFlag.Name)
796 797
	case file != "":
		if key, err = crypto.LoadECDSA(file); err != nil {
798
			Fatalf("Option %q: %v", NodeKeyFileFlag.Name, err)
799
		}
800
		cfg.PrivateKey = key
801 802
	case hex != "":
		if key, err = crypto.HexToECDSA(hex); err != nil {
803
			Fatalf("Option %q: %v", NodeKeyHexFlag.Name, err)
804
		}
805
		cfg.PrivateKey = key
806 807 808
	}
}

809 810
// setNodeUserIdent creates the user identifier from CLI flags.
func setNodeUserIdent(ctx *cli.Context, cfg *node.Config) {
811
	if identity := ctx.GlobalString(IdentityFlag.Name); len(identity) > 0 {
812
		cfg.UserIdent = identity
813 814 815
	}
}

816
// setBootstrapNodes creates a list of bootstrap nodes from the command line
817
// flags, reverting to pre-configured ones if none have been specified.
818
func setBootstrapNodes(ctx *cli.Context, cfg *p2p.Config) {
819
	urls := params.MainnetBootnodes
820
	switch {
821 822
	case ctx.GlobalIsSet(BootnodesFlag.Name) || ctx.GlobalIsSet(LegacyBootnodesV4Flag.Name):
		if ctx.GlobalIsSet(LegacyBootnodesV4Flag.Name) {
823
			urls = SplitAndTrim(ctx.GlobalString(LegacyBootnodesV4Flag.Name))
824
		} else {
825
			urls = SplitAndTrim(ctx.GlobalString(BootnodesFlag.Name))
826
		}
827 828
	case ctx.GlobalBool(LegacyTestnetFlag.Name) || ctx.GlobalBool(RopstenFlag.Name):
		urls = params.RopstenBootnodes
829 830
	case ctx.GlobalBool(RinkebyFlag.Name):
		urls = params.RinkebyBootnodes
831 832
	case ctx.GlobalBool(GoerliFlag.Name):
		urls = params.GoerliBootnodes
833 834
	case ctx.GlobalBool(YoloV3Flag.Name):
		urls = params.YoloV3Bootnodes
835 836
	case cfg.BootstrapNodes != nil:
		return // already set, don't apply defaults.
837 838
	}

839
	cfg.BootstrapNodes = make([]*enode.Node, 0, len(urls))
840
	for _, url := range urls {
841
		if url != "" {
842
			node, err := enode.Parse(enode.ValidSchemes, url)
843 844
			if err != nil {
				log.Crit("Bootstrap URL invalid", "enode", url, "err", err)
845
				continue
846 847
			}
			cfg.BootstrapNodes = append(cfg.BootstrapNodes, node)
848 849 850 851
		}
	}
}

852
// setBootstrapNodesV5 creates a list of bootstrap nodes from the command line
853
// flags, reverting to pre-configured ones if none have been specified.
854
func setBootstrapNodesV5(ctx *cli.Context, cfg *p2p.Config) {
855
	urls := params.V5Bootnodes
856
	switch {
857 858
	case ctx.GlobalIsSet(BootnodesFlag.Name) || ctx.GlobalIsSet(LegacyBootnodesV5Flag.Name):
		if ctx.GlobalIsSet(LegacyBootnodesV5Flag.Name) {
859
			urls = SplitAndTrim(ctx.GlobalString(LegacyBootnodesV5Flag.Name))
860
		} else {
861
			urls = SplitAndTrim(ctx.GlobalString(BootnodesFlag.Name))
862
		}
863
	case cfg.BootstrapNodesV5 != nil:
864
		return // already set, don't apply defaults.
865 866
	}

867
	cfg.BootstrapNodesV5 = make([]*enode.Node, 0, len(urls))
868
	for _, url := range urls {
869
		if url != "" {
870
			node, err := enode.Parse(enode.ValidSchemes, url)
871 872 873 874 875
			if err != nil {
				log.Error("Bootstrap URL invalid", "enode", url, "err", err)
				continue
			}
			cfg.BootstrapNodesV5 = append(cfg.BootstrapNodesV5, node)
876 877 878 879
		}
	}
}

880
// setListenAddress creates a TCP listening address string from set command
881
// line flags.
882 883 884 885
func setListenAddress(ctx *cli.Context, cfg *p2p.Config) {
	if ctx.GlobalIsSet(ListenPortFlag.Name) {
		cfg.ListenAddr = fmt.Sprintf(":%d", ctx.GlobalInt(ListenPortFlag.Name))
	}
886 887
}

888 889 890 891 892 893 894 895
// setNAT creates a port mapper from command line flags.
func setNAT(ctx *cli.Context, cfg *p2p.Config) {
	if ctx.GlobalIsSet(NATFlag.Name) {
		natif, err := nat.Parse(ctx.GlobalString(NATFlag.Name))
		if err != nil {
			Fatalf("Option %s: %v", NATFlag.Name, err)
		}
		cfg.NAT = natif
896 897 898
	}
}

899
// SplitAndTrim splits input separated by a comma
900
// and trims excessive white space from the substrings.
901
func SplitAndTrim(input string) (ret []string) {
902 903
	l := strings.Split(input, ",")
	for _, r := range l {
904
		if r = strings.TrimSpace(r); r != "" {
905 906
			ret = append(ret, r)
		}
907
	}
908
	return ret
909 910
}

911
// setHTTP creates the HTTP RPC listener interface string from the set
912
// command line flags, returning empty if the HTTP endpoint is disabled.
913
func setHTTP(ctx *cli.Context, cfg *node.Config) {
914 915
	if ctx.GlobalBool(LegacyRPCEnabledFlag.Name) && cfg.HTTPHost == "" {
		log.Warn("The flag --rpc is deprecated and will be removed in the future, please use --http")
916
		cfg.HTTPHost = "127.0.0.1"
917 918 919
		if ctx.GlobalIsSet(LegacyRPCListenAddrFlag.Name) {
			cfg.HTTPHost = ctx.GlobalString(LegacyRPCListenAddrFlag.Name)
			log.Warn("The flag --rpcaddr is deprecated and will be removed in the future, please use --http.addr")
920 921
		}
	}
922 923 924 925 926
	if ctx.GlobalBool(HTTPEnabledFlag.Name) && cfg.HTTPHost == "" {
		cfg.HTTPHost = "127.0.0.1"
		if ctx.GlobalIsSet(HTTPListenAddrFlag.Name) {
			cfg.HTTPHost = ctx.GlobalString(HTTPListenAddrFlag.Name)
		}
927
	}
928 929 930 931

	if ctx.GlobalIsSet(LegacyRPCPortFlag.Name) {
		cfg.HTTPPort = ctx.GlobalInt(LegacyRPCPortFlag.Name)
		log.Warn("The flag --rpcport is deprecated and will be removed in the future, please use --http.port")
932
	}
933 934
	if ctx.GlobalIsSet(HTTPPortFlag.Name) {
		cfg.HTTPPort = ctx.GlobalInt(HTTPPortFlag.Name)
935
	}
936 937

	if ctx.GlobalIsSet(LegacyRPCCORSDomainFlag.Name) {
938
		cfg.HTTPCors = SplitAndTrim(ctx.GlobalString(LegacyRPCCORSDomainFlag.Name))
939 940 941
		log.Warn("The flag --rpccorsdomain is deprecated and will be removed in the future, please use --http.corsdomain")
	}
	if ctx.GlobalIsSet(HTTPCORSDomainFlag.Name) {
942
		cfg.HTTPCors = SplitAndTrim(ctx.GlobalString(HTTPCORSDomainFlag.Name))
943 944 945
	}

	if ctx.GlobalIsSet(LegacyRPCApiFlag.Name) {
946
		cfg.HTTPModules = SplitAndTrim(ctx.GlobalString(LegacyRPCApiFlag.Name))
947 948 949
		log.Warn("The flag --rpcapi is deprecated and will be removed in the future, please use --http.api")
	}
	if ctx.GlobalIsSet(HTTPApiFlag.Name) {
950
		cfg.HTTPModules = SplitAndTrim(ctx.GlobalString(HTTPApiFlag.Name))
951 952 953
	}

	if ctx.GlobalIsSet(LegacyRPCVirtualHostsFlag.Name) {
954
		cfg.HTTPVirtualHosts = SplitAndTrim(ctx.GlobalString(LegacyRPCVirtualHostsFlag.Name))
955 956 957
		log.Warn("The flag --rpcvhosts is deprecated and will be removed in the future, please use --http.vhosts")
	}
	if ctx.GlobalIsSet(HTTPVirtualHostsFlag.Name) {
958
		cfg.HTTPVirtualHosts = SplitAndTrim(ctx.GlobalString(HTTPVirtualHostsFlag.Name))
959
	}
960 961 962 963

	if ctx.GlobalIsSet(HTTPPathPrefixFlag.Name) {
		cfg.HTTPPathPrefix = ctx.GlobalString(HTTPPathPrefixFlag.Name)
	}
964 965
}

966 967 968 969
// setGraphQL creates the GraphQL listener interface string from the set
// command line flags, returning empty if the GraphQL endpoint is disabled.
func setGraphQL(ctx *cli.Context, cfg *node.Config) {
	if ctx.GlobalIsSet(GraphQLCORSDomainFlag.Name) {
970
		cfg.GraphQLCors = SplitAndTrim(ctx.GlobalString(GraphQLCORSDomainFlag.Name))
971 972
	}
	if ctx.GlobalIsSet(GraphQLVirtualHostsFlag.Name) {
973
		cfg.GraphQLVirtualHosts = SplitAndTrim(ctx.GlobalString(GraphQLVirtualHostsFlag.Name))
974 975 976
	}
}

977
// setWS creates the WebSocket RPC listener interface string from the set
978
// command line flags, returning empty if the HTTP endpoint is disabled.
979 980 981
func setWS(ctx *cli.Context, cfg *node.Config) {
	if ctx.GlobalBool(WSEnabledFlag.Name) && cfg.WSHost == "" {
		cfg.WSHost = "127.0.0.1"
982 983 984 985
		if ctx.GlobalIsSet(LegacyWSListenAddrFlag.Name) {
			cfg.WSHost = ctx.GlobalString(LegacyWSListenAddrFlag.Name)
			log.Warn("The flag --wsaddr is deprecated and will be removed in the future, please use --ws.addr")
		}
986 987 988 989
		if ctx.GlobalIsSet(WSListenAddrFlag.Name) {
			cfg.WSHost = ctx.GlobalString(WSListenAddrFlag.Name)
		}
	}
990 991 992 993
	if ctx.GlobalIsSet(LegacyWSPortFlag.Name) {
		cfg.WSPort = ctx.GlobalInt(LegacyWSPortFlag.Name)
		log.Warn("The flag --wsport is deprecated and will be removed in the future, please use --ws.port")
	}
994 995 996
	if ctx.GlobalIsSet(WSPortFlag.Name) {
		cfg.WSPort = ctx.GlobalInt(WSPortFlag.Name)
	}
997 998

	if ctx.GlobalIsSet(LegacyWSAllowedOriginsFlag.Name) {
999
		cfg.WSOrigins = SplitAndTrim(ctx.GlobalString(LegacyWSAllowedOriginsFlag.Name))
1000 1001
		log.Warn("The flag --wsorigins is deprecated and will be removed in the future, please use --ws.origins")
	}
1002
	if ctx.GlobalIsSet(WSAllowedOriginsFlag.Name) {
1003
		cfg.WSOrigins = SplitAndTrim(ctx.GlobalString(WSAllowedOriginsFlag.Name))
1004
	}
1005 1006

	if ctx.GlobalIsSet(LegacyWSApiFlag.Name) {
1007
		cfg.WSModules = SplitAndTrim(ctx.GlobalString(LegacyWSApiFlag.Name))
1008 1009
		log.Warn("The flag --wsapi is deprecated and will be removed in the future, please use --ws.api")
	}
1010
	if ctx.GlobalIsSet(WSApiFlag.Name) {
1011
		cfg.WSModules = SplitAndTrim(ctx.GlobalString(WSApiFlag.Name))
1012
	}
1013 1014 1015 1016

	if ctx.GlobalIsSet(WSPathPrefixFlag.Name) {
		cfg.WSPathPrefix = ctx.GlobalString(WSPathPrefixFlag.Name)
	}
1017 1018 1019 1020 1021
}

// setIPC creates an IPC path configuration from the set command line flags,
// returning an empty string if IPC was explicitly disabled, or the set path.
func setIPC(ctx *cli.Context, cfg *node.Config) {
1022
	CheckExclusive(ctx, IPCDisabledFlag, IPCPathFlag)
1023 1024 1025 1026 1027
	switch {
	case ctx.GlobalBool(IPCDisabledFlag.Name):
		cfg.IPCPath = ""
	case ctx.GlobalIsSet(IPCPathFlag.Name):
		cfg.IPCPath = ctx.GlobalString(IPCPathFlag.Name)
1028 1029 1030
	}
}

1031
// setLes configures the les server and ultra light client settings from the command line flags.
1032
func setLes(ctx *cli.Context, cfg *ethconfig.Config) {
1033 1034 1035
	if ctx.GlobalIsSet(LegacyLightServFlag.Name) {
		cfg.LightServ = ctx.GlobalInt(LegacyLightServFlag.Name)
		log.Warn("The flag --lightserv is deprecated and will be removed in the future, please use --light.serve")
1036 1037 1038 1039 1040 1041
	}
	if ctx.GlobalIsSet(LightServeFlag.Name) {
		cfg.LightServ = ctx.GlobalInt(LightServeFlag.Name)
	}
	if ctx.GlobalIsSet(LightIngressFlag.Name) {
		cfg.LightIngress = ctx.GlobalInt(LightIngressFlag.Name)
1042
	}
1043 1044
	if ctx.GlobalIsSet(LightEgressFlag.Name) {
		cfg.LightEgress = ctx.GlobalInt(LightEgressFlag.Name)
1045
	}
1046 1047 1048
	if ctx.GlobalIsSet(LegacyLightPeersFlag.Name) {
		cfg.LightPeers = ctx.GlobalInt(LegacyLightPeersFlag.Name)
		log.Warn("The flag --lightpeers is deprecated and will be removed in the future, please use --light.maxpeers")
1049
	}
1050 1051
	if ctx.GlobalIsSet(LightMaxPeersFlag.Name) {
		cfg.LightPeers = ctx.GlobalInt(LightMaxPeersFlag.Name)
1052
	}
1053 1054
	if ctx.GlobalIsSet(UltraLightServersFlag.Name) {
		cfg.UltraLightServers = strings.Split(ctx.GlobalString(UltraLightServersFlag.Name), ",")
1055
	}
1056 1057
	if ctx.GlobalIsSet(UltraLightFractionFlag.Name) {
		cfg.UltraLightFraction = ctx.GlobalInt(UltraLightFractionFlag.Name)
1058
	}
1059
	if cfg.UltraLightFraction <= 0 && cfg.UltraLightFraction > 100 {
1060 1061
		log.Error("Ultra light fraction is invalid", "had", cfg.UltraLightFraction, "updated", ethconfig.Defaults.UltraLightFraction)
		cfg.UltraLightFraction = ethconfig.Defaults.UltraLightFraction
1062
	}
1063 1064
	if ctx.GlobalIsSet(UltraLightOnlyAnnounceFlag.Name) {
		cfg.UltraLightOnlyAnnounce = ctx.GlobalBool(UltraLightOnlyAnnounceFlag.Name)
1065
	}
1066 1067 1068
	if ctx.GlobalIsSet(LightNoPruneFlag.Name) {
		cfg.LightNoPrune = ctx.GlobalBool(LightNoPruneFlag.Name)
	}
1069 1070
}

1071
// makeDatabaseHandles raises out the number of allowed file handles per process
1072
// for Geth and returns half of the allowance to assign to the database.
1073
func makeDatabaseHandles() int {
1074
	limit, err := fdlimit.Maximum()
1075
	if err != nil {
1076
		Fatalf("Failed to retrieve file descriptor allowance: %v", err)
1077
	}
1078 1079
	raised, err := fdlimit.Raise(uint64(limit))
	if err != nil {
1080
		Fatalf("Failed to raise file descriptor allowance: %v", err)
1081
	}
1082
	return int(raised / 2) // Leave half for networking and other stuff
1083 1084
}

1085 1086
// MakeAddress converts an account specified directly as a hex encoded string or
// a key index in the key store to an internal account representation.
1087
func MakeAddress(ks *keystore.KeyStore, account string) (accounts.Account, error) {
1088 1089
	// If the specified account is a valid address, return it
	if common.IsHexAddress(account) {
F
Felix Lange 已提交
1090
		return accounts.Account{Address: common.HexToAddress(account)}, nil
1091 1092 1093
	}
	// Otherwise try to interpret the account as a keystore index
	index, err := strconv.Atoi(account)
1094
	if err != nil || index < 0 {
F
Felix Lange 已提交
1095
		return accounts.Account{}, fmt.Errorf("invalid account address or index %q", account)
1096
	}
1097 1098 1099 1100 1101 1102
	log.Warn("-------------------------------------------------------------------")
	log.Warn("Referring to accounts by order in the keystore folder is dangerous!")
	log.Warn("This functionality is deprecated and will be removed in the future!")
	log.Warn("Please use explicit addresses! (can search via `geth account list`)")
	log.Warn("-------------------------------------------------------------------")

1103 1104 1105 1106 1107
	accs := ks.Accounts()
	if len(accs) <= index {
		return accounts.Account{}, fmt.Errorf("index %d higher than number of accounts %d", index, len(accs))
	}
	return accs[index], nil
1108 1109
}

1110
// setEtherbase retrieves the etherbase either from the directly specified
1111
// command line flags or from the keystore if CLI indexed.
1112
func setEtherbase(ctx *cli.Context, ks *keystore.KeyStore, cfg *ethconfig.Config) {
1113 1114
	// Extract the current etherbase, new flag overriding legacy one
	var etherbase string
1115 1116 1117 1118
	if ctx.GlobalIsSet(LegacyMinerEtherbaseFlag.Name) {
		etherbase = ctx.GlobalString(LegacyMinerEtherbaseFlag.Name)
		log.Warn("The flag --etherbase is deprecated and will be removed in the future, please use --miner.etherbase")

1119 1120 1121 1122 1123 1124
	}
	if ctx.GlobalIsSet(MinerEtherbaseFlag.Name) {
		etherbase = ctx.GlobalString(MinerEtherbaseFlag.Name)
	}
	// Convert the etherbase into an address and configure it
	if etherbase != "" {
1125 1126 1127 1128 1129
		if ks != nil {
			account, err := MakeAddress(ks, etherbase)
			if err != nil {
				Fatalf("Invalid miner etherbase: %v", err)
			}
1130
			cfg.Miner.Etherbase = account.Address
1131 1132
		} else {
			Fatalf("No etherbase configured")
1133
		}
1134
	}
1135 1136
}

1137
// MakePasswordList reads password lines from the file specified by the global --password flag.
1138
func MakePasswordList(ctx *cli.Context) []string {
1139 1140 1141 1142 1143 1144
	path := ctx.GlobalString(PasswordFileFlag.Name)
	if path == "" {
		return nil
	}
	text, err := ioutil.ReadFile(path)
	if err != nil {
1145
		Fatalf("Failed to read password file: %v", err)
1146 1147 1148 1149 1150
	}
	lines := strings.Split(string(text), "\n")
	// Sanitise DOS line endings.
	for i := range lines {
		lines[i] = strings.TrimRight(lines[i], "\r")
1151
	}
1152
	return lines
1153 1154
}

1155 1156 1157 1158 1159 1160 1161
func SetP2PConfig(ctx *cli.Context, cfg *p2p.Config) {
	setNodeKey(ctx, cfg)
	setNAT(ctx, cfg)
	setListenAddress(ctx, cfg)
	setBootstrapNodes(ctx, cfg)
	setBootstrapNodesV5(ctx, cfg)

1162
	lightClient := ctx.GlobalString(SyncModeFlag.Name) == "light"
1163
	lightServer := (ctx.GlobalInt(LegacyLightServFlag.Name) != 0 || ctx.GlobalInt(LightServeFlag.Name) != 0)
1164

1165
	lightPeers := ctx.GlobalInt(LegacyLightPeersFlag.Name)
1166 1167 1168
	if ctx.GlobalIsSet(LightMaxPeersFlag.Name) {
		lightPeers = ctx.GlobalInt(LightMaxPeersFlag.Name)
	}
1169
	if lightClient && !ctx.GlobalIsSet(LegacyLightPeersFlag.Name) && !ctx.GlobalIsSet(LightMaxPeersFlag.Name) {
1170 1171 1172 1173
		// dynamic default - for clients we use 1/10th of the default for servers
		lightPeers /= 10
	}

1174 1175
	if ctx.GlobalIsSet(MaxPeersFlag.Name) {
		cfg.MaxPeers = ctx.GlobalInt(MaxPeersFlag.Name)
1176
		if lightServer && !ctx.GlobalIsSet(LegacyLightPeersFlag.Name) && !ctx.GlobalIsSet(LightMaxPeersFlag.Name) {
1177 1178
			cfg.MaxPeers += lightPeers
		}
1179 1180 1181 1182
	} else {
		if lightServer {
			cfg.MaxPeers += lightPeers
		}
1183
		if lightClient && (ctx.GlobalIsSet(LegacyLightPeersFlag.Name) || ctx.GlobalIsSet(LightMaxPeersFlag.Name)) && cfg.MaxPeers < lightPeers {
1184 1185
			cfg.MaxPeers = lightPeers
		}
1186
	}
1187 1188 1189 1190 1191 1192 1193 1194 1195
	if !(lightClient || lightServer) {
		lightPeers = 0
	}
	ethPeers := cfg.MaxPeers - lightPeers
	if lightClient {
		ethPeers = 0
	}
	log.Info("Maximum peer count", "ETH", ethPeers, "LES", lightPeers, "total", cfg.MaxPeers)

1196 1197 1198
	if ctx.GlobalIsSet(MaxPendingPeersFlag.Name) {
		cfg.MaxPendingPeers = ctx.GlobalInt(MaxPendingPeersFlag.Name)
	}
1199
	if ctx.GlobalIsSet(NoDiscoverFlag.Name) || lightClient {
1200
		cfg.NoDiscovery = true
1201 1202
	}

1203 1204 1205
	// if we're running a light client or server, force enable the v5 peer discovery
	// unless it is explicitly disabled with --nodiscover note that explicitly specifying
	// --v5disc overrides --nodiscover, in which case the later only disables v4 discovery
1206
	forceV5Discovery := (lightClient || lightServer) && !ctx.GlobalBool(NoDiscoverFlag.Name)
1207 1208 1209 1210
	if ctx.GlobalIsSet(DiscoveryV5Flag.Name) {
		cfg.DiscoveryV5 = ctx.GlobalBool(DiscoveryV5Flag.Name)
	} else if forceV5Discovery {
		cfg.DiscoveryV5 = true
1211
	}
1212

1213 1214 1215
	if netrestrict := ctx.GlobalString(NetrestrictFlag.Name); netrestrict != "" {
		list, err := netutil.ParseNetlist(netrestrict)
		if err != nil {
1216
			Fatalf("Option %q: %v", NetrestrictFlag.Name, err)
1217
		}
1218
		cfg.NetRestrict = list
1219 1220
	}

1221
	if ctx.GlobalBool(DeveloperFlag.Name) {
1222 1223 1224 1225 1226
		// --dev mode can't use p2p networking.
		cfg.MaxPeers = 0
		cfg.ListenAddr = ":0"
		cfg.NoDiscovery = true
		cfg.DiscoveryV5 = false
1227 1228 1229
	}
}

1230 1231 1232 1233 1234
// SetNodeConfig applies node-related command line flags to the config.
func SetNodeConfig(ctx *cli.Context, cfg *node.Config) {
	SetP2PConfig(ctx, &cfg.P2P)
	setIPC(ctx, cfg)
	setHTTP(ctx, cfg)
1235
	setGraphQL(ctx, cfg)
1236 1237
	setWS(ctx, cfg)
	setNodeUserIdent(ctx, cfg)
1238
	setDataDir(ctx, cfg)
1239
	setSmartCard(ctx, cfg)
1240

1241 1242 1243 1244
	if ctx.GlobalIsSet(ExternalSignerFlag.Name) {
		cfg.ExternalSigner = ctx.GlobalString(ExternalSignerFlag.Name)
	}

1245 1246 1247 1248 1249 1250
	if ctx.GlobalIsSet(KeyStoreDirFlag.Name) {
		cfg.KeyStoreDir = ctx.GlobalString(KeyStoreDirFlag.Name)
	}
	if ctx.GlobalIsSet(LightKDFFlag.Name) {
		cfg.UseLightweightKDF = ctx.GlobalBool(LightKDFFlag.Name)
	}
1251
	if ctx.GlobalIsSet(NoUSBFlag.Name) || cfg.NoUSB {
1252
		log.Warn("Option nousb is deprecated and USB is deactivated by default. Use --usb to enable")
1253
	}
1254 1255 1256
	if ctx.GlobalIsSet(USBFlag.Name) {
		cfg.USB = ctx.GlobalBool(USBFlag.Name)
	}
1257 1258 1259
	if ctx.GlobalIsSet(InsecureUnlockAllowedFlag.Name) {
		cfg.InsecureUnlockAllowed = ctx.GlobalBool(InsecureUnlockAllowedFlag.Name)
	}
1260 1261
}

1262 1263 1264 1265 1266 1267 1268 1269 1270
func setSmartCard(ctx *cli.Context, cfg *node.Config) {
	// Skip enabling smartcards if no path is set
	path := ctx.GlobalString(SmartCardDaemonPathFlag.Name)
	if path == "" {
		return
	}
	// Sanity check that the smartcard path is valid
	fi, err := os.Stat(path)
	if err != nil {
1271
		log.Info("Smartcard socket not found, disabling", "err", err)
1272 1273 1274 1275 1276 1277 1278 1279 1280 1281
		return
	}
	if fi.Mode()&os.ModeType != os.ModeSocket {
		log.Error("Invalid smartcard daemon path", "path", path, "type", fi.Mode().String())
		return
	}
	// Smartcard daemon path exists and is a socket, enable it
	cfg.SmartCardDaemonPath = path
}

1282 1283 1284 1285 1286 1287
func setDataDir(ctx *cli.Context, cfg *node.Config) {
	switch {
	case ctx.GlobalIsSet(DataDirFlag.Name):
		cfg.DataDir = ctx.GlobalString(DataDirFlag.Name)
	case ctx.GlobalBool(DeveloperFlag.Name):
		cfg.DataDir = "" // unless explicitly requested, use memory databases
1288 1289 1290 1291 1292 1293 1294 1295 1296 1297
	case (ctx.GlobalBool(LegacyTestnetFlag.Name) || ctx.GlobalBool(RopstenFlag.Name)) && cfg.DataDir == node.DefaultDataDir():
		// Maintain compatibility with older Geth configurations storing the
		// Ropsten database in `testnet` instead of `ropsten`.
		legacyPath := filepath.Join(node.DefaultDataDir(), "testnet")
		if _, err := os.Stat(legacyPath); !os.IsNotExist(err) {
			log.Warn("Using the deprecated `testnet` datadir. Future versions will store the Ropsten chain in `ropsten`.")
			cfg.DataDir = legacyPath
		} else {
			cfg.DataDir = filepath.Join(node.DefaultDataDir(), "ropsten")
		}
1298
	case ctx.GlobalBool(RinkebyFlag.Name) && cfg.DataDir == node.DefaultDataDir():
1299
		cfg.DataDir = filepath.Join(node.DefaultDataDir(), "rinkeby")
1300
	case ctx.GlobalBool(GoerliFlag.Name) && cfg.DataDir == node.DefaultDataDir():
1301
		cfg.DataDir = filepath.Join(node.DefaultDataDir(), "goerli")
1302
	case ctx.GlobalBool(YoloV3Flag.Name) && cfg.DataDir == node.DefaultDataDir():
1303
		cfg.DataDir = filepath.Join(node.DefaultDataDir(), "yolo-v2")
1304 1305 1306
	}
}

1307 1308 1309 1310
func setGPO(ctx *cli.Context, cfg *gasprice.Config, light bool) {
	// If we are running the light client, apply another group
	// settings for gas oracle.
	if light {
1311 1312
		cfg.Blocks = ethconfig.LightClientGPO.Blocks
		cfg.Percentile = ethconfig.LightClientGPO.Percentile
1313
	}
1314 1315 1316 1317
	if ctx.GlobalIsSet(LegacyGpoBlocksFlag.Name) {
		cfg.Blocks = ctx.GlobalInt(LegacyGpoBlocksFlag.Name)
		log.Warn("The flag --gpoblocks is deprecated and will be removed in the future, please use --gpo.blocks")
	}
1318 1319 1320
	if ctx.GlobalIsSet(GpoBlocksFlag.Name) {
		cfg.Blocks = ctx.GlobalInt(GpoBlocksFlag.Name)
	}
1321 1322 1323 1324
	if ctx.GlobalIsSet(LegacyGpoPercentileFlag.Name) {
		cfg.Percentile = ctx.GlobalInt(LegacyGpoPercentileFlag.Name)
		log.Warn("The flag --gpopercentile is deprecated and will be removed in the future, please use --gpo.percentile")
	}
1325 1326 1327
	if ctx.GlobalIsSet(GpoPercentileFlag.Name) {
		cfg.Percentile = ctx.GlobalInt(GpoPercentileFlag.Name)
	}
1328 1329 1330
	if ctx.GlobalIsSet(GpoMaxGasPriceFlag.Name) {
		cfg.MaxPrice = big.NewInt(ctx.GlobalInt64(GpoMaxGasPriceFlag.Name))
	}
1331 1332
}

1333
func setTxPool(ctx *cli.Context, cfg *core.TxPoolConfig) {
1334 1335 1336 1337 1338 1339 1340 1341 1342 1343
	if ctx.GlobalIsSet(TxPoolLocalsFlag.Name) {
		locals := strings.Split(ctx.GlobalString(TxPoolLocalsFlag.Name), ",")
		for _, account := range locals {
			if trimmed := strings.TrimSpace(account); !common.IsHexAddress(trimmed) {
				Fatalf("Invalid account in --txpool.locals: %s", trimmed)
			} else {
				cfg.Locals = append(cfg.Locals, common.HexToAddress(account))
			}
		}
	}
1344 1345 1346
	if ctx.GlobalIsSet(TxPoolNoLocalsFlag.Name) {
		cfg.NoLocals = ctx.GlobalBool(TxPoolNoLocalsFlag.Name)
	}
1347 1348 1349 1350 1351 1352
	if ctx.GlobalIsSet(TxPoolJournalFlag.Name) {
		cfg.Journal = ctx.GlobalString(TxPoolJournalFlag.Name)
	}
	if ctx.GlobalIsSet(TxPoolRejournalFlag.Name) {
		cfg.Rejournal = ctx.GlobalDuration(TxPoolRejournalFlag.Name)
	}
1353 1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368 1369 1370 1371 1372 1373 1374 1375
	if ctx.GlobalIsSet(TxPoolPriceLimitFlag.Name) {
		cfg.PriceLimit = ctx.GlobalUint64(TxPoolPriceLimitFlag.Name)
	}
	if ctx.GlobalIsSet(TxPoolPriceBumpFlag.Name) {
		cfg.PriceBump = ctx.GlobalUint64(TxPoolPriceBumpFlag.Name)
	}
	if ctx.GlobalIsSet(TxPoolAccountSlotsFlag.Name) {
		cfg.AccountSlots = ctx.GlobalUint64(TxPoolAccountSlotsFlag.Name)
	}
	if ctx.GlobalIsSet(TxPoolGlobalSlotsFlag.Name) {
		cfg.GlobalSlots = ctx.GlobalUint64(TxPoolGlobalSlotsFlag.Name)
	}
	if ctx.GlobalIsSet(TxPoolAccountQueueFlag.Name) {
		cfg.AccountQueue = ctx.GlobalUint64(TxPoolAccountQueueFlag.Name)
	}
	if ctx.GlobalIsSet(TxPoolGlobalQueueFlag.Name) {
		cfg.GlobalQueue = ctx.GlobalUint64(TxPoolGlobalQueueFlag.Name)
	}
	if ctx.GlobalIsSet(TxPoolLifetimeFlag.Name) {
		cfg.Lifetime = ctx.GlobalDuration(TxPoolLifetimeFlag.Name)
	}
}

1376
func setEthash(ctx *cli.Context, cfg *ethconfig.Config) {
1377
	if ctx.GlobalIsSet(EthashCacheDirFlag.Name) {
1378
		cfg.Ethash.CacheDir = ctx.GlobalString(EthashCacheDirFlag.Name)
1379 1380
	}
	if ctx.GlobalIsSet(EthashDatasetDirFlag.Name) {
1381
		cfg.Ethash.DatasetDir = ctx.GlobalString(EthashDatasetDirFlag.Name)
1382 1383
	}
	if ctx.GlobalIsSet(EthashCachesInMemoryFlag.Name) {
1384
		cfg.Ethash.CachesInMem = ctx.GlobalInt(EthashCachesInMemoryFlag.Name)
1385 1386
	}
	if ctx.GlobalIsSet(EthashCachesOnDiskFlag.Name) {
1387
		cfg.Ethash.CachesOnDisk = ctx.GlobalInt(EthashCachesOnDiskFlag.Name)
1388
	}
1389 1390 1391
	if ctx.GlobalIsSet(EthashCachesLockMmapFlag.Name) {
		cfg.Ethash.CachesLockMmap = ctx.GlobalBool(EthashCachesLockMmapFlag.Name)
	}
1392
	if ctx.GlobalIsSet(EthashDatasetsInMemoryFlag.Name) {
1393
		cfg.Ethash.DatasetsInMem = ctx.GlobalInt(EthashDatasetsInMemoryFlag.Name)
1394 1395
	}
	if ctx.GlobalIsSet(EthashDatasetsOnDiskFlag.Name) {
1396
		cfg.Ethash.DatasetsOnDisk = ctx.GlobalInt(EthashDatasetsOnDiskFlag.Name)
1397
	}
1398 1399 1400
	if ctx.GlobalIsSet(EthashDatasetsLockMmapFlag.Name) {
		cfg.Ethash.DatasetsLockMmap = ctx.GlobalBool(EthashDatasetsLockMmapFlag.Name)
	}
1401 1402
}

1403 1404 1405 1406
func setMiner(ctx *cli.Context, cfg *miner.Config) {
	if ctx.GlobalIsSet(MinerNotifyFlag.Name) {
		cfg.Notify = strings.Split(ctx.GlobalString(MinerNotifyFlag.Name), ",")
	}
1407 1408 1409
	if ctx.GlobalIsSet(LegacyMinerExtraDataFlag.Name) {
		cfg.ExtraData = []byte(ctx.GlobalString(LegacyMinerExtraDataFlag.Name))
		log.Warn("The flag --extradata is deprecated and will be removed in the future, please use --miner.extradata")
1410 1411 1412 1413
	}
	if ctx.GlobalIsSet(MinerExtraDataFlag.Name) {
		cfg.ExtraData = []byte(ctx.GlobalString(MinerExtraDataFlag.Name))
	}
1414 1415 1416
	if ctx.GlobalIsSet(LegacyMinerGasTargetFlag.Name) {
		cfg.GasFloor = ctx.GlobalUint64(LegacyMinerGasTargetFlag.Name)
		log.Warn("The flag --targetgaslimit is deprecated and will be removed in the future, please use --miner.gastarget")
1417 1418 1419 1420 1421 1422 1423
	}
	if ctx.GlobalIsSet(MinerGasTargetFlag.Name) {
		cfg.GasFloor = ctx.GlobalUint64(MinerGasTargetFlag.Name)
	}
	if ctx.GlobalIsSet(MinerGasLimitFlag.Name) {
		cfg.GasCeil = ctx.GlobalUint64(MinerGasLimitFlag.Name)
	}
1424 1425 1426
	if ctx.GlobalIsSet(LegacyMinerGasPriceFlag.Name) {
		cfg.GasPrice = GlobalBig(ctx, LegacyMinerGasPriceFlag.Name)
		log.Warn("The flag --gasprice is deprecated and will be removed in the future, please use --miner.gasprice")
1427 1428 1429 1430 1431
	}
	if ctx.GlobalIsSet(MinerGasPriceFlag.Name) {
		cfg.GasPrice = GlobalBig(ctx, MinerGasPriceFlag.Name)
	}
	if ctx.GlobalIsSet(MinerRecommitIntervalFlag.Name) {
1432
		cfg.Recommit = ctx.GlobalDuration(MinerRecommitIntervalFlag.Name)
1433 1434
	}
	if ctx.GlobalIsSet(MinerNoVerfiyFlag.Name) {
1435
		cfg.Noverify = ctx.GlobalBool(MinerNoVerfiyFlag.Name)
1436 1437 1438
	}
}

1439
func setWhitelist(ctx *cli.Context, cfg *ethconfig.Config) {
1440 1441 1442 1443 1444 1445 1446 1447 1448
	whitelist := ctx.GlobalString(WhitelistFlag.Name)
	if whitelist == "" {
		return
	}
	cfg.Whitelist = make(map[uint64]common.Hash)
	for _, entry := range strings.Split(whitelist, ",") {
		parts := strings.Split(entry, "=")
		if len(parts) != 2 {
			Fatalf("Invalid whitelist entry: %s", entry)
1449
		}
1450 1451 1452 1453 1454 1455 1456 1457 1458
		number, err := strconv.ParseUint(parts[0], 0, 64)
		if err != nil {
			Fatalf("Invalid whitelist block number %s: %v", parts[0], err)
		}
		var hash common.Hash
		if err = hash.UnmarshalText([]byte(parts[1])); err != nil {
			Fatalf("Invalid whitelist hash %s: %v", parts[1], err)
		}
		cfg.Whitelist[number] = hash
1459 1460 1461
	}
}

1462
// CheckExclusive verifies that only a single instance of the provided flags was
1463 1464
// set by the user. Each flag might optionally be followed by a string type to
// specialize it further.
1465
func CheckExclusive(ctx *cli.Context, args ...interface{}) {
1466
	set := make([]string, 0, 1)
1467 1468 1469 1470 1471 1472 1473 1474 1475 1476 1477 1478
	for i := 0; i < len(args); i++ {
		// Make sure the next argument is a flag and skip if not set
		flag, ok := args[i].(cli.Flag)
		if !ok {
			panic(fmt.Sprintf("invalid argument, not cli.Flag type: %T", args[i]))
		}
		// Check if next arg extends current and expand its name if so
		name := flag.GetName()

		if i+1 < len(args) {
			switch option := args[i+1].(type) {
			case string:
1479
				// Extended flag check, make sure value set doesn't conflict with passed in option
1480 1481
				if ctx.GlobalString(flag.GetName()) == option {
					name += "=" + option
1482
					set = append(set, "--"+name)
1483
				}
1484
				// shift arguments and continue
1485
				i++
1486
				continue
1487 1488 1489 1490 1491 1492 1493

			case cli.Flag:
			default:
				panic(fmt.Sprintf("invalid argument, not cli.Flag or string extension: %T", args[i+1]))
			}
		}
		// Mark the flag if it's set
1494
		if ctx.GlobalIsSet(flag.GetName()) {
1495
			set = append(set, "--"+name)
1496 1497
		}
	}
1498
	if len(set) > 1 {
1499
		Fatalf("Flags %v can't be used at the same time", strings.Join(set, ", "))
1500
	}
1501 1502
}

1503
// SetShhConfig applies shh-related command line flags to the config.
1504 1505 1506 1507 1508 1509
func SetShhConfig(ctx *cli.Context, stack *node.Node) {
	if ctx.GlobalIsSet(WhisperEnabledFlag.Name) ||
		ctx.GlobalIsSet(WhisperMaxMessageSizeFlag.Name) ||
		ctx.GlobalIsSet(WhisperMinPOWFlag.Name) ||
		ctx.GlobalIsSet(WhisperRestrictConnectionBetweenLightClientsFlag.Name) {
		log.Warn("Whisper support has been deprecated and the code has been moved to github.com/ethereum/whisper")
1510
	}
1511 1512
}

1513
// SetEthConfig applies eth-related command line flags to the config.
1514
func SetEthConfig(ctx *cli.Context, stack *node.Node, cfg *ethconfig.Config) {
1515
	// Avoid conflicting network flags
1516
	CheckExclusive(ctx, MainnetFlag, DeveloperFlag, LegacyTestnetFlag, RopstenFlag, RinkebyFlag, GoerliFlag, YoloV3Flag)
1517
	CheckExclusive(ctx, LegacyLightServFlag, LightServeFlag, SyncModeFlag, "light")
1518
	CheckExclusive(ctx, DeveloperFlag, ExternalSignerFlag) // Can't use both ephemeral unlocked and external signer
1519
	CheckExclusive(ctx, GCModeFlag, "archive", TxLookupLimitFlag)
1520 1521 1522
	if (ctx.GlobalIsSet(LegacyLightServFlag.Name) || ctx.GlobalIsSet(LightServeFlag.Name)) && ctx.GlobalIsSet(TxLookupLimitFlag.Name) {
		log.Warn("LES server cannot serve old transaction status and cannot connect below les/4 protocol version if transaction lookup index is limited")
	}
1523 1524 1525 1526
	var ks *keystore.KeyStore
	if keystores := stack.AccountManager().Backends(keystore.KeyStoreType); len(keystores) > 0 {
		ks = keystores[0].(*keystore.KeyStore)
	}
1527
	setEtherbase(ctx, ks, cfg)
1528
	setGPO(ctx, &cfg.GPO, ctx.GlobalString(SyncModeFlag.Name) == "light")
1529
	setTxPool(ctx, &cfg.TxPool)
1530
	setEthash(ctx, cfg)
1531
	setMiner(ctx, &cfg.Miner)
1532
	setWhitelist(ctx, cfg)
1533
	setLes(ctx, cfg)
1534

1535
	if ctx.GlobalIsSet(SyncModeFlag.Name) {
1536 1537 1538
		cfg.SyncMode = *GlobalTextMarshaler(ctx, SyncModeFlag.Name).(*downloader.SyncMode)
	}
	if ctx.GlobalIsSet(NetworkIdFlag.Name) {
1539
		cfg.NetworkId = ctx.GlobalUint64(NetworkIdFlag.Name)
1540
	}
1541 1542
	if ctx.GlobalIsSet(CacheFlag.Name) || ctx.GlobalIsSet(CacheDatabaseFlag.Name) {
		cfg.DatabaseCache = ctx.GlobalInt(CacheFlag.Name) * ctx.GlobalInt(CacheDatabaseFlag.Name) / 100
1543 1544
	}
	cfg.DatabaseHandles = makeDatabaseHandles()
1545 1546 1547
	if ctx.GlobalIsSet(AncientFlag.Name) {
		cfg.DatabaseFreezer = ctx.GlobalString(AncientFlag.Name)
	}
1548

1549 1550 1551
	if gcmode := ctx.GlobalString(GCModeFlag.Name); gcmode != "full" && gcmode != "archive" {
		Fatalf("--%s must be either 'full' or 'archive'", GCModeFlag.Name)
	}
1552 1553 1554 1555 1556 1557
	if ctx.GlobalIsSet(GCModeFlag.Name) {
		cfg.NoPruning = ctx.GlobalString(GCModeFlag.Name) == "archive"
	}
	if ctx.GlobalIsSet(CacheNoPrefetchFlag.Name) {
		cfg.NoPrefetch = ctx.GlobalBool(CacheNoPrefetchFlag.Name)
	}
1558 1559 1560 1561 1562 1563
	// Read the value from the flag no matter if it's set or not.
	cfg.Preimages = ctx.GlobalBool(CachePreimagesFlag.Name)
	if cfg.NoPruning && !cfg.Preimages {
		cfg.Preimages = true
		log.Info("Enabling recording of key preimages since archive mode is used")
	}
1564 1565 1566
	if ctx.GlobalIsSet(TxLookupLimitFlag.Name) {
		cfg.TxLookupLimit = ctx.GlobalUint64(TxLookupLimitFlag.Name)
	}
1567 1568 1569
	if ctx.GlobalIsSet(CacheFlag.Name) || ctx.GlobalIsSet(CacheTrieFlag.Name) {
		cfg.TrieCleanCache = ctx.GlobalInt(CacheFlag.Name) * ctx.GlobalInt(CacheTrieFlag.Name) / 100
	}
1570 1571 1572 1573 1574 1575
	if ctx.GlobalIsSet(CacheTrieJournalFlag.Name) {
		cfg.TrieCleanCacheJournal = ctx.GlobalString(CacheTrieJournalFlag.Name)
	}
	if ctx.GlobalIsSet(CacheTrieRejournalFlag.Name) {
		cfg.TrieCleanCacheRejournal = ctx.GlobalDuration(CacheTrieRejournalFlag.Name)
	}
1576
	if ctx.GlobalIsSet(CacheFlag.Name) || ctx.GlobalIsSet(CacheGCFlag.Name) {
1577
		cfg.TrieDirtyCache = ctx.GlobalInt(CacheFlag.Name) * ctx.GlobalInt(CacheGCFlag.Name) / 100
1578
	}
1579 1580 1581
	if ctx.GlobalIsSet(CacheFlag.Name) || ctx.GlobalIsSet(CacheSnapshotFlag.Name) {
		cfg.SnapshotCache = ctx.GlobalInt(CacheFlag.Name) * ctx.GlobalInt(CacheSnapshotFlag.Name) / 100
	}
1582
	if !ctx.GlobalIsSet(SnapshotFlag.Name) {
1583 1584 1585 1586 1587 1588 1589
		// If snap-sync is requested, this flag is also required
		if cfg.SyncMode == downloader.SnapSync {
			log.Info("Snap sync requested, enabling --snapshot")
		} else {
			cfg.TrieCleanCache += cfg.SnapshotCache
			cfg.SnapshotCache = 0 // Disabled
		}
1590
	}
1591 1592 1593 1594 1595 1596 1597
	if ctx.GlobalIsSet(DocRootFlag.Name) {
		cfg.DocRoot = ctx.GlobalString(DocRootFlag.Name)
	}
	if ctx.GlobalIsSet(VMEnableDebugFlag.Name) {
		// TODO(fjl): force-enable this in --dev mode
		cfg.EnablePreimageRecording = ctx.GlobalBool(VMEnableDebugFlag.Name)
	}
1598

1599 1600 1601 1602 1603 1604 1605
	if ctx.GlobalIsSet(EWASMInterpreterFlag.Name) {
		cfg.EWASMInterpreter = ctx.GlobalString(EWASMInterpreterFlag.Name)
	}

	if ctx.GlobalIsSet(EVMInterpreterFlag.Name) {
		cfg.EVMInterpreter = ctx.GlobalString(EVMInterpreterFlag.Name)
	}
1606 1607
	if ctx.GlobalIsSet(RPCGlobalGasCapFlag.Name) {
		cfg.RPCGasCap = ctx.GlobalUint64(RPCGlobalGasCapFlag.Name)
1608 1609 1610 1611 1612
	}
	if cfg.RPCGasCap != 0 {
		log.Info("Set global gas cap", "cap", cfg.RPCGasCap)
	} else {
		log.Info("Global gas cap disabled")
1613
	}
1614 1615
	if ctx.GlobalIsSet(RPCGlobalTxFeeCapFlag.Name) {
		cfg.RPCTxFeeCap = ctx.GlobalFloat64(RPCGlobalTxFeeCapFlag.Name)
1616
	}
1617
	if ctx.GlobalIsSet(NoDiscoverFlag.Name) {
1618
		cfg.EthDiscoveryURLs, cfg.SnapDiscoveryURLs = []string{}, []string{}
1619
	} else if ctx.GlobalIsSet(DNSDiscoveryFlag.Name) {
1620 1621
		urls := ctx.GlobalString(DNSDiscoveryFlag.Name)
		if urls == "" {
1622
			cfg.EthDiscoveryURLs = []string{}
1623
		} else {
1624
			cfg.EthDiscoveryURLs = SplitAndTrim(urls)
1625 1626
		}
	}
1627
	// Override any default configs for hard coded networks.
1628
	switch {
1629 1630 1631 1632 1633 1634
	case ctx.GlobalBool(MainnetFlag.Name):
		if !ctx.GlobalIsSet(NetworkIdFlag.Name) {
			cfg.NetworkId = 1
		}
		cfg.Genesis = core.DefaultGenesisBlock()
		SetDNSDiscoveryDefaults(cfg, params.MainnetGenesisHash)
1635
	case ctx.GlobalBool(LegacyTestnetFlag.Name) || ctx.GlobalBool(RopstenFlag.Name):
1636
		if !ctx.GlobalIsSet(NetworkIdFlag.Name) {
1637
			cfg.NetworkId = 3
1638
		}
1639
		cfg.Genesis = core.DefaultRopstenGenesisBlock()
1640
		SetDNSDiscoveryDefaults(cfg, params.RopstenGenesisHash)
1641 1642 1643 1644 1645
	case ctx.GlobalBool(RinkebyFlag.Name):
		if !ctx.GlobalIsSet(NetworkIdFlag.Name) {
			cfg.NetworkId = 4
		}
		cfg.Genesis = core.DefaultRinkebyGenesisBlock()
1646
		SetDNSDiscoveryDefaults(cfg, params.RinkebyGenesisHash)
1647 1648 1649 1650 1651
	case ctx.GlobalBool(GoerliFlag.Name):
		if !ctx.GlobalIsSet(NetworkIdFlag.Name) {
			cfg.NetworkId = 5
		}
		cfg.Genesis = core.DefaultGoerliGenesisBlock()
1652
		SetDNSDiscoveryDefaults(cfg, params.GoerliGenesisHash)
1653
	case ctx.GlobalBool(YoloV3Flag.Name):
1654
		if !ctx.GlobalIsSet(NetworkIdFlag.Name) {
1655
			cfg.NetworkId = new(big.Int).SetBytes([]byte("yolov3")).Uint64() // "yolov3"
1656
		}
1657
		cfg.Genesis = core.DefaultYoloV3GenesisBlock()
1658
	case ctx.GlobalBool(DeveloperFlag.Name):
1659 1660 1661
		if !ctx.GlobalIsSet(NetworkIdFlag.Name) {
			cfg.NetworkId = 1337
		}
1662 1663
		// Create new developer account or reuse existing one
		var (
1664 1665 1666
			developer  accounts.Account
			passphrase string
			err        error
1667
		)
1668 1669 1670 1671 1672 1673 1674 1675 1676 1677
		if list := MakePasswordList(ctx); len(list) > 0 {
			// Just take the first value. Although the function returns a possible multiple values and
			// some usages iterate through them as attempts, that doesn't make sense in this setting,
			// when we're definitely concerned with only one account.
			passphrase = list[0]
		}
		// setEtherbase has been called above, configuring the miner address from command line flags.
		if cfg.Miner.Etherbase != (common.Address{}) {
			developer = accounts.Account{Address: cfg.Miner.Etherbase}
		} else if accs := ks.Accounts(); len(accs) > 0 {
1678 1679
			developer = ks.Accounts()[0]
		} else {
1680
			developer, err = ks.NewAccount(passphrase)
1681 1682 1683 1684
			if err != nil {
				Fatalf("Failed to create developer account: %v", err)
			}
		}
1685
		if err := ks.Unlock(developer, passphrase); err != nil {
1686 1687 1688 1689
			Fatalf("Failed to unlock developer account: %v", err)
		}
		log.Info("Using developer account", "address", developer.Address)

1690
		// Create a new developer genesis block or reuse existing one
1691
		cfg.Genesis = core.DeveloperGenesisBlock(uint64(ctx.GlobalInt(DeveloperPeriodFlag.Name)), developer.Address)
1692 1693 1694 1695 1696 1697 1698 1699 1700
		if ctx.GlobalIsSet(DataDirFlag.Name) {
			// Check if we have an already initialized chain and fall back to
			// that if so. Otherwise we need to generate a new genesis spec.
			chaindb := MakeChainDatabase(ctx, stack)
			if rawdb.ReadCanonicalHash(chaindb, 0) != (common.Hash{}) {
				cfg.Genesis = nil // fallback to db content
			}
			chaindb.Close()
		}
1701
		if !ctx.GlobalIsSet(MinerGasPriceFlag.Name) && !ctx.GlobalIsSet(LegacyMinerGasPriceFlag.Name) {
1702
			cfg.Miner.GasPrice = big.NewInt(1)
1703
		}
1704 1705
	default:
		if cfg.NetworkId == 1 {
1706
			SetDNSDiscoveryDefaults(cfg, params.MainnetGenesisHash)
1707 1708 1709 1710
		}
	}
}

1711
// SetDNSDiscoveryDefaults configures DNS discovery with the given URL if
1712
// no URLs are set.
1713
func SetDNSDiscoveryDefaults(cfg *ethconfig.Config, genesis common.Hash) {
1714
	if cfg.EthDiscoveryURLs != nil {
1715 1716
		return // already set through flags/config
	}
1717
	protocol := "all"
1718 1719 1720 1721
	if cfg.SyncMode == downloader.LightSync {
		protocol = "les"
	}
	if url := params.KnownDNSNetwork(genesis, protocol); url != "" {
1722 1723 1724 1725 1726 1727
		cfg.EthDiscoveryURLs = []string{url}
	}
	if cfg.SyncMode == downloader.SnapSync {
		if url := params.KnownDNSNetwork(genesis, "snap"); url != "" {
			cfg.SnapDiscoveryURLs = []string{url}
		}
1728
	}
1729
}
1730

1731
// RegisterEthService adds an Ethereum client to the stack.
1732
func RegisterEthService(stack *node.Node, cfg *ethconfig.Config) ethapi.Backend {
1733
	if cfg.SyncMode == downloader.LightSync {
R
rene 已提交
1734 1735 1736 1737
		backend, err := les.New(stack, cfg)
		if err != nil {
			Fatalf("Failed to register the Ethereum service: %v", err)
		}
1738
		stack.RegisterAPIs(tracers.APIs(backend.ApiBackend))
R
rene 已提交
1739
		return backend.ApiBackend
1740 1741 1742 1743 1744 1745 1746
	}
	backend, err := eth.New(stack, cfg)
	if err != nil {
		Fatalf("Failed to register the Ethereum service: %v", err)
	}
	if cfg.LightServ > 0 {
		_, err := les.NewLesServer(stack, backend, cfg)
R
rene 已提交
1747
		if err != nil {
1748
			Fatalf("Failed to create the LES server: %v", err)
R
rene 已提交
1749
		}
1750
	}
1751
	stack.RegisterAPIs(tracers.APIs(backend.APIBackend))
1752
	return backend.APIBackend
1753 1754
}

1755
// RegisterEthStatsService configures the Ethereum Stats daemon and adds it to
S
Sarlor 已提交
1756
// the given node.
R
rene 已提交
1757 1758
func RegisterEthStatsService(stack *node.Node, backend ethapi.Backend, url string) {
	if err := ethstats.New(stack, backend, backend.Engine(), url); err != nil {
1759
		Fatalf("Failed to register the Ethereum Stats service: %v", err)
1760 1761 1762
	}
}

1763
// RegisterGraphQLService is a utility function to construct a new service and register it against a node.
R
rene 已提交
1764 1765
func RegisterGraphQLService(stack *node.Node, backend ethapi.Backend, cfg node.Config) {
	if err := graphql.New(stack, backend, cfg.GraphQLCors, cfg.GraphQLVirtualHosts); err != nil {
1766 1767 1768 1769
		Fatalf("Failed to register the GraphQL service: %v", err)
	}
}

1770 1771 1772
func SetupMetrics(ctx *cli.Context) {
	if metrics.Enabled {
		log.Info("Enabling metrics collection")
1773

1774 1775 1776 1777 1778 1779 1780 1781 1782
		var (
			enableExport = ctx.GlobalBool(MetricsEnableInfluxDBFlag.Name)
			endpoint     = ctx.GlobalString(MetricsInfluxDBEndpointFlag.Name)
			database     = ctx.GlobalString(MetricsInfluxDBDatabaseFlag.Name)
			username     = ctx.GlobalString(MetricsInfluxDBUsernameFlag.Name)
			password     = ctx.GlobalString(MetricsInfluxDBPasswordFlag.Name)
		)

		if enableExport {
1783 1784
			tagsMap := SplitTagsFlag(ctx.GlobalString(MetricsInfluxDBTagsFlag.Name))

1785
			log.Info("Enabling metrics export to InfluxDB")
1786 1787 1788

			go influxdb.InfluxDBWithTags(metrics.DefaultRegistry, 10*time.Second, endpoint, database, username, password, "geth.", tagsMap)
		}
1789 1790 1791 1792 1793 1794

		if ctx.GlobalIsSet(MetricsHTTPFlag.Name) {
			address := fmt.Sprintf("%s:%d", ctx.GlobalString(MetricsHTTPFlag.Name), ctx.GlobalInt(MetricsPortFlag.Name))
			log.Info("Enabling stand-alone metrics HTTP endpoint", "address", address)
			exp.Setup(address)
		}
1795 1796 1797 1798 1799 1800 1801 1802 1803 1804 1805 1806 1807 1808
	}
}

func SplitTagsFlag(tagsFlag string) map[string]string {
	tags := strings.Split(tagsFlag, ",")
	tagsMap := map[string]string{}

	for _, t := range tags {
		if t != "" {
			kv := strings.Split(t, "=")

			if len(kv) == 2 {
				tagsMap[kv[0]] = kv[1]
			}
1809 1810
		}
	}
1811 1812

	return tagsMap
1813 1814
}

1815
// MakeChainDatabase open an LevelDB using the flags passed to the client and will hard crash if it fails.
1816
func MakeChainDatabase(ctx *cli.Context, stack *node.Node) ethdb.Database {
1817
	var (
1818
		cache   = ctx.GlobalInt(CacheFlag.Name) * ctx.GlobalInt(CacheDatabaseFlag.Name) / 100
1819
		handles = makeDatabaseHandles()
1820 1821 1822

		err     error
		chainDb ethdb.Database
1823
	)
1824
	if ctx.GlobalString(SyncModeFlag.Name) == "light" {
1825 1826 1827 1828 1829
		name := "lightchaindata"
		chainDb, err = stack.OpenDatabase(name, cache, handles, "")
	} else {
		name := "chaindata"
		chainDb, err = stack.OpenDatabaseWithFreezer(name, cache, handles, ctx.GlobalString(AncientFlag.Name), "")
1830
	}
1831
	if err != nil {
1832
		Fatalf("Could not open database: %v", err)
1833
	}
1834 1835 1836
	return chainDb
}

F
Felix Lange 已提交
1837 1838 1839
func MakeGenesis(ctx *cli.Context) *core.Genesis {
	var genesis *core.Genesis
	switch {
1840 1841
	case ctx.GlobalBool(LegacyTestnetFlag.Name) || ctx.GlobalBool(RopstenFlag.Name):
		genesis = core.DefaultRopstenGenesisBlock()
1842 1843
	case ctx.GlobalBool(RinkebyFlag.Name):
		genesis = core.DefaultRinkebyGenesisBlock()
1844 1845
	case ctx.GlobalBool(GoerliFlag.Name):
		genesis = core.DefaultGoerliGenesisBlock()
1846 1847
	case ctx.GlobalBool(YoloV3Flag.Name):
		genesis = core.DefaultYoloV3GenesisBlock()
1848 1849
	case ctx.GlobalBool(DeveloperFlag.Name):
		Fatalf("Developer chains are ephemeral")
F
Felix Lange 已提交
1850 1851 1852 1853
	}
	return genesis
}

1854
// MakeChain creates a chain manager from set command line flags.
1855
func MakeChain(ctx *cli.Context, stack *node.Node, readOnly bool) (chain *core.BlockChain, chainDb ethdb.Database) {
1856
	var err error
1857
	chainDb = MakeChainDatabase(ctx, stack)
F
Felix Lange 已提交
1858 1859 1860 1861
	config, _, err := core.SetupGenesisBlock(chainDb, MakeGenesis(ctx))
	if err != nil {
		Fatalf("%v", err)
	}
1862 1863 1864 1865 1866 1867
	var engine consensus.Engine
	if config.Clique != nil {
		engine = clique.New(config.Clique, chainDb)
	} else {
		engine = ethash.NewFaker()
		if !ctx.GlobalBool(FakePoWFlag.Name) {
1868
			engine = ethash.New(ethash.Config{
1869 1870 1871 1872 1873 1874 1875 1876
				CacheDir:         stack.ResolvePath(ethconfig.Defaults.Ethash.CacheDir),
				CachesInMem:      ethconfig.Defaults.Ethash.CachesInMem,
				CachesOnDisk:     ethconfig.Defaults.Ethash.CachesOnDisk,
				CachesLockMmap:   ethconfig.Defaults.Ethash.CachesLockMmap,
				DatasetDir:       stack.ResolvePath(ethconfig.Defaults.Ethash.DatasetDir),
				DatasetsInMem:    ethconfig.Defaults.Ethash.DatasetsInMem,
				DatasetsOnDisk:   ethconfig.Defaults.Ethash.DatasetsOnDisk,
				DatasetsLockMmap: ethconfig.Defaults.Ethash.DatasetsLockMmap,
1877
			}, nil, false)
1878 1879
		}
	}
1880 1881 1882 1883
	if gcmode := ctx.GlobalString(GCModeFlag.Name); gcmode != "full" && gcmode != "archive" {
		Fatalf("--%s must be either 'full' or 'archive'", GCModeFlag.Name)
	}
	cache := &core.CacheConfig{
1884
		TrieCleanLimit:      ethconfig.Defaults.TrieCleanCache,
1885
		TrieCleanNoPrefetch: ctx.GlobalBool(CacheNoPrefetchFlag.Name),
1886
		TrieDirtyLimit:      ethconfig.Defaults.TrieDirtyCache,
1887
		TrieDirtyDisabled:   ctx.GlobalString(GCModeFlag.Name) == "archive",
1888 1889
		TrieTimeLimit:       ethconfig.Defaults.TrieTimeout,
		SnapshotLimit:       ethconfig.Defaults.SnapshotCache,
1890 1891 1892 1893 1894
		Preimages:           ctx.GlobalBool(CachePreimagesFlag.Name),
	}
	if cache.TrieDirtyDisabled && !cache.Preimages {
		cache.Preimages = true
		log.Info("Enabling recording of key preimages since archive mode is used")
1895
	}
1896 1897 1898
	if !ctx.GlobalIsSet(SnapshotFlag.Name) {
		cache.SnapshotLimit = 0 // Disabled
	}
1899 1900
	if ctx.GlobalIsSet(CacheFlag.Name) || ctx.GlobalIsSet(CacheTrieFlag.Name) {
		cache.TrieCleanLimit = ctx.GlobalInt(CacheFlag.Name) * ctx.GlobalInt(CacheTrieFlag.Name) / 100
1901 1902
	}
	if ctx.GlobalIsSet(CacheFlag.Name) || ctx.GlobalIsSet(CacheGCFlag.Name) {
1903
		cache.TrieDirtyLimit = ctx.GlobalInt(CacheFlag.Name) * ctx.GlobalInt(CacheGCFlag.Name) / 100
1904
	}
F
Felix Lange 已提交
1905
	vmcfg := vm.Config{EnablePreimageRecording: ctx.GlobalBool(VMEnableDebugFlag.Name)}
1906 1907 1908 1909 1910 1911
	var limit *uint64
	if ctx.GlobalIsSet(TxLookupLimitFlag.Name) && !readOnly {
		l := ctx.GlobalUint64(TxLookupLimitFlag.Name)
		limit = &l
	}
	chain, err = core.NewBlockChain(chainDb, cache, config, engine, vmcfg, nil, limit)
O
obscuren 已提交
1912
	if err != nil {
F
Felix Lange 已提交
1913
		Fatalf("Can't create BlockChain: %v", err)
O
obscuren 已提交
1914
	}
1915
	return chain, chainDb
1916
}
1917 1918 1919 1920 1921 1922 1923 1924 1925

// MakeConsolePreloads retrieves the absolute paths for the console JavaScript
// scripts to preload before starting.
func MakeConsolePreloads(ctx *cli.Context) []string {
	// Skip preloading if there's nothing to preload
	if ctx.GlobalString(PreloadJSFlag.Name) == "" {
		return nil
	}
	// Otherwise resolve absolute paths and return them
1926
	var preloads []string
1927 1928

	for _, file := range strings.Split(ctx.GlobalString(PreloadJSFlag.Name), ",") {
1929
		preloads = append(preloads, strings.TrimSpace(file))
1930 1931 1932
	}
	return preloads
}
1933 1934 1935 1936 1937 1938 1939 1940 1941 1942 1943 1944 1945 1946 1947 1948 1949 1950 1951 1952 1953 1954 1955 1956

// MigrateFlags sets the global flag from a local flag when it's set.
// This is a temporary function used for migrating old command/flags to the
// new format.
//
// e.g. geth account new --keystore /tmp/mykeystore --lightkdf
//
// is equivalent after calling this method with:
//
// geth --keystore /tmp/mykeystore --lightkdf account new
//
// This allows the use of the existing configuration functionality.
// When all flags are migrated this function can be removed and the existing
// configuration functionality must be changed that is uses local flags
func MigrateFlags(action func(ctx *cli.Context) error) func(*cli.Context) error {
	return func(ctx *cli.Context) error {
		for _, name := range ctx.FlagNames() {
			if ctx.IsSet(name) {
				ctx.GlobalSet(name, ctx.String(name))
			}
		}
		return action(ctx)
	}
}