@ledgerhq/coin-internet_computer
Version:
Ledger Internet Computer integration
90 lines • 4.31 kB
JavaScript
import { log } from "@ledgerhq/logs";
import { FETCH_TXNS_LIMIT, MAINNET_INDEX_CANISTER_ID, MAINNET_LEDGER_CANISTER_ID, ICP_NETWORK_URL, } from "../consts";
import { ledgerIdlFactory, indexIdlFactory, getCanisterIdlFunc, Principal, encodeCanisterIdlFunc, decodeCanisterIdlFunc, } from "@zondax/ledger-live-icp";
import BigNumber from "bignumber.js";
import { fromNullable } from "@zondax/ledger-live-icp/utils";
import { getAgent } from "@zondax/ledger-live-icp/agent";
import invariant from "invariant";
export const fetchBlockHeight = async () => {
const canisterId = Principal.fromText(MAINNET_LEDGER_CANISTER_ID);
const queryBlocksRawRequest = {
start: BigInt(0),
length: BigInt(1),
};
const queryBlocksIdlFunc = getCanisterIdlFunc(ledgerIdlFactory, "query_blocks");
const queryBlocksargs = encodeCanisterIdlFunc(queryBlocksIdlFunc, [queryBlocksRawRequest]);
const agent = await getAgent(ICP_NETWORK_URL);
const blockHeightRes = await agent.query(canisterId, {
arg: queryBlocksargs,
methodName: "query_blocks",
});
invariant(blockHeightRes.status === "replied", "[ICP](fetchBlockHeight) Query failed");
const decodedIdl = decodeCanisterIdlFunc(queryBlocksIdlFunc, blockHeightRes.reply.arg);
const decoded = fromNullable(decodedIdl);
invariant(decoded, "[ICP](fetchBlockHeight) Decoding failed");
return BigNumber(decoded.chain_length.toString());
};
export const broadcastTxn = async (payload, canisterId, type) => {
log("debug", `[ICP] Broadcasting ${type} to ${canisterId}, body: ${payload.toString("hex")}`);
const res = await fetch(`${ICP_NETWORK_URL}/api/v3/canister/${canisterId}/${type}`, {
body: payload,
method: "POST",
headers: {
"Content-Type": "application/cbor",
},
});
if (res.status === 200) {
return await res.arrayBuffer();
}
throw new Error(`Failed to broadcast transaction: ${res.text()}`);
};
export const fetchBalance = async (address) => {
const agent = await getAgent(ICP_NETWORK_URL);
const indexCanister = Principal.fromText(MAINNET_INDEX_CANISTER_ID);
const getBalanceIdlFunc = getCanisterIdlFunc(indexIdlFactory, "get_account_identifier_balance");
const getBalanceArgs = encodeCanisterIdlFunc(getBalanceIdlFunc, [address]);
const balanceRes = await agent.query(indexCanister, {
arg: getBalanceArgs,
methodName: "get_account_identifier_balance",
});
if (balanceRes.status !== "replied") {
log("debug", `[ICP](fetchBalance) Query failed: ${balanceRes.status}`);
return BigNumber(0);
}
const decodedBalance = decodeCanisterIdlFunc(getBalanceIdlFunc, balanceRes.reply.arg);
const balance = fromNullable(decodedBalance);
if (!balance) {
return BigNumber(0);
}
return BigNumber(balance.toString());
};
export const fetchTxns = async (address, startBlockHeight, stopBlockHeight = BigInt(0)) => {
if (startBlockHeight <= stopBlockHeight) {
return [];
}
const agent = await getAgent(ICP_NETWORK_URL);
const canisterId = Principal.fromText(MAINNET_INDEX_CANISTER_ID);
const transactionsRawRequest = {
account_identifier: address,
start: [startBlockHeight],
max_results: BigInt(FETCH_TXNS_LIMIT),
};
const getTransactionsIdlFunc = getCanisterIdlFunc(indexIdlFactory, "get_account_identifier_transactions");
const getTransactionsArgs = encodeCanisterIdlFunc(getTransactionsIdlFunc, [
transactionsRawRequest,
]);
const transactionsRes = await agent.query(canisterId, {
arg: getTransactionsArgs,
methodName: "get_account_identifier_transactions",
});
invariant(transactionsRes.status === "replied", "[ICP](fetchTxns) Query failed");
const decodedTransactions = decodeCanisterIdlFunc(getTransactionsIdlFunc, transactionsRes.reply.arg);
const response = fromNullable(decodedTransactions);
invariant(response, "[ICP](fetchTxns) Decoding failed");
if (response.Ok.transactions.length === 0) {
return [];
}
const nextTxns = await fetchTxns(address, response.Ok.transactions.at(-1)?.id ?? BigInt(0), stopBlockHeight);
return [...response.Ok.transactions, ...nextTxns];
};
//# sourceMappingURL=api.js.map