UNPKG

@ledgerhq/coin-celo

Version:
162 lines 7.17 kB
import { celoKit } from "../network/sdk"; import { BigNumber } from "bignumber.js"; import { getPendingStakingOperationAmounts, getVote } from "../logic"; import { findSubAccountById } from "@ledgerhq/coin-framework/account/index"; import { CELO_STABLE_TOKENS, getStableTokenEnum, MAX_FEES_THRESHOLD_MULTIPLIER, } from "../constants"; const buildTransaction = async (account, transaction) => { const kit = celoKit(); const tokenAccount = findSubAccountById(account, transaction.subAccountId || ""); const isTokenTransaction = tokenAccount?.type === "TokenAccount"; let value = transactionValue(account, transaction); let celoTransaction; if (transaction.mode === "lock") { const lockedGold = await kit.contracts.getLockedGold(); celoTransaction = { from: account.freshAddress, value: value.toFixed(), to: lockedGold.address, data: lockedGold.lock().txo.encodeABI(), gas: await lockedGold.lock().txo.estimateGas({ from: account.freshAddress, value: value.toFixed(), }), }; } else if (transaction.mode === "unlock") { const lockedGold = await kit.contracts.getLockedGold(); celoTransaction = { from: account.freshAddress, to: lockedGold.address, data: lockedGold.unlock(value).txo.encodeABI(), gas: await lockedGold.unlock(value).txo.estimateGas({ from: account.freshAddress, }), }; } else if (transaction.mode === "withdraw") { const lockedGold = await kit.contracts.getLockedGold(); celoTransaction = { from: account.freshAddress, to: lockedGold.address, data: lockedGold.withdraw(transaction.index || 0).txo.encodeABI(), gas: await lockedGold.withdraw(transaction.index || 0).txo.estimateGas({ from: account.freshAddress, value: value.toFixed(), }), }; } else if (transaction.mode === "vote") { const election = await kit.contracts.getElection(); const vote = await election.vote(transaction.recipient, new BigNumber(value)); celoTransaction = { from: account.freshAddress, to: election.address, data: vote.txo.encodeABI(), gas: await vote.txo.estimateGas({ from: account.freshAddress }), }; } else if (transaction.mode === "revoke") { const election = await kit.contracts.getElection(); const accounts = await kit.contracts.getAccounts(); const voteSignerAccount = await accounts.voteSignerToAccount(account.freshAddress); const revokes = await election.revoke(voteSignerAccount, transaction.recipient, new BigNumber(value)); const revoke = revokes.find(transactionObject => { return (transactionObject.txo._method.name === (transaction.index === 0 ? "revokePending" : "revokeActive")); }); if (!revoke) throw new Error("No votes to revoke"); celoTransaction = { from: account.freshAddress, to: election.address, data: revoke.txo.encodeABI(), gas: await revoke.txo.estimateGas({ from: account.freshAddress }), }; } else if (transaction.mode === "activate") { const election = await kit.contracts.getElection(); const accounts = await kit.contracts.getAccounts(); const voteSignerAccount = await accounts.voteSignerToAccount(account.freshAddress); const activates = await election.activate(voteSignerAccount); const activate = activates.find(a => a.txo.arguments[0] === transaction.recipient); if (!activate) throw new Error("No votes to activate"); celoTransaction = { from: account.freshAddress, to: election.address, data: activate.txo.encodeABI(), gas: await activate.txo.estimateGas({ from: account.freshAddress, }), }; } else if (transaction.mode === "register") { const accounts = await kit.contracts.getAccounts(); celoTransaction = { from: account.freshAddress, to: accounts.address, data: accounts.createAccount().txo.encodeABI(), gas: await accounts.createAccount().txo.estimateGas({ from: account.freshAddress }), }; } else if (isTokenTransaction) { value = transaction.useAllAmount ? tokenAccount.balance : transaction.amount; const block = await kit.connection.web3.eth.getBlock("latest"); const maxPriorityFeePerGas = await kit.connection.getMaxPriorityFeePerGas(); const baseFee = BigInt(block.baseFeePerGas || maxPriorityFeePerGas); const maxFeePerGas = baseFee + BigInt(maxPriorityFeePerGas); let token; if (CELO_STABLE_TOKENS.includes(tokenAccount.token.id)) { token = await kit.contracts.getStableToken(getStableTokenEnum(tokenAccount.token.id)); } else { token = await kit.contracts.getErc20(tokenAccount.token.contractAddress); } celoTransaction = { from: account.freshAddress, to: transaction.recipient, data: token.transfer(transaction.recipient, value.toFixed()).txo.encodeABI(), value: value.toFixed(), }; } else { // Send celoTransaction = { from: account.freshAddress, to: transaction.recipient, value: value.toFixed(), }; } const gas = ((await kit.connection.estimateGasWithInflationFactor(celoTransaction)) * MAX_FEES_THRESHOLD_MULTIPLIER).toFixed(); const tx = { ...celoTransaction, gas, chainId: await kit.connection.chainId(), nonce: await kit.connection.nonce(account.freshAddress), }; return tx; }; const transactionValue = (account, transaction) => { let value = transaction.amount; if (transaction.useAllAmount) { if ((transaction.mode === "unlock" || transaction.mode === "vote") && account.celoResources) { // Deduct the amount of pending vote transactions from // the total non-voting locked balance to get the true non-voting locked balance. const pendingOperationAmounts = getPendingStakingOperationAmounts(account); const pendingOperationAmount = transaction.mode === "vote" ? pendingOperationAmounts.vote : new BigNumber(0); value = account.celoResources.nonvotingLockedBalance.minus(pendingOperationAmount); } else if (transaction.mode === "revoke" && account.celoResources) { const revoke = getVote(account, transaction.recipient, transaction.index); if (revoke?.amount) value = revoke.amount; } else { value = account.spendableBalance.minus(transaction.fees || 0); } } return value; }; export default buildTransaction; //# sourceMappingURL=buildTransaction.js.map