From 6b230670e300669eb490b0f36b1e8bf2fa8c640a Mon Sep 17 00:00:00 2001 From: Peter Smith Date: Thu, 3 Oct 2024 19:26:19 +0100 Subject: [PATCH 01/37] chore: added tests around all `abi-contract` tests --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 1236 +++++++++++++++++ packages/fuel-gauge/src/abi/constants.ts | 63 + packages/fuel-gauge/src/abi/vitest.matcher.ts | 20 + .../test/fixtures/forc-projects/Forc.toml | 1 + .../forc-projects/abi-contract/Forc.toml | 7 + .../abi-contract/src/data_structures.sw | 137 ++ .../abi-contract/src/equality.sw | 375 +++++ .../forc-projects/abi-contract/src/main.sw | 871 ++++++++++++ .../forc-projects/abi-contract/src/utils.sw | 26 + .../forc-projects/abi-library/Forc.toml | 7 + .../forc-projects/abi-library/src/lib.sw | 22 + 11 files changed, 2765 insertions(+) create mode 100644 packages/fuel-gauge/src/abi/abi-coder.test.ts create mode 100644 packages/fuel-gauge/src/abi/constants.ts create mode 100644 packages/fuel-gauge/src/abi/vitest.matcher.ts create mode 100644 packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/Forc.toml create mode 100644 packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/data_structures.sw create mode 100644 packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw create mode 100644 packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw create mode 100644 packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/utils.sw create mode 100644 packages/fuel-gauge/test/fixtures/forc-projects/abi-library/Forc.toml create mode 100644 packages/fuel-gauge/test/fixtures/forc-projects/abi-library/src/lib.sw diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts new file mode 100644 index 00000000000..8dda146d4d6 --- /dev/null +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -0,0 +1,1236 @@ +import { Contract, FuelError, Interface } from 'fuels'; +import type { + AssetId, + BigNumberish, + DecodedValue, + EvmAddress, + RawSlice, + WalletUnlocked, +} from 'fuels'; +import { expectToThrowFuelError, launchTestNode } from 'fuels/test-utils'; + +import { AbiContract, AbiContractFactory } from '../../test/typegen'; +import { EnumWithNativeInput, ExternalEnumInput } from '../../test/typegen/contracts/AbiContract'; +import type { + EnumWithBuiltinTypeInput, + EnumWithBuiltinTypeOutput, + EnumWithVectorInput, + EnumWithVectorOutput, + IdentityInput, + IdentityOutput, + StructDoubleGenericInput, + StructSimpleInput, + StructSimpleOutput, + StructWithGenericArrayInput, + StructWithMultiOptionInput, + StructWithMultiOptionOutput, + StructCInput, + StructWithNestedArrayInput, + StructWithNestedTupleInput, + StructSingleGenericInput, +} from '../../test/typegen/contracts/AbiContract'; +import type { Option, Result, Vec } from '../../test/typegen/contracts/common'; + +import { + U16_MAX, + U16_MIN, + U256_MAX, + U256_MIN, + U32_MAX, + U32_MIN, + U64_MAX, + U64_MIN, + U8_MAX, + U8_MIN, +} from './constants'; +import { toEqualBn } from './vitest.matcher'; + +expect.extend({ toEqualBn }); + +/** + * @group node + */ +describe('AbiCoder', () => { + let contract: AbiContract; + let wallet: WalletUnlocked; + let cleanup: () => void; + + beforeAll(async () => { + const launched = await launchTestNode({ + contractsConfigs: [{ factory: AbiContractFactory }], + }); + + const oldAbi = new Interface(AbiContract.abi); + + wallet = launched.wallets[0]; + contract = new Contract(launched.contracts[0].id, oldAbi, wallet) as AbiContract; + cleanup = launched.cleanup; + }); + + afterAll(() => { + cleanup(); + }); + + describe('types_u8', () => { + test('should encode/decode just fine', async () => { + const input = 8; + const expected = 255; + + const fn = contract.functions.types_u8(input); + + const { waitForResult } = await fn.call(); + + const { value } = await waitForResult(); + expect(value).toBe(expected); + }); + + test('should fail to encode/decode [min - 1]', async () => { + const input = U8_MIN - 1; + + await expectToThrowFuelError( + () => contract.functions.types_u8(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid u8.') + ); + }); + + test('should fail to encode/decode [max + 1]', async () => { + const input = U8_MAX + 1; + + await expectToThrowFuelError( + () => contract.functions.types_u8(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid u8, too many bytes.') + ); + }); + }); + + describe('types_u16', () => { + it('should encode/decode just fine', async () => { + const input = 16; + const expected = 65535; + + const { waitForResult } = await contract.functions.types_u16(input).call(); + + const { value } = await waitForResult(); + expect(value).toBe(expected); + }); + + it('should fail to encode/decode [min - 1]', async () => { + const input = U16_MIN - 1; + + await expectToThrowFuelError( + () => contract.functions.types_u16(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid u16.') + ); + }); + + it('should fail to encode/decode [max + 1]', async () => { + const input = U16_MAX + 1; + + await expectToThrowFuelError( + () => contract.functions.types_u16(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid u16, too many bytes.') + ); + }); + }); + + describe('types_u32', () => { + it('should encode/decode just fine', async () => { + const input = 32; + const expected = 4294967295; + + const { waitForResult } = await contract.functions.types_u32(input).call(); + + const { value } = await waitForResult(); + expect(value).toBe(expected); + }); + + it('should fail to encode/decode [min - 1]', async () => { + const input = U32_MIN - 1; + + await expectToThrowFuelError( + () => contract.functions.types_u32(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid u32.') + ); + }); + + it('should fail to encode/decode [max + 1]', async () => { + const input = U32_MAX + 1; + + await expectToThrowFuelError( + () => contract.functions.types_u32(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid u32, too many bytes.') + ); + }); + }); + + describe('types_u64', () => { + it('should encode/decode just fine', async () => { + const input = 64; + const expected = '4294967295000'; + + const { waitForResult } = await contract.functions.types_u64(input).call(); + + const { value } = await waitForResult(); + const actual = value.toString(); + expect(actual).toBe(expected); + }); + + it('should fail to encode/decode [min - 1]', async () => { + const input = U64_MIN - 1; + + await expectToThrowFuelError( + () => contract.functions.types_u64(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid u64.') + ); + }); + + it('should fail to encode/decode [max + 1]', async () => { + const input = U64_MAX.add(1); + + await expectToThrowFuelError( + () => contract.functions.types_u64(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid u64.') + ); + }); + }); + + describe('types_u256', () => { + it('should encode/decode just fine', async () => { + const input = 256; + const expected = '0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff'; + + const { waitForResult } = await contract.functions.types_u256(input).call(); + + const { value } = await waitForResult(); + const actual = value.toHex(); + expect(actual).toEqual(expected); + }); + + it('should fail to encode/decode [min - 1]', async () => { + const input = U256_MIN - 1; + + await expectToThrowFuelError( + () => contract.functions.types_u256(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid u256.') + ); + }); + + it('should fail to encode/decode [max + 1]', async () => { + const input = U256_MAX.add(1); + + await expectToThrowFuelError( + () => contract.functions.types_u256(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid u256.') + ); + }); + }); + + describe('types_bool', () => { + it('should encode/decode just fine', async () => { + const input = false; + const expected = true; + + const { waitForResult } = await contract.functions.types_bool(input).call(); + + const { value } = await waitForResult(); + expect(value).toBe(expected); + }); + + it('should fail to encode/decode [number]', async () => { + const input = 2; + + await expectToThrowFuelError( + () => contract.functions.types_bool(input as unknown as boolean).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid boolean value.') + ); + }); + + it('should fail to encode/decode [string]', async () => { + const input = '2'; + + await expectToThrowFuelError( + () => contract.functions.types_bool(input as unknown as boolean).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid boolean value.') + ); + }); + }); + + describe('types_b256', () => { + it('should encode/decode just fine', async () => { + const input = `0x${'a'.repeat(64)}`; + const expected = `0x${'0'.repeat(64)}`; + + const { waitForResult } = await contract.functions.types_b256(input).call(); + + const { value } = await waitForResult(); + expect(value).toBe(expected); + }); + + it('should fail to encode/decode [too short]', async () => { + const input = `0x${'a'.repeat(63)}`; + + await expectToThrowFuelError( + () => contract.functions.types_b256(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid b256.') + ); + }); + + it('should fail to encode/decode [too long]', async () => { + const input = `0x${'a'.repeat(65)}`; + + await expectToThrowFuelError( + () => contract.functions.types_b256(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid b256.') + ); + }); + + it('should fail to encode/decode [not a hex]', async () => { + const input = 'not a hex value'; + + await expectToThrowFuelError( + () => contract.functions.types_b256(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid b256.') + ); + }); + }); + + describe('types_b512', () => { + it('should encode/decode just fine', async () => { + const input = `0xbd0c9b8792876713afa8bff383eebf31c43437823ed761cc3600d0016de5110c44ac566bd156b4fc71a4a4cb2655d3dd360c695edb17dc3b64d611e122fea23d`; + const expected = `0xbd0c9b8792876713afa8bff383eebf31c43437823ed761cc3600d0016de5110c44ac566bd156b4fc71a4a4cb2655d3dd360c695edb17dc3b64d611e122fea23d`; + + const { waitForResult } = await contract.functions.types_b512(input).call(); + + const { value } = await waitForResult(); + expect(value).toBe(expected); + }); + + it('should fail to encode/decode [too short]', async () => { + const input = `0x${'a'.repeat(127)}`; + + await expectToThrowFuelError( + () => contract.functions.types_b512(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid struct B512.') + ); + }); + + it('should fail to encode/decode [too long]', async () => { + const input = `0x${'a'.repeat(129)}`; + + await expectToThrowFuelError( + () => contract.functions.types_b512(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid struct B512.') + ); + }); + + it('should fail to encode/decode [not a hex]', async () => { + const input = 'not a hex value'; + + await expectToThrowFuelError( + () => contract.functions.types_b512(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Invalid struct B512.') + ); + }); + }); + + describe('types_bytes', () => { + it('should encode/decode just fine [Uint8Array]', async () => { + const input = Uint8Array.from([1, 2, 3]); + const expected = Uint8Array.from([3, 2, 1]); + const { waitForResult } = await contract.functions.types_bytes(input).call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + + it('should encode/decode just fine [number]', async () => { + const input = [1, 2, 3]; + const expected = Uint8Array.from([3, 2, 1]); + const { waitForResult } = await contract.functions.types_bytes(input).call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + + /** + * Strings + */ + describe('types_str', () => { + it('should encode/decode just fine [length = 5]', async () => { + const input = 'Input'; + const expected = 'Hello'; + + const { waitForResult } = await contract.functions.types_str(input).call(); + + const { value } = await waitForResult(); + expect(value).toBe(expected); + }); + + it('should fail to encode/decode [length - 1]', async () => { + const input = 'a'.repeat(4); + + await expectToThrowFuelError( + () => contract.functions.types_str(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Value length mismatch during encode.') + ); + }); + + it('should fail to encode/decode [length + 1]', async () => { + const input = 'a'.repeat(6); + + await expectToThrowFuelError( + () => contract.functions.types_str(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Value length mismatch during encode.') + ); + }); + }); + describe('types_str_slice', () => { + it('should encode/decode just fine', async () => { + const input = 'Input'; + const expected = 'Output'; + + const { waitForResult } = await contract.functions.types_str_slice(input).call(); + + const { value } = await waitForResult(); + expect(value).toBe(expected); + }); + }); + describe('types_std_string', () => { + it('should encode/decode just fine', async () => { + const input = 'Input'; + const expected = 'Output'; + + const { waitForResult } = await contract.functions.types_std_string(input).call(); + + const { value } = await waitForResult(); + expect(value).toBe(expected); + }); + }); + describe('types_raw_slice', () => { + it('should encode/decode just fine', async () => { + const input: RawSlice = [1, 2, 3]; + const expected: RawSlice = [4, 3, 2, 1]; + + const { waitForResult } = await contract.functions.types_raw_slice(input).call(); + const { value } = await waitForResult(); + + expect(value).toStrictEqual(expected); + }); + }); + + /** + * Arrays + */ + describe('types_array', () => { + it('should encode/decode just fine', async () => { + const input = [1, 2, 3, 4] as [number, number, number, number]; + const expected = [4, 3, 2, 1] as [number, number, number, number]; + + const { waitForResult } = await contract.functions.types_array(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + + it('should fail to encode/decode [empty]', async () => { + const input = [] as unknown as [number, number, number, number]; + + await expectToThrowFuelError( + () => contract.functions.types_array(input).call(), + new FuelError(FuelError.CODES.ENCODE_ERROR, 'Types/values length mismatch.') + ); + }); + }); + + describe('types_array_struct', () => { + it('should encode/decode just fine', async () => { + const input = [ + { a: true, b: 10 }, + { a: true, b: 10 }, + { a: true, b: 10 }, + ] as [{ a: boolean; b: number }, { a: boolean; b: number }, { a: boolean; b: number }]; // @TODO removed once typegen remastered + const expected = [ + { a: false, b: 30 }, + { a: false, b: 30 }, + { a: false, b: 30 }, + ]; + + const { waitForResult } = await contract.functions.types_array_struct(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + + describe('types_array_with_generic_struct', () => { + it('should encode/decode just fine', async () => { + const INPUT_STRUCT = { + a: { + a: 10, + }, + b: 'A', + }; + const input = [INPUT_STRUCT, INPUT_STRUCT]; + + const EXPECTED_STRUCT = { + a: { + // @ts-expect-error: Custom matcher 'toEqualBn' + a: expect.toEqualBn(20), + }, + b: 'B', + }; + const expected = [EXPECTED_STRUCT, EXPECTED_STRUCT]; + + const { waitForResult } = await contract.functions + // @ts-expect-error - @TODO remove once typegen remastered + .types_array_with_generic_struct(input) + .call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + + describe('types_array_with_vector', () => { + it('should encode/decode just fine', async () => { + const input = [[1, 2, 3]]; + const expected = [[3, 2, 1]]; + + // @ts-expect-error - @TODO remove once typegen remastered + const { waitForResult } = await contract.functions.types_array_with_vector(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + + /** + * Tuples + */ + describe('types_tuple', () => { + it('should encode/decode just fine', async () => { + const input = [1, 2, 3] as [number, number, number]; + const expected = [3, 2, 1] as [number, number, number]; + + const { waitForResult } = await contract.functions.types_tuple(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_tuple_complex', () => { + it('should encode/decode just fine', async () => { + const input = [1, { a: { a: 10 } }, 'ABC']; + // @ts-expect-error: Custom matcher 'toEqualBn' + // @todo resolve this issue. + const expected = [3, { a: { a: expect.toEqualBn(30) } }, 'CBA']; + + // @ts-expect-error - @TODO remove once typegen remastered + const { waitForResult } = await contract.functions.types_tuple_complex(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_tuple_with_native_types', () => { + it('should encode/decode just fine', async () => { + const A: AssetId = { + bits: '0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa', + }; + const B: AssetId = { + bits: '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb', + }; + const input = [A, B, true]; + const expected = [B, A, false]; + + const { waitForResult } = await contract.functions + // @ts-expect-error - @TODO remove once typegen remastered + .types_tuple_with_native_types(input) + .call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_alias_tuple_with_native_types', () => { + it('should encode/decode just fine', async () => { + const A: AssetId = { + bits: '0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa', + }; + const B: AssetId = { + bits: '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb', + }; + const input = [A, B, true]; + const expected = [B, A, false]; + + const { waitForResult } = await contract.functions + // @ts-expect-error - @TODO remove once typegen remastered + .types_alias_tuple_with_native_types(input) + .call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + + /** + * Structs + */ + describe('types_struct_simple', () => { + it('should encode/decode just fine', async () => { + const input = { a: true, b: 10 }; + const expected = { a: false, b: 30 }; + + const { waitForResult } = await contract.functions.types_struct_simple(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_struct_generic', () => { + it('should encode/decode just fine', async () => { + const input = { a: 10 }; + const expected = { a: 20 }; + + const { waitForResult } = await contract.functions.types_struct_generic(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_struct_with_tuple', () => { + it('should encode/decode just fine', async () => { + const input: StructSingleGenericInput<[boolean, BigNumberish]> = { a: [true, 10] }; + // @ts-expect-error: Custom matcher 'toEqualBn' + const expected = { a: [false, expect.toEqualBn(20)] }; + + const { waitForResult } = await contract.functions.types_struct_with_tuple(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_struct_double_generic', () => { + it('should encode/decode just fine', async () => { + const input = { a: 10, b: { a: 10 } }; + const expected = { a: 20, b: { b: 10 } }; + + const { waitForResult } = await contract.functions.types_struct_double_generic(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('type_struct_external', () => { + it('should encode/decode just fine', async () => { + const input = { value: 10 }; + // @ts-expect-error: Custom matcher 'toEqualBn' + const expected = { value: expect.toEqualBn(20) }; + + const { waitForResult } = await contract.functions.type_struct_external(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_struct_with_nested_array', () => { + it('should encode/decode just fine', async () => { + const INPUT_STRUCT = { a: { a: 10 }, b: 'A' }; + const input: StructWithNestedArrayInput = { a: [INPUT_STRUCT, INPUT_STRUCT] }; + // @ts-expect-error: Custom matcher 'toEqualBn' + // @todo resolve this issue. + const EXPECTED_STRUCT = { a: { a: expect.toEqualBn(20) }, b: 'B' }; + const EXPECTED = { a: [EXPECTED_STRUCT, EXPECTED_STRUCT] }; + + const { waitForResult } = await contract.functions + .types_struct_with_nested_array(input) + .call(); + + const { value } = await waitForResult(); + expect(value).toEqual(EXPECTED); + }); + }); + describe('types_struct_with_nested_tuple', () => { + it('should encode/decode just fine', async () => { + const input: StructWithNestedTupleInput = { a: [10, { a: { a: 20 } }, 'ABC'] }; + // @ts-expect-error: Custom matcher 'toEqualBn' + // @todo resolve this issue. + const expected = { a: [30, { a: { a: expect.toEqualBn(40) } }, 'CBA'] }; + + const { waitForResult } = await contract.functions + .types_struct_with_nested_tuple(input) + .call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_struct_with_nested_struct', () => { + it('should encode/decode just fine', async () => { + const input = { a: { a: { a: 10 }, b: 20 } }; + const expected = { a: { a: { a: 30 }, b: 40 } }; + + const { waitForResult } = await contract.functions + .types_struct_with_nested_struct(input) + .call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe.todo('types_struct_with_multiple_struct_params', () => { + it('should encode/decode just fine', async () => { + const STRUCT_A = { propA1: 10 }; + const STRUCT_B = { propB1: STRUCT_A, propB2: 20 }; + + const INPUT_X = STRUCT_A; + const INPUT_Y = STRUCT_B; + const INPUT_Z: StructCInput = { + propC1: STRUCT_A, + propC2: [STRUCT_B], + propC3: { + propD1: [{ propE1: STRUCT_A, propE2: STRUCT_B, propE3: 30 }], + propD2: 40, + propD3: { propF1: 50, propF2: 'A' }, + }, + }; + + const { waitForResult } = await contract.functions + .types_struct_with_multiple_struct_params(INPUT_X, INPUT_Y, INPUT_Z) + .call(); + + const { value } = await waitForResult(); + // expect(value).toEqual(expected); + }); + }); + describe.todo('types_struct_with_implicit_generics', () => {}); + + // @todo Investigate: returning the input as the output + describe.skip('types_struct_with_array', () => { + it('should encode/decode just fine', async () => { + // Inputs + const inputB256: string = + '0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'; + const inputStruct: StructDoubleGenericInput = { + a: inputB256, + b: 10, + }; + const input: StructWithGenericArrayInput = { + a: [inputStruct, inputStruct, inputStruct], + }; + + // Expected + const expectedB256: string = + '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb'; + const expectedStruct: StructDoubleGenericInput = { + a: expectedB256, + b: 20, + }; + const expected: StructWithGenericArrayInput = { + a: [expectedStruct, expectedStruct, expectedStruct], + }; + + const { waitForResult } = await contract.functions.types_struct_with_array(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe.todo('types_struct_with_vector'); + describe.todo('types_struct_with_array_of_enums'); + describe.todo('types_struct_with_complex_nested_struct'); + describe.todo('types_struct_with_single_option'); + + /** + * Enums + */ + describe('types_enum', () => { + it('should encode/decode just fine', async () => { + const input = EnumWithNativeInput.Checked; + const expected = EnumWithNativeInput.Pending; + + const { waitForResult } = await contract.functions.types_enum(input).call(); + + const { value } = await waitForResult(); + expect(value).toBe(expected); + }); + }); + describe('types_enum_with_builtin_type', () => { + it('should encode/decode just fine', async () => { + const input: EnumWithBuiltinTypeInput = { a: true }; + // @ts-expect-error: Custom matcher 'toEqualBn' + // @todo resolve this issue. + const expected: EnumWithBuiltinTypeOutput = { b: expect.toEqualBn(20) }; + + const { waitForResult } = await contract.functions.types_enum_with_builtin_type(input).call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + describe('types_enum_with_vector', () => { + it('should encode/decode just fine', async () => { + const input: EnumWithVectorInput = { a: 10 }; + const expected: EnumWithVectorOutput = { b: [1, 2, 3] }; + + const { waitForResult } = await contract.functions.types_enum_with_vector(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_generic_enum', () => { + it('should encode/decode just fine', async () => { + const input = { a: 10 }; + const expected = { b: 20 }; + + const { waitForResult } = await contract.functions.types_generic_enum(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_enum_external', () => { + // @TODO revist this one, can't return B from this Sway function. + it('should encode/decode just fine', async () => { + const input = ExternalEnumInput.A; + const expected = ExternalEnumInput.A; // Should be B + + const { waitForResult } = await contract.functions.types_enum_external(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_enum_with_structs', () => { + it('should encode/decode just fine', async () => { + const input = { a: EnumWithNativeInput.Checked }; + const expected = { b: { a: true, b: 10 } }; + + const { waitForResult } = await contract.functions.types_enum_with_structs(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + + /** + * Vectors + */ + describe('types_vector_u8', () => { + it('should encode/decode just fine', async () => { + const input = [1, 2, 3]; + const expected = [3, 2, 1]; + + const { waitForResult } = await contract.functions.types_vector_u8(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_vector_boolean', () => { + it('should encode/decode just fine', async () => { + const input = [true, false, true, false]; + const expected = [false, true, false, true]; + + const { waitForResult } = await contract.functions.types_vector_boolean(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_vector_inside_vector', () => { + it('should encode/decode just fine', async () => { + const input = [[1, 2, 3]]; + const expected = [ + [3, 2, 1], + [6, 5, 4], + ]; + + const { waitForResult } = await contract.functions.types_vector_inside_vector(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_vector_with_struct', () => { + it('should encode/decode just fine', async () => { + const input = [{ a: true, b: 10 }]; + const expected = [{ a: false, b: 30 }]; + + const { waitForResult } = await contract.functions.types_vector_with_struct(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('types_vector_option', () => { + it('should encode/decode just fine', async () => { + const input: Vec = [{ a: [1, 2, 3, 4, 5] }]; + const expected: Vec = [{ a: [5, 4, 3, 2, 1] }]; + + const { waitForResult } = await contract.functions.types_vector_option(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + + /** + * Options + */ + // @todo Investigate: returning the input as the output + describe.skip('types_option', () => { + it('should encode/decode just fine', async () => { + const input: Option = 10; // Some + const expected: Option = undefined; // None + + const { waitForResult } = await contract.functions.types_option(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + // @todo Investigate: returning the input as the output + describe.skip('types_option_geo', () => { + it('should encode/decode just fine', async () => { + const input: Option = { + a: true, + b: 10, + }; + const expected: Option = undefined; + + const { waitForResult } = await contract.functions.types_option_geo(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + + /** + * Native types + */ + describe('types_asset_id', () => { + it('should encode/decode just fine', async () => { + const input: AssetId = { + bits: '0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa', + }; + const expected: AssetId = { + bits: '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb', + }; + + const { waitForResult } = await contract.functions.types_asset_id(input).call(); + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + describe('type_identity_address', () => { + it('should encode/decode just fine', async () => { + const input: IdentityInput = { + Address: { bits: '0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' }, + }; + const expected: IdentityOutput = { + Address: { bits: '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb' }, + }; + + const { waitForResult } = await contract.functions.type_identity_address(input).call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + describe('type_identity_contract_id', () => { + it('should encode/decode just fine', async () => { + const input: IdentityInput = { + ContractId: { bits: '0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' }, + }; + const expected: IdentityOutput = { + ContractId: { bits: '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb' }, + }; + + const { waitForResult } = await contract.functions.type_identity_contract_id(input).call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + describe('type_address', () => { + it('should encode/decode just fine', async () => { + const input = { bits: '0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' }; + const expected = { + bits: '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb', + }; + + const { waitForResult } = await contract.functions.type_address(input).call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + describe('type_contract_id', () => { + it('should encode/decode just fine', async () => { + const input = { bits: '0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' }; + const expected = { + bits: '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb', + }; + + const { waitForResult } = await contract.functions.type_contract_id(input).call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + describe('types_evm_address', () => { + it('should encode/decode just fine', async () => { + const input: EvmAddress = { + bits: '0x000000000000000000000000AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA', + }; + const expected = { + bits: '0x000000000000000000000000bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb', + }; + + const { waitForResult } = await contract.functions.types_evm_address(input).call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + describe('types_result', () => { + it('should accept result just fine [Ok - 10]', async () => { + const input: Result = { + Ok: 10, + }; + const expected: Result = { + // @ts-expect-error: Custom matcher 'toEqualBn' + // @todo resolve this issue. + Ok: expect.toEqualBn(2), + }; + + const { waitForResult } = await contract.functions.types_result(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + + it('should accept result just fine [Err - divide by zero]', async () => { + const input: Result = { + Ok: 0, + }; + const expected: Result = { + Err: 'DivisError', + }; + + const { waitForResult } = await contract.functions.types_result(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + + it('should accept result just fine [Err - 10]', async () => { + const input: Result = { + Err: 10, + }; + const expected: Result = { + Err: 'InputError', + }; + + const { waitForResult } = await contract.functions.types_result(input).call(); + + const { value } = await waitForResult(); + expect(value).toEqual(expected); + }); + }); + + /** + * Void + */ + describe('types_void', () => { + it('should encode/decode just fine', async () => { + const input = undefined; + const expected = undefined; + + const { waitForResult } = await contract.functions.types_void(input).call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + + it('should encode/decode just fine [omit optional args]', async () => { + const expected = undefined; + + const { waitForResult } = await contract.functions.types_void().call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + describe('types_void_then_value', () => { + it('should encode/decode just fine', async () => { + const inputX = undefined; + const inputY = 10; + const expected = undefined; + + const { waitForResult } = await contract.functions + .types_void_then_value(inputX, inputY) + .call(); + + const { value } = await waitForResult(); + expect(value).toBe(expected); + }); + }); + describe('types_value_then_void', () => { + it('should encode/decode just fine', async () => { + const inputX = 10; + const inputY = undefined; + const { waitForResult } = await contract.functions + .types_value_then_void(inputX, inputY) + .call(); + + const { value } = await waitForResult(); + expect(value).toBeUndefined(); + }); + + it('should encode/decode just fine [omitting optional args]', async () => { + const inputX = 10; + + const { waitForResult } = await contract.functions.types_value_then_void(inputX).call(); + + const { value } = await waitForResult(); + expect(value).toBeUndefined(); + }); + }); + describe('types_value_then_void_then_value', () => { + it('should encode/decode just fine', async () => { + const inputX = 10; + const inputY = undefined; + const inputZ = 20; + + const { waitForResult } = await contract.functions + .types_value_then_void_then_value(inputX, inputY, inputZ) + .call(); + + const { value } = await waitForResult(); + expect(value).toBeUndefined(); + }); + }); + describe('types_value_then_value_then_void_then_void', () => { + it('should encode/decode just fine', async () => { + const inputX = 10; + const inputY = 20; + const inputZ = undefined; + const inputA = undefined; + + const { waitForResult } = await contract.functions + .types_value_then_value_then_void_then_void(inputX, inputY, inputZ, inputA) + .call(); + + const { value } = await waitForResult(); + expect(value).toBeUndefined(); + }); + + it('should encode/decode just fine [omitting optional args]', async () => { + const inputX = 10; + const inputY = 20; + + const { waitForResult } = await contract.functions + .types_value_then_value_then_void_then_void(inputX, inputY) + .call(); + + const { value } = await waitForResult(); + expect(value).toBeUndefined(); + }); + }); + + /** + * Multi-arg + * + * @todo resolve the below issue. + * Most of these are suffering from the similar issue around returning the input as the output. + */ + describe('multi_arg_u64_u64', () => { + it('should encode/decode just fine', async () => { + const inputX = 1; + const inputY = 2; + // @ts-expect-error: Custom matcher 'toEqualBn's + const expected = expect.toEqualBn(3); + + const { waitForResult } = await contract.functions.multi_arg_u64_u64(inputX, inputY).call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + // @todo Investigate: returning the input as the output + describe.skip('multi_arg_b256_bool', () => { + // @todo investigate, this is returning the input as the output. + it('should encode/decode just fine', async () => { + const inputX = '0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'; + const inputY = true; + const expected = [ + '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb', + false, + ]; + + const { waitForResult } = await contract.functions.multi_arg_b256_bool(inputX, inputY).call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + // @todo Investigate: returning the input as the output + describe.skip('multi_arg_vector_vector', () => { + it('should encode/decode just fine', async () => { + const inputX = [1, 2, 3]; + const inputY = [4, 5, 6]; + const expected = [ + [7, 8, 9], + [10, 11, 12], + ]; + + const { waitForResult } = await contract.functions + .multi_arg_vector_vector(inputX, inputY) + .call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + // @todo Investigate: returning the input as the output + describe.skip('multi_arg_vector_b256', () => { + it('should encode/decode just fine', async () => { + const inputX = [1, 2, 3]; + const inputY = '0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'; + const expected = [ + [7, 8, 9], + '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb', + ]; + + const { waitForResult } = await contract.functions + .multi_arg_vector_b256(inputX, inputY) + .call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + // @todo Investigate: returning the input as the output + describe.skip('multi_arg_struct_vector', () => { + it('should encode/decode just fine', async () => { + const inputX = { a: true, b: 1 }; + const inputY = [1, 2, 3]; + const expected = [{ a: false, b: 2 }, [4, 5, 6]]; + + const { waitForResult } = await contract.functions + .multi_arg_struct_vector(inputX, inputY) + .call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); + describe.skip('multi_arg_u64_struct'); + describe.skip('multi_arg_str_str'); + describe.skip('multi_arg_u32_vector_vector'); + describe.skip('multi_arg_complex'); +}); diff --git a/packages/fuel-gauge/src/abi/constants.ts b/packages/fuel-gauge/src/abi/constants.ts new file mode 100644 index 00000000000..a81232d91c1 --- /dev/null +++ b/packages/fuel-gauge/src/abi/constants.ts @@ -0,0 +1,63 @@ +import { bn } from 'fuels'; + +export const U8_MIN = 0; +export const U8_MAX = 2 ** 8 - 1; +export const U8_MAX_ENCODED = new Uint8Array([255]); +export const U8_MIN_ENCODED = new Uint8Array([0]); +export const U16_MIN = 0; +export const U16_MAX = 2 ** 16 - 1; +export const U16_MAX_ENCODED = new Uint8Array([255, 255]); +export const U16_MIN_ENCODED = new Uint8Array([0, 0]); +export const U32_MIN = 0; +export const U32_MAX = 2 ** 32 - 1; +export const U32_MAX_ENCODED = new Uint8Array([255, 255, 255, 255]); +export const U32_MIN_ENCODED = new Uint8Array([0, 0, 0, 0]); +export const U64_MIN = 0; +export const U64_MAX = bn(2).pow(64).sub(1); +export const U64_MAX_ENCODED = new Uint8Array([255, 255, 255, 255, 255, 255, 255, 255]); +export const U64_MIN_ENCODED = new Uint8Array([0, 0, 0, 0, 0, 0, 0, 0]); +export const U256_MIN = 0; +export const U256_MAX = bn(2).pow(256).sub(1); +export const U256_MAX_ENCODED = new Uint8Array(32).fill(255); +export const U256_MIN_ENCODED = new Uint8Array(32).fill(0); + +export const EMPTY_8_BYTE_ARRAY = new Uint8Array([0, 0, 0, 0, 0, 0, 0, 0]); +export const ENUM_FIRST_INDEX = EMPTY_8_BYTE_ARRAY; +export const ENUM_SECOND_INDEX = new Uint8Array([0, 0, 0, 0, 0, 0, 0, 1]); +export const ENUM_THIRD_INDEX = new Uint8Array([0, 0, 0, 0, 0, 0, 0, 2]); + +export const STRING_MIN_DECODED = ''; +export const STRING_MIN_ENCODED = new Uint8Array(); +export const STRING_MAX_DECODED = 'a'.repeat(U8_MAX); +export const STRING_MAX_ENCODED = new Uint8Array([ + ...Array.from(Array(U8_MAX + 1).fill(97, 0, U8_MAX)), +]); + +export const B256_DECODED = '0xd5579c46dfcc7f18207013e65b44e4cb4e2c2298f4ac457ba8f82743f31e930b'; +export const B256_ENCODED = new Uint8Array([ + 213, 87, 156, 70, 223, 204, 127, 24, 32, 112, 19, 230, 91, 68, 228, 203, 78, 44, 34, 152, 244, + 172, 69, 123, 168, 248, 39, 67, 243, 30, 147, 11, +]); +export const B256_ZERO_DECODED = + '0x0000000000000000000000000000000000000000000000000000000000000000'; +export const B256_ZERO_ENCODED = new Uint8Array(32); + +export const BYTE_MIN_DECODED = 0; +export const BYTE_MIN_ENCODED = new Uint8Array([0, 0, 0, 0, 0, 0, 0, 0]); +export const BYTE_MAX_DECODED = U8_MAX; +export const BYTE_MAX_ENCODED = new Uint8Array([0, 0, 0, 0, 0, 0, 0, 255]); + +export const BOOL_TRUE_ENCODED = new Uint8Array([1]); +export const BOOL_FALSE_ENCODED = new Uint8Array([0]); + +export const B512_DECODED = + '0x8e9dda6f7793745ac5aacf9e907cae30b2a01fdf0d23b7750a85c6a44fca0c29f0906f9d1f1e92e6a1fb3c3dcef3cc3b3cdbaae27e47b9d9a4c6a4fce4cf16b2'; +export const B512_ENCODED = new Uint8Array([ + 142, 157, 218, 111, 119, 147, 116, 90, 197, 170, 207, 158, 144, 124, 174, 48, 178, 160, 31, 223, + 13, 35, 183, 117, 10, 133, 198, 164, 79, 202, 12, 41, 240, 144, 111, 157, 31, 30, 146, 230, 161, + 251, 60, 61, 206, 243, 204, 59, 60, 219, 170, 226, 126, 71, 185, 217, 164, 198, 164, 252, 228, + 207, 22, 178, +]); +export const B512_ZERO_DECODED = + '0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000'; +export const B512_ZERO_ENCODED = new Uint8Array(64); diff --git a/packages/fuel-gauge/src/abi/vitest.matcher.ts b/packages/fuel-gauge/src/abi/vitest.matcher.ts new file mode 100644 index 00000000000..32a1cd3b4a8 --- /dev/null +++ b/packages/fuel-gauge/src/abi/vitest.matcher.ts @@ -0,0 +1,20 @@ +import { bn } from 'fuels'; +import type { BNInput } from 'fuels'; + +export const toEqualBn = (_received: BNInput, _argument: BNInput) => { + const received = bn(_received); + const argument = bn(_argument); + + const pass = received.eq(argument); + + if (pass) { + return { + message: () => `Expected ${received.toString()} not to equal ${argument.toString()}`, + pass: true, + }; + } + return { + message: () => `expected ${received.toString()} to equal ${argument.toString()}`, + pass: false, + }; +}; diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/Forc.toml b/packages/fuel-gauge/test/fixtures/forc-projects/Forc.toml index a7dc648d366..144df4c78ab 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/Forc.toml +++ b/packages/fuel-gauge/test/fixtures/forc-projects/Forc.toml @@ -1,5 +1,6 @@ [workspace] members = [ + "abi-contract", "advanced-logging", "advanced-logging-abi", "advanced-logging-other-contract", diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/Forc.toml b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/Forc.toml new file mode 100644 index 00000000000..7e941f84b42 --- /dev/null +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/Forc.toml @@ -0,0 +1,7 @@ +[project] +authors = ["Fuel Labs "] +license = "Apache-2.0" +name = "abi-contract" + +[dependencies] +abi-library = { path = "../abi-library" } diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/data_structures.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/data_structures.sw new file mode 100644 index 00000000000..d3834b64754 --- /dev/null +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/data_structures.sw @@ -0,0 +1,137 @@ +library; + +pub struct Configurables { + pub U8_VALUE: u8, + pub BOOL_VALUE: bool, + pub B256_VALUE: b256, + pub OPTION_U8_VALUE: Option, + pub GENERIC_STRUCT_VALUE: StructDoubleGeneric, u32>, +} + +pub enum EnumWithNative { + pub Checked: (), + pub Pending: (), +} + +pub enum EnumWithVector { + pub a: u8, + pub b: Vec, +} + +pub enum EnumWithBuiltinType { + pub a: bool, + pub b: u64, +} + +pub enum EnumDoubleGeneric { + pub a: T1, + pub b: T2, +} + +pub enum EnumWithStructs { + pub a: EnumWithNative, + pub b: StructSimple, + pub c: StructDoubleGeneric, +} + +pub struct StructSimple { + pub a: bool, + pub b: u32, +} + +pub struct StructWithEnumArray { + pub a: [EnumWithNative; 3], +} + +pub struct StructWithMultiOption { + pub a: [Option; 5], +} + +pub struct StructWithSingleOption { + pub b: Option, +} + +pub struct StructWithVector { + pub a: u8, + pub b: Vec, +} + +pub struct StructSingleGeneric { + pub a: T, +} + +pub struct StructDoubleGeneric { + pub a: T1, + pub b: T2, +} + +pub struct StructGenericWithEnum { + pub a: T1, + pub b: EnumDoubleGeneric, +} + +pub struct StructWithImplicitGenerics { + pub a: [E; 3], + pub b: (E, F), +} + +pub struct StructWithGenericArray { + pub a: [StructDoubleGeneric; 3], +} + +pub struct StructWithNestedArray { + pub a: [StructDoubleGeneric, str[1]>; 2], +} + +pub struct StructWithNestedTuple { + pub a: (u8, StructSingleGeneric>, str[3]), +} + +pub struct StructWithNestedStruct { + pub a: StructDoubleGeneric, u16>, +} + +pub struct StructA { + pub propA1: u8, +} + +pub struct StructB { + pub propB1: StructA, + pub propB2: u16, +} + +pub struct StructC { + pub propC1: StructA, + pub propC2: Vec, + pub propC3: StructD>, + // propC4: Vec>>, + // propC5: Vec>>>, +} + +pub struct StructD { + pub propD1: Vec>, + pub propD2: U, + pub propD3: V, +} + +pub struct StructE { + pub propE1: StructA, + pub propE2: StructB, + pub propE3: T, +} + +pub struct StructF { + pub propF1: u64, + pub propF2: T, +} + +pub struct StructG { + pub propG1: u8, +} + +pub enum MyContractError { + pub DivisionByZero: (), +} + +pub type TupleWithNativeAssets = (AssetId, AssetId, bool); + diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw new file mode 100644 index 00000000000..4190341b3b6 --- /dev/null +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw @@ -0,0 +1,375 @@ +library; + +use ::data_structures::*; +use core::ops::Eq; + +impl Eq for [u8; 4] { + fn eq(self, other: Self) -> bool { + self[0] == other[0] && + self[1] == other[1] && + self[2] == other[2] && + self[3] == other[3] + } +} + +impl Eq for StructSimple { + fn eq(self, other: Self) -> bool { + self.a == other.a && self.b == other.b + } +} + +impl Eq for [StructSimple; 3] { + fn eq(self, other: Self) -> bool { + self[0] == other[0] && self[1] == other[1] && self[2] == other[2] + } +} + +impl Eq for StructSingleGeneric { + fn eq(self, other: Self) -> bool { + self.a == other.a + } +} + +impl Eq for str[1] { + fn eq(self, other: Self) -> bool { + // @TODO work out how to equal str[1] + // self[0] == other[0] + true + } +} + +impl Eq for str[3] { + fn eq(self, other: Self) -> bool { + // @TODO work out how to equal str[3] + // self[0] == other[0] && self[1] == other[1] && self[2] == other[2] + true + } +} + +impl Eq for StructDoubleGeneric, str[1]> { + fn eq(self, other: Self) -> bool { + self.a == other.a && self.b == other.b + } +} + +impl Eq for [StructDoubleGeneric, str[1]>; 2] { + fn eq(self, other: Self) -> bool { + self[0] == other[0] && self[1] == other[1] + } +} + +impl Eq for Vec { + fn eq(self, other: Self) -> bool { + if self.len() != other.len() { + return false; + } + let mut i = 0; + while i < self.len() { + if self.get(i).unwrap() != other.get(i).unwrap() { + return false; + } + i += 1; + } + true + } +} + +impl Eq for [Vec; 1] { + fn eq(self, other: Self) -> bool { + self[0] == other[0] + } +} + +impl Eq for (u8, u8, u8) { + fn eq(self, other: Self) -> bool { + self.0 == other.0 && self.1 == other.1 && self.2 == other.2 + } +} + +impl Eq for StructSingleGeneric> { + fn eq(self, other: Self) -> bool { + self.a == other.a + } +} + +impl Eq for (u8, StructSingleGeneric>, str[3]) { + fn eq(self, other: Self) -> bool { + self.0 == other.0 && self.1 == other.1 && self.2 == other.2 + } +} + +impl Eq for (AssetId, AssetId, bool) { + fn eq(self, other: Self) -> bool { + self.0 == other.0 && self.1 == other.1 && self.2 == other.2 + } +} + +impl Eq for StructSingleGeneric { + fn eq(self, other: Self) -> bool { + self.a == other.a + } +} + +impl Eq for (bool, u64) { + fn eq(self, other: Self) -> bool { + self.0 == other.0 && self.1 == other.1 + } +} + +impl Eq for StructSingleGeneric<(bool, u64)> { + fn eq(self, other: Self) -> bool { + self.a == other.a + } +} + +impl Eq for StructWithNestedArray { + fn eq(self, other: Self) -> bool { + self.a == other.a + } +} + +impl Eq for EnumDoubleGeneric { + fn eq(self, other: Self) -> bool { + match (self, other) { + (EnumDoubleGeneric::a(a), EnumDoubleGeneric::a(b)) => a == b, + (EnumDoubleGeneric::b(a), EnumDoubleGeneric::b(b)) => a == b, + _ => false, + } + } +} + +impl Eq for StructGenericWithEnum { + fn eq(self, other: Self) -> bool { + self.a == other.a && self.b == other.b + } +} + +impl Eq for StructWithNestedTuple { + fn eq(self, other: Self) -> bool { + self.a == other.a + } +} + +impl Eq for StructDoubleGeneric, u16> { + fn eq(self, other: Self) -> bool { + self.a == other.a && self.b == other.b + } +} + +impl Eq for StructWithNestedStruct { + fn eq(self, other: Self) -> bool { + self.a == other.a + } +} + +impl Eq for StructA { + fn eq(self, other: Self) -> bool { + self.propA1 == other.propA1 + } +} + +impl Eq for StructB { + fn eq(self, other: Self) -> bool { + self.propB1 == other.propB1 && self.propB2 == other.propB2 + } +} + +impl Eq for StructE { + fn eq(self, other: Self) -> bool { + self.propE1 == other.propE1 && self.propE2 == other.propE2 && self.propE3 == other.propE3 + } +} + +impl Eq for Vec> { + fn eq(self, other: Self) -> bool { + if self.len() != other.len() { + return false; + } + let mut i = 0; + while i < self.len() { + if self.get(i).unwrap() != other.get(i).unwrap() { + return false; + } + i += 1; + } + true + } +} + +impl Eq for StructF { + fn eq(self, other: Self) -> bool { + self.propF1 == other.propF1 && self.propF2 == other.propF2 + } +} + +impl Eq for StructD> { + fn eq(self, other: Self) -> bool { + self.propD1 == other.propD1 && self.propD2 == other.propD2 && self.propD3 == other.propD3 + } +} + +impl Eq for Vec { + fn eq(self, other: Self) -> bool { + if self.len() != other.len() { + return false; + } + let mut i = 0; + while i < self.len() { + if self.get(i).unwrap() != other.get(i).unwrap() { + return false; + } + i += 1; + } + true + } +} + +impl Eq for StructC { + fn eq(self, other: Self) -> bool { + self.propC1 == other.propC1 && self.propC2 == other.propC2 && self.propC3 == other.propC3 + } +} + +impl Eq for EnumWithNative { + fn eq(self, other: Self) -> bool { + match (self, other) { + (EnumWithNative::Checked, EnumWithNative::Checked) => true, + (EnumWithNative::Pending, EnumWithNative::Pending) => true, + _ => false, + } + } +} + +impl Eq for EnumWithBuiltinType { + fn eq(self, other: Self) -> bool { + match (self, other) { + (EnumWithBuiltinType::a(a), EnumWithBuiltinType::a(b)) => a == b, + (EnumWithBuiltinType::b(a), EnumWithBuiltinType::b(b)) => a == b, + _ => false, + } + } +} + +impl Eq for Vec { + fn eq(self, other: Self) -> bool { + if self.len() != other.len() { + return false; + } + let mut i = 0; + while i < self.len() { + if self.get(i).unwrap() != other.get(i).unwrap() { + return false; + } + i += 1; + } + true + } +} + +impl Eq for EnumWithVector { + fn eq(self, other: Self) -> bool { + match (self, other) { + (EnumWithVector::a(a), EnumWithVector::a(b)) => a == b, + (EnumWithVector::b(a), EnumWithVector::b(b)) => a == b, + _ => false, + } + } +} + +impl Eq for StructDoubleGeneric { + fn eq(self, other: Self) -> bool { + self.a == other.a && self.b == other.b + } +} + +impl Eq for EnumWithStructs { + fn eq(self, other: Self) -> bool { + match (self, other) { + (EnumWithStructs::a(a), EnumWithStructs::a(b)) => a == b, + (EnumWithStructs::b(a), EnumWithStructs::b(b)) => a == b, + (EnumWithStructs::c(a), EnumWithStructs::c(b)) => a == b, + _ => false, + } + } +} + +impl Eq for Vec { + fn eq(self, other: Self) -> bool { + if self.len() != other.len() { + return false; + } + let mut i = 0; + while i < self.len() { + if self.get(i).unwrap() != other.get(i).unwrap() { + return false; + } + i += 1; + } + true + } +} + +impl Eq for Vec> { + fn eq(self, other: Self) -> bool { + if self.len() != other.len() { + return false; + } + let mut i = 0; + while i < self.len() { + if self.get(i).unwrap() != other.get(i).unwrap() { + return false; + } + i += 1; + } + true + } +} + +impl Eq for Vec { + fn eq(self, other: Self) -> bool { + if self.len() != other.len() { + return false; + } + let mut i = 0; + while i < self.len() { + if self.get(i).unwrap() != other.get(i).unwrap() { + return false; + } + i += 1; + } + true + } +} + +impl Eq for [Option; 5] { + fn eq(self, other: Self) -> bool { + self[0] == other[0] && + self[1] == other[1] && + self[2] == other[2] && + self[3] == other[3] && + self[4] == other[4] + } +} + +impl Eq for StructWithMultiOption { + fn eq(self, other: Self) -> bool { + self.a == other.a + } +} + + +impl Eq for Vec { + fn eq(self, other: Self) -> bool { + if self.len() != other.len() { + return false; + } + let mut i = 0; + while i < self.len() { + if self.get(i).unwrap() != other.get(i).unwrap() { + return false; + } + i += 1; + } + true + } +} \ No newline at end of file diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw new file mode 100644 index 00000000000..d8c75629a2c --- /dev/null +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -0,0 +1,871 @@ +contract; + +mod data_structures; +mod equality; +mod utils; + +use data_structures::*; +use equality::*; +use utils::*; + +use abi_library::ExternalStruct; +use abi_library::ExternalEnum; +use std::vm::evm::evm_address::EvmAddress; +use std::b512::B512; +use std::string::String; +use std::bytes::Bytes; + + +fn divide(numerator: u64, denominator: u64) -> Result { + if (denominator == 0) { + return Err(MyContractError::DivisionByZero); + } else { + Ok(numerator / denominator) + } +} + +abi MyContract { + fn configurables() -> Configurables; + + fn types_u8(x: u8) -> u8; + fn types_u16(x: u16) -> u16; + fn types_u32(x: u32) -> u32; + fn types_u64(x: u64) -> u64; + fn types_u256(x: u256) -> u256; + fn types_bool(x: bool) -> bool; + fn types_b256(x: b256) -> b256; + fn types_b512(x: B512) -> B512; + fn types_bytes(x: Bytes) -> Bytes; + + fn types_str(x: str[5]) -> str[5]; + fn types_str_slice(x: str) -> str; + fn types_raw_slice(x: raw_slice) -> raw_slice; + fn types_std_string(x: String) -> String; + + fn types_array(x: [u8; 4]) -> [u8; 4]; + fn types_array_struct(x: [StructSimple; 3]) -> [StructSimple; 3]; + fn types_array_with_generic_struct( + x: [StructDoubleGeneric, str[1]>; 2], + ) -> [StructDoubleGeneric, str[1]>; 2]; + fn types_array_with_vector(x: [Vec; 1]) -> [Vec; 1]; + + fn types_struct_simple(x: StructSimple) -> StructSimple; + fn types_struct_generic(x: StructSingleGeneric) -> StructSingleGeneric; + fn types_struct_with_tuple( + x: StructSingleGeneric<(bool, u64)>, + ) -> StructSingleGeneric<(bool, u64)>; + fn types_struct_double_generic( + x: StructGenericWithEnum, + ) -> StructGenericWithEnum; + fn type_struct_external(x: ExternalStruct) -> ExternalStruct; + fn types_struct_with_implicit_generics( + x: StructWithImplicitGenerics, + ) -> StructWithImplicitGenerics; + fn types_struct_with_array(x: StructWithGenericArray) -> StructWithGenericArray; + fn types_struct_with_vector(x: StructWithVector) -> StructWithVector; + fn types_struct_with_array_of_enums(x: StructWithEnumArray) -> StructWithEnumArray; + fn types_struct_with_nested_array(x: StructWithNestedArray) -> StructWithNestedArray; + fn types_struct_with_nested_tuple(x: StructWithNestedTuple) -> StructWithNestedTuple; + fn types_struct_with_nested_struct(x: StructWithNestedStruct) -> StructWithNestedStruct; + fn types_struct_with_multiple_struct_params(x: StructA, y: StructB, z: StructC) -> bool; + fn types_struct_with_complex_nested_struct(x: StructD>>) -> bool; + fn types_struct_with_single_option(x: StructWithSingleOption) -> StructWithSingleOption; + + fn types_tuple(x: (u8, u8, u8)) -> (u8, u8, u8); + fn types_tuple_complex( + x: (u8, StructSingleGeneric>, str[3]), + ) -> (u8, StructSingleGeneric>, str[3]); + fn types_tuple_with_native_types(x: (AssetId, AssetId, bool)) -> (AssetId, AssetId, bool); + fn types_alias_tuple_with_native_types(x: TupleWithNativeAssets) -> TupleWithNativeAssets; + + fn types_enum(x: EnumWithNative) -> EnumWithNative; + fn types_enum_with_builtin_type(x: EnumWithBuiltinType) -> EnumWithBuiltinType; + fn types_enum_with_vector(x: EnumWithVector) -> EnumWithVector; + fn types_generic_enum(x: EnumDoubleGeneric) -> EnumDoubleGeneric; + fn types_enum_external(x: ExternalEnum) -> ExternalEnum; + fn types_enum_with_structs(x: EnumWithStructs) -> EnumWithStructs; + + fn types_vector_u8(x: Vec) -> Vec; + fn types_vector_boolean(x: Vec) -> Vec; + fn types_vector_inside_vector(x: Vec>) -> Vec>; + fn types_vector_with_struct(x: Vec) -> Vec; + fn types_vector_option(x: Vec) -> Vec; + + fn types_option(x: Option) -> Option; + fn types_option_geo(x: Option) -> Option; + + fn type_identity_address(x: Identity) -> Identity; + fn type_identity_contract_id(x: Identity) -> Identity; + fn type_address(x: Address) -> Address; + fn type_contract_id(x: ContractId) -> ContractId; + fn types_asset_id(x: AssetId) -> AssetId; + fn types_evm_address(x: EvmAddress) -> EvmAddress; + fn types_result(x: Result) -> Result; + + fn types_void(x: ()) -> (); + fn types_void_then_value(x: (), y: u8) -> (); + fn types_value_then_void(x: u8, y: ()) -> (); + fn types_value_then_void_then_value(x: u8, y: (), z: u8) -> (); + fn types_value_then_value_then_void_then_void(x: u8, y: u8, z: (), a: ()) -> (); + + fn multi_arg_u64_u64(x: u64, y: u64) -> u64; + fn multi_arg_b256_bool(x: b256, y: bool) -> (b256, bool); + fn multi_arg_vector_vector(x: Vec, y: Vec) -> (Vec, Vec); + fn multi_arg_vector_b256(x: Vec, y: b256) -> (Vec, b256); + fn multi_arg_struct_vector(x: StructSimple, y: Vec) -> (StructSimple, Vec); + fn multi_arg_u64_struct(x: u64, y: StructSimple) -> (u64, StructSimple); + fn multi_arg_str_str(x: str[5], y: str[5]) -> (str[5], str[5]); + fn multi_arg_u32_vector_vector(x: u32, y: Vec, z: Vec) -> (u32, Vec, Vec); + fn multi_arg_complex( + x: StructDoubleGeneric<[b256; 3], u8>, + y: [StructDoubleGeneric; 4], + z: (str[5], bool), + a: StructSimple, + ); +} + + +configurable { + U8_VALUE: u8 = 10, + BOOL_VALUE: bool = true, + B256_VALUE: b256 = 0x38966262edb5997574be45f94c665aedb41a1663f5b0528e765f355086eebf96, + OPTION_U8_VALUE: Option = Option::None, + GENERIC_STRUCT_VALUE: StructDoubleGeneric, u32> = StructDoubleGeneric { + a: StructDoubleGeneric { a: 4, b: 257 }, + b: 57000, + }, +} + +impl MyContract for Contract { + fn configurables() -> Configurables { + Configurables { + U8_VALUE: U8_VALUE, + BOOL_VALUE: BOOL_VALUE, + B256_VALUE: B256_VALUE, + OPTION_U8_VALUE: OPTION_U8_VALUE, + GENERIC_STRUCT_VALUE: GENERIC_STRUCT_VALUE, + } + } + fn types_u8(x: u8) -> u8 { + assert_eq(x, 8); + 255 + } + fn types_u16(x: u16) -> u16 { + assert_eq(x, 16); + 65535 + } + fn types_u32(x: u32) -> u32 { + assert_eq(x, 32); + 4294967295 + } + fn types_u64(x: u64) -> u64 { + assert_eq(x, 64); + 4294967295000 + } + fn types_u256(x: u256) -> u256 { + assert_eq(x, 256); + 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFu256 + } + + fn types_bool(x: bool) -> bool { + const INPUT: bool = false; + assert_eq(x, INPUT); + + const EXPECTED: bool = true; + return EXPECTED + } + fn types_b256(x: b256) -> b256 { + const INPUT: b256 = 0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa; + assert_eq(x, INPUT); + + const EXPECTED: b256 = 0x0000000000000000000000000000000000000000000000000000000000000000; + return EXPECTED + } + // @TODO + fn types_b512(x: B512) -> B512 { + // HIGH_BIT and **LOW_BIT** + const HI_BITS = 0xbd0c9b8792876713afa8bff383eebf31c43437823ed761cc3600d0016de5110c; + const LO_BITS = 0x44ac566bd156b4fc71a4a4cb2655d3dd360c695edb17dc3b64d611e122fea23d; + const INPUT: B512 = B512::from((HI_BITS, LO_BITS)); + assert_eq(x, INPUT); + + // HIGH_BIT and **LOW_BIT2** + const LO_BITS2 = 0x54ac566bd156b4fc71a4a4cb2655d3dd360c695edb17dc3b64d611e122fea23d; + const EXPECTED: B512 = B512::from((HI_BITS, LO_BITS2)); + return INPUT + } + + fn types_bytes(x: Bytes) -> Bytes { + let mut INPUT = Bytes::new(); + INPUT.push(1u8); + INPUT.push(2u8); + INPUT.push(3u8); + assert_eq(x, INPUT); + + let mut EXPECTED = Bytes::new(); + EXPECTED.push(3u8); + EXPECTED.push(2u8); + EXPECTED.push(1u8); + return EXPECTED + // let EXPECTED = Bytes::from_hex("0xabcdef9012345678"); + + } + + /** + * Strings + */ + fn types_str(x: str[5]) -> str[5] { + // @TODO + // const INPUT: str[5] = __to_str_array("Input"); + // assert_eq(x, INPUT); + + const EXPECTED: str[5] = __to_str_array("Hello"); + return EXPECTED; + } + fn types_str_slice(x: str) -> str { + let INPUT = "Input"; + assert(x == INPUT); + + let EXPECTED = "Output"; + return EXPECTED; + } + fn types_std_string(x: String) -> String { + let INPUT = "Input"; + assert_eq(x, String::from_ascii_str(INPUT)); + + let EXPECTED = "Output"; + return String::from_ascii_str(EXPECTED); + } + fn types_raw_slice(x: raw_slice) -> raw_slice { + let vec: Vec = Vec::from(x); + require(vec.len() == 3, "raw slice len is not 3"); + require( + vec + .get(2) + .unwrap() == 3, + "expected 3rd slice entry to be 3", + ); + require( + vec + .get(1) + .unwrap() == 2, + "expected 2nd slice entry to be 2", + ); + require( + vec + .get(0) + .unwrap() == 1, + "expected 1st slice entry to be 1", + ); + + + let mut vec_expected: Vec = Vec::new(); + vec_expected.push(4); + vec_expected.push(3); + vec_expected.push(2); + vec_expected.push(1); + let EXPECTED = vec_expected.as_raw_slice(); + return EXPECTED + } + + /** + * Arrays + */ + fn types_array(x: [u8; 4]) -> [u8; 4] { + const INPUT: [u8; 4] = [1, 2, 3, 4]; + assert(x == INPUT); + + const EXPECTED: [u8; 4] = [4, 3, 2, 1]; + return EXPECTED + } + fn types_array_struct(x: [StructSimple; 3]) -> [StructSimple; 3] { + const INPUT_STRUCT_1: StructSimple = StructSimple { a: true, b: 10 }; + const INPUT = [INPUT_STRUCT_1, INPUT_STRUCT_1, INPUT_STRUCT_1]; + assert(x == INPUT); + + const EXPECTED_STRUCT: StructSimple = StructSimple { a: false, b: 30 }; + [EXPECTED_STRUCT, EXPECTED_STRUCT, EXPECTED_STRUCT] + } + fn types_array_with_generic_struct( + x: [StructDoubleGeneric, str[1]>; 2], + ) -> [StructDoubleGeneric, str[1]>; 2] { + const INPUT_STRUCT: StructDoubleGeneric, str[1]> = + StructDoubleGeneric { + a: StructSingleGeneric { a: 10 }, + b: __to_str_array("A"), + }; + const INPUT = [INPUT_STRUCT, INPUT_STRUCT]; + assert(x == INPUT); + + const EXPECTED_STRUCT: StructDoubleGeneric, str[1]> = + StructDoubleGeneric { + a: StructSingleGeneric { a: 20 }, + b: __to_str_array("B"), + }; + [EXPECTED_STRUCT, EXPECTED_STRUCT] + } + fn types_array_with_vector(x: [Vec; 1]) -> [Vec; 1] { + let INPUT_VEC = vec_32_from([1, 2, 3]); + let INPUT = [INPUT_VEC]; + assert(x == INPUT); + + let EXPECTED_VEC: Vec = vec_32_from([3, 2, 1]); + let EXPECTED: [Vec; 1] = [EXPECTED_VEC]; + return EXPECTED + } + + /** + * Tuples + */ + fn types_tuple(x: (u8, u8, u8)) -> (u8, u8, u8) { + const INPUT: (u8, u8, u8) = (1, 2, 3); + assert(x == INPUT); + + const EXPECTED: (u8, u8, u8) = (3, 2, 1); + return EXPECTED + } + fn types_tuple_complex( + x: (u8, StructSingleGeneric>, str[3]), + ) -> (u8, StructSingleGeneric>, str[3]) { + let INPUT: (u8, StructSingleGeneric>, str[3]) = + (1, StructSingleGeneric { a: StructSingleGeneric { a: 10 } }, __to_str_array("ABC")); + assert(x == INPUT); + + let EXPECTED: (u8, StructSingleGeneric>, str[3]) = + (3, StructSingleGeneric { a: StructSingleGeneric { a: 30 } }, __to_str_array("CBA")); + return EXPECTED + } + fn types_tuple_with_native_types(x: (AssetId, AssetId, bool)) -> (AssetId, AssetId, bool) { + const A = AssetId::from(0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA); + const B = AssetId::from(0xBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB); + const C = true; + const INPUT: (AssetId, AssetId, bool) = (A, B, C); + assert(x == INPUT); + + const F = false; + const EXPECTED: (AssetId, AssetId, bool) = (B, A, F); + return EXPECTED + } + fn types_alias_tuple_with_native_types(x: TupleWithNativeAssets) -> TupleWithNativeAssets { + const A = AssetId::from(0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA); + const B = AssetId::from(0xBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB); + const C = true; + const INPUT: (AssetId, AssetId, bool) = (A, B, C); + assert(x == INPUT); + + const F = false; + const EXPECTED: (AssetId, AssetId, bool) = (B, A, F); + return EXPECTED + } + + /** + * Structs + */ + fn types_struct_simple(x: StructSimple) -> StructSimple { + const INPUT: StructSimple = StructSimple { a: true, b: 10 }; + assert(x == INPUT); + + const EXPECTED: StructSimple = StructSimple { a: false, b: 30 }; + return EXPECTED + } + fn types_struct_generic(x: StructSingleGeneric) -> StructSingleGeneric { + const INPUT: StructSingleGeneric = StructSingleGeneric { a: 10 }; + assert(x == INPUT); + + const EXPECTED: StructSingleGeneric = StructSingleGeneric { a: 20 }; + return EXPECTED + } + fn types_struct_with_tuple( + x: StructSingleGeneric<(bool, u64)>, + ) -> StructSingleGeneric<(bool, u64)> { + const INPUT: StructSingleGeneric<(bool, u64)> = StructSingleGeneric { a: (true, 10) }; + assert(x == INPUT); + + const EXPECTED: StructSingleGeneric<(bool, u64)> = StructSingleGeneric { a: (false, 20) }; + return EXPECTED + } + fn types_struct_double_generic( + x: StructGenericWithEnum, + ) -> StructGenericWithEnum { + const INPUT: StructGenericWithEnum = StructGenericWithEnum { + a: 10, + b: EnumDoubleGeneric::a(10), + }; + assert(x == INPUT); + + const EXPECTED: StructGenericWithEnum = StructGenericWithEnum { + a: 20, + b: EnumDoubleGeneric::b(10), + }; + return EXPECTED + } + fn type_struct_external(x: ExternalStruct) -> ExternalStruct { + const INPUT: ExternalStruct = ExternalStruct { value: 10 }; + assert(x == INPUT); + + const EXPECTED: ExternalStruct = ExternalStruct { value: 20 }; + return EXPECTED + } + fn types_struct_with_nested_array(x: StructWithNestedArray) -> StructWithNestedArray { + const INPUT_STRUCT: StructDoubleGeneric, str[1]> = + StructDoubleGeneric { + a: StructSingleGeneric { a: 10 }, + b: __to_str_array("A"), + }; + const INPUT = StructWithNestedArray { a: [INPUT_STRUCT, INPUT_STRUCT] }; + assert(x == INPUT); + + const EXPECTED_STRUCT: StructDoubleGeneric, str[1]> = + StructDoubleGeneric { + a: StructSingleGeneric { a: 20 }, + b: __to_str_array("B"), + }; + const EXPECTED = StructWithNestedArray { a: [EXPECTED_STRUCT, EXPECTED_STRUCT] }; + return EXPECTED + } + fn types_struct_with_nested_tuple(x: StructWithNestedTuple) -> StructWithNestedTuple { + const INPUT: StructWithNestedTuple = StructWithNestedTuple { + a: (10, StructSingleGeneric { a: StructSingleGeneric { a: 20 } }, __to_str_array("ABC")), + }; + assert(x == INPUT); + + const EXPECTED: StructWithNestedTuple = StructWithNestedTuple { + a: (30, StructSingleGeneric { a: StructSingleGeneric { a: 40 } }, __to_str_array("CBA")), + }; + return EXPECTED + } + fn types_struct_with_nested_struct(x: StructWithNestedStruct) -> StructWithNestedStruct { + const INPUT: StructWithNestedStruct = StructWithNestedStruct { + a: StructDoubleGeneric { + a: StructSingleGeneric { a: 10 }, + b: 20, + }, + }; + assert(x == INPUT); + + const EXPECTED: StructWithNestedStruct = StructWithNestedStruct { + a: StructDoubleGeneric { + a: StructSingleGeneric { a: 30 }, + b: 40, + }, + }; + return EXPECTED + } + fn types_struct_with_multiple_struct_params(x: StructA, y: StructB, z: StructC) -> bool { + const STRUCT_A: StructA = StructA { propA1: 10 }; + assert(x == STRUCT_A); + + const STRUCT_B: StructB = StructB { propB1: STRUCT_A, propB2: 20 }; + assert(y == STRUCT_B); + + // PropC2 + let mut propC2 = Vec::new(); + propC2.push(STRUCT_B); + + // PropC3 + const STRUCT_E: StructE = StructE { + propE1: STRUCT_A, + propE2: STRUCT_B, + propE3: 30, + }; + let mut propD1 = Vec::new(); + propD1.push(STRUCT_E); + + const STRUCT_F: StructF = StructF { propF1: 50, propF2: __to_str_array("A") }; + let propC3: StructD> = StructD { + propD1: propD1, + propD2: 40, + propD3: STRUCT_F, + }; + + + let STRUCT_C: StructC = StructC { + propC1: STRUCT_A, + propC2: propC2, + propC3: propC3, + // propC4: [STRUCT_D], + // propC5: [STRUCT_D], + }; + + assert(z == STRUCT_C); + + return true; + + + + + + // const STRUCT_C4: StructD> = StructD { + // propD1: [StructE { propE1: STRUCT_A, propE2: STRUCT_B, propE3: 30 }], + // propD2: 40, + // propD3: StructF { propF1: 50, propF2: true }, + // }; + + // const STRUCT_C5: StructD>> = StructD> { + // propD1: [StructE { propE1: STRUCT_A, propE2: STRUCT_B, propE3: 30 }], + // propD2: 40, + // propD3: StructF { propF1: 50, propF2: [StructG { propG1: 60 }] }, + // }; + + // const STRUCT_C: StructC = StructC { + // propC1: STRUCT_A, + // propC2: [STRUCT_B], + // propC3: STRUCT_C3, + // propC4: [STRUCT_C4], + // propC5: [STRUCT_C5], + // }; + // const STRUCT_B: StructB = StructB { propB1: INPUT_X, propB2: 20 }; + // const STRUCT_C: StructC = StructC { propC1: INPUT_X, propC2: [INPUT_Y], propC3: INPUT_D, propC4: [INPUT_D], propC5: [INPUT_D] }; + // const STRUCT_D: StructD = StructD { + // propD1: [StructE { propE1: INPUT_X, propE2: INPUT_Y, propE3: 30 }], + // propD2: 40, + // propD3: StructF { propF1: 50, propF2: __to_str_array("ABC") }, + // }; + // assert(y == INPUT_Y); + + } + fn types_struct_with_complex_nested_struct(x: StructD>>) -> bool { + false + } + fn types_struct_with_single_option(x: StructWithSingleOption) -> StructWithSingleOption { + x + } + fn types_struct_with_implicit_generics( + x: StructWithImplicitGenerics, + ) -> StructWithImplicitGenerics { + x + } + // @todo - unable to create this struct. + fn types_struct_with_array(x: StructWithGenericArray) -> StructWithGenericArray { + const INPUT_B256: b256 = 0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa; + const INPUT_STRUCT: StructDoubleGeneric = StructDoubleGeneric { + a: INPUT_B256, + b: 10 + }; + const INPUT = StructWithGenericArray { + a: [INPUT_STRUCT, INPUT_STRUCT, INPUT_STRUCT] + } + assert(x === INPUT) + + const EXPECTED_B256: b256 = 0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb; + const EXPECTED_STRUCT: StructDoubleGeneric = StructDoubleGeneric { + a: EXPECTED_B256, + b: 20 + }; + const EXPECTED = StructWithGenericArray { + a: [EXPECTED_STRUCT, EXPECTED_STRUCT, EXPECTED_STRUCT] + } + + EXPECTED + } + // @todo - need help + fn types_struct_with_vector(x: StructWithVector) -> StructWithVector { + // let INPUT_VEC: Vec = vec_u8_from([1, 2, 3]); + // let INPUT: StructWithVector = { + // a: 1, + // b: INPUT_VEC + // } + + x + } + + // @todo - also broken + fn types_struct_with_array_of_enums(x: StructWithEnumArray) -> StructWithEnumArray { + // const INPUT_ENUM = EnumWithNative::Checked; + // const INPUT: StructWithEnumArray = StructWithEnumArray { + // a: [INPUT_ENUM, INPUT_ENUM, INPUT_ENUM] + // } + // assert(x == INPUT); + + // return INPUT + + return x + } + + /** + * Enums + */ + fn types_enum(x: EnumWithNative) -> EnumWithNative { + assert(x == EnumWithNative::Checked); + + EnumWithNative::Pending + } + fn types_enum_with_builtin_type(x: EnumWithBuiltinType) -> EnumWithBuiltinType { + assert(x == EnumWithBuiltinType::a(true)); + + EnumWithBuiltinType::b(20) + } + fn types_enum_with_vector(x: EnumWithVector) -> EnumWithVector { + assert(x == EnumWithVector::a(10)); + + let EXPECTED_VEC = vec_u8_from([1, 2, 3]); + return EnumWithVector::b(EXPECTED_VEC) + } + fn types_generic_enum(x: EnumDoubleGeneric) -> EnumDoubleGeneric { + const INPUT: EnumDoubleGeneric = EnumDoubleGeneric::a(10); + assert(x == INPUT); + + const EXPECTED: EnumDoubleGeneric = EnumDoubleGeneric::b(20); + return EXPECTED + } + fn types_enum_external(x: ExternalEnum) -> ExternalEnum { + assert_eq(x, ExternalEnum::A); + + return ExternalEnum::B; + } + fn types_enum_with_structs(x: EnumWithStructs) -> EnumWithStructs { + const INPUT: EnumWithStructs = EnumWithStructs::a(EnumWithNative::Checked); + assert(x == INPUT); + + const EXPECTED: EnumWithStructs = EnumWithStructs::b(StructSimple { a: true, b: 10 }); + return EXPECTED + } + + /** + * Vectors + */ + fn types_vector_u8(x: Vec) -> Vec { + let INPUT = vec_u8_from([1, 2, 3]); + assert(x == INPUT); + + let EXPECTED = vec_u8_from([3, 2, 1]); + return EXPECTED + } + fn types_vector_boolean(x: Vec) -> Vec { + let INPUT = vec_bool_from([true, false, true, false]); + assert(x == INPUT); + + let EXPECTED = vec_bool_from([false, true, false, true]); + return EXPECTED + } + fn types_vector_inside_vector(x: Vec>) -> Vec> { + let mut INPUT = Vec::new(); + INPUT.push(vec_32_from([1, 2, 3])); + assert(x == INPUT); + + let mut EXPECTED = Vec::new(); + EXPECTED.push(vec_32_from([3, 2, 1])); + EXPECTED.push(vec_32_from([6, 5, 4])); + return EXPECTED + } + fn types_vector_with_struct(x: Vec) -> Vec { + let mut INPUT = Vec::new(); + INPUT.push(StructSimple { a: true, b: 10 }); + assert(x == INPUT); + + let mut EXPECTED = Vec::new(); + EXPECTED.push(StructSimple { a: false, b: 30 }); + return EXPECTED + } + fn types_vector_option(x: Vec) -> Vec { + let mut INPUT = Vec::new(); + INPUT.push(StructWithMultiOption { + a: [Some(1), Some(2), Some(3), Some(4), Some(5)], + }); + assert(x == INPUT); + + let mut EXPECTED = Vec::new(); + EXPECTED.push(StructWithMultiOption { + a: [Some(5), Some(4), Some(3), Some(2), Some(1)], + }); + return EXPECTED + } + + /** + * Options + */ + fn types_option(x: Option) -> Option { + const INPUT: Option = Option::Some(10); + assert(x === INPUT); + + const EXPECTED: Option = Option::None; + return EXPECTED + } + fn types_option_geo(x: Option) -> Option { + const INPUT_STRUCT: StructSimple = StructSimple { + a: true, + b: 10 + } + const INPUT: Option = Option::Some(INPUT_STRUCT); + assert(x === INPUT); + + const EXPECTED: Option = Option::None; + return EXPECTED + } + + /** + * Native types + */ + fn types_asset_id(x: AssetId) -> AssetId { + const INPUT = AssetId::from(0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA); + assert(x == INPUT); + + const EXPECTED = AssetId::from(0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb); + return EXPECTED + } + fn type_identity_address(x: Identity) -> Identity { + const ADDRESS = Address::from(0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA); + const INPUT = Identity::Address(ADDRESS); + assert(x == INPUT); + + const EXPECTED_ADDRESS = Address::from(0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb); + const EXPECTED = Identity::Address(EXPECTED_ADDRESS); + return EXPECTED + } + fn type_identity_contract_id(x: Identity) -> Identity { + const CONTRACT_ID = ContractId::from(0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA); + const INPUT = Identity::ContractId(CONTRACT_ID); + assert(x == INPUT); + + const EXPECTED_ADDRESS = ContractId::from(0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb); + const EXPECTED = Identity::ContractId(EXPECTED_ADDRESS); + return EXPECTED + } + fn type_address(x: Address) -> Address { + const INPUT = Address::from(0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA); + assert(x == INPUT); + + const EXPECTED = Address::from(0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb); + return EXPECTED + } + fn type_contract_id(x: ContractId) -> ContractId { + const INPUT = ContractId::from(0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA); + assert(x == INPUT); + + const EXPECTED = ContractId::from(0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb); + return EXPECTED + } + fn types_evm_address(x: EvmAddress) -> EvmAddress { + let INPUT = EvmAddress::from(0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA); + assert(x == INPUT); + + let EXPECTED = EvmAddress::from(0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb); + return EXPECTED + } + fn types_result(x: Result) -> Result { + if (x.is_err()) { + return Err(__to_str_array("InputError")); + } + + let result = divide(20, x.unwrap()); + match result { + Ok(value) => Ok(value), + Err(MyContractError::DivisionByZero) => Err(__to_str_array("DivisError")), + } + } + + /** + * Void + */ + fn types_void(x: ()) -> () { + x + } + fn types_void_then_value(x: (), y: u8) -> () { + const inputY = 10; + assert(y == inputY); + + () + } + fn types_value_then_void(x: u8, y: ()) -> () { + const inputX = 10; + assert(x == inputX); + + () + } + fn types_value_then_void_then_value(x: u8, y: (), z: u8) -> () { + const inputX = 10; + assert(x == inputX); + + const inputZ = 20; + assert(z == inputZ); + + () + } + fn types_value_then_value_then_void_then_void(x: u8, y: u8, z: (), a: ()) -> () { + const inputX = 10; + assert(x == inputX); + + const inputY = 20; + assert(z == inputZ); + + () + } + + /** + * Multi-args + * @TODO revisit these after we can resolve the issue around the input being returned as the output. + */ + fn multi_arg_u64_u64(x: u64, y: u64) -> u64 { + const INPUT_X = 1; + const INPUT_Y = 2; + assert(x == INPUT_X); + assert(y == INPUT_Y); + + const EXPECTED = 3; + return EXPECTED; + } + fn multi_arg_b256_bool(x: b256, y: bool) -> (b256, bool) { + const INPUT_X: b256 = 0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa; + const INPUT_Y: bool = true; + assert_eq(x, INPUT); + + const EXPECTED: (b256, bool) = (0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb, false); + return EXPECTED + } + fn multi_arg_vector_vector(x: Vec, y: Vec) -> (Vec, Vec) { + let INPUT_X = vec_u8_from([1, 2, 3]); + let INPUT_Y = vec_u8_from([4, 5, 6]); + expect(x === INPUT_X); + expect(y === INPUT_Y); + + const EXPECTED_X = vec_u8_from([7, 8, 9]); + const EXPECTED_X = vec_u8_from([10, 11, 12]); + (EXPECTED_X, EXPECTED_Y) + } + fn multi_arg_vector_b256(x: Vec, y: b256) -> (Vec, b256) { + let INPUT_X = vec_u8_from([1, 2, 3]); + let INPUT_Y: b256 = 0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa; + expect(x === INPUT_X); + expect(y === INPUT_Y); + + const EXPECTED_X = vec_u8_from([7, 8, 9]); + const EXPECTED_Y = 0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb; + const EXPECTED = (EXPECTED_X, EXPECTED_Y); + return EXPECTED + } + fn multi_arg_struct_vector(x: StructSimple, y: Vec) -> (StructSimple, Vec) { + const INPUT_X = StructSimple { + a: true, + b: 1 + }; + const INPUT_Y = vec_u8_from([1, 2, 3]); + expect(x === INPUT_X); + expect(y === INPUT_Y); + + + const EXPECTED_X = StructSimple { + a: false, + b: 2 + }; + const EXPECTED_Y = vec_u8_from([4, 5, 6]); + const EXPECTED = (EXPECTED_X, EXPECTED_Y); + return EXPECTED + } + fn multi_arg_u64_struct(x: u64, y: StructSimple) -> (u64, StructSimple) { + (x, y) + } + fn multi_arg_str_str(x: str[5], y: str[5]) -> (str[5], str[5]) { + (x, y) + } + fn multi_arg_u32_vector_vector(x: u32, y: Vec, z: Vec) -> (u32, Vec, Vec) { + (x, y, z) + } + fn multi_arg_complex( + x: StructDoubleGeneric<[b256; 3], u8>, + y: [StructDoubleGeneric; 4], + z: (str[5], bool), + a: StructSimple, + ) { + () + } +} diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/utils.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/utils.sw new file mode 100644 index 00000000000..feea71b922f --- /dev/null +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/utils.sw @@ -0,0 +1,26 @@ +library; + +pub fn vec_32_from(vals: [u32; 3]) -> Vec { + let mut vec = Vec::new(); + vec.push(vals[0]); + vec.push(vals[1]); + vec.push(vals[2]); + vec +} + +pub fn vec_u8_from(vals: [u8; 3]) -> Vec { + let mut vec = Vec::new(); + vec.push(vals[0]); + vec.push(vals[1]); + vec.push(vals[2]); + vec +} + +pub fn vec_bool_from(vals: [bool; 4]) -> Vec { + let mut vec = Vec::new(); + vec.push(vals[0]); + vec.push(vals[1]); + vec.push(vals[2]); + vec.push(vals[3]); + vec +} \ No newline at end of file diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-library/Forc.toml b/packages/fuel-gauge/test/fixtures/forc-projects/abi-library/Forc.toml new file mode 100644 index 00000000000..d28ab3240d9 --- /dev/null +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-library/Forc.toml @@ -0,0 +1,7 @@ +[project] +authors = ["Fuel Labs "] +entry = "lib.sw" +license = "Apache-2.0" +name = "abi-library" + +[dependencies] diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-library/src/lib.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-library/src/lib.sw new file mode 100644 index 00000000000..8e59a3145ad --- /dev/null +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-library/src/lib.sw @@ -0,0 +1,22 @@ +library; + +// anything `pub` here will be exported as a part of this library's API +pub struct ExternalStruct { + pub value: u64, +} + +pub enum ExternalEnum { + A: (), + B: (), +} + +impl Eq for ExternalStruct { + fn eq(self, other: Self) -> bool { + self.value == other.value + } +} +impl Eq for ExternalEnum { + fn eq(self, other: Self) -> bool { + self == other + } +} \ No newline at end of file From b0002815bbbb09a55795a4c8559673fe751d202a Mon Sep 17 00:00:00 2001 From: Peter Smith Date: Thu, 3 Oct 2024 19:26:39 +0100 Subject: [PATCH 02/37] chore: changeset --- .changeset/long-ducks-jump.md | 2 ++ 1 file changed, 2 insertions(+) create mode 100644 .changeset/long-ducks-jump.md diff --git a/.changeset/long-ducks-jump.md b/.changeset/long-ducks-jump.md new file mode 100644 index 00000000000..a845151cc84 --- /dev/null +++ b/.changeset/long-ducks-jump.md @@ -0,0 +1,2 @@ +--- +--- From 96a74ea7e47f099cc31dffcbf72d3b5436f4ff26 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 09:51:29 -0300 Subject: [PATCH 03/37] fix Eq for ExternalEnum --- .../test/fixtures/forc-projects/abi-library/src/lib.sw | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-library/src/lib.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-library/src/lib.sw index 8e59a3145ad..90c6c70044a 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-library/src/lib.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-library/src/lib.sw @@ -15,8 +15,13 @@ impl Eq for ExternalStruct { self.value == other.value } } + impl Eq for ExternalEnum { fn eq(self, other: Self) -> bool { - self == other + match (self, other) { + (ExternalEnum::A, ExternalEnum::A) => true, + (ExternalEnum::B, ExternalEnum::B) => true, + _ => false, + } } -} \ No newline at end of file +} From 0932fd645e8935381891e84e76678faae2ecb5be Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 09:57:10 -0300 Subject: [PATCH 04/37] fix equals use --- .../forc-projects/abi-contract/src/main.sw | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index d8c75629a2c..05bf686447b 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -545,7 +545,7 @@ impl MyContract for Contract { const INPUT = StructWithGenericArray { a: [INPUT_STRUCT, INPUT_STRUCT, INPUT_STRUCT] } - assert(x === INPUT) + assert(x == INPUT) const EXPECTED_B256: b256 = 0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb; const EXPECTED_STRUCT: StructDoubleGeneric = StructDoubleGeneric { @@ -676,7 +676,7 @@ impl MyContract for Contract { */ fn types_option(x: Option) -> Option { const INPUT: Option = Option::Some(10); - assert(x === INPUT); + assert(x == INPUT); const EXPECTED: Option = Option::None; return EXPECTED @@ -687,7 +687,7 @@ impl MyContract for Contract { b: 10 } const INPUT: Option = Option::Some(INPUT_STRUCT); - assert(x === INPUT); + assert(x == INPUT); const EXPECTED: Option = Option::None; return EXPECTED @@ -815,8 +815,8 @@ impl MyContract for Contract { fn multi_arg_vector_vector(x: Vec, y: Vec) -> (Vec, Vec) { let INPUT_X = vec_u8_from([1, 2, 3]); let INPUT_Y = vec_u8_from([4, 5, 6]); - expect(x === INPUT_X); - expect(y === INPUT_Y); + expect(x == INPUT_X); + expect(y == INPUT_Y); const EXPECTED_X = vec_u8_from([7, 8, 9]); const EXPECTED_X = vec_u8_from([10, 11, 12]); @@ -825,8 +825,8 @@ impl MyContract for Contract { fn multi_arg_vector_b256(x: Vec, y: b256) -> (Vec, b256) { let INPUT_X = vec_u8_from([1, 2, 3]); let INPUT_Y: b256 = 0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa; - expect(x === INPUT_X); - expect(y === INPUT_Y); + expect(x == INPUT_X); + expect(y == INPUT_Y); const EXPECTED_X = vec_u8_from([7, 8, 9]); const EXPECTED_Y = 0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb; @@ -839,8 +839,8 @@ impl MyContract for Contract { b: 1 }; const INPUT_Y = vec_u8_from([1, 2, 3]); - expect(x === INPUT_X); - expect(y === INPUT_Y); + expect(x == INPUT_X); + expect(y == INPUT_Y); const EXPECTED_X = StructSimple { From d91f6fcb41bfe3040e49c7bfd7df9ae66fbf2160 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 10:08:37 -0300 Subject: [PATCH 05/37] linting --- .../forc-projects/abi-contract/src/main.sw | 22 +++++++++---------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index 05bf686447b..ba3944e30c3 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -540,21 +540,21 @@ impl MyContract for Contract { const INPUT_B256: b256 = 0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa; const INPUT_STRUCT: StructDoubleGeneric = StructDoubleGeneric { a: INPUT_B256, - b: 10 + b: 10, }; - const INPUT = StructWithGenericArray { - a: [INPUT_STRUCT, INPUT_STRUCT, INPUT_STRUCT] - } - assert(x == INPUT) + const INPUT: StructWithGenericArray = StructWithGenericArray { + a: [INPUT_STRUCT, INPUT_STRUCT, INPUT_STRUCT], + }; + assert(x == INPUT); const EXPECTED_B256: b256 = 0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb; const EXPECTED_STRUCT: StructDoubleGeneric = StructDoubleGeneric { a: EXPECTED_B256, - b: 20 + b: 20, + }; + const EXPECTED: StructWithGenericArray = StructWithGenericArray { + a: [EXPECTED_STRUCT, EXPECTED_STRUCT, EXPECTED_STRUCT], }; - const EXPECTED = StructWithGenericArray { - a: [EXPECTED_STRUCT, EXPECTED_STRUCT, EXPECTED_STRUCT] - } EXPECTED } @@ -684,8 +684,8 @@ impl MyContract for Contract { fn types_option_geo(x: Option) -> Option { const INPUT_STRUCT: StructSimple = StructSimple { a: true, - b: 10 - } + b: 10, + }; const INPUT: Option = Option::Some(INPUT_STRUCT); assert(x == INPUT); From 801c02dd991dbefb4be9fb93e5995061ca53bcdf Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 10:14:18 -0300 Subject: [PATCH 06/37] minor fixes within contract code --- .../forc-projects/abi-contract/src/main.sw | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index ba3944e30c3..0c394061228 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -807,7 +807,7 @@ impl MyContract for Contract { fn multi_arg_b256_bool(x: b256, y: bool) -> (b256, bool) { const INPUT_X: b256 = 0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa; const INPUT_Y: bool = true; - assert_eq(x, INPUT); + assert_eq(x, INPUT_X); const EXPECTED: (b256, bool) = (0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb, false); return EXPECTED @@ -815,18 +815,18 @@ impl MyContract for Contract { fn multi_arg_vector_vector(x: Vec, y: Vec) -> (Vec, Vec) { let INPUT_X = vec_u8_from([1, 2, 3]); let INPUT_Y = vec_u8_from([4, 5, 6]); - expect(x == INPUT_X); - expect(y == INPUT_Y); + assert(x == INPUT_X); + assert(y == INPUT_Y); const EXPECTED_X = vec_u8_from([7, 8, 9]); - const EXPECTED_X = vec_u8_from([10, 11, 12]); + const EXPECTED_Y = vec_u8_from([10, 11, 12]); (EXPECTED_X, EXPECTED_Y) } fn multi_arg_vector_b256(x: Vec, y: b256) -> (Vec, b256) { let INPUT_X = vec_u8_from([1, 2, 3]); let INPUT_Y: b256 = 0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa; - expect(x == INPUT_X); - expect(y == INPUT_Y); + assert(x == INPUT_X); + assert(y == INPUT_Y); const EXPECTED_X = vec_u8_from([7, 8, 9]); const EXPECTED_Y = 0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb; @@ -839,8 +839,8 @@ impl MyContract for Contract { b: 1 }; const INPUT_Y = vec_u8_from([1, 2, 3]); - expect(x == INPUT_X); - expect(y == INPUT_Y); + assert(x == INPUT_X); + assert(y == INPUT_Y); const EXPECTED_X = StructSimple { From d8b58e5eca50295554204e0064b637826fb4d88d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 10:19:52 -0300 Subject: [PATCH 07/37] fix comparison at types_value_then_value_then_void_then_void --- .../test/fixtures/forc-projects/abi-contract/src/main.sw | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index 0c394061228..a6b4d8132fe 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -786,7 +786,7 @@ impl MyContract for Contract { assert(x == inputX); const inputY = 20; - assert(z == inputZ); + assert(y == inputY); () } From f6f6cd4962e71bcd5f6cda820ff18121b26dd5f5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 11:06:38 -0300 Subject: [PATCH 08/37] implementing missing Eq --- .../abi-contract/src/equality.sw | 21 ++++++++++++++++++- 1 file changed, 20 insertions(+), 1 deletion(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw index 4190341b3b6..1da1a8a5854 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw @@ -52,6 +52,18 @@ impl Eq for StructDoubleGeneric, str[1]> { } } +impl Eq for StructDoubleGeneric { + fn eq(self, other: Self) -> bool { + self.a == other.a && self.b == other.b + } +} + +impl Eq for [StructDoubleGeneric; 3] { + fn eq(self, other: Self) -> bool { + self[0] == other[0] && self[1] == other[1] && self[2] == other[2] + } +} + impl Eq for [StructDoubleGeneric, str[1]>; 2] { fn eq(self, other: Self) -> bool { self[0] == other[0] && self[1] == other[1] @@ -372,4 +384,11 @@ impl Eq for Vec { } true } -} \ No newline at end of file +} + +impl Eq for StructWithGenericArray { + fn eq(self, other: Self) -> bool { + self.a == other.a + } +} + From 54af3cc59e9e4c8f9974b2bf8c6333ac12eaabd0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 11:06:49 -0300 Subject: [PATCH 09/37] using let for heap types --- .../forc-projects/abi-contract/src/main.sw | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index a6b4d8132fe..5eb85047275 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -818,8 +818,8 @@ impl MyContract for Contract { assert(x == INPUT_X); assert(y == INPUT_Y); - const EXPECTED_X = vec_u8_from([7, 8, 9]); - const EXPECTED_Y = vec_u8_from([10, 11, 12]); + let EXPECTED_X = vec_u8_from([7, 8, 9]); + let EXPECTED_Y = vec_u8_from([10, 11, 12]); (EXPECTED_X, EXPECTED_Y) } fn multi_arg_vector_b256(x: Vec, y: b256) -> (Vec, b256) { @@ -828,9 +828,9 @@ impl MyContract for Contract { assert(x == INPUT_X); assert(y == INPUT_Y); - const EXPECTED_X = vec_u8_from([7, 8, 9]); + let EXPECTED_X = vec_u8_from([7, 8, 9]); const EXPECTED_Y = 0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb; - const EXPECTED = (EXPECTED_X, EXPECTED_Y); + let EXPECTED = (EXPECTED_X, EXPECTED_Y); return EXPECTED } fn multi_arg_struct_vector(x: StructSimple, y: Vec) -> (StructSimple, Vec) { @@ -838,7 +838,7 @@ impl MyContract for Contract { a: true, b: 1 }; - const INPUT_Y = vec_u8_from([1, 2, 3]); + let INPUT_Y = vec_u8_from([1, 2, 3]); assert(x == INPUT_X); assert(y == INPUT_Y); @@ -847,8 +847,8 @@ impl MyContract for Contract { a: false, b: 2 }; - const EXPECTED_Y = vec_u8_from([4, 5, 6]); - const EXPECTED = (EXPECTED_X, EXPECTED_Y); + let EXPECTED_Y = vec_u8_from([4, 5, 6]); + let EXPECTED = (EXPECTED_X, EXPECTED_Y); return EXPECTED } fn multi_arg_u64_struct(x: u64, y: StructSimple) -> (u64, StructSimple) { From fa6e12adc37ac27de1a0e105d2b5849a8f83e639 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 11:25:47 -0300 Subject: [PATCH 10/37] rename vec_32_from --- .../fixtures/forc-projects/abi-contract/src/main.sw | 10 +++++----- .../fixtures/forc-projects/abi-contract/src/utils.sw | 4 ++-- 2 files changed, 7 insertions(+), 7 deletions(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index 5eb85047275..fb8e3352983 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -305,11 +305,11 @@ impl MyContract for Contract { [EXPECTED_STRUCT, EXPECTED_STRUCT] } fn types_array_with_vector(x: [Vec; 1]) -> [Vec; 1] { - let INPUT_VEC = vec_32_from([1, 2, 3]); + let INPUT_VEC = vec_u32_from([1, 2, 3]); let INPUT = [INPUT_VEC]; assert(x == INPUT); - let EXPECTED_VEC: Vec = vec_32_from([3, 2, 1]); + let EXPECTED_VEC: Vec = vec_u32_from([3, 2, 1]); let EXPECTED: [Vec; 1] = [EXPECTED_VEC]; return EXPECTED } @@ -640,12 +640,12 @@ impl MyContract for Contract { } fn types_vector_inside_vector(x: Vec>) -> Vec> { let mut INPUT = Vec::new(); - INPUT.push(vec_32_from([1, 2, 3])); + INPUT.push(vec_u32_from([1, 2, 3])); assert(x == INPUT); let mut EXPECTED = Vec::new(); - EXPECTED.push(vec_32_from([3, 2, 1])); - EXPECTED.push(vec_32_from([6, 5, 4])); + EXPECTED.push(vec_u32_from([3, 2, 1])); + EXPECTED.push(vec_u32_from([6, 5, 4])); return EXPECTED } fn types_vector_with_struct(x: Vec) -> Vec { diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/utils.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/utils.sw index feea71b922f..27338730220 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/utils.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/utils.sw @@ -1,6 +1,6 @@ library; -pub fn vec_32_from(vals: [u32; 3]) -> Vec { +pub fn vec_u32_from(vals: [u32; 3]) -> Vec { let mut vec = Vec::new(); vec.push(vals[0]); vec.push(vals[1]); @@ -23,4 +23,4 @@ pub fn vec_bool_from(vals: [bool; 4]) -> Vec { vec.push(vals[2]); vec.push(vals[3]); vec -} \ No newline at end of file +} From e2cb4ccb47764e2021d2254458b3016af7d08ab3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 11:28:26 -0300 Subject: [PATCH 11/37] rename some functions --- .../fixtures/forc-projects/full/src/main.sw | 8 +++--- .../test/fixtures/templates/contract/main.hbs | 12 ++++----- packages/fuel-gauge/src/abi/abi-coder.test.ts | 25 +++++++------------ .../forc-projects/abi-contract/src/main.sw | 16 ++++++------ 4 files changed, 27 insertions(+), 34 deletions(-) diff --git a/packages/abi-typegen/test/fixtures/forc-projects/full/src/main.sw b/packages/abi-typegen/test/fixtures/forc-projects/full/src/main.sw index 500f8639108..b10f05a2a0f 100644 --- a/packages/abi-typegen/test/fixtures/forc-projects/full/src/main.sw +++ b/packages/abi-typegen/test/fixtures/forc-projects/full/src/main.sw @@ -88,8 +88,8 @@ abi MyContract { fn types_str_slice(x: str) -> str; fn types_std_string(x: String) -> String; fn types_result(x: Result) -> Result; - fn type_address(x: Address) -> Address; - fn type_contract_id(x: ContractId) -> ContractId; + fn types_address(x: Address) -> Address; + fn types_contract_id(x: ContractId) -> ContractId; fn type_identity(x: Identity) -> Identity; fn type_external_struct(x: ExternalStruct) -> ExternalStruct; fn type_external_enum(x: ExternalEnum) -> ExternalEnum; @@ -211,10 +211,10 @@ impl MyContract for Contract { Err(MyContractError::DivisionByZero) => Err(__to_str_array("DivisError")), } } - fn type_address(x: Address) -> Address { + fn types_address(x: Address) -> Address { x } - fn type_contract_id(x: ContractId) -> ContractId { + fn types_contract_id(x: ContractId) -> ContractId { x } fn type_identity(x: Identity) -> Identity { diff --git a/packages/abi-typegen/test/fixtures/templates/contract/main.hbs b/packages/abi-typegen/test/fixtures/templates/contract/main.hbs index 942b70a5042..316a9a639d8 100644 --- a/packages/abi-typegen/test/fixtures/templates/contract/main.hbs +++ b/packages/abi-typegen/test/fixtures/templates/contract/main.hbs @@ -692,7 +692,7 @@ const abi = { "concreteTypeId": "f597b637c3b0f588fb8d7086c6f4735caa3122b85f0423b82e489f9bb58e2308" } ], - "name": "type_address", + "name": "types_address", "output": "f597b637c3b0f588fb8d7086c6f4735caa3122b85f0423b82e489f9bb58e2308", "attributes": null }, @@ -703,7 +703,7 @@ const abi = { "concreteTypeId": "29c10735d33b5159f0c71ee1dbd17b36a3e69e41f00fab0d42e1bd9f428d8a54" } ], - "name": "type_contract_id", + "name": "types_contract_id", "output": "29c10735d33b5159f0c71ee1dbd17b36a3e69e41f00fab0d42e1bd9f428d8a54", "attributes": null }, @@ -1157,8 +1157,8 @@ export class MyContractInterface extends Interface { declare functions: { alias_types_tuple_with_native_types: FunctionFragment; - type_address: FunctionFragment; - type_contract_id: FunctionFragment; + types_address: FunctionFragment; + types_contract_id: FunctionFragment; type_external_enum: FunctionFragment; type_external_struct: FunctionFragment; type_identity: FunctionFragment; @@ -1206,8 +1206,8 @@ export class MyContract extends Contract { declare interface: MyContractInterface; declare functions: { alias_types_tuple_with_native_types: InvokeFunction<[x: [AssetIdInput, AssetIdInput, boolean]], [AssetIdOutput, AssetIdOutput, boolean]>; - type_address: InvokeFunction<[x: AddressInput], AddressOutput>; - type_contract_id: InvokeFunction<[x: ContractIdInput], ContractIdOutput>; + types_address: InvokeFunction<[x: AddressInput], AddressOutput>; + types_contract_id: InvokeFunction<[x: ContractIdInput], ContractIdOutput>; type_external_enum: InvokeFunction<[x: ExternalEnumInput], ExternalEnumOutput>; type_external_struct: InvokeFunction<[x: ExternalStructInput], ExternalStructOutput>; type_identity: InvokeFunction<[x: IdentityInput], IdentityOutput>; diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 8dda146d4d6..9fe1bd6ed7b 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -1,12 +1,5 @@ import { Contract, FuelError, Interface } from 'fuels'; -import type { - AssetId, - BigNumberish, - DecodedValue, - EvmAddress, - RawSlice, - WalletUnlocked, -} from 'fuels'; +import type { AssetId, BigNumberish, EvmAddress, RawSlice, WalletUnlocked } from 'fuels'; import { expectToThrowFuelError, launchTestNode } from 'fuels/test-utils'; import { AbiContract, AbiContractFactory } from '../../test/typegen'; @@ -925,7 +918,7 @@ describe('AbiCoder', () => { expect(value).toEqual(expected); }); }); - describe('type_identity_address', () => { + describe('types_identity_address', () => { it('should encode/decode just fine', async () => { const input: IdentityInput = { Address: { bits: '0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' }, @@ -934,13 +927,13 @@ describe('AbiCoder', () => { Address: { bits: '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb' }, }; - const { waitForResult } = await contract.functions.type_identity_address(input).call(); + const { waitForResult } = await contract.functions.types_identity_address(input).call(); const { value } = await waitForResult(); expect(value).toStrictEqual(expected); }); }); - describe('type_identity_contract_id', () => { + describe('types_identity_contract_id', () => { it('should encode/decode just fine', async () => { const input: IdentityInput = { ContractId: { bits: '0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' }, @@ -949,33 +942,33 @@ describe('AbiCoder', () => { ContractId: { bits: '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb' }, }; - const { waitForResult } = await contract.functions.type_identity_contract_id(input).call(); + const { waitForResult } = await contract.functions.types_identity_contract_id(input).call(); const { value } = await waitForResult(); expect(value).toStrictEqual(expected); }); }); - describe('type_address', () => { + describe('types_address', () => { it('should encode/decode just fine', async () => { const input = { bits: '0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' }; const expected = { bits: '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb', }; - const { waitForResult } = await contract.functions.type_address(input).call(); + const { waitForResult } = await contract.functions.types_address(input).call(); const { value } = await waitForResult(); expect(value).toStrictEqual(expected); }); }); - describe('type_contract_id', () => { + describe('types_contract_id', () => { it('should encode/decode just fine', async () => { const input = { bits: '0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' }; const expected = { bits: '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb', }; - const { waitForResult } = await contract.functions.type_contract_id(input).call(); + const { waitForResult } = await contract.functions.types_contract_id(input).call(); const { value } = await waitForResult(); expect(value).toStrictEqual(expected); diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index fb8e3352983..fcb4b0c97ef 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -94,10 +94,10 @@ abi MyContract { fn types_option(x: Option) -> Option; fn types_option_geo(x: Option) -> Option; - fn type_identity_address(x: Identity) -> Identity; - fn type_identity_contract_id(x: Identity) -> Identity; - fn type_address(x: Address) -> Address; - fn type_contract_id(x: ContractId) -> ContractId; + fn types_identity_address(x: Identity) -> Identity; + fn types_identity_contract_id(x: Identity) -> Identity; + fn types_address(x: Address) -> Address; + fn types_contract_id(x: ContractId) -> ContractId; fn types_asset_id(x: AssetId) -> AssetId; fn types_evm_address(x: EvmAddress) -> EvmAddress; fn types_result(x: Result) -> Result; @@ -703,7 +703,7 @@ impl MyContract for Contract { const EXPECTED = AssetId::from(0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb); return EXPECTED } - fn type_identity_address(x: Identity) -> Identity { + fn types_identity_address(x: Identity) -> Identity { const ADDRESS = Address::from(0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA); const INPUT = Identity::Address(ADDRESS); assert(x == INPUT); @@ -712,7 +712,7 @@ impl MyContract for Contract { const EXPECTED = Identity::Address(EXPECTED_ADDRESS); return EXPECTED } - fn type_identity_contract_id(x: Identity) -> Identity { + fn types_identity_contract_id(x: Identity) -> Identity { const CONTRACT_ID = ContractId::from(0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA); const INPUT = Identity::ContractId(CONTRACT_ID); assert(x == INPUT); @@ -721,14 +721,14 @@ impl MyContract for Contract { const EXPECTED = Identity::ContractId(EXPECTED_ADDRESS); return EXPECTED } - fn type_address(x: Address) -> Address { + fn types_address(x: Address) -> Address { const INPUT = Address::from(0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA); assert(x == INPUT); const EXPECTED = Address::from(0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb); return EXPECTED } - fn type_contract_id(x: ContractId) -> ContractId { + fn types_contract_id(x: ContractId) -> ContractId { const INPUT = ContractId::from(0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA); assert(x == INPUT); From 33dd29dfd1eeaefc8fe9eb5f45e153f7d1680c03 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 11:29:27 -0300 Subject: [PATCH 12/37] rename types_option_geo --- .../test/fixtures/forc-projects/full/src/main.sw | 4 ++-- .../abi-typegen/test/fixtures/templates/contract/main.hbs | 6 +++--- packages/fuel-gauge/src/abi/abi-coder.test.ts | 4 ++-- .../test/fixtures/forc-projects/abi-contract/src/main.sw | 4 ++-- 4 files changed, 9 insertions(+), 9 deletions(-) diff --git a/packages/abi-typegen/test/fixtures/forc-projects/full/src/main.sw b/packages/abi-typegen/test/fixtures/forc-projects/full/src/main.sw index b10f05a2a0f..74af47c4e97 100644 --- a/packages/abi-typegen/test/fixtures/forc-projects/full/src/main.sw +++ b/packages/abi-typegen/test/fixtures/forc-projects/full/src/main.sw @@ -81,7 +81,7 @@ abi MyContract { fn types_vector_geo(x: Vec) -> Vec; fn types_vector_option(x: Vec) -> Vec; fn types_option(x: Option) -> Option; - fn types_option_geo(x: Option) -> Option; + fn types_option_struct(x: Option) -> Option; fn types_evm_address(x: EvmAddress) -> EvmAddress; fn types_bytes(x: Bytes) -> Bytes; fn types_raw_slice(x: raw_slice) -> raw_slice; @@ -182,7 +182,7 @@ impl MyContract for Contract { fn types_option(x: Option) -> Option { x } - fn types_option_geo(x: Option) -> Option { + fn types_option_struct(x: Option) -> Option { x } fn types_evm_address(x: EvmAddress) -> EvmAddress { diff --git a/packages/abi-typegen/test/fixtures/templates/contract/main.hbs b/packages/abi-typegen/test/fixtures/templates/contract/main.hbs index 316a9a639d8..700db8e1719 100644 --- a/packages/abi-typegen/test/fixtures/templates/contract/main.hbs +++ b/packages/abi-typegen/test/fixtures/templates/contract/main.hbs @@ -905,7 +905,7 @@ const abi = { "concreteTypeId": "3597e0782bd4dbaf5c8025b40ff3a325845ee34caa713a6d664bda034a31d02a" } ], - "name": "types_option_geo", + "name": "types_option_struct", "output": "3597e0782bd4dbaf5c8025b40ff3a325845ee34caa713a6d664bda034a31d02a", "attributes": null }, @@ -1176,7 +1176,7 @@ export class MyContractInterface extends Interface { types_generic_enum: FunctionFragment; types_generic_struct: FunctionFragment; types_option: FunctionFragment; - types_option_geo: FunctionFragment; + types_option_struct: FunctionFragment; types_raw_slice: FunctionFragment; types_result: FunctionFragment; types_std_string: FunctionFragment; @@ -1225,7 +1225,7 @@ export class MyContract extends Contract { types_generic_enum: InvokeFunction<[x: GenericEnumInput], GenericEnumOutput>; types_generic_struct: InvokeFunction<[x: GenericStructWithEnumInput], GenericStructWithEnumOutput>; types_option: InvokeFunction<[x?: Option], Option>; - types_option_geo: InvokeFunction<[x?: Option], Option>; + types_option_struct: InvokeFunction<[x?: Option], Option>; types_raw_slice: InvokeFunction<[x: RawSlice], RawSlice>; types_result: InvokeFunction<[x: Result], Result>; types_std_string: InvokeFunction<[x: StdString], StdString>; diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 9fe1bd6ed7b..0178e03961f 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -886,7 +886,7 @@ describe('AbiCoder', () => { }); }); // @todo Investigate: returning the input as the output - describe.skip('types_option_geo', () => { + describe.skip('types_option_struct', () => { it('should encode/decode just fine', async () => { const input: Option = { a: true, @@ -894,7 +894,7 @@ describe('AbiCoder', () => { }; const expected: Option = undefined; - const { waitForResult } = await contract.functions.types_option_geo(input).call(); + const { waitForResult } = await contract.functions.types_option_struct(input).call(); const { value } = await waitForResult(); expect(value).toEqual(expected); diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index fcb4b0c97ef..134010493d4 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -92,7 +92,7 @@ abi MyContract { fn types_vector_option(x: Vec) -> Vec; fn types_option(x: Option) -> Option; - fn types_option_geo(x: Option) -> Option; + fn types_option_struct(x: Option) -> Option; fn types_identity_address(x: Identity) -> Identity; fn types_identity_contract_id(x: Identity) -> Identity; @@ -681,7 +681,7 @@ impl MyContract for Contract { const EXPECTED: Option = Option::None; return EXPECTED } - fn types_option_geo(x: Option) -> Option { + fn types_option_struct(x: Option) -> Option { const INPUT_STRUCT: StructSimple = StructSimple { a: true, b: 10, From 15453dc610ad4d8ffe37649b67b24a0d67d31538 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 11:30:16 -0300 Subject: [PATCH 13/37] type_struct_external --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 4 ++-- .../test/fixtures/forc-projects/abi-contract/src/main.sw | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 0178e03961f..86d0439881a 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -615,13 +615,13 @@ describe('AbiCoder', () => { expect(value).toEqual(expected); }); }); - describe('type_struct_external', () => { + describe('types_struct_external', () => { it('should encode/decode just fine', async () => { const input = { value: 10 }; // @ts-expect-error: Custom matcher 'toEqualBn' const expected = { value: expect.toEqualBn(20) }; - const { waitForResult } = await contract.functions.type_struct_external(input).call(); + const { waitForResult } = await contract.functions.types_struct_external(input).call(); const { value } = await waitForResult(); expect(value).toEqual(expected); diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index 134010493d4..fbed734e8c2 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -57,7 +57,7 @@ abi MyContract { fn types_struct_double_generic( x: StructGenericWithEnum, ) -> StructGenericWithEnum; - fn type_struct_external(x: ExternalStruct) -> ExternalStruct; + fn types_struct_external(x: ExternalStruct) -> ExternalStruct; fn types_struct_with_implicit_generics( x: StructWithImplicitGenerics, ) -> StructWithImplicitGenerics; @@ -399,7 +399,7 @@ impl MyContract for Contract { }; return EXPECTED } - fn type_struct_external(x: ExternalStruct) -> ExternalStruct { + fn types_struct_external(x: ExternalStruct) -> ExternalStruct { const INPUT: ExternalStruct = ExternalStruct { value: 10 }; assert(x == INPUT); From dd7108d5953808c36e5b9be4eea8b45e86cf35d8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 11:32:27 -0300 Subject: [PATCH 14/37] rename contract abi --- .../test/fixtures/forc-projects/abi-contract/src/main.sw | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index fbed734e8c2..a0de70d66c3 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -24,7 +24,7 @@ fn divide(numerator: u64, denominator: u64) -> Result { } } -abi MyContract { +abi AbiContract { fn configurables() -> Configurables; fn types_u8(x: u8) -> u8; @@ -136,7 +136,7 @@ configurable { }, } -impl MyContract for Contract { +impl AbiContract for Contract { fn configurables() -> Configurables { Configurables { U8_VALUE: U8_VALUE, From b2df95f61b2b5a2d9a4aeb7b9c217f511ce21140 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 11:51:13 -0300 Subject: [PATCH 15/37] fix test case types_enum_external --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 86d0439881a..e577b0a8dfe 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -785,10 +785,9 @@ describe('AbiCoder', () => { }); }); describe('types_enum_external', () => { - // @TODO revist this one, can't return B from this Sway function. it('should encode/decode just fine', async () => { const input = ExternalEnumInput.A; - const expected = ExternalEnumInput.A; // Should be B + const expected = ExternalEnumInput.B; const { waitForResult } = await contract.functions.types_enum_external(input).call(); From fba792fd0a986a0059ca40b5a4fec42dacfabab0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 12:17:49 -0300 Subject: [PATCH 16/37] implement Eqs related to types_struct_with_implicit_generics fn --- .../forc-projects/abi-contract/src/equality.sw | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw index 1da1a8a5854..2320fdb0886 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw @@ -392,3 +392,20 @@ impl Eq for StructWithGenericArray { } } +impl Eq for [b256; 3] { + fn eq(self, other: Self) -> bool { + self[0] == other[0] && self[1] == other[1] && self[2] == other[2] + } +} + +impl Eq for (b256, u8) { + fn eq(self, other: Self) -> bool { + self.0 == other.0 && self.1 == other.1 + } +} + +impl Eq for StructWithImplicitGenerics { + fn eq(self, other: Self) -> bool { + self.a == other.a && self.b == other.b + } +} From f0cd0ed3f8c6ce2e960b3e51d19719e9390bada6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 12:17:59 -0300 Subject: [PATCH 17/37] implementing types_struct_with_implicit_generics --- .../forc-projects/abi-contract/src/main.sw | 16 +++++++++++++++- 1 file changed, 15 insertions(+), 1 deletion(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index a0de70d66c3..a3af1ff54be 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -533,7 +533,21 @@ impl AbiContract for Contract { fn types_struct_with_implicit_generics( x: StructWithImplicitGenerics, ) -> StructWithImplicitGenerics { - x + const INPUT_B256: b256 = 0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa; + const INPUT: StructWithImplicitGenerics = StructWithImplicitGenerics { + a: [INPUT_B256, INPUT_B256, INPUT_B256], + b: (INPUT_B256, 10), + }; + + assert(x == INPUT); + + const EXPECTED_B256: b256 = 0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb; + const EXPECTED: StructWithImplicitGenerics = StructWithImplicitGenerics { + a: [EXPECTED_B256, EXPECTED_B256, EXPECTED_B256], + b: (EXPECTED_B256, 25), + }; + + EXPECTED } // @todo - unable to create this struct. fn types_struct_with_array(x: StructWithGenericArray) -> StructWithGenericArray { From 7687b3a7aa5965cfe33cb45d2f5c48e798ac9fb9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 12:18:12 -0300 Subject: [PATCH 18/37] add test for types_struct_with_implicit_generics --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 25 ++++++++++++++++++- 1 file changed, 24 insertions(+), 1 deletion(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index e577b0a8dfe..7a315457701 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -21,6 +21,7 @@ import type { StructWithNestedArrayInput, StructWithNestedTupleInput, StructSingleGenericInput, + StructWithImplicitGenericsInput, } from '../../test/typegen/contracts/AbiContract'; import type { Option, Result, Vec } from '../../test/typegen/contracts/common'; @@ -697,7 +698,29 @@ describe('AbiCoder', () => { // expect(value).toEqual(expected); }); }); - describe.todo('types_struct_with_implicit_generics', () => {}); + describe('types_struct_with_implicit_generics', () => { + it('should encode/decode just fine', async () => { + const INPUT_B256 = '0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'; + const INPUT: StructWithImplicitGenericsInput = { + a: [INPUT_B256, INPUT_B256, INPUT_B256], + b: [INPUT_B256, 10], + }; + + const EXPECTED_B256 = '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb'; + + const EXPECTED: StructWithImplicitGenericsInput = { + a: [EXPECTED_B256, EXPECTED_B256, EXPECTED_B256], + b: [EXPECTED_B256, 25], + }; + + const { waitForResult } = await contract.functions + .types_struct_with_implicit_generics(INPUT) + .call(); + + const { value } = await waitForResult(); + expect(value).toEqual(EXPECTED); + }); + }); // @todo Investigate: returning the input as the output describe.skip('types_struct_with_array', () => { From 6de8b9991d02cf219d917ef7d1c76241b9f2a463 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 12:20:29 -0300 Subject: [PATCH 19/37] unskip test --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 7a315457701..1754f59108d 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -722,8 +722,7 @@ describe('AbiCoder', () => { }); }); - // @todo Investigate: returning the input as the output - describe.skip('types_struct_with_array', () => { + describe('types_struct_with_array', () => { it('should encode/decode just fine', async () => { // Inputs const inputB256: string = From 767821bcdf0a480efe65507b35851c95c180685d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 12:33:12 -0300 Subject: [PATCH 20/37] use let instead of const to avoid run-time compile error --- .../test/fixtures/forc-projects/abi-contract/src/main.sw | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index a3af1ff54be..84cb9262520 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -696,12 +696,12 @@ impl AbiContract for Contract { return EXPECTED } fn types_option_struct(x: Option) -> Option { - const INPUT_STRUCT: StructSimple = StructSimple { + let input_struct: StructSimple = StructSimple { a: true, b: 10, }; - const INPUT: Option = Option::Some(INPUT_STRUCT); - assert(x == INPUT); + let input: Option = Option::Some(input_struct); + assert(x == input); const EXPECTED: Option = Option::None; return EXPECTED From 2dd95c5a87d47bc020ddc2ec7f7e20e7ee0a6a41 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 12:33:54 -0300 Subject: [PATCH 21/37] fixing some tests --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 1754f59108d..ffacf805936 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -894,8 +894,7 @@ describe('AbiCoder', () => { /** * Options */ - // @todo Investigate: returning the input as the output - describe.skip('types_option', () => { + describe('types_option', () => { it('should encode/decode just fine', async () => { const input: Option = 10; // Some const expected: Option = undefined; // None @@ -906,8 +905,7 @@ describe('AbiCoder', () => { expect(value).toEqual(expected); }); }); - // @todo Investigate: returning the input as the output - describe.skip('types_option_struct', () => { + describe('types_option_struct', () => { it('should encode/decode just fine', async () => { const input: Option = { a: true, From cbd8b13647b4f11944cb75447374151e663829f1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 12:37:50 -0300 Subject: [PATCH 22/37] fix multi_arg_b256_bool fn and its test --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 4 +--- .../test/fixtures/forc-projects/abi-contract/src/main.sw | 1 + 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index ffacf805936..80c39a71cbd 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -1173,9 +1173,7 @@ describe('AbiCoder', () => { expect(value).toStrictEqual(expected); }); }); - // @todo Investigate: returning the input as the output - describe.skip('multi_arg_b256_bool', () => { - // @todo investigate, this is returning the input as the output. + describe('multi_arg_b256_bool', () => { it('should encode/decode just fine', async () => { const inputX = '0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'; const inputY = true; diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index 84cb9262520..73a828bdbae 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -822,6 +822,7 @@ impl AbiContract for Contract { const INPUT_X: b256 = 0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa; const INPUT_Y: bool = true; assert_eq(x, INPUT_X); + assert_eq(y, INPUT_Y); const EXPECTED: (b256, bool) = (0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb, false); return EXPECTED From f8e8d7a048d76dbbc0017e15f0df89bacbffc882 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 12:39:22 -0300 Subject: [PATCH 23/37] unskip multi_arg_vector_vector --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 80c39a71cbd..0ae00165054 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -1188,8 +1188,7 @@ describe('AbiCoder', () => { expect(value).toStrictEqual(expected); }); }); - // @todo Investigate: returning the input as the output - describe.skip('multi_arg_vector_vector', () => { + describe('multi_arg_vector_vector', () => { it('should encode/decode just fine', async () => { const inputX = [1, 2, 3]; const inputY = [4, 5, 6]; From 18ffc3586c6defd88718399062f41d7f33de039e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 12:40:37 -0300 Subject: [PATCH 24/37] unskip multi_arg_vector_b256 --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 0ae00165054..58e947f41ad 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -1205,8 +1205,7 @@ describe('AbiCoder', () => { expect(value).toStrictEqual(expected); }); }); - // @todo Investigate: returning the input as the output - describe.skip('multi_arg_vector_b256', () => { + describe('multi_arg_vector_b256', () => { it('should encode/decode just fine', async () => { const inputX = [1, 2, 3]; const inputY = '0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'; From d9b30d222b03224ba34edd81c15494ac95f7b841 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 12:41:32 -0300 Subject: [PATCH 25/37] unskip multi_arg_struct_vector --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 58e947f41ad..2d7d4057c95 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -1222,8 +1222,7 @@ describe('AbiCoder', () => { expect(value).toStrictEqual(expected); }); }); - // @todo Investigate: returning the input as the output - describe.skip('multi_arg_struct_vector', () => { + describe('multi_arg_struct_vector', () => { it('should encode/decode just fine', async () => { const inputX = { a: true, b: 1 }; const inputY = [1, 2, 3]; From 23385f07b5364638c0b31c26f44c5e88bb5bc157 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 12:49:15 -0300 Subject: [PATCH 26/37] implement multi_arg_u64_struct --- .../forc-projects/abi-contract/src/main.sw | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index 73a828bdbae..19fed0ddc64 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -867,7 +867,20 @@ impl AbiContract for Contract { return EXPECTED } fn multi_arg_u64_struct(x: u64, y: StructSimple) -> (u64, StructSimple) { - (x, y) + const INPUT_X = 99u64; + let input_y = StructSimple { + a: true, + b: 51 + }; + assert(x == INPUT_X); + assert(y == input_y); + + const EXPECTED_X = 3u64; + let expected_y = StructSimple { + a: false, + b: 4 + }; + return (EXPECTED_X, expected_y); } fn multi_arg_str_str(x: str[5], y: str[5]) -> (str[5], str[5]) { (x, y) From f54a2a2c355f78bddbe654ecee94a406a7d1677d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 12:49:19 -0300 Subject: [PATCH 27/37] add test for multi_arg_u64_struct --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 17 +++++++++++++++-- 1 file changed, 15 insertions(+), 2 deletions(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 2d7d4057c95..41dc137a94d 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -1,4 +1,4 @@ -import { Contract, FuelError, Interface } from 'fuels'; +import { bn, Contract, FuelError, Interface } from 'fuels'; import type { AssetId, BigNumberish, EvmAddress, RawSlice, WalletUnlocked } from 'fuels'; import { expectToThrowFuelError, launchTestNode } from 'fuels/test-utils'; @@ -1236,7 +1236,20 @@ describe('AbiCoder', () => { expect(value).toStrictEqual(expected); }); }); - describe.skip('multi_arg_u64_struct'); + describe('multi_arg_u64_struct', () => { + it('should encode/decode just fine', async () => { + const inputX = bn(99); + const inputY: StructSimpleInput = { a: true, b: 51 }; + const expected = [bn(3), { a: false, b: 4 }]; + + const { waitForResult } = await contract.functions + .multi_arg_u64_struct(inputX, inputY) + .call(); + + const { value } = await waitForResult(); + expect(JSON.stringify(value)).toEqual(JSON.stringify(expected)); + }); + }); describe.skip('multi_arg_str_str'); describe.skip('multi_arg_u32_vector_vector'); describe.skip('multi_arg_complex'); From 84ae89cd513fb8a710ad954d934167a9b103e860 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 13:03:48 -0300 Subject: [PATCH 28/37] implement multi_arg_str_str --- .../fixtures/forc-projects/abi-contract/src/main.sw | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index 19fed0ddc64..26fb53cf2f4 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -883,7 +883,16 @@ impl AbiContract for Contract { return (EXPECTED_X, expected_y); } fn multi_arg_str_str(x: str[5], y: str[5]) -> (str[5], str[5]) { - (x, y) + let input_x: str = "Input"; + let input_y: str = "False"; + + assert_eq(from_str_array(x), input_x); + assert_eq(from_str_array(y), input_y); + + let EXPECTED_X: str[5] = __to_str_array("Fuuel"); + let EXPECTED_Y: str[5] = __to_str_array("Niice"); + + (EXPECTED_X, EXPECTED_Y) } fn multi_arg_u32_vector_vector(x: u32, y: Vec, z: Vec) -> (u32, Vec, Vec) { (x, y, z) From 978ad9b90cb46a47fa83f03075ec7f158c0f3d15 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 13:03:55 -0300 Subject: [PATCH 29/37] add test for multi_arg_str_str --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 14 +++++++++++++- 1 file changed, 13 insertions(+), 1 deletion(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 41dc137a94d..01a3c318d2f 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -1250,7 +1250,19 @@ describe('AbiCoder', () => { expect(JSON.stringify(value)).toEqual(JSON.stringify(expected)); }); }); - describe.skip('multi_arg_str_str'); + describe('multi_arg_str_str', () => { + it('should encode/decode just fine', async () => { + const inputX = 'Input'; + const inputY = 'False'; + + const expected = ['Fuuel', 'Niice']; + + const { waitForResult } = await contract.functions.multi_arg_str_str(inputX, inputY).call(); + + const { value } = await waitForResult(); + expect(value).toStrictEqual(expected); + }); + }); describe.skip('multi_arg_u32_vector_vector'); describe.skip('multi_arg_complex'); }); From 411ca8f5199407db5e21c48cb4577ba3ffda8f7d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 13:33:17 -0300 Subject: [PATCH 30/37] implement Eq for Vec --- .../forc-projects/abi-contract/src/equality.sw | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw index 2320fdb0886..6d829e5dd14 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw @@ -86,6 +86,22 @@ impl Eq for Vec { } } +impl Eq for Vec { + fn eq(self, other: Self) -> bool { + if self.len() != other.len() { + return false; + } + let mut i = 0; + while i < self.len() { + if self.get(i).unwrap() != other.get(i).unwrap() { + return false; + } + i += 1; + } + true + } +} + impl Eq for [Vec; 1] { fn eq(self, other: Self) -> bool { self[0] == other[0] From 50ab5a8753723734e1e84b870b217563a9009f2b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 13:33:25 -0300 Subject: [PATCH 31/37] implement fn multi_arg_u32_vector_vector --- .../forc-projects/abi-contract/src/main.sw | 29 ++++++++++++++++++- 1 file changed, 28 insertions(+), 1 deletion(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index 26fb53cf2f4..ebe524002ab 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -895,7 +895,34 @@ impl AbiContract for Contract { (EXPECTED_X, EXPECTED_Y) } fn multi_arg_u32_vector_vector(x: u32, y: Vec, z: Vec) -> (u32, Vec, Vec) { - (x, y, z) + const INPUT_X = 1u32; + + let mut input_y: Vec = Vec::new(); + input_y.push(10020); + input_y.push(1231231); + input_y.push(777657); + + let mut input_z: Vec = Vec::new(); + input_z.push(99); + input_z.push(101); + + assert(x == INPUT_X); + assert(y == input_y); + assert(z == input_z); + + const EXPECTED_X = 2u32; + + let mut expected_y: Vec = Vec::new(); + expected_y.push(7); + expected_y.push(8); + expected_y.push(9); + + let mut expected_z: Vec = Vec::new(); + expected_z.push(10); + expected_z.push(11); + expected_z.push(12); + + return (EXPECTED_X, expected_y, expected_z); } fn multi_arg_complex( x: StructDoubleGeneric<[b256; 3], u8>, From eef5009e16a33faa4a56feef88845fcbfae0b80f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 13:33:37 -0300 Subject: [PATCH 32/37] add test for multi_arg_u32_vector_vector --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 17 ++++++++++++++++- 1 file changed, 16 insertions(+), 1 deletion(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 01a3c318d2f..14cfe7382c3 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -1263,6 +1263,21 @@ describe('AbiCoder', () => { expect(value).toStrictEqual(expected); }); }); - describe.skip('multi_arg_u32_vector_vector'); + describe('multi_arg_u32_vector_vector', async () => { + it('should encode/decode just fine', async () => { + const inputX = 1; + const inputY = [bn(10020), bn(1231231), bn(777657)]; + const inputZ = [bn(99), bn(101)]; + + const expected = [2, [bn(7), bn(8), bn(9)], [bn(10), bn(11), bn(12)]]; + + const { waitForResult } = await contract.functions + .multi_arg_u32_vector_vector(inputX, inputY, inputZ) + .call(); + + const { value } = await waitForResult(); + expect(JSON.stringify(value)).toEqual(JSON.stringify(expected)); + }); + }); describe.skip('multi_arg_complex'); }); From 9b6913f339098401ce52aff870976c3ff0298fb0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Fri, 4 Oct 2024 13:34:06 -0300 Subject: [PATCH 33/37] remove wrong async --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 14cfe7382c3..0fe0239e540 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -1263,7 +1263,7 @@ describe('AbiCoder', () => { expect(value).toStrictEqual(expected); }); }); - describe('multi_arg_u32_vector_vector', async () => { + describe('multi_arg_u32_vector_vector', () => { it('should encode/decode just fine', async () => { const inputX = 1; const inputY = [bn(10020), bn(1231231), bn(777657)]; From 9ec757462575abd64177bbee99999ba7266c3d66 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Mon, 7 Oct 2024 10:00:17 -0300 Subject: [PATCH 34/37] adding multiple equalies func --- .../abi-contract/src/equality.sw | 74 ++++++++++++++----- 1 file changed, 56 insertions(+), 18 deletions(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw index 6d829e5dd14..d83723abf54 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/equality.sw @@ -30,19 +30,33 @@ impl Eq for StructSingleGeneric { } } +impl Eq for [b256; 3] { + fn eq(self, other: Self) -> bool { + self[0] == other[0] && self[1] == other[1] && self[2] == other[2] + } +} + +impl Eq for (b256, u8) { + fn eq(self, other: Self) -> bool { + self.0 == other.0 && self.1 == other.1 + } +} + impl Eq for str[1] { fn eq(self, other: Self) -> bool { - // @TODO work out how to equal str[1] - // self[0] == other[0] - true + from_str_array(self) == from_str_array(other) } } impl Eq for str[3] { fn eq(self, other: Self) -> bool { - // @TODO work out how to equal str[3] - // self[0] == other[0] && self[1] == other[1] && self[2] == other[2] - true + from_str_array(self) == from_str_array(other) + } +} + +impl Eq for str[5] { + fn eq(self, other: Self) -> bool { + from_str_array(self) == from_str_array(other) } } @@ -58,6 +72,12 @@ impl Eq for StructDoubleGeneric { } } +impl Eq for StructDoubleGeneric { + fn eq(self, other: Self) -> bool { + self.a == other.a + } +} + impl Eq for [StructDoubleGeneric; 3] { fn eq(self, other: Self) -> bool { self[0] == other[0] && self[1] == other[1] && self[2] == other[2] @@ -70,6 +90,30 @@ impl Eq for [StructDoubleGeneric, str[1]>; 2] { } } +impl Eq for [StructDoubleGeneric; 4] { + fn eq(self, other: Self) -> bool { + self[0] == other[0] && self[1] == other[1] && self[2] == other[2] && self[3] == other[3] + } +} + +impl Eq for StructSingleGeneric<[b256; 3]> { + fn eq(self, other: Self) -> bool { + self.a == other.a + } +} + +impl Eq for StructDoubleGeneric<[b256; 3], u8> { + fn eq(self, other: Self) -> bool { + self.a == other.a + } +} + +impl Eq for StructDoubleGeneric, u8> { + fn eq(self, other: Self) -> bool { + self.a == other.a && self.b == other.b + } +} + impl Eq for Vec { fn eq(self, other: Self) -> bool { if self.len() != other.len() { @@ -144,6 +188,12 @@ impl Eq for (bool, u64) { } } +impl Eq for (str[5], bool) { + fn eq(self, other: Self) -> bool { + self.0 == other.0 && self.1 == other.1 + } +} + impl Eq for StructSingleGeneric<(bool, u64)> { fn eq(self, other: Self) -> bool { self.a == other.a @@ -408,18 +458,6 @@ impl Eq for StructWithGenericArray { } } -impl Eq for [b256; 3] { - fn eq(self, other: Self) -> bool { - self[0] == other[0] && self[1] == other[1] && self[2] == other[2] - } -} - -impl Eq for (b256, u8) { - fn eq(self, other: Self) -> bool { - self.0 == other.0 && self.1 == other.1 - } -} - impl Eq for StructWithImplicitGenerics { fn eq(self, other: Self) -> bool { self.a == other.a && self.b == other.b From 0f32f153c2e96a3689e9f091dfe6c2ca0dc41a61 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Mon, 7 Oct 2024 10:00:48 -0300 Subject: [PATCH 35/37] fixing fn return type --- .../test/fixtures/forc-projects/abi-contract/src/main.sw | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index ebe524002ab..12ed99229da 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -121,6 +121,11 @@ abi AbiContract { y: [StructDoubleGeneric; 4], z: (str[5], bool), a: StructSimple, + ) -> ( + StructDoubleGeneric<[b256; 3], u8>, + [StructDoubleGeneric; 4], + (str[5], bool), + StructSimple, ); } From 6536404c0bba7b6e84ccd2ea2dd7f5875a8ab183 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Mon, 7 Oct 2024 10:01:30 -0300 Subject: [PATCH 36/37] implement multi_arg_complex --- .../forc-projects/abi-contract/src/main.sw | 53 ++++++++++++++++++- 1 file changed, 52 insertions(+), 1 deletion(-) diff --git a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw index 12ed99229da..86a215798d7 100644 --- a/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw +++ b/packages/fuel-gauge/test/fixtures/forc-projects/abi-contract/src/main.sw @@ -934,7 +934,58 @@ impl AbiContract for Contract { y: [StructDoubleGeneric; 4], z: (str[5], bool), a: StructSimple, + ) -> ( + StructDoubleGeneric<[b256; 3], u8>, + [StructDoubleGeneric; 4], + (str[5], bool), + StructSimple, ) { - () + + let input_x: StructDoubleGeneric<[b256; 3], u8> = StructDoubleGeneric { + a: [ + 0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa, + 0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb, + 0xcccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc, + ], + b: 10, + }; + + let input_y: [StructDoubleGeneric; 4] = [ + StructDoubleGeneric { a: 99u64, b: false }, + StructDoubleGeneric { a: 199u64, b: false }, + StructDoubleGeneric { a: 2000u64, b: false }, + StructDoubleGeneric { a: 31u64, b: true }, + ]; + + let input_z: (str[5], bool) = (__to_str_array("Input"), true); + + let input_a: StructSimple = StructSimple { a: true, b: 10 }; + + assert(x == input_x); + assert(y == input_y); + assert(z == input_z); + assert(a == input_a); + + let expected_x: StructDoubleGeneric<[b256; 3], u8> = StructDoubleGeneric { + a: [ + 0xdddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddd, + 0xeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee, + 0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff, + ], + b: 99, + }; + + let expected_y: [StructDoubleGeneric; 4] = [ + StructDoubleGeneric { a: 11u64, b: true }, + StructDoubleGeneric { a: 99u64, b: true }, + StructDoubleGeneric { a: 567u64, b: true }, + StructDoubleGeneric { a: 971u64, b: false }, + ]; + + let expected_z: (str[5], bool) = (__to_str_array("tupni"), false); + + let expected_a: StructSimple = StructSimple { a: false, b: 57 }; + + return (expected_x, expected_y, expected_z, expected_a); } } From db4a6c07a4fc430facb02043cb64d8a63a9c982f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Torres?= <30977845+Torres-ssf@users.noreply.github.com> Date: Mon, 7 Oct 2024 10:01:35 -0300 Subject: [PATCH 37/37] add test case --- packages/fuel-gauge/src/abi/abi-coder.test.ts | 66 ++++++++++++++++++- 1 file changed, 65 insertions(+), 1 deletion(-) diff --git a/packages/fuel-gauge/src/abi/abi-coder.test.ts b/packages/fuel-gauge/src/abi/abi-coder.test.ts index 0fe0239e540..1cd944c0e37 100644 --- a/packages/fuel-gauge/src/abi/abi-coder.test.ts +++ b/packages/fuel-gauge/src/abi/abi-coder.test.ts @@ -1279,5 +1279,69 @@ describe('AbiCoder', () => { expect(JSON.stringify(value)).toEqual(JSON.stringify(expected)); }); }); - describe.skip('multi_arg_complex'); + describe('multi_arg_complex', () => { + it('should encode/decode just fine', async () => { + const inputX: StructDoubleGenericInput<[string, string, string], number> = { + a: [ + '0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa', + '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb', + '0xcccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + ], + b: 10, + }; + + const inputY: [ + StructDoubleGenericInput, + StructDoubleGenericInput, + StructDoubleGenericInput, + StructDoubleGenericInput, + ] = [ + { a: bn(99), b: false }, + { a: bn(199), b: false }, + { a: bn(2000), b: false }, + { a: bn(31), b: true }, + ]; + + const inputZ: [string, boolean] = ['Input', true]; + + const inputA = { a: true, b: 10 }; + + const expectedX: StructDoubleGenericInput<[string, string, string], number> = { + a: [ + '0xdddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddd', + '0xeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee', + '0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff', + ], + b: 99, + }; + + const expectedY: [ + StructDoubleGenericInput, + StructDoubleGenericInput, + StructDoubleGenericInput, + StructDoubleGenericInput, + ] = [ + { a: bn(11), b: true }, + { a: bn(99), b: true }, + { a: bn(567), b: true }, + { a: bn(971), b: false }, + ]; + + const expectedZ: [string, boolean] = ['tupni', false]; + + const expectedA = { + a: false, + b: 57, + }; + + const { waitForResult } = await contract.functions + .multi_arg_complex(inputX, inputY, inputZ, inputA) + .call(); + + const { value } = await waitForResult(); + expect(JSON.stringify(value)).toEqual( + JSON.stringify([expectedX, expectedY, expectedZ, expectedA]) + ); + }); + }); });