UNPKG

@ledgerhq/coin-ton

Version:
381 lines (321 loc) 13.7 kB
import { encodeOperationId } from "@ledgerhq/coin-framework/lib/operation"; import BigNumber from "bignumber.js"; // eslint-disable-next-line no-restricted-imports import { Builder, Slice } from "@ton/core"; import flatMap from "lodash/flatMap"; import { TonJettonTransfer, TonTransaction } from "../../bridge/bridgeHelpers/api.types"; import { dataToSlice, decodeForwardPayload, loadSnakeBytes, mapJettonTxToOps, mapTxToOps, } from "../../bridge/bridgeHelpers/txn"; import { jettonTransferResponse, mockAccountId, mockAddress, tonTransactionResponse, } from "../fixtures/common.fixtures"; describe("Transaction functions", () => { describe("mapTxToOps", () => { it("should map an IN failed ton transaction without total_fees to a ledger operation", async () => { const { now, lt, hash, in_msg, total_fees, mc_block_seqno } = tonTransactionResponse.transactions[0]; const finalOperation = flatMap( tonTransactionResponse.transactions, mapTxToOps(mockAccountId, mockAddress, tonTransactionResponse.address_book), ); expect(finalOperation).toEqual([ { accountId: mockAccountId, blockHash: null, blockHeight: mc_block_seqno, date: new Date(now * 1000), // now is defined in seconds extra: { comment: { isEncrypted: false, text: "" }, explorerHash: hash, lt }, fee: BigNumber(total_fees), hasFailed: true, hash: in_msg?.hash, id: encodeOperationId(mockAccountId, in_msg?.hash ?? "", "IN"), recipients: [in_msg?.destination], senders: ["EQCVnqqL0OOiZi2BQnjVGm-ZeUYgfUhHgAi-vn9F8-94HwrH"], type: "IN", value: BigNumber(in_msg?.value ?? 0), subOperations: undefined, }, ]); }); it("should map an IN ton transaction with total_fees to a ledger operation", async () => { const transactions = [{ ...tonTransactionResponse.transactions[0], total_fees: "15" }]; const { now, lt, hash, in_msg, total_fees, mc_block_seqno } = transactions[0]; const finalOperation = flatMap( transactions, mapTxToOps(mockAccountId, mockAddress, tonTransactionResponse.address_book), ); expect(finalOperation).toEqual([ { accountId: mockAccountId, blockHash: null, blockHeight: mc_block_seqno, date: new Date(now * 1000), // now is defined in seconds extra: { comment: { isEncrypted: false, text: "" }, explorerHash: hash, lt }, fee: BigNumber(total_fees), hasFailed: true, hash: in_msg?.hash, id: encodeOperationId(mockAccountId, in_msg?.hash ?? "", "IN"), recipients: [in_msg?.destination], senders: ["EQCVnqqL0OOiZi2BQnjVGm-ZeUYgfUhHgAi-vn9F8-94HwrH"], type: "IN", value: BigNumber(in_msg?.value ?? 0), subOperations: [ { id: encodeOperationId(mockAccountId, in_msg?.hash ?? "", "NONE"), hash: in_msg?.hash, type: "NONE", value: BigNumber(total_fees), fee: BigNumber(0), blockHeight: mc_block_seqno, blockHash: null, hasFailed: true, accountId: mockAccountId, senders: [mockAddress], recipients: [], date: new Date(now * 1000), extra: { comment: { isEncrypted: false, text: "" }, explorerHash: hash, lt }, }, ], }, ]); }); it("should map a failed OUT ton transaction to a ledger operation", async () => { // The IN transaction will be used as OUT transaction and it will be adjusted const transactions: TonTransaction[] = [ { ...tonTransactionResponse.transactions[0], in_msg: null, }, ]; if (tonTransactionResponse.transactions[0].in_msg) { transactions[0].out_msgs = [ { ...tonTransactionResponse.transactions[0].in_msg, source: transactions[0].account }, ]; } const { now, lt, hash, out_msgs, total_fees, mc_block_seqno } = transactions[0]; const finalOperation = flatMap( transactions, mapTxToOps(mockAccountId, mockAddress, tonTransactionResponse.address_book), ); expect(finalOperation).toEqual([ { id: encodeOperationId(mockAccountId, hash, "OUT"), hash: out_msgs?.[0].hash, type: "OUT", value: BigNumber(out_msgs?.[0].value ?? 0), fee: BigNumber(total_fees), blockHeight: mc_block_seqno, blockHash: null, hasFailed: true, accountId: mockAccountId, senders: [transactions[0].account], recipients: ["EQDzd8aeBOU-jqYw_ZSuZjceI5p-F4b7HMprAsUJAtRPbJfg"], date: new Date(now * 1000), // now is defined in seconds extra: { comment: { isEncrypted: false, text: "" }, explorerHash: hash, lt }, }, ]); }); }); describe("mapJettonToOps", () => { it("should map an IN ton transaction without total_fees to a ledger operation", async () => { const { transaction_hash, amount, transaction_now, transaction_lt } = jettonTransferResponse.jetton_transfers[0]; const finalOperation = flatMap( jettonTransferResponse.jetton_transfers, mapJettonTxToOps(mockAccountId, mockAddress, tonTransactionResponse.address_book), ); const tokenByCurrencyAddress = `${mockAccountId}+ton%2Fjetton%2Feqavlwfdxgf2lxm67y4yzc17wykd9a0guwpkms1gosm~!underscore!~~!underscore!~not`; expect(finalOperation).toEqual([ { id: encodeOperationId(tokenByCurrencyAddress, transaction_hash, "IN"), hash: transaction_hash, type: "IN", value: BigNumber(amount), fee: BigNumber(0), blockHeight: 1, blockHash: null, hasFailed: false, accountId: tokenByCurrencyAddress, senders: ["EQDnqcVSV4S9m2Y9gLAQrDerQktKSx2I1uhs6r5o_H8VT9G-"], recipients: [mockAddress], date: new Date(transaction_now * 1000), // now is defined in seconds extra: { comment: { isEncrypted: false, text: "" }, explorerHash: transaction_hash, lt: transaction_lt, }, }, ]); }); it("should map an OUT jetton transaction to a ledger operation", async () => { // The IN jetton transaction will be used as OUT transaction and it will be adjusted const jettonTransfers: TonJettonTransfer[] = [ { ...jettonTransferResponse.jetton_transfers[0], }, ]; jettonTransfers[0].source = jettonTransfers[0].destination; jettonTransfers[0].destination = jettonTransferResponse.jetton_transfers[0].source; const { transaction_hash, amount, transaction_now, transaction_lt } = jettonTransfers[0]; const finalOperation = flatMap( jettonTransfers, mapJettonTxToOps(mockAccountId, mockAddress, tonTransactionResponse.address_book), ); const tokenByCurrencyAddress = `${mockAccountId}+ton%2Fjetton%2Feqavlwfdxgf2lxm67y4yzc17wykd9a0guwpkms1gosm~!underscore!~~!underscore!~not`; expect(finalOperation).toEqual([ { id: encodeOperationId(tokenByCurrencyAddress, transaction_hash, "OUT"), hash: transaction_hash, type: "OUT", value: BigNumber(amount), fee: BigNumber(0), blockHeight: 1, blockHash: null, hasFailed: false, accountId: tokenByCurrencyAddress, recipients: ["EQDnqcVSV4S9m2Y9gLAQrDerQktKSx2I1uhs6r5o_H8VT9G-"], senders: [mockAddress], date: new Date(transaction_now * 1000), // now is defined in seconds extra: { comment: { isEncrypted: false, text: "" }, explorerHash: transaction_hash, lt: transaction_lt, }, }, ]); }); }); }); describe("TON Payload Processing Functions", () => { describe("dataToSlice", () => { it("should convert base64 string to Slice when it's a valid BOC", () => { // Create a Cell from a string and convert to BOC const cell = new Builder().storeUint(123, 32).endCell(); const bocBase64 = cell.toBoc().toString("base64"); const result = dataToSlice(bocBase64); expect(result).toBeInstanceOf(Slice); expect(result?.loadUint(32)).toBe(123); }); it("should fallback to BitString when the data is not a valid BOC", () => { const invalidBocBase64 = "aW52YWxpZCB0b24gZGF0YQ=="; // "invalid ton data" const result = dataToSlice(invalidBocBase64); expect(result).toBeInstanceOf(Slice); }); it("should return undefined for non-string input", () => { // @ts-expect-error - Testing invalid input const result = dataToSlice(null); expect(result).toBeUndefined(); }); }); describe("loadSnakeBytes", () => { it("should load bytes from a simple slice without refs", () => { const cell = new Builder().storeBuffer(Buffer.from("Slice", "utf-8")).endCell(); const slice = cell.beginParse(); const result = loadSnakeBytes(slice); expect(result.toString("utf-8")).toBe("Slice"); }); it("should load bytes from a slice with refs (snake structure)", () => { // Create a chain of cells (snake structure) const cell2 = new Builder().storeBuffer(Buffer.from(" Data", "utf-8")).endCell(); const cell1 = new Builder() .storeBuffer(Buffer.from("Slice", "utf-8")) .storeRef(cell2) .endCell(); const slice = cell1.beginParse(); const result = loadSnakeBytes(slice); expect(result.toString("utf-8")).toBe("Slice Data"); }); it("should handle empty slice", () => { const cell = new Builder().endCell(); const slice = cell.beginParse(); const result = loadSnakeBytes(slice); expect(result.length).toBe(0); }); it("should handle slice with multiple refs in chain", () => { // Create a longer chain of cells (snake structure) const cell3 = new Builder().storeBuffer(Buffer.from("Part3", "utf-8")).endCell(); const cell2 = new Builder() .storeBuffer(Buffer.from("Part2", "utf-8")) .storeRef(cell3) .endCell(); const cell1 = new Builder() .storeBuffer(Buffer.from("Part1", "utf-8")) .storeRef(cell2) .endCell(); const slice = cell1.beginParse(); const result = loadSnakeBytes(slice); expect(result.toString("utf-8")).toBe("Part1Part2Part3"); }); }); describe("decodeForwardPayload", () => { it("should return empty string for null payload", () => { const result = decodeForwardPayload(null); expect(result).toBe(""); }); it("should decode a valid payload with opcode 0 containing text", () => { // Create a cell with opcode 0 followed by a text string const cell = new Builder() .storeUint(0, 32) // opcode 0 .storeBuffer(Buffer.from("This is the comment", "utf-8")) .endCell(); const bocBase64 = cell.toBoc().toString("base64"); const result = decodeForwardPayload(bocBase64); expect(result).toBe("This is the comment"); }); it("should return empty string for payloads with non-zero opcode", () => { // Create a cell with opcode 1 followed by some data const cell = new Builder() .storeUint(1, 32) // non-zero opcode .storeBuffer(Buffer.from("Should be ignored", "utf-8")) .endCell(); const bocBase64 = cell.toBoc().toString("base64"); const result = decodeForwardPayload(bocBase64); expect(result).toBe(""); }); it("should handle payload with unicode characters", () => { // Create a cell with opcode 0 followed by a text with unicode const cell = new Builder() .storeUint(0, 32) // opcode 0 .storeBuffer(Buffer.from("Unicode: 你好, мир, 🚀", "utf-8")) .endCell(); const bocBase64 = cell.toBoc().toString("base64"); const result = decodeForwardPayload(bocBase64); expect(result).toBe("Unicode: 你好, мир, 🚀"); }); it("should handle snake format payloads correctly", () => { // Create a chain of cells with opcode 0 followed by a long message const cell2 = new Builder() .storeBuffer(Buffer.from(" would need multiple cells to store.", "utf-8")) .endCell(); const cell1 = new Builder() .storeUint(0, 32) // opcode 0 .storeBuffer(Buffer.from("This is a very long message that", "utf-8")) .storeRef(cell2) .endCell(); const bocBase64 = cell1.toBoc().toString("base64"); const result = decodeForwardPayload(bocBase64); expect(result).toBe("This is a very long message that would need multiple cells to store."); }); it("should handle invalid payloads gracefully by returning empty string", () => { // Create an invalid base64 string const invalidBase64 = "!@#$%^&*()"; const result = decodeForwardPayload(invalidBase64); expect(result).toBe(""); }); it("should handle valid base64 but invalid BOC payloads", () => { // Valid base64 but not a valid BOC const validBase64NotBoc = "aW52YWxpZCB0b24gZGF0YQ=="; // "invalid ton data" in base64 const result = decodeForwardPayload(validBase64NotBoc); // Should return empty string as it's not a valid Cell expect(result).toBe(""); }); }); });