Skip to content

Commit

Permalink
feat: HIP-904 Implement TokenClaimAirdrop System Contract (#16054)
Browse files Browse the repository at this point in the history
Signed-off-by: Stanimir Stoyanov <stanimir.stoyanov@limechain.tech>
  • Loading branch information
stoyanov-st authored and Evdokia-Georgieva committed Oct 31, 2024
1 parent f0ebaa7 commit 745d624
Show file tree
Hide file tree
Showing 16 changed files with 1,786 additions and 1 deletion.
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,8 @@ public enum DispatchType {
UTIL_PRNG(HederaFunctionality.UTIL_PRNG, DEFAULT),
TOKEN_INFO(HederaFunctionality.TOKEN_GET_INFO, DEFAULT),
UPDATE_TOKEN_CUSTOM_FEES(HederaFunctionality.TOKEN_FEE_SCHEDULE_UPDATE, DEFAULT),
TOKEN_AIRDROP(HederaFunctionality.TOKEN_AIRDROP, DEFAULT);
TOKEN_AIRDROP(HederaFunctionality.TOKEN_AIRDROP, DEFAULT),
TOKEN_CLAIM_AIRDROP(HederaFunctionality.TOKEN_CLAIM_AIRDROP, DEFAULT);

private final HederaFunctionality functionality;
private final SubType subtype;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.hts.associations.AssociationsTranslator;
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.hts.balanceof.BalanceOfTranslator;
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.hts.burn.BurnTranslator;
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.hts.claimairdrops.TokenClaimAirdropTranslator;
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.hts.create.CreateTranslator;
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.hts.customfees.TokenCustomFeesTranslator;
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.hts.decimals.DecimalsTranslator;
Expand Down Expand Up @@ -448,4 +449,13 @@ static CallTranslator<HtsCallAttempt> provideTokenAirdropTranslator(
@NonNull final TokenAirdropTranslator translator) {
return translator;
}

@Provides
@Singleton
@IntoSet
@Named("HtsTranslators")
static CallTranslator<HtsCallAttempt> provideTokenClaimAirdropDecoder(
@NonNull final TokenClaimAirdropTranslator translator) {
return translator;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,146 @@
/*
* 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.contract.impl.exec.systemcontracts.hts.claimairdrops;

import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_TOKEN_ID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.PENDING_AIRDROP_ID_LIST_TOO_LONG;
import static com.hedera.node.app.service.contract.impl.utils.ConversionUtils.asTokenId;
import static com.hedera.node.app.spi.workflows.HandleException.validateFalse;
import static com.hedera.node.app.spi.workflows.HandleException.validateTrue;

import com.esaulpaugh.headlong.abi.Address;
import com.esaulpaugh.headlong.abi.Tuple;
import com.hedera.hapi.node.base.AccountID;
import com.hedera.hapi.node.base.NftID;
import com.hedera.hapi.node.base.PendingAirdropId;
import com.hedera.hapi.node.base.TokenID;
import com.hedera.hapi.node.base.TokenType;
import com.hedera.hapi.node.token.TokenClaimAirdropTransactionBody;
import com.hedera.hapi.node.transaction.TransactionBody;
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.hts.HtsCallAttempt;
import com.hedera.node.config.data.TokensConfig;
import edu.umd.cs.findbugs.annotations.NonNull;
import java.util.ArrayList;
import java.util.Arrays;
import javax.inject.Inject;
import javax.inject.Singleton;

@Singleton
public class TokenClaimAirdropDecoder {

// Tuple indexes
private static final int TRANSFER_LIST = 0;
private static final int SENDER = 0;
private static final int RECEIVER = 1;
private static final int TOKEN = 2;
private static final int SERIAL = 3;
private static final int HRC_SENDER = 0;
private static final int HRC_SERIAL = 1;

@Inject
public TokenClaimAirdropDecoder() {
// Dagger2
}

public TransactionBody decodeTokenClaimAirdrop(@NonNull final HtsCallAttempt attempt) {
final var call = TokenClaimAirdropTranslator.CLAIM_AIRDROP.decodeCall(attempt.inputBytes());
final var maxPendingAirdropsToClaim =
attempt.configuration().getConfigData(TokensConfig.class).maxAllowedPendingAirdropsToClaim();
validateFalse(((Tuple[]) call.get(0)).length > maxPendingAirdropsToClaim, PENDING_AIRDROP_ID_LIST_TOO_LONG);

final var transferList = (Tuple[]) call.get(TRANSFER_LIST);
final var pendingAirdrops = new ArrayList<PendingAirdropId>();
Arrays.stream(transferList).forEach(transfer -> {
final var senderAddress = (Address) transfer.get(SENDER);
final var receiverAddress = (Address) transfer.get(RECEIVER);
final var tokenAddress = (Address) transfer.get(TOKEN);
final var serial = (long) transfer.get(SERIAL);

final var senderId = attempt.addressIdConverter().convert(senderAddress);
final var receiverId = attempt.addressIdConverter().convert(receiverAddress);
final var tokenId = asTokenId(tokenAddress);

final var token = attempt.enhancement().nativeOperations().getToken(tokenId.tokenNum());
validateTrue(token != null, INVALID_TOKEN_ID);
if (token.tokenType().equals(TokenType.FUNGIBLE_COMMON)) {
pendingAirdrops.add(pendingFTAirdrop(senderId, receiverId, tokenId));
} else {
pendingAirdrops.add(pendingNFTAirdrop(senderId, receiverId, tokenId, serial));
}
});

return TransactionBody.newBuilder()
.tokenClaimAirdrop(TokenClaimAirdropTransactionBody.newBuilder().pendingAirdrops(pendingAirdrops))
.build();
}

public TransactionBody decodeHrcClaimAirdropFt(@NonNull final HtsCallAttempt attempt) {
final var call = TokenClaimAirdropTranslator.HRC_CLAIM_AIRDROP_FT.decodeCall(attempt.inputBytes());

// As the Token Claim is an operation for the receiver of an Airdrop,
// hence the `transaction sender` in the HRC scenario is in reality the `Airdrop receiver`.
final var receiverId = attempt.senderId();
final var senderAddress = (Address) call.get(HRC_SENDER);
final var token = attempt.redirectTokenId();
validateTrue(token != null, INVALID_TOKEN_ID);
final var senderId = attempt.addressIdConverter().convert(senderAddress);

return TransactionBody.newBuilder()
.tokenClaimAirdrop(TokenClaimAirdropTransactionBody.newBuilder()
.pendingAirdrops(pendingFTAirdrop(senderId, receiverId, token)))
.build();
}

public TransactionBody decodeHrcClaimAirdropNft(@NonNull final HtsCallAttempt attempt) {
final var call = TokenClaimAirdropTranslator.HRC_CLAIM_AIRDROP_NFT.decodeCall(attempt.inputBytes());

// As the Token Claim is an operation for the receiver of an Airdrop,
// hence the `transaction sender` in the HRC scenario is in reality the `Airdrop receiver`.
final var receiverId = attempt.senderId();
final var senderAddress = (Address) call.get(HRC_SENDER);
final var serial = (long) call.get(HRC_SERIAL);
final var token = attempt.redirectTokenId();
validateTrue(token != null, INVALID_TOKEN_ID);
final var senderId = attempt.addressIdConverter().convert(senderAddress);

return TransactionBody.newBuilder()
.tokenClaimAirdrop(TokenClaimAirdropTransactionBody.newBuilder()
.pendingAirdrops(pendingNFTAirdrop(senderId, receiverId, token, serial)))
.build();
}

private PendingAirdropId pendingFTAirdrop(
@NonNull final AccountID senderId, @NonNull final AccountID receiverId, @NonNull final TokenID tokenId) {
return PendingAirdropId.newBuilder()
.senderId(senderId)
.receiverId(receiverId)
.fungibleTokenType(tokenId)
.build();
}

private PendingAirdropId pendingNFTAirdrop(
@NonNull final AccountID senderId,
@NonNull final AccountID receiverId,
@NonNull final TokenID tokenId,
final long serial) {
return PendingAirdropId.newBuilder()
.senderId(senderId)
.receiverId(receiverId)
.nonFungibleToken(NftID.newBuilder().tokenId(tokenId).serialNumber(serial))
.build();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,85 @@
/*
* 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.contract.impl.exec.systemcontracts.hts.claimairdrops;

import com.esaulpaugh.headlong.abi.Function;
import com.hedera.hapi.node.base.AccountID;
import com.hedera.hapi.node.transaction.TransactionBody;
import com.hedera.node.app.service.contract.impl.exec.gas.DispatchType;
import com.hedera.node.app.service.contract.impl.exec.gas.SystemContractGasCalculator;
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.common.AbstractCallTranslator;
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.common.Call;
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.hts.DispatchForResponseCodeHtsCall;
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.hts.HtsCallAttempt;
import com.hedera.node.app.service.contract.impl.exec.systemcontracts.hts.ReturnTypes;
import com.hedera.node.app.service.contract.impl.hevm.HederaWorldUpdater;
import com.hedera.node.config.data.ContractsConfig;
import edu.umd.cs.findbugs.annotations.NonNull;
import javax.inject.Inject;

public class TokenClaimAirdropTranslator extends AbstractCallTranslator<HtsCallAttempt> {
public static final Function CLAIM_AIRDROP =
new Function("claimAirdrops((address,address,address,int64)[])", ReturnTypes.INT_64);
public static final Function HRC_CLAIM_AIRDROP_FT = new Function("claimAirdropFT(address)", ReturnTypes.INT_64);
public static final Function HRC_CLAIM_AIRDROP_NFT =
new Function("claimAirdropNFT(address,int64)", ReturnTypes.INT_64);

private final TokenClaimAirdropDecoder decoder;

@Inject
public TokenClaimAirdropTranslator(@NonNull final TokenClaimAirdropDecoder decoder) {
this.decoder = decoder;
}

@Override
public boolean matches(@NonNull final HtsCallAttempt attempt) {
final var claimAirdropEnabled =
attempt.configuration().getConfigData(ContractsConfig.class).systemContractClaimAirdropsEnabled();
return attempt.isTokenRedirect()
? attempt.isSelectorIfConfigEnabled(HRC_CLAIM_AIRDROP_FT, claimAirdropEnabled)
|| attempt.isSelectorIfConfigEnabled(HRC_CLAIM_AIRDROP_NFT, claimAirdropEnabled)
: attempt.isSelectorIfConfigEnabled(CLAIM_AIRDROP, claimAirdropEnabled);
}

@Override
public Call callFrom(@NonNull final HtsCallAttempt attempt) {
return new DispatchForResponseCodeHtsCall(
attempt,
attempt.isSelector(CLAIM_AIRDROP) ? bodyForClassic(attempt) : bodyForHRC(attempt),
TokenClaimAirdropTranslator::gasRequirement);
}

public static long gasRequirement(
@NonNull final TransactionBody body,
@NonNull final SystemContractGasCalculator systemContractGasCalculator,
@NonNull final HederaWorldUpdater.Enhancement enhancement,
@NonNull final AccountID payerId) {
return systemContractGasCalculator.gasRequirement(body, DispatchType.TOKEN_CLAIM_AIRDROP, payerId);
}

private TransactionBody bodyForClassic(@NonNull final HtsCallAttempt attempt) {
return decoder.decodeTokenClaimAirdrop(attempt);
}

private TransactionBody bodyForHRC(@NonNull final HtsCallAttempt attempt) {
if (attempt.isSelector(HRC_CLAIM_AIRDROP_FT)) {
return decoder.decodeHrcClaimAirdropFt(attempt);
} else {
return decoder.decodeHrcClaimAirdropNft(attempt);
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -126,4 +126,28 @@ public static HasCallAttempt prepareHasAttemptWithSelector(
List.of(translator),
false);
}

public static HtsCallAttempt prepareHtsAttemptWithSelectorForRedirectWithConfig(
final Function function,
final CallTranslator<HtsCallAttempt> translator,
final HederaWorldUpdater.Enhancement enhancement,
final AddressIdConverter addressIdConverter,
final VerificationStrategies verificationStrategies,
final SystemContractGasCalculator gasCalculator,
final Configuration config) {
final var input = TestHelpers.bytesForRedirect(function.selector(), NON_SYSTEM_LONG_ZERO_ADDRESS);

return new HtsCallAttempt(
input,
OWNER_BESU_ADDRESS,
OWNER_BESU_ADDRESS,
false,
enhancement,
config,
addressIdConverter,
verificationStrategies,
gasCalculator,
List.of(translator),
false);
}
}
Loading

0 comments on commit 745d624

Please sign in to comment.