Merge pull request #346 from chriseth/importAliases

Allow aliases during import.
This commit is contained in:
chriseth 2016-01-13 17:29:34 +01:00
commit d2f18c73f7
4 changed files with 81 additions and 16 deletions

View File

@ -49,6 +49,8 @@ Declaration const* DeclarationContainer::conflictingDeclaration(
if (!dynamic_cast<FunctionDefinition const*>(declaration)) if (!dynamic_cast<FunctionDefinition const*>(declaration))
return declaration; return declaration;
} }
else if (declarations.size() == 1 && declarations.front() == &_declaration)
return nullptr;
else if (!declarations.empty()) else if (!declarations.empty())
return declarations.front(); return declarations.front();
@ -73,13 +75,12 @@ bool DeclarationContainer::registerDeclaration(
m_declarations.erase(*_name); m_declarations.erase(*_name);
m_invisibleDeclarations.erase(*_name); m_invisibleDeclarations.erase(*_name);
} }
else if (conflictingDeclaration(_declaration)) else if (conflictingDeclaration(_declaration, _name))
return false; return false;
if (_invisible) vector<Declaration const*>& decls = _invisible ? m_invisibleDeclarations[*_name] : m_declarations[*_name];
m_invisibleDeclarations[*_name].push_back(&_declaration); if (!contains(decls, &_declaration))
else decls.push_back(&_declaration);
m_declarations[*_name].push_back(&_declaration);
return true; return true;
} }

View File

@ -76,22 +76,57 @@ bool NameAndTypeResolver::performImports(SourceUnit& _sourceUnit, map<string, So
string const& path = imp->annotation().absolutePath; string const& path = imp->annotation().absolutePath;
if (!_sourceUnits.count(path)) if (!_sourceUnits.count(path))
{ {
reportDeclarationError( node->location(), reportDeclarationError(
"Import \"" + imp->location(),
"Import \"" + path + "\" (referenced as \"" + imp->path() + "\") not found."
);
error = true;
continue;
}
auto scope = m_scopes.find(_sourceUnits.at(path));
solAssert(scope != end(m_scopes), "");
if (!imp->symbolAliases().empty())
for (auto const& alias: imp->symbolAliases())
{
auto declarations = scope->second->resolveName(alias.first->name(), false);
if (declarations.empty())
{
reportDeclarationError(
imp->location(),
"Declaration \"" +
alias.first->name() +
"\" not found in \"" +
path + path +
"\" (referenced as \"" + "\" (referenced as \"" +
imp->path() + imp->path() +
"\") not found." "\")."
); );
error = true; error = true;
} }
else if (imp->name().empty()) else
for (Declaration const* declaration: declarations)
{ {
auto scope = m_scopes.find(_sourceUnits.at(path)); ASTString const* name = alias.second ? alias.second.get() : &declaration->name();
solAssert(scope != end(m_scopes), ""); if (!target.registerDeclaration(*declaration, name))
{
reportDeclarationError(
imp->location(),
"Identifier \"" + *name + "\" already declared."
);
error = true;
}
}
}
else if (imp->name().empty())
for (auto const& nameAndDeclaration: scope->second->declarations()) for (auto const& nameAndDeclaration: scope->second->declarations())
for (auto const& declaration: nameAndDeclaration.second) for (auto const& declaration: nameAndDeclaration.second)
target.registerDeclaration(*declaration, &nameAndDeclaration.first); if (!target.registerDeclaration(*declaration, &nameAndDeclaration.first))
{
reportDeclarationError(
imp->location(),
"Identifier \"" + nameAndDeclaration.first + "\" already declared."
);
error = true;
} }
} }
return !error; return !error;

View File

@ -201,6 +201,10 @@ public:
virtual void accept(ASTConstVisitor& _visitor) const override; virtual void accept(ASTConstVisitor& _visitor) const override;
ASTString const& path() const { return *m_path; } ASTString const& path() const { return *m_path; }
std::vector<std::pair<ASTPointer<Identifier>, ASTPointer<ASTString>>> const& symbolAliases() const
{
return m_symbolAliases;
}
virtual ImportAnnotation& annotation() const override; virtual ImportAnnotation& annotation() const override;
virtual TypePointer type() const override; virtual TypePointer type() const override;

View File

@ -101,6 +101,31 @@ BOOST_AUTO_TEST_CASE(simple_alias)
BOOST_CHECK(c.compile()); BOOST_CHECK(c.compile());
} }
BOOST_AUTO_TEST_CASE(complex_import)
{
CompilerStack c;
c.addSource("a", "contract A {} contract B {} contract C { struct S { uint a; } }");
c.addSource("b", "import \"a\" as x; import {B as b, C as c, C} from \"a\"; "
"contract D is b { function f(c.S var1, x.C.S var2, C.S var3) internal {} }");
BOOST_CHECK(c.compile());
}
BOOST_AUTO_TEST_CASE(name_clash_in_import)
{
CompilerStack c;
c.addSource("a", "contract A {}");
c.addSource("b", "import \"a\"; contract A {} ");
BOOST_CHECK(!c.compile());
c.addSource("b", "import \"a\" as A; contract A {} ");
BOOST_CHECK(!c.compile());
c.addSource("b", "import {A as b} from \"a\"; contract b {} ");
BOOST_CHECK(!c.compile());
c.addSource("b", "import {A} from \"a\"; contract A {} ");
BOOST_CHECK(!c.compile());
c.addSource("b", "import {A} from \"a\"; contract B {} ");
BOOST_CHECK(c.compile());
}
BOOST_AUTO_TEST_SUITE_END() BOOST_AUTO_TEST_SUITE_END()
} }