-
-
Notifications
You must be signed in to change notification settings - Fork 932
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: Improve return type for toSignature #2928
base: main
Are you sure you want to change the base?
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
--- | ||
"viem": minor | ||
--- | ||
|
||
Improved return type for toSignature |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,139 @@ | ||
import { expectTypeOf, test } from 'vitest' | ||
import { normalizeSignature } from './normalizeSignature.js' | ||
|
||
/** | ||
* Cases where we should be able to get proper types for this, but where the feature just isn't implemented in Viem | ||
*/ | ||
type TodoType = string | ||
|
||
test('foo()', () => { | ||
expectTypeOf(normalizeSignature('event foo()')).toMatchTypeOf<'foo()'>() | ||
}) | ||
|
||
test('bar(uint foo)', () => { | ||
expectTypeOf(normalizeSignature('bar(uint foo)')).toMatchTypeOf<TodoType>() | ||
}) | ||
|
||
test('processAccount(uint256 , address )', () => { | ||
expectTypeOf( | ||
normalizeSignature('processAccount(uint256 , address )'), | ||
).toMatchTypeOf<TodoType>() | ||
}) | ||
|
||
test('function bar()', () => { | ||
expectTypeOf(normalizeSignature('function bar()')).toMatchTypeOf<'bar()'>() | ||
}) | ||
|
||
test('function bar()', () => { | ||
expectTypeOf(normalizeSignature('function bar( )')).toMatchTypeOf<TodoType>() | ||
}) | ||
|
||
test('event Bar()', () => { | ||
expectTypeOf(normalizeSignature('event Bar()')).toMatchTypeOf<'Bar()'>() | ||
}) | ||
|
||
test('function bar(uint foo)', () => { | ||
expectTypeOf( | ||
normalizeSignature('function bar(uint foo)'), | ||
).toMatchTypeOf<'bar(uint256)'>() | ||
}) | ||
|
||
test('function bar(uint foo, address baz)', () => { | ||
expectTypeOf( | ||
normalizeSignature('function bar(uint foo, address baz)'), | ||
).toMatchTypeOf<'bar(uint256,address)'>() | ||
}) | ||
|
||
test('event Barry(uint foo)', () => { | ||
expectTypeOf( | ||
normalizeSignature('event Barry(uint foo)'), | ||
).toMatchTypeOf<'Barry(uint256)'>() | ||
}) | ||
|
||
test('Barry(uint indexed foo)', () => { | ||
expectTypeOf( | ||
normalizeSignature('Barry(uint indexed foo)'), | ||
).toMatchTypeOf<TodoType>() | ||
}) | ||
|
||
test('event Barry(uint indexed foo)', () => { | ||
expectTypeOf( | ||
normalizeSignature('event Barry(uint indexed foo)'), | ||
).toMatchTypeOf<'Barry(uint256)'>() | ||
}) | ||
|
||
test('function _compound(uint256 a, uint256 b, uint256 c)', () => { | ||
expectTypeOf( | ||
normalizeSignature('function _compound(uint256 a, uint256 b, uint256 c)'), | ||
).toMatchTypeOf<'_compound(uint256,uint256,uint256)'>() | ||
}) | ||
|
||
test('bar(uint foo, (uint baz))', () => { | ||
expectTypeOf( | ||
normalizeSignature('bar(uint foo, (uint baz))'), | ||
).toMatchTypeOf<TodoType>() | ||
}) | ||
|
||
test('bar(uint foo, (uint baz, bool x))', () => { | ||
expectTypeOf( | ||
normalizeSignature('bar(uint foo, (uint baz, bool x))'), | ||
).toMatchTypeOf<TodoType>() | ||
}) | ||
|
||
test('bar(uint foo, (uint baz, bool x) foo)', () => { | ||
expectTypeOf( | ||
normalizeSignature('bar(uint foo, (uint baz, bool x) foo)'), | ||
).toMatchTypeOf<TodoType>() | ||
}) | ||
|
||
test('bar(uint[] foo, (uint baz, bool x))', () => { | ||
expectTypeOf( | ||
normalizeSignature('bar(uint[] foo, (uint baz, bool x))'), | ||
).toMatchTypeOf<TodoType>() | ||
}) | ||
|
||
test('bar(uint[] foo, (uint baz, bool x)[])', () => { | ||
expectTypeOf( | ||
normalizeSignature('bar(uint[] foo, (uint baz, bool x)[])'), | ||
).toMatchTypeOf<TodoType>() | ||
}) | ||
|
||
test('foo(uint bar)', () => { | ||
expectTypeOf( | ||
normalizeSignature('foo(uint bar) payable returns (uint)'), | ||
).toMatchTypeOf<TodoType>() | ||
}) | ||
|
||
test('function submitBlocksWithCallbacks(bool isDataCompressed, bytes calldata data, ((uint16,(uint16,uint16,uint16,bytes)[])[], address[]) calldata config)', () => { | ||
expectTypeOf( | ||
normalizeSignature( | ||
'function submitBlocksWithCallbacks(bool isDataCompressed, bytes calldata data, ((uint16,(uint16,uint16,uint16,bytes)[])[], address[]) calldata config)', | ||
), | ||
).toMatchTypeOf<'submitBlocksWithCallbacks(bool,bytes,((uint16,(uint16,uint16,uint16,bytes)[])[],address[]))'>( | ||
'submitBlocksWithCallbacks(bool,bytes,((uint16,(uint16,uint16,uint16,bytes)[])[],address[]))', | ||
) | ||
}) | ||
|
||
test('function createEdition(string name, string symbol, uint64 editionSize, uint16 royaltyBPS, address fundsRecipient, address defaultAdmin, (uint104 publicSalePrice, uint32 maxSalePurchasePerAddress, uint64 publicSaleStart, uint64 publicSaleEnd, uint64 presaleStart, uint64 presaleEnd, bytes32 presaleMerkleRoot) saleConfig, string description, string animationURI, string imageURI) returns (address)', () => { | ||
expectTypeOf( | ||
normalizeSignature( | ||
'function createEdition(string name, string symbol, uint64 editionSize, uint16 royaltyBPS, address fundsRecipient, address defaultAdmin, (uint104 publicSalePrice, uint32 maxSalePurchasePerAddress, uint64 publicSaleStart, uint64 publicSaleEnd, uint64 presaleStart, uint64 presaleEnd, bytes32 presaleMerkleRoot) saleConfig, string description, string animationURI, string imageURI) returns (address)', | ||
), | ||
).toMatchTypeOf<'createEdition(string,string,uint64,uint16,address,address,(uint104,uint32,uint64,uint64,uint64,uint64,bytes32),string,string,string)'>() | ||
}) | ||
|
||
test('trim spaces', () => { | ||
expectTypeOf( | ||
normalizeSignature( | ||
'function createEdition(string name,string symbol, uint64 editionSize , uint16 royaltyBPS, address fundsRecipient, address defaultAdmin, ( uint104 publicSalePrice , uint32 maxSalePurchasePerAddress, uint64 publicSaleStart, uint64 publicSaleEnd, uint64 presaleStart, uint64 presaleEnd, bytes32 presaleMerkleRoot) saleConfig , string description, string animationURI, string imageURI ) returns (address)', | ||
), | ||
).toMatchTypeOf<TodoType>() | ||
}) | ||
|
||
test('error: invalid signatures', () => { | ||
// TODO: these should be errors, but they aren't yet | ||
// assertType<never>(normalizeSignature('bar')) | ||
// assertType<never>(normalizeSignature('bar(uint foo')) | ||
// assertType<never>(normalizeSignature('baruint foo)')) | ||
// assertType<never>(normalizeSignature('bar(uint foo, (uint baz)')) | ||
}) |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,13 +1,44 @@ | ||
import type { | ||
AbiEvent, | ||
AbiFunction, | ||
ParseAbiItem, | ||
SignatureAbiItem, | ||
} from 'abitype' | ||
import { BaseError } from '../../errors/base.js' | ||
import type { ErrorType } from '../../errors/utils.js' | ||
|
||
type NormalizeSignatureParameters = string | ||
type NormalizeSignatureReturnType = string | ||
/** | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. There are multiple TODOs here and I'm not sure which one of these would have to be fixed to be considered good enough to merge (given that any type is better than no type, but the wrong type is also worse than no type) I wanted to make the original commit for this PR contain as detailed information before we start truncating out cases to support the subset that makes sense Here are the 4 options in increase level of work required (each option requires the previous option to be completed): Option 1: we can make that the Option 2: we make Option 3: we try to make all valid Option 4: make all |
||
* TODO: ParseAbiItem<Signature> is too strict here since | ||
* 1. normalizeSignature allows loose formatting (ex: spacing) | ||
* 2. normalization doesn't require the string to conform to abitype's human-readable encoding | ||
* ex: 'function bar( )' should still results in `bar()` as the result | ||
* ex: `event foo(tuple(uint16, uint16))` results in a broken type response | ||
* for now, we return`string` in cases where abitype fails | ||
* but this doesn't handle error cases where abitype is just wrong like | ||
* TODO: Some obvious errors get the return type `string` | ||
* this is because we fallback to `string` if `ParseAbiItem<Signature>` fails | ||
* but it could fail because not because it's too strict, but because the string is plain incorrect | ||
* ex: normalizeSignature('bar') falls back to `string` as the response type | ||
* since it can't tell if this is wrong since `ParseAbiItem<Signature>` is too strict or if it really is invalid | ||
* TODO: Some normalizations are missing | ||
* ex 1: (JS wrong, type correct) 'event Barry(uint foo)' results in type `Barry(uint256)`, but the real return value is `Barry(uint)`. | ||
* ex 2: (Both wrong) tuples doesn't get removed `foo(tuple(uint16, uint16))` does not get normalized to 'foo((uint16, uint16))' | ||
*/ | ||
export type ToSignature<Signature extends string> = | ||
ParseAbiItem<Signature> extends never | ||
? string | ||
: ParseAbiItem<Signature> extends AbiFunction | AbiEvent | ||
? SignatureAbiItem<ParseAbiItem<Signature>> | ||
: never | ||
|
||
type NormalizeSignatureParameters<Signature extends string> = Signature | ||
type NormalizeSignatureReturnType<Signature extends string> = | ||
ToSignature<Signature> | ||
export type NormalizeSignatureErrorType = ErrorType | ||
|
||
export function normalizeSignature( | ||
signature: NormalizeSignatureParameters, | ||
): NormalizeSignatureReturnType { | ||
export function normalizeSignature<Signature extends string>( | ||
signature: NormalizeSignatureParameters<Signature>, | ||
): NormalizeSignatureReturnType<Signature> { | ||
let active = true | ||
let current = '' | ||
let level = 0 | ||
|
@@ -60,5 +91,5 @@ export function normalizeSignature( | |
|
||
if (!valid) throw new BaseError('Unable to normalize signature.') | ||
|
||
return result | ||
return result as NormalizeSignatureReturnType<Signature> | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
note: this type (
SignatureAbiItem
) depends on wevm/abitype#255 getting merged and released