mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Fix interface type conversion internal to structs.
This commit is contained in:
parent
823e67bf40
commit
36a90289e6
@ -1783,6 +1783,8 @@ TypePointer StructType::interfaceType(bool _inLibrary) const
|
||||
if (_inLibrary && location() == DataLocation::Storage)
|
||||
return shared_from_this();
|
||||
else if (!recursive())
|
||||
// TODO this might not be enough, we have to convert all members to
|
||||
// their interfaceType
|
||||
return copyForLocation(DataLocation::Memory, true);
|
||||
else
|
||||
return TypePointer();
|
||||
@ -1805,7 +1807,9 @@ string StructType::signatureInExternalFunction(bool _structsByName) const
|
||||
auto memberTypeStrings = memberTypes | boost::adaptors::transformed([&](TypePointer _t) -> string
|
||||
{
|
||||
solAssert(_t, "Parameter should have external type.");
|
||||
return _t->signatureInExternalFunction(_structsByName);
|
||||
auto t = _t->interfaceType(_structsByName);
|
||||
solAssert(t, "");
|
||||
return t->signatureInExternalFunction(_structsByName);
|
||||
});
|
||||
return "(" + boost::algorithm::join(memberTypeStrings, ",") + ")";
|
||||
}
|
||||
|
@ -152,7 +152,9 @@ Json::Value ABI::formatType(string const& _name, Type const& _type, bool _forLib
|
||||
for (auto const& member: structType->members(nullptr))
|
||||
{
|
||||
solAssert(member.type, "");
|
||||
ret["type"].append(formatType(member.name, *member.type, _forLibrary));
|
||||
auto t = member.type->interfaceType(_forLibrary);
|
||||
solAssert(t, "");
|
||||
ret["type"].append(formatType(member.name, *t, _forLibrary));
|
||||
}
|
||||
}
|
||||
else
|
||||
|
@ -980,6 +980,42 @@ BOOST_AUTO_TEST_CASE(return_structs)
|
||||
checkInterface(text, interface);
|
||||
}
|
||||
|
||||
BOOST_AUTO_TEST_CASE(return_structs_with_contracts)
|
||||
{
|
||||
char const* text = R"(
|
||||
contract C {
|
||||
struct S { C[] x; C y; }
|
||||
function f() returns (S s, C c) {
|
||||
}
|
||||
}
|
||||
)";
|
||||
char const* interface = R"(
|
||||
[
|
||||
{
|
||||
"constant" : false,
|
||||
"payable": false,
|
||||
"inputs": [],
|
||||
"name": "f",
|
||||
"outputs" : [{
|
||||
"name" : "s",
|
||||
"type" : [{
|
||||
"name" : "x",
|
||||
"type" : "address[]"
|
||||
}, {
|
||||
"name" : "y",
|
||||
"type" : "address"
|
||||
}]
|
||||
}, {
|
||||
"name" : "c",
|
||||
"type" : "address"
|
||||
}],
|
||||
"type" : "function"
|
||||
}
|
||||
]
|
||||
)";
|
||||
checkInterface(text, interface);
|
||||
}
|
||||
|
||||
BOOST_AUTO_TEST_CASE(event_structs)
|
||||
{
|
||||
char const* text = R"(
|
||||
|
@ -609,7 +609,7 @@ BOOST_AUTO_TEST_CASE(external_structs)
|
||||
struct Nested { X[2][] a; mapping(uint => uint) m; uint y; }
|
||||
struct X { bytes32 x; Test t; Empty[] e; }
|
||||
function f(ActionChoices, uint, Empty) external {}
|
||||
function g(Nested) external {}
|
||||
function g(Test, Nested) external {}
|
||||
function h(function(Nested) external returns (uint)[]) external {}
|
||||
function i(Nested[]) external {}
|
||||
}
|
||||
@ -620,10 +620,8 @@ BOOST_AUTO_TEST_CASE(external_structs)
|
||||
{
|
||||
auto functions = contract->definedFunctions();
|
||||
BOOST_REQUIRE(!functions.empty());
|
||||
for (auto const& f: functions)
|
||||
cout << f->externalSignature() << endl;
|
||||
BOOST_CHECK_EQUAL("f(uint8,uint256,())", functions[0]->externalSignature());
|
||||
BOOST_CHECK_EQUAL("g(((bytes32,address,()[])[2][],uint256))", functions[1]->externalSignature());
|
||||
BOOST_CHECK_EQUAL("g(address,((bytes32,address,()[])[2][],uint256))", functions[1]->externalSignature());
|
||||
BOOST_CHECK_EQUAL("h(function[])", functions[2]->externalSignature());
|
||||
BOOST_CHECK_EQUAL("i(((bytes32,address,()[])[2][],uint256)[])", functions[3]->externalSignature());
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user