mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-11-04 02:26:41 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			288 lines
		
	
	
	
		
			8.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			288 lines
		
	
	
	
		
			8.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
#include <iostream>
 | 
						|
#include <vector>
 | 
						|
#include <exception>
 | 
						|
 | 
						|
#include <boost/program_options.hpp>
 | 
						|
#include <boost/filesystem.hpp>
 | 
						|
#include <boost/filesystem/fstream.hpp>
 | 
						|
 | 
						|
#include <components/bsa/bsa_file.hpp>
 | 
						|
 | 
						|
#define BSATOOL_VERSION 1.1
 | 
						|
 | 
						|
// Create local aliases for brevity
 | 
						|
namespace bpo = boost::program_options;
 | 
						|
namespace bfs = boost::filesystem;
 | 
						|
 | 
						|
struct Arguments
 | 
						|
{
 | 
						|
    std::string mode;
 | 
						|
    std::string filename;
 | 
						|
    std::string extractfile;
 | 
						|
    std::string outdir;
 | 
						|
 | 
						|
    bool longformat;
 | 
						|
    bool fullpath;
 | 
						|
};
 | 
						|
 | 
						|
void replaceAll(std::string& str, const std::string& needle, const std::string& substitute)
 | 
						|
{
 | 
						|
    int pos = str.find(needle);
 | 
						|
    while(pos != -1)
 | 
						|
    {
 | 
						|
        str.replace(pos, needle.size(), substitute);
 | 
						|
        pos = str.find(needle);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
bool parseOptions (int argc, char** argv, Arguments &info)
 | 
						|
{
 | 
						|
    bpo::options_description desc("Inspect and extract files from Bethesda BSA archives\n\n"
 | 
						|
            "Usages:\n"
 | 
						|
            "  bsatool list [-l] archivefile\n"
 | 
						|
            "      List the files presents in the input archive.\n\n"
 | 
						|
            "  bsatool extract [-f] archivefile [file_to_extract] [output_directory]\n"
 | 
						|
            "      Extract a file from the input archive.\n\n"
 | 
						|
            "  bsatool extractall archivefile [output_directory]\n"
 | 
						|
            "      Extract all files from the input archive.\n\n"
 | 
						|
            "Allowed options");
 | 
						|
 | 
						|
    desc.add_options()
 | 
						|
        ("help,h", "print help message.")
 | 
						|
        ("version,v", "print version information and quit.")
 | 
						|
        ("long,l", "Include extra information in archive listing.")
 | 
						|
        ("full-path,f", "Create diretory hierarchy on file extraction "
 | 
						|
         "(always true for extractall).")
 | 
						|
        ;
 | 
						|
 | 
						|
    // input-file is hidden and used as a positional argument
 | 
						|
    bpo::options_description hidden("Hidden Options");
 | 
						|
 | 
						|
    hidden.add_options()
 | 
						|
        ( "mode,m", bpo::value<std::string>(), "bsatool mode")
 | 
						|
        ( "input-file,i", bpo::value< std::vector<std::string> >(), "input file")
 | 
						|
        ;
 | 
						|
 | 
						|
    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
 | 
						|
    {
 | 
						|
        bpo::parsed_options valid_opts = bpo::command_line_parser(argc, argv)
 | 
						|
            .options(all).positional(p).run();
 | 
						|
        bpo::store(valid_opts, variables);
 | 
						|
    }
 | 
						|
    catch(std::exception &e)
 | 
						|
    {
 | 
						|
        std::cout << "ERROR parsing arguments: " << e.what() << "\n\n"
 | 
						|
            << desc << std::endl;
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    bpo::notify(variables);
 | 
						|
 | 
						|
    if (variables.count ("help"))
 | 
						|
    {
 | 
						|
        std::cout << desc << std::endl;
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
    if (variables.count ("version"))
 | 
						|
    {
 | 
						|
        std::cout << "BSATool version " << BSATOOL_VERSION << std::endl;
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
    if (!variables.count("mode"))
 | 
						|
    {
 | 
						|
        std::cout << "ERROR: no mode specified!\n\n"
 | 
						|
            << desc << std::endl;
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    info.mode = variables["mode"].as<std::string>();
 | 
						|
    if (!(info.mode == "list" || info.mode == "extract" || info.mode == "extractall"))
 | 
						|
    {
 | 
						|
        std::cout << std::endl << "ERROR: invalid mode \"" << info.mode << "\"\n\n"
 | 
						|
            << desc << std::endl;
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    if (!variables.count("input-file"))
 | 
						|
    {
 | 
						|
        std::cout << "\nERROR: missing BSA archive\n\n"
 | 
						|
            << desc << std::endl;
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
    info.filename = variables["input-file"].as< std::vector<std::string> >()[0];
 | 
						|
 | 
						|
    // Default output to the working directory
 | 
						|
    info.outdir = ".";
 | 
						|
 | 
						|
    if (info.mode == "extract")
 | 
						|
    {
 | 
						|
        if (variables["input-file"].as< std::vector<std::string> >().size() < 2)
 | 
						|
        {
 | 
						|
            std::cout << "\nERROR: file to extract unspecified\n\n"
 | 
						|
                << desc << std::endl;
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
        if (variables["input-file"].as< std::vector<std::string> >().size() > 1)
 | 
						|
            info.extractfile = variables["input-file"].as< std::vector<std::string> >()[1];
 | 
						|
        if (variables["input-file"].as< std::vector<std::string> >().size() > 2)
 | 
						|
            info.outdir = variables["input-file"].as< std::vector<std::string> >()[2];
 | 
						|
    }
 | 
						|
    else if (variables["input-file"].as< std::vector<std::string> >().size() > 1)
 | 
						|
        info.outdir = variables["input-file"].as< std::vector<std::string> >()[1];
 | 
						|
 | 
						|
    info.longformat = variables.count("long");
 | 
						|
    info.fullpath = variables.count("full-path");
 | 
						|
 | 
						|
    return true;
 | 
						|
}
 | 
						|
 | 
						|
int list(Bsa::BSAFile& bsa, Arguments& info);
 | 
						|
int extract(Bsa::BSAFile& bsa, Arguments& info);
 | 
						|
int extractAll(Bsa::BSAFile& bsa, Arguments& info);
 | 
						|
 | 
						|
int main(int argc, char** argv)
 | 
						|
{
 | 
						|
    Arguments info;
 | 
						|
    if(!parseOptions (argc, argv, info))
 | 
						|
        return 1;
 | 
						|
 | 
						|
    // Open file
 | 
						|
    Bsa::BSAFile bsa;
 | 
						|
    try
 | 
						|
    {
 | 
						|
        bsa.open(info.filename);
 | 
						|
    }
 | 
						|
    catch(std::exception &e)
 | 
						|
    {
 | 
						|
        std::cout << "ERROR reading BSA archive '" << info.filename
 | 
						|
            << "'\nDetails:\n" << e.what() << std::endl;
 | 
						|
        return 2;
 | 
						|
    }
 | 
						|
 | 
						|
    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
 | 
						|
    {
 | 
						|
        std::cout << "Unsupported mode. That is not supposed to happen." << std::endl;
 | 
						|
        return 1;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
int list(Bsa::BSAFile& bsa, Arguments& info)
 | 
						|
{
 | 
						|
    // List all files
 | 
						|
    const Bsa::BSAFile::FileList &files = bsa.getList();
 | 
						|
    for(int i=0; i<files.size(); i++)
 | 
						|
    {
 | 
						|
        if(info.longformat)
 | 
						|
        {
 | 
						|
            // Long format
 | 
						|
            std::cout << std::setw(50) << std::left << files[i].name;
 | 
						|
            std::cout << std::setw(8) << std::left << std::dec << files[i].fileSize;
 | 
						|
            std::cout << "@ 0x" << std::hex << files[i].offset << std::endl;
 | 
						|
        }
 | 
						|
        else
 | 
						|
            std::cout << files[i].name << std::endl;
 | 
						|
    }
 | 
						|
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
int extract(Bsa::BSAFile& bsa, Arguments& info)
 | 
						|
{
 | 
						|
    std::string archivePath = info.extractfile;
 | 
						|
    replaceAll(archivePath, "/", "\\");
 | 
						|
 | 
						|
    std::string extractPath = info.extractfile;
 | 
						|
    replaceAll(extractPath, "\\", "/");
 | 
						|
 | 
						|
    if (!bsa.exists(archivePath.c_str()))
 | 
						|
    {
 | 
						|
        std::cout << "ERROR: file '" << archivePath << "' not found\n";
 | 
						|
        std::cout << "In archive: " << info.filename << std::endl;
 | 
						|
        return 3;
 | 
						|
    }
 | 
						|
 | 
						|
    // Get the target path (the path the file will be extracted to)
 | 
						|
    bfs::path relPath (extractPath);
 | 
						|
    bfs::path outdir (info.outdir);
 | 
						|
 | 
						|
    bfs::path target;
 | 
						|
    if (info.fullpath)
 | 
						|
        target = outdir / relPath;
 | 
						|
    else
 | 
						|
        target = outdir / relPath.filename();
 | 
						|
 | 
						|
    // Create the directory hierarchy
 | 
						|
    bfs::create_directories(target.parent_path());
 | 
						|
 | 
						|
    bfs::file_status s = bfs::status(target.parent_path());
 | 
						|
    if (!bfs::is_directory(s))
 | 
						|
    {
 | 
						|
        std::cout << "ERROR: " << target.parent_path() << " is not a directory." << std::endl;
 | 
						|
        return 3;
 | 
						|
    }
 | 
						|
 | 
						|
    // Get a stream for the file to extract
 | 
						|
    Ogre::DataStreamPtr data = bsa.getFile(archivePath.c_str());
 | 
						|
    bfs::ofstream out(target, std::ios::binary);
 | 
						|
 | 
						|
    // Write the file to disk
 | 
						|
    std::cout << "Extracting " << info.extractfile << " to " << target << std::endl;
 | 
						|
    out.write(data->getAsString().c_str(), data->size());
 | 
						|
    out.close();
 | 
						|
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
int extractAll(Bsa::BSAFile& bsa, Arguments& info)
 | 
						|
{
 | 
						|
    // Get the list of files present in the archive
 | 
						|
    Bsa::BSAFile::FileList list = bsa.getList();
 | 
						|
 | 
						|
    // Iter on the list
 | 
						|
    for(Bsa::BSAFile::FileList::iterator it = list.begin(); it != list.end(); ++it) {
 | 
						|
        const char* archivePath = it->name;
 | 
						|
 | 
						|
        std::string extractPath (archivePath);
 | 
						|
        replaceAll(extractPath, "\\", "/");
 | 
						|
 | 
						|
        // Get the target path (the path the file will be extracted to)
 | 
						|
        bfs::path target (info.outdir);
 | 
						|
        target /= extractPath;
 | 
						|
 | 
						|
        // Create the directory hierarchy
 | 
						|
        bfs::create_directories(target.parent_path());
 | 
						|
 | 
						|
        bfs::file_status s = bfs::status(target.parent_path());
 | 
						|
        if (!bfs::is_directory(s))
 | 
						|
        {
 | 
						|
            std::cout << "ERROR: " << target.parent_path() << " is not a directory." << std::endl;
 | 
						|
            return 3;
 | 
						|
        }
 | 
						|
 | 
						|
        // Get a stream for the file to extract
 | 
						|
        // (inefficient because getFile iter on the list again)
 | 
						|
        Ogre::DataStreamPtr data = bsa.getFile(archivePath);
 | 
						|
        bfs::ofstream out(target, std::ios::binary);
 | 
						|
 | 
						|
        // Write the file to disk
 | 
						|
        std::cout << "Extracting " << target << std::endl;
 | 
						|
        out.write(data->getAsString().c_str(), data->size());
 | 
						|
        out.close();
 | 
						|
    }
 | 
						|
 | 
						|
    return 0;
 | 
						|
}
 |