| // Copyright 2018 The Fuchsia Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <fidl/flat_ast.h> |
| #include <fidl/lexer.h> |
| #include <fidl/names.h> |
| #include <fidl/parser.h> |
| #include <fidl/source_file.h> |
| #include <unittest/unittest.h> |
| |
| #include "test_library.h" |
| |
| namespace { |
| |
| bool valid_using() { |
| BEGIN_TEST; |
| |
| SharedAmongstLibraries shared; |
| TestLibrary dependency("dependent.fidl", R"FIDL( |
| library dependent; |
| |
| struct Bar { |
| int8 s; |
| }; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(dependency.Compile()); |
| |
| TestLibrary library("example.fidl", R"FIDL( |
| library example; |
| |
| using dependent; |
| |
| struct Foo { |
| dependent.Bar dep; |
| }; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(library.AddDependentLibrary(std::move(dependency))); |
| ASSERT_TRUE(library.Compile()); |
| |
| END_TEST; |
| } |
| |
| bool valid_using_with_as_refs_through_both() { |
| BEGIN_TEST; |
| |
| SharedAmongstLibraries shared; |
| TestLibrary dependency("dependent.fidl", R"FIDL( |
| library dependent; |
| |
| struct Bar { |
| int8 s; |
| }; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(dependency.Compile()); |
| |
| TestLibrary library("example.fidl", R"FIDL( |
| library example; |
| |
| using dependent as the_alias; |
| |
| struct Foo { |
| dependent.Bar dep1; |
| the_alias.Bar dep2; |
| }; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(library.AddDependentLibrary(std::move(dependency))); |
| ASSERT_TRUE(library.Compile()); |
| |
| END_TEST; |
| } |
| |
| bool valid_using_with_as_ref_only_through_fqn() { |
| BEGIN_TEST; |
| |
| SharedAmongstLibraries shared; |
| TestLibrary dependency("dependent.fidl", R"FIDL( |
| library dependent; |
| |
| struct Bar { |
| int8 s; |
| }; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(dependency.Compile()); |
| |
| TestLibrary library("example.fidl", R"FIDL( |
| library example; |
| |
| using dependent as the_alias; |
| |
| struct Foo { |
| dependent.Bar dep1; |
| }; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(library.AddDependentLibrary(std::move(dependency))); |
| ASSERT_TRUE(library.Compile()); |
| |
| END_TEST; |
| } |
| |
| bool valid_using_with_as_ref_only_through_alias() { |
| BEGIN_TEST; |
| |
| SharedAmongstLibraries shared; |
| TestLibrary dependency("dependent.fidl", R"FIDL( |
| library dependent; |
| |
| struct Bar { |
| int8 s; |
| }; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(dependency.Compile()); |
| |
| TestLibrary library("example.fidl", R"FIDL( |
| library example; |
| |
| using dependent as the_alias; |
| |
| struct Foo { |
| the_alias.Bar dep1; |
| }; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(library.AddDependentLibrary(std::move(dependency))); |
| ASSERT_TRUE(library.Compile()); |
| |
| END_TEST; |
| } |
| |
| bool invalid_missing_using() { |
| BEGIN_TEST; |
| |
| TestLibrary library(R"FIDL( |
| library example; |
| |
| // missing using. |
| |
| struct Foo { |
| dependent.Bar dep; |
| }; |
| |
| )FIDL"); |
| ASSERT_FALSE(library.Compile()); |
| const auto& errors = library.errors(); |
| ASSERT_EQ(1, errors.size()); |
| ASSERT_STR_STR(errors[0].c_str(), "unknown type dependent.Bar"); |
| |
| END_TEST; |
| } |
| |
| bool invalid_unknown_using() { |
| BEGIN_TEST; |
| |
| TestLibrary library(R"FIDL( |
| library example; |
| |
| using dependent; // unknown using. |
| |
| struct Foo { |
| dependent.Bar dep; |
| }; |
| |
| )FIDL"); |
| ASSERT_FALSE(library.Compile()); |
| const auto& errors = library.errors(); |
| ASSERT_EQ(1, errors.size()); |
| ASSERT_STR_STR( |
| errors[0].c_str(), |
| "Could not find library named dependent. Did you include its sources with --files?"); |
| |
| END_TEST; |
| } |
| |
| bool invalid_duplicate_using() { |
| BEGIN_TEST; |
| |
| SharedAmongstLibraries shared; |
| TestLibrary dependency("dependent.fidl", R"FIDL( |
| library dependent; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(dependency.Compile()); |
| |
| TestLibrary library("example.fidl", R"FIDL( |
| library example; |
| |
| using dependent; |
| using dependent; // duplicated |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(library.AddDependentLibrary(std::move(dependency))); |
| ASSERT_FALSE(library.Compile()); |
| const auto& errors = library.errors(); |
| ASSERT_EQ(1, errors.size()); |
| ASSERT_STR_STR(errors[0].c_str(), |
| "Library dependent already imported. Did you require it twice?"); |
| |
| END_TEST; |
| } |
| |
| bool invalid_unused_using() { |
| BEGIN_TEST; |
| |
| SharedAmongstLibraries shared; |
| TestLibrary dependency("dependent.fidl", R"FIDL( |
| library dependent; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(dependency.Compile()); |
| |
| TestLibrary library("example.fidl", R"FIDL( |
| library example; |
| |
| using dependent; |
| |
| struct Foo { |
| int64 does_not; |
| int32 use_dependent; |
| }; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(library.AddDependentLibrary(std::move(dependency))); |
| ASSERT_FALSE(library.Compile()); |
| |
| const auto& errors = library.errors(); |
| ASSERT_EQ(1, errors.size()); |
| ASSERT_STR_STR(errors[0].c_str(), |
| "Library example imports dependent but does not use it. Either use dependent, or " |
| "remove import."); |
| |
| END_TEST; |
| } |
| |
| bool invalid_too_many_provided_libraries() { |
| BEGIN_TEST; |
| |
| SharedAmongstLibraries shared; |
| |
| TestLibrary dependency("notused.fidl", "library not.used;", &shared); |
| ASSERT_TRUE(dependency.Compile()); |
| |
| TestLibrary library("example.fidl", "library example;", &shared); |
| ASSERT_TRUE(library.AddDependentLibrary(std::move(dependency))); |
| ASSERT_TRUE(library.Compile()); |
| |
| auto unused = shared.all_libraries.Unused(library.library()); |
| ASSERT_EQ(1, unused.size()); |
| ASSERT_STR_EQ("not.used", fidl::NameLibrary(*unused.begin()).c_str()); |
| |
| END_TEST; |
| } |
| |
| bool library_declaration_name_collision() { |
| BEGIN_TEST; |
| |
| SharedAmongstLibraries shared; |
| TestLibrary dependency("dep.fidl", R"FIDL( |
| library dep; |
| |
| struct A{}; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(dependency.Compile()); |
| TestLibrary library("lib.fidl", |
| R"FIDL( |
| library lib; |
| |
| using dep; |
| |
| struct dep{}; |
| |
| struct B{dep.A a;}; // So the import is used. |
| |
| )FIDL", |
| &shared); |
| |
| ASSERT_TRUE(library.AddDependentLibrary(std::move(dependency))); |
| ASSERT_FALSE(library.Compile()); |
| const auto& errors = library.errors(); |
| ASSERT_EQ(1, errors.size()); |
| ASSERT_STR_STR(errors[0].c_str(), |
| "Declaration name 'dep' conflicts with a library import; consider using the 'as' " |
| "keyword to import the library under a different name."); |
| END_TEST; |
| } |
| |
| bool aliased_library_declaration_name_collision() { |
| BEGIN_TEST; |
| |
| SharedAmongstLibraries shared; |
| TestLibrary dependency("dep.fidl", R"FIDL( |
| library dep; |
| |
| struct A{}; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(dependency.Compile()); |
| TestLibrary library("lib.fidl", |
| R"FIDL( |
| library lib; |
| |
| using dep as x; |
| |
| struct x{}; |
| |
| struct B{dep.A a;}; // So the import is used. |
| |
| )FIDL", |
| &shared); |
| |
| ASSERT_TRUE(library.AddDependentLibrary(std::move(dependency))); |
| ASSERT_FALSE(library.Compile()); |
| const auto& errors = library.errors(); |
| ASSERT_EQ(1, errors.size()); |
| ASSERT_STR_STR(errors[0].c_str(), |
| "Declaration name 'x' conflicts with a library import; consider using the 'as' " |
| "keyword to import the library under a different name."); |
| END_TEST; |
| } |
| |
| bool aliased_library_nonaliased_declaration_name_collision() { |
| BEGIN_TEST; |
| |
| SharedAmongstLibraries shared; |
| TestLibrary dependency("dep.fidl", R"FIDL( |
| library dep; |
| |
| struct A{}; |
| |
| )FIDL", |
| &shared); |
| ASSERT_TRUE(dependency.Compile()); |
| TestLibrary library("lib.fidl", |
| R"FIDL( |
| library lib; |
| |
| using dep as depnoconflict; |
| |
| struct dep{}; |
| |
| struct B{depnoconflict.A a;}; // So the import is used. |
| |
| )FIDL", |
| &shared); |
| |
| ASSERT_TRUE(library.AddDependentLibrary(std::move(dependency))); |
| ASSERT_FALSE(library.Compile()); |
| const auto& errors = library.errors(); |
| ASSERT_EQ(1, errors.size()); |
| ASSERT_STR_STR( |
| errors[0].c_str(), |
| R"ERROR(lib.fidl:6:8: error: Declaration name 'x' conflicts with a library import; consider using the 'as' keyword to import the library under a different name. |
| struct x{}; |
| ^)ERROR"); |
| END_TEST; |
| } |
| |
| } // namespace |
| |
| BEGIN_TEST_CASE(using_tests) |
| RUN_TEST(valid_using) |
| RUN_TEST(valid_using_with_as_refs_through_both) |
| RUN_TEST(valid_using_with_as_ref_only_through_fqn) |
| RUN_TEST(valid_using_with_as_ref_only_through_alias) |
| RUN_TEST(invalid_missing_using) |
| RUN_TEST(invalid_unknown_using) |
| RUN_TEST(invalid_duplicate_using) |
| RUN_TEST(invalid_unused_using) |
| RUN_TEST(invalid_too_many_provided_libraries) |
| RUN_TEST(library_declaration_name_collision) |
| RUN_TEST(aliased_library_declaration_name_collision) |
| END_TEST_CASE(using_tests) |