diff --git a/ethereum/api/src/main/java/org/hyperledger/besu/ethereum/api/jsonrpc/internal/privacy/methods/eea/EeaSendRawTransaction.java b/ethereum/api/src/main/java/org/hyperledger/besu/ethereum/api/jsonrpc/internal/privacy/methods/eea/EeaSendRawTransaction.java index 8d247c1786d..181e948003b 100644 --- a/ethereum/api/src/main/java/org/hyperledger/besu/ethereum/api/jsonrpc/internal/privacy/methods/eea/EeaSendRawTransaction.java +++ b/ethereum/api/src/main/java/org/hyperledger/besu/ethereum/api/jsonrpc/internal/privacy/methods/eea/EeaSendRawTransaction.java @@ -14,10 +14,13 @@ */ package org.hyperledger.besu.ethereum.api.jsonrpc.internal.privacy.methods.eea; +import static org.apache.logging.log4j.LogManager.getLogger; import static org.hyperledger.besu.ethereum.api.jsonrpc.JsonRpcEnclaveErrorConverter.convertEnclaveInvalidReason; import static org.hyperledger.besu.ethereum.api.jsonrpc.JsonRpcErrorConverter.convertTransactionInvalidReason; import static org.hyperledger.besu.ethereum.api.jsonrpc.internal.response.JsonRpcError.DECODE_ERROR; import static org.hyperledger.besu.ethereum.api.jsonrpc.internal.response.JsonRpcError.PRIVATE_FROM_DOES_NOT_MATCH_ENCLAVE_PUBLIC_KEY; +import static org.hyperledger.besu.ethereum.privacy.PrivacyGroupUtil.findOffchainPrivacyGroup; +import static org.hyperledger.besu.ethereum.privacy.PrivacyGroupUtil.findOnchainPrivacyGroup; import org.hyperledger.besu.enclave.types.PrivacyGroup; import org.hyperledger.besu.ethereum.api.jsonrpc.RpcMethod; @@ -40,14 +43,17 @@ import java.util.Optional; +import org.apache.logging.log4j.Logger; import org.apache.tuweni.bytes.Bytes; import org.apache.tuweni.bytes.Bytes32; public class EeaSendRawTransaction implements JsonRpcMethod { + private static final Logger LOG = getLogger(); private final TransactionPool transactionPool; private final PrivacyController privacyController; private final EnclavePublicKeyProvider enclavePublicKeyProvider; + /* Temporarily adding this flag to this method to avoid being able to use offchain and onchain privacy groups at the same time. Later on this check will be done in a better place. @@ -86,30 +92,25 @@ public JsonRpcResponse response(final JsonRpcRequestContext requestContext) { return new JsonRpcErrorResponse(id, PRIVATE_FROM_DOES_NOT_MATCH_ENCLAVE_PUBLIC_KEY); } - Optional maybePrivacyGroup = null; final Optional maybePrivacyGroupId = privateTransaction.getPrivacyGroupId(); - if (onchainPrivacyGroupsEnabled) { - if (!maybePrivacyGroupId.isPresent()) { - return new JsonRpcErrorResponse(id, JsonRpcError.ONCHAIN_PRIVACY_GROUP_ID_NOT_AVAILABLE); - } - maybePrivacyGroup = - privacyController.retrieveOnChainPrivacyGroupWithToBeAddedMembers( - maybePrivacyGroupId.get(), enclavePublicKey, privateTransaction); - if (maybePrivacyGroup.isEmpty()) { - return new JsonRpcErrorResponse(id, JsonRpcError.ONCHAIN_PRIVACY_GROUP_DOES_NOT_EXIST); - } - } else { // !onchainPrivacyGroupEnabled - if (maybePrivacyGroupId.isPresent()) { - maybePrivacyGroup = - privacyController.retrieveOffChainPrivacyGroup( - maybePrivacyGroupId.get().toBase64String(), enclavePublicKey); - } else { - maybePrivacyGroup = Optional.empty(); - } + + if (onchainPrivacyGroupsEnabled && maybePrivacyGroupId.isEmpty()) { + return new JsonRpcErrorResponse(id, JsonRpcError.ONCHAIN_PRIVACY_GROUP_ID_NOT_AVAILABLE); + } + + final Optional maybePrivacyGroup = + onchainPrivacyGroupsEnabled + ? findOnchainPrivacyGroup( + privacyController, maybePrivacyGroupId, enclavePublicKey, privateTransaction) + : findOffchainPrivacyGroup(privacyController, maybePrivacyGroupId, enclavePublicKey); + + if (onchainPrivacyGroupsEnabled && maybePrivacyGroup.isEmpty()) { + return new JsonRpcErrorResponse(id, JsonRpcError.ONCHAIN_PRIVACY_GROUP_DOES_NOT_EXIST); } final ValidationResult validationResult = privacyController.validatePrivateTransaction(privateTransaction, enclavePublicKey); + if (!validationResult.isValid()) { return new JsonRpcErrorResponse( id, convertTransactionInvalidReason(validationResult.getInvalidReason())); @@ -140,10 +141,10 @@ public JsonRpcResponse response(final JsonRpcRequestContext requestContext) { id, privateTransaction, privateTransactionLookupId, Address.DEFAULT_PRIVACY); } } catch (final IllegalArgumentException | RLPException e) { + LOG.error(e); return new JsonRpcErrorResponse(id, DECODE_ERROR); } catch (final Exception e) { - final String message = e.getMessage(); - return new JsonRpcErrorResponse(id, convertEnclaveInvalidReason(message)); + return new JsonRpcErrorResponse(id, convertEnclaveInvalidReason(e.getMessage())); } } diff --git a/ethereum/api/src/main/java/org/hyperledger/besu/ethereum/api/jsonrpc/internal/privacy/methods/priv/PrivDistributeRawTransaction.java b/ethereum/api/src/main/java/org/hyperledger/besu/ethereum/api/jsonrpc/internal/privacy/methods/priv/PrivDistributeRawTransaction.java index eee8b3e9d96..1b113e6bbde 100644 --- a/ethereum/api/src/main/java/org/hyperledger/besu/ethereum/api/jsonrpc/internal/privacy/methods/priv/PrivDistributeRawTransaction.java +++ b/ethereum/api/src/main/java/org/hyperledger/besu/ethereum/api/jsonrpc/internal/privacy/methods/priv/PrivDistributeRawTransaction.java @@ -20,6 +20,8 @@ import static org.hyperledger.besu.ethereum.api.jsonrpc.internal.response.JsonRpcError.DECODE_ERROR; import static org.hyperledger.besu.ethereum.api.jsonrpc.internal.response.JsonRpcError.ENCLAVE_ERROR; import static org.hyperledger.besu.ethereum.api.jsonrpc.internal.response.JsonRpcError.PRIVATE_FROM_DOES_NOT_MATCH_ENCLAVE_PUBLIC_KEY; +import static org.hyperledger.besu.ethereum.privacy.PrivacyGroupUtil.findOffchainPrivacyGroup; +import static org.hyperledger.besu.ethereum.privacy.PrivacyGroupUtil.findOnchainPrivacyGroup; import org.hyperledger.besu.enclave.types.PrivacyGroup; import org.hyperledger.besu.ethereum.api.jsonrpc.RpcMethod; @@ -81,30 +83,25 @@ public JsonRpcResponse response(final JsonRpcRequestContext requestContext) { return new JsonRpcErrorResponse(id, PRIVATE_FROM_DOES_NOT_MATCH_ENCLAVE_PUBLIC_KEY); } - Optional maybePrivacyGroup = null; final Optional maybePrivacyGroupId = privateTransaction.getPrivacyGroupId(); - if (onchainPrivacyGroupsEnabled) { - if (!maybePrivacyGroupId.isPresent()) { - return new JsonRpcErrorResponse(id, JsonRpcError.ONCHAIN_PRIVACY_GROUP_ID_NOT_AVAILABLE); - } - maybePrivacyGroup = - privacyController.retrieveOnChainPrivacyGroupWithToBeAddedMembers( - maybePrivacyGroupId.get(), enclavePublicKey, privateTransaction); - if (maybePrivacyGroup.isEmpty()) { - return new JsonRpcErrorResponse(id, JsonRpcError.ONCHAIN_PRIVACY_GROUP_DOES_NOT_EXIST); - } - } else { // !onchainPrivacyGroupEnabled - if (maybePrivacyGroupId.isPresent()) { - maybePrivacyGroup = - privacyController.retrieveOffChainPrivacyGroup( - maybePrivacyGroupId.get().toBase64String(), enclavePublicKey); - } else { - maybePrivacyGroup = Optional.empty(); - } + + if (onchainPrivacyGroupsEnabled && maybePrivacyGroupId.isEmpty()) { + return new JsonRpcErrorResponse(id, JsonRpcError.ONCHAIN_PRIVACY_GROUP_ID_NOT_AVAILABLE); + } + + final Optional maybePrivacyGroup = + onchainPrivacyGroupsEnabled + ? findOnchainPrivacyGroup( + privacyController, maybePrivacyGroupId, enclavePublicKey, privateTransaction) + : findOffchainPrivacyGroup(privacyController, maybePrivacyGroupId, enclavePublicKey); + + if (onchainPrivacyGroupsEnabled && maybePrivacyGroup.isEmpty()) { + return new JsonRpcErrorResponse(id, JsonRpcError.ONCHAIN_PRIVACY_GROUP_DOES_NOT_EXIST); } final ValidationResult validationResult = privacyController.validatePrivateTransaction(privateTransaction, enclavePublicKey); + if (!validationResult.isValid()) { return new JsonRpcErrorResponse( id, convertTransactionInvalidReason(validationResult.getInvalidReason())); @@ -118,6 +115,7 @@ public JsonRpcResponse response(final JsonRpcRequestContext requestContext) { LOG.error("Unauthorized privacy multi-tenancy rpc request. {}", e.getMessage()); return new JsonRpcErrorResponse(id, ENCLAVE_ERROR); } catch (final IllegalArgumentException | RLPException e) { + LOG.error(e); return new JsonRpcErrorResponse(id, DECODE_ERROR); } catch (final Exception e) { return new JsonRpcErrorResponse(id, convertEnclaveInvalidReason(e.getMessage())); diff --git a/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/privacy/PrivacyGroupUtil.java b/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/privacy/PrivacyGroupUtil.java index d01c8ebc815..91946560385 100644 --- a/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/privacy/PrivacyGroupUtil.java +++ b/ethereum/core/src/main/java/org/hyperledger/besu/ethereum/privacy/PrivacyGroupUtil.java @@ -15,12 +15,14 @@ package org.hyperledger.besu.ethereum.privacy; import org.hyperledger.besu.crypto.Hash; +import org.hyperledger.besu.enclave.types.PrivacyGroup; import org.hyperledger.besu.ethereum.rlp.BytesValueRLPOutput; import java.util.ArrayList; import java.util.Arrays; import java.util.Comparator; import java.util.List; +import java.util.Optional; import java.util.stream.Collectors; import org.apache.tuweni.bytes.Bytes; @@ -52,4 +54,25 @@ public static Bytes32 calculateEeaPrivacyGroupId( return Hash.keccak256(bytesValueRLPOutput.encoded()); } + + public static Optional findOnchainPrivacyGroup( + final PrivacyController privacyController, + final Optional maybePrivacyGroupId, + final String enclavePublicKey, + final PrivateTransaction privateTransaction) { + return maybePrivacyGroupId.flatMap( + privacyGroupId -> + privacyController.retrieveOnChainPrivacyGroupWithToBeAddedMembers( + privacyGroupId, enclavePublicKey, privateTransaction)); + } + + public static Optional findOffchainPrivacyGroup( + final PrivacyController privacyController, + final Optional maybePrivacyGroupId, + final String enclavePublicKey) { + return maybePrivacyGroupId.flatMap( + privacyGroupId -> + privacyController.retrieveOffChainPrivacyGroup( + privacyGroupId.toBase64String(), enclavePublicKey)); + } }