/* * Copyright (c) Meta Platforms, Inc. and affiliates. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include #include #include using namespace apache::thrift::compiler; TEST(ParserTest, type_resolution) { auto source_mgr = source_manager(); source_mgr.add_virtual_file("test.thrift", R"( struct S { 1: MissingType field; } )"); auto diag = std::optional(); auto diags = diagnostics_engine(source_mgr, [&diag](diagnostic d) { diag = d; }); // Types must be resolved in parse_ast. auto programs = parse_ast(source_mgr, diags, "test.thrift", {}); EXPECT_TRUE(diags.has_errors()); EXPECT_EQ(diag->message(), "Type `test.MissingType` not defined."); // Programs must be non-null even in case of an error. EXPECT_TRUE(programs != nullptr); } TEST(ParserTest, missing_includes) { auto source_mgr = source_manager(); source_mgr.add_virtual_file("test.thrift", "include \"nonexist.thrift\"\n"); { int count = 0; auto diags = diagnostics_engine(source_mgr, [&count](diagnostic d) { EXPECT_EQ(d.level(), diagnostic_level::error); EXPECT_EQ(d.message(), "Could not find include file nonexist.thrift"); ++count; }); auto programs = parse_ast(source_mgr, diags, "test.thrift", {}); EXPECT_EQ(programs, nullptr); EXPECT_TRUE(diags.has_errors()); EXPECT_EQ(count, 1); } { int count = 0; auto diags = diagnostics_engine(source_mgr, [&count](diagnostic d) { EXPECT_EQ(d.level(), diagnostic_level::warning); // We should get both errors since we did not end parsing on first error. if (count == 0) { EXPECT_EQ(d.message(), "Could not find include file nonexist.thrift"); } else { EXPECT_EQ(d.message(), "failed to open file: nonexist.thrift"); } ++count; }); parsing_params params; params.allow_missing_includes = true; auto programs = parse_ast(source_mgr, diags, "test.thrift", params); EXPECT_NE(programs, nullptr); EXPECT_FALSE(diags.has_errors()); EXPECT_EQ(count, 2); } } TEST(ParserTest, struct_doc) { auto source_mgr = source_manager(); source_mgr.add_virtual_file("test.thrift", R"( /// struct doc struct S { /// field doc 1: i32 field; /** * multiline doc */ 2: i32 field; 3: i32 field2; // ignored } )"); auto diag = std::optional(); auto diags = diagnostics_engine(source_mgr, [&diag](diagnostic d) { diag = d; }); // No parsing errors auto programs = parse_ast(source_mgr, diags, "test.thrift", {}); EXPECT_FALSE(diags.has_errors()); auto s = programs->get_root_program()->structs_and_unions()[0]; EXPECT_EQ(s->doc(), "struct doc\n"); EXPECT_EQ(s->fields()[0].doc(), "field doc\n"); EXPECT_EQ(s->fields()[1].doc(), "multiline doc\n"); EXPECT_FALSE(s->fields()[2].has_doc()); } TEST(ParserTest, enum_doc) { auto source_mgr = source_manager(); source_mgr.add_virtual_file("test.thrift", R"( /// enum doc enum E { /// value doc A = 1, /** * multiline doc */ B = 2, C = 3, // ignored } )"); auto diag = std::optional(); auto diags = diagnostics_engine(source_mgr, [&diag](diagnostic d) { diag = d; }); // No parsing errors auto programs = parse_ast(source_mgr, diags, "test.thrift", {}); EXPECT_FALSE(diags.has_errors()); auto e = programs->get_root_program()->enums()[0]; EXPECT_EQ(e->doc(), "enum doc\n"); EXPECT_EQ(e->values()[0].doc(), "value doc\n"); EXPECT_EQ(e->values()[1].doc(), "multiline doc\n"); EXPECT_FALSE(e->values()[2].has_doc()); } TEST(ParserTest, struct_annotation) { auto source_mgr = source_manager(); source_mgr.add_virtual_file("test.thrift", R"( include "thrift/annotation/rust.thrift" struct S { @rust.Type{name = "Foo"} 1: i32 field; 2: i32 field; 3: i32 field2; } )"); auto diag = std::optional(); auto diags = diagnostics_engine(source_mgr, [&diag](diagnostic d) { diag = d; }); parsing_params params; if (auto* includes = std::getenv("IMPLICIT_INCLUDES")) { params.incl_searchpath.push_back(includes); } auto programs = parse_ast(source_mgr, diags, "test.thrift", params); EXPECT_FALSE(diags.has_errors()); auto& s = programs->get_root_program()->structs_and_unions()[0]; EXPECT_EQ(s->fields()[0].structured_annotations().size(), 1); EXPECT_EQ( s->fields()[0].structured_annotations()[0].type()->uri(), "facebook.com/thrift/annotation/rust/Type"); auto a = s->fields()[0].find_structured_annotation_or_null( "facebook.com/thrift/annotation/rust/Type"); EXPECT_TRUE(a != nullptr); auto v = s->fields()[0].structured_annotations()[0].value(); auto& m = v->get_map(); ASSERT_EQ(m.size(), 1); ASSERT_EQ(m[0].first->get_string(), "name"); ASSERT_EQ(m[0].second->get_string(), "Foo"); }