2015-07-07 03:54:22 +03:00
|
|
|
// Copyright 2014 The go-ethereum Authors
|
2015-07-22 19:48:40 +03:00
|
|
|
// This file is part of the go-ethereum library.
|
2015-07-07 03:54:22 +03:00
|
|
|
//
|
2015-07-23 19:35:11 +03:00
|
|
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
2015-07-07 03:54:22 +03:00
|
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
2015-07-22 19:48:40 +03:00
|
|
|
// The go-ethereum library is distributed in the hope that it will be useful,
|
2015-07-07 03:54:22 +03:00
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2015-07-22 19:48:40 +03:00
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2015-07-07 03:54:22 +03:00
|
|
|
// GNU Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
2015-07-22 19:48:40 +03:00
|
|
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
2015-07-07 03:54:22 +03:00
|
|
|
|
2015-07-07 06:08:16 +03:00
|
|
|
// Package eth implements the Ethereum protocol.
|
2014-12-14 20:03:24 +02:00
|
|
|
package eth
|
|
|
|
|
|
|
|
import (
|
2024-03-22 20:53:53 +03:00
|
|
|
"encoding/json"
|
2015-01-04 15:20:16 +02:00
|
|
|
"fmt"
|
2017-05-16 22:07:27 +03:00
|
|
|
"math/big"
|
2017-04-12 17:27:23 +03:00
|
|
|
"runtime"
|
2016-05-12 20:32:04 +03:00
|
|
|
"sync"
|
2014-12-14 20:03:24 +02:00
|
|
|
|
2015-02-26 14:22:09 +02:00
|
|
|
"github.com/ethereum/go-ethereum/accounts"
|
2015-03-18 14:00:01 +02:00
|
|
|
"github.com/ethereum/go-ethereum/common"
|
2017-04-12 17:27:23 +03:00
|
|
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
2017-04-05 01:16:29 +03:00
|
|
|
"github.com/ethereum/go-ethereum/consensus"
|
2014-12-14 20:03:24 +02:00
|
|
|
"github.com/ethereum/go-ethereum/core"
|
2017-08-29 14:13:11 +03:00
|
|
|
"github.com/ethereum/go-ethereum/core/bloombits"
|
2018-05-07 14:35:06 +03:00
|
|
|
"github.com/ethereum/go-ethereum/core/rawdb"
|
all: bloom-filter based pruning mechanism (#21724)
* cmd, core, tests: initial state pruner
core: fix db inspector
cmd/geth: add verify-state
cmd/geth: add verification tool
core/rawdb: implement flatdb
cmd, core: fix rebase
core/state: use new contract code layout
core/state/pruner: avoid deleting genesis state
cmd/geth: add helper function
core, cmd: fix extract genesis
core: minor fixes
contracts: remove useless
core/state/snapshot: plugin stacktrie
core: polish
core/state/snapshot: iterate storage concurrently
core/state/snapshot: fix iteration
core: add comments
core/state/snapshot: polish code
core/state: polish
core/state/snapshot: rebase
core/rawdb: add comments
core/rawdb: fix tests
core/rawdb: improve tests
core/state/snapshot: fix concurrent iteration
core/state: run pruning during the recovery
core, trie: implement martin's idea
core, eth: delete flatdb and polish pruner
trie: fix import
core/state/pruner: add log
core/state/pruner: fix issues
core/state/pruner: don't read back
core/state/pruner: fix contract code write
core/state/pruner: check root node presence
cmd, core: polish log
core/state: use HEAD-127 as the target
core/state/snapshot: improve tests
cmd/geth: fix verification tool
cmd/geth: use HEAD as the verification default target
all: replace the bloomfilter with martin's fork
cmd, core: polish code
core, cmd: forcibly delete state root
core/state/pruner: add hash64
core/state/pruner: fix blacklist
core/state: remove blacklist
cmd, core: delete trie clean cache before pruning
cmd, core: fix lint
cmd, core: fix rebase
core/state: fix the special case for clique networks
core/state/snapshot: remove useless code
core/state/pruner: capping the snapshot after pruning
cmd, core, eth: fixes
core/rawdb: update db inspector
cmd/geth: polish code
core/state/pruner: fsync bloom filter
cmd, core: print warning log
core/state/pruner: adjust the parameters for bloom filter
cmd, core: create the bloom filter by size
core: polish
core/state/pruner: sanitize invalid bloomfilter size
cmd: address comments
cmd/geth: address comments
cmd/geth: address comment
core/state/pruner: address comments
core/state/pruner: rename homedir to datadir
cmd, core: address comments
core/state/pruner: address comment
core/state: address comments
core, cmd, tests: address comments
core: address comments
core/state/pruner: release the iterator after each commit
core/state/pruner: improve pruner
cmd, core: adjust bloom paramters
core/state/pruner: fix lint
core/state/pruner: fix tests
core: fix rebase
core/state/pruner: remove atomic rename
core/state/pruner: address comments
all: run go mod tidy
core/state/pruner: avoid false-positive for the middle state roots
core/state/pruner: add checks for middle roots
cmd/geth: replace crit with error
* core/state/pruner: fix lint
* core: drop legacy bloom filter
* core/state/snapshot: improve pruner
* core/state/snapshot: polish concurrent logs to report ETA vs. hashes
* core/state/pruner: add progress report for pruning and compaction too
* core: fix snapshot test API
* core/state: fix some pruning logs
* core/state/pruner: support recovering from bloom flush fail
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-02-08 14:16:30 +03:00
|
|
|
"github.com/ethereum/go-ethereum/core/state/pruner"
|
2022-10-24 16:13:55 +03:00
|
|
|
"github.com/ethereum/go-ethereum/core/txpool"
|
2023-07-27 13:45:35 +03:00
|
|
|
"github.com/ethereum/go-ethereum/core/txpool/blobpool"
|
2023-06-16 15:29:40 +03:00
|
|
|
"github.com/ethereum/go-ethereum/core/txpool/legacypool"
|
2015-03-13 19:34:43 +02:00
|
|
|
"github.com/ethereum/go-ethereum/core/types"
|
2017-01-17 14:19:50 +03:00
|
|
|
"github.com/ethereum/go-ethereum/core/vm"
|
2015-04-13 18:22:32 +03:00
|
|
|
"github.com/ethereum/go-ethereum/eth/downloader"
|
2021-02-05 15:51:15 +03:00
|
|
|
"github.com/ethereum/go-ethereum/eth/ethconfig"
|
2015-12-16 05:26:23 +02:00
|
|
|
"github.com/ethereum/go-ethereum/eth/gasprice"
|
2020-12-14 12:27:15 +03:00
|
|
|
"github.com/ethereum/go-ethereum/eth/protocols/eth"
|
|
|
|
"github.com/ethereum/go-ethereum/eth/protocols/snap"
|
2024-03-22 20:53:53 +03:00
|
|
|
"github.com/ethereum/go-ethereum/eth/tracers"
|
2015-01-04 15:20:16 +02:00
|
|
|
"github.com/ethereum/go-ethereum/ethdb"
|
2014-12-14 20:03:24 +02:00
|
|
|
"github.com/ethereum/go-ethereum/event"
|
2015-12-16 05:26:23 +02:00
|
|
|
"github.com/ethereum/go-ethereum/internal/ethapi"
|
2021-12-17 17:18:51 +03:00
|
|
|
"github.com/ethereum/go-ethereum/internal/shutdowncheck"
|
2017-02-22 15:10:07 +03:00
|
|
|
"github.com/ethereum/go-ethereum/log"
|
2015-02-17 13:24:51 +02:00
|
|
|
"github.com/ethereum/go-ethereum/miner"
|
2015-11-17 18:33:25 +02:00
|
|
|
"github.com/ethereum/go-ethereum/node"
|
2014-12-14 20:03:24 +02:00
|
|
|
"github.com/ethereum/go-ethereum/p2p"
|
2021-05-04 12:29:32 +03:00
|
|
|
"github.com/ethereum/go-ethereum/p2p/dnsdisc"
|
2020-02-13 16:38:30 +03:00
|
|
|
"github.com/ethereum/go-ethereum/p2p/enode"
|
2016-10-20 14:36:29 +03:00
|
|
|
"github.com/ethereum/go-ethereum/params"
|
2017-04-12 17:27:23 +03:00
|
|
|
"github.com/ethereum/go-ethereum/rlp"
|
2015-12-16 11:58:01 +02:00
|
|
|
"github.com/ethereum/go-ethereum/rpc"
|
2014-12-14 20:03:24 +02:00
|
|
|
)
|
|
|
|
|
2021-02-05 15:51:15 +03:00
|
|
|
// Config contains the configuration options of the ETH protocol.
|
|
|
|
// Deprecated: use ethconfig.Config instead.
|
|
|
|
type Config = ethconfig.Config
|
|
|
|
|
2016-06-30 13:03:26 +03:00
|
|
|
// Ethereum implements the Ethereum full node service.
|
|
|
|
type Ethereum struct {
|
2024-08-15 23:14:42 +03:00
|
|
|
// core protocol objects
|
|
|
|
config *ethconfig.Config
|
|
|
|
txPool *txpool.TxPool
|
|
|
|
blockchain *core.BlockChain
|
2017-09-05 19:18:28 +03:00
|
|
|
|
2024-08-15 23:14:42 +03:00
|
|
|
handler *handler
|
|
|
|
discmix *enode.FairMix
|
2017-09-05 19:18:28 +03:00
|
|
|
|
2015-12-16 05:26:23 +02:00
|
|
|
// DB interfaces
|
|
|
|
chainDb ethdb.Database // Block chain database
|
2014-12-14 20:03:24 +02:00
|
|
|
|
2015-12-16 05:26:23 +02:00
|
|
|
eventMux *event.TypeMux
|
2017-04-05 01:16:29 +03:00
|
|
|
engine consensus.Engine
|
2015-12-16 05:26:23 +02:00
|
|
|
accountManager *accounts.Manager
|
2015-05-26 15:17:43 +03:00
|
|
|
|
2020-03-27 16:03:20 +03:00
|
|
|
bloomRequests chan chan *bloombits.Retrieval // Channel receiving bloom data retrieval requests
|
|
|
|
bloomIndexer *core.ChainIndexer // Bloom indexer operating during block imports
|
|
|
|
closeBloomHandler chan struct{}
|
2017-08-18 22:52:20 +03:00
|
|
|
|
2018-05-09 10:59:00 +03:00
|
|
|
APIBackend *EthAPIBackend
|
2015-10-27 00:24:09 +03:00
|
|
|
|
2024-03-06 15:45:03 +03:00
|
|
|
miner *miner.Miner
|
|
|
|
gasPrice *big.Int
|
2014-12-14 20:03:24 +02:00
|
|
|
|
2018-06-14 13:14:52 +03:00
|
|
|
networkID uint64
|
2022-06-21 12:05:43 +03:00
|
|
|
netRPCService *ethapi.NetAPI
|
2017-05-29 10:21:34 +03:00
|
|
|
|
2020-08-03 20:40:46 +03:00
|
|
|
p2pServer *p2p.Server
|
2014-12-14 20:03:24 +02:00
|
|
|
|
2020-08-03 20:40:46 +03:00
|
|
|
lock sync.RWMutex // Protects the variadic fields (e.g. gas price and etherbase)
|
2021-12-17 17:18:51 +03:00
|
|
|
|
|
|
|
shutdownTracker *shutdowncheck.ShutdownTracker // Tracks if and when the node has shutdown ungracefully
|
all: on-chain oracle checkpoint syncing (#19543)
* all: implement simple checkpoint syncing
cmd, les, node: remove callback mechanism
cmd, node: remove callback definition
les: simplify the registrar
les: expose checkpoint rpc services in the light client
les, light: don't store untrusted receipt
cmd, contracts, les: discard stale checkpoint
cmd, contracts/registrar: loose restriction of registeration
cmd, contracts: add replay-protection
all: off-chain multi-signature contract
params: deploy checkpoint contract for rinkeby
cmd/registrar: add raw signing mode for registrar
cmd/registrar, contracts/registrar, les: fixed messages
* cmd/registrar, contracts/registrar: fix lints
* accounts/abi/bind, les: address comments
* cmd, contracts, les, light, params: minor checkpoint sync cleanups
* cmd, eth, les, light: move checkpoint config to config file
* cmd, eth, les, params: address comments
* eth, les, params: address comments
* cmd: polish up the checkpoint admin CLI
* cmd, contracts, params: deploy new version contract
* cmd/checkpoint-admin: add another flag for clef mode signing
* cmd, contracts, les: rename and regen checkpoint oracle with abigen
2019-06-28 10:34:02 +03:00
|
|
|
}
|
|
|
|
|
2024-03-25 20:03:44 +03:00
|
|
|
// New creates a new Ethereum object (including the initialisation of the common Ethereum object),
|
|
|
|
// whose lifecycle will be managed by the provided node.
|
2021-02-05 15:51:15 +03:00
|
|
|
func New(stack *node.Node, config *ethconfig.Config) (*Ethereum, error) {
|
2018-08-23 13:02:36 +03:00
|
|
|
// Ensure configuration values are compatible and sane
|
2017-04-12 17:27:23 +03:00
|
|
|
if !config.SyncMode.IsValid() {
|
|
|
|
return nil, fmt.Errorf("invalid sync mode %d", config.SyncMode)
|
|
|
|
}
|
2024-04-09 13:14:30 +03:00
|
|
|
if config.Miner.GasPrice == nil || config.Miner.GasPrice.Sign() <= 0 {
|
2021-02-05 15:51:15 +03:00
|
|
|
log.Warn("Sanitizing invalid miner gas price", "provided", config.Miner.GasPrice, "updated", ethconfig.Defaults.Miner.GasPrice)
|
|
|
|
config.Miner.GasPrice = new(big.Int).Set(ethconfig.Defaults.Miner.GasPrice)
|
2018-08-23 13:02:36 +03:00
|
|
|
}
|
2019-02-05 13:49:59 +03:00
|
|
|
if config.NoPruning && config.TrieDirtyCache > 0 {
|
2020-05-06 13:01:01 +03:00
|
|
|
if config.SnapshotCache > 0 {
|
|
|
|
config.TrieCleanCache += config.TrieDirtyCache * 3 / 5
|
|
|
|
config.SnapshotCache += config.TrieDirtyCache * 2 / 5
|
|
|
|
} else {
|
|
|
|
config.TrieCleanCache += config.TrieDirtyCache
|
|
|
|
}
|
2019-02-05 13:49:59 +03:00
|
|
|
config.TrieDirtyCache = 0
|
|
|
|
}
|
|
|
|
log.Info("Allocated trie memory caches", "clean", common.StorageSize(config.TrieCleanCache)*1024*1024, "dirty", common.StorageSize(config.TrieDirtyCache)*1024*1024)
|
|
|
|
|
2018-08-23 13:02:36 +03:00
|
|
|
// Assemble the Ethereum object
|
2021-03-22 21:06:30 +03:00
|
|
|
chainDb, err := stack.OpenDatabaseWithFreezer("chaindata", config.DatabaseCache, config.DatabaseHandles, config.DatabaseFreezer, "eth/db/chaindata/", false)
|
2015-01-04 15:20:16 +02:00
|
|
|
if err != nil {
|
2015-11-17 18:33:25 +02:00
|
|
|
return nil, err
|
2015-01-04 15:20:16 +02:00
|
|
|
}
|
2023-10-11 11:27:44 +03:00
|
|
|
scheme, err := rawdb.ParseStateScheme(config.StateScheme, chainDb)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
all: activate pbss as experimental feature (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-10 22:21:36 +03:00
|
|
|
// Try to recover offline state pruning only in hash-based.
|
2023-10-11 11:27:44 +03:00
|
|
|
if scheme == rawdb.HashScheme {
|
all: activate pbss as experimental feature (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-10 22:21:36 +03:00
|
|
|
if err := pruner.RecoverPruning(stack.ResolvePath(""), chainDb); err != nil {
|
|
|
|
log.Error("Failed to recover state", "error", err)
|
|
|
|
}
|
all: bloom-filter based pruning mechanism (#21724)
* cmd, core, tests: initial state pruner
core: fix db inspector
cmd/geth: add verify-state
cmd/geth: add verification tool
core/rawdb: implement flatdb
cmd, core: fix rebase
core/state: use new contract code layout
core/state/pruner: avoid deleting genesis state
cmd/geth: add helper function
core, cmd: fix extract genesis
core: minor fixes
contracts: remove useless
core/state/snapshot: plugin stacktrie
core: polish
core/state/snapshot: iterate storage concurrently
core/state/snapshot: fix iteration
core: add comments
core/state/snapshot: polish code
core/state: polish
core/state/snapshot: rebase
core/rawdb: add comments
core/rawdb: fix tests
core/rawdb: improve tests
core/state/snapshot: fix concurrent iteration
core/state: run pruning during the recovery
core, trie: implement martin's idea
core, eth: delete flatdb and polish pruner
trie: fix import
core/state/pruner: add log
core/state/pruner: fix issues
core/state/pruner: don't read back
core/state/pruner: fix contract code write
core/state/pruner: check root node presence
cmd, core: polish log
core/state: use HEAD-127 as the target
core/state/snapshot: improve tests
cmd/geth: fix verification tool
cmd/geth: use HEAD as the verification default target
all: replace the bloomfilter with martin's fork
cmd, core: polish code
core, cmd: forcibly delete state root
core/state/pruner: add hash64
core/state/pruner: fix blacklist
core/state: remove blacklist
cmd, core: delete trie clean cache before pruning
cmd, core: fix lint
cmd, core: fix rebase
core/state: fix the special case for clique networks
core/state/snapshot: remove useless code
core/state/pruner: capping the snapshot after pruning
cmd, core, eth: fixes
core/rawdb: update db inspector
cmd/geth: polish code
core/state/pruner: fsync bloom filter
cmd, core: print warning log
core/state/pruner: adjust the parameters for bloom filter
cmd, core: create the bloom filter by size
core: polish
core/state/pruner: sanitize invalid bloomfilter size
cmd: address comments
cmd/geth: address comments
cmd/geth: address comment
core/state/pruner: address comments
core/state/pruner: rename homedir to datadir
cmd, core: address comments
core/state/pruner: address comment
core/state: address comments
core, cmd, tests: address comments
core: address comments
core/state/pruner: release the iterator after each commit
core/state/pruner: improve pruner
cmd, core: adjust bloom paramters
core/state/pruner: fix lint
core/state/pruner: fix tests
core: fix rebase
core/state/pruner: remove atomic rename
core/state/pruner: address comments
all: run go mod tidy
core/state/pruner: avoid false-positive for the middle state roots
core/state/pruner: add checks for middle roots
cmd/geth: replace crit with error
* core/state/pruner: fix lint
* core: drop legacy bloom filter
* core/state/snapshot: improve pruner
* core/state/snapshot: polish concurrent logs to report ETA vs. hashes
* core/state/pruner: add progress report for pruning and compaction too
* core: fix snapshot test API
* core/state: fix some pruning logs
* core/state/pruner: support recovering from bloom flush fail
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-02-08 14:16:30 +03:00
|
|
|
}
|
2022-08-30 19:22:28 +03:00
|
|
|
// Transfer mining-related config to the ethash config.
|
2023-05-03 12:58:39 +03:00
|
|
|
chainConfig, err := core.LoadChainConfig(chainDb, config.Genesis)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
engine, err := ethconfig.CreateConsensusEngine(chainConfig, chainDb)
|
2022-08-30 19:22:28 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-11-07 16:21:46 +03:00
|
|
|
networkID := config.NetworkId
|
|
|
|
if networkID == 0 {
|
|
|
|
networkID = chainConfig.ChainID.Uint64()
|
|
|
|
}
|
2016-06-30 13:03:26 +03:00
|
|
|
eth := &Ethereum{
|
2020-03-27 16:03:20 +03:00
|
|
|
config: config,
|
|
|
|
chainDb: chainDb,
|
2020-08-03 20:40:46 +03:00
|
|
|
eventMux: stack.EventMux(),
|
|
|
|
accountManager: stack.AccountManager(),
|
2022-08-30 19:22:28 +03:00
|
|
|
engine: engine,
|
2020-03-27 16:03:20 +03:00
|
|
|
closeBloomHandler: make(chan struct{}),
|
2023-11-07 16:21:46 +03:00
|
|
|
networkID: networkID,
|
2020-03-27 16:03:20 +03:00
|
|
|
gasPrice: config.Miner.GasPrice,
|
|
|
|
bloomRequests: make(chan chan *bloombits.Retrieval),
|
2021-02-05 15:51:15 +03:00
|
|
|
bloomIndexer: core.NewBloomIndexer(chainDb, params.BloomBitsBlocks, params.BloomConfirms),
|
2020-08-03 20:40:46 +03:00
|
|
|
p2pServer: stack.Server(),
|
2024-08-15 23:14:42 +03:00
|
|
|
discmix: enode.NewFairMix(0),
|
2021-12-17 17:18:51 +03:00
|
|
|
shutdownTracker: shutdowncheck.NewShutdownTracker(chainDb),
|
2015-07-10 15:29:40 +03:00
|
|
|
}
|
2019-02-21 16:14:35 +03:00
|
|
|
bcVersion := rawdb.ReadDatabaseVersion(chainDb)
|
|
|
|
var dbVer = "<nil>"
|
|
|
|
if bcVersion != nil {
|
|
|
|
dbVer = fmt.Sprintf("%d", *bcVersion)
|
|
|
|
}
|
2023-11-07 16:21:46 +03:00
|
|
|
log.Info("Initialising Ethereum protocol", "network", networkID, "dbversion", dbVer)
|
2015-12-16 05:26:23 +02:00
|
|
|
|
2015-04-13 11:13:52 +03:00
|
|
|
if !config.SkipBcVersionCheck {
|
2019-01-11 14:49:12 +03:00
|
|
|
if bcVersion != nil && *bcVersion > core.BlockChainVersion {
|
|
|
|
return nil, fmt.Errorf("database version is v%d, Geth %s only supports v%d", *bcVersion, params.VersionWithMeta, core.BlockChainVersion)
|
2019-02-21 16:14:35 +03:00
|
|
|
} else if bcVersion == nil || *bcVersion < core.BlockChainVersion {
|
2021-05-03 15:42:43 +03:00
|
|
|
if bcVersion != nil { // only print warning on upgrade, not on init
|
|
|
|
log.Warn("Upgrade blockchain database version", "from", dbVer, "to", core.BlockChainVersion)
|
|
|
|
}
|
2019-02-21 16:14:35 +03:00
|
|
|
rawdb.WriteDatabaseVersion(chainDb, core.BlockChainVersion)
|
2015-04-13 11:13:52 +03:00
|
|
|
}
|
|
|
|
}
|
2018-02-05 19:40:32 +03:00
|
|
|
var (
|
2018-09-20 10:44:35 +03:00
|
|
|
vmConfig = vm.Config{
|
|
|
|
EnablePreimageRecording: config.EnablePreimageRecording,
|
2024-06-11 11:10:07 +03:00
|
|
|
EnableWitnessCollection: config.EnableWitnessCollection,
|
2018-09-20 10:44:35 +03:00
|
|
|
}
|
2019-04-01 11:52:11 +03:00
|
|
|
cacheConfig = &core.CacheConfig{
|
|
|
|
TrieCleanLimit: config.TrieCleanCache,
|
|
|
|
TrieCleanNoPrefetch: config.NoPrefetch,
|
|
|
|
TrieDirtyLimit: config.TrieDirtyCache,
|
|
|
|
TrieDirtyDisabled: config.NoPruning,
|
|
|
|
TrieTimeLimit: config.TrieTimeout,
|
2019-11-26 10:48:29 +03:00
|
|
|
SnapshotLimit: config.SnapshotCache,
|
2020-11-18 12:51:33 +03:00
|
|
|
Preimages: config.Preimages,
|
all: activate pbss as experimental feature (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-10 22:21:36 +03:00
|
|
|
StateHistory: config.StateHistory,
|
2023-10-11 11:27:44 +03:00
|
|
|
StateScheme: scheme,
|
2019-04-01 11:52:11 +03:00
|
|
|
}
|
2018-02-05 19:40:32 +03:00
|
|
|
)
|
2024-03-22 20:53:53 +03:00
|
|
|
if config.VMTrace != "" {
|
|
|
|
var traceConfig json.RawMessage
|
2024-04-24 08:53:16 +03:00
|
|
|
if config.VMTraceJsonConfig != "" {
|
|
|
|
traceConfig = json.RawMessage(config.VMTraceJsonConfig)
|
2024-03-22 20:53:53 +03:00
|
|
|
}
|
|
|
|
t, err := tracers.LiveDirectory.New(config.VMTrace, traceConfig)
|
|
|
|
if err != nil {
|
2024-05-28 14:44:40 +03:00
|
|
|
return nil, fmt.Errorf("failed to create tracer %s: %v", config.VMTrace, err)
|
2024-03-22 20:53:53 +03:00
|
|
|
}
|
|
|
|
vmConfig.Tracer = t
|
|
|
|
}
|
2022-08-30 19:22:28 +03:00
|
|
|
// Override the chain config with provided settings.
|
|
|
|
var overrides core.ChainOverrides
|
2023-04-26 18:17:37 +03:00
|
|
|
if config.OverrideCancun != nil {
|
|
|
|
overrides.OverrideCancun = config.OverrideCancun
|
2022-08-30 19:22:28 +03:00
|
|
|
}
|
2023-06-28 12:08:48 +03:00
|
|
|
if config.OverrideVerkle != nil {
|
|
|
|
overrides.OverrideVerkle = config.OverrideVerkle
|
|
|
|
}
|
2024-03-06 15:45:03 +03:00
|
|
|
// TODO (MariusVanDerWijden) get rid of shouldPreserve in a follow-up PR
|
|
|
|
shouldPreserve := func(header *types.Header) bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
eth.blockchain, err = core.NewBlockChain(chainDb, cacheConfig, config.Genesis, &overrides, eth.engine, vmConfig, shouldPreserve, &config.TransactionHistory)
|
2015-06-08 13:12:13 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-10-24 16:19:09 +03:00
|
|
|
eth.bloomIndexer.Start(eth.blockchain)
|
2017-03-02 16:03:33 +03:00
|
|
|
|
2023-07-27 13:45:35 +03:00
|
|
|
if config.BlobPool.Datadir != "" {
|
|
|
|
config.BlobPool.Datadir = stack.ResolvePath(config.BlobPool.Datadir)
|
|
|
|
}
|
|
|
|
blobPool := blobpool.New(config.BlobPool, eth.blockchain)
|
|
|
|
|
2017-07-28 16:09:39 +03:00
|
|
|
if config.TxPool.Journal != "" {
|
2020-08-03 20:40:46 +03:00
|
|
|
config.TxPool.Journal = stack.ResolvePath(config.TxPool.Journal)
|
2017-07-28 16:09:39 +03:00
|
|
|
}
|
2023-06-16 15:29:40 +03:00
|
|
|
legacyPool := legacypool.New(config.TxPool, eth.blockchain)
|
2015-06-15 12:33:08 +03:00
|
|
|
|
2024-02-13 12:10:11 +03:00
|
|
|
eth.txPool, err = txpool.New(config.TxPool.PriceLimit, eth.blockchain, []txpool.SubPool{legacyPool, blobPool})
|
2023-06-16 15:29:40 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-05-13 15:28:01 +03:00
|
|
|
// Permit the downloader to use the trie cache allowance during fast sync
|
2019-11-26 10:48:29 +03:00
|
|
|
cacheLimit := cacheConfig.TrieCleanLimit + cacheConfig.TrieDirtyLimit + cacheConfig.SnapshotLimit
|
2020-12-14 12:27:15 +03:00
|
|
|
if eth.handler, err = newHandler(&handlerConfig{
|
2024-03-02 23:39:22 +03:00
|
|
|
NodeID: eth.p2pServer.Self().ID(),
|
2022-05-04 19:55:17 +03:00
|
|
|
Database: chainDb,
|
|
|
|
Chain: eth.blockchain,
|
|
|
|
TxPool: eth.txPool,
|
2023-11-07 16:21:46 +03:00
|
|
|
Network: networkID,
|
2022-05-04 19:55:17 +03:00
|
|
|
Sync: config.SyncMode,
|
|
|
|
BloomCache: uint64(cacheLimit),
|
|
|
|
EventMux: eth.eventMux,
|
|
|
|
RequiredBlocks: config.RequiredBlocks,
|
2020-12-14 12:27:15 +03:00
|
|
|
}); err != nil {
|
2015-09-01 17:35:14 +03:00
|
|
|
return nil, err
|
|
|
|
}
|
2021-04-16 22:29:22 +03:00
|
|
|
|
2024-03-06 15:45:03 +03:00
|
|
|
eth.miner = miner.New(eth, config.Miner, eth.engine)
|
2019-04-23 10:08:51 +03:00
|
|
|
eth.miner.SetExtra(makeExtraData(config.Miner.ExtraData))
|
2015-07-25 18:33:56 +03:00
|
|
|
|
2021-02-23 15:09:19 +03:00
|
|
|
eth.APIBackend = &EthAPIBackend{stack.Config().ExtRPCEnabled(), stack.Config().AllowUnprotectedTxs, eth, nil}
|
|
|
|
if eth.APIBackend.allowUnprotectedTxs {
|
|
|
|
log.Info("Unprotected transactions allowed")
|
|
|
|
}
|
2024-07-22 10:58:53 +03:00
|
|
|
eth.APIBackend.gpo = gasprice.NewOracle(eth.APIBackend, config.GPO, config.Miner.GasPrice)
|
2015-12-16 05:26:23 +02:00
|
|
|
|
2020-08-03 20:40:46 +03:00
|
|
|
// Start the RPC service
|
2023-11-07 16:21:46 +03:00
|
|
|
eth.netRPCService = ethapi.NewNetAPI(eth.p2pServer, networkID)
|
2020-08-03 20:40:46 +03:00
|
|
|
|
|
|
|
// Register the backend on the node
|
|
|
|
stack.RegisterAPIs(eth.APIs())
|
|
|
|
stack.RegisterProtocols(eth.Protocols())
|
|
|
|
stack.RegisterLifecycle(eth)
|
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 14:23:02 +03:00
|
|
|
|
2021-12-17 17:18:51 +03:00
|
|
|
// Successful startup; push a marker and check previous unclean shutdowns.
|
|
|
|
eth.shutdownTracker.MarkStartup()
|
|
|
|
|
2014-12-14 20:03:24 +02:00
|
|
|
return eth, nil
|
|
|
|
}
|
|
|
|
|
2017-04-12 17:27:23 +03:00
|
|
|
func makeExtraData(extra []byte) []byte {
|
|
|
|
if len(extra) == 0 {
|
|
|
|
// create default extradata
|
|
|
|
extra, _ = rlp.EncodeToBytes([]interface{}{
|
|
|
|
uint(params.VersionMajor<<16 | params.VersionMinor<<8 | params.VersionPatch),
|
|
|
|
"geth",
|
|
|
|
runtime.Version(),
|
|
|
|
runtime.GOOS,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
if uint64(len(extra)) > params.MaximumExtraDataSize {
|
|
|
|
log.Warn("Miner extra data exceed limit", "extra", hexutil.Bytes(extra), "limit", params.MaximumExtraDataSize)
|
|
|
|
extra = nil
|
|
|
|
}
|
|
|
|
return extra
|
|
|
|
}
|
|
|
|
|
2018-05-24 15:55:20 +03:00
|
|
|
// APIs return the collection of RPC services the ethereum package offers.
|
2015-10-15 17:07:19 +03:00
|
|
|
// NOTE, some of these services probably need to be moved to somewhere else.
|
2016-06-30 13:03:26 +03:00
|
|
|
func (s *Ethereum) APIs() []rpc.API {
|
2018-05-09 10:59:00 +03:00
|
|
|
apis := ethapi.GetAPIs(s.APIBackend)
|
2017-04-05 01:16:29 +03:00
|
|
|
|
|
|
|
// Append any APIs exposed explicitly by the consensus engine
|
|
|
|
apis = append(apis, s.engine.APIs(s.BlockChain())...)
|
|
|
|
|
|
|
|
// Append all the local APIs and return
|
|
|
|
return append(apis, []rpc.API{
|
2015-10-15 17:07:19 +03:00
|
|
|
{
|
2022-06-21 12:05:43 +03:00
|
|
|
Namespace: "miner",
|
|
|
|
Service: NewMinerAPI(s),
|
2015-10-15 17:07:19 +03:00
|
|
|
}, {
|
|
|
|
Namespace: "eth",
|
2024-01-22 23:05:18 +03:00
|
|
|
Service: downloader.NewDownloaderAPI(s.handler.downloader, s.blockchain, s.eventMux),
|
2015-12-04 20:56:11 +02:00
|
|
|
}, {
|
|
|
|
Namespace: "admin",
|
2022-06-21 12:05:43 +03:00
|
|
|
Service: NewAdminAPI(s),
|
2015-12-04 20:56:11 +02:00
|
|
|
}, {
|
|
|
|
Namespace: "debug",
|
2022-06-21 12:05:43 +03:00
|
|
|
Service: NewDebugAPI(s),
|
2015-12-16 11:58:01 +02:00
|
|
|
}, {
|
|
|
|
Namespace: "net",
|
|
|
|
Service: s.netRPCService,
|
2015-10-15 17:07:19 +03:00
|
|
|
},
|
2015-12-16 05:26:23 +02:00
|
|
|
}...)
|
2015-10-15 17:07:19 +03:00
|
|
|
}
|
|
|
|
|
2016-06-30 13:03:26 +03:00
|
|
|
func (s *Ethereum) ResetWithGenesisBlock(gb *types.Block) {
|
2015-08-31 18:09:50 +03:00
|
|
|
s.blockchain.ResetWithGenesisBlock(gb)
|
2015-03-13 19:34:43 +02:00
|
|
|
}
|
|
|
|
|
2016-06-30 13:03:26 +03:00
|
|
|
func (s *Ethereum) Miner() *miner.Miner { return s.miner }
|
|
|
|
|
|
|
|
func (s *Ethereum) AccountManager() *accounts.Manager { return s.accountManager }
|
|
|
|
func (s *Ethereum) BlockChain() *core.BlockChain { return s.blockchain }
|
2022-10-24 16:13:55 +03:00
|
|
|
func (s *Ethereum) TxPool() *txpool.TxPool { return s.txPool }
|
2016-06-30 13:03:26 +03:00
|
|
|
func (s *Ethereum) EventMux() *event.TypeMux { return s.eventMux }
|
2017-04-05 01:16:29 +03:00
|
|
|
func (s *Ethereum) Engine() consensus.Engine { return s.engine }
|
2016-06-30 13:03:26 +03:00
|
|
|
func (s *Ethereum) ChainDb() ethdb.Database { return s.chainDb }
|
|
|
|
func (s *Ethereum) IsListening() bool { return true } // Always listening
|
2020-12-14 12:27:15 +03:00
|
|
|
func (s *Ethereum) Downloader() *downloader.Downloader { return s.handler.downloader }
|
core, accounts, eth, trie: handle genesis state missing (#28171)
* core, accounts, eth, trie: handle genesis state missing
* core, eth, trie: polish
* core: manage txpool subscription in mainpool
* eth/backend: fix test
* cmd, eth: fix test
* core/rawdb, trie/triedb/pathdb: address comments
* eth, trie: address comments
* eth: inline the function
* eth: use synced flag
* core/txpool: revert changes in txpool
* core, eth, trie: rename functions
2023-09-28 10:00:53 +03:00
|
|
|
func (s *Ethereum) Synced() bool { return s.handler.synced.Load() }
|
all: activate pbss as experimental feature (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-10 22:21:36 +03:00
|
|
|
func (s *Ethereum) SetSynced() { s.handler.enableSyncedFeatures() }
|
2019-06-11 10:40:32 +03:00
|
|
|
func (s *Ethereum) ArchiveMode() bool { return s.config.NoPruning }
|
2020-08-03 20:40:46 +03:00
|
|
|
func (s *Ethereum) BloomIndexer() *core.ChainIndexer { return s.bloomIndexer }
|
2015-01-28 19:14:28 +02:00
|
|
|
|
2020-08-03 20:40:46 +03:00
|
|
|
// Protocols returns all the currently configured
|
2015-11-17 18:33:25 +02:00
|
|
|
// network protocols to start.
|
2016-06-30 13:03:26 +03:00
|
|
|
func (s *Ethereum) Protocols() []p2p.Protocol {
|
2024-08-15 23:14:42 +03:00
|
|
|
protos := eth.MakeProtocols((*ethHandler)(s.handler), s.networkID, s.discmix)
|
2020-12-14 12:27:15 +03:00
|
|
|
if s.config.SnapshotCache > 0 {
|
2024-08-15 23:14:42 +03:00
|
|
|
protos = append(protos, snap.MakeProtocols((*snapHandler)(s.handler))...)
|
2019-07-08 18:53:47 +03:00
|
|
|
}
|
|
|
|
return protos
|
2015-11-17 18:33:25 +02:00
|
|
|
}
|
2015-04-22 13:46:41 +03:00
|
|
|
|
2020-08-03 20:40:46 +03:00
|
|
|
// Start implements node.Lifecycle, starting all internal goroutines needed by the
|
2016-06-30 13:03:26 +03:00
|
|
|
// Ethereum protocol implementation.
|
2020-08-03 20:40:46 +03:00
|
|
|
func (s *Ethereum) Start() error {
|
2024-08-15 23:14:42 +03:00
|
|
|
s.setupDiscovery()
|
2019-07-08 18:53:47 +03:00
|
|
|
|
2017-08-29 14:13:11 +03:00
|
|
|
// Start the bloom bits servicing goroutines
|
2018-08-28 10:08:16 +03:00
|
|
|
s.startBloomHandlers(params.BloomBitsBlocks)
|
2017-08-29 14:13:11 +03:00
|
|
|
|
2021-12-17 17:18:51 +03:00
|
|
|
// Regularly update shutdown marker
|
|
|
|
s.shutdownTracker.Start()
|
|
|
|
|
2024-08-15 12:42:39 +03:00
|
|
|
// Start the networking layer
|
|
|
|
s.handler.Start(s.p2pServer.MaxPeers)
|
2014-12-14 20:03:24 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-08-15 23:14:42 +03:00
|
|
|
func (s *Ethereum) setupDiscovery() error {
|
|
|
|
eth.StartENRUpdater(s.blockchain, s.p2pServer.LocalNode())
|
|
|
|
|
|
|
|
// Add eth nodes from DNS.
|
|
|
|
dnsclient := dnsdisc.NewClient(dnsdisc.Config{})
|
|
|
|
if len(s.config.EthDiscoveryURLs) > 0 {
|
|
|
|
iter, err := dnsclient.NewIterator(s.config.EthDiscoveryURLs...)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
s.discmix.AddSource(iter)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add snap nodes from DNS.
|
|
|
|
if len(s.config.SnapDiscoveryURLs) > 0 {
|
|
|
|
iter, err := dnsclient.NewIterator(s.config.SnapDiscoveryURLs...)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
s.discmix.AddSource(iter)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add DHT nodes from discv5.
|
|
|
|
if s.p2pServer.DiscoveryV5() != nil {
|
|
|
|
filter := eth.NewNodeFilter(s.blockchain)
|
|
|
|
iter := enode.Filter(s.p2pServer.DiscoveryV5().RandomNodes(), filter)
|
|
|
|
s.discmix.AddSource(iter)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-08-03 20:40:46 +03:00
|
|
|
// Stop implements node.Lifecycle, terminating all internal goroutines used by the
|
2015-11-17 18:33:25 +02:00
|
|
|
// Ethereum protocol.
|
2016-06-30 13:03:26 +03:00
|
|
|
func (s *Ethereum) Stop() error {
|
2020-03-27 16:03:20 +03:00
|
|
|
// Stop all the peer-related stuff first.
|
2024-08-15 23:14:42 +03:00
|
|
|
s.discmix.Close()
|
2020-12-14 12:27:15 +03:00
|
|
|
s.handler.Stop()
|
2020-03-27 16:03:20 +03:00
|
|
|
|
|
|
|
// Then stop everything else.
|
|
|
|
s.bloomIndexer.Close()
|
|
|
|
close(s.closeBloomHandler)
|
2023-06-16 15:29:40 +03:00
|
|
|
s.txPool.Close()
|
2020-03-27 16:03:20 +03:00
|
|
|
s.blockchain.Stop()
|
|
|
|
s.engine.Close()
|
2021-12-17 17:18:51 +03:00
|
|
|
|
|
|
|
// Clean shutdown marker as the last thing before closing db
|
|
|
|
s.shutdownTracker.Stop()
|
|
|
|
|
2015-09-14 10:45:40 +03:00
|
|
|
s.chainDb.Close()
|
2020-03-27 16:03:20 +03:00
|
|
|
s.eventMux.Stop()
|
2020-12-14 12:27:15 +03:00
|
|
|
|
2015-11-17 18:33:25 +02:00
|
|
|
return nil
|
2014-12-14 20:03:24 +02:00
|
|
|
}
|
2024-03-05 17:13:28 +03:00
|
|
|
|
|
|
|
// SyncMode retrieves the current sync mode, either explicitly set, or derived
|
|
|
|
// from the chain status.
|
|
|
|
func (s *Ethereum) SyncMode() downloader.SyncMode {
|
|
|
|
// If we're in snap sync mode, return that directly
|
|
|
|
if s.handler.snapSync.Load() {
|
|
|
|
return downloader.SnapSync
|
|
|
|
}
|
|
|
|
// We are probably in full sync, but we might have rewound to before the
|
|
|
|
// snap sync pivot, check if we should re-enable snap sync.
|
|
|
|
head := s.blockchain.CurrentBlock()
|
|
|
|
if pivot := rawdb.ReadLastPivotNumber(s.chainDb); pivot != nil {
|
|
|
|
if head.Number.Uint64() < *pivot {
|
|
|
|
return downloader.SnapSync
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// We are in a full sync, but the associated head state is missing. To complete
|
|
|
|
// the head state, forcefully rerun the snap sync. Note it doesn't mean the
|
|
|
|
// persistent state is corrupted, just mismatch with the head block.
|
|
|
|
if !s.blockchain.HasState(head.Root) {
|
|
|
|
log.Info("Reenabled snap sync as chain is stateless")
|
|
|
|
return downloader.SnapSync
|
|
|
|
}
|
|
|
|
// Nope, we're really full syncing
|
|
|
|
return downloader.FullSync
|
|
|
|
}
|