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

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

There is a newer version: 0.57.3
Show newest version
/*
 * Copyright (C) 2022-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.INVALID_ACCOUNT_ID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_AUTORENEW_ACCOUNT;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_CUSTOM_FEE_COLLECTOR;
import static com.hedera.hapi.node.base.ResponseCodeEnum.MAX_ENTITIES_IN_PRICE_REGIME_HAVE_BEEN_CREATED;
import static com.hedera.node.app.hapi.fees.usage.SingletonUsageProperties.USAGE_PROPERTIES;
import static com.hedera.node.app.hapi.fees.usage.token.TokenOpsUsageUtils.TOKEN_OPS_USAGE_UTILS;
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.TokenHandlerHelper.verifyNotEmptyKey;
import static com.hedera.node.app.service.token.impl.validators.CustomFeesValidator.SENTINEL_TOKEN_ID;
import static com.hedera.node.app.spi.validation.ExpiryMeta.NA;
import static com.hedera.node.app.spi.workflows.HandleException.validateTrue;
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.SubType;
import com.hedera.hapi.node.base.TokenID;
import com.hedera.hapi.node.base.TokenType;
import com.hedera.hapi.node.state.token.Token;
import com.hedera.hapi.node.token.TokenCreateTransactionBody;
import com.hedera.hapi.node.transaction.CustomFee;
import com.hedera.hapi.node.transaction.TransactionBody;
import com.hedera.node.app.hapi.utils.CommonPbjConverters;
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.WritableTokenRelationStore;
import com.hedera.node.app.service.token.impl.WritableTokenStore;
import com.hedera.node.app.service.token.impl.util.TokenHandlerHelper;
import com.hedera.node.app.service.token.impl.validators.CustomFeesValidator;
import com.hedera.node.app.service.token.impl.validators.TokenCreateValidator;
import com.hedera.node.app.service.token.records.TokenCreateStreamBuilder;
import com.hedera.node.app.spi.fees.FeeContext;
import com.hedera.node.app.spi.fees.Fees;
import com.hedera.node.app.spi.validation.ExpiryMeta;
import com.hedera.node.app.spi.workflows.HandleContext;
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.EntitiesConfig;
import com.hedera.node.config.data.TokensConfig;
import com.hedera.pbj.runtime.io.buffer.Bytes;
import edu.umd.cs.findbugs.annotations.NonNull;
import java.util.List;
import javax.inject.Inject;
import javax.inject.Singleton;

/**
 * This class contains all workflow-related functionality regarding {@link
 * HederaFunctionality#TOKEN_CREATE}.
 */
@Singleton
public class TokenCreateHandler extends BaseTokenHandler implements TransactionHandler {
    private final CustomFeesValidator customFeesValidator;
    private final TokenCreateValidator tokenCreateValidator;

    /**
     * Default constructor for injection.
     * @param customFeesValidator custom fees validator
     * @param tokenCreateValidator token create validator
     */
    @Inject
    public TokenCreateHandler(
            @NonNull final CustomFeesValidator customFeesValidator,
            @NonNull final TokenCreateValidator tokenCreateValidator) {
        requireNonNull(customFeesValidator);
        requireNonNull(tokenCreateValidator);

        this.customFeesValidator = customFeesValidator;
        this.tokenCreateValidator = tokenCreateValidator;
    }

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

        final var tokenCreateTxnBody = txn.tokenCreationOrThrow();
        if (tokenCreateTxnBody.hasTreasury()) {
            final var treasuryId = tokenCreateTxnBody.treasuryOrThrow();
            // Note: should throw INVALID_TREASURY_ACCOUNT_FOR_TOKEN after modularization
            context.requireKeyOrThrow(treasuryId, INVALID_ACCOUNT_ID);
        }
        if (tokenCreateTxnBody.hasAutoRenewAccount()) {
            final var autoRenewalAccountId = tokenCreateTxnBody.autoRenewAccountOrThrow();
            context.requireKeyOrThrow(autoRenewalAccountId, INVALID_AUTORENEW_ACCOUNT);
        }
        if (tokenCreateTxnBody.hasAdminKey()) {
            context.requireKey(tokenCreateTxnBody.adminKeyOrThrow());
        }
        final var customFees = tokenCreateTxnBody.customFees();
        addCustomFeeCollectorKeys(context, customFees);
    }

    @Override
    public void pureChecks(@NonNull final TransactionBody txn) throws PreCheckException {
        tokenCreateValidator.pureChecks(txn.tokenCreationOrThrow());
    }

    @Override
    public void handle(@NonNull final HandleContext context) {
        requireNonNull(context);
        final var txn = context.body();
        final var op = txn.tokenCreationOrThrow();
        // Create or get needed config and stores
        final var tokensConfig = context.configuration().getConfigData(TokensConfig.class);
        final var storeFactory = context.storeFactory();
        final var accountStore = storeFactory.writableStore(WritableAccountStore.class);
        final var tokenStore = storeFactory.writableStore(WritableTokenStore.class);
        final var tokenRelationStore = storeFactory.writableStore(WritableTokenRelationStore.class);

        final var recordBuilder = context.savepointStack().getBaseBuilder(TokenCreateStreamBuilder.class);

        /* Validate if the current token can be created */
        validateTrue(
                tokenStore.sizeOfState() + 1 <= tokensConfig.maxNumber(),
                MAX_ENTITIES_IN_PRICE_REGIME_HAVE_BEEN_CREATED);

        // validate fields in the transaction body that involves checking with
        // dynamic properties or state.
        final var resolvedExpiryMeta = validateSemantics(context, accountStore, op, tokensConfig);

        // build a new token
        final var newTokenNum = context.entityNumGenerator().newEntityNum();
        final var newTokenId = TokenID.newBuilder().tokenNum(newTokenNum).build();
        final var newToken = buildToken(newTokenNum, op, resolvedExpiryMeta);

        // validate custom fees and get back list of fees with created token denomination
        final var feesSetNeedingCollectorAutoAssociation = customFeesValidator.validateForCreation(
                newToken, accountStore, tokenRelationStore, tokenStore, op.customFees());
        // Put token into modifications map
        tokenStore.put(newToken);
        // associate token with treasury and collector ids of custom fees whose token denomination
        // is set to sentinel value
        associateAccounts(
                context,
                newToken,
                accountStore,
                tokenRelationStore,
                feesSetNeedingCollectorAutoAssociation,
                recordBuilder);

        // Since we have associated treasury and needed fee collector accounts in the previous step,
        // this relation must exist
        final var treasuryRel = requireNonNull(tokenRelationStore.get(op.treasuryOrThrow(), newTokenId));
        if (op.initialSupply() > 0) {
            // This keeps modified token with minted balance into modifications in token store
            mintFungible(newToken, treasuryRel, op.initialSupply(), accountStore, tokenStore, tokenRelationStore);
        }
        // Increment treasury's title count
        final var treasuryAccount = requireNonNull(accountStore.get(treasuryRel.accountIdOrThrow()));
        accountStore.put(treasuryAccount
                .copyBuilder()
                .numberTreasuryTitles(treasuryAccount.numberTreasuryTitles() + 1)
                .build());

        // Update record with newly created token id
        recordBuilder.tokenID(newTokenId);
        recordBuilder.tokenType(newToken.tokenType());
    }

    /**
     * Associate treasury account and the collector accounts of custom fees whose token denomination
     * is set to sentinel value, to use denomination as newly created token.
     *
     * @param newToken newly created token
     * @param accountStore account store
     * @param tokenRelStore token relation store
     * @param requireCollectorAutoAssociation set of custom fees whose token denomination is set to sentinel value
     * @param recordBuilder
     */
    private void associateAccounts(
            final HandleContext context,
            final Token newToken,
            final WritableAccountStore accountStore,
            @NonNull final WritableTokenRelationStore tokenRelStore,
            final List requireCollectorAutoAssociation,
            final TokenCreateStreamBuilder recordBuilder) {
        final var tokensConfig = context.configuration().getConfigData(TokensConfig.class);
        final var entitiesConfig = context.configuration().getConfigData(EntitiesConfig.class);

        // This should exist as it is validated in validateSemantics
        final var treasury = accountStore.get(newToken.treasuryAccountId());
        // Validate if token relation can be created between treasury and new token
        // If this succeeds, create and link token relation.
        tokenCreateValidator.validateAssociation(entitiesConfig, tokensConfig, treasury, newToken, tokenRelStore);
        createAndLinkTokenRels(treasury, List.of(newToken), accountStore, tokenRelStore);
        recordBuilder.addAutomaticTokenAssociation(asTokenAssociation(newToken.tokenId(), treasury.accountId()));

        for (final var customFee : requireCollectorAutoAssociation) {
            // This should exist as it is validated in validateSemantics
            final var collector = accountStore.get(customFee.feeCollectorAccountIdOrThrow());
            if (treasury.accountId().equals(collector.accountId())) {
                continue;
            }

            // Ensure no duplicate relations are created
            final var existingTokenRel = tokenRelStore.get(collector.accountId(), newToken.tokenId());
            if (existingTokenRel != null) {
                continue;
            }

            // Validate if token relation can be created between collector and new token
            // If this succeeds, create and link token relation.
            tokenCreateValidator.validateAssociation(entitiesConfig, tokensConfig, collector, newToken, tokenRelStore);
            createAndLinkTokenRels(collector, List.of(newToken), accountStore, tokenRelStore);
            recordBuilder.addAutomaticTokenAssociation(asTokenAssociation(newToken.tokenId(), collector.accountId()));
        }
    }

    /**
     * Create a new token with the given parameters.
     *
     * @param newTokenNum new token number
     * @param op token creation transaction body
     * @param resolvedExpiryMeta resolved expiry meta
     * @return newly created token
     */
    private Token buildToken(
            final long newTokenNum, final TokenCreateTransactionBody op, final ExpiryMeta resolvedExpiryMeta) {
        return new Token(
                asToken(newTokenNum),
                op.name(),
                op.symbol(),
                op.decimals(),
                0, // is this correct ?
                op.treasury(),
                op.adminKey(),
                op.kycKey(),
                op.freezeKey(),
                op.wipeKey(),
                op.supplyKey(),
                op.feeScheduleKey(),
                op.pauseKey(),
                0,
                false,
                op.tokenType(),
                op.supplyType(),
                resolvedExpiryMeta.autoRenewAccountId(),
                // We want to return 0 instead of ExpiryMeta.NA when querying this token's info
                resolvedExpiryMeta.hasAutoRenewPeriod() ? resolvedExpiryMeta.autoRenewPeriod() : 0L,
                resolvedExpiryMeta.expiry(),
                op.memo(),
                op.maxSupply(),
                false,
                op.freezeDefault(),
                false,
                modifyCustomFeesWithSentinelValues(op.customFees(), newTokenNum),
                op.metadata(),
                op.metadataKey());
    }

    /**
     * Modify the custom fees with the newly created token number as the token denomination.
     * For any custom fixed fees that has 0.0.0 as denominating tokenId, it should be changed
     * to the newly created token number before setting it to the token.
     *
     * @param customFees list of custom fees
     * @param newTokenNum newly created token number
     * @return modified custom fees
     */
    private List modifyCustomFeesWithSentinelValues(
            final List customFees, final long newTokenNum) {
        return customFees.stream()
                .map(fee -> {
                    if (fee.hasFixedFee()
                            && fee.fixedFeeOrThrow().hasDenominatingTokenId()
                            && fee.fixedFeeOrThrow()
                                    .denominatingTokenIdOrThrow()
                                    .equals(SENTINEL_TOKEN_ID)) {
                        final var newTokenId =
                                TokenID.newBuilder().tokenNum(newTokenNum).build();
                        final var modifiedFixedFee = fee.fixedFeeOrThrow()
                                .copyBuilder()
                                .denominatingTokenId(newTokenId)
                                .build();
                        // Assign the modified fee back to the original fee object
                        return fee.copyBuilder().fixedFee(modifiedFixedFee).build();
                    }
                    return fee; // Return unmodified fee if conditions are not met
                })
                .toList();
    }

    /**
     * Get the expiry metadata for the token to be created from the transaction body.
     *
     * @param op token creation transaction body
     * @return given expiry metadata
     */
    private ExpiryMeta getExpiryMeta(@NonNull final TokenCreateTransactionBody op) {
        final var impliedExpiry = op.hasExpiry() ? op.expiry().seconds() : NA;

        return new ExpiryMeta(
                impliedExpiry,
                op.hasAutoRenewPeriod() ? op.autoRenewPeriod().seconds() : NA,
                // Shard and realm will be ignored if num is NA
                op.autoRenewAccount());
    }

    /**
     * Validate the semantics of the token creation transaction body, that involves checking with
     * dynamic properties or state.
     *
     * @param context handle context
     * @param accountStore account store
     * @param op token creation transaction body
     * @param config tokens configuration
     * @return resolved expiry metadata
     */
    private ExpiryMeta validateSemantics(
            @NonNull final HandleContext context,
            @NonNull final ReadableAccountStore accountStore,
            @NonNull final TokenCreateTransactionBody op,
            @NonNull final TokensConfig config) {
        requireNonNull(context);
        requireNonNull(accountStore);
        requireNonNull(op);
        requireNonNull(config);

        // validate different token create fields
        tokenCreateValidator.validate(context, accountStore, op, config);

        // validate expiration and auto-renew account if present
        final var givenExpiryMeta = getExpiryMeta(op);
        final var resolvedExpiryMeta = context.expiryValidator()
                .resolveCreationAttempt(false, givenExpiryMeta, HederaFunctionality.TOKEN_CREATE);

        // validate auto-renew account exists
        if (resolvedExpiryMeta.hasAutoRenewAccountId()) {
            TokenHandlerHelper.getIfUsableForAutoRenew(
                    resolvedExpiryMeta.autoRenewAccountId(),
                    accountStore,
                    context.expiryValidator(),
                    INVALID_AUTORENEW_ACCOUNT);
        }
        return resolvedExpiryMeta;
    }

    /* --------------- Helper methods --------------- */

    /**
     * Validates the collector key from the custom fees.
     *
     * @param context given context
     * @param customFeesList list with the custom fees
     */
    private void addCustomFeeCollectorKeys(
            @NonNull final PreHandleContext context, @NonNull final List customFeesList)
            throws PreCheckException {

        for (final var customFee : customFeesList) {
            final var collector = customFee.feeCollectorAccountIdOrElse(AccountID.DEFAULT);

            // Verify that the collector either has a non-empty alias OR a mutable key
            final var acctStore = context.createStore(ReadableAccountStore.class);
            final var collectorAcct = acctStore.getAccountById(collector);
            if (collectorAcct != null
                    && (collectorAcct.alias() == null || Bytes.EMPTY.equals(collectorAcct.alias()))
                    && (collectorAcct.hasKey())) {
                verifyNotEmptyKey(collectorAcct.key(), INVALID_CUSTOM_FEE_COLLECTOR);
            }

            /* A fractional fee collector and a collector for a fixed fee denominated
            in the units of the newly created token both must always sign a TokenCreate,
            since these are automatically associated to the newly created token. */
            if (customFee.hasFixedFee()) {
                final var fixedFee = customFee.fixedFeeOrThrow();
                final var alwaysAdd = fixedFee.hasDenominatingTokenId()
                        && fixedFee.denominatingTokenIdOrThrow().tokenNum() == 0L;
                addAccount(context, collector, alwaysAdd);
            } else if (customFee.hasFractionalFee()) {
                context.requireKeyOrThrow(collector, INVALID_CUSTOM_FEE_COLLECTOR);
            } else if (customFee.hasRoyaltyFee()) {
                // TODO: Need to validate if this is actually needed
                final var royaltyFee = customFee.royaltyFeeOrThrow();
                var alwaysAdd = false;
                if (royaltyFee.hasFallbackFee()) {
                    final var fFee = royaltyFee.fallbackFeeOrThrow();
                    alwaysAdd = fFee.hasDenominatingTokenId()
                            && fFee.denominatingTokenIdOrThrow().tokenNum() == 0;
                }
                addAccount(context, collector, alwaysAdd);
            }
        }
    }

    /**
     * Signs the metadata or adds failure status.
     *
     * @param context given context
     * @param collector the ID of the collector
     * @param alwaysAdd if true, will always add the key
     */
    private void addAccount(
            @NonNull final PreHandleContext context, @NonNull final AccountID collector, final boolean alwaysAdd)
            throws PreCheckException {
        if (alwaysAdd) {
            context.requireKeyOrThrow(collector, INVALID_CUSTOM_FEE_COLLECTOR);
        } else {
            context.requireKeyIfReceiverSigRequired(collector, INVALID_CUSTOM_FEE_COLLECTOR);
        }
    }

    @NonNull
    @Override
    public Fees calculateFees(@NonNull final FeeContext feeContext) {
        requireNonNull(feeContext);
        final var body = feeContext.body();
        final var meta = TOKEN_OPS_USAGE_UTILS.tokenCreateUsageFrom(CommonPbjConverters.fromPbj(body));
        final var op = body.tokenCreationOrThrow();
        final var type = op.tokenType();

        final long tokenSizes = TOKEN_ENTITY_SIZES.bytesUsedToRecordTokenTransfers(
                        meta.getNumTokens(), meta.getFungibleNumTransfers(), meta.getNftsTransfers())
                * USAGE_PROPERTIES.legacyReceiptStorageSecs();

        return feeContext
                .feeCalculatorFactory()
                .feeCalculator(tokenSubTypeFrom(
                        type, op.hasFeeScheduleKey() || !op.customFees().isEmpty()))
                .addBytesPerTransaction(meta.getBaseSize())
                .addRamByteSeconds(tokenSizes)
                .addNetworkRamByteSeconds(meta.getNetworkRecordRb() * USAGE_PROPERTIES.legacyReceiptStorageSecs())
                .addRamByteSeconds((meta.getBaseSize() + meta.getCustomFeeScheduleSize()) * meta.getLifeTime())
                .calculate();
    }

    /**
     * Get the token subtype to be used for the fees calculation.
     * @param tokenType the token type
     * @param hasCustomFees if the token has custom fees
     * @return the token subtype
     */
    public static SubType tokenSubTypeFrom(final TokenType tokenType, boolean hasCustomFees) {
        return switch (tokenType) {
            case FUNGIBLE_COMMON -> hasCustomFees
                    ? SubType.TOKEN_FUNGIBLE_COMMON_WITH_CUSTOM_FEES
                    : SubType.TOKEN_FUNGIBLE_COMMON;
            case NON_FUNGIBLE_UNIQUE -> hasCustomFees
                    ? SubType.TOKEN_NON_FUNGIBLE_UNIQUE_WITH_CUSTOM_FEES
                    : SubType.TOKEN_NON_FUNGIBLE_UNIQUE;
        };
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy