From 20f518ed43612cc6133b727e7706ddcea3bacec9 Mon Sep 17 00:00:00 2001 From: Simon Dudley Date: Wed, 25 Oct 2023 12:53:41 +1000 Subject: [PATCH] Decouple TrieLogManager and CachedWorldStorageManager (#6072) Separate out the concepts of world state caching and trie log management. Remove AbstractTrieLogManager and make TrieLogManager the concrete implementation. Signed-off-by: Simon Dudley --- .../bonsai/BonsaiWorldStateProvider.java | 36 ++-- .../cache/CachedWorldStorageManager.java | 135 +++---------- .../trielog/AbstractTrieLogManager.java | 164 ---------------- .../bonsai/trielog/TrieLogManager.java | 178 ++++++++++++++++-- .../bonsai/worldview/BonsaiWorldState.java | 12 +- .../bonsai/BonsaiSnapshotIsolationTests.java | 7 +- .../bonsai/BonsaiWorldStateArchiveTest.java | 12 +- .../bonsai/trielog/TrieLogManagerTests.java | 13 +- .../BonsaiReferenceTestWorldState.java | 70 ++++--- 9 files changed, 283 insertions(+), 344 deletions(-) delete mode 100644 ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/trielog/AbstractTrieLogManager.java diff --git a/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/BonsaiWorldStateProvider.java b/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/BonsaiWorldStateProvider.java index 299b826369c..bcb7f9f65fe 100644 --- a/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/BonsaiWorldStateProvider.java +++ b/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/BonsaiWorldStateProvider.java @@ -62,6 +62,7 @@ public class BonsaiWorldStateProvider implements WorldStateArchive { private final Blockchain blockchain; + private final CachedWorldStorageManager cachedWorldStorageManager; private final TrieLogManager trieLogManager; private final BonsaiWorldState persistedState; private final BonsaiWorldStateKeyValueStorage worldStateStorage; @@ -91,15 +92,12 @@ public BonsaiWorldStateProvider( final ObservableMetricsSystem metricsSystem, final BesuContext pluginContext) { + this.cachedWorldStorageManager = + new CachedWorldStorageManager(this, worldStateStorage, metricsSystem); // TODO: de-dup constructors this.trieLogManager = - new CachedWorldStorageManager( - this, - blockchain, - worldStateStorage, - metricsSystem, - maxLayersToLoad.orElse(RETAINED_LAYERS), - pluginContext); + new TrieLogManager( + blockchain, worldStateStorage, maxLayersToLoad.orElse(RETAINED_LAYERS), pluginContext); this.blockchain = blockchain; this.worldStateStorage = worldStateStorage; this.cachedMerkleTrieLoader = cachedMerkleTrieLoader; @@ -108,16 +106,18 @@ public BonsaiWorldStateProvider( .getBlockHeader(persistedState.getWorldStateBlockHash()) .ifPresent( blockHeader -> - this.trieLogManager.addCachedLayer( + this.cachedWorldStorageManager.addCachedLayer( blockHeader, persistedState.getWorldStateRootHash(), persistedState)); } @VisibleForTesting BonsaiWorldStateProvider( + final CachedWorldStorageManager cachedWorldStorageManager, final TrieLogManager trieLogManager, final BonsaiWorldStateKeyValueStorage worldStateStorage, final Blockchain blockchain, final CachedMerkleTrieLoader cachedMerkleTrieLoader) { + this.cachedWorldStorageManager = cachedWorldStorageManager; this.trieLogManager = trieLogManager; this.blockchain = blockchain; this.worldStateStorage = worldStateStorage; @@ -127,13 +127,13 @@ public BonsaiWorldStateProvider( .getBlockHeader(persistedState.getWorldStateBlockHash()) .ifPresent( blockHeader -> - this.trieLogManager.addCachedLayer( + this.cachedWorldStorageManager.addCachedLayer( blockHeader, persistedState.getWorldStateRootHash(), persistedState)); } @Override public Optional get(final Hash rootHash, final Hash blockHash) { - return trieLogManager + return cachedWorldStorageManager .getWorldState(blockHash) .or( () -> { @@ -148,7 +148,7 @@ public Optional get(final Hash rootHash, final Hash blockHash) { @Override public boolean isWorldStateAvailable(final Hash rootHash, final Hash blockHash) { - return trieLogManager.containWorldStateStorage(blockHash) + return cachedWorldStorageManager.containWorldStateStorage(blockHash) || persistedState.blockHash().equals(blockHash) || worldStateStorage.isWorldStateAvailable(rootHash, blockHash); } @@ -167,10 +167,10 @@ public Optional getMutable( trieLogManager.getMaxLayersToLoad()); return Optional.empty(); } - return trieLogManager + return cachedWorldStorageManager .getWorldState(blockHeader.getHash()) - .or(() -> trieLogManager.getNearestWorldState(blockHeader)) - .or(() -> trieLogManager.getHeadWorldState(blockchain::getBlockHeader)) + .or(() -> cachedWorldStorageManager.getNearestWorldState(blockHeader)) + .or(() -> cachedWorldStorageManager.getHeadWorldState(blockchain::getBlockHeader)) .flatMap( bonsaiWorldState -> rollMutableStateToBlockHash(bonsaiWorldState, blockHeader.getHash())) @@ -354,11 +354,15 @@ public TrieLogManager getTrieLogManager() { return trieLogManager; } + public CachedWorldStorageManager getCachedWorldStorageManager() { + return cachedWorldStorageManager; + } + @Override public void resetArchiveStateTo(final BlockHeader blockHeader) { persistedState.resetWorldStateTo(blockHeader); - this.trieLogManager.reset(); - this.trieLogManager.addCachedLayer( + this.cachedWorldStorageManager.reset(); + this.cachedWorldStorageManager.addCachedLayer( blockHeader, persistedState.getWorldStateRootHash(), persistedState); } diff --git a/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/cache/CachedWorldStorageManager.java b/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/cache/CachedWorldStorageManager.java index d0f1be97d89..f2434bbf912 100644 --- a/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/cache/CachedWorldStorageManager.java +++ b/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/cache/CachedWorldStorageManager.java @@ -20,17 +20,9 @@ import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiWorldStateKeyValueStorage; import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiWorldStateKeyValueStorage.BonsaiStorageSubscriber; import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiWorldStateLayerStorage; -import org.hyperledger.besu.ethereum.bonsai.trielog.AbstractTrieLogManager; -import org.hyperledger.besu.ethereum.bonsai.trielog.TrieLogFactoryImpl; import org.hyperledger.besu.ethereum.bonsai.worldview.BonsaiWorldState; -import org.hyperledger.besu.ethereum.chain.Blockchain; import org.hyperledger.besu.ethereum.core.BlockHeader; import org.hyperledger.besu.metrics.ObservableMetricsSystem; -import org.hyperledger.besu.plugin.BesuContext; -import org.hyperledger.besu.plugin.services.TrieLogService; -import org.hyperledger.besu.plugin.services.trielogs.TrieLog; -import org.hyperledger.besu.plugin.services.trielogs.TrieLogFactory; -import org.hyperledger.besu.plugin.services.trielogs.TrieLogProvider; import java.util.ArrayList; import java.util.Comparator; @@ -39,52 +31,39 @@ import java.util.Optional; import java.util.concurrent.ConcurrentHashMap; import java.util.function.Function; -import java.util.stream.LongStream; -import java.util.stream.Stream; -import com.google.common.annotations.VisibleForTesting; import org.apache.tuweni.bytes.Bytes32; import org.slf4j.Logger; import org.slf4j.LoggerFactory; -public class CachedWorldStorageManager extends AbstractTrieLogManager - implements BonsaiStorageSubscriber { +public class CachedWorldStorageManager implements BonsaiStorageSubscriber { + public static final long RETAINED_LAYERS = 512; // at least 256 + typical rollbacks private static final Logger LOG = LoggerFactory.getLogger(CachedWorldStorageManager.class); private final BonsaiWorldStateProvider archive; private final ObservableMetricsSystem metricsSystem; - CachedWorldStorageManager( + private final BonsaiWorldStateKeyValueStorage rootWorldStateStorage; + private final Map cachedWorldStatesByHash; + + private CachedWorldStorageManager( final BonsaiWorldStateProvider archive, - final Blockchain blockchain, final BonsaiWorldStateKeyValueStorage worldStateStorage, - final long maxLayersToLoad, final Map cachedWorldStatesByHash, - final BesuContext pluginContext, final ObservableMetricsSystem metricsSystem) { - super(blockchain, worldStateStorage, maxLayersToLoad, cachedWorldStatesByHash, pluginContext); worldStateStorage.subscribe(this); + this.rootWorldStateStorage = worldStateStorage; + this.cachedWorldStatesByHash = cachedWorldStatesByHash; this.archive = archive; this.metricsSystem = metricsSystem; } public CachedWorldStorageManager( final BonsaiWorldStateProvider archive, - final Blockchain blockchain, final BonsaiWorldStateKeyValueStorage worldStateStorage, - final ObservableMetricsSystem metricsSystem, - final long maxLayersToLoad, - final BesuContext pluginContext) { - this( - archive, - blockchain, - worldStateStorage, - maxLayersToLoad, - new ConcurrentHashMap<>(), - pluginContext, - metricsSystem); + final ObservableMetricsSystem metricsSystem) { + this(archive, worldStateStorage, new ConcurrentHashMap<>(), metricsSystem); } - @Override public synchronized void addCachedLayer( final BlockHeader blockHeader, final Hash worldStateRootHash, @@ -132,7 +111,20 @@ public synchronized void addCachedLayer( scrubCachedLayers(blockHeader.getNumber()); } - @Override + private synchronized void scrubCachedLayers(final long newMaxHeight) { + if (cachedWorldStatesByHash.size() > RETAINED_LAYERS) { + final long waterline = newMaxHeight - RETAINED_LAYERS; + cachedWorldStatesByHash.values().stream() + .filter(layer -> layer.getBlockNumber() < waterline) + .toList() + .forEach( + layer -> { + cachedWorldStatesByHash.remove(layer.getBlockHash()); + layer.close(); + }); + } + } + public Optional getWorldState(final Hash blockHash) { if (cachedWorldStatesByHash.containsKey(blockHash)) { // return a new worldstate using worldstate storage and an isolated copy of the updater @@ -150,7 +142,6 @@ public Optional getWorldState(final Hash blockHash) { return Optional.empty(); } - @Override public Optional getNearestWorldState(final BlockHeader blockHeader) { LOG.atDebug() .setMessage("getting nearest worldstate for {}") @@ -183,7 +174,6 @@ public Optional getNearestWorldState(final BlockHeader blockHe archive, new BonsaiWorldStateLayerStorage(storage))); } - @Override public Optional getHeadWorldState( final Function> hashBlockHeaderFunction) { @@ -203,7 +193,10 @@ public Optional getHeadWorldState( }); } - @Override + public boolean containWorldStateStorage(final Hash blockHash) { + return cachedWorldStatesByHash.containsKey(blockHash); + } + public void reset() { this.cachedWorldStatesByHash.clear(); } @@ -227,76 +220,4 @@ public void onClearTrieLog() { public void onCloseStorage() { this.cachedWorldStatesByHash.clear(); } - - @VisibleForTesting - @Override - protected TrieLogFactory setupTrieLogFactory(final BesuContext pluginContext) { - // if we have a TrieLogService from pluginContext, use it. - var trieLogServicez = - Optional.ofNullable(pluginContext) - .flatMap(context -> context.getService(TrieLogService.class)); - - if (trieLogServicez.isPresent()) { - var trieLogService = trieLogServicez.get(); - // push the TrieLogProvider into the TrieLogService - trieLogService.configureTrieLogProvider(getTrieLogProvider()); - - // configure plugin observers: - trieLogService.getObservers().forEach(trieLogObservers::subscribe); - - // return the TrieLogFactory implementation from the TrieLogService - return trieLogService.getTrieLogFactory(); - } else { - // Otherwise default to TrieLogFactoryImpl - return new TrieLogFactoryImpl(); - } - } - - @VisibleForTesting - TrieLogProvider getTrieLogProvider() { - return new TrieLogProvider() { - @Override - public Optional getTrieLogLayer(final Hash blockHash) { - return CachedWorldStorageManager.this.getTrieLogLayer(blockHash); - } - - @Override - public Optional getTrieLogLayer(final long blockNumber) { - return CachedWorldStorageManager.this - .blockchain - .getBlockHeader(blockNumber) - .map(BlockHeader::getHash) - .flatMap(CachedWorldStorageManager.this::getTrieLogLayer); - } - - @Override - public List getTrieLogsByRange( - final long fromBlockNumber, final long toBlockNumber) { - return rangeAsStream(fromBlockNumber, toBlockNumber) - .map(blockchain::getBlockHeader) - .map( - headerOpt -> - headerOpt.flatMap( - header -> - CachedWorldStorageManager.this - .getTrieLogLayer(header.getBlockHash()) - .map( - layer -> - new TrieLogRangeTuple( - header.getBlockHash(), header.getNumber(), layer)))) - .filter(Optional::isPresent) - .map(Optional::get) - .toList(); - } - - Stream rangeAsStream(final long fromBlockNumber, final long toBlockNumber) { - if (Math.abs(toBlockNumber - fromBlockNumber) > LOG_RANGE_LIMIT) { - throw new IllegalArgumentException("Requested Range too large"); - } - long left = Math.min(fromBlockNumber, toBlockNumber); - long right = Math.max(fromBlockNumber, toBlockNumber); - return LongStream.range(left, right).boxed(); - } - }; - } } diff --git a/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/trielog/AbstractTrieLogManager.java b/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/trielog/AbstractTrieLogManager.java deleted file mode 100644 index 90aff386e83..00000000000 --- a/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/trielog/AbstractTrieLogManager.java +++ /dev/null @@ -1,164 +0,0 @@ -/* - * Copyright Hyperledger Besu Contributors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with - * the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on - * an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the - * specific language governing permissions and limitations under the License. - * - * SPDX-License-Identifier: Apache-2.0 - * - */ -package org.hyperledger.besu.ethereum.bonsai.trielog; - -import org.hyperledger.besu.datatypes.Hash; -import org.hyperledger.besu.ethereum.bonsai.cache.CachedBonsaiWorldView; -import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiWorldStateKeyValueStorage; -import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiWorldStateKeyValueStorage.BonsaiUpdater; -import org.hyperledger.besu.ethereum.bonsai.worldview.BonsaiWorldState; -import org.hyperledger.besu.ethereum.bonsai.worldview.BonsaiWorldStateUpdateAccumulator; -import org.hyperledger.besu.ethereum.chain.Blockchain; -import org.hyperledger.besu.ethereum.core.BlockHeader; -import org.hyperledger.besu.plugin.BesuContext; -import org.hyperledger.besu.plugin.services.trielogs.TrieLog; -import org.hyperledger.besu.plugin.services.trielogs.TrieLogEvent.TrieLogObserver; -import org.hyperledger.besu.plugin.services.trielogs.TrieLogFactory; -import org.hyperledger.besu.util.Subscribers; - -import java.util.Map; -import java.util.Optional; - -import com.google.common.annotations.VisibleForTesting; -import org.apache.tuweni.bytes.Bytes32; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -public abstract class AbstractTrieLogManager implements TrieLogManager { - private static final Logger LOG = LoggerFactory.getLogger(AbstractTrieLogManager.class); - public static final long RETAINED_LAYERS = 512; // at least 256 + typical rollbacks - public static final long LOG_RANGE_LIMIT = 1000; // restrict trielog range queries to 1k logs - protected final Blockchain blockchain; - protected final BonsaiWorldStateKeyValueStorage rootWorldStateStorage; - - protected final Map cachedWorldStatesByHash; - protected final long maxLayersToLoad; - protected final Subscribers trieLogObservers = Subscribers.create(); - - protected final TrieLogFactory trieLogFactory; - - protected AbstractTrieLogManager( - final Blockchain blockchain, - final BonsaiWorldStateKeyValueStorage worldStateStorage, - final long maxLayersToLoad, - final Map cachedWorldStatesByHash, - final BesuContext pluginContext) { - this.blockchain = blockchain; - this.rootWorldStateStorage = worldStateStorage; - this.cachedWorldStatesByHash = cachedWorldStatesByHash; - this.maxLayersToLoad = maxLayersToLoad; - this.trieLogFactory = setupTrieLogFactory(pluginContext); - } - - protected abstract TrieLogFactory setupTrieLogFactory(final BesuContext pluginContext); - - @Override - public synchronized void saveTrieLog( - final BonsaiWorldStateUpdateAccumulator localUpdater, - final Hash forWorldStateRootHash, - final BlockHeader forBlockHeader, - final BonsaiWorldState forWorldState) { - // do not overwrite a trielog layer that already exists in the database. - // if it's only in memory we need to save it - // for example, in case of reorg we don't replace a trielog layer - if (rootWorldStateStorage.getTrieLog(forBlockHeader.getHash()).isEmpty()) { - final BonsaiUpdater stateUpdater = forWorldState.getWorldStateStorage().updater(); - boolean success = false; - try { - final TrieLog trieLog = prepareTrieLog(forBlockHeader, localUpdater); - persistTrieLog(forBlockHeader, forWorldStateRootHash, trieLog, stateUpdater); - - // notify trie log added observers, synchronously - trieLogObservers.forEach(o -> o.onTrieLogAdded(new TrieLogAddedEvent(trieLog))); - - success = true; - } finally { - if (success) { - stateUpdater.commit(); - } else { - stateUpdater.rollback(); - } - } - } - } - - @VisibleForTesting - TrieLog prepareTrieLog( - final BlockHeader blockHeader, final BonsaiWorldStateUpdateAccumulator localUpdater) { - LOG.atDebug() - .setMessage("Adding layered world state for {}") - .addArgument(blockHeader::toLogString) - .log(); - final TrieLog trieLog = trieLogFactory.create(localUpdater, blockHeader); - trieLog.freeze(); - return trieLog; - } - - public synchronized void scrubCachedLayers(final long newMaxHeight) { - if (cachedWorldStatesByHash.size() > RETAINED_LAYERS) { - final long waterline = newMaxHeight - RETAINED_LAYERS; - cachedWorldStatesByHash.values().stream() - .filter(layer -> layer.getBlockNumber() < waterline) - .toList() - .forEach( - layer -> { - cachedWorldStatesByHash.remove(layer.getBlockHash()); - layer.close(); - }); - } - } - - private void persistTrieLog( - final BlockHeader blockHeader, - final Hash worldStateRootHash, - final TrieLog trieLog, - final BonsaiUpdater stateUpdater) { - LOG.atDebug() - .setMessage("Persisting trie log for block hash {} and world state root {}") - .addArgument(blockHeader::toLogString) - .addArgument(worldStateRootHash::toHexString) - .log(); - - stateUpdater - .getTrieLogStorageTransaction() - .put(blockHeader.getHash().toArrayUnsafe(), trieLogFactory.serialize(trieLog)); - } - - @Override - public boolean containWorldStateStorage(final Hash blockHash) { - return cachedWorldStatesByHash.containsKey(blockHash); - } - - @Override - public long getMaxLayersToLoad() { - return maxLayersToLoad; - } - - @Override - public Optional getTrieLogLayer(final Hash blockHash) { - return rootWorldStateStorage.getTrieLog(blockHash).map(trieLogFactory::deserialize); - } - - @Override - public synchronized long subscribe(final TrieLogObserver sub) { - return trieLogObservers.subscribe(sub); - } - - @Override - public synchronized void unsubscribe(final long id) { - trieLogObservers.unsubscribe(id); - } -} diff --git a/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/trielog/TrieLogManager.java b/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/trielog/TrieLogManager.java index 5da83180a1c..cabd5a2d300 100644 --- a/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/trielog/TrieLogManager.java +++ b/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/trielog/TrieLogManager.java @@ -16,42 +16,188 @@ package org.hyperledger.besu.ethereum.bonsai.trielog; import org.hyperledger.besu.datatypes.Hash; +import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiWorldStateKeyValueStorage; import org.hyperledger.besu.ethereum.bonsai.worldview.BonsaiWorldState; import org.hyperledger.besu.ethereum.bonsai.worldview.BonsaiWorldStateUpdateAccumulator; +import org.hyperledger.besu.ethereum.chain.Blockchain; import org.hyperledger.besu.ethereum.core.BlockHeader; +import org.hyperledger.besu.plugin.BesuContext; +import org.hyperledger.besu.plugin.services.TrieLogService; import org.hyperledger.besu.plugin.services.trielogs.TrieLog; import org.hyperledger.besu.plugin.services.trielogs.TrieLogEvent; +import org.hyperledger.besu.plugin.services.trielogs.TrieLogFactory; +import org.hyperledger.besu.plugin.services.trielogs.TrieLogProvider; +import org.hyperledger.besu.util.Subscribers; +import java.util.List; import java.util.Optional; -import java.util.function.Function; +import java.util.stream.LongStream; +import java.util.stream.Stream; -public interface TrieLogManager { +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; - void saveTrieLog( +public class TrieLogManager { + private static final Logger LOG = LoggerFactory.getLogger(TrieLogManager.class); + public static final long LOG_RANGE_LIMIT = 1000; // restrict trielog range queries to 1k logs + protected final Blockchain blockchain; + protected final BonsaiWorldStateKeyValueStorage rootWorldStateStorage; + + protected final long maxLayersToLoad; + protected final Subscribers trieLogObservers = Subscribers.create(); + + protected final TrieLogFactory trieLogFactory; + + public TrieLogManager( + final Blockchain blockchain, + final BonsaiWorldStateKeyValueStorage worldStateStorage, + final long maxLayersToLoad, + final BesuContext pluginContext) { + this.blockchain = blockchain; + this.rootWorldStateStorage = worldStateStorage; + this.maxLayersToLoad = maxLayersToLoad; + this.trieLogFactory = setupTrieLogFactory(pluginContext); + } + + public synchronized void saveTrieLog( final BonsaiWorldStateUpdateAccumulator localUpdater, final Hash forWorldStateRootHash, final BlockHeader forBlockHeader, - final BonsaiWorldState forWorldState); + final BonsaiWorldState forWorldState) { + // do not overwrite a trielog layer that already exists in the database. + // if it's only in memory we need to save it + // for example, in case of reorg we don't replace a trielog layer + if (rootWorldStateStorage.getTrieLog(forBlockHeader.getHash()).isEmpty()) { + final BonsaiWorldStateKeyValueStorage.BonsaiUpdater stateUpdater = + forWorldState.getWorldStateStorage().updater(); + boolean success = false; + try { + final TrieLog trieLog = prepareTrieLog(forBlockHeader, localUpdater); + persistTrieLog(forBlockHeader, forWorldStateRootHash, trieLog, stateUpdater); + + // notify trie log added observers, synchronously + trieLogObservers.forEach(o -> o.onTrieLogAdded(new TrieLogAddedEvent(trieLog))); + + success = true; + } finally { + if (success) { + stateUpdater.commit(); + } else { + stateUpdater.rollback(); + } + } + } + } + + private TrieLog prepareTrieLog( + final BlockHeader blockHeader, final BonsaiWorldStateUpdateAccumulator localUpdater) { + LOG.atDebug() + .setMessage("Adding layered world state for {}") + .addArgument(blockHeader::toLogString) + .log(); + final TrieLog trieLog = trieLogFactory.create(localUpdater, blockHeader); + trieLog.freeze(); + return trieLog; + } + + private void persistTrieLog( + final BlockHeader blockHeader, + final Hash worldStateRootHash, + final TrieLog trieLog, + final BonsaiWorldStateKeyValueStorage.BonsaiUpdater stateUpdater) { + LOG.atDebug() + .setMessage("Persisting trie log for block hash {} and world state root {}") + .addArgument(blockHeader::toLogString) + .addArgument(worldStateRootHash::toHexString) + .log(); + + stateUpdater + .getTrieLogStorageTransaction() + .put(blockHeader.getHash().toArrayUnsafe(), trieLogFactory.serialize(trieLog)); + } + + public long getMaxLayersToLoad() { + return maxLayersToLoad; + } + + public Optional getTrieLogLayer(final Hash blockHash) { + return rootWorldStateStorage.getTrieLog(blockHash).map(trieLogFactory::deserialize); + } - void addCachedLayer( - BlockHeader blockHeader, Hash worldStateRootHash, BonsaiWorldState forWorldState); + public synchronized long subscribe(final TrieLogEvent.TrieLogObserver sub) { + return trieLogObservers.subscribe(sub); + } - boolean containWorldStateStorage(final Hash blockHash); + public synchronized void unsubscribe(final long id) { + trieLogObservers.unsubscribe(id); + } - Optional getWorldState(final Hash blockHash); + private TrieLogFactory setupTrieLogFactory(final BesuContext pluginContext) { + // if we have a TrieLogService from pluginContext, use it. + var trieLogServicez = + Optional.ofNullable(pluginContext) + .flatMap(context -> context.getService(TrieLogService.class)); - Optional getNearestWorldState(final BlockHeader blockHeader); + if (trieLogServicez.isPresent()) { + var trieLogService = trieLogServicez.get(); + // push the TrieLogProvider into the TrieLogService + trieLogService.configureTrieLogProvider(getTrieLogProvider()); - Optional getHeadWorldState( - final Function> hashBlockHeaderFunction); + // configure plugin observers: + trieLogService.getObservers().forEach(trieLogObservers::subscribe); - long getMaxLayersToLoad(); + // return the TrieLogFactory implementation from the TrieLogService + return trieLogService.getTrieLogFactory(); + } else { + // Otherwise default to TrieLogFactoryImpl + return new TrieLogFactoryImpl(); + } + } - void reset(); + private TrieLogProvider getTrieLogProvider() { + return new TrieLogProvider() { + @Override + public Optional getTrieLogLayer(final Hash blockHash) { + return TrieLogManager.this.getTrieLogLayer(blockHash); + } - Optional getTrieLogLayer(final Hash blockHash); + @Override + public Optional getTrieLogLayer(final long blockNumber) { + return TrieLogManager.this + .blockchain + .getBlockHeader(blockNumber) + .map(BlockHeader::getHash) + .flatMap(TrieLogManager.this::getTrieLogLayer); + } - long subscribe(final TrieLogEvent.TrieLogObserver sub); + @Override + public List getTrieLogsByRange( + final long fromBlockNumber, final long toBlockNumber) { + return rangeAsStream(fromBlockNumber, toBlockNumber) + .map(blockchain::getBlockHeader) + .map( + headerOpt -> + headerOpt.flatMap( + header -> + TrieLogManager.this + .getTrieLogLayer(header.getBlockHash()) + .map( + layer -> + new TrieLogRangeTuple( + header.getBlockHash(), header.getNumber(), layer)))) + .filter(Optional::isPresent) + .map(Optional::get) + .toList(); + } - void unsubscribe(final long id); + Stream rangeAsStream(final long fromBlockNumber, final long toBlockNumber) { + if (Math.abs(toBlockNumber - fromBlockNumber) > LOG_RANGE_LIMIT) { + throw new IllegalArgumentException("Requested Range too large"); + } + long left = Math.min(fromBlockNumber, toBlockNumber); + long right = Math.max(fromBlockNumber, toBlockNumber); + return LongStream.range(left, right).boxed(); + } + }; + } } diff --git a/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/worldview/BonsaiWorldState.java b/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/worldview/BonsaiWorldState.java index ed81c90f439..d4c4c223a24 100644 --- a/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/worldview/BonsaiWorldState.java +++ b/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/bonsai/worldview/BonsaiWorldState.java @@ -28,6 +28,7 @@ import org.hyperledger.besu.ethereum.bonsai.BonsaiValue; import org.hyperledger.besu.ethereum.bonsai.BonsaiWorldStateProvider; import org.hyperledger.besu.ethereum.bonsai.cache.CachedMerkleTrieLoader; +import org.hyperledger.besu.ethereum.bonsai.cache.CachedWorldStorageManager; import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiSnapshotWorldStateKeyValueStorage; import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiWorldStateKeyValueStorage; import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiWorldStateKeyValueStorage.BonsaiStorageSubscriber; @@ -69,6 +70,7 @@ public class BonsaiWorldState protected BonsaiWorldStateKeyValueStorage worldStateStorage; protected final CachedMerkleTrieLoader cachedMerkleTrieLoader; + protected final CachedWorldStorageManager cachedWorldStorageManager; protected final TrieLogManager trieLogManager; private BonsaiWorldStateUpdateAccumulator accumulator; @@ -79,12 +81,17 @@ public class BonsaiWorldState public BonsaiWorldState( final BonsaiWorldStateProvider archive, final BonsaiWorldStateKeyValueStorage worldStateStorage) { - this(worldStateStorage, archive.getCachedMerkleTrieLoader(), archive.getTrieLogManager()); + this( + worldStateStorage, + archive.getCachedMerkleTrieLoader(), + archive.getCachedWorldStorageManager(), + archive.getTrieLogManager()); } protected BonsaiWorldState( final BonsaiWorldStateKeyValueStorage worldStateStorage, final CachedMerkleTrieLoader cachedMerkleTrieLoader, + final CachedWorldStorageManager cachedWorldStorageManager, final TrieLogManager trieLogManager) { this.worldStateStorage = worldStateStorage; this.worldStateRootHash = @@ -101,6 +108,7 @@ protected BonsaiWorldState( (addr, value) -> cachedMerkleTrieLoader.preLoadStorageSlot(getWorldStateStorage(), addr, value)); this.cachedMerkleTrieLoader = cachedMerkleTrieLoader; + this.cachedWorldStorageManager = cachedWorldStorageManager; this.trieLogManager = trieLogManager; } @@ -396,7 +404,7 @@ public void persist(final BlockHeader blockHeader) { trieLogManager.saveTrieLog(localCopy, newWorldStateRootHash, blockHeader, this); // not save a frozen state in the cache if (!isFrozen) { - trieLogManager.addCachedLayer(blockHeader, newWorldStateRootHash, this); + cachedWorldStorageManager.addCachedLayer(blockHeader, newWorldStateRootHash, this); } }; diff --git a/ethereum/core/src/test/java/org/hyperledger/besu/ethereum/bonsai/BonsaiSnapshotIsolationTests.java b/ethereum/core/src/test/java/org/hyperledger/besu/ethereum/bonsai/BonsaiSnapshotIsolationTests.java index ed4b4549674..03407d2868f 100644 --- a/ethereum/core/src/test/java/org/hyperledger/besu/ethereum/bonsai/BonsaiSnapshotIsolationTests.java +++ b/ethereum/core/src/test/java/org/hyperledger/besu/ethereum/bonsai/BonsaiSnapshotIsolationTests.java @@ -61,8 +61,11 @@ public void testIsolatedFromHead_behindHead() { assertThat(res.isSuccessful()).isTrue(); assertThat(res2.isSuccessful()).isTrue(); - assertThat(archive.getTrieLogManager().containWorldStateStorage(firstBlock.getHash())).isTrue(); - assertThat(archive.getTrieLogManager().containWorldStateStorage(secondBlock.getHash())) + assertThat( + archive.getCachedWorldStorageManager().containWorldStateStorage(firstBlock.getHash())) + .isTrue(); + assertThat( + archive.getCachedWorldStorageManager().containWorldStateStorage(secondBlock.getHash())) .isTrue(); assertThat(archive.getMutable().get(testAddress)).isNotNull(); diff --git a/ethereum/core/src/test/java/org/hyperledger/besu/ethereum/bonsai/BonsaiWorldStateArchiveTest.java b/ethereum/core/src/test/java/org/hyperledger/besu/ethereum/bonsai/BonsaiWorldStateArchiveTest.java index b492b67ffdb..0721522ffb1 100644 --- a/ethereum/core/src/test/java/org/hyperledger/besu/ethereum/bonsai/BonsaiWorldStateArchiveTest.java +++ b/ethereum/core/src/test/java/org/hyperledger/besu/ethereum/bonsai/BonsaiWorldStateArchiveTest.java @@ -32,6 +32,7 @@ import org.hyperledger.besu.datatypes.Hash; import org.hyperledger.besu.ethereum.bonsai.cache.CachedMerkleTrieLoader; +import org.hyperledger.besu.ethereum.bonsai.cache.CachedWorldStorageManager; import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiWorldStateKeyValueStorage; import org.hyperledger.besu.ethereum.bonsai.trielog.TrieLogFactoryImpl; import org.hyperledger.besu.ethereum.bonsai.trielog.TrieLogLayer; @@ -74,6 +75,7 @@ public class BonsaiWorldStateArchiveTest { @Mock SegmentedKeyValueStorageTransaction segmentedKeyValueStorageTransaction; BonsaiWorldStateProvider bonsaiWorldStateArchive; + @Mock CachedWorldStorageManager cachedWorldStorageManager; @Mock TrieLogManager trieLogManager; @BeforeEach @@ -100,6 +102,7 @@ public void testGetMutableReturnPersistedStateWhenNeeded() { .thenReturn(Optional.of(chainHead.getHash().toArrayUnsafe())); bonsaiWorldStateArchive = new BonsaiWorldStateProvider( + cachedWorldStorageManager, trieLogManager, new BonsaiWorldStateKeyValueStorage(storageProvider, new NoOpMetricsSystem()), blockchain, @@ -123,7 +126,7 @@ public void testGetMutableReturnEmptyWhenLoadMoreThanLimitLayersBack() { final BlockHeader chainHead = blockBuilder.number(512).buildHeader(); when(blockchain.getChainHeadHeader()).thenReturn(chainHead); assertThat(bonsaiWorldStateArchive.getMutable(blockHeader, false)).isEmpty(); - verify(trieLogManager, Mockito.never()).getWorldState(any(Hash.class)); + verify(cachedWorldStorageManager, Mockito.never()).getWorldState(any(Hash.class)); } @Test @@ -131,6 +134,7 @@ public void testGetMutableWhenLoadLessThanLimitLayersBack() { bonsaiWorldStateArchive = new BonsaiWorldStateProvider( + cachedWorldStorageManager, trieLogManager, new BonsaiWorldStateKeyValueStorage(storageProvider, new NoOpMetricsSystem()), blockchain, @@ -142,7 +146,7 @@ public void testGetMutableWhenLoadLessThanLimitLayersBack() { when(mockWorldState.freeze()).thenReturn(mockWorldState); when(trieLogManager.getMaxLayersToLoad()).thenReturn(Long.valueOf(512)); - when(trieLogManager.getWorldState(blockHeader.getHash())) + when(cachedWorldStorageManager.getWorldState(blockHeader.getHash())) .thenReturn(Optional.of(mockWorldState)); when(blockchain.getChainHeadHeader()).thenReturn(chainHead); assertThat(bonsaiWorldStateArchive.getMutable(blockHeader, false)) @@ -162,6 +166,7 @@ public void testGetMutableWithStorageInconsistencyRollbackTheState() { bonsaiWorldStateArchive = spy( new BonsaiWorldStateProvider( + cachedWorldStorageManager, trieLogManager, worldStateStorage, blockchain, @@ -186,6 +191,7 @@ public void testGetMutableWithStorageConsistencyNotRollbackTheState() { bonsaiWorldStateArchive = spy( new BonsaiWorldStateProvider( + cachedWorldStorageManager, trieLogManager, worldStateStorage, blockchain, @@ -222,6 +228,7 @@ public void testGetMutableWithStorageConsistencyToRollbackAndRollForwardTheState bonsaiWorldStateArchive = spy( new BonsaiWorldStateProvider( + cachedWorldStorageManager, trieLogManager, worldStateStorage, blockchain, @@ -261,6 +268,7 @@ public void testGetMutableWithRollbackNotOverrideTrieLogLayer() { bonsaiWorldStateArchive = spy( new BonsaiWorldStateProvider( + cachedWorldStorageManager, trieLogManager, new BonsaiWorldStateKeyValueStorage(storageProvider, new NoOpMetricsSystem()), blockchain, diff --git a/ethereum/core/src/test/java/org/hyperledger/besu/ethereum/bonsai/trielog/TrieLogManagerTests.java b/ethereum/core/src/test/java/org/hyperledger/besu/ethereum/bonsai/trielog/TrieLogManagerTests.java index 35ce1b77c06..0c911d33556 100644 --- a/ethereum/core/src/test/java/org/hyperledger/besu/ethereum/bonsai/trielog/TrieLogManagerTests.java +++ b/ethereum/core/src/test/java/org/hyperledger/besu/ethereum/bonsai/trielog/TrieLogManagerTests.java @@ -19,15 +19,12 @@ import static org.mockito.Mockito.spy; import org.hyperledger.besu.datatypes.Hash; -import org.hyperledger.besu.ethereum.bonsai.BonsaiWorldStateProvider; -import org.hyperledger.besu.ethereum.bonsai.cache.CachedWorldStorageManager; import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiWorldStateKeyValueStorage; import org.hyperledger.besu.ethereum.bonsai.worldview.BonsaiWorldState; import org.hyperledger.besu.ethereum.bonsai.worldview.BonsaiWorldStateUpdateAccumulator; import org.hyperledger.besu.ethereum.chain.Blockchain; import org.hyperledger.besu.ethereum.core.BlockHeader; import org.hyperledger.besu.ethereum.core.BlockHeaderTestFixture; -import org.hyperledger.besu.metrics.noop.NoOpMetricsSystem; import java.util.concurrent.atomic.AtomicBoolean; @@ -48,7 +45,6 @@ public class TrieLogManagerTests { @Mock BonsaiWorldStateKeyValueStorage bonsaiWorldStateKeyValueStorage; @Mock BonsaiWorldState worldState; - @Mock BonsaiWorldStateProvider archive; @Mock Blockchain blockchain; BonsaiWorldStateUpdateAccumulator bonsaiUpdater = spy(new BonsaiWorldStateUpdateAccumulator(worldState, (__, ___) -> {}, (__, ___) -> {})); @@ -57,14 +53,7 @@ public class TrieLogManagerTests { @BeforeEach public void setup() { - trieLogManager = - new CachedWorldStorageManager( - archive, - blockchain, - bonsaiWorldStateKeyValueStorage, - new NoOpMetricsSystem(), - 512, - null); + trieLogManager = new TrieLogManager(blockchain, bonsaiWorldStateKeyValueStorage, 512, null); } @Test diff --git a/ethereum/referencetests/src/main/java/org/hyperledger/besu/ethereum/referencetests/BonsaiReferenceTestWorldState.java b/ethereum/referencetests/src/main/java/org/hyperledger/besu/ethereum/referencetests/BonsaiReferenceTestWorldState.java index a5b46085ada..59eb9329f33 100644 --- a/ethereum/referencetests/src/main/java/org/hyperledger/besu/ethereum/referencetests/BonsaiReferenceTestWorldState.java +++ b/ethereum/referencetests/src/main/java/org/hyperledger/besu/ethereum/referencetests/BonsaiReferenceTestWorldState.java @@ -17,10 +17,10 @@ import org.hyperledger.besu.datatypes.Address; import org.hyperledger.besu.datatypes.Hash; import org.hyperledger.besu.ethereum.bonsai.cache.CachedMerkleTrieLoader; +import org.hyperledger.besu.ethereum.bonsai.cache.CachedWorldStorageManager; import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiPreImageProxy; import org.hyperledger.besu.ethereum.bonsai.storage.BonsaiWorldStateKeyValueStorage; import org.hyperledger.besu.ethereum.bonsai.trielog.TrieLogAddedEvent; -import org.hyperledger.besu.ethereum.bonsai.trielog.TrieLogFactoryImpl; import org.hyperledger.besu.ethereum.bonsai.trielog.TrieLogManager; import org.hyperledger.besu.ethereum.bonsai.worldview.BonsaiWorldState; import org.hyperledger.besu.ethereum.bonsai.worldview.BonsaiWorldStateUpdateAccumulator; @@ -31,8 +31,6 @@ import org.hyperledger.besu.metrics.noop.NoOpMetricsSystem; import org.hyperledger.besu.plugin.services.trielogs.TrieLog; import org.hyperledger.besu.plugin.services.trielogs.TrieLogEvent; -import org.hyperledger.besu.plugin.services.trielogs.TrieLogFactory; -import org.hyperledger.besu.util.Subscribers; import java.util.Map; import java.util.Optional; @@ -52,9 +50,10 @@ public class BonsaiReferenceTestWorldState extends BonsaiWorldState protected BonsaiReferenceTestWorldState( final BonsaiReferenceTestWorldStateStorage worldStateStorage, final CachedMerkleTrieLoader cachedMerkleTrieLoader, + final CachedWorldStorageManager cachedWorldStorageManager, final TrieLogManager trieLogManager, final BonsaiPreImageProxy preImageProxy) { - super(worldStateStorage, cachedMerkleTrieLoader, trieLogManager); + super(worldStateStorage, cachedMerkleTrieLoader, cachedWorldStorageManager, trieLogManager); this.refTestStorage = worldStateStorage; this.preImageProxy = preImageProxy; setAccumulator( @@ -72,7 +71,11 @@ protected BonsaiReferenceTestWorldState( public ReferenceTestWorldState copy() { var layerCopy = new BonsaiReferenceTestWorldStateStorage(worldStateStorage, preImageProxy); return new BonsaiReferenceTestWorldState( - layerCopy, cachedMerkleTrieLoader, trieLogManager, preImageProxy); + layerCopy, + cachedMerkleTrieLoader, + cachedWorldStorageManager, + trieLogManager, + preImageProxy); } /** @@ -102,9 +105,16 @@ public static BonsaiReferenceTestWorldState create( new InMemoryKeyValueStorageProvider(), metricsSystem), preImageProxy); + final NoOpCachedWorldStorageManager noOpCachedWorldStorageManager = + new NoOpCachedWorldStorageManager(); + final BonsaiReferenceTestWorldState worldState = new BonsaiReferenceTestWorldState( - worldStateStorage, cachedMerkleTrieLoader, trieLogManager, preImageProxy); + worldStateStorage, + cachedMerkleTrieLoader, + noOpCachedWorldStorageManager, + trieLogManager, + preImageProxy); final WorldUpdater updater = worldState.updater(); for (final Map.Entry entry : accounts.entrySet()) { @@ -120,19 +130,14 @@ public Stream streamAccounts(final Bytes32 startKeyHash, fina return this.refTestStorage.streamAccounts(this, startKeyHash, limit); } - static class NoOpTrieLogManager implements TrieLogManager { - private final Subscribers trieLogObservers = Subscribers.create(); - private final TrieLogFactory trieLogFactory = new TrieLogFactoryImpl(); + static class NoOpCachedWorldStorageManager extends CachedWorldStorageManager { - @Override - public void saveTrieLog( - final BonsaiWorldStateUpdateAccumulator localUpdater, - final Hash forWorldStateRootHash, - final BlockHeader forBlockHeader, - final BonsaiWorldState forWorldState) { - // notify trie log added observers, synchronously - TrieLog trieLog = trieLogFactory.create(localUpdater, forBlockHeader); - trieLogObservers.forEach(o -> o.onTrieLogAdded(new TrieLogAddedEvent(trieLog))); + public NoOpCachedWorldStorageManager() { + super( + null, + new BonsaiWorldStateKeyValueStorage( + new InMemoryKeyValueStorageProvider(), new NoOpMetricsSystem()), + new NoOpMetricsSystem()); } @Override @@ -165,17 +170,36 @@ public Optional getHeadWorldState( } @Override - public long getMaxLayersToLoad() { - return 0; + public void reset() { + // reference test world states are not re-used + } + } + + static class NoOpTrieLogManager extends TrieLogManager { + + public NoOpTrieLogManager() { + super(null, null, 0, null); } + @SuppressWarnings("UnsynchronizedOverridesSynchronized") @Override - public void reset() { - // reference test world states are not re-used + public void saveTrieLog( + final BonsaiWorldStateUpdateAccumulator localUpdater, + final Hash forWorldStateRootHash, + final BlockHeader forBlockHeader, + final BonsaiWorldState forWorldState) { + // notify trie log added observers, synchronously + TrieLog trieLog = trieLogFactory.create(localUpdater, forBlockHeader); + trieLogObservers.forEach(o -> o.onTrieLogAdded(new TrieLogAddedEvent(trieLog))); + } + + @Override + public long getMaxLayersToLoad() { + return 0; } @Override - public Optional getTrieLogLayer(final Hash blockHash) { + public Optional getTrieLogLayer(final Hash blockHash) { return Optional.empty(); }