1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-22 04:23:53 +00:00
openmw/apps/bsatool/bsatool.cpp

352 lines
11 KiB
C++
Raw Normal View History

2022-05-22 16:52:05 +00:00
#include <filesystem>
#include <fstream>
2015-05-09 22:36:04 +00:00
#include <iomanip>
2022-09-22 18:26:05 +00:00
#include <iostream>
2013-03-07 20:05:56 +00:00
#include <vector>
#include <boost/program_options.hpp>
2023-03-02 15:51:04 +00:00
#include <components/bsa/ba2dx10file.hpp>
2023-03-02 15:49:09 +00:00
#include <components/bsa/ba2gnrlfile.hpp>
2020-10-23 11:04:21 +00:00
#include <components/bsa/compressedbsafile.hpp>
#include <components/files/configurationmanager.hpp>
#include <components/files/conversion.hpp>
2022-09-22 18:26:05 +00:00
#include <components/misc/strings/algorithm.hpp>
#include <components/misc/strings/conversion.hpp>
2013-03-04 23:45:25 +00:00
2013-03-07 20:05:56 +00:00
#define BSATOOL_VERSION 1.1
2013-03-04 23:45:25 +00:00
2013-03-07 20:05:56 +00:00
// Create local aliases for brevity
namespace bpo = boost::program_options;
2013-03-04 23:45:25 +00:00
2013-03-07 20:05:56 +00:00
struct Arguments
2013-03-04 23:45:25 +00:00
{
2013-03-07 20:05:56 +00:00
std::string mode;
std::filesystem::path filename;
std::filesystem::path extractfile;
std::filesystem::path addfile;
std::filesystem::path outdir;
2013-03-04 23:45:25 +00:00
2013-03-07 20:05:56 +00:00
bool longformat;
bool fullpath;
};
2013-03-04 23:45:25 +00:00
2022-09-22 18:26:05 +00:00
bool parseOptions(int argc, char** argv, Arguments& info)
2013-03-07 20:05:56 +00:00
{
bpo::options_description desc(R"(Inspect and extract files from Bethesda BSA archives
Usages:
bsatool list [-l] archivefile\n
List the files presents in the input archive.
bsatool extract [-f] archivefile [file_to_extract] [output_directory]
Extract a file from the input archive.
bsatool extractall archivefile [output_directory]
Extract all files from the input archive.
bsatool add [-a] archivefile file_to_add
Add a file to the input archive.
bsatool create [-c] archivefile
Create an archive.
Allowed options)");
2013-03-07 20:05:56 +00:00
auto addOption = desc.add_options();
addOption("help,h", "print help message.");
addOption("version,v", "print version information and quit.");
addOption("long,l", "Include extra information in archive listing.");
addOption("full-path,f", "Create directory hierarchy on file extraction (always true for extractall).");
2013-03-07 20:05:56 +00:00
// input-file is hidden and used as a positional argument
bpo::options_description hidden("Hidden Options");
auto addHiddenOption = hidden.add_options();
addHiddenOption("mode,m", bpo::value<std::string>(), "bsatool mode");
2022-09-22 18:26:05 +00:00
addHiddenOption("input-file,i", bpo::value<Files::MaybeQuotedPathContainer>(), "input file");
2013-03-07 20:05:56 +00:00
bpo::positional_options_description p;
p.add("mode", 1).add("input-file", 3);
// there might be a better way to do this
bpo::options_description all;
all.add(desc).add(hidden);
bpo::variables_map variables;
try
{
2022-09-22 18:26:05 +00:00
bpo::parsed_options valid_opts = bpo::command_line_parser(argc, argv).options(all).positional(p).run();
2013-03-07 20:05:56 +00:00
bpo::store(valid_opts, variables);
}
2022-09-22 18:26:05 +00:00
catch (std::exception& e)
2013-03-04 23:45:25 +00:00
{
2022-09-22 18:26:05 +00:00
std::cout << "ERROR parsing arguments: " << e.what() << "\n\n" << desc << std::endl;
2013-03-07 20:05:56 +00:00
return false;
2013-03-04 23:45:25 +00:00
}
2013-03-07 20:05:56 +00:00
bpo::notify(variables);
2022-09-22 18:26:05 +00:00
if (variables.count("help"))
2013-03-07 20:05:56 +00:00
{
std::cout << desc << std::endl;
return false;
}
2022-09-22 18:26:05 +00:00
if (variables.count("version"))
2013-03-07 20:05:56 +00:00
{
std::cout << "BSATool version " << BSATOOL_VERSION << std::endl;
return false;
}
if (!variables.count("mode"))
{
2022-09-22 18:26:05 +00:00
std::cout << "ERROR: no mode specified!\n\n" << desc << std::endl;
2013-03-07 20:05:56 +00:00
return false;
}
info.mode = variables["mode"].as<std::string>();
2022-09-22 18:26:05 +00:00
if (!(info.mode == "list" || info.mode == "extract" || info.mode == "extractall" || info.mode == "add"
|| info.mode == "create"))
2013-03-07 20:05:56 +00:00
{
2022-09-22 18:26:05 +00:00
std::cout << std::endl << "ERROR: invalid mode \"" << info.mode << "\"\n\n" << desc << std::endl;
2013-03-07 20:05:56 +00:00
return false;
}
if (!variables.count("input-file"))
2013-03-04 23:45:25 +00:00
{
2022-09-22 18:26:05 +00:00
std::cout << "\nERROR: missing BSA archive\n\n" << desc << std::endl;
2013-03-07 20:05:56 +00:00
return false;
}
2022-09-22 18:26:05 +00:00
auto inputFiles = variables["input-file"].as<Files::MaybeQuotedPathContainer>();
2022-09-22 18:26:05 +00:00
info.filename = inputFiles[0].u8string(); // This call to u8string is redundant, but required to build on MSVC 14.26
// due to implementation bugs.
2013-03-04 23:45:25 +00:00
2013-03-07 20:05:56 +00:00
// Default output to the working directory
info.outdir = std::filesystem::current_path();
2013-03-07 20:05:56 +00:00
if (info.mode == "extract")
{
if (inputFiles.size() < 2)
2013-03-04 23:45:25 +00:00
{
2022-09-22 18:26:05 +00:00
std::cout << "\nERROR: file to extract unspecified\n\n" << desc << std::endl;
2013-03-07 20:05:56 +00:00
return false;
2013-03-04 23:45:25 +00:00
}
if (inputFiles.size() > 1)
2022-09-22 18:26:05 +00:00
info.extractfile = inputFiles[1].u8string(); // This call to u8string is redundant, but required to build on
// MSVC 14.26 due to implementation bugs.
if (inputFiles.size() > 2)
2022-09-22 18:26:05 +00:00
info.outdir = inputFiles[2].u8string(); // This call to u8string is redundant, but required to build on
// MSVC 14.26 due to implementation bugs.
2013-03-07 20:05:56 +00:00
}
2021-04-07 16:58:46 +00:00
else if (info.mode == "add")
{
if (inputFiles.empty())
2021-04-07 16:58:46 +00:00
{
2022-09-22 18:26:05 +00:00
std::cout << "\nERROR: file to add unspecified\n\n" << desc << std::endl;
2021-04-07 16:58:46 +00:00
return false;
}
if (inputFiles.size() > 1)
2022-09-22 18:26:05 +00:00
info.addfile = inputFiles[1].u8string(); // This call to u8string is redundant, but required to build on
// MSVC 14.26 due to implementation bugs.
2021-04-07 16:58:46 +00:00
}
else if (inputFiles.size() > 1)
2022-09-22 18:26:05 +00:00
info.outdir = inputFiles[1].u8string(); // This call to u8string is redundant, but required to build on
// MSVC 14.26 due to implementation bugs.
2013-03-04 23:45:25 +00:00
2015-01-22 18:04:59 +00:00
info.longformat = variables.count("long") != 0;
info.fullpath = variables.count("full-path") != 0;
2013-03-04 23:45:25 +00:00
2013-03-07 20:05:56 +00:00
return true;
}
2013-03-04 23:45:25 +00:00
2022-09-22 18:26:05 +00:00
template <typename File>
int list(std::unique_ptr<File>& bsa, Arguments& info)
2013-03-07 20:05:56 +00:00
{
2013-03-04 23:45:25 +00:00
// List all files
2022-09-22 18:26:05 +00:00
const auto& files = bsa->getList();
2020-06-10 06:30:37 +00:00
for (const auto& file : files)
2013-03-04 23:45:25 +00:00
{
2022-09-22 18:26:05 +00:00
if (info.longformat)
2013-03-04 23:45:25 +00:00
{
// Long format
std::ios::fmtflags f(std::cout.flags());
2021-04-07 16:58:46 +00:00
std::cout << std::setw(50) << std::left << file.name();
2020-06-10 06:30:37 +00:00
std::cout << std::setw(8) << std::left << std::dec << file.fileSize;
std::cout << "@ 0x" << std::hex << file.offset << std::endl;
std::cout.flags(f);
2013-03-04 23:45:25 +00:00
}
else
2021-04-07 16:58:46 +00:00
std::cout << file.name() << std::endl;
2013-03-04 23:45:25 +00:00
}
2013-03-07 20:05:56 +00:00
return 0;
}
2022-09-22 18:26:05 +00:00
template <typename File>
int extract(std::unique_ptr<File>& bsa, Arguments& info)
2013-03-07 20:05:56 +00:00
{
auto archivePath = info.extractfile.u8string();
Misc::StringUtils::replaceAll(archivePath, u8"/", u8"\\");
2013-03-07 20:05:56 +00:00
auto extractPath = info.extractfile.u8string();
Misc::StringUtils::replaceAll(extractPath, u8"\\", u8"/");
2013-03-07 20:05:56 +00:00
Files::IStreamPtr stream;
// Get a stream for the file to extract
for (auto it = bsa->getList().rbegin(); it != bsa->getList().rend(); ++it)
{
auto streamPath = Misc::StringUtils::stringToU8String(it->name());
if (Misc::StringUtils::ciEqual(streamPath, archivePath) || Misc::StringUtils::ciEqual(streamPath, extractPath))
{
stream = bsa->getFile(&*it);
break;
}
}
if (!stream)
2013-03-07 20:05:56 +00:00
{
std::cout << "ERROR: file '" << Misc::StringUtils::u8StringToString(archivePath) << "' not found\n";
std::cout << "In archive: " << Files::pathToUnicodeString(info.filename) << std::endl;
2013-03-07 20:05:56 +00:00
return 3;
}
// Get the target path (the path the file will be extracted to)
2022-09-22 18:26:05 +00:00
std::filesystem::path relPath(extractPath);
2013-03-07 20:05:56 +00:00
2022-05-22 16:52:05 +00:00
std::filesystem::path target;
2013-03-07 20:05:56 +00:00
if (info.fullpath)
target = info.outdir / relPath;
2013-03-07 20:05:56 +00:00
else
target = info.outdir / relPath.filename();
2013-03-07 20:05:56 +00:00
// Create the directory hierarchy
2022-05-22 16:52:05 +00:00
std::filesystem::create_directories(target.parent_path());
2013-03-07 20:05:56 +00:00
2022-05-22 16:52:05 +00:00
std::filesystem::file_status s = std::filesystem::status(target.parent_path());
if (!std::filesystem::is_directory(s))
2013-03-07 20:05:56 +00:00
{
2022-09-22 18:26:05 +00:00
std::cout << "ERROR: " << Files::pathToUnicodeString(target.parent_path()) << " is not a directory."
<< std::endl;
2013-03-07 20:05:56 +00:00
return 3;
}
2022-05-22 16:52:05 +00:00
std::ofstream out(target, std::ios::binary);
2013-03-07 20:05:56 +00:00
// Write the file to disk
2022-09-22 18:26:05 +00:00
std::cout << "Extracting " << Files::pathToUnicodeString(info.extractfile) << " to "
<< Files::pathToUnicodeString(target) << std::endl;
2015-04-22 16:06:29 +00:00
out << stream->rdbuf();
2013-03-07 20:05:56 +00:00
out.close();
return 0;
}
2022-09-22 18:26:05 +00:00
template <typename File>
int extractAll(std::unique_ptr<File>& bsa, Arguments& info)
2013-03-07 20:05:56 +00:00
{
2022-09-22 18:26:05 +00:00
for (const auto& file : bsa->getList())
2020-06-10 06:30:37 +00:00
{
2021-04-07 16:58:46 +00:00
std::string extractPath(file.name());
2020-10-23 11:04:21 +00:00
Misc::StringUtils::replaceAll(extractPath, "\\", "/");
2013-03-07 20:05:56 +00:00
// Get the target path (the path the file will be extracted to)
auto target = info.outdir;
target /= Misc::StringUtils::stringToU8String(extractPath);
2013-03-07 20:05:56 +00:00
// Create the directory hierarchy
2022-05-22 16:52:05 +00:00
std::filesystem::create_directories(target.parent_path());
2013-03-07 20:05:56 +00:00
2022-05-22 16:52:05 +00:00
std::filesystem::file_status s = std::filesystem::status(target.parent_path());
if (!std::filesystem::is_directory(s))
2013-03-07 20:05:56 +00:00
{
std::cout << "ERROR: " << target.parent_path() << " is not a directory." << std::endl;
return 3;
}
// Get a stream for the file to extract
Files::IStreamPtr data = bsa->getFile(&file);
2022-05-22 16:52:05 +00:00
std::ofstream out(target, std::ios::binary);
2013-03-07 20:05:56 +00:00
// Write the file to disk
std::cout << "Extracting " << Files::pathToUnicodeString(target) << std::endl;
2015-04-22 16:06:29 +00:00
out << data->rdbuf();
2013-03-07 20:05:56 +00:00
out.close();
}
2013-03-04 23:45:25 +00:00
return 0;
}
2021-04-07 16:58:46 +00:00
2022-09-22 18:26:05 +00:00
template <typename File>
int add(std::unique_ptr<File>& bsa, Arguments& info)
2021-04-07 16:58:46 +00:00
{
2022-05-22 16:52:05 +00:00
std::fstream stream(info.addfile, std::ios_base::binary | std::ios_base::out | std::ios_base::in);
bsa->addFile(Files::pathToUnicodeString(info.addfile), stream);
2021-04-07 16:58:46 +00:00
return 0;
}
2022-09-22 18:26:05 +00:00
template <typename File>
int call(Arguments& info)
{
std::unique_ptr<File> bsa = std::make_unique<File>();
if (info.mode == "create")
{
bsa->open(info.filename);
return 0;
}
bsa->open(info.filename);
if (info.mode == "list")
return list(bsa, info);
else if (info.mode == "extract")
return extract(bsa, info);
else if (info.mode == "extractall")
return extractAll(bsa, info);
else if (info.mode == "add")
return add(bsa, info);
else
{
std::cout << "Unsupported mode. That is not supposed to happen." << std::endl;
return 1;
}
}
int main(int argc, char** argv)
{
try
{
Arguments info;
if (!parseOptions(argc, argv, info))
return 1;
// Open file
2023-10-09 03:59:28 +00:00
// TODO: add a version argument for this mode after compressed BSA writing is a thing
if (info.mode == "create")
return call<Bsa::BSAFile>(info);
2023-03-02 15:49:09 +00:00
Bsa::BsaVersion bsaVersion = Bsa::BSAFile::detectVersion(info.filename);
2023-03-02 15:49:09 +00:00
switch (bsaVersion)
{
2024-01-30 00:29:38 +00:00
case Bsa::BsaVersion::Unknown:
break;
case Bsa::BsaVersion::Uncompressed:
return call<Bsa::BSAFile>(info);
case Bsa::BsaVersion::Compressed:
2023-03-02 15:49:09 +00:00
return call<Bsa::CompressedBSAFile>(info);
2024-01-30 00:29:38 +00:00
case Bsa::BsaVersion::BA2GNRL:
2023-03-02 15:49:09 +00:00
return call<Bsa::BA2GNRLFile>(info);
2024-01-30 00:29:38 +00:00
case Bsa::BsaVersion::BA2DX10:
2023-03-02 15:51:04 +00:00
return call<Bsa::BA2DX10File>(info);
2023-03-02 15:49:09 +00:00
}
2024-01-30 00:29:38 +00:00
throw std::runtime_error("Unrecognised BSA archive");
}
catch (std::exception& e)
{
std::cerr << "ERROR reading BSA archive\nDetails:\n" << e.what() << std::endl;
return 2;
}
}