#include "Parsing/Commands/Sequence/SequenceAction.h" #include "Parsing/Mock/MockLexer.h" #include "Parsing/PostProcessing/CreateMemberInformationPostProcessor.h" #include "Parsing/PostProcessing/CreateStructureInformationPostProcessor.h" #include "Persistence/InMemory/InMemoryRepository.h" #include "Utils/ClassUtils.h" #include #include namespace test::parsing::commands::sequence::sequence_action { class CommandsSequenceTestsHelper { public: std::unique_ptr m_repository; std::unique_ptr m_state; std::unique_ptr> m_lexer; StructDefinition* m_test_struct_raw; StructureInformation* m_test_struct; StructDefinition* m_test_struct2_raw; StructureInformation* m_test_struct2; StructDefinition* m_arg_struct_raw; StructureInformation* m_arg_struct; StructDefinition* m_arg_struct2_raw; StructureInformation* m_arg_struct2; StructDefinition* m_arg_struct3_raw; StructureInformation* m_arg_struct3; unsigned m_consumed_token_count; private: void RetrieveInformationPointers() { m_test_struct = m_repository->GetInformationFor(m_test_struct_raw); REQUIRE(m_test_struct != nullptr); m_test_struct2 = m_repository->GetInformationFor(m_test_struct2_raw); REQUIRE(m_test_struct2 != nullptr); m_arg_struct = m_repository->GetInformationFor(m_arg_struct_raw); REQUIRE(m_arg_struct != nullptr); m_arg_struct2 = m_repository->GetInformationFor(m_arg_struct2_raw); REQUIRE(m_arg_struct2 != nullptr); m_arg_struct3 = m_repository->GetInformationFor(m_arg_struct3_raw); REQUIRE(m_arg_struct3 != nullptr); } _NODISCARD bool CreateInformation() const { auto createStructureInformation = std::make_unique(); auto createMemberInformation = std::make_unique(); return createStructureInformation->PostProcess(m_repository.get()) && createMemberInformation->PostProcess(m_repository.get()); } void AddSampleData() { auto def = std::make_unique("", "test_struct_t", 8); def->m_members.emplace_back(std::make_shared("m_test", std::make_unique(BaseTypeDefinition::BOOL))); m_test_struct_raw = def.get(); m_repository->Add(std::move(def)); def = std::make_unique("", "container_struct_t", 8); def->m_members.emplace_back(std::make_shared("m_child", std::make_unique(m_test_struct_raw))); m_test_struct2_raw = def.get(); m_repository->Add(std::move(def)); def = std::make_unique("", "arg_t", 8); def->m_members.emplace_back(std::make_shared("m_hello_world", std::make_unique(BaseTypeDefinition::INT))); m_arg_struct_raw = def.get(); m_repository->Add(std::move(def)); def = std::make_unique("", "another_arg_t", 8); def->m_members.emplace_back(std::make_shared("m_hello_galaxy", std::make_unique(BaseTypeDefinition::DOUBLE))); def->m_members.emplace_back(std::make_shared("m_hello_universe", std::make_unique(BaseTypeDefinition::CHAR))); m_arg_struct2_raw = def.get(); m_repository->Add(std::move(def)); def = std::make_unique("", "yeet_t", 8); def->m_members.emplace_back(std::make_shared("m_ya", std::make_unique(m_arg_struct_raw))); def->m_members.emplace_back(std::make_shared("m_yeeeeeet", std::make_unique(m_arg_struct2_raw))); m_arg_struct3_raw = def.get(); m_repository->Add(std::move(def)); } public: CommandsSequenceTestsHelper() : m_repository(std::make_unique()), m_state(std::make_unique(m_repository.get())), m_test_struct_raw(nullptr), m_test_struct(nullptr), m_test_struct2_raw(nullptr), m_test_struct2(nullptr), m_arg_struct_raw(nullptr), m_arg_struct(nullptr), m_arg_struct2_raw(nullptr), m_arg_struct2(nullptr), m_arg_struct3_raw(nullptr), m_arg_struct3(nullptr), m_consumed_token_count(0u) { AddSampleData(); REQUIRE(CreateInformation()); RetrieveInformationPointers(); } void Tokens(std::initializer_list> tokens) { m_lexer = std::make_unique>(tokens, CommandsParserValue::EndOfFile(TokenPos())); } bool PerformTest() { REQUIRE(m_lexer); const auto sequence = std::make_unique(); return sequence->MatchSequence(m_lexer.get(), m_state.get(), m_consumed_token_count); } }; TEST_CASE("SequenceAction: Ensure can parse simple action directive", "[parsing][sequence]") { CommandsSequenceTestsHelper helper; const TokenPos pos; helper.Tokens({ CommandsParserValue::Identifier(pos, new std::string("set")), CommandsParserValue::Identifier(pos, new std::string("action")), CommandsParserValue::Identifier(pos, new std::string("test_struct_t")), CommandsParserValue::Identifier(pos, new std::string("TestMethod")), CommandsParserValue::Character(pos, '('), CommandsParserValue::Character(pos, ')'), CommandsParserValue::Character(pos, ';'), CommandsParserValue::EndOfFile(pos), }); auto result = helper.PerformTest(); REQUIRE(result); REQUIRE(helper.m_consumed_token_count == 7); REQUIRE(helper.m_test_struct->m_post_load_action); REQUIRE(helper.m_test_struct->m_post_load_action->m_action_name == "TestMethod"); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types.empty()); } TEST_CASE("SequenceAction: Ensure can parse simple action directive with one arg", "[parsing][sequence]") { CommandsSequenceTestsHelper helper; const TokenPos pos; helper.Tokens({ CommandsParserValue::Identifier(pos, new std::string("set")), CommandsParserValue::Identifier(pos, new std::string("action")), CommandsParserValue::Identifier(pos, new std::string("test_struct_t")), CommandsParserValue::Identifier(pos, new std::string("TestMethod")), CommandsParserValue::Character(pos, '('), CommandsParserValue::Identifier(pos, new std::string("arg_t")), CommandsParserValue::Character(pos, ')'), CommandsParserValue::Character(pos, ';'), CommandsParserValue::EndOfFile(pos), }); auto result = helper.PerformTest(); REQUIRE(result); REQUIRE(helper.m_consumed_token_count == 8); REQUIRE(helper.m_test_struct->m_post_load_action); REQUIRE(helper.m_test_struct->m_post_load_action->m_action_name == "TestMethod"); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types.size() == 1); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types[0] == helper.m_arg_struct_raw); } TEST_CASE("SequenceAction: Ensure can parse simple action directive with two args", "[parsing][sequence]") { CommandsSequenceTestsHelper helper; const TokenPos pos; helper.Tokens({ CommandsParserValue::Identifier(pos, new std::string("set")), CommandsParserValue::Identifier(pos, new std::string("action")), CommandsParserValue::Identifier(pos, new std::string("test_struct_t")), CommandsParserValue::Identifier(pos, new std::string("TestMethod")), CommandsParserValue::Character(pos, '('), CommandsParserValue::Identifier(pos, new std::string("arg_t")), CommandsParserValue::Character(pos, ','), CommandsParserValue::Identifier(pos, new std::string("another_arg_t")), CommandsParserValue::Character(pos, ')'), CommandsParserValue::Character(pos, ';'), CommandsParserValue::EndOfFile(pos), }); auto result = helper.PerformTest(); REQUIRE(result); REQUIRE(helper.m_consumed_token_count == 10); REQUIRE(helper.m_test_struct->m_post_load_action); REQUIRE(helper.m_test_struct->m_post_load_action->m_action_name == "TestMethod"); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types.size() == 2); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types[0] == helper.m_arg_struct_raw); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types[1] == helper.m_arg_struct2_raw); } TEST_CASE("SequenceAction: Ensure can parse simple action directive with three args", "[parsing][sequence]") { CommandsSequenceTestsHelper helper; const TokenPos pos; helper.Tokens({ CommandsParserValue::Identifier(pos, new std::string("set")), CommandsParserValue::Identifier(pos, new std::string("action")), CommandsParserValue::Identifier(pos, new std::string("test_struct_t")), CommandsParserValue::Identifier(pos, new std::string("TestMethod")), CommandsParserValue::Character(pos, '('), CommandsParserValue::Identifier(pos, new std::string("arg_t")), CommandsParserValue::Character(pos, ','), CommandsParserValue::Identifier(pos, new std::string("another_arg_t")), CommandsParserValue::Character(pos, ','), CommandsParserValue::Identifier(pos, new std::string("yeet_t")), CommandsParserValue::Character(pos, ')'), CommandsParserValue::Character(pos, ';'), CommandsParserValue::EndOfFile(pos), }); auto result = helper.PerformTest(); REQUIRE(result); REQUIRE(helper.m_consumed_token_count == 12); REQUIRE(helper.m_test_struct->m_post_load_action); REQUIRE(helper.m_test_struct->m_post_load_action->m_action_name == "TestMethod"); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types.size() == 3); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types[0] == helper.m_arg_struct_raw); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types[1] == helper.m_arg_struct2_raw); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types[2] == helper.m_arg_struct3_raw); } TEST_CASE("SequenceAction: Fails if base typename does not exist", "[parsing][sequence]") { CommandsSequenceTestsHelper helper; const TokenPos pos; helper.Tokens({ CommandsParserValue::Identifier(pos, new std::string("set")), CommandsParserValue::Identifier(pos, new std::string("action")), CommandsParserValue::Identifier(pos, new std::string("unknown_struct_t")), CommandsParserValue::Identifier(pos, new std::string("TestMethod")), CommandsParserValue::Character(pos, '('), CommandsParserValue::Character(pos, ')'), CommandsParserValue::Character(pos, ';'), CommandsParserValue::EndOfFile(pos), }); REQUIRE_THROWS_AS(helper.PerformTest(), ParsingException); REQUIRE(helper.m_test_struct->m_post_load_action == nullptr); } TEST_CASE("SequenceAction: Fails if arg typename does not exist", "[parsing][sequence]") { CommandsSequenceTestsHelper helper; const TokenPos pos; helper.Tokens({ CommandsParserValue::Identifier(pos, new std::string("set")), CommandsParserValue::Identifier(pos, new std::string("action")), CommandsParserValue::Identifier(pos, new std::string("test_struct_t")), CommandsParserValue::Identifier(pos, new std::string("TestMethod")), CommandsParserValue::Character(pos, '('), CommandsParserValue::Identifier(pos, new std::string("arg_t")), CommandsParserValue::Character(pos, ','), CommandsParserValue::Identifier(pos, new std::string("another_arg_t")), CommandsParserValue::Character(pos, ','), CommandsParserValue::Identifier(pos, new std::string("unknown_type_t")), CommandsParserValue::Character(pos, ')'), CommandsParserValue::Character(pos, ';'), CommandsParserValue::EndOfFile(pos), }); REQUIRE_THROWS_AS(helper.PerformTest(), ParsingException); REQUIRE(helper.m_test_struct->m_post_load_action == nullptr); } TEST_CASE("SequenceAction: Fails if no type and no use", "[parsing][sequence]") { CommandsSequenceTestsHelper helper; const TokenPos pos; helper.Tokens({ CommandsParserValue::Identifier(pos, new std::string("set")), CommandsParserValue::Identifier(pos, new std::string("action")), CommandsParserValue::Identifier(pos, new std::string("TestMethod")), CommandsParserValue::Character(pos, '('), CommandsParserValue::Character(pos, ')'), CommandsParserValue::Character(pos, ';'), CommandsParserValue::EndOfFile(pos), }); REQUIRE_THROWS_AS(helper.PerformTest(), ParsingException); REQUIRE(helper.m_test_struct->m_post_load_action == nullptr); } TEST_CASE("SequenceAction: Ensure can parse action directive with used type", "[parsing][sequence]") { CommandsSequenceTestsHelper helper; const TokenPos pos; helper.Tokens({ CommandsParserValue::Identifier(pos, new std::string("set")), CommandsParserValue::Identifier(pos, new std::string("action")), CommandsParserValue::Identifier(pos, new std::string("TestMethod")), CommandsParserValue::Character(pos, '('), CommandsParserValue::Character(pos, ')'), CommandsParserValue::Character(pos, ';'), CommandsParserValue::EndOfFile(pos), }); helper.m_state->SetInUse(helper.m_test_struct); auto result = helper.PerformTest(); REQUIRE(result); REQUIRE(helper.m_consumed_token_count == 6); REQUIRE(helper.m_test_struct->m_post_load_action); REQUIRE(helper.m_test_struct->m_post_load_action->m_action_name == "TestMethod"); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types.empty()); } TEST_CASE("SequenceAction: Ensure can parse action directive with type from member", "[parsing][sequence]") { CommandsSequenceTestsHelper helper; const TokenPos pos; helper.Tokens({ CommandsParserValue::Identifier(pos, new std::string("set")), CommandsParserValue::Identifier(pos, new std::string("action")), CommandsParserValue::Identifier(pos, new std::string("container_struct_t")), CommandsParserValue::Character(pos, ':'), CommandsParserValue::Character(pos, ':'), CommandsParserValue::Identifier(pos, new std::string("m_child")), CommandsParserValue::Identifier(pos, new std::string("TestMethod")), CommandsParserValue::Character(pos, '('), CommandsParserValue::Character(pos, ')'), CommandsParserValue::Character(pos, ';'), CommandsParserValue::EndOfFile(pos), }); auto result = helper.PerformTest(); REQUIRE(result); REQUIRE(helper.m_consumed_token_count == 10); REQUIRE(helper.m_test_struct->m_post_load_action); REQUIRE(helper.m_test_struct->m_post_load_action->m_action_name == "TestMethod"); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types.empty()); } TEST_CASE("SequenceAction: Ensure can parse action directive with type from member and in use", "[parsing][sequence]") { CommandsSequenceTestsHelper helper; const TokenPos pos; helper.Tokens({ CommandsParserValue::Identifier(pos, new std::string("set")), CommandsParserValue::Identifier(pos, new std::string("action")), CommandsParserValue::Identifier(pos, new std::string("m_child")), CommandsParserValue::Identifier(pos, new std::string("TestMethod")), CommandsParserValue::Character(pos, '('), CommandsParserValue::Character(pos, ')'), CommandsParserValue::Character(pos, ';'), CommandsParserValue::EndOfFile(pos), }); helper.m_state->SetInUse(helper.m_test_struct2); auto result = helper.PerformTest(); REQUIRE(result); REQUIRE(helper.m_consumed_token_count == 7); REQUIRE(helper.m_test_struct->m_post_load_action); REQUIRE(helper.m_test_struct->m_post_load_action->m_action_name == "TestMethod"); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types.empty()); } TEST_CASE("SequenceAction: Ensure can use different struct even though something is used", "[parsing][sequence]") { CommandsSequenceTestsHelper helper; const TokenPos pos; helper.Tokens({ CommandsParserValue::Identifier(pos, new std::string("set")), CommandsParserValue::Identifier(pos, new std::string("action")), CommandsParserValue::Identifier(pos, new std::string("container_struct_t")), CommandsParserValue::Character(pos, ':'), CommandsParserValue::Character(pos, ':'), CommandsParserValue::Identifier(pos, new std::string("m_child")), CommandsParserValue::Identifier(pos, new std::string("TestMethod")), CommandsParserValue::Character(pos, '('), CommandsParserValue::Character(pos, ')'), CommandsParserValue::Character(pos, ';'), CommandsParserValue::EndOfFile(pos), }); helper.m_state->SetInUse(helper.m_arg_struct2); auto result = helper.PerformTest(); REQUIRE(result); REQUIRE(helper.m_consumed_token_count == 10); REQUIRE(helper.m_test_struct->m_post_load_action); REQUIRE(helper.m_test_struct->m_post_load_action->m_action_name == "TestMethod"); REQUIRE(helper.m_test_struct->m_post_load_action->m_parameter_types.empty()); } TEST_CASE("SequenceAction: Ensure member must be type with members", "[parsing][sequence]") { CommandsSequenceTestsHelper helper; const TokenPos pos; helper.Tokens({ CommandsParserValue::Identifier(pos, new std::string("set")), CommandsParserValue::Identifier(pos, new std::string("action")), CommandsParserValue::Identifier(pos, new std::string("test_struct_t")), CommandsParserValue::Character(pos, ':'), CommandsParserValue::Character(pos, ':'), CommandsParserValue::Identifier(pos, new std::string("m_test")), CommandsParserValue::Identifier(pos, new std::string("TestMethod")), CommandsParserValue::Character(pos, '('), CommandsParserValue::Character(pos, ')'), CommandsParserValue::Character(pos, ';'), CommandsParserValue::EndOfFile(pos), }); REQUIRE_THROWS_AS(helper.PerformTest(), ParsingException); REQUIRE(helper.m_test_struct->m_post_load_action == nullptr); } } // namespace test::parsing::commands::sequence::sequence_action