UNPKG

aptos

Version:
481 lines (405 loc) 18.2 kB
// Copyright © Aptos Foundation // SPDX-License-Identifier: Apache-2.0 import { HexString } from "../../utils"; import { AccountAddress, Identifier, objectStructTag, optionStructTag, StructTag, TransactionArgumentAddress, TransactionArgumentBool, TransactionArgumentU128, TransactionArgumentU64, TransactionArgumentU8, TransactionArgumentU8Vector, TypeTagAddress, TypeTagBool, TypeTagParser, TypeTagStruct, TypeTagU128, TypeTagU16, TypeTagU256, TypeTagU32, TypeTagU64, TypeTagU8, TypeTagVector, } from "../../aptos_types"; import { Serializer } from "../../bcs"; import { argToTransactionArgument, serializeArg, ensureBoolean, ensureNumber, ensureBigInt, } from "../../transaction_builder/builder_utils"; import { stringStructTag } from "../../aptos_types/type_tag"; describe("BuilderUtils", () => { it("parses a bool TypeTag", async () => { expect(new TypeTagParser("bool").parseTypeTag() instanceof TypeTagBool).toBeTruthy(); }); it("parses a u8 TypeTag", async () => { expect(new TypeTagParser("u8").parseTypeTag() instanceof TypeTagU8).toBeTruthy(); }); it("parses a u16 TypeTag", async () => { expect(new TypeTagParser("u16").parseTypeTag() instanceof TypeTagU16).toBeTruthy(); }); it("parses a u32 TypeTag", async () => { expect(new TypeTagParser("u32").parseTypeTag() instanceof TypeTagU32).toBeTruthy(); }); it("parses a u64 TypeTag", async () => { expect(new TypeTagParser("u64").parseTypeTag() instanceof TypeTagU64).toBeTruthy(); }); it("parses a u128 TypeTag", async () => { expect(new TypeTagParser("u128").parseTypeTag() instanceof TypeTagU128).toBeTruthy(); }); it("parses a u256 TypeTag", async () => { expect(new TypeTagParser("u256").parseTypeTag() instanceof TypeTagU256).toBeTruthy(); }); it("parses a address TypeTag", async () => { expect(new TypeTagParser("address").parseTypeTag() instanceof TypeTagAddress).toBeTruthy(); }); it("parses a vector TypeTag", async () => { const vectorAddress = new TypeTagParser("vector<address>").parseTypeTag(); expect(vectorAddress instanceof TypeTagVector).toBeTruthy(); expect((vectorAddress as TypeTagVector).value instanceof TypeTagAddress).toBeTruthy(); const vectorU64 = new TypeTagParser(" vector < u64 > ").parseTypeTag(); expect(vectorU64 instanceof TypeTagVector).toBeTruthy(); expect((vectorU64 as TypeTagVector).value instanceof TypeTagU64).toBeTruthy(); }); it("parses a struct TypeTag", async () => { const assertStruct = (struct: TypeTagStruct, accountAddress: string, moduleName: string, structName: string) => { expect(HexString.fromUint8Array(struct.value.address.address).toShortString()).toBe(accountAddress); expect(struct.value.module_name.value).toBe(moduleName); expect(struct.value.name.value).toBe(structName); }; const coin = new TypeTagParser("0x1::test_coin::Coin").parseTypeTag(); expect(coin instanceof TypeTagStruct).toBeTruthy(); assertStruct(coin as TypeTagStruct, "0x1", "test_coin", "Coin"); const aptosCoin = new TypeTagParser( "0x1::coin::CoinStore < 0x1::test_coin::AptosCoin1 , 0x1::test_coin::AptosCoin2 > ", ).parseTypeTag(); expect(aptosCoin instanceof TypeTagStruct).toBeTruthy(); assertStruct(aptosCoin as TypeTagStruct, "0x1", "coin", "CoinStore"); const aptosCoinTrailingComma = new TypeTagParser( "0x1::coin::CoinStore < 0x1::test_coin::AptosCoin1 , 0x1::test_coin::AptosCoin2, > ", ).parseTypeTag(); expect(aptosCoinTrailingComma instanceof TypeTagStruct).toBeTruthy(); assertStruct(aptosCoinTrailingComma as TypeTagStruct, "0x1", "coin", "CoinStore"); const structTypeTags = (aptosCoin as TypeTagStruct).value.type_args; expect(structTypeTags.length).toBe(2); const structTypeTag1 = structTypeTags[0]; assertStruct(structTypeTag1 as TypeTagStruct, "0x1", "test_coin", "AptosCoin1"); const structTypeTag2 = structTypeTags[1]; assertStruct(structTypeTag2 as TypeTagStruct, "0x1", "test_coin", "AptosCoin2"); const coinComplex = new TypeTagParser( // eslint-disable-next-line max-len "0x1::coin::CoinStore < 0x2::coin::LPCoin < 0x1::test_coin::AptosCoin1 <u8>, vector<0x1::test_coin::AptosCoin2 > > >", ).parseTypeTag(); expect(coinComplex instanceof TypeTagStruct).toBeTruthy(); assertStruct(coinComplex as TypeTagStruct, "0x1", "coin", "CoinStore"); const coinComplexTypeTag = (coinComplex as TypeTagStruct).value.type_args[0]; assertStruct(coinComplexTypeTag as TypeTagStruct, "0x2", "coin", "LPCoin"); expect(() => { new TypeTagParser("0x1::test_coin").parseTypeTag(); }).toThrow("Invalid type tag."); expect(() => { new TypeTagParser("0x1::test_coin::CoinStore<0x1::test_coin::AptosCoin").parseTypeTag(); }).toThrow("Invalid type tag."); expect(() => { new TypeTagParser("0x1::test_coin::CoinStore<0x1::test_coin>").parseTypeTag(); }).toThrow("Invalid type tag."); expect(() => { new TypeTagParser("0x1:test_coin::AptosCoin").parseTypeTag(); }).toThrow("Unrecognized token."); expect(() => { new TypeTagParser("0x!::test_coin::AptosCoin").parseTypeTag(); }).toThrow("Unrecognized token."); expect(() => { new TypeTagParser("0x1::test_coin::AptosCoin<").parseTypeTag(); }).toThrow("Invalid type tag."); expect(() => { new TypeTagParser("0x1::test_coin::CoinStore<0x1::test_coin::AptosCoin,").parseTypeTag(); }).toThrow("Invalid type tag."); expect(() => { new TypeTagParser("").parseTypeTag(); }).toThrow("Invalid type tag."); expect(() => { new TypeTagParser("0x1::<::CoinStore<0x1::test_coin::AptosCoin,").parseTypeTag(); }).toThrow("Invalid type tag."); expect(() => { new TypeTagParser("0x1::test_coin::><0x1::test_coin::AptosCoin,").parseTypeTag(); }).toThrow("Invalid type tag."); expect(() => { new TypeTagParser("u3").parseTypeTag(); }).toThrow("Invalid type tag."); }); it("serializes a boolean arg", async () => { let serializer = new Serializer(); serializeArg(true, new TypeTagBool(), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0x01])); }); it("throws on serializing an invalid boolean arg", async () => { let serializer = new Serializer(); expect(() => { serializeArg(123, new TypeTagBool(), serializer); }).toThrow(/Invalid arg/); }); it("serializes a u8 arg", async () => { let serializer = new Serializer(); serializeArg(255, new TypeTagU8(), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0xff])); }); it("throws on serializing an invalid u8 arg", async () => { let serializer = new Serializer(); expect(() => { serializeArg("u8", new TypeTagU8(), serializer); }).toThrow(/Invalid number string/); }); it("serializes a u16 arg", async () => { let serializer = new Serializer(); serializeArg(0x7fff, new TypeTagU16(), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0xff, 0x7f])); }); it("throws on serializing an invalid u16 arg", async () => { let serializer = new Serializer(); expect(() => { serializeArg("u16", new TypeTagU16(), serializer); }).toThrow(/Invalid number string/); }); it("serializes a u32 arg", async () => { let serializer = new Serializer(); serializeArg(0x01020304, new TypeTagU32(), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0x04, 0x03, 0x02, 0x01])); }); it("throws on serializing an invalid u32 arg", async () => { let serializer = new Serializer(); expect(() => { serializeArg("u32", new TypeTagU32(), serializer); }).toThrow(/Invalid number string/); }); it("serializes a u64 arg", async () => { let serializer = new Serializer(); serializeArg(BigInt("18446744073709551615"), new TypeTagU64(), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff])); }); it("throws on serializing an invalid u64 arg", async () => { let serializer = new Serializer(); expect(() => { serializeArg("u64", new TypeTagU64(), serializer); }).toThrow(/^Cannot convert/); }); it("serializes a u128 arg", async () => { let serializer = new Serializer(); serializeArg(BigInt("340282366920938463463374607431768211455"), new TypeTagU128(), serializer); expect(serializer.getBytes()).toEqual( new Uint8Array([0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff]), ); }); it("throws on serializing an invalid u128 arg", async () => { let serializer = new Serializer(); expect(() => { serializeArg("u128", new TypeTagU128(), serializer); }).toThrow(/^Cannot convert/); }); it("serializes a u256 arg", async () => { let serializer = new Serializer(); serializeArg( BigInt("0x0001020304050607080910111213141516171819202122232425262728293031"), new TypeTagU256(), serializer, ); expect(serializer.getBytes()).toEqual( new Uint8Array([ 0x31, 0x30, 0x29, 0x28, 0x27, 0x26, 0x25, 0x24, 0x23, 0x22, 0x21, 0x20, 0x19, 0x18, 0x17, 0x16, 0x15, 0x14, 0x13, 0x12, 0x11, 0x10, 0x09, 0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01, 0x00, ]), ); }); it("throws on serializing an invalid u256 arg", async () => { let serializer = new Serializer(); expect(() => { serializeArg("u256", new TypeTagU256(), serializer); }).toThrow(/^Cannot convert/); }); it("serializes an AccountAddress arg", async () => { let serializer = new Serializer(); serializeArg("0x1", new TypeTagAddress(), serializer); expect(HexString.fromUint8Array(serializer.getBytes()).toShortString()).toEqual("0x1"); serializer = new Serializer(); serializeArg(HexString.ensure("0x1"), new TypeTagAddress(), serializer); expect(HexString.fromUint8Array(serializer.getBytes()).toShortString()).toEqual("0x1"); serializer = new Serializer(); serializeArg(AccountAddress.fromHex("0x1"), new TypeTagAddress(), serializer); expect(HexString.fromUint8Array(serializer.getBytes()).toShortString()).toEqual("0x1"); }); it("throws on serializing an invalid AccountAddress arg", async () => { let serializer = new Serializer(); expect(() => { serializeArg(123456, new TypeTagAddress(), serializer); }).toThrow("Invalid account address."); }); it("serializes a vector arg", async () => { let serializer = new Serializer(); serializeArg([255], new TypeTagVector(new TypeTagU8()), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0x1, 0xff])); }); it("serializes a vector u8 arg from string characters", async () => { let serializer = new Serializer(); serializeArg("abc", new TypeTagVector(new TypeTagU8()), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0x3, 0x61, 0x62, 0x63])); }); it("serializes a vector u8 arg from a hex string", async () => { let serializer = new Serializer(); serializeArg(HexString.ensure("0x010203"), new TypeTagVector(new TypeTagU8()), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0x3, 0x01, 0x02, 0x03])); }); it("serializes a vector u8 arg from a uint8array", async () => { let serializer = new Serializer(); serializeArg(new Uint8Array([0x61, 0x62, 0x63]), new TypeTagVector(new TypeTagU8()), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0x3, 0x61, 0x62, 0x63])); }); it("serializes a vector of Objects", async () => { let serializer = new Serializer(); serializeArg(["0xbeef"], new TypeTagVector(new TypeTagStruct(objectStructTag(new TypeTagU8()))), serializer); expect(serializer.getBytes()).toEqual( new Uint8Array([ 0x1, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xbe, 0xef, ]), ); }); it("throws error when serializing a mismatched type", async () => { let serializer = new Serializer(); expect(() => { serializeArg(123456, new TypeTagVector(new TypeTagU8()), serializer); }).toThrow("Invalid vector args."); }); it("serializes a string arg", async () => { let serializer = new Serializer(); serializeArg("abc", new TypeTagStruct(stringStructTag), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0x3, 0x61, 0x62, 0x63])); }); it("serializes an empty option arg", async () => { let serializer = new Serializer(); serializeArg(undefined, new TypeTagStruct(optionStructTag(new TypeTagU8())), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0x0])); let serializer2 = new Serializer(); serializeArg(null, new TypeTagStruct(optionStructTag(new TypeTagU8())), serializer2); expect(serializer2.getBytes()).toEqual(new Uint8Array([0x0])); }); it("serializes an option num arg", async () => { let serializer = new Serializer(); serializeArg("1", new TypeTagStruct(optionStructTag(new TypeTagU8())), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0x1, 0x1])); }); it("serializes an option string arg", async () => { let serializer = new Serializer(); serializeArg("abc", new TypeTagStruct(optionStructTag(new TypeTagStruct(stringStructTag))), serializer); expect(serializer.getBytes()).toEqual(new Uint8Array([0x1, 0x3, 0x61, 0x62, 0x63])); }); it("serializes a optional Object", async () => { let serializer = new Serializer(); serializeArg( "0x01", new TypeTagStruct(optionStructTag(new TypeTagStruct(objectStructTag(new TypeTagU8())))), serializer, ); //00 00 00 00 00000000 00000000 00000000 00000000 00000000 00000000 00000000 expect(serializer.getBytes()).toEqual( new Uint8Array([ 0x1, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, ]), ); }); it("throws when unsupported struct type", async () => { let serializer = new Serializer(); expect(() => { serializeArg( "abc", new TypeTagStruct( new StructTag(AccountAddress.fromHex("0x3"), new Identifier("token"), new Identifier("Token"), []), ), serializer, ); }).toThrow("Unsupported struct type in function argument"); }); it("throws at unrecognized arg types", async () => { const serializer = new Serializer(); expect(() => { // @ts-ignore serializeArg(123456, "unknown_type", serializer); }).toThrow("Unsupported arg type."); }); it("converts a boolean TransactionArgument", async () => { const res = argToTransactionArgument(true, new TypeTagBool()); expect((res as TransactionArgumentBool).value).toEqual(true); expect(() => { argToTransactionArgument(123, new TypeTagBool()); }).toThrow(/Invalid arg/); }); it("converts a u8 TransactionArgument", async () => { const res = argToTransactionArgument(123, new TypeTagU8()); expect((res as TransactionArgumentU8).value).toEqual(123); expect(() => { argToTransactionArgument("u8", new TypeTagBool()); }).toThrow(/Invalid boolean string/); }); it("converts a u64 TransactionArgument", async () => { const res = argToTransactionArgument(123, new TypeTagU64()); expect((res as TransactionArgumentU64).value).toEqual(BigInt(123)); expect(() => { argToTransactionArgument("u64", new TypeTagU64()); }).toThrow(/Cannot convert/); }); it("converts a u128 TransactionArgument", async () => { const res = argToTransactionArgument(123, new TypeTagU128()); expect((res as TransactionArgumentU128).value).toEqual(BigInt(123)); expect(() => { argToTransactionArgument("u128", new TypeTagU128()); }).toThrow(/Cannot convert/); }); it("converts an AccountAddress TransactionArgument", async () => { let res = argToTransactionArgument("0x1", new TypeTagAddress()) as TransactionArgumentAddress; expect(HexString.fromUint8Array(res.value.address).toShortString()).toEqual("0x1"); res = argToTransactionArgument(AccountAddress.fromHex("0x2"), new TypeTagAddress()) as TransactionArgumentAddress; expect(HexString.fromUint8Array(res.value.address).toShortString()).toEqual("0x2"); expect(() => { argToTransactionArgument(123456, new TypeTagAddress()); }).toThrow("Invalid account address."); }); it("converts a vector TransactionArgument", async () => { const res = argToTransactionArgument( new Uint8Array([0x1]), new TypeTagVector(new TypeTagU8()), ) as TransactionArgumentU8Vector; expect(res.value).toEqual(new Uint8Array([0x1])); expect(() => { argToTransactionArgument(123456, new TypeTagVector(new TypeTagU8())); }).toThrow(/.*should be an instance of Uint8Array$/); }); it("throws at unrecognized TransactionArgument types", async () => { expect(() => { // @ts-ignore argToTransactionArgument(123456, "unknown_type"); }).toThrow("Unknown type for TransactionArgument."); }); it("ensures a boolean", async () => { expect(ensureBoolean(false)).toBe(false); expect(ensureBoolean(true)).toBe(true); expect(ensureBoolean("true")).toBe(true); expect(ensureBoolean("false")).toBe(false); expect(() => ensureBoolean("True")).toThrow("Invalid boolean string."); }); it("ensures a number", async () => { expect(ensureNumber(10)).toBe(10); expect(ensureNumber("123")).toBe(123); expect(() => ensureNumber("True")).toThrow("Invalid number string."); }); it("ensures a bigint", async () => { expect(ensureBigInt(10)).toBe(BigInt(10)); expect(ensureBigInt("123")).toBe(BigInt(123)); expect(() => ensureBigInt("True")).toThrow(/^Cannot convert/); }); });