-
Notifications
You must be signed in to change notification settings - Fork 5.9k
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
Invalid and inconsistent types in ABIs for library functions #9278
Comments
Yeah, I'm sorry this is really a mess. Someone should create a consistent overview of the current state of affairs. The ABI for library functions somehow should include which string is used to compute the function selector and potentially how to encode it. One speciality for library functions are of course storage pointers, which are not part of the ABI. So for |
Yes, sorry, to be clear, I was implicitly omitting functions that take or return storage pointers, since those aren't put in the ABI. I said "pure or view functions", but what I really meant was "pure or view functions that neither take nor return storage pointers". Once you omit those, then, well... you're left with the fact that these ABIs are not valid. Except for structs. Those ones are valid for some reason. Just not enums and contracts. While I don't know the history here, I can't help but speculate that the decision was made to change it to this broken way in 0.1.5 to make selectors work better, and then this never got revisited, even when structs got added and were done the valid way. This does of course leave the problem of how to alter the ABI to make computing the selector possible, and there are several possible solutions to that, but I'll skip discussing those because that's a separate issue that's already been much discussed already[0]. Because I think that first of all before that, these invalid ABIs need to be changed to conform to the spec. Sure, tools may get the selector wrong, but that's better than having them, like, just choke on the things. And I mean, who cares if the selector is right if you can't actually encode? [0]OK I'll state what I see as the two possible solutions briefly. They are:
...but again these both have been discussed already elsewhere, so... |
Also just to be clear -- I think it makes sense to fix this particular issue before sorting out how to handle this problem more generally. Like right now it's malformed and not even consistent (the one benefit of this way is rendered moot by the fact that structs don't work this way -- not that they should (OK, it's also rendered moot by other factors as mentioned above)). May as well handle that first, since that should be pretty easy, and figure out the bigger problem later. |
Today I have encountered a bug stemming from the fact that people are generally not aware of this inconsistency (here specifically in case of enums in library ABI): dethcrypto/TypeChain#216 @chriseth So, the current behavior clearly needs to be documented so I'm labeling this as |
This actually appears to be a pretty widespread problem. I checked I reported this as bugs (ethers-io/ethers.js#1126, ethereum/eth-abi#146) but if so many tools have problems with it then maybe we should not worry about backwards compatibility here - they don't work with the current ABI anyway. |
I just want to point out that this is basically the same point I made above. :) :P |
Is there is actually anyway that ethers can handle this? Is the information required encoded some other way in the ABI? |
@haltman-at Right. I read that comment a while ago and now missed that point. In any case, I agree with you. I was just investigating a different issue related to enums and while doing it I also checked what the situation is regarding this one. And it confirms what you said - it's bad :(. @ricmoo I don't think it is, but @chriseth knows more about the ABI so maybe he'd be able to suggest something. In case of enums people from TypeChain hard-coded it to We're going to limit enums to |
Note that the Hm, I suppose this actually might be more handleable than I thought, given that:
That's pretty hacky. I mean, it's the sort of thing I might just do, but probably not the sort of thing I'd expect anyone else to do. :) |
We discussed it on the call today. We really don't want to break library selectors so it would be best to just add extra fields with necessary information. I created a separate issue for the docs to at least push that part forward since it's pretty clear-cut that we need them: #10201. @ricmoo Actually, you don't need to know if an enum is |
@cameel but I do need to know the type for encoding the selector, no? Or is uint256 always used for that too? |
I am receiving the following error from abigen: $ abigen --abi abi.json --pkg versioning --type Versioning --out versioning.go
Fatal: Failed to generate ABI binding: unsupported arg type: State For simplicity I have produced a minimal reproducible copy of what is causing the error, from my actual code, with an example code shown below. // SPDX-License-Identifier: MIT
pragma solidity ^0.8.5;
enum State {
UNINITIALISED,
INIT,
OPEN
}
library Versioning {
function versionState(
State state,
string calldata version
) external pure returns (string memory versionedState) {
if (state == State.OPEN) {
return string(abi.encodePacked(version, "OPEN"));
}
if (state == State.INIT) {
return string(abi.encodePacked(version, "INIT"));
}
if (state == State.UNINITIALISED) {
return string(abi.encodePacked(version, "UNINITIALISED"));
}
return "UNKNOWN STATE";
}
} I am able to reproduce the code generation error, when running the following set of commands: $ forge build mrc.sol
$ jq .abi ./out/mrc.sol/Versioning.json > abi.json
$ abigen --abi abi.json --pkg versioning --type Versioning --out versioning.go
Fatal: Failed to generate ABI binding: unsupported arg type: State
$ typechain --target ethers-v6 --out-dir . ./out/mrc.sol/Versioning.json ...
versionState: TypedContractMethod<
[state: BigNumberish, version: string],
[string],
"view"
>;
... I would expect the enum in question to be converted to some integer type in Go, // mrc.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.5;
enum Example {
THIS,
IS,
AN,
EXAMPLE,
ENUM
}
enum State {
UNINITIALISED,
INIT,
OPEN
}
library Versioning {
function versionState(
State state,
string calldata version
) external pure returns (string memory versionedState) {
if (state == State.OPEN) {
return string(abi.encodePacked(version, "OPEN"));
}
if (state == State.INIT) {
return string(abi.encodePacked(version, "INIT"));
}
if (state == State.UNINITIALISED) {
return string(abi.encodePacked(version, "UNINITIALISED"));
}
return "UNKNOWN STATE";
}
}
// import.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.5;
import { Example } from "./mrc.sol";
contract ExampleContract {
constructor() {}
function exampleType(
Example exm
) external pure returns (string memory example) {
return string(abi.encodePacked("example", exm));
}
} Using the commands below generates the following Go code: $ forge buid mrc.sol import.sol
$ jq .abi ./out/import.sol/ExampleContract.json > abi.json
$ abigen --abi abi.json --pkg example --type Example --out example.go
# No errors ...
// ExampleType is a free data retrieval call binding the contract method 0xd19b25d7.
//
// Solidity: function exampleType(uint8 exm) pure returns(string example)
func (_Example *ExampleCaller) ExampleType(opts *bind.CallOpts, exm uint8) (string, error) {
var out []interface{}
err := _Example.contract.Call(opts, &out, "exampleType", exm)
if err != nil {
return *new(string), err
}
out0 := *abi.ConvertType(out[0], new(string)).(*string)
return out0, err
}
... |
Description
While most library functions don't go in the ABI,
pure
andview
functions do. Of course, actually using these is infamously difficult, due to the fact that library selectors are computed differently from other selectors, but they're there. The specific types that differ are structs, enums, and contracts, which in the signature use the internal name of the type, rather than the usual conversion to an ABI type.The problem is this: In nearly every version of Solidity, including 0.6.10, the ABI parameter entries for enums and contracts put this internal name of the type in the
type
field, rather than the name of the corresponding ABI type.Now, this does have the advantage that it allows the selector to be computed correctly by the naive method, without specifically knowing that you're dealing with a library! (As the ABI still lacks anything to indicate the contract/library distinction.) But it has the disadvantage that it screws up any more sophisticated attempt to process the ABI, that is expecting something that conforms to the spec. (Like: You can't actually encode with these types! Without other information, there's nothing to tell us whether a
Blank
is the name of a contract type or the name of a globally-declared enum type. Web3 and Ethers sure can't handle it, and I'm not about to make Truffle try.)I suspect this is simply a bug. Further evidence for this is that this only seems to happen to enums and contracts -- structs are unaffected. Even if it's deliberate... well, it should be changed, and the library problem should be handled a different way. (Having
internalType
goes a long way here, even if we still need something in the ABI to distinguish contracts from libraries.)Now, on noticing this, I had to check which versions are affected... it turns out that this behavior goes all the way back to 0.1.5, a mere two versions after libraries were introduced in 0.1.3. Given how long this has been around, one might be tempted to say, this isn't a bug, this is just how it works, at least for now. But I'm still calling it a bug: it breaks the spec; it breaks any processing more sophisticated than computing the selector, such as encoding or decoding; and structs don't work this way and have never worked this way, not since 0.4.17 when it first became possible to pass a struct externally.
Environment
Steps to Reproduce
Try compiling the following:
(comment things out as appropriate to get it to compile with earlier versions; changing
calldata
tomemory
andexternal
topublic
may also be necessary; also changingview
toconstant
(or just omitting that, since everything went in the ABI prior to 0.5.6))The text was updated successfully, but these errors were encountered: