UNPKG

@myria/airdrop-js

Version:

Airdrop in L1 with claim based approach

392 lines 28.7 kB
"use strict"; /** * Set of single functions in our airdrop-js * to let consumers pick and use with their strategies * @module Transaction/Core */ Object.defineProperty(exports, "__esModule", { value: true }); exports.retryPrepareAndSubmitRawTransaction = exports.sendTransactionAndWaitForReceipt = exports.saveSnapshotByOwner = exports.saveMerkleRootByOwner = exports.generateMerkleTreeInfoERC20ForWhitelist = exports.approveAirdropAsSpender = exports.claimAirdropToken = exports.getThirdwebContract = exports.getOwnerOfContract = exports.getNextNonce = exports.getGasFeeInfo = exports.getRpcClientByChain = exports.setMerkleRoot = exports.saveSnapshot = void 0; // re-exports: forward export from thirdweb var airdrop_1 = require("thirdweb/extensions/airdrop"); Object.defineProperty(exports, "saveSnapshot", { enumerable: true, get: function () { return airdrop_1.saveSnapshot; } }); Object.defineProperty(exports, "setMerkleRoot", { enumerable: true, get: function () { return airdrop_1.setMerkleRoot; } }); // External imports below this line const thirdweb_1 = require("thirdweb"); const airdrop_2 = require("thirdweb/extensions/airdrop"); const erc20_1 = require("thirdweb/extensions/erc20"); // Internal imports below this line const Retry_1 = require("../common/Retry"); const ConstType_1 = require("../type/ConstType"); const internalUse_1 = require("./internalUse"); /** * Returns an RPC request that can be used to make JSON-RPC requests * * @param {ThirdwebClient} client - Thirdweb client. * @param {CreateRpcClientOptions} options - Options to create RpcClient. */ /* eslint-disable @typescript-eslint/explicit-function-return-type, @typescript-eslint/explicit-module-boundary-types */ const getRpcClientByChain = (client, options) => { const { selectingChain, chain } = options; if (chain) { return (0, thirdweb_1.getRpcClient)({ client, chain, }); } return (0, thirdweb_1.getRpcClient)({ client, chain: (0, internalUse_1.getThirdWebChain)(selectingChain), }); }; exports.getRpcClientByChain = getRpcClientByChain; /** * This callback type is called `transactionCallback` and is displayed as a global symbol. * * @callback transactionCallback * @param {TransactionReceipt} result - The transaction receipt received from submitting. */ /** * Calculate gas fee info need to paid to submit a transaction * @see {@link https://ethereum.org/en/developers/docs/gas|Ethereum's gas} or @see {@link https://support.metamask.io/transactions-and-gas/gas-fees/user-guide-gas/|Metamask's gas} * @see {@link https://etherscan.io/gastracker|Gas Tracker} * * @param {EstimateGasOptions} options - The options for estimating gas. * @param {boolean} isLogResult - Whether to log the result or not. Default true. * @returns {Promise<GasFeeInfo>} Promise object represents the gas fee info to perform an on-chain transaction * @throws An error if the account is missed. */ const getGasFeeInfo = async (options, extraGasOptions = {}, isLogResult = true) => { const { transaction, account } = options; if (!account) { throw Error('require the account as sender of the transaction'); } const { extraGasPercentage = ConstType_1.DEFAULT_EXTRA_GAS_PERCENTAGE, extraMaxPriorityFeePerGasPercentage = ConstType_1.DEFAULT_EXTRA_PRIORITY_TIP_PERCENTAGE, extraOnRetryPercentage = ConstType_1.DEFAULT_EXTRA_ON_RETRY_PERCENTAGE, } = extraGasOptions; const { chain, client } = transaction; const rpcClient = (0, exports.getRpcClientByChain)(client, { chain }); // Estimate gas use for the transaction // -> gas limit const gasLimit = await (0, thirdweb_1.estimateGas)({ transaction, account, }); // -> gas extra const extraGas = (gasLimit / BigInt(100)) * BigInt(extraGasPercentage + extraOnRetryPercentage); // Gas price per unit of gas // -> base fee const gasPrice = await (0, thirdweb_1.eth_gasPrice)(rpcClient); // -> priority & max fee const maxPriorityFeePerGas = await (0, thirdweb_1.eth_maxPriorityFeePerGas)(rpcClient); const extraMaxPriorityFeePerGas = (maxPriorityFeePerGas / BigInt(100)) * BigInt(extraMaxPriorityFeePerGasPercentage + extraOnRetryPercentage); const maxFeePerGas = gasPrice + maxPriorityFeePerGas + extraMaxPriorityFeePerGas; // Estimate Gas Cost to compare with real transaction const gasCost = await (0, thirdweb_1.estimateGasCost)({ transaction, account }); console.log(` [getGasFeeInfo] gasCost.ether = ${gasCost.ether} - gasCost.wei = ${gasCost.wei}`); const gasFeeInfo = { gas: gasLimit, gasPrice, maxPriorityFeePerGas, maxFeePerGas, extraGas, }; if (isLogResult) { console.log(` [getGasFeeInfo]: gasFeeInfo = ${JSON.stringify(gasFeeInfo)}`); } return gasFeeInfo; }; exports.getGasFeeInfo = getGasFeeInfo; /** * Retrieves the transaction count (nonce) for a given Ethereum address. * * @see {@link https://ethereum.org/en/developers/docs/gas} for GAS AND FEES. * @see {@link https://etherscan.io/gastracker|Gas Tracker} * * @param {Address} address - The Ethereum address of Sender. * @param {ThirdwebClient} client - Thirdweb client. * @param {CreateRpcClientOptions} options - Options to create RpcClient. * @param {boolean} isLogResult - Whether to log the result or not. Default true. * @returns {Promise<number>} Promise object represents the next transaction nonce */ const getNextNonce = async (address, client, options, isLogResult = true) => { const startTime = (0, internalUse_1.logFunctionTrackStartTime)(exports.getNextNonce.name); const rpcClient = (0, exports.getRpcClientByChain)(client, options); const transactionNonce = await (0, thirdweb_1.eth_getTransactionCount)(rpcClient, { address, }); if (isLogResult) { // TODO: replace with the logger library for better format or other targets console.log(` [getNextNonce]>[response] transactionNonce = ${transactionNonce}`); } (0, internalUse_1.logFunctionDuration)(exports.getNextNonce.name, startTime); return transactionNonce; }; exports.getNextNonce = getNextNonce; /** * Reads owner of a smart contract. * * @param {ThirdwebContract} contract - The Thirdweb contract. * @returns {Promise<string>} A promise that resolves with the result of the owner ethereum address. */ const getOwnerOfContract = async (contract) => { return await (0, thirdweb_1.readContract)({ contract, // Pass a snippet of the ABI for the method you want to call. method: { type: 'function', name: 'owner', inputs: [], outputs: [ { type: 'address', name: '', internalType: 'address', }, ], stateMutability: 'view', }, params: [], }); }; exports.getOwnerOfContract = getOwnerOfContract; /** * Creates a Thirdweb contract by combining the Thirdweb client and contract options. * * @param {Address} address - The ethereum smart contract address. * @param {ThirdwebClient} client - Thirdweb client. * @param {SupportingChain} selectingChain - The selecting chain. * @returns The Thirdweb contract. */ /* eslint-disable @typescript-eslint/explicit-function-return-type*/ const getThirdwebContract = (address, client, selectingChain) => { return (0, thirdweb_1.getContract)({ client, chain: (0, internalUse_1.getThirdWebChain)(selectingChain), address, }); }; exports.getThirdwebContract = getThirdwebContract; /** * End-User connects wallet to trigger Claim from Client side * * @param {Address} tokenAddress - The token address to claim. * @param {Account} account - The Account represent as sender @see {@link https://ethereum.org/en/glossary/#account|Account's Ethereum}. * @param {ThirdwebContract} airdropContract - The airdrop Thirdweb contract. * @param {transactionCallback} callback - The callback that handles the post-submit state. * @param {boolean} isLogResult - Whether to log the result or not. Default true. * @returns {Promise<TransactionReceipt>} A promise that resolves to the confirmed transaction receipt. * @throws An error if the wallet is not connected. * @example * ```ts * import { claimAirdropToken } from "./index"; * * const transactionReceipt = await claimAirdropToken( * tokenAddress, * account, * airdropContract * ); * ``` */ const claimAirdropToken = async (tokenAddress, account, airdropContract, callback, isLogResult = true) => { const claimTransaction = (0, airdrop_2.claimERC20)({ contract: airdropContract, tokenAddress, recipient: account.address, }); // Send the transaction return (0, internalUse_1.doSubmitTransaction)({ transaction: claimTransaction, account }, callback, isLogResult); }; exports.claimAirdropToken = claimAirdropToken; /** * Token contract owner approve airdrop contract address as spender with amount * * @param {Address} spender - The airdrop smart contract address as spender. * @param {number} amount - The total airdrop amount in ether format. * @param {Account} account - The Account represent as sender @see {@link https://ethereum.org/en/glossary/#account|Account's Ethereum}. * @param {ThirdwebContract} tokenContract - The token Thirdweb contract to airdrop * @param {transactionCallback} callback - The callback that handles the post-submit state. * @param {boolean} isLogResult - Whether to log the result or not. Default true. * @returns {Promise<TransactionReceipt>} A promise that resolves to the confirmed transaction receipt. * @throws An error if the amount <= 0. * @throws An error if the wallet is not connected. * @example * ```ts * import { approveAirdropAsSpender } from "./index"; * * const transactionReceipt = await approveAirdropAsSpender( * spender, * amount, * account, * tokenContract * ); * ``` */ const approveAirdropAsSpender = async (spender, amount, account, tokenContract, callback, isLogResult = true) => { if (amount <= 0) { throw Error('total airdrop amount must be greater than 0'); } const startTime = (0, internalUse_1.logFunctionTrackStartTime)(exports.approveAirdropAsSpender.name); const transaction = (0, erc20_1.approve)({ contract: tokenContract, spender, amount, }); // Send the transaction const result = await (0, internalUse_1.doSubmitTransaction)({ transaction, account }, callback, isLogResult); (0, internalUse_1.logFunctionDuration)(exports.approveAirdropAsSpender.name, startTime); return result; }; exports.approveAirdropAsSpender = approveAirdropAsSpender; /** * Generate merkle tree info for a whitelist * * @param {WhiteListItem[]} whitelist - The list of items is available for airdrop. * @param {ThirdwebContract} airdropContract - The Airdrop Thirdweb contract. * @param {Address} tokenAddress - The token address to claim. * @param {boolean} isLogResult - Whether to log the result or not. Default true. * @returns {Promise<GenerateMerkleTreeInfo>} A promise that resolves to the generated info. * @throws An error if the wallet is zero. * @example * ```ts * import { generateMerkleTreeForWhitelist } from "./index"; * * const generateMerkleTreeInfo = await generateMerkleTreeForWhitelist( * whitelist, * airdropContract, * tokenAddress, * tokenContract * ); * ``` */ const generateMerkleTreeInfoERC20ForWhitelist = async (whitelist, airdropContract, tokenAddress, isLogResult = true) => { const startTime = (0, internalUse_1.logFunctionTrackStartTime)(exports.generateMerkleTreeInfoERC20ForWhitelist.name); const params = { contract: airdropContract, snapshot: whitelist, tokenAddress, }; const generateMerkleTreeInfo = await (0, airdrop_2.generateMerkleTreeInfoERC20)(params); if (isLogResult) { // TODO: replace with the logger library for better format or other targets console.log(' [generateMerkleTreeForWhitelist]'); console.log(` [request] params = ${JSON.stringify(params)}`); console.log(` [response] result = ${JSON.stringify(generateMerkleTreeInfo)}`); } (0, internalUse_1.logFunctionDuration)(exports.generateMerkleTreeInfoERC20ForWhitelist.name, startTime); return generateMerkleTreeInfo; }; exports.generateMerkleTreeInfoERC20ForWhitelist = generateMerkleTreeInfoERC20ForWhitelist; /** * Airdrop's owner saved merkleRoot to on-chain. * * @remarks * MUST execute saveSnapshotByOwner first. Order master * * @param {Account} account - The Account represent as sender @see {@link https://ethereum.org/en/glossary/#account|Account's Ethereum}. * @param {ThirdwebContract} airdropContract - The airdrop Thirdweb contract. * @param {Address} tokenAddress - The token address to claim. * @param {string} merkleRoot - The generated merkleRoot from whitelist @see {@link generateMerkleTreeInfoERC20ForWhitelist|Generate merkleRoot} * @param {RetryOptions} retryOptions - The configuration on retry * @param {ExtraGasOptions} extraGasOptions - The extra gas options bidding for your transaction to be included in the next block. * @returns {Promise<TransactionReceipt>} A promise that resolves to the confirmed transaction receipt. * @throws An error if the wallet is not connected. */ const saveMerkleRootByOwner = async (account, airdropContract, tokenAddress, merkleRoot, retryOptions = {}, extraGasOptions = {}) => { const startTime = (0, internalUse_1.logFunctionTrackStartTime)(exports.saveMerkleRootByOwner.name); const merkleRootTransaction = (0, airdrop_2.setMerkleRoot)({ contract: airdropContract, token: `0x${tokenAddress.replace('0x', '')}`, tokenMerkleRoot: `0x${merkleRoot.replace('0x', '')}`, resetClaimStatus: true, }); const result = await retryPrepareAndSubmitRawTransaction(merkleRootTransaction, account, retryOptions, extraGasOptions); (0, internalUse_1.logFunctionDuration)(exports.saveMerkleRootByOwner.name, startTime); return result; }; exports.saveMerkleRootByOwner = saveMerkleRootByOwner; /** * Airdrop's owner saved snapshotUri to on-chain. * * @param {Account} account - The Account represent as sender @see {@link https://ethereum.org/en/glossary/#account|Account's Ethereum}. * @param {ThirdwebContract} airdropContract - The airdrop Thirdweb contract. * @param {string} merkleRoot - The generated merkleRoot from whitelist @see {@link generateMerkleTreeInfoERC20ForWhitelist|Generate merkleRoot} * @param {string} snapshotUri - The generated snapshotUri from whitelist @see {@link generateMerkleTreeInfoERC20ForWhitelist|Generate snapshotUri} * @param {RetryOptions} retryOptions - The configuration on retry * @param {ExtraGasOptions} extraGasOptions - The extra gas options bidding for your transaction to be included in the next block. * @returns {Promise<TransactionReceipt>} A promise that resolves to the confirmed transaction receipt. * @throws An error if the wallet is not connected. */ const saveSnapshotByOwner = async (account, airdropContract, merkleRoot, snapshotUri, retryOptions = {}, extraGasOptions = {}) => { const startTime = (0, internalUse_1.logFunctionTrackStartTime)(exports.saveSnapshotByOwner.name); const saveSnapshotTransaction = (0, airdrop_2.saveSnapshot)({ contract: airdropContract, merkleRoot, snapshotUri, }); const result = await retryPrepareAndSubmitRawTransaction(saveSnapshotTransaction, account, retryOptions, extraGasOptions); (0, internalUse_1.logFunctionDuration)(exports.saveSnapshotByOwner.name, startTime); return result; }; exports.saveSnapshotByOwner = saveSnapshotByOwner; /** * Sends a transaction using the provided wallet. * @param {SendTransactionOptions} options - The options for sending the transaction. * @param {number} maxBlocksWaitTime - The maximum of blocks to wait for confirmation before considering success. * @returns {Promise<TransactionReceipt>} A promise that resolves to the confirmed transaction receipt. * @throws An error if the wallet is not connected. * @example * ```ts * import { sendAndConfirmTransaction } from "./index"; * * const transactionReceipt = await sendAndConfirmTransaction( * options, * maxBlocksWaitTime * ); * ``` */ async function sendTransactionAndWaitForReceipt(options, maxBlocksWaitTime = ConstType_1.DEFAULT_MAX_BLOCKS_WAIT_TIME) { const submittedTx = await (0, thirdweb_1.sendTransaction)(options); return (0, thirdweb_1.waitForReceipt)({ ...submittedTx, maxBlocksWaitTime, }); } exports.sendTransactionAndWaitForReceipt = sendTransactionAndWaitForReceipt; /** * Retry on preparing the gas fee, and nonce, and send a transaction using the provided wallet. * * @param {PreparedTransaction} transaction - The raw transaction to submit * @param {Account} account - The Account represent as sender @see {@link https://ethereum.org/en/glossary/#account|Account's Ethereum}. * @param {ExtraGasOptions} extraGasOptions - The extra gas options bidding for your transaction to be included in the next block. Otherwise, Use default our sdk {@link Type | Default Variables} * @param {RetryOptions} retryOptions - The configuration on retry * @returns {Promise<TransactionReceipt>} A promise that resolves to the confirmed transaction receipt. * @throws An error if the wallet is not connected. */ /* eslint-disable @typescript-eslint/no-explicit-any */ async function retryPrepareAndSubmitRawTransaction(transaction, account, retryOptions = {}, extraGasOptions = {}) { const { extraGasPercentage = ConstType_1.DEFAULT_EXTRA_GAS_PERCENTAGE, extraMaxPriorityFeePerGasPercentage = ConstType_1.DEFAULT_EXTRA_PRIORITY_TIP_PERCENTAGE, extraOnRetryPercentage = ConstType_1.DEFAULT_EXTRA_ON_RETRY_PERCENTAGE, } = extraGasOptions; const { client, chain } = transaction; return (0, Retry_1.retry)(async (retryCount) => { // Get transaction nonce const nextNonce = await (0, exports.getNextNonce)(account.address, client, { chain, }); // Calculate gas fee of a transaction const gasFeeInfo = await (0, exports.getGasFeeInfo)({ transaction, account }, { extraGasPercentage, extraMaxPriorityFeePerGasPercentage, extraOnRetryPercentage: extraOnRetryPercentage * retryCount, }); const sendingSnapshotTransaction = { ...transaction, ...gasFeeInfo, nonce: nextNonce, }; return await sendTransactionAndWaitForReceipt({ transaction: sendingSnapshotTransaction, account: account, }); }, retryOptions); } exports.retryPrepareAndSubmitRawTransaction = retryPrepareAndSubmitRawTransaction; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"core.js","sourceRoot":"","sources":["../../src/transaction/core.ts"],"names":[],"mappings":"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"}