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

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

/*
 * 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.CUSTOM_FEES_LIST_TOO_LONG;
import static com.hedera.hapi.node.base.ResponseCodeEnum.CUSTOM_SCHEDULE_ALREADY_HAS_NO_FEES;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_CUSTOM_FEE_COLLECTOR;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_TOKEN_ID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.TOKEN_HAS_NO_FEE_SCHEDULE_KEY;
import static com.hedera.node.app.hapi.fees.usage.SingletonEstimatorUtils.ESTIMATOR_UTILS;
import static com.hedera.node.app.hapi.fees.usage.token.TokenOpsUsage.LONG_BASIC_ENTITY_ID_SIZE;
import static com.hedera.node.app.spi.workflows.HandleException.validateTrue;
import static java.util.Collections.emptyList;
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.state.token.Token;
import com.hedera.hapi.node.token.TokenFeeScheduleUpdateTransactionBody;
import com.hedera.hapi.node.transaction.CustomFee;
import com.hedera.hapi.node.transaction.TransactionBody;
import com.hedera.node.app.hapi.fees.usage.token.TokenOpsUsage;
import com.hedera.node.app.hapi.utils.CommonPbjConverters;
import com.hedera.node.app.service.token.ReadableAccountStore;
import com.hedera.node.app.service.token.ReadableTokenRelationStore;
import com.hedera.node.app.service.token.ReadableTokenStore;
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.records.TokenBaseStreamBuilder;
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.TokensConfig;
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_FEE_SCHEDULE_UPDATE}.
 */
@Singleton
public class TokenFeeScheduleUpdateHandler implements TransactionHandler {
    private final CustomFeesValidator customFeesValidator;

    /**
     * Default constructor for injection.
     * @param customFeesValidator the custom fees validator
     */
    @Inject
    public TokenFeeScheduleUpdateHandler(@NonNull final CustomFeesValidator customFeesValidator) {
        requireNonNull(customFeesValidator);
        this.customFeesValidator = customFeesValidator;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public void preHandle(@NonNull final PreHandleContext context) throws PreCheckException {
        requireNonNull(context);
        pureChecks(context.body());

        final var op = context.body().tokenFeeScheduleUpdateOrThrow();
        final var tokenId = op.tokenIdOrElse(TokenID.DEFAULT);

        final var tokenStore = context.createStore(ReadableTokenStore.class);
        final var tokenMetadata = tokenStore.getTokenMeta(tokenId);
        if (tokenMetadata == null) throw new PreCheckException(INVALID_TOKEN_ID);
        if (tokenMetadata.hasFeeScheduleKey()) {
            context.requireKey(tokenMetadata.feeScheduleKey());
            for (final var customFee : op.customFees()) {
                final var collector = customFee.feeCollectorAccountIdOrElse(AccountID.DEFAULT);
                context.requireKeyIfReceiverSigRequired(collector, INVALID_CUSTOM_FEE_COLLECTOR);
            }
        }
        // we do not set a failure status if a fee schedule key is not present for the token,
        // we choose to fail with TOKEN_HAS_NO_FEE_SCHEDULE_KEY in the handle() method
    }

    /**
     * Handles a transaction with {@link HederaFunctionality#TOKEN_FEE_SCHEDULE_UPDATE}.
     * @param context the context of the transaction
     */
    @Override
    public void handle(@NonNull final HandleContext context) {
        requireNonNull(context);

        final var txn = context.body();

        // get the latest configuration
        final var config = context.configuration().getConfigData(TokensConfig.class);
        final var op = txn.tokenFeeScheduleUpdateOrThrow();

        // validate checks in handle
        final var storeFactory = context.storeFactory();
        final var tokenStore = storeFactory.writableStore(WritableTokenStore.class);
        final var token = validateSemantics(op, tokenStore, config);

        // create readable stores from the context
        final var readableAccountStore = storeFactory.readableStore(ReadableAccountStore.class);
        final var readableTokenRelsStore = storeFactory.readableStore(ReadableTokenRelationStore.class);

        if (token.customFees().isEmpty() && op.customFees().isEmpty()) {
            throw new HandleException(CUSTOM_SCHEDULE_ALREADY_HAS_NO_FEES);
        }
        // validate custom fees before committing
        customFeesValidator.validateForFeeScheduleUpdate(
                token, readableAccountStore, readableTokenRelsStore, tokenStore, op.customFees());
        // set the custom fees on token
        final var copy = token.copyBuilder().customFees(op.customFees());
        // add token to the modifications map
        tokenStore.put(copy.build());

        final var record = context.savepointStack().getBaseBuilder(TokenBaseStreamBuilder.class);
        record.tokenType(token.tokenType());
    }

    /**
     * Validate semantics of the transaction in handle call. This method is called before the
     * transaction is handled.
     * @param op the transaction body
     * @param tokenStore the token store
     * @param config the token service config
     * @return the token
     */
    private Token validateSemantics(
            @NonNull final TokenFeeScheduleUpdateTransactionBody op,
            @NonNull final WritableTokenStore tokenStore,
            @NonNull final TokensConfig config) {
        var token = TokenHandlerHelper.getIfUsable(op.tokenIdOrElse(TokenID.DEFAULT), tokenStore);
        validateTrue(token.hasFeeScheduleKey(), TOKEN_HAS_NO_FEE_SCHEDULE_KEY);
        validateTrue(op.customFees().size() <= config.maxCustomFeesAllowed(), CUSTOM_FEES_LIST_TOO_LONG);
        return token;
    }

    @Override
    public void pureChecks(@NonNull final TransactionBody txn) throws PreCheckException {
        final var op = txn.tokenFeeScheduleUpdateOrThrow();
        if (!op.hasTokenId()) {
            throw new PreCheckException(INVALID_TOKEN_ID);
        }
    }

    @NonNull
    @Override
    public Fees calculateFees(@NonNull final FeeContext feeContext) {
        final var body = feeContext.body();
        final var op = body.tokenFeeScheduleUpdateOrThrow();
        final var token = feeContext.readableStore(ReadableTokenStore.class).get(op.tokenIdOrThrow());
        final var tokenOpsUsage = new TokenOpsUsage();
        var newFees = op.customFees();

        // Ensure no null values for denominatingTokenId
        newFees = newFees.stream()
                .map(fee -> {
                    if (fee.hasFixedFee() && fee.fixedFee().denominatingTokenId() == null) {
                        return fee.copyBuilder()
                                .fixedFee(fee.fixedFee()
                                        .copyBuilder()
                                        .denominatingTokenId(TokenID.DEFAULT)
                                        .build())
                                .build();
                    }
                    return fee;
                })
                .toList();

        final var newReprBytes = tokenOpsUsage.bytesNeededToRepr(
                newFees.stream().map(CommonPbjConverters::fromPbj).toList());
        final var effConsTime =
                body.transactionIDOrThrow().transactionValidStartOrThrow().seconds();
        final var lifetime = Math.max(0, token == null ? 0 : token.expirationSecond() - effConsTime);
        final List customFees = token == null ? emptyList() : token.customFees();

        final var existingFeeReprBytes = currentFeeScheduleSize(customFees, tokenOpsUsage);
        final var rbsDelta = ESTIMATOR_UTILS.changeInBsUsage(existingFeeReprBytes, lifetime, newReprBytes, lifetime);
        return feeContext
                .feeCalculatorFactory()
                .feeCalculator(SubType.DEFAULT)
                .addBytesPerTransaction(LONG_BASIC_ENTITY_ID_SIZE + newReprBytes)
                .addRamByteSeconds(rbsDelta)
                .calculate();
    }

    private int currentFeeScheduleSize(List feeSchedule, final TokenOpsUsage tokenOpsUsage) {
        int numFixedHbarFees = 0;
        int numFixedHtsFees = 0;
        int numFractionalFees = 0;
        int numRoyaltyNoFallbackFees = 0;
        int numRoyaltyHtsFallbackFees = 0;
        int numRoyaltyHbarFallbackFees = 0;
        for (var fee : feeSchedule) {
            if (fee.fee().kind().equals(CustomFee.FeeOneOfType.FIXED_FEE)) {
                if (fee.fixedFee().hasDenominatingTokenId()) {
                    numFixedHtsFees++;
                } else {
                    numFixedHbarFees++;
                }
            } else if (fee.fee().kind().equals(CustomFee.FeeOneOfType.FRACTIONAL_FEE)) {
                numFractionalFees++;
            } else {
                final var royaltyFee = fee.royaltyFee();
                final var fallbackFee = royaltyFee.fallbackFee();
                if (fallbackFee != null) {
                    if (fallbackFee.hasDenominatingTokenId()) {
                        numRoyaltyHtsFallbackFees++;
                    } else {
                        numRoyaltyHbarFallbackFees++;
                    }
                } else {
                    numRoyaltyNoFallbackFees++;
                }
            }
        }
        return tokenOpsUsage.bytesNeededToRepr(
                numFixedHbarFees,
                numFixedHtsFees,
                numFractionalFees,
                numRoyaltyNoFallbackFees,
                numRoyaltyHtsFallbackFees,
                numRoyaltyHbarFallbackFees);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy