Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add script to merge and split coins #255

Merged
merged 54 commits into from
Aug 21, 2024
Merged
Show file tree
Hide file tree
Changes from 9 commits
Commits
Show all changes
54 commits
Select commit Hold shift + click to select a range
c6b7044
feat: add manage-coins script
npty Jun 3, 2024
67e25cb
feat: complete merge coins
npty Jun 4, 2024
7f9a837
feat: add split coins
npty Jun 4, 2024
11b8667
feat: group coins-related functions into class
npty Jun 4, 2024
3700c2d
chore: fix not get all coins
npty Jun 4, 2024
1ffbfd8
chore: simply return instead of exit program
npty Jun 4, 2024
b8d8c3b
chore: fix can't split all coins including gas token
npty Jun 4, 2024
9e94132
chore: fix lint
npty Jun 4, 2024
10bda3b
feat: add transfer option
npty Jun 5, 2024
2ca5c07
chore: fix lint
npty Jun 5, 2024
c60959f
chore: remove receipt
npty Jun 5, 2024
7549c0a
chore: update coinType ternary condition
npty Jun 6, 2024
318a3ed
chore: update program name and description
npty Jun 6, 2024
969e4c1
chore: update console.log to printInfo
npty Jun 6, 2024
511a61a
chore: refactor mergeCoin
npty Jun 6, 2024
14ab718
chore: use validateParameters
npty Jun 6, 2024
4775434
chore: export coinId in utils and use validateParameters
npty Jun 6, 2024
a832bb8
chore: split sui coin if coin type is not specified
npty Jun 6, 2024
b9ed184
chore: accept sui amount in full denom
npty Jun 6, 2024
08271f6
Merge branch 'main' into feat/script-merge-and-split-coins
npty Jun 6, 2024
5951312
chore: remove newline from printInfo
npty Jun 7, 2024
c707c77
chore: replace remaining console.log
npty Jun 7, 2024
3b8aa17
chore: print message when no coins to merge
npty Jun 7, 2024
74c3106
chore: remove doSplitCoins
npty Jun 7, 2024
09fd172
chore: remove doMergeCoin
npty Jun 7, 2024
a8557d7
Merge branch 'feat/script-merge-and-split-coins' of github.com:axelar…
npty Jun 7, 2024
e4c7552
chore: rename functions for consistency
npty Jun 7, 2024
638616c
chore: export isGasToken to utils
npty Jun 7, 2024
72ceefa
chore: refactor splitCoins
npty Jun 7, 2024
256f271
chore: refactor to use subcommands
npty Jun 7, 2024
580a22f
chore: reorder programs
npty Jun 7, 2024
78dc4fa
Merge branch 'main' into feat/script-merge-and-split-coins
blockchainguyy Jun 19, 2024
1b6e324
chore: add a util function for iterating over pagination response
npty Jun 21, 2024
2e09354
Merge branch 'feat/script-merge-and-split-coins' of github.com:axelar…
npty Jun 21, 2024
30cf4c4
chore: fix split coins
npty Jun 21, 2024
4576195
chore: fix object selection to split coin
npty Jun 21, 2024
862a837
Merge branch 'main' into feat/script-merge-and-split-coins
npty Jun 25, 2024
72d5614
Merge branch 'main' into feat/script-merge-and-split-coins
npty Jul 2, 2024
42a0a08
chore: replace signAndBroadcast
npty Jul 2, 2024
c7a4cc3
Merge branch 'main' into feat/script-merge-and-split-coins
npty Aug 1, 2024
25dc2f6
chore: refactor
npty Aug 2, 2024
7ad46d8
chore: adjust params
npty Aug 2, 2024
a5a63f0
chore: remove evm utils
npty Aug 2, 2024
657c6fc
chore: use addOptionsToCommands
npty Aug 2, 2024
3cf8a56
chore: update readme
npty Aug 2, 2024
f83fe44
chore: reword readme
npty Aug 2, 2024
344dd54
Merge branch 'main' into feat/script-merge-and-split-coins
blockchainguyy Aug 5, 2024
92255c2
chore: prettier
npty Aug 5, 2024
1be3d6d
Merge branch 'feat/script-merge-and-split-coins' of github.com:axelar…
npty Aug 5, 2024
6211df4
Merge branch 'main' into feat/script-merge-and-split-coins
npty Aug 13, 2024
51ad53e
Merge branch 'main' into feat/script-merge-and-split-coins
npty Aug 21, 2024
ff2576d
chore: update imports
npty Aug 21, 2024
9c7c9a8
chore: prettier
npty Aug 21, 2024
f1a5231
Merge branch 'main' into feat/script-merge-and-split-coins
npty Aug 21, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
112 changes: 48 additions & 64 deletions sui/manage-coins.js → sui/tokens.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,14 +3,14 @@ const { saveConfig } = require('../evm/utils');
const { Command } = require('commander');
const { addBaseOptions } = require('./cli-utils');
const { getWallet } = require('./sign-utils');
const { printInfo } = require('../evm/utils');
const { printInfo, printError, validateParameters } = require('../evm/utils');
const chalk = require('chalk');
const { loadSuiConfig } = require('./utils');

const SUI_COIN_ID = '0x2::sui::SUI';
const {
utils: { parseUnits },
} = require('ethers');
const { loadSuiConfig, SUI_COIN_ID } = require('./utils');

class CoinManager {

static async getAllCoins(client, account) {
let cursor;
const coinTypeToCoins = {};
Expand Down Expand Up @@ -43,48 +43,23 @@ class CoinManager {
}

static async splitCoins(tx, coinTypeToCoins, options) {
CoinManager.checkSplitAmount(options);
const splitAmount = BigInt(options.split);

console.log('\n==== Splitting Coins ====');
printInfo('\n==== Splitting Coins ====');
npty marked this conversation as resolved.
Show resolved Hide resolved
npty marked this conversation as resolved.
Show resolved Hide resolved

// Set coin type to given coin type or the first coin type if there's only one if it's a SUI token.
const hasOnlyGasToken = Object.keys(coinTypeToCoins).length === 1 && coinTypeToCoins[SUI_COIN_ID];
const coinType = options.coinType || hasOnlyGasToken ? SUI_COIN_ID : undefined;

if (coinType) {
// Throw an error if the coin type is specified but no coins are found
CoinManager.checkCoinType(coinType, coinTypeToCoins);
const coins = coinTypeToCoins[coinType];
const [coin] = CoinManager.doSplitCoins(tx, coins, splitAmount);
const splitAmount = BigInt(options.split);
const coinType = options.coinType || SUI_COIN_ID;
npty marked this conversation as resolved.
Show resolved Hide resolved

if (options.transfer) {
CoinManager.doTransfer(tx, coin, options.transfer);
}
} else {
for (const coinType in coinTypeToCoins) {
const coins = coinTypeToCoins[coinType];
if (this.isGasToken(coins.data[0])) continue;
const [coin] = CoinManager.doSplitCoins(tx, coins, splitAmount);

if (options.transfer) {
CoinManager.doTransfer(tx, coin, options.transfer);
}
}
}
const coins = coinTypeToCoins[coinType];
const [coin] = CoinManager.doSplitCoins(tx, coins, splitAmount);

if (options.transfer) {
console.log(`\nTransfer ${splitAmount} coins for every split coin to ${chalk.green(options.transfer)}`);
tx.transferObjects([coin], options.transfer);
printInfo(`\nTransfer ${splitAmount} coins for every split coin to ${chalk.green(options.transfer)}`);
}

// The transaction will fail if the gas budget is not set for splitting coins transaction
tx.setGasBudget(1e8);
}

static doTransfer(tx, coin, recipient) {
tx.transferObjects([coin], recipient);
}

static doSplitCoins(tx, coins, splitAmount) {
npty marked this conversation as resolved.
Show resolved Hide resolved
const firstObjectId = this.isGasToken(coins.data[0]) ? tx.gas : coins.data[0].coinObjectId;
const response = tx.splitCoins(firstObjectId, [splitAmount]);
npty marked this conversation as resolved.
Show resolved Hide resolved
Expand All @@ -93,20 +68,17 @@ class CoinManager {
}

static async mergeCoin(tx, coinTypeToCoins, options) {
const coinType = options.coinType;
console.log('\n==== Merging Coins ====');

// Throw an error if the coin type is specified but no coins are found
CoinManager.checkCoinType(coinType, coinTypeToCoins);
const coinTypes = options.coinType ? [options.coinType] : Object.keys(coinTypeToCoins);
printInfo('\n==== Merging Coins ====');

if (coinType) {
for (const coinType of coinTypes) {
const coins = coinTypeToCoins[coinType];
await CoinManager.doMergeCoin(tx, coins);
} else {
for (const coinType in coinTypeToCoins) {
const coins = coinTypeToCoins[coinType];
await CoinManager.doMergeCoin(tx, coins);

if (!coins) {
throw new Error(`No coins found for coin type ${coinType}`);
}

await CoinManager.doMergeCoin(tx, coins);
npty marked this conversation as resolved.
Show resolved Hide resolved
}
}

Expand Down Expand Up @@ -142,7 +114,27 @@ class CoinManager {
if (options.merge) {
await CoinManager.mergeCoin(tx, coinTypeToCoins, options);
} else if (options.split) {
await CoinManager.splitCoins(tx, coinTypeToCoins, options);
validateParameters({
isValidNumber: { split: options.split },
});

const coinType = options.coinType || SUI_COIN_ID;
const metadata = await client.getCoinMetadata({
coinType,
});

if (!metadata) {
printError(`No metadata found for coin type ${coinType}`);
process.exit(0);
}

const splitAmount = parseUnits(options.split, metadata.decimals);

await CoinManager.splitCoins(tx, coinTypeToCoins, {
...options,
split: splitAmount.toString(),
coinType,
npty marked this conversation as resolved.
Show resolved Hide resolved
});
}

const requireBroadcast = options.merge || options.split;
Expand All @@ -167,31 +159,22 @@ class CoinManager {

for (const coinType in coinTypeToCoins) {
const coins = coinTypeToCoins[coinType];
console.log(`Coin Type: ${chalk.green(coinType)}`);
console.log(`Total Balance: ${chalk.green(coins.totalBalance)}`);
console.log(`Total Objects: ${chalk.green(coins.data.length)}`);
printInfo(`Coin Type`, coinType);
printInfo(`Total Balance`, coins.totalBalance);
printInfo(`Total Objects`, coins.data.length);
}
}

static checkCoinType(coinType, coinTypeToCoins) {
if (coinType && !coinTypeToCoins[coinType]) {
console.error(`No coins found for coin type ${coinType}`);
printError(`No coins found for coin type ${coinType}`);
process.exit(0);
}
}

static isGasToken(coin) {
return coin.coinType === '0x2::sui::SUI';
}

static checkSplitAmount(options) {
try {
parseInt(options.split);
} catch (e) {
console.error('\nError: Please specify a valid split amount');
process.exit(0);
}
}
}

async function mainProcessor(options, processor) {
Expand All @@ -202,13 +185,14 @@ async function mainProcessor(options, processor) {

if (require.main === module) {
const program = new Command();
program.name('manage-coins').description('Merge or split coins for an account');

program.name('tokens').description('Token management tool (e.g. show balance, merge, split etc.)');

addBaseOptions(program);

program
.option('--merge', 'Merge all coins')
.option('--split <amount>', 'Split coins')
.option('--split <amount>', 'Split coins. The amount is expected to be in the full coin unit (e.g. 1.5 for 1_500_000_000 coins)')
.option('--coin-type <coinType>', 'Coin type to merge/split')
.option('--transfer <recipientAddress>', 'Used with --split to transfer the split coins to the recipient address')
.action((options) => {
Expand Down
3 changes: 3 additions & 0 deletions sui/utils.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,8 @@ const {
} = ethers;
const { CosmWasmClient } = require('@cosmjs/cosmwasm-stargate');

const SUI_COIN_ID = '0x2::sui::SUI';

const getAmplifierSigners = async (config, chain) => {
const client = await CosmWasmClient.connect(config.axelar.rpc);
const workerSet = await client.queryContractSmart(config.axelar.contracts.MultisigProver[chain].address, 'get_worker_set');
Expand Down Expand Up @@ -45,6 +47,7 @@ const loadSuiConfig = (env) => {
};

module.exports = {
SUI_COIN_ID,
getAmplifierSigners,
loadSuiConfig,
};
Loading