#include "resource_pack.hpp" #include // Para replace #include // Para path, recursive_directory_iterator, directory_entry, exists, relative #include // Para basic_ifstream, basic_ostream, basic_ofstream, operator<<, basic_ios, basic_istream::read, basic_ostream::write, endl, ios, basic_istream, ifstream, operator|, basic_istream::seekg, basic_istream::tellg, ofstream, streamsize #include // Para cerr #include // Para pair const std::string ResourcePack::DEFAULT_ENCRYPT_KEY = "CCAE_RESOURCES__2024"; ResourcePack::ResourcePack() : loaded_(false) {} ResourcePack::~ResourcePack() { clear(); } auto ResourcePack::calculateChecksum(const std::vector& data) -> uint32_t { uint32_t checksum = 0x12345678; for (unsigned char i : data) { checksum = ((checksum << 5) + checksum) + i; } return checksum; } void ResourcePack::encryptData(std::vector& data, const std::string& key) { if (key.empty()) { return; } for (size_t i = 0; i < data.size(); ++i) { data[i] ^= key[i % key.length()]; } } void ResourcePack::decryptData(std::vector& data, const std::string& key) { encryptData(data, key); } auto ResourcePack::loadPack(const std::string& pack_file) -> bool { std::ifstream file(pack_file, std::ios::binary); if (!file) { std::cerr << "Error: Could not open pack file: " << pack_file << '\n'; return false; } char header[4]; file.read(header, 4); if (std::string(header, 4) != "CCAE") { std::cerr << "Error: Invalid pack file format" << '\n'; return false; } uint32_t version; file.read(reinterpret_cast(&version), sizeof(version)); if (version != 1) { std::cerr << "Error: Unsupported pack version: " << version << '\n'; return false; } uint32_t resource_count; file.read(reinterpret_cast(&resource_count), sizeof(resource_count)); resources_.clear(); resources_.reserve(resource_count); for (uint32_t i = 0; i < resource_count; ++i) { uint32_t filename_length; file.read(reinterpret_cast(&filename_length), sizeof(filename_length)); std::string filename(filename_length, '\0'); file.read(filename.data(), filename_length); ResourceEntry entry; entry.filename = filename; file.read(reinterpret_cast(&entry.offset), sizeof(entry.offset)); file.read(reinterpret_cast(&entry.size), sizeof(entry.size)); file.read(reinterpret_cast(&entry.checksum), sizeof(entry.checksum)); resources_[filename] = entry; } uint64_t data_size; file.read(reinterpret_cast(&data_size), sizeof(data_size)); data_.resize(data_size); file.read(reinterpret_cast(data_.data()), data_size); decryptData(data_, DEFAULT_ENCRYPT_KEY); loaded_ = true; return true; } auto ResourcePack::savePack(const std::string& pack_file) -> bool { std::ofstream file(pack_file, std::ios::binary); if (!file) { std::cerr << "Error: Could not create pack file: " << pack_file << '\n'; return false; } file.write("CCAE", 4); uint32_t version = 1; file.write(reinterpret_cast(&version), sizeof(version)); auto resource_count = static_cast(resources_.size()); file.write(reinterpret_cast(&resource_count), sizeof(resource_count)); for (const auto& [filename, entry] : resources_) { auto filename_length = static_cast(filename.length()); file.write(reinterpret_cast(&filename_length), sizeof(filename_length)); file.write(filename.c_str(), filename_length); file.write(reinterpret_cast(&entry.offset), sizeof(entry.offset)); file.write(reinterpret_cast(&entry.size), sizeof(entry.size)); file.write(reinterpret_cast(&entry.checksum), sizeof(entry.checksum)); } std::vector encrypted_data = data_; encryptData(encrypted_data, DEFAULT_ENCRYPT_KEY); uint64_t data_size = encrypted_data.size(); file.write(reinterpret_cast(&data_size), sizeof(data_size)); file.write(reinterpret_cast(encrypted_data.data()), data_size); return true; } auto ResourcePack::addFile(const std::string& filename, const std::string& filepath) -> bool { std::ifstream file(filepath, std::ios::binary | std::ios::ate); if (!file) { std::cerr << "Error: Could not open file: " << filepath << '\n'; return false; } std::streamsize file_size = file.tellg(); file.seekg(0, std::ios::beg); std::vector file_data(file_size); if (!file.read(reinterpret_cast(file_data.data()), file_size)) { std::cerr << "Error: Could not read file: " << filepath << '\n'; return false; } ResourceEntry entry; entry.filename = filename; entry.offset = data_.size(); entry.size = file_data.size(); entry.checksum = calculateChecksum(file_data); data_.insert(data_.end(), file_data.begin(), file_data.end()); resources_[filename] = entry; return true; } auto ResourcePack::addDirectory(const std::string& directory) -> bool { if (!std::filesystem::exists(directory)) { std::cerr << "Error: Directory does not exist: " << directory << '\n'; return false; } for (const auto& entry : std::filesystem::recursive_directory_iterator(directory)) { if (entry.is_regular_file()) { std::string filepath = entry.path().string(); std::string filename = std::filesystem::relative(entry.path(), directory).string(); std::ranges::replace(filename, '\\', '/'); if (!addFile(filename, filepath)) { return false; } } } return true; } auto ResourcePack::getResource(const std::string& filename) -> std::vector { auto it = resources_.find(filename); if (it == resources_.end()) { std::cerr << "Error: Resource not found: " << filename << '\n'; return {}; } const ResourceEntry& entry = it->second; if (entry.offset + entry.size > data_.size()) { std::cerr << "Error: Invalid resource data: " << filename << '\n'; return {}; } std::vector result(data_.begin() + entry.offset, data_.begin() + entry.offset + entry.size); uint32_t checksum = calculateChecksum(result); if (checksum != entry.checksum) { std::cerr << "Warning: Checksum mismatch for resource: " << filename << '\n'; } return result; } auto ResourcePack::hasResource(const std::string& filename) const -> bool { return resources_.find(filename) != resources_.end(); } void ResourcePack::clear() { resources_.clear(); data_.clear(); loaded_ = false; } auto ResourcePack::getResourceCount() const -> size_t { return resources_.size(); } auto ResourcePack::getResourceList() const -> std::vector { std::vector result; result.reserve(resources_.size()); for (const auto& [filename, entry] : resources_) { result.push_back(filename); } return result; }