RestrictedFlexibleEeaSendRawTransaction.java

/*
 * Copyright ConsenSys AG.
 *
 * 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.api.jsonrpc.internal.privacy.methods.eea;

import static org.hyperledger.besu.ethereum.core.PrivacyParameters.FLEXIBLE_PRIVACY;

import org.hyperledger.besu.datatypes.Address;
import org.hyperledger.besu.enclave.types.PrivacyGroup;
import org.hyperledger.besu.ethereum.api.jsonrpc.internal.privacy.methods.PrivacyIdProvider;
import org.hyperledger.besu.ethereum.api.jsonrpc.internal.response.RpcErrorType;
import org.hyperledger.besu.ethereum.core.Transaction;
import org.hyperledger.besu.ethereum.eth.transactions.TransactionPool;
import org.hyperledger.besu.ethereum.mainnet.ValidationResult;
import org.hyperledger.besu.ethereum.privacy.FlexibleUtil;
import org.hyperledger.besu.ethereum.privacy.PrivacyController;
import org.hyperledger.besu.ethereum.privacy.PrivateTransaction;
import org.hyperledger.besu.ethereum.transaction.TransactionInvalidReason;
import org.hyperledger.besu.ethereum.util.NonceProvider;
import org.hyperledger.besu.plugin.data.Restriction;
import org.hyperledger.besu.plugin.services.privacy.PrivateMarkerTransactionFactory;

import java.util.List;
import java.util.Optional;

import io.vertx.ext.auth.User;
import org.apache.tuweni.bytes.Bytes;

public class RestrictedFlexibleEeaSendRawTransaction extends AbstractEeaSendRawTransaction {

  private final PrivacyController privacyController;
  private final PrivacyIdProvider privacyIdProvider;

  public RestrictedFlexibleEeaSendRawTransaction(
      final TransactionPool transactionPool,
      final PrivacyIdProvider privacyIdProvider,
      final PrivateMarkerTransactionFactory privateMarkerTransactionFactory,
      final NonceProvider publicNonceProvider,
      final PrivacyController privacyController) {
    super(transactionPool, privacyIdProvider, privateMarkerTransactionFactory, publicNonceProvider);
    this.privacyController = privacyController;
    this.privacyIdProvider = privacyIdProvider;
  }

  @Override
  protected ValidationResult<TransactionInvalidReason> validatePrivateTransaction(
      final PrivateTransaction privateTransaction, final Optional<User> user) {
    if (!privateTransaction.getRestriction().equals(Restriction.RESTRICTED)) {
      return ValidationResult.invalid(
          TransactionInvalidReason.PRIVATE_UNIMPLEMENTED_TRANSACTION_TYPE);
    }
    return privacyController.validatePrivateTransaction(
        privateTransaction, privacyIdProvider.getPrivacyUserId(user));
  }

  @Override
  protected Transaction createPrivateMarkerTransaction(
      final Address sender,
      final PrivateTransaction privateTransaction,
      final Optional<User> user) {
    final Optional<Bytes> maybePrivacyGroupId = privateTransaction.getPrivacyGroupId();
    if (maybePrivacyGroupId.isEmpty()) {
      throw new JsonRpcErrorResponseException(RpcErrorType.FLEXIBLE_PRIVACY_GROUP_ID_NOT_AVAILABLE);
    }
    final Bytes privacyGroupId = maybePrivacyGroupId.get();

    final String privacyUserId = privacyIdProvider.getPrivacyUserId(user);

    Optional<PrivacyGroup> maybePrivacyGroup =
        privacyController.findPrivacyGroupByGroupId(privacyGroupId.toBase64String(), privacyUserId);

    final boolean isGroupAdditionTransaction =
        FlexibleUtil.isGroupAdditionTransaction(privateTransaction);
    if (maybePrivacyGroup.isEmpty() && !isGroupAdditionTransaction) {
      throw new JsonRpcErrorResponseException(RpcErrorType.FLEXIBLE_PRIVACY_GROUP_DOES_NOT_EXIST);
    }

    if (isGroupAdditionTransaction) {
      final List<String> participantsFromParameter =
          FlexibleUtil.getParticipantsFromParameter(privateTransaction.getPayload());
      if (maybePrivacyGroup.isEmpty()) {
        maybePrivacyGroup =
            Optional.of(
                new PrivacyGroup(
                    privacyGroupId.toBase64String(),
                    PrivacyGroup.Type.FLEXIBLE,
                    null,
                    null,
                    participantsFromParameter));
      } else {
        maybePrivacyGroup.get().addMembers(participantsFromParameter);
      }
    }

    if (!maybePrivacyGroup.get().getMembers().contains(privacyUserId)) {
      throw new JsonRpcErrorResponseException(RpcErrorType.FLEXIBLE_PRIVACY_GROUP_DOES_NOT_EXIST);
    }

    final String pmtPayload =
        privacyController.createPrivateMarkerTransactionPayload(
            privateTransaction, privacyUserId, maybePrivacyGroup);

    return createPrivateMarkerTransaction(
        sender, FLEXIBLE_PRIVACY, pmtPayload, privateTransaction, privacyUserId);
  }

  @Override
  protected long getGasLimit(final PrivateTransaction privateTransaction, final String pmtPayload) {
    return privateTransaction.getGasLimit();
  }
}