All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.hedera.node.app.service.token.impl.handlers.TokenRejectHandler Maven / Gradle / Ivy

/*
 * Copyright (C) 2024 Hedera Hashgraph, LLC
 *
 * 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.
 */

package com.hedera.node.app.service.token.impl.handlers;

import static com.hedera.hapi.node.base.ResponseCodeEnum.ACCOUNT_IS_IMMUTABLE;
import static com.hedera.hapi.node.base.ResponseCodeEnum.ACCOUNT_IS_TREASURY;
import static com.hedera.hapi.node.base.ResponseCodeEnum.EMPTY_TOKEN_REFERENCE_LIST;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INSUFFICIENT_TOKEN_BALANCE;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_ACCOUNT_ID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_NFT_ID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_OWNER_ID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_TOKEN_ID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_TOKEN_NFT_SERIAL_NUMBER;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_TRANSACTION_BODY;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_TREASURY_ACCOUNT_FOR_TOKEN;
import static com.hedera.hapi.node.base.ResponseCodeEnum.NOT_SUPPORTED;
import static com.hedera.hapi.node.base.ResponseCodeEnum.TOKEN_REFERENCE_LIST_SIZE_LIMIT_EXCEEDED;
import static com.hedera.hapi.node.base.ResponseCodeEnum.TOKEN_REFERENCE_REPEATED;
import static com.hedera.hapi.node.base.SubType.TOKEN_FUNGIBLE_COMMON;
import static com.hedera.hapi.node.base.SubType.TOKEN_NON_FUNGIBLE_UNIQUE;
import static com.hedera.node.app.hapi.fees.usage.SingletonUsageProperties.USAGE_PROPERTIES;
import static com.hedera.node.app.hapi.fees.usage.token.entities.TokenEntitySizes.TOKEN_ENTITY_SIZES;
import static com.hedera.node.app.service.token.impl.util.CryptoTransferHelper.createFungibleTransfer;
import static com.hedera.node.app.service.token.impl.util.CryptoTransferHelper.createNftTransfer;
import static com.hedera.node.app.service.token.impl.util.CryptoTransferHelper.nftTransfer;
import static com.hedera.node.app.service.token.impl.util.TokenHandlerHelper.getIfUsable;
import static com.hedera.node.app.service.token.impl.util.TokenHandlerHelper.getIfUsableForAliasedId;
import static com.hedera.node.app.spi.key.KeyUtils.isValid;
import static com.hedera.node.app.spi.validation.Validations.validateAccountID;
import static com.hedera.node.app.spi.workflows.HandleException.validateTrue;
import static com.hedera.node.app.spi.workflows.PreCheckException.validateFalsePreCheck;
import static com.hedera.node.app.spi.workflows.PreCheckException.validateTruePreCheck;
import static java.util.Objects.requireNonNull;

import com.hedera.hapi.node.base.AccountID;
import com.hedera.hapi.node.base.HederaFunctionality;
import com.hedera.hapi.node.base.NftID;
import com.hedera.hapi.node.base.NftTransfer;
import com.hedera.hapi.node.base.SubType;
import com.hedera.hapi.node.base.TokenID;
import com.hedera.hapi.node.base.TokenTransferList;
import com.hedera.hapi.node.state.token.Account;
import com.hedera.hapi.node.token.TokenReference;
import com.hedera.hapi.node.transaction.TransactionBody;
import com.hedera.node.app.service.token.ReadableAccountStore;
import com.hedera.node.app.service.token.impl.WritableAccountStore;
import com.hedera.node.app.service.token.impl.WritableNftStore;
import com.hedera.node.app.service.token.impl.WritableTokenRelationStore;
import com.hedera.node.app.service.token.impl.WritableTokenStore;
import com.hedera.node.app.service.token.impl.handlers.transfer.AdjustFungibleTokenChangesStep;
import com.hedera.node.app.service.token.impl.handlers.transfer.NFTOwnersChangeStep;
import com.hedera.node.app.service.token.impl.handlers.transfer.TransferContextImpl;
import com.hedera.node.app.spi.fees.FeeContext;
import com.hedera.node.app.spi.fees.Fees;
import com.hedera.node.app.spi.workflows.HandleContext;
import com.hedera.node.app.spi.workflows.HandleException;
import com.hedera.node.app.spi.workflows.PreCheckException;
import com.hedera.node.app.spi.workflows.PreHandleContext;
import com.hedera.node.app.spi.workflows.TransactionHandler;
import com.hedera.node.config.data.FeesConfig;
import com.hedera.node.config.data.LedgerConfig;
import com.hedera.node.config.data.TokensConfig;
import com.swirlds.base.utility.Pair;
import edu.umd.cs.findbugs.annotations.NonNull;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.stream.Collectors;
import javax.inject.Inject;
import javax.inject.Singleton;

/**
 * This class contains all workflow-related functionality regarding {@link HederaFunctionality#TOKEN_REJECT}.
 * This transaction type is used to reject tokens from an account and send them back to the treasury.
 */
@Singleton
public class TokenRejectHandler extends BaseTokenHandler implements TransactionHandler {

    @Inject
    public TokenRejectHandler() {}

    @Override
    public void preHandle(@NonNull final PreHandleContext context) throws PreCheckException {
        requireNonNull(context);
        final var txn = context.body();
        final var op = txn.tokenRejectOrThrow();

        // If owner account is specified, verify that it's valid and require that it signs the transaction.
        // If not specified, payer account is considered as the one rejecting the tokens.
        if (op.hasOwner()) {
            final var accountStore = context.createStore(ReadableAccountStore.class);
            verifyOwnerAndRequireKey(op.ownerOrThrow(), context, accountStore);
        }
    }

    /**
     * Verifies that the owner account exists and ensures the account's key is valid and required for the transaction.
     *
     * @param ownerId The AccountID of the owner whose key needs to be validated.
     * @param context The PreHandleContext providing transaction context.
     * @param accountStore The store to access readable account information.
     * @throws PreCheckException If the sender's account is immutable or the sender's account ID is invalid.
     */
    private void verifyOwnerAndRequireKey(
            @NonNull final AccountID ownerId,
            @NonNull final PreHandleContext context,
            @NonNull final ReadableAccountStore accountStore)
            throws PreCheckException {

        final var ownerAccount = accountStore.getAliasedAccountById(ownerId);
        validateTruePreCheck(ownerAccount != null, INVALID_OWNER_ID);

        // If the sender account is immutable, then we throw an exception.
        final var key = ownerAccount.key();
        if (key == null || !isValid(key)) {
            throw new PreCheckException(ACCOUNT_IS_IMMUTABLE);
        }
        context.requireKey(key);
    }

    @SuppressWarnings("java:S2259")
    @Override
    public void pureChecks(@NonNull final TransactionBody txn) throws PreCheckException {
        requireNonNull(txn, "Transaction body cannot be null");
        final var op = txn.tokenRejectOrThrow();

        validateFalsePreCheck(op.rejections().isEmpty(), EMPTY_TOKEN_REFERENCE_LIST);
        if (op.hasOwner()) {
            validateAccountID(op.owner(), null);
        }

        final var uniqueTokenReferences = new HashSet();
        for (final var rejection : op.rejections()) {
            if (!uniqueTokenReferences.add(rejection)) {
                throw new PreCheckException(TOKEN_REFERENCE_REPEATED);
            }
            // Ensure one token type per single rejection reference.
            validateFalsePreCheck(rejection.hasFungibleToken() && rejection.hasNft(), INVALID_TRANSACTION_BODY);

            if (rejection.hasFungibleToken()) {
                final var tokenID = rejection.fungibleToken();
                validateTruePreCheck(tokenID != null && !tokenID.equals(TokenID.DEFAULT), INVALID_TOKEN_ID);
            }
            if (rejection.hasNft()) {
                final var nftID = rejection.nft();
                validateTruePreCheck(nftID != null && nftID.tokenId() != null, INVALID_NFT_ID);
                validateTruePreCheck(nftID.serialNumber() > 0, INVALID_TOKEN_NFT_SERIAL_NUMBER);
            }
        }
    }

    @Override
    public void handle(@NonNull final HandleContext context) throws HandleException {
        requireNonNull(context);
        final var op = context.body().tokenRejectOrThrow();
        final var rejectingAccountID = op.ownerOrElse(context.payer());
        final var rejections = op.rejections();

        final var ledgerConfig = context.configuration().getConfigData(LedgerConfig.class);
        final var tokensConfig = context.configuration().getConfigData(TokensConfig.class);
        validateTrue(tokensConfig.tokenRejectEnabled(), NOT_SUPPORTED);
        validateTrue(rejections.size() <= ledgerConfig.tokenRejectsMaxLen(), TOKEN_REFERENCE_LIST_SIZE_LIMIT_EXCEEDED);

        final var accountStore = context.storeFactory().writableStore(WritableAccountStore.class);
        final var rejectingAccount = getIfUsableForAliasedId(
                rejectingAccountID, accountStore, context.expiryValidator(), INVALID_ACCOUNT_ID);

        // Process the rejections to prepare for the transfers
        final var processedRejectTransfers = processRejectionsForTransfer(rejections, context, rejectingAccount);

        // Apply all changes to the handleContext's states by performing the transfer to the treasuries
        // Allowances will not be affected with the token reject
        final var transferContext = new TransferContextImpl(context);
        final var fungibleTokensStep = new AdjustFungibleTokenChangesStep(processedRejectTransfers, context.payer());
        final var nftOwnersChangeStep = new NFTOwnersChangeStep(processedRejectTransfers, context.payer());
        fungibleTokensStep.doIn(transferContext);
        nftOwnersChangeStep.doIn(transferContext);
    }

    /**
     * Processes the rejections specified in the transaction to prepare for the transfers to the treasuries.
     *
     * @param rejections The list of TokenReferences representing the rejections to process.
     * @param context The HandleContext providing the current handling context.
     * @param rejectingAccount The Account rejecting its tokens.
     * @return A List of TokenTransferLists representing the transfers for the rejected tokens to the treasuries
     */
    private List processRejectionsForTransfer(
            @NonNull final List rejections,
            @NonNull final HandleContext context,
            @NonNull final Account rejectingAccount) {

        final WritableTokenStore tokenStore = context.storeFactory().writableStore(WritableTokenStore.class);
        final WritableNftStore nftStore = context.storeFactory().writableStore(WritableNftStore.class);
        final WritableTokenRelationStore relStore =
                context.storeFactory().writableStore(WritableTokenRelationStore.class);

        final var accountID = rejectingAccount.accountIdOrThrow();

        // Validate fungible tokens and collect token transfer lists to the treasuries
        final var fungibleTransfers = rejections.stream()
                .filter(TokenReference::hasFungibleToken)
                .map(rejection -> processFungibleTokenRejection(rejection, rejectingAccount, tokenStore, relStore))
                .toList();

        // Validate NFTs and collect token transfer lists to the treasuries
        // NFTTransfers with same TokenID are grouped under the same TokenTransferList
        final var nftTransfers = rejections.stream()
                .filter(TokenReference::hasNft)
                .map(rejection -> processNFTTokenRejection(rejection.nftOrThrow(), accountID, nftStore, tokenStore))
                .collect(Collectors.groupingBy(Pair::left, Collectors.mapping(Pair::right, Collectors.toList())))
                .entrySet()
                .stream()
                .map(tokenIdNFTPair -> createNftTransfer(tokenIdNFTPair.getKey(), tokenIdNFTPair.getValue()))
                .toList();

        final var tokenTransferLists = new ArrayList();
        tokenTransferLists.addAll(fungibleTransfers);
        tokenTransferLists.addAll(nftTransfers);
        return tokenTransferLists;
    }

    /**
     * Processes a single NFT rejection by performing validations and converting it to a Pair of TokenID & NftTransfer.
     *
     * @param nftID The NftID of the NFT being rejected.
     * @param rejectingAccountID The AccountID of the owner rejecting the NFT.
     * @param nftStore Access to writable NFT data.
     * @param tokenStore Access to writable token data.
     */
    private Pair processNFTTokenRejection(
            @NonNull final NftID nftID,
            @NonNull final AccountID rejectingAccountID,
            @NonNull final WritableNftStore nftStore,
            @NonNull final WritableTokenStore tokenStore) {
        final var nft = getIfUsable(nftID, nftStore);
        final var token = getIfUsable(nftID.tokenIdOrThrow(), tokenStore);
        final var tokenTreasury = token.treasuryAccountIdOrThrow();
        validateTrue(!tokenTreasury.equals(rejectingAccountID), ACCOUNT_IS_TREASURY);
        validateTrue(nft.hasOwnerId() && rejectingAccountID.equals(nft.ownerId()), INVALID_OWNER_ID);

        return Pair.of(token.tokenId(), nftTransfer(rejectingAccountID, tokenTreasury, nftID.serialNumber()));
    }

    /**
     * Processes a single fungible token rejection by performing validations and builds the transfer for the rejected
     * token.
     *
     * @param rejection The token reference detailing the rejection.
     * @param rejectingAccount The Account rejecting the token.
     * @param tokenStore Access to writable token data.
     * @param relStore Access to writable token relations.
     */
    private TokenTransferList processFungibleTokenRejection(
            @NonNull final TokenReference rejection,
            @NonNull final Account rejectingAccount,
            @NonNull final WritableTokenStore tokenStore,
            @NonNull final WritableTokenRelationStore relStore) {
        final var accountID = rejectingAccount.accountIdOrThrow();
        final var token = getIfUsable(rejection.fungibleTokenOrThrow(), tokenStore);
        final var tokenId = token.tokenIdOrThrow();
        final var tokenRelation = getIfUsable(accountID, tokenId, relStore);
        validateTrue(token.treasuryAccountId() != null, INVALID_TREASURY_ACCOUNT_FOR_TOKEN);
        validateTrue(!token.treasuryAccountId().equals(accountID), ACCOUNT_IS_TREASURY);
        validateTrue(tokenRelation.balance() > 0, INSUFFICIENT_TOKEN_BALANCE);

        return createFungibleTransfer(tokenId, accountID, tokenRelation.balance(), token.treasuryAccountId());
    }

    @NonNull
    @Override
    public Fees calculateFees(final FeeContext feeContext) {
        final var body = feeContext.body();
        final var op = body.tokenRejectOrThrow();
        final var config = feeContext.configuration();
        final var tokenMultiplier =
                requireNonNull(config).getConfigData(FeesConfig.class).tokenTransferUsageMultiplier();

        final int totalRejections = op.rejections().size();
        int numOfFungibleTokenRejections = 0;
        int numOfNFTRejections = 0;
        for (final var rejection : op.rejections()) {
            if (rejection.hasFungibleToken()) {
                // Each fungible token rejection involves 2 AccountAmount transfers
                // We add 2 in order to match CryptoTransfer's bpt & rbs fee calculation
                numOfFungibleTokenRejections += 2;
            } else {
                numOfNFTRejections++;
            }
        }

        final int weightedTokensInvolved = tokenMultiplier * totalRejections;
        final int weightedFungibleTokens = tokenMultiplier * numOfFungibleTokenRejections;
        final long bpt =
                calculateBytesPerTransaction(weightedTokensInvolved, weightedFungibleTokens, numOfNFTRejections);
        final long rbs = USAGE_PROPERTIES.legacyReceiptStorageSecs() * bpt;

        return feeContext
                .feeCalculatorFactory()
                .feeCalculator(getSubType(numOfNFTRejections))
                .addBytesPerTransaction(bpt)
                .addRamByteSeconds(rbs)
                .calculate();
    }

    private SubType getSubType(final int numOfNFTRejections) {
        return numOfNFTRejections != 0 ? TOKEN_NON_FUNGIBLE_UNIQUE : TOKEN_FUNGIBLE_COMMON;
    }

    private long calculateBytesPerTransaction(
            final int weightedTokensInvolved, final int weightedFungibleTokens, final int numNFTRejections) {
        return TOKEN_ENTITY_SIZES.bytesUsedToRecordTokenTransfers(
                weightedTokensInvolved, weightedFungibleTokens, numNFTRejections);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy