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

com.hedera.node.app.service.token.impl.util.AirdropHandlerHelper 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.util;

import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_ACCOUNT_ID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_PENDING_AIRDROP_ID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_RECEIVING_NODE_ACCOUNT;
import static com.hedera.hapi.node.base.ResponseCodeEnum.PENDING_AIRDROP_ID_REPEATED;
import static com.hedera.node.app.service.token.impl.handlers.BaseTokenHandler.UNLIMITED_AUTOMATIC_ASSOCIATIONS;
import static com.hedera.node.app.service.token.impl.util.TokenHandlerHelper.getIfUsableForAliasedId;
import static com.hedera.node.app.spi.workflows.HandleException.validateFalse;
import static com.hedera.node.app.spi.workflows.HandleException.validateTrue;
import static java.util.Objects.requireNonNull;

import com.hedera.hapi.node.base.AccountAmount;
import com.hedera.hapi.node.base.AccountID;
import com.hedera.hapi.node.base.NftID;
import com.hedera.hapi.node.base.NftTransfer;
import com.hedera.hapi.node.base.PendingAirdropId;
import com.hedera.hapi.node.base.PendingAirdropValue;
import com.hedera.hapi.node.base.TokenID;
import com.hedera.hapi.node.state.token.Account;
import com.hedera.hapi.node.state.token.AccountPendingAirdrop;
import com.hedera.hapi.node.state.token.TokenRelation;
import com.hedera.hapi.node.transaction.PendingAirdropRecord;
import com.hedera.node.app.service.token.ReadableAccountStore;
import com.hedera.node.app.service.token.ReadableAirdropStore;
import com.hedera.node.app.service.token.ReadableTokenRelationStore;
import com.hedera.node.app.spi.workflows.HandleContext;
import com.hedera.node.app.spi.workflows.HandleException;
import edu.umd.cs.findbugs.annotations.NonNull;
import edu.umd.cs.findbugs.annotations.Nullable;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Set;

/**
 * Utility class that provides static methods.
 */
public class AirdropHandlerHelper {
    private static final Long LAST_RESERVED_SYSTEM_ACCOUNT = 1000L;

    /**
     * Given an account store and a list of validated pending airdrop ids, standardizes the pending airdrop ids
     * to use only the {@code 0.0.X} numeric ids for both sender and receiver.
     * @param accountStore the account store to look up aliases in
     * @param airdropIds the list of pending airdrop ids to standardize
     * @param knownExtantIdTypes the set of id types that are known to exist in the system
     * @return a list of standardized pending airdrop ids
     * @throws HandleException with INVALID_PENDING_AIRDROP_ID if any of the pending airdrop ids are invalid
     */
    public static List standardizeAirdropIds(
            @NonNull final ReadableAccountStore accountStore,
            @NonNull final ReadableAirdropStore airdropStore,
            @NonNull final List airdropIds,
            @NonNull final Set knownExtantIdTypes) {
        final List standardAirdropIds = new ArrayList<>();
        for (final var airdropId : airdropIds) {
            final var sender = knownExtantIdTypes.contains(IdType.SENDER)
                    ? requireNonNull(accountStore.getAliasedAccountById(airdropId.senderIdOrThrow()))
                    : accountStore.getAliasedAccountById(airdropId.senderIdOrElse(AccountID.DEFAULT));
            validateTrue(sender != null, INVALID_PENDING_AIRDROP_ID);
            final var receiver = knownExtantIdTypes.contains(IdType.RECEIVER)
                    ? requireNonNull(accountStore.getAliasedAccountById(airdropId.receiverIdOrThrow()))
                    : accountStore.getAliasedAccountById(airdropId.receiverIdOrElse(AccountID.DEFAULT));
            validateTrue(receiver != null, INVALID_PENDING_AIRDROP_ID);
            // Airdrop ids always have 0.0.X form of both sender and receiver ids
            final var validatedId = airdropId
                    .copyBuilder()
                    .senderId(sender.accountIdOrThrow())
                    .receiverId(receiver.accountIdOrThrow())
                    .build();
            validateTrue(airdropStore.exists(validatedId), INVALID_PENDING_AIRDROP_ID);
            standardAirdropIds.add(validatedId);
        }
        final var uniqueAirdropIds = Set.copyOf(standardAirdropIds);
        validateTrue(standardAirdropIds.size() == uniqueAirdropIds.size(), PENDING_AIRDROP_ID_REPEATED);
        return standardAirdropIds;
    }

    /**
     * A type of airdrop-involved id to look up in the account store.
     */
    public enum IdType {
        SENDER,
        RECEIVER
    }

    public record FungibleAirdropLists(
            @NonNull List transferFungibleAmounts,
            @NonNull List pendingFungibleAmounts,
            int transfersNeedingAutoAssociation) {}

    public record NftAirdropLists(
            @NonNull List transferNftList,
            @NonNull List pendingNftList,
            int transfersNeedingAutoAssociation) {}

    private AirdropHandlerHelper() {
        throw new UnsupportedOperationException("Utility class only");
    }

    /**
     * Checks every {@link AccountAmount} from given transfer list and separate it in to two lists.
     * One containing transfers that should be added to pending airdrop state and the other list - transfers that should
     * be executed. The check is done by account's available auto associations slots and the existence of account-token
     * relation {@link #isPendingAirdrop(HandleContext, Account, TokenRelation)}
     *
     * @param context {@link HandleContext} used to obtain state stores
     * @param tokenId token id
     * @param transfers list of {@link AccountAmount}
     * @return {@link FungibleAirdropLists} a record containing two lists - transfers to be added in pending state and transfers to be executed
     */
    public static FungibleAirdropLists separateFungibleTransfers(
            @NonNull final HandleContext context,
            @NonNull final TokenID tokenId,
            @NonNull final List transfers) {
        final var transferFungibleAmounts = new ArrayList();
        final var pendingFungibleAmounts = new ArrayList();
        final var transfersNeedingAutoAssociation = new LinkedHashSet();

        final var tokenRelStore = context.storeFactory().readableStore(ReadableTokenRelationStore.class);
        final var accountStore = context.storeFactory().readableStore(ReadableAccountStore.class);
        for (final var aa : transfers) {
            final var accountId = aa.accountIDOrElse(AccountID.DEFAULT);
            // Treat a transfer to a missing account as an auto-creation attempt
            if (accountStore.isMissing(accountId)) {
                transferFungibleAmounts.add(aa);
                transfersNeedingAutoAssociation.add(accountId);
                continue;
            }

            final var account =
                    getIfUsableForAliasedId(accountId, accountStore, context.expiryValidator(), INVALID_ACCOUNT_ID);
            final var tokenRel = tokenRelStore.get(accountId, tokenId);
            var isPendingAirdrop = false;
            if (aa.amount() > 0) {
                validateFalse(isSystemAccount(account), INVALID_RECEIVING_NODE_ACCOUNT);
                isPendingAirdrop = isPendingAirdrop(context, account, tokenRel);
            }

            if (isPendingAirdrop) {
                pendingFungibleAmounts.add(aa);
            } else {
                transferFungibleAmounts.add(aa);
                // Any transfer that is with no explicitly associated token will need to be charged $0.1
                // So we charge $0.05 pending airdrop fee and $0.05 is charged in CryptoTransferHandler during
                // auto-association
                if (tokenRel == null) {
                    transfersNeedingAutoAssociation.add(accountId);
                }
            }
        }

        return new FungibleAirdropLists(
                transferFungibleAmounts, pendingFungibleAmounts, transfersNeedingAutoAssociation.size());
    }

    /**
     * Checks every {@link NftTransfer} from given transfer list and separate it in to two lists.
     * One containing transfers that should be added to pending airdrop state and the other list - transfers that should
     * be executed. The check is done by account's available auto associations slots and the existence of account-token
     * relation {@link #isPendingAirdrop(HandleContext, Account, TokenRelation)}
     *
     * @param context context
     * @param tokenId token id
     * @param transfers list of nft transfers
     * @return {@link NftAirdropLists} a record containing two lists - transfers to be added in pending state and transfers to be executed
     */
    public static NftAirdropLists separateNftTransfers(
            HandleContext context, TokenID tokenId, List transfers) {
        List transferNftList = new ArrayList<>();
        List pendingNftList = new ArrayList<>();
        Set transfersNeedingAutoAssociation = new HashSet<>();

        for (final var nftTransfer : transfers) {
            final var tokenRelStore = context.storeFactory().readableStore(ReadableTokenRelationStore.class);
            final var accountStore = context.storeFactory().readableStore(ReadableAccountStore.class);

            final var receiverId = nftTransfer.receiverAccountIDOrElse(AccountID.DEFAULT);
            // Treat a transfer to a missing account as an auto-creation attempt
            if (accountStore.isMissing(receiverId)) {
                transferNftList.add(nftTransfer);
                transfersNeedingAutoAssociation.add(receiverId);
                continue;
            }

            final var account =
                    getIfUsableForAliasedId(receiverId, accountStore, context.expiryValidator(), INVALID_ACCOUNT_ID);
            validateFalse(isSystemAccount(account), INVALID_RECEIVING_NODE_ACCOUNT);
            var tokenRel = tokenRelStore.get(receiverId, tokenId);
            if (isPendingAirdrop(context, account, tokenRel)) {
                pendingNftList.add(nftTransfer);
            } else {
                transferNftList.add(nftTransfer);
                // Any transfer that is with no explicitly associated token will need to be charged $0.1
                // So we charge $0.05 pending airdrop fee and $0.05 is charged in CryptoTransferHandler during
                // auto-association
                if (tokenRel == null) {
                    transfersNeedingAutoAssociation.add(receiverId);
                }
            }
        }
        return new NftAirdropLists(transferNftList, pendingNftList, transfersNeedingAutoAssociation.size());
    }

    /**
     * Checks if the airdrop to an account is pending airdrop. An airdrop will result into a pending airdrop, iuf
     * the receiver has not signed the transaction when receiverSigRequired is set to true, or if the
     * receiver is no yet associated to teh token but there are no open slots available. In other scenarios,
     * the airdrop is not pending and will be transferred.
     * @param context the context
     * @param receiver the receiver account
     * @param tokenRelation the token relation
     * @return boolean value indicating if the airdrop is pending
     */
    private static boolean isPendingAirdrop(
            @NonNull final HandleContext context,
            @NonNull final Account receiver,
            @Nullable final TokenRelation tokenRelation) {
        // Check if the receiver has signed the transaction. If not, it should result in a pending airdrop
        if (receiver.receiverSigRequired()) {
            var sigVerification = context.keyVerifier().verificationFor(requireNonNull(receiver.key()));
            if (sigVerification.failed()) {
                return true;
            }
        }
        return tokenRelation == null && isAutoAssociationLimitReached(receiver);
    }

    /**
     * Creates a {@link PendingAirdropId} for a fungible token.
     *
     * @param tokenId the ID of the token
     * @param sender the sender's account
     * @param receiver the receiver's account
     * @return {@link PendingAirdropId} for storing in the state
     */
    public static PendingAirdropId createFungibleTokenPendingAirdropId(
            TokenID tokenId, Account sender, Account receiver) {
        return PendingAirdropId.newBuilder()
                .receiverId(receiver.accountIdOrThrow())
                .senderId(sender.accountIdOrThrow())
                .fungibleTokenType(tokenId)
                .build();
    }

    /**
     * Creates a {@link PendingAirdropId} for a fungible token.
     *
     * @param tokenId the ID of the token
     * @param senderId the sender's account ID
     * @param receiverId the receiver's account ID
     * @return {@link PendingAirdropId} for storing in the state
     */
    public static PendingAirdropId createFungibleTokenPendingAirdropId(
            TokenID tokenId, AccountID senderId, AccountID receiverId) {
        return PendingAirdropId.newBuilder()
                .receiverId(receiverId)
                .senderId(senderId)
                .fungibleTokenType(tokenId)
                .build();
    }

    /**
     * Creates a {@link PendingAirdropId} for a non-fungible token.
     *
     * @param tokenId the ID of the token
     * @param serialNumber the serial number of the token
     * @param sender the sender's account
     * @param receiver the receiver's account
     * @return {@link PendingAirdropId} for storing in the state
     */
    public static PendingAirdropId createNftPendingAirdropId(
            TokenID tokenId, long serialNumber, Account sender, Account receiver) {
        var nftId =
                NftID.newBuilder().tokenId(tokenId).serialNumber(serialNumber).build();
        return PendingAirdropId.newBuilder()
                .receiverId(receiver.accountIdOrThrow())
                .senderId(sender.accountIdOrThrow())
                .nonFungibleToken(nftId)
                .build();
    }

    /**
     * Creates a {@link AccountPendingAirdrop} for a fungible token.
     *
     * @param pendingAirdropValue the amount of fungible token
     * @return {@link AccountPendingAirdrop} for storing in the state
     */
    public static AccountPendingAirdrop createFirstAccountPendingAirdrop(PendingAirdropValue pendingAirdropValue) {
        return createAccountPendingAirdrop(pendingAirdropValue, null);
    }

    /**
     * Creates a {@link AccountPendingAirdrop} for a fungible token.
     *
     * @param pendingAirdropValue the amount of fungible token
     * @return {@link AccountPendingAirdrop} for storing in the state
     */
    public static AccountPendingAirdrop createAccountPendingAirdrop(
            @Nullable final PendingAirdropValue pendingAirdropValue, @Nullable final PendingAirdropId next) {
        return AccountPendingAirdrop.newBuilder()
                .pendingAirdropValue(pendingAirdropValue)
                .nextAirdrop(next)
                .build();
    }

    /**
     * Creates a {@link PendingAirdropRecord} for externalizing pending airdrop records.
     *
     * @param pendingAirdropId the ID of the pending airdrop
     * @param pendingAirdropValue the value of the pending airdrop
     * @return {@link PendingAirdropRecord}
     */
    public static PendingAirdropRecord createPendingAirdropRecord(
            @NonNull final PendingAirdropId pendingAirdropId, @Nullable final PendingAirdropValue pendingAirdropValue) {
        return PendingAirdropRecord.newBuilder()
                .pendingAirdropId(pendingAirdropId)
                .pendingAirdropValue(pendingAirdropValue)
                .build();
    }

    /**
     * Returns if given account is a system account.
     *
     * @param account the account that need to be validated
     * @return boolean value indicating if the account is a system account
     */
    public static boolean isSystemAccount(@NonNull Account account) {
        requireNonNull(account);
        return account.accountIdOrThrow().accountNumOrThrow() <= LAST_RESERVED_SYSTEM_ACCOUNT;
    }

    private static boolean isAutoAssociationLimitReached(@NonNull final Account receiver) {
        return receiver.maxAutoAssociations() <= receiver.usedAutoAssociations()
                && receiver.maxAutoAssociations() != UNLIMITED_AUTOMATIC_ASSOCIATIONS;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy