From 8877b336b38f8686973b881893096d32e2580760 Mon Sep 17 00:00:00 2001 From: Nazar Hussain Date: Mon, 6 Dec 2021 21:39:49 +0100 Subject: [PATCH] Add contract static type support --- packages/web3-eth-abi/src/api/events_api.ts | 4 +- .../web3-eth-abi/src/api/functions_api.ts | 6 +- packages/web3-eth-abi/src/api/logs_api.ts | 12 +- .../web3-eth-abi/src/api/parameters_api.ts | 2 +- packages/web3-eth-abi/src/number_map_type.ts | 7 + packages/web3-eth-abi/src/types.ts | 206 ++++++++++++++---- packages/web3-eth-abi/src/utils.ts | 60 ++--- packages/web3-eth-contract/src/contract.ts | 54 +++-- packages/web3-eth-contract/src/encoding.ts | 21 +- packages/web3-eth-contract/src/types.ts | 80 ++++++- .../web3-eth-contract/test/fixtures/abi.json | 58 +++++ .../fixtures/types/web3-v1-contracts/abi.ts | 48 ++++ .../fixtures/types/web3-v1-contracts/types.ts | 64 ++++++ packages/web3-utils/src/types.ts | 41 ++++ 14 files changed, 554 insertions(+), 109 deletions(-) create mode 100644 packages/web3-eth-abi/src/number_map_type.ts create mode 100644 packages/web3-eth-contract/test/fixtures/abi.json create mode 100644 packages/web3-eth-contract/test/fixtures/types/web3-v1-contracts/abi.ts create mode 100644 packages/web3-eth-contract/test/fixtures/types/web3-v1-contracts/types.ts diff --git a/packages/web3-eth-abi/src/api/events_api.ts b/packages/web3-eth-abi/src/api/events_api.ts index 17f3d1d370c..4a89cb3aacd 100644 --- a/packages/web3-eth-abi/src/api/events_api.ts +++ b/packages/web3-eth-abi/src/api/events_api.ts @@ -1,12 +1,12 @@ import { AbiError } from 'web3-common'; import { sha3Raw } from 'web3-utils'; -import { JsonAbiEventFragment } from '../types'; +import { AbiEventFragment } from '../types'; import { jsonInterfaceMethodToString, isAbiEventFragment } from '../utils'; /** * Encodes the event name to its ABI signature, which are the sha3 hash of the event name including input types.. */ -export const encodeEventSignature = (functionName: string | JsonAbiEventFragment): string => { +export const encodeEventSignature = (functionName: string | AbiEventFragment): string => { if (typeof functionName !== 'string' && !isAbiEventFragment(functionName)) { throw new AbiError('Invalid parameter value in encodeEventSignature'); } diff --git a/packages/web3-eth-abi/src/api/functions_api.ts b/packages/web3-eth-abi/src/api/functions_api.ts index 143581d6c7f..ae832fb50fc 100644 --- a/packages/web3-eth-abi/src/api/functions_api.ts +++ b/packages/web3-eth-abi/src/api/functions_api.ts @@ -1,13 +1,13 @@ import { AbiError } from 'web3-common'; import { sha3Raw } from 'web3-utils'; import { isAbiFunctionFragment, jsonInterfaceMethodToString } from '../utils'; -import { JsonAbiFunctionFragment } from '../types'; +import { AbiFunctionFragment } from '../types'; import { encodeParameters } from './parameters_api'; /** * Encodes the function name to its ABI representation, which are the first 4 bytes of the sha3 of the function name including types. */ -export const encodeFunctionSignature = (functionName: string | JsonAbiFunctionFragment): string => { +export const encodeFunctionSignature = (functionName: string | AbiFunctionFragment): string => { if (typeof functionName !== 'string' && !isAbiFunctionFragment(functionName)) { throw new AbiError('Invalid parameter value in encodeFunctionSignature'); } @@ -27,7 +27,7 @@ export const encodeFunctionSignature = (functionName: string | JsonAbiFunctionFr * Encodes a function call from its json interface and parameters. */ export const encodeFunctionCall = ( - jsonInterface: JsonAbiFunctionFragment, + jsonInterface: AbiFunctionFragment, params: unknown[], ): string => { if (!isAbiFunctionFragment(jsonInterface)) { diff --git a/packages/web3-eth-abi/src/api/logs_api.ts b/packages/web3-eth-abi/src/api/logs_api.ts index ab2d705ba65..97ad08f468c 100644 --- a/packages/web3-eth-abi/src/api/logs_api.ts +++ b/packages/web3-eth-abi/src/api/logs_api.ts @@ -1,5 +1,5 @@ import { HexString } from 'web3-utils'; -import { JsonAbiParameter } from '../types'; +import { AbiParameter } from '../types'; import { decodeParameter, decodeParametersWith } from './parameters_api'; const STATIC_TYPES = ['bool', 'string', 'int', 'uint', 'address', 'fixed', 'ufixed']; @@ -8,7 +8,7 @@ const STATIC_TYPES = ['bool', 'string', 'int', 'uint', 'address', 'fixed', 'ufix * Decodes ABI-encoded log data and indexed topic data */ export const decodeLog = >( - inputs: Array, + inputs: Array, data: HexString, topics: string | string[], ) => { @@ -16,20 +16,20 @@ export const decodeLog = >( const clonedData = data ?? ''; - const notIndexedInputs: Array = []; - const indexedParams: Array = []; + const notIndexedInputs: Array = []; + const indexedParams: Array = []; let topicCount = 0; // TODO check for anonymous logs? for (const [i, input] of inputs.entries()) { if (input.indexed) { indexedParams[i] = STATIC_TYPES.some(s => input.type.startsWith(s)) - ? (decodeParameter(input.type, clonedTopics[topicCount]) as JsonAbiParameter) + ? (decodeParameter(input.type, clonedTopics[topicCount]) as AbiParameter) : clonedTopics[topicCount]; topicCount += 1; } else { - notIndexedInputs[i] = input as unknown as JsonAbiParameter; + notIndexedInputs[i] = input as unknown as AbiParameter; } } diff --git a/packages/web3-eth-abi/src/api/parameters_api.ts b/packages/web3-eth-abi/src/api/parameters_api.ts index a6a676a2efc..3994d9a23a4 100644 --- a/packages/web3-eth-abi/src/api/parameters_api.ts +++ b/packages/web3-eth-abi/src/api/parameters_api.ts @@ -8,7 +8,7 @@ import { formatParam, isAbiFragment, mapTypes, modifyParams } from '../utils'; /** * Should be used to encode list of params */ -export const encodeParameters = (abi: AbiInput[], params: unknown[]): string => { +export const encodeParameters = (abi: ReadonlyArray, params: unknown[]): string => { try { const modifiedTypes = mapTypes(Array.isArray(abi) ? abi : [abi]); const modifiedParams: Array = []; diff --git a/packages/web3-eth-abi/src/number_map_type.ts b/packages/web3-eth-abi/src/number_map_type.ts new file mode 100644 index 00000000000..c80f5eba30c --- /dev/null +++ b/packages/web3-eth-abi/src/number_map_type.ts @@ -0,0 +1,7 @@ +// TODO: Convert it to static file +type _SolidityIndexRange = 1 | 2 | 3 | 4 | 5; + +export type ConvertToNumber< + T extends string, + Range extends number = _SolidityIndexRange, +> = Range extends unknown ? (`${Range}` extends T ? Range : never) : never; diff --git a/packages/web3-eth-abi/src/types.ts b/packages/web3-eth-abi/src/types.ts index e1db1f567bb..76a8e0695b2 100644 --- a/packages/web3-eth-abi/src/types.ts +++ b/packages/web3-eth-abi/src/types.ts @@ -1,57 +1,187 @@ -// TODO: Adding reference of source definition/doc for these types -export interface JsonAbiParameter { - readonly name: string; - readonly type: string; - readonly baseType?: string; - readonly indexed?: boolean; - readonly components?: Array; - readonly arrayLength?: number; - readonly arrayChildren?: Array; -} +import { + Address, + Bytes, + Numbers, + ObjectValueToTuple, + ArrayToIndexObject, + FixedSizeArray, +} from 'web3-utils'; +import { ConvertToNumber } from './number_map_type'; -export interface JsonAbiStruct { +export interface AbiStruct { [key: string]: unknown; name?: string; type: string; } -export interface JsonAbiCoderStruct extends JsonAbiStruct { +export interface AbiCoderStruct extends AbiStruct { [key: string]: unknown; - components?: Array; + components?: Array; } +// https://docs.soliditylang.org/en/latest/abi-spec.html#json +export type AbiParameter = { + readonly name: string; + readonly type: string; + readonly baseType?: string; + readonly indexed?: boolean; + readonly components?: ReadonlyArray; + readonly arrayLength?: number; + readonly arrayChildren?: ReadonlyArray; +}; + type FragmentTypes = 'constructor' | 'event' | 'function'; -export interface JsonAbiBaseFragment { - name?: string; - type: FragmentTypes; - inputs?: Array; -} +export type AbiBaseFragment = { + readonly type: FragmentTypes; +}; -export interface JsonAbiConstructorFragment extends JsonAbiBaseFragment { - type: 'constructor'; - stateMutability: 'nonpayable' | 'payable'; -} +// https://docs.soliditylang.org/en/latest/abi-spec.html#json +export type AbiConstructorFragment = AbiBaseFragment & { + readonly type: 'constructor'; + readonly stateMutability: 'nonpayable' | 'payable'; + readonly inputs: ReadonlyArray; +}; -export interface JsonAbiFunctionFragment extends JsonAbiBaseFragment { - type: 'function'; - stateMutability: 'nonpayable' | 'payable' | 'pure' | 'view'; - outputs?: Array; +// https://docs.soliditylang.org/en/latest/abi-spec.html#json +export type AbiFunctionFragment = AbiBaseFragment & { + readonly name: string; + readonly type: 'function'; + readonly stateMutability: 'nonpayable' | 'payable' | 'pure' | 'view'; + readonly inputs: ReadonlyArray; + readonly outputs: ReadonlyArray; // legacy properties - constant?: boolean; // stateMutability == 'pure' or stateMutability == 'view' - payable?: boolean; // stateMutability == 'payable' -} + readonly constant?: boolean; // stateMutability == 'pure' or stateMutability == 'view' + readonly payable?: boolean; // stateMutability == 'payable' +}; -export interface JsonAbiEventFragment extends JsonAbiBaseFragment { - type: 'event'; - anonymous?: boolean; -} +// https://docs.soliditylang.org/en/latest/abi-spec.html#json +export type AbiEventFragment = AbiBaseFragment & { + readonly name: string; + readonly type: 'event'; + readonly inputs: ReadonlyArray; + readonly anonymous?: boolean; +}; // https://docs.soliditylang.org/en/latest/abi-spec.html#json -export type JsonAbiFragment = - | JsonAbiConstructorFragment - | JsonAbiFunctionFragment - | JsonAbiEventFragment; +export type AbiFragment = AbiConstructorFragment | AbiFunctionFragment | AbiEventFragment; + +export type ContractAbi = ReadonlyArray; + +export type AbiInput = string | AbiParameter | { readonly [key: string]: unknown }; + +export type FilterAbis = Abi extends Filter + ? Abi + : never; + +type _TypedArray = Size extends '' + ? Type[] + : FixedSizeArray>; + +export type PrimitiveAddressType = Type extends `address[${infer Size}]` + ? _TypedArray + : Type extends 'address' + ? Address + : never; + +export type PrimitiveStringType = Type extends `string${string}[${infer Size}]` + ? _TypedArray + : Type extends 'string' | `string${string}` + ? string + : never; + +export type PrimitiveBooleanType = Type extends `bool[${infer Size}]` + ? _TypedArray + : Type extends 'bool' + ? boolean + : never; + +export type PrimitiveIntegerType = Type extends `uint${string}[${infer Size}]` + ? _TypedArray + : Type extends 'uint' | 'int' | `int${string}` | `uint${string}` + ? Numbers + : never; + +export type PrimitiveBytesType = Type extends `bytes${string}[${infer Size}]` + ? _TypedArray + : Type extends 'bytes' | `bytes${string}` + ? Bytes + : never; + +export type PrimitiveTupleType< + Type extends string, + Components extends ReadonlyArray | undefined = [], +> = Components extends ReadonlyArray + ? Type extends 'tuple' + ? { + // eslint-disable-next-line no-use-before-define + [Param in Components[number] as Param['name']]: MatchPrimitiveType< + Param['type'], + Param['components'] + >; + } + : Type extends `tuple[${infer Size}]` + ? _TypedArray< + { + // eslint-disable-next-line no-use-before-define + [Param in Components[number] as Param['name']]: MatchPrimitiveType< + Param['type'], + Param['components'] + >; + }, + Size + > + : never + : never; + +export type MatchPrimitiveType< + Type extends string, + Components extends ReadonlyArray | undefined, +> = + | PrimitiveAddressType + | PrimitiveStringType + | PrimitiveBooleanType + | PrimitiveIntegerType + | PrimitiveBytesType + | PrimitiveTupleType + | Type; + +// Only intended to use locally so why not exported +// TODO: Inspect Record not working constraint +// eslint-disable-next-line @typescript-eslint/no-explicit-any +type _ExtractParameterType> = { + [K in keyof T]: MatchPrimitiveType; +}; + +export type ContractMethodOutputParameters> = + ObjectValueToTuple<_ExtractParameterType>>; + +export type ContractMethodInputParameters> = { + [Param in Params[number] as Param['name']]: MatchPrimitiveType< + Param['type'], + Param['components'] + >; +}; + +export type ContractConstructor = { + [Abi in FilterAbis as 'constructor']: { + readonly Abi: Abi; + readonly Inputs: ContractMethodInputParameters; + }; +}['constructor']; + +export type ContractMethods = { + [Abi in FilterAbis as Abi['name']]: { + readonly Abi: Abi; + readonly Inputs: ContractMethodInputParameters; + readonly Outputs: ContractMethodOutputParameters; + }; +}; -export type AbiInput = string | JsonAbiParameter | { readonly [key: string]: unknown }; +export type ContractEvents = { + [Abi in FilterAbis as Abi['name']]: { + readonly Abi: Abi; + readonly Inputs: ContractMethodInputParameters; + }; +}; diff --git a/packages/web3-eth-abi/src/utils.ts b/packages/web3-eth-abi/src/utils.ts index 4eac49aee2e..20cce30a41a 100644 --- a/packages/web3-eth-abi/src/utils.ts +++ b/packages/web3-eth-abi/src/utils.ts @@ -4,37 +4,37 @@ import { leftPad, rightPad, toHex } from 'web3-utils'; import ethersAbiCoder from './ethers_abi_coder'; import { AbiInput, - JsonAbiCoderStruct, - JsonAbiFragment, - JsonAbiParameter, - JsonAbiStruct, - JsonAbiEventFragment, - JsonAbiFunctionFragment, - JsonAbiConstructorFragment, + AbiCoderStruct, + AbiFragment, + AbiParameter, + AbiStruct, + AbiEventFragment, + AbiFunctionFragment, + AbiConstructorFragment, } from './types'; -export const isAbiFragment = (item: unknown): item is JsonAbiFragment => +export const isAbiFragment = (item: unknown): item is AbiFragment => item !== undefined && item !== null && typeof item === 'object' && (item as { type: string }).type !== undefined && ['function', 'event', 'constructor'].includes((item as { type: string }).type); -export const isAbiEventFragment = (item: unknown): item is JsonAbiEventFragment => +export const isAbiEventFragment = (item: unknown): item is AbiEventFragment => item !== undefined && item !== null && typeof item === 'object' && (item as { type: string }).type !== undefined && (item as { type: string }).type === 'event'; -export const isAbiFunctionFragment = (item: unknown): item is JsonAbiFunctionFragment => +export const isAbiFunctionFragment = (item: unknown): item is AbiFunctionFragment => item !== undefined && item !== null && typeof item === 'object' && (item as { type: string }).type !== undefined && (item as { type: string }).type === 'function'; -export const isAbiConstructorFragment = (item: unknown): item is JsonAbiConstructorFragment => +export const isAbiConstructorFragment = (item: unknown): item is AbiConstructorFragment => item !== undefined && item !== null && typeof item === 'object' && @@ -45,8 +45,8 @@ export const isAbiConstructorFragment = (item: unknown): item is JsonAbiConstruc * Check if type is simplified struct format */ export const isSimplifiedStructFormat = ( - type: string | Partial, -): type is Omit => + type: string | Partial, +): type is Omit => typeof type === 'object' && typeof (type as { components: unknown }).components === 'undefined' && typeof (type as { name: unknown }).name === 'undefined'; @@ -54,7 +54,7 @@ export const isSimplifiedStructFormat = ( /** * Maps the correct tuple type and name when the simplified format in encode/decodeParameter is used */ -export const mapStructNameAndType = (structName: string): JsonAbiStruct => +export const mapStructNameAndType = (structName: string): AbiStruct => structName.includes('[]') ? { type: 'tuple[]', name: structName.slice(0, -2) } : { type: 'tuple', name: structName }; @@ -62,8 +62,8 @@ export const mapStructNameAndType = (structName: string): JsonAbiStruct => /** * Maps the simplified format in to the expected format of the ABICoder */ -export const mapStructToCoderFormat = (struct: JsonAbiStruct): Array => { - const components: Array = []; +export const mapStructToCoderFormat = (struct: AbiStruct): Array => { + const components: Array = []; for (const key of Object.keys(struct)) { const item = struct[key]; @@ -71,7 +71,7 @@ export const mapStructToCoderFormat = (struct: JsonAbiStruct): Array> => { - const mappedTypes: Array> = []; +): Array> => { + const mappedTypes: Array> = []; for (const type of types) { let modifiedType = type; @@ -112,8 +112,8 @@ export const mapTypes = ( mappedTypes.push({ ...mapStructNameAndType(structName), components: mapStructToCoderFormat( - modifiedType[structName] as unknown as JsonAbiStruct, - ) as unknown as JsonAbiParameter[], + modifiedType[structName] as unknown as AbiStruct, + ) as unknown as AbiParameter[], }); } else { mappedTypes.push(modifiedType); @@ -217,7 +217,10 @@ export const modifyParams = ( * used to flatten json abi inputs/outputs into an array of type-representing-strings */ -export const flattenTypes = (includeTuple: boolean, puts: JsonAbiParameter[]): string[] => { +export const flattenTypes = ( + includeTuple: boolean, + puts: ReadonlyArray, +): string[] => { const types: string[] = []; puts.forEach(param => { @@ -250,10 +253,15 @@ export const flattenTypes = (includeTuple: boolean, puts: JsonAbiParameter[]): s * Should be used to create full function/event name from json abi * returns a string */ -export const jsonInterfaceMethodToString = (json: JsonAbiFragment): string => { - if (json.name?.includes('(')) { - return json.name; +export const jsonInterfaceMethodToString = (json: AbiFragment): string => { + if (isAbiEventFragment(json) || isAbiFunctionFragment(json)) { + if (json.name?.includes('(')) { + return json.name; + } + + return `${json.name ?? ''}(${flattenTypes(false, json.inputs ?? []).join(',')})`; } - return `${json.name ?? ''}(${flattenTypes(false, json.inputs ?? []).join(',')})`; + // Constructor fragment + return `(${flattenTypes(false, json.inputs ?? []).join(',')})`; }; diff --git a/packages/web3-eth-contract/src/contract.ts b/packages/web3-eth-contract/src/contract.ts index 0ca38714fa3..f17c13dd1d7 100644 --- a/packages/web3-eth-contract/src/contract.ts +++ b/packages/web3-eth-contract/src/contract.ts @@ -1,25 +1,37 @@ -import { EthExecutionAPI, inputAddressFormatter } from 'web3-common'; +// eslint-disable-next-line max-classes-per-file +import { EthExecutionAPI, inputAddressFormatter, Web3EventEmitter } from 'web3-common'; import { Web3Context } from 'web3-core'; import { + AbiEventFragment, + AbiFunctionFragment, + ContractAbi, + ContractEvents, + ContractMethods, encodeEventSignature, encodeFunctionSignature, isAbiEventFragment, isAbiFunctionFragment, - JsonAbiEventFragment, - JsonAbiFragment, - JsonAbiFunctionFragment, jsonInterfaceMethodToString, } from 'web3-eth-abi'; import { Address, toChecksumAddress } from 'web3-utils'; -import { ContractInitOptions, ContractOptions } from './types'; +import { + ContractEventEmitterInterface, + ContractEventsInterface, + ContractInitOptions, + ContractMethodsInterface, + ContractOptions, +} from './types'; type ContractBoundFunction = string; type ContractBoundEvent = string; -export class Contract extends Web3Context { +export class Contract + extends Web3Context + implements Web3EventEmitter>> +{ public readonly options: ContractOptions; - private _jsonInterface!: JsonAbiFragment[]; + private _jsonInterface!: Abi; private _address?: Address | null; private _functions: Record< string, @@ -31,13 +43,17 @@ export class Contract extends Web3Context { > = {}; private _events: Record = {}; - public constructor( - jsonInterface: JsonAbiFragment[], - address?: Address, - options?: ContractInitOptions, - ) { + public readonly methods: ContractMethodsInterface>; + public readonly events: ContractEventsInterface>; + + public constructor(jsonInterface: Abi, address?: Address, options?: ContractInitOptions) { super(options?.provider ?? ''); + // TODO: Implement these methods + this.methods = {} as ContractMethodsInterface>; + // TODO: Implement these events + this.events = {} as ContractEventsInterface>; + this._parseAndSetAddress(address); this._parseAndSetJsonInterface(jsonInterface); @@ -56,7 +72,7 @@ export class Contract extends Web3Context { }); Object.defineProperty(this.options, 'jsonInterface', { - set: (value: JsonAbiFragment[]) => this._parseAndSetJsonInterface(value), + set: (value: ContractAbi) => this._parseAndSetJsonInterface(value), get: () => this._jsonInterface, }); } @@ -65,10 +81,10 @@ export class Contract extends Web3Context { this._address = value ? toChecksumAddress(inputAddressFormatter(value)) : null; } - private _parseAndSetJsonInterface(abis: JsonAbiFragment[]) { + private _parseAndSetJsonInterface(abis: ContractAbi) { this._functions = {}; this._events = {}; - const result: JsonAbiFragment[] = []; + let result: ContractAbi = []; for (const a of abis) { const abi = { @@ -110,22 +126,22 @@ export class Contract extends Web3Context { this._events[signature] = event; } - result.push(abi); + result = [...result, abi]; } - this._jsonInterface = abis; + this._jsonInterface = [...result] as unknown as Abi; } // eslint-disable-next-line class-methods-use-this private _createContractFunction( name: string, - _abi: JsonAbiFunctionFragment, + _abi: AbiFunctionFragment, ): ContractBoundFunction { return name; } // eslint-disable-next-line class-methods-use-this - private _createContractEvent(name: string, _abi: JsonAbiEventFragment): ContractBoundEvent { + private _createContractEvent(name: string, _abi: AbiEventFragment): ContractBoundEvent { return name; } } diff --git a/packages/web3-eth-contract/src/encoding.ts b/packages/web3-eth-contract/src/encoding.ts index e84f8f65bbb..2ddce254213 100644 --- a/packages/web3-eth-contract/src/encoding.ts +++ b/packages/web3-eth-contract/src/encoding.ts @@ -5,16 +5,17 @@ import { encodeParameter, encodeParameters, isAbiConstructorFragment, - JsonAbiConstructorFragment, - JsonAbiEventFragment, - JsonAbiFunctionFragment, + AbiConstructorFragment, + AbiEventFragment, + AbiFunctionFragment, + isAbiFunctionFragment, } from 'web3-eth-abi'; import { HexString, Uint } from 'web3-utils'; import { ContractOptions } from './types'; export const encodeEventABI = ( { address }: ContractOptions, - event: JsonAbiEventFragment & { signature: string }, + event: AbiEventFragment & { signature: string }, options?: { fromBlock?: Uint; toBlock?: Uint; @@ -84,7 +85,7 @@ export const encodeEventABI = ( }; export const decodeEventABI = ( - event: JsonAbiEventFragment & { signature: string }, + event: AbiEventFragment & { signature: string }, data: LogsInput, ) => { let modifiedEvent = { ...event }; @@ -121,7 +122,7 @@ export const decodeEventABI = ( return { ...result, - returnValue: decodeLog(event.inputs ?? [], data.data, argTopics), + returnValue: decodeLog([...event.inputs], data.data, argTopics), event: event.name, signature: event.anonymous || !data.topics[0] ? null : data.topics[0], raw: { @@ -132,11 +133,11 @@ export const decodeEventABI = ( }; export const encodeMethodABI = ( - abi: (JsonAbiFunctionFragment | JsonAbiConstructorFragment) & { signature?: string }, + abi: (AbiFunctionFragment | AbiConstructorFragment) & { signature?: string }, args: unknown[], deployData?: HexString, ) => { - if (abi.signature && abi.name !== abi.signature) { + if (isAbiFunctionFragment(abi) && abi.signature && abi.name !== abi.signature) { throw new Error('The ABI can not match with given signature'); } @@ -176,7 +177,7 @@ export const encodeMethodABI = ( }; export const decodeMethodReturn = ( - abi: (JsonAbiFunctionFragment | JsonAbiConstructorFragment) & { signature?: string }, + abi: (AbiFunctionFragment | AbiConstructorFragment) & { signature?: string }, returnValues?: HexString, ) => { if (!returnValues) { @@ -184,7 +185,7 @@ export const decodeMethodReturn = ( } const value = returnValues.length >= 2 ? returnValues.slice(2) : returnValues; - const result = decodeParameters(abi.inputs ?? [], value); + const result = decodeParameters([...abi.inputs], value); if (result.__length__ === 1) { return result[0]; diff --git a/packages/web3-eth-contract/src/types.ts b/packages/web3-eth-contract/src/types.ts index 57a2def6b84..d13752e0039 100644 --- a/packages/web3-eth-contract/src/types.ts +++ b/packages/web3-eth-contract/src/types.ts @@ -1,14 +1,17 @@ -import { Uint, Address, Bytes } from 'web3-utils'; +import { EventEmitter } from 'events'; +import { BlockNumberOrTag, EthExecutionAPI } from 'web3-common'; import { SupportedProviders } from 'web3-core'; -import { EthExecutionAPI } from 'web3-common'; -import { JsonAbiFragment } from 'web3-eth-abi'; +import { ContractAbi, ContractEvents, ContractMethods } from 'web3-eth-abi'; +import { Address, Bytes, Numbers, Uint, HexString } from 'web3-utils'; + +type Callback = (error: Error, result: T) => void; export interface ContractOptions { readonly gas: Uint | null; readonly gasPrice: Uint | null; readonly from?: Address; readonly data?: Bytes; - jsonInterface: JsonAbiFragment[]; + jsonInterface: ContractAbi; address?: Address | null; } @@ -20,3 +23,72 @@ export interface ContractInitOptions { readonly gasLimit: Uint; readonly provider: SupportedProviders | string; } + +// TODO: Add correct type +// eslint-disable-next-line @typescript-eslint/no-empty-interface +export interface TransactionReceipt {} + +// TODO: Add correct type +// eslint-disable-next-line +export interface PromiEvent { + T: T; +} + +export interface NonPayableTx { + nonce?: Numbers; + chainId?: Numbers; + from?: Address; + to?: Address; + data?: HexString; + gas?: Numbers; + maxPriorityFeePerGas?: Numbers; + maxFeePerGas?: Numbers; + gasPrice?: Numbers; +} + +export interface PayableTx extends NonPayableTx { + value?: Numbers; +} + +export interface NonPayableTransactionObject { + arguments: T; + call(tx?: NonPayableTx, block?: BlockNumberOrTag): Promise; + send(tx?: NonPayableTx): PromiEvent; + estimateGas(tx?: NonPayableTx): Promise; + encodeABI(): string; +} + +export interface PayableTransactionObject { + arguments: T; + call(tx?: PayableTx, block?: BlockNumberOrTag): Promise; + send(tx?: PayableTx): PromiEvent; + estimateGas(tx?: PayableTx): Promise; + encodeABI(): string; +} + +export type ContractMethodsInterface< + Abi extends ContractAbi, + Methods extends ContractMethods, +> = { + [Name in keyof Methods]: { + call: ( + args: Methods[Name]['Inputs'], + // TODO: Debug why the `Abi` object is not accessible. + // eslint-disable-next-line @typescript-eslint/ban-ts-comment + // @ts-expect-error + ) => Methods[Name]['Abi']['stateMutability'] extends 'payable' | 'pure' + ? PayableTransactionObject + : NonPayableTransactionObject; + }; +}; + +export type ContractEventsInterface> = { + [Name in keyof Events]: (cb?: Callback) => EventEmitter; +}; + +export type ContractEventEmitterInterface< + Abi extends ContractAbi, + Events extends ContractEvents, +> = { + [Name in keyof Events]: Events[Name]['Inputs']; +}; diff --git a/packages/web3-eth-contract/test/fixtures/abi.json b/packages/web3-eth-contract/test/fixtures/abi.json new file mode 100644 index 00000000000..ff72dfe974e --- /dev/null +++ b/packages/web3-eth-contract/test/fixtures/abi.json @@ -0,0 +1,58 @@ +[ + { + "inputs": [ + { + "internalType": "uint256", + "name": "a", + "type": "uint256" + } + ], + "stateMutability": "nonpayable", + "type": "constructor" + }, + { + "anonymous": false, + "inputs": [ + { + "indexed": true, + "internalType": "uint256", + "name": "oldVal", + "type": "uint256" + }, + { + "indexed": true, + "internalType": "uint256", + "name": "newVal", + "type": "uint256" + } + ], + "name": "ValueSet", + "type": "event" + }, + { + "inputs": [], + "name": "retrieve", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "uint256", + "name": "num", + "type": "uint256" + } + ], + "name": "store", + "outputs": [], + "stateMutability": "nonpayable", + "type": "function" + } +] diff --git a/packages/web3-eth-contract/test/fixtures/types/web3-v1-contracts/abi.ts b/packages/web3-eth-contract/test/fixtures/types/web3-v1-contracts/abi.ts new file mode 100644 index 00000000000..3c737205754 --- /dev/null +++ b/packages/web3-eth-contract/test/fixtures/types/web3-v1-contracts/abi.ts @@ -0,0 +1,48 @@ +/* Autogenerated file. Do not edit manually. */ +/* tslint:disable */ +/* eslint-disable */ + +import BN from 'bn.js'; +import { ContractOptions } from 'web3-eth-contract'; +import { EventLog } from 'web3-core'; +import { EventEmitter } from 'events'; +import { + Callback, + PayableTransactionObject, + None, + BlockType, + ContractEventLog, + BaseContract, +} from './types'; + +export interface EventOptions { + filter?: object; + fromBlock?: BlockType; + topics?: string[]; +} + +export type ValueSet = ContractEventLog<{ + oldVal: string; + newVal: string; + 0: string; + 1: string; +}>; + +export interface Abi extends BaseContract { + constructor(jsonInterface: any[], address?: string, options?: ContractOptions): Abi; + clone(): Abi; + methods: { + retrieve(): NonPayableTransactionObject; + + store(num: number | string | BN): NonPayableTransactionObject; + }; + events: { + ValueSet(cb?: Callback): EventEmitter; + ValueSet(options?: EventOptions, cb?: Callback): EventEmitter; + + allEvents(options?: EventOptions, cb?: Callback): EventEmitter; + }; + + once(event: 'ValueSet', cb: Callback): void; + once(event: 'ValueSet', options: EventOptions, cb: Callback): void; +} diff --git a/packages/web3-eth-contract/test/fixtures/types/web3-v1-contracts/types.ts b/packages/web3-eth-contract/test/fixtures/types/web3-v1-contracts/types.ts new file mode 100644 index 00000000000..6eb50763e06 --- /dev/null +++ b/packages/web3-eth-contract/test/fixtures/types/web3-v1-contracts/types.ts @@ -0,0 +1,64 @@ +/* Autogenerated file. Do not edit manually. */ +/* tslint:disable */ +/* eslint-disable */ +import BN from 'bn.js'; +import { EventEmitter } from 'events'; +import { EventLog, PromiEvent, TransactionReceipt } from 'web3-core/types'; +import { Contract } from 'web3-eth-contract'; + +export interface EstimateGasOptions { + from?: string; + gas?: number; + value?: number | string | BN; +} + +export interface EventOptions { + filter?: object; + fromBlock?: BlockType; + topics?: string[]; +} + +export type Callback = (error: Error, result: T) => void; +export interface ContractEventLog extends EventLog { + returnValues: T; +} +export interface ContractEventEmitter extends EventEmitter { + on(event: 'connected', listener: (subscriptionId: string) => void): this; + on(event: 'data' | 'changed', listener: (event: ContractEventLog) => void): this; + on(event: 'error', listener: (error: Error) => void): this; +} + +export interface NonPayableTx { + nonce?: string | number | BN; + chainId?: string | number | BN; + from?: string; + to?: string; + data?: string; + gas?: string | number | BN; + maxPriorityFeePerGas?: string | number | BN; + maxFeePerGas?: string | number | BN; + gasPrice?: string | number | BN; +} + +export interface PayableTx extends NonPayableTx { + value?: string | number | BN; +} + +export interface NonPayableTransactionObject { + arguments: any[]; + call(tx?: NonPayableTx, block?: BlockType): Promise; + send(tx?: NonPayableTx): PromiEvent; + estimateGas(tx?: NonPayableTx): Promise; + encodeABI(): string; +} + +export interface PayableTransactionObject { + arguments: any[]; + call(tx?: PayableTx, block?: BlockType): Promise; + send(tx?: PayableTx): PromiEvent; + estimateGas(tx?: PayableTx): Promise; + encodeABI(): string; +} + +export type BlockType = 'latest' | 'pending' | 'genesis' | 'earliest' | number | BN; +export type BaseContract = Omit; diff --git a/packages/web3-utils/src/types.ts b/packages/web3-utils/src/types.ts index 78aaf1318e7..360bb6ccfd1 100644 --- a/packages/web3-utils/src/types.ts +++ b/packages/web3-utils/src/types.ts @@ -28,3 +28,44 @@ export type TypedObjectAbbreviated = { t: string; v: EncodingTypes; }; + +export type IndexKeysForArray = Exclude; + +export type IntersectionOfUnion = (U extends unknown ? (k: U) => void : never) extends ( + k: infer I, +) => void + ? I + : never; + +export type LastOf = IntersectionOfUnion< + T extends unknown ? () => T : never +> extends () => infer R + ? R + : never; + +export type Push = [...T, V]; + +export type UnionToTuple, N = [T] extends [never] ? true : false> = true extends N + ? [] + : Push>, L>; + +export type ObjectValueToTuple< + T, + KS extends unknown[] = UnionToTuple, + R extends unknown[] = [], +> = KS extends [infer K, ...infer KT] ? ObjectValueToTuple : R; + +export type ArrayToIndexObject> = { + [K in IndexKeysForArray]: T[K]; +}; + +type _Grow> = ((x: T, ...xs: A) => void) extends (...a: infer X) => void + ? X + : never; + +export type GrowToSize, N extends number> = { + 0: A; + 1: GrowToSize, N>; +}[A['length'] extends N ? 0 : 1]; + +export type FixedSizeArray = GrowToSize;