com.hedera.node.app.service.token.impl.handlers.BaseTokenHandler Maven / Gradle / Ivy
Show all versions of app-service-token-impl Show documentation
/*
* Copyright (C) 2023-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_KYC_NOT_GRANTED_FOR_TOKEN;
import static com.hedera.hapi.node.base.ResponseCodeEnum.FAIL_INVALID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INSUFFICIENT_TOKEN_BALANCE;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_TOKEN_MINT_AMOUNT;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_TREASURY_ACCOUNT_FOR_TOKEN;
import static com.hedera.hapi.node.base.ResponseCodeEnum.MAX_ENTITIES_IN_PRICE_REGIME_HAVE_BEEN_CREATED;
import static com.hedera.hapi.node.base.ResponseCodeEnum.NO_REMAINING_AUTOMATIC_ASSOCIATIONS;
import static com.hedera.hapi.node.base.ResponseCodeEnum.TOKENS_PER_ACCOUNT_LIMIT_EXCEEDED;
import static com.hedera.hapi.node.base.ResponseCodeEnum.TOKEN_ALREADY_ASSOCIATED_TO_ACCOUNT;
import static com.hedera.hapi.node.base.ResponseCodeEnum.TOKEN_MAX_SUPPLY_REACHED;
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.AccountID;
import com.hedera.hapi.node.base.ResponseCodeEnum;
import com.hedera.hapi.node.base.TokenAssociation;
import com.hedera.hapi.node.base.TokenID;
import com.hedera.hapi.node.base.TokenSupplyType;
import com.hedera.hapi.node.state.token.Account;
import com.hedera.hapi.node.state.token.Token;
import com.hedera.hapi.node.state.token.TokenRelation;
import com.hedera.hapi.node.token.TokenUpdateTransactionBody;
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.util.TokenKey;
import com.hedera.node.app.spi.validation.ExpiryValidator;
import com.hedera.node.config.data.EntitiesConfig;
import com.hedera.node.config.data.TokensConfig;
import com.swirlds.config.api.Configuration;
import edu.umd.cs.findbugs.annotations.NonNull;
import java.util.ArrayList;
import java.util.EnumSet;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
/**
* Provides common functionality for token handlers.
*/
public class BaseTokenHandler {
private static final Logger log = LogManager.getLogger(BaseTokenHandler.class);
/**
* The set of token keys that are not admin keys.
*/
protected static final Set NON_ADMIN_TOKEN_KEYS = EnumSet.complementOf(EnumSet.of(TokenKey.ADMIN_KEY));
/**
* The set of all token keys.
*/
protected static final Set TOKEN_KEYS = EnumSet.allOf(TokenKey.class);
/**
* The value for unlimited automatic associations.
*/
public static final int UNLIMITED_AUTOMATIC_ASSOCIATIONS = -1;
/**
* Mints fungible tokens. This method is called in both token create and mint.
*
* @param token the new or existing token to mint
* @param treasuryRel the treasury relation for the token
* @param amount the amount to mint
* @param accountStore the account store
* @param tokenStore the token store
* @param tokenRelationStore the token relation store
* @param expiryValidator the expiry validator
*/
protected long mintFungible(
@NonNull final Token token,
@NonNull final TokenRelation treasuryRel,
final long amount,
@NonNull final WritableAccountStore accountStore,
@NonNull final WritableTokenStore tokenStore,
@NonNull final WritableTokenRelationStore tokenRelationStore,
@NonNull final ExpiryValidator expiryValidator) {
requireNonNull(token);
requireNonNull(treasuryRel);
return changeSupply(
token,
treasuryRel,
+amount,
INVALID_TOKEN_MINT_AMOUNT,
accountStore,
tokenStore,
tokenRelationStore,
expiryValidator);
}
/**
* Since token mint and token burn change the supply on the token and treasury account,
* this method is used to change the supply.
*
*
* Note: This method assumes the given token has a non-null supply key!
*
* @param token the token that is minted or burned
* @param treasuryRel the treasury relation for the token
* @param amount the amount to mint or burn
* @param invalidSupplyCode the invalid supply code to use if the supply is invalid
* @param accountStore the account store
* @param tokenStore the token store
* @param tokenRelationStore the token relation store
* @param expiryValidator the expiry validator
*/
protected long changeSupply(
@NonNull final Token token,
@NonNull final TokenRelation treasuryRel,
final long amount,
@NonNull final ResponseCodeEnum invalidSupplyCode,
@NonNull final WritableAccountStore accountStore,
@NonNull final WritableTokenStore tokenStore,
@NonNull final WritableTokenRelationStore tokenRelationStore,
@NonNull final ExpiryValidator expiryValidator) {
requireNonNull(token);
requireNonNull(treasuryRel);
requireNonNull(invalidSupplyCode);
validateTrue(
treasuryRel.accountId().equals(token.treasuryAccountId())
&& token.tokenId().equals(treasuryRel.tokenId()),
FAIL_INVALID);
final long newTotalSupply = token.totalSupply() + amount;
// validate that the new total supply is not negative after mint or burn or wipe
// FUTURE - All these checks that return FAIL_INVALID probably should end up in a
// finalize method in token service to validate everything before we commit
validateTrue(newTotalSupply >= 0, invalidSupplyCode);
if (token.supplyType() == TokenSupplyType.FINITE) {
validateTrue(token.maxSupply() >= newTotalSupply, TOKEN_MAX_SUPPLY_REACHED);
}
final var treasuryAccount = TokenHandlerHelper.getIfUsable(
treasuryRel.accountId(), accountStore, expiryValidator, INVALID_TREASURY_ACCOUNT_FOR_TOKEN);
final long newTreasuryBalance = treasuryRel.balance() + amount;
validateTrue(newTreasuryBalance >= 0, INSUFFICIENT_TOKEN_BALANCE);
// copy the token, treasury account and treasury relation
final var copyTreasuryAccount = treasuryAccount.copyBuilder();
final var copyToken = token.copyBuilder();
final var copyTreasuryRel = treasuryRel.copyBuilder();
if (treasuryRel.balance() == 0 && amount > 0) {
// On an account positive balances are incremented for newly added tokens.
// If treasury relation did mint any for this token till now, only then increment
// total positive balances on treasury account.
copyTreasuryAccount.numberPositiveBalances(treasuryAccount.numberPositiveBalances() + 1);
} else if (newTreasuryBalance == 0 && amount < 0) {
// On an account positive balances are decremented for burning tokens completely.
// If treasury relation did not burn any for this token till now or if this burn makes the balance to 0,
// only then decrement total positive balances on treasury account.
copyTreasuryAccount.numberPositiveBalances(treasuryAccount.numberPositiveBalances() - 1);
}
// since we have either minted or burned tokens, we need to update the total supply
copyToken.totalSupply(newTotalSupply);
copyTreasuryRel.balance(newTreasuryBalance);
// put the changed token, treasury account and treasury relation
accountStore.put(copyTreasuryAccount.build());
tokenStore.put(copyToken.build());
tokenRelationStore.put(copyTreasuryRel.build());
return newTotalSupply;
}
/**
* Creates {@link TokenRelation} object for each token association to account and links tokens to the account.
* This is used in both token associate logic and also token create logic
* @param account the account to link the tokens to
* @param tokens the tokens to link to the account
* @param accountStore the account store
* @param tokenRelStore the token relation store
*/
protected void createAndLinkTokenRels(
@NonNull final Account account,
@NonNull final List tokens,
@NonNull final WritableAccountStore accountStore,
@NonNull final WritableTokenRelationStore tokenRelStore) {
// create list of token relations to be added
final var newTokenRels = createTokenRelsToAccount(account, tokens);
// FUTURE - We may need to return a proper error status when tokens are empty
if (!newTokenRels.isEmpty()) {
// Link the new token relations to the account
linkTokenRels(account, newTokenRels, tokenRelStore);
// Now replace the account's old head token number with the new head token number. This is
// how we link the new tokenRels to the account
final var firstOfNewTokenRels = newTokenRels.get(0);
final var updatedAcct = account.copyBuilder()
// replace the head token number with the first token number of the new tokenRels
.headTokenId(firstOfNewTokenRels.tokenId())
// and also update the account's total number of token associations
.numberAssociations(account.numberAssociations() + newTokenRels.size())
.build();
// Save the results
accountStore.put(updatedAcct);
newTokenRels.forEach(tokenRelStore::put);
}
}
/**
* Link all the new token relations created for the account together, and then link them to the account.
* @param account the account to link the tokens to
* @param newTokenRels the new token relations to link to the account
* @param tokenRelStore the token relation store
*/
private void linkTokenRels(
@NonNull final Account account,
@NonNull final List newTokenRels,
@NonNull final WritableTokenRelationStore tokenRelStore) {
// Now all the NEW token relations are linked together, but they are not yet linked to the account. First,
// compute where the account's current head token number should go in the linked list of tokens
final var currentHeadTokenNum = account.headTokenId();
// NOTE: if currentHeadTokenNum is less than 1, it means the account isn't associated with any tokens yet, so
// we'll just set the head to the first token, i.e. the first token ID list from the transaction (since the new
// tokenRels are all linked, and in the order of the token IDs as they appeared in the original list)
if (isValidTokenId(currentHeadTokenNum)) {
// The account is already associated with some tokens, so we need to insert the new
// tokenRels at the beginning of the list of existing token numbers first. We start by
// retrieving the token rel object with the currentHeadTokenNum at the head of the
// account
final var headTokenRel = tokenRelStore.get(account.accountId(), currentHeadTokenNum);
if (headTokenRel != null) {
// Recreate the current head token's tokenRel, but with its previous pointer set to
// the last of the new tokenRels. This links the new token rels to the rest of the
// token rels connected via the old head token rel
final var lastOfNewTokenRels = newTokenRels.remove(newTokenRels.size() - 1);
final var headTokenAsNonHeadTokenRel = headTokenRel
.copyBuilder()
.previousToken(lastOfNewTokenRels.tokenId())
.build(); // the old head token rel is no longer the head
// Also connect the last of the new tokenRels to the old head token rel
newTokenRels.add(lastOfNewTokenRels
.copyBuilder()
.nextToken(headTokenAsNonHeadTokenRel.tokenId())
.build());
tokenRelStore.put(headTokenAsNonHeadTokenRel);
} else {
// This shouldn't happen, but if it does we'll log the error and continue with creating the token
// associations
AccountID accountId = account.accountId();
log.error(
"Unable to get head tokenRel for account {}.{}.{}, token {}.{}.{}! "
+ "Linked-list relations are likely in a bad state",
accountId.shardNum(),
accountId.realmNum(),
accountId.accountNum(),
currentHeadTokenNum.shardNum(),
currentHeadTokenNum.realmNum(),
currentHeadTokenNum.tokenNum());
}
}
}
/**
* Creates list of {@link TokenRelation}s for each token association to account.
* @param account the account to link the tokens to
* @param tokens the tokens to link to the account
* @return the list of token relations to be added
*/
private List createTokenRelsToAccount(
@NonNull final Account account, @NonNull final List tokens) {
final var newTokenRels = new ArrayList();
for (int i = 0; i < tokens.size(); i++) {
final var token = tokens.get(i);
// Link each of the new token IDs together in a doubly-linked list way by setting each
// token relation's previous and next token IDs.
// Compute the previous and next token IDs.
TokenID prevTokenId = null;
TokenID nextTokenId = null;
if (i - 1 >= 0) { // if there is a previous token
prevTokenId = Optional.ofNullable(tokens.get(i - 1))
.map(Token::tokenId)
.orElse(null);
}
if (i + 1 < tokens.size()) { // if there is a next token
nextTokenId = Optional.ofNullable(tokens.get(i + 1))
.map(Token::tokenId)
.orElse(null);
}
// Create the new token relation
boolean isTreasuryAccount = Objects.equals(token.treasuryAccountId(), account.accountId());
final var isFrozen = token.hasFreezeKey() && token.accountsFrozenByDefault() && !isTreasuryAccount;
final var kycGranted = !token.hasKycKey() || isTreasuryAccount;
final var newTokenRel = new TokenRelation(
token.tokenId(), account.accountId(), 0, isFrozen, kycGranted, false, prevTokenId, nextTokenId);
newTokenRels.add(newTokenRel);
}
return newTokenRels;
}
/**
* Creates a new {@link TokenRelation} with the account and token. This is called when there is
* no association yet, but have open slots for maxAutoAssociations on the account.
* @param accountId the accountId to link the tokens to
* @param token the token to link to the account
* @param accountStore the account store
* @param tokenRelStore the token relation store
* @param config the configuration
* @return the new token relation added
*/
protected TokenRelation autoAssociate(
@NonNull final AccountID accountId,
@NonNull final Token token,
@NonNull final WritableAccountStore accountStore,
@NonNull final WritableTokenRelationStore tokenRelStore,
@NonNull final Configuration config) {
final var tokensConfig = config.getConfigData(TokensConfig.class);
final var entitiesConfig = config.getConfigData(EntitiesConfig.class);
final var tokenId = token.tokenIdOrThrow();
// If token is already associated, no need to associate again
final var existingRel = tokenRelStore.get(accountId, tokenId);
validateTrue(existingRel == null, TOKEN_ALREADY_ASSOCIATED_TO_ACCOUNT);
validateTrue(
tokenRelStore.sizeOfState() + 1 < tokensConfig.maxAggregateRels(),
MAX_ENTITIES_IN_PRICE_REGIME_HAVE_BEEN_CREATED);
final var account = accountStore.get(accountId);
// Check is number of used associations is less than maxAutoAssociations
final var numAssociations = account.numberAssociations();
validateFalse(
entitiesConfig.limitTokenAssociations() && numAssociations >= tokensConfig.maxPerAccount(),
TOKENS_PER_ACCOUNT_LIMIT_EXCEEDED);
final var maxAutoAssociations = account.maxAutoAssociations();
final var usedAutoAssociations = account.usedAutoAssociations();
// only validate remaining associations if unlimitedAutoAssociations is disabled for the account
if (!hasUnlimitedAutoAssociations(account, entitiesConfig)) {
validateFalse(usedAutoAssociations >= maxAutoAssociations, NO_REMAINING_AUTOMATIC_ASSOCIATIONS);
}
// Create new token relation and commit to store
final var newTokenRel = TokenRelation.newBuilder()
.tokenId(tokenId)
.accountId(accountId)
.automaticAssociation(true)
.kycGranted(!token.hasKycKey())
.frozen(token.hasFreezeKey() && token.accountsFrozenByDefault())
.previousToken((TokenID) null)
.nextToken(account.headTokenId())
.build();
final var copyAccount = account.copyBuilder()
.numberAssociations(numAssociations + 1)
.usedAutoAssociations(usedAutoAssociations + 1)
.headTokenId(tokenId)
.build();
final var existingFirstTokenId = account.headTokenId();
if (existingFirstTokenId != null) {
final var existingFirstTokenRel = tokenRelStore.get(accountId, existingFirstTokenId);
if (existingFirstTokenRel != null) {
tokenRelStore.put(existingFirstTokenRel
.copyBuilder()
.previousToken(tokenId)
.build());
}
}
accountStore.put(copyAccount);
tokenRelStore.put(newTokenRel);
return newTokenRel;
}
/**
* Adjust fungible balances for the given token relationship and account by the given adjustment.
* NOTE: This updates account's numOfPositiveBalances and puts to modifications on state.
* @param tokenRel token relationship
* @param account account to be adjusted
* @param adjustment adjustment to be made
* @param tokenRelStore token relationship store
* @param accountStore account store
*/
protected void adjustBalance(
@NonNull final TokenRelation tokenRel,
@NonNull final Account account,
final long adjustment,
@NonNull final WritableTokenRelationStore tokenRelStore,
@NonNull final WritableAccountStore accountStore) {
requireNonNull(tokenRel);
requireNonNull(account);
requireNonNull(tokenRelStore);
requireNonNull(accountStore);
final var originalBalance = tokenRel.balance();
final var newBalance = originalBalance + adjustment;
validateTrue(newBalance >= 0, INSUFFICIENT_TOKEN_BALANCE);
final var copyRel = tokenRel.copyBuilder();
tokenRelStore.put(copyRel.balance(newBalance).build());
var numPositiveBalances = account.numberPositiveBalances();
// If the original balance is zero, then the receiving account's numPositiveBalances has to
// be increased
// and if the newBalance is zero, then the sending account's numPositiveBalances has to be
// decreased
if (newBalance == 0 && adjustment < 0) {
numPositiveBalances--;
} else if (originalBalance == 0 && adjustment > 0) {
numPositiveBalances++;
}
final var copyAccount = account.copyBuilder();
accountStore.put(copyAccount.numberPositiveBalances(numPositiveBalances).build());
}
protected void validateNotFrozenAndKycOnRelation(@NonNull final TokenRelation rel) {
validateTrue(!rel.frozen(), ResponseCodeEnum.ACCOUNT_FROZEN_FOR_TOKEN);
validateTrue(rel.kycGranted(), ACCOUNT_KYC_NOT_GRANTED_FOR_TOKEN);
}
/* ------------------------- Helper functions ------------------------- */
/**
* Checks if the given op updates any of the non-key token properties that can only be
* changed given the admin key signature.
*
* @param op the token update op to check
* @return true if it requires admin key signature
*/
protected static boolean updatesAdminOnlyNonKeyTokenProperty(@NonNull final TokenUpdateTransactionBody op) {
return !op.symbol().isEmpty() || !op.name().isEmpty() || op.hasAutoRenewPeriod() || op.hasMemo();
}
/**
* Returns a new {@link TokenID} with the given number.
* @param num the token number
* @return the new token ID
*/
@NonNull
public static TokenID asToken(final long num) {
return TokenID.newBuilder().tokenNum(num).build();
}
/**
* Determines if a given token is valid.
*
* @param tokenId the token to check
* @return true if the token is valid
*/
public static boolean isValidTokenId(final TokenID tokenId) {
if (tokenId == null) return false;
return tokenId.tokenNum() > 0;
}
/**
* Creates a new {@link TokenAssociation} with the given tokenId and accountId.
* @param tokenId the token id
* @param accountId the account id
* @return the new token association
*/
public static TokenAssociation asTokenAssociation(final TokenID tokenId, final AccountID accountId) {
return TokenAssociation.newBuilder()
.tokenId(tokenId)
.accountId(accountId)
.build();
}
/**
* Checks if the given account has unlimited auto-associations enabled.
*
* @param account the account to check; must not be null
* @param entitiesConfig the configuration settings to check against; must not be null
* @return {@code true} if unlimited auto-associations is enabled and the account's
* max auto-associations is set to {@code UNLIMITED_AUTOMATIC_ASSOCIATIONS},
* otherwise {@code false}
* @throws NullPointerException if either {@code account} or {@code entitiesConfig} is null
*/
public static boolean hasUnlimitedAutoAssociations(
@NonNull final Account account, @NonNull EntitiesConfig entitiesConfig) {
return entitiesConfig.unlimitedAutoAssociationsEnabled()
&& account.maxAutoAssociations() == UNLIMITED_AUTOMATIC_ASSOCIATIONS;
}
}