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

com.hedera.node.app.service.token.impl.handlers.CryptoCreateHandler 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.ACCOUNT_DELETED;
import static com.hedera.hapi.node.base.ResponseCodeEnum.ALIAS_ALREADY_ASSIGNED;
import static com.hedera.hapi.node.base.ResponseCodeEnum.BAD_ENCODING;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INSUFFICIENT_PAYER_BALANCE;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_ACCOUNT_ID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_ALIAS_KEY;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_INITIAL_BALANCE;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_MAX_AUTO_ASSOCIATIONS;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_PAYER_ACCOUNT_ID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_RECEIVE_RECORD_THRESHOLD;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_RENEWAL_PERIOD;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_SEND_RECORD_THRESHOLD;
import static com.hedera.hapi.node.base.ResponseCodeEnum.KEY_NOT_PROVIDED;
import static com.hedera.hapi.node.base.ResponseCodeEnum.KEY_REQUIRED;
import static com.hedera.hapi.node.base.ResponseCodeEnum.MAX_ENTITIES_IN_PRICE_REGIME_HAVE_BEEN_CREATED;
import static com.hedera.hapi.node.base.ResponseCodeEnum.NOT_SUPPORTED;
import static com.hedera.hapi.node.base.ResponseCodeEnum.PROXY_ACCOUNT_ID_FIELD_IS_DEPRECATED;
import static com.hedera.node.app.hapi.fees.usage.SingletonUsageProperties.USAGE_PROPERTIES;
import static com.hedera.node.app.hapi.fees.usage.crypto.CryptoOpsUsage.CREATE_SLOT_MULTIPLIER;
import static com.hedera.node.app.hapi.fees.usage.crypto.entities.CryptoEntitySizes.CRYPTO_ENTITY_SIZES;
import static com.hedera.node.app.hapi.utils.fee.FeeBuilder.BASIC_ENTITY_ID_SIZE;
import static com.hedera.node.app.hapi.utils.fee.FeeBuilder.BOOL_SIZE;
import static com.hedera.node.app.hapi.utils.fee.FeeBuilder.INT_SIZE;
import static com.hedera.node.app.hapi.utils.fee.FeeBuilder.LONG_SIZE;
import static com.hedera.node.app.hapi.utils.fee.FeeBuilder.getAccountKeyStorageSize;
import static com.hedera.node.app.service.token.AliasUtils.asKeyFromAlias;
import static com.hedera.node.app.service.token.AliasUtils.asKeyFromAliasPreCheck;
import static com.hedera.node.app.service.token.AliasUtils.extractEvmAddress;
import static com.hedera.node.app.service.token.AliasUtils.isEntityNumAlias;
import static com.hedera.node.app.service.token.AliasUtils.isKeyAlias;
import static com.hedera.node.app.service.token.AliasUtils.isOfEvmAddressSize;
import static com.hedera.node.app.service.token.impl.handlers.BaseTokenHandler.UNLIMITED_AUTOMATIC_ASSOCIATIONS;
import static com.hedera.node.app.service.token.impl.handlers.staking.StakingUtilities.NOT_REWARDED_SINCE_LAST_STAKING_META_CHANGE;
import static com.hedera.node.app.service.token.impl.handlers.staking.StakingUtilities.NO_STAKE_PERIOD_START;
import static com.hedera.node.app.service.token.impl.util.TokenHandlerHelper.getIfUsable;
import static com.hedera.node.app.spi.key.KeyUtils.IMMUTABILITY_SENTINEL_KEY;
import static com.hedera.node.app.spi.key.KeyUtils.isEmpty;
import static com.hedera.node.app.spi.key.KeyUtils.isValid;
import static com.hedera.node.app.spi.workflows.HandleException.validateFalse;
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.Duration;
import com.hedera.hapi.node.base.HederaFunctionality;
import com.hedera.hapi.node.base.Key;
import com.hedera.hapi.node.base.SubType;
import com.hedera.hapi.node.state.token.Account;
import com.hedera.hapi.node.token.CryptoCreateTransactionBody;
import com.hedera.hapi.node.token.CryptoUpdateTransactionBody;
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.validators.CryptoCreateValidator;
import com.hedera.node.app.service.token.impl.validators.StakingValidator;
import com.hedera.node.app.service.token.records.CryptoCreateStreamBuilder;
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.app.spi.workflows.record.StreamBuilder;
import com.hedera.node.config.data.AccountsConfig;
import com.hedera.node.config.data.CryptoCreateWithAliasConfig;
import com.hedera.node.config.data.EntitiesConfig;
import com.hedera.node.config.data.HederaConfig;
import com.hedera.node.config.data.LedgerConfig;
import com.hedera.node.config.data.StakingConfig;
import com.hedera.node.config.data.TokensConfig;
import com.hedera.pbj.runtime.io.buffer.Bytes;
import edu.umd.cs.findbugs.annotations.NonNull;
import javax.inject.Inject;
import javax.inject.Singleton;

/**
 * This class contains all workflow-related functionality regarding {@link HederaFunctionality#CRYPTO_CREATE}. A
 * "crypto create" is the creation of a new account on the Hedera network.
 */
@Singleton
public class CryptoCreateHandler extends BaseCryptoHandler implements TransactionHandler {
    private static final TransactionBody UPDATE_TXN_BODY_BUILDER = TransactionBody.newBuilder()
            .cryptoUpdateAccount(CryptoUpdateTransactionBody.newBuilder()
                    .key(Key.newBuilder().ecdsaSecp256k1(Bytes.EMPTY).build()))
            .build();

    private final CryptoCreateValidator cryptoCreateValidator;

    /**
     * Constructs a {@link CryptoCreateHandler} with the given {@link CryptoCreateValidator} and {@link StakingValidator}.
     * @param cryptoCreateValidator the validator for the crypto create transaction
     */
    @Inject
    public CryptoCreateHandler(@NonNull final CryptoCreateValidator cryptoCreateValidator) {
        this.cryptoCreateValidator =
                requireNonNull(cryptoCreateValidator, "The supplied argument 'cryptoCreateValidator' must not be null");
    }

    @Override
    public void pureChecks(@NonNull final TransactionBody txn) throws PreCheckException {
        final var op = txn.cryptoCreateAccountOrThrow();
        // Note: validation lives here for now but should take place in handle in the future
        validateTruePreCheck(op.hasAutoRenewPeriod(), INVALID_RENEWAL_PERIOD);
        validateTruePreCheck(op.autoRenewPeriodOrThrow().seconds() >= 0, INVALID_RENEWAL_PERIOD);
        if (op.hasShardID()) {
            validateTruePreCheck(op.shardIDOrThrow().shardNum() == 0, INVALID_ACCOUNT_ID);
        }
        if (op.hasRealmID()) {
            validateTruePreCheck(op.realmIDOrThrow().realmNum() == 0, INVALID_ACCOUNT_ID);
        }
        // HIP 904 now allows for unlimited auto-associations
        validateTruePreCheck(
                op.maxAutomaticTokenAssociations() >= UNLIMITED_AUTOMATIC_ASSOCIATIONS, INVALID_MAX_AUTO_ASSOCIATIONS);
        validateTruePreCheck(op.initialBalance() >= 0L, INVALID_INITIAL_BALANCE);
        // FUTURE: should this return SEND_RECORD_THRESHOLD_FIELD_IS_DEPRECATED
        validateTruePreCheck(op.sendRecordThreshold() >= 0L, INVALID_SEND_RECORD_THRESHOLD);
        // FUTURE: should this return RECEIVE_RECORD_THRESHOLD_FIELD_IS_DEPRECATED
        validateTruePreCheck(op.receiveRecordThreshold() >= 0L, INVALID_RECEIVE_RECORD_THRESHOLD);
        validateTruePreCheck(
                op.proxyAccountIDOrElse(AccountID.DEFAULT).equals(AccountID.DEFAULT),
                PROXY_ACCOUNT_ID_FIELD_IS_DEPRECATED);
        // sendRecordThreshold, receiveRecordThreshold and proxyAccountID are deprecated. So no need to check them.
        validateFalsePreCheck(op.hasProxyAccountID(), PROXY_ACCOUNT_ID_FIELD_IS_DEPRECATED);
        // You can never set the alias to be an "entity num alias" (sometimes called "long-zero").
        final var alias = op.alias();
        validateFalsePreCheck(isEntityNumAlias(alias), INVALID_ALIAS_KEY);
        // The alias, if set, must be of EVM address size, or it must be a valid key.
        validateTruePreCheck(alias.length() == 0 || isOfEvmAddressSize(alias) || isKeyAlias(alias), INVALID_ALIAS_KEY);
        // There must be a key provided, and it must not be empty, unless in one very particular case, where the
        // transactionID is null. This code is very particular about which error code to throw in various cases.
        // FUTURE: Clean up the error codes to be consistent.
        final var key = op.key();
        final var isInternal = !txn.hasTransactionID();
        final var keyIsEmpty = isEmpty(key);
        if (!isInternal && keyIsEmpty) {
            if (key == null) {
                throw new PreCheckException(alias.length() > 0 ? INVALID_ALIAS_KEY : KEY_REQUIRED);
            } else if (key.hasThresholdKey() || key.hasKeyList()) {
                throw new PreCheckException(KEY_REQUIRED);
            } else {
                throw new PreCheckException(BAD_ENCODING);
            }
        }
        validateTruePreCheck(key != null, KEY_NOT_PROVIDED);
    }

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

        // SPEC(HIP-583): If the alias is set during CryptoCreate, then ownership of the alias must be proven by the
        // caller by signing the transaction with the key represented by the alias. It is not necessary for the key on
        // the newly created account to match the alias, but it is required for the transaction to be signed by the key
        // represented by the alias.
        final var alias = op.alias(); // will never be null
        if (alias.length() > 0) {
            // We will only require additional signing if the alias (whether evm-address or key-encoded) is different
            // from the payer key (since checking the payer key is done in every case)
            final var payerKey = context.payerKey();
            // We already know from pure checks that this isn't a long-zero address, and that the alias is either
            // an EVM address alias, or it is a key-alias. Since checking the EVM address alias is cheap, and checking
            // the key alias is more expensive, we'll check for the EVM address alias case first
            if (isOfEvmAddressSize(alias)) {
                // Convert they payer key to an EVM Address, and only gather the alias if the payer key cannot be
                // converted into the same EVM address as the alias
                final var payerEvmAddress = extractEvmAddress(payerKey);
                final var accountKeyEvmAddress = extractEvmAddress(op.keyOrThrow());
                if (!alias.equals(payerEvmAddress) && !alias.equals(accountKeyEvmAddress)) {
                    // Verify there is a signature that matches the EVM address
                    context.requireSignatureForHollowAccountCreation(alias);
                }
            } else if (context.isSyntheticTransaction()) {
                // We allow key-based aliases on _synthetic_ (non-user created) transactions, such as when an account is
                // automatically created based on a crypto transfer to a new account by alias.
                //
                // Try to extract the key. We already know from pure checks that if it isn't an evm address then it
                // must be a key account. So this extraction must work.
                final var aliasAsKey = asKeyFromAliasPreCheck(alias);
                if (!aliasAsKey.equals(payerKey)) {
                    context.requireKey(aliasAsKey);
                }
            } else {
                // We do NOT allow a crypto-create transaction sent from the user (i.e. a user transaction) to define
                // a key-based alias. Technically we could, but key-aliases are deprecated, so we don't allow it.
                throw new PreCheckException(INVALID_ALIAS_KEY);
            }
        }

        // You cannot set receiverSigRequired without asserting ownership of the key. Since these are cryptographic
        // keys, if the key was a contract key or delegatable contract key, then it would fail to be verified.
        // FUTURE Maybe a more specific error code would be better here rather than just failing at key verification.
        final var receiverSigReq = op.receiverSigRequired();
        if (receiverSigReq) {
            context.requireKey(op.keyOrThrow());
        }
    }

    /**
     * This method is called during the handle workflow. It executes the {@code CryptoCreate} transaction, creating a
     * new account with the given properties.
     *
     * 

If the transaction is successful, the account is created and the payer account is charged the transaction fee * and the initial balance of new account and the balance of the new account is set to the initial balance. * *

If the transaction is not successful, then the account is not created and the payer account will be charged * the transaction fee. * * @throws NullPointerException if one of the arguments is {@code null} * @throws HandleException if the transaction is not successful due to payer account being deleted or has * insufficient balance or the account is not created due to the usage of a price * regime */ @Override public void handle(@NonNull final HandleContext context) { requireNonNull(context); final var txnBody = context.body(); final var op = txnBody.cryptoCreateAccountOrThrow(); final var accountStore = context.storeFactory().writableStore(WritableAccountStore.class); // FUTURE: Use the config and check if accounts can be created. Currently, this check is being done in // `finishCryptoCreate` before `commit` // Validate fields in the transaction body that involves checking with dynamic properties or state validateSemantics(context, accountStore, op); // Now that we have fully validated the transaction inputs, it is time to create an account! // First, charge the payer for whatever initial balance there is. if (op.initialBalance() > 0) { final var payer = getIfUsable(context.payer(), accountStore, context.expiryValidator(), INVALID_PAYER_ACCOUNT_ID); final long newPayerBalance = payer.tinybarBalance() - op.initialBalance(); validatePayer(payer, newPayerBalance); // Change payer's balance to reflect the deduction of the initial balance for the new account final var modifiedPayer = payer.copyBuilder().tinybarBalance(newPayerBalance).build(); accountStore.put(modifiedPayer); } // Build the new account to be persisted based on the transaction body and save the newly created account // number in the record builder final var accountCreated = buildAccount(op, context); accountStore.put(accountCreated); final var createdAccountID = accountCreated.accountIdOrThrow(); final var recordBuilder = context.savepointStack().getBaseBuilder(CryptoCreateStreamBuilder.class); recordBuilder.accountID(createdAccountID); // Put if any new alias is associated with the account into account store final var alias = op.alias(); if (alias.length() > 0) { // If we have been given an EVM address, then we can just put it into the store if (isOfEvmAddressSize(alias)) { accountStore.putAlias(alias, createdAccountID); } else { // The only other kind of alias it could be is a key-alias. And in that case, it could be an ED25519 // protobuf-encoded key, or it could be an ECDSA_SECP256K1 protobuf-encoded key. In this latter case, // we will actually store two things in the map. We'll store the raw alias, and we'll also extract the // EVM address from the key and store a second mapping from that EVM address to account ID. This makes // it trivial at look up time to see if an EVM address is already in use, or to look up the account ID // for an EVM address, and leads to simpler, more correct code. final var key = asKeyFromAlias(alias); validateTrue(isValid(key), INVALID_ALIAS_KEY); // In case the protobuf encoded key is BOGUS! final var evmAddress = extractEvmAddress(key); if (evmAddress != null) { accountStore.putAlias(evmAddress, createdAccountID); recordBuilder.evmAddress(evmAddress); } accountStore.putAlias(alias, createdAccountID); } } } /* ----------- Helper Methods ----------- */ /** * Validate the fields in the transaction body that involves checking with dynamic * properties or state. This check is done as part of the handle workflow. * @param context handle context * @param accountStore account store * @param op crypto create transaction body */ private void validateSemantics( @NonNull final HandleContext context, @NonNull final ReadableAccountStore accountStore, @NonNull final CryptoCreateTransactionBody op) { final var cryptoCreateWithAliasConfig = context.configuration().getConfigData(CryptoCreateWithAliasConfig.class); final var ledgerConfig = context.configuration().getConfigData(LedgerConfig.class); final var entitiesConfig = context.configuration().getConfigData(EntitiesConfig.class); final var tokensConfig = context.configuration().getConfigData(TokensConfig.class); final var accountConfig = context.configuration().getConfigData(AccountsConfig.class); final var alias = op.alias(); // We have a limit on the total maximum number of entities that can be created on the network, for different // types of entities. We need to verify that creating a new account won't exceed that number. if (accountStore.getNumberOfAccounts() + 1 > accountConfig.maxNumber()) { throw new HandleException(MAX_ENTITIES_IN_PRICE_REGIME_HAVE_BEEN_CREATED); } // Aliases are fully supported in mainnet, but we still have this feature flag. If it is disabled, then // you cannot create an account with an alias. FUTURE: We may be able to remove this flag. final var hasAlias = alias.length() > 0; if (hasAlias && !cryptoCreateWithAliasConfig.enabled()) { throw new HandleException(NOT_SUPPORTED); } // We have to check the memo, which may be too long or in some other way be invalid. context.attributeValidator().validateMemo(op.memo()); // If there is an alias, then we need to make sure no other account or contract account is using that alias. if (hasAlias) { // find account by alias and check if it was deleted var accountId = accountStore.getAccountIDByAlias(alias); var account = accountId != null ? accountStore.getAccountById(accountId) : null; var isDeleted = account == null || account.deleted(); validateTrue(accountId == null || isDeleted, ALIAS_ALREADY_ASSIGNED); } // We've already validated in pureChecks that there is a renewal period. Deeper validation. context.attributeValidator() .validateAutoRenewPeriod(op.autoRenewPeriodOrThrow().seconds()); // When the account is created, it can be created with some auto-association slots. But we have some // additional ledger-wide limits we need to check as well. validateFalse( cryptoCreateValidator.tooManyAutoAssociations( op.maxAutomaticTokenAssociations(), ledgerConfig, entitiesConfig, tokensConfig), INVALID_MAX_AUTO_ASSOCIATIONS); // This proxy field has been deprecated. We do not allow people to use it. validateFalse( op.hasProxyAccountID() && !op.proxyAccountIDOrThrow().equals(AccountID.DEFAULT), PROXY_ACCOUNT_ID_FIELD_IS_DEPRECATED); // Validates the key. One special twist is that if there is no transaction ID, then the creation call came from // another internal service, and not through the HAPI. In that case, we have some special validation. See the // method for details. cryptoCreateValidator.validateKey( op.keyOrThrow(), // cannot be null by this point context.attributeValidator(), context.savepointStack().getBaseBuilder(StreamBuilder.class).isInternalDispatch()); // Validate the staking information included in this account creation. if (op.hasStakedAccountId() || op.hasStakedNodeId()) { StakingValidator.validateStakedIdForCreation( context.configuration().getConfigData(StakingConfig.class).isEnabled(), op.declineReward(), op.stakedId().kind().name(), op.stakedAccountId(), op.stakedNodeId(), accountStore, context.networkInfo()); } } /** * Validates the payer account exists and has enough balance to cover the initial balance of the * account to be created. * * @param payer the payer account * @param newPayerBalance the initial balance of the account to be created */ private void validatePayer(@NonNull final Account payer, final long newPayerBalance) { // If the payer account is deleted, throw an exception if (payer.deleted()) { throw new HandleException(ACCOUNT_DELETED); } if (newPayerBalance < 0) { throw new HandleException(INSUFFICIENT_PAYER_BALANCE); } // FUTURE: check if payer account is detached when we have started expiring accounts ? } /** * Builds an account based on the transaction body and the consensus time. * * @param op the transaction body * @param handleContext the handle context * @return the account created */ @NonNull private Account buildAccount(CryptoCreateTransactionBody op, HandleContext handleContext) { final var autoRenewPeriod = op.autoRenewPeriodOrThrow().seconds(); final var consensusTime = handleContext.consensusNow().getEpochSecond(); final var expiry = consensusTime + autoRenewPeriod; var builder = Account.newBuilder() .memo(op.memo()) .expirationSecond(expiry) .autoRenewSeconds(autoRenewPeriod) .receiverSigRequired(op.receiverSigRequired()) .maxAutoAssociations(op.maxAutomaticTokenAssociations()) .tinybarBalance(op.initialBalance()) .declineReward(op.declineReward()) .key(op.keyOrThrow()) .stakeAtStartOfLastRewardedPeriod(NOT_REWARDED_SINCE_LAST_STAKING_META_CHANGE) .stakePeriodStart(NO_STAKE_PERIOD_START) .alias(op.alias()); // We do this separately because we want to let the protobuf object remain UNSET for the staked ID if neither // of the staking information was set in the transaction body. if (op.hasStakedAccountId()) { builder.stakedAccountId(op.stakedAccountId()); } else if (op.hasStakedNodeId()) { builder.stakedNodeId(op.stakedNodeIdOrThrow()); } // Set the new account number final var hederaConfig = handleContext.configuration().getConfigData(HederaConfig.class); builder.accountId(AccountID.newBuilder() .shardNum(hederaConfig.shard()) .realmNum(hederaConfig.realm()) .accountNum(handleContext.entityNumGenerator().newEntityNum()) .build()); return builder.build(); } @Override @NonNull public Fees calculateFees(@NonNull final FeeContext feeContext) { // Variable bytes plus two additional longs for balance and auto-renew period; plus a boolean for receiver sig // required. final var op = feeContext.body().cryptoCreateAccountOrThrow(); final var keySize = op.hasKey() ? getAccountKeyStorageSize(CommonPbjConverters.fromPbj(op.keyOrElse(Key.DEFAULT))) : 0L; final var unlimitedAutoAssociations = feeContext.configuration().getConfigData(EntitiesConfig.class).unlimitedAutoAssociationsEnabled(); final var maxAutoAssociationsSize = !unlimitedAutoAssociations && op.maxAutomaticTokenAssociations() > 0 ? INT_SIZE : 0L; final var baseSize = op.memo().length() + keySize + maxAutoAssociationsSize; final var lifeTime = op.autoRenewPeriodOrElse(Duration.DEFAULT).seconds(); final var feeCalculator = feeContext.feeCalculatorFactory().feeCalculator(SubType.DEFAULT); final var fee = feeCalculator .addBytesPerTransaction(baseSize + (2 * LONG_SIZE) + BOOL_SIZE) .addRamByteSeconds((CRYPTO_ENTITY_SIZES.fixedBytesInAccountRepr() + baseSize) * lifeTime) .addNetworkRamByteSeconds(BASIC_ENTITY_ID_SIZE * USAGE_PROPERTIES.legacyReceiptStorageSecs()); if (!unlimitedAutoAssociations && op.maxAutomaticTokenAssociations() > 0) { fee.addRamByteSeconds(op.maxAutomaticTokenAssociations() * lifeTime * CREATE_SLOT_MULTIPLIER); } if (IMMUTABILITY_SENTINEL_KEY.equals(op.key())) { final var lazyCreationFee = feeContext.dispatchComputeFees(UPDATE_TXN_BODY_BUILDER, feeContext.payer()); return fee.calculate().plus(lazyCreationFee); } return fee.calculate(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy