// // EuropaTools // // (C) 2021-2025 modeco80 // // SPDX-License-Identifier: LGPL-3.0-or-later // #ifndef EUROPA_IO_PAKFILE_H #define EUROPA_IO_PAKFILE_H #include #include #include #include #include #include #include namespace europa::io { struct PakReader; struct PakWriter; /// sumtype struct PakFileData { // clang-format off using Variant = std::variant< // File data std::vector, // Path std::filesystem::path >; // clang-format on static PakFileData InitAsBuffer(std::vector&& buffer) { return PakFileData { .variant_ = Variant(std::move(buffer)) }; } static PakFileData InitAsPath(const std::filesystem::path& path) { return PakFileData { .variant_ = Variant(path) }; } std::uint32_t GetSize() const { // FIXME: make this just a overloaded lambda struct SizeVisitor { std::uint32_t& size; // bleh void operator()(const std::vector& buffer) { size = static_cast(buffer.size()); } void operator()(const std::filesystem::path& fsPath) { if(!std::filesystem::exists(fsPath) && !std::filesystem::is_regular_file(fsPath)) throw std::runtime_error("invalid path in path file"); size = static_cast(std::filesystem::file_size(fsPath)); } }; std::uint32_t size {}; std::visit(SizeVisitor { size }, variant_); return size; } template const T* GetIf() const { return std::get_if(&variant_); } template auto Visit(Visitor&& v) const { return std::visit(v, variant_); } // private: PakFileData::Variant variant_; }; /// Repressents a package file. /// FIXME: Maybe make this not hold a buffer at some point, /// or a sumtype which can contain either buffer OR path to os file /// (which we can then efficiently tee into) struct PakFile { using DataType = PakFileData; template void InitWithExistingTocEntry(const T& value) { toc = value; } void InitAs(structs::PakVersion version) { switch(version) { case structs::PakVersion::Ver3: toc = structs::PakHeader_V3::TocEntry {}; break; case structs::PakVersion::Ver4: toc = structs::PakHeader_V4::TocEntry {}; break; case structs::PakVersion::Ver5: toc = structs::PakHeader_V5::TocEntry {}; break; default: throw std::invalid_argument("Invalid PAK version to initalize TOC entry"); break; } } bool HasData() const { return fileData.has_value(); } /** * Get the file data. */ [[nodiscard]] const DataType& GetData() const { if(!fileData.has_value()) throw std::runtime_error("no file data to get!"); return fileData.value(); } /// Sets data. void SetData(DataType&& data) { this->fileData = std::move(data); // Update the TOC size. std::visit([&](auto& entry) { entry.size = this->fileData.value().GetSize(); }, toc); } /// Purge read file data. void PurgeData() { this->fileData = std::nullopt; } /** * Get the TOC entry responsible. */ template [[nodiscard]] const T& GetTOCEntry() const { return std::get(toc); } std::uint32_t GetCreationUnixTime() const { std::uint32_t time {}; std::visit([&](auto& entry) { time = entry.creationUnixTime; }, toc); return time; } std::uint32_t GetOffset() const { std::uint32_t size {}; std::visit([&](auto& entry) { size = entry.offset; }, toc); return size; } std::uint32_t GetSize() const { std::uint32_t size {}; std::visit([&](auto& entry) { size = entry.size; }, toc); return size; } template void Visit(const Cb& cb) { std::visit(cb, toc); } private: friend PakReader; friend PakWriter; std::optional fileData; structs::PakTocEntryVariant toc; }; } // namespace europa::io #endif // EUROPA_IO_PAKFILE_H