mirror of https://github.com/hyperledger/besu
parent
06553be973
commit
1ba51845e2
@ -0,0 +1,103 @@ |
|||||||
|
/* |
||||||
|
* Copyright contributors to Hyperledger Besu. |
||||||
|
* |
||||||
|
* 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.eth.manager.peertask.task; |
||||||
|
|
||||||
|
import org.hyperledger.besu.datatypes.Hash; |
||||||
|
import org.hyperledger.besu.ethereum.core.BlockHeader; |
||||||
|
import org.hyperledger.besu.ethereum.core.TransactionReceipt; |
||||||
|
import org.hyperledger.besu.ethereum.eth.EthProtocol; |
||||||
|
import org.hyperledger.besu.ethereum.eth.manager.peertask.InvalidPeerTaskResponseException; |
||||||
|
import org.hyperledger.besu.ethereum.eth.manager.peertask.PeerTask; |
||||||
|
import org.hyperledger.besu.ethereum.eth.manager.peertask.PeerTaskBehavior; |
||||||
|
import org.hyperledger.besu.ethereum.eth.messages.GetReceiptsMessage; |
||||||
|
import org.hyperledger.besu.ethereum.eth.messages.ReceiptsMessage; |
||||||
|
import org.hyperledger.besu.ethereum.mainnet.BodyValidation; |
||||||
|
import org.hyperledger.besu.ethereum.p2p.rlpx.wire.MessageData; |
||||||
|
|
||||||
|
import java.util.ArrayList; |
||||||
|
import java.util.Collection; |
||||||
|
import java.util.HashMap; |
||||||
|
import java.util.List; |
||||||
|
import java.util.Map; |
||||||
|
|
||||||
|
public class GetReceiptsFromPeerTask |
||||||
|
implements PeerTask<Map<BlockHeader, List<TransactionReceipt>>> { |
||||||
|
|
||||||
|
private final Collection<BlockHeader> blockHeaders; |
||||||
|
private final Map<Hash, List<BlockHeader>> headersByReceiptsRoot = new HashMap<>(); |
||||||
|
|
||||||
|
public GetReceiptsFromPeerTask(final Collection<BlockHeader> blockHeaders) { |
||||||
|
this.blockHeaders = blockHeaders; |
||||||
|
blockHeaders.forEach( |
||||||
|
header -> |
||||||
|
headersByReceiptsRoot |
||||||
|
.computeIfAbsent(header.getReceiptsRoot(), key -> new ArrayList<>()) |
||||||
|
.add(header)); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public String getSubProtocol() { |
||||||
|
return EthProtocol.NAME; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public long getRequiredBlockNumber() { |
||||||
|
return blockHeaders.stream() |
||||||
|
.mapToLong(BlockHeader::getNumber) |
||||||
|
.max() |
||||||
|
.orElse(BlockHeader.GENESIS_BLOCK_NUMBER); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public MessageData getRequestMessage() { |
||||||
|
// Since we have to match up the data by receipt root, we only need to request receipts
|
||||||
|
// for one of the headers with each unique receipt root.
|
||||||
|
final List<Hash> blockHashes = |
||||||
|
headersByReceiptsRoot.values().stream() |
||||||
|
.map(headers -> headers.getFirst().getHash()) |
||||||
|
.toList(); |
||||||
|
return GetReceiptsMessage.create(blockHashes); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public Map<BlockHeader, List<TransactionReceipt>> parseResponse(final MessageData messageData) |
||||||
|
throws InvalidPeerTaskResponseException { |
||||||
|
if (messageData == null) { |
||||||
|
throw new InvalidPeerTaskResponseException(); |
||||||
|
} |
||||||
|
final ReceiptsMessage receiptsMessage = ReceiptsMessage.readFrom(messageData); |
||||||
|
final List<List<TransactionReceipt>> receiptsByBlock = receiptsMessage.receipts(); |
||||||
|
if (receiptsByBlock.isEmpty() || receiptsByBlock.size() > blockHeaders.size()) { |
||||||
|
throw new InvalidPeerTaskResponseException(); |
||||||
|
} |
||||||
|
|
||||||
|
final Map<BlockHeader, List<TransactionReceipt>> receiptsByHeader = new HashMap<>(); |
||||||
|
for (final List<TransactionReceipt> receiptsInBlock : receiptsByBlock) { |
||||||
|
final List<BlockHeader> blockHeaders = |
||||||
|
headersByReceiptsRoot.get(BodyValidation.receiptsRoot(receiptsInBlock)); |
||||||
|
if (blockHeaders == null) { |
||||||
|
// Contains receipts that we didn't request, so mustn't be the response we're looking for.
|
||||||
|
throw new InvalidPeerTaskResponseException(); |
||||||
|
} |
||||||
|
blockHeaders.forEach(header -> receiptsByHeader.put(header, receiptsInBlock)); |
||||||
|
} |
||||||
|
return receiptsByHeader; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public Collection<PeerTaskBehavior> getPeerTaskBehaviors() { |
||||||
|
return List.of(PeerTaskBehavior.RETRY_WITH_OTHER_PEERS, PeerTaskBehavior.RETRY_WITH_SAME_PEER); |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue