#include "CodeGenerator.h" #include "Domain/Computations/StructureComputations.h" #include "Templates/AssetStructTestsTemplate.h" #include "Templates/ZoneLoadTemplate.h" #include "Templates/ZoneMarkTemplate.h" #include "Templates/ZoneWriteTemplate.h" #include "Utils/Logging/Log.h" #include #include #include #include namespace fs = std::filesystem; CodeGenerator::CodeGenerator(const ZoneCodeGeneratorArguments* args) : m_args(args) { SetupTemplates(); } void CodeGenerator::SetupTemplates() { m_template_mapping["zoneload"] = std::make_unique(); m_template_mapping["zonemark"] = std::make_unique(); m_template_mapping["zonewrite"] = std::make_unique(); m_template_mapping["assetstructtests"] = std::make_unique(); } bool CodeGenerator::GenerateCodeForTemplate(const RenderingContext& context, ICodeTemplate* codeTemplate) const { for (const auto& codeFile : codeTemplate->GetFilesToRender(context)) { fs::path p(m_args->m_output_directory); p.append(codeFile.m_file_name); auto parentFolder(p); parentFolder.remove_filename(); create_directories(parentFolder); std::ofstream stream(p, std::fstream::out | std::fstream::binary); if (!stream.is_open()) { con::error("Failed to open file '{}'", p.string()); return false; } codeTemplate->RenderFile(stream, codeFile.m_tag, context); stream.close(); } return true; } bool CodeGenerator::GetAssetWithName(IDataRepository* repository, const std::string& name, StructureInformation*& asset) { auto* def = repository->GetDataDefinitionByName(name); if (def == nullptr) { con::error("Could not find type with name '{}'", name); return false; } auto* defWithMembers = dynamic_cast(def); asset = defWithMembers != nullptr ? repository->GetInformationFor(defWithMembers) : nullptr; if (asset == nullptr) { con::error("Could not find type with name '{}'", name); return false; } if (!StructureComputations(asset).IsAsset()) { con::error("Type is not an asset '{}'", name); return false; } return true; } bool CodeGenerator::GenerateCode(IDataRepository* repository) { std::vector assets; for (auto* info : repository->GetAllStructureInformation()) { StructureComputations computations(info); if (computations.IsAsset()) assets.push_back(info); } const auto start = std::chrono::steady_clock::now(); for (const auto& generationTask : m_args->m_generation_tasks) { auto templateName = generationTask.m_template_name; for (auto& c : templateName) c = static_cast(tolower(c)); const auto foundTemplate = m_template_mapping.find(templateName); if (foundTemplate == m_template_mapping.end()) { con::error("Unknown template '{}'.", generationTask.m_template_name); return false; } if (generationTask.m_all_assets) { for (auto* asset : assets) { auto context = RenderingContext::BuildContext(repository, asset); if (!GenerateCodeForTemplate(*context, foundTemplate->second.get())) { con::error("Failed to generate code for asset '{}' with preset '{}'", asset->m_definition->GetFullName(), foundTemplate->first); return false; } con::info("Successfully generated code for asset '{}' with preset '{}'", asset->m_definition->GetFullName(), foundTemplate->first); } } else { StructureInformation* asset; if (!GetAssetWithName(repository, generationTask.m_asset_name, asset)) return false; auto context = RenderingContext::BuildContext(repository, asset); if (!GenerateCodeForTemplate(*context, foundTemplate->second.get())) return false; } } const auto end = std::chrono::steady_clock::now(); con::debug("Generating code took {}ms", std::chrono::duration_cast(end - start).count()); return true; }