openmw-tes3coop/apps/esmtool/esmtool.cpp
2012-09-17 11:37:50 +04:00

922 lines
28 KiB
C++

#include <iostream>
#include <list>
#include <map>
#include <boost/program_options.hpp>
#include <components/esm/esm_reader.hpp>
#include <components/esm/esm_writer.hpp>
#include <components/esm/records.hpp>
#define ESMTOOL_VERSION 1.1
using namespace std;
using namespace ESM;
// Create a local alias for brevity
namespace bpo = boost::program_options;
struct ESMData
{
std::string author;
string description;
int version;
int type;
ESMReader::MasterList masters;
list<Record*> records;
map<Record*, list<CellRef> > cellRefs;
};
// Based on the legacy struct
struct Arguments
{
unsigned int raw_given;
unsigned int quiet_given;
unsigned int loadcells_given;
string mode;
string encoding;
string filename;
string outname;
ESMData data;
ESMReader reader;
ESMWriter writer;
};
bool parseOptions (int argc, char** argv, Arguments &info)
{
bpo::options_description desc("Inspect and extract from Morrowind ES files (ESM, ESP, ESS)\nSyntax: esmtool [options] mode infile [outfile]\nAllowed modes:\n dump\t Dumps all readable data from the input file.\n clone\t Clones the input file to the output file.\n comp\t Compares the given files.\n\nAllowed options");
desc.add_options()
("help,h", "print help message.")
("version,v", "print version information and quit.")
("raw,r", "Show an unformattet list of all records and subrecords.")
("quiet,q", "Supress all record information. Useful for speed tests.")
("loadcells,C", "Browse through contents of all cells.")
( "encoding,e", bpo::value<string>(&(info.encoding))->
default_value("win1252"),
"Character encoding used in ESMTool:\n"
"\n\twin1250 - Central and Eastern European such as Polish, Czech, Slovak, Hungarian, Slovene, Bosnian, Croatian, Serbian (Latin script), Romanian and Albanian languages\n"
"\n\twin1251 - Cyrillic alphabet such as Russian, Bulgarian, Serbian Cyrillic and other languages\n"
"\n\twin1252 - Western European (Latin) alphabet, used by default")
;
string finalText = "\nIf no option is given, the default action is to parse all records in the archive\nand display diagnostic information.";
// input-file is hidden and used as a positional argument
bpo::options_description hidden("Hidden Options");
hidden.add_options()
( "mode,m", bpo::value<string>(), "esmtool mode")
( "input-file,i", bpo::value< vector<string> >(), "input file")
;
bpo::positional_options_description p;
p.add("mode", 1).add("input-file", 2);
// there might be a better way to do this
bpo::options_description all;
all.add(desc).add(hidden);
bpo::parsed_options valid_opts = bpo::command_line_parser(argc, argv)
.options(all).positional(p).run();
bpo::variables_map variables;
bpo::store(valid_opts, variables);
bpo::notify(variables);
if (variables.count ("help"))
{
cout << desc << finalText << endl;
return false;
}
if (variables.count ("version"))
{
cout << "ESMTool version " << ESMTOOL_VERSION << endl;
return false;
}
if (!variables.count("mode"))
{
cout << "No mode specified!" << endl << endl
<< desc << finalText << endl;
return false;
}
info.mode = variables["mode"].as<string>();
if (!(info.mode == "dump" || info.mode == "clone" || info.mode == "comp"))
{
cout << endl << "ERROR: invalid mode \"" << info.mode << "\"" << endl << endl
<< desc << finalText << endl;
return false;
}
if ( !variables.count("input-file") )
{
cout << "\nERROR: missing ES file\n\n";
cout << desc << finalText << endl;
return false;
}
// handling gracefully the user adding multiple files
/* if (variables["input-file"].as< vector<string> >().size() > 1)
{
cout << "\nERROR: more than one ES file specified\n\n";
cout << desc << finalText << endl;
return false;
}*/
info.filename = variables["input-file"].as< vector<string> >()[0];
if (variables["input-file"].as< vector<string> >().size() > 1)
info.outname = variables["input-file"].as< vector<string> >()[1];
info.raw_given = variables.count ("raw");
info.quiet_given = variables.count ("quiet");
info.loadcells_given = variables.count ("loadcells");
// Font encoding settings
info.encoding = variables["encoding"].as<string>();
if (info.encoding == "win1250")
{
cout << "Using Central and Eastern European font encoding." << endl;
}
else if (info.encoding == "win1251")
{
cout << "Using Cyrillic font encoding." << endl;
}
else
{
if(info.encoding != "win1252")
{
cout << info.encoding << " is not a valid encoding option." << endl;
info.encoding = "win1252";
}
cout << "Using default (English) font encoding." << endl;
}
return true;
}
void printRaw(ESMReader &esm);
void loadCell(Cell &cell, ESMReader &esm, Arguments& info);
int load(Arguments& info);
int clone(Arguments& info);
int comp(Arguments& info);
int main(int argc, char**argv)
{
Arguments info;
if(!parseOptions (argc, argv, info))
return 1;
if (info.mode == "dump")
return load(info);
else if (info.mode == "clone")
return clone(info);
else if (info.mode == "comp")
return comp(info);
else
{
cout << "Invalid or no mode specified, dying horribly. Have a nice day." << endl;
return 1;
}
return 0;
}
void loadCell(Cell &cell, ESMReader &esm, Arguments& info)
{
bool quiet = (info.quiet_given || info.mode == "clone");
bool save = (info.mode == "clone");
// Skip back to the beginning of the reference list
cell.restore(esm);
// Loop through all the references
CellRef ref;
if(!quiet) cout << " References:\n";
while(cell.getNextRef(esm, ref))
{
if (save)
info.data.cellRefs[&cell].push_back(ref);
if(quiet) continue;
cout << " Refnum: " << ref.mRefnum << endl;
cout << " ID: '" << ref.mRefID << "'\n";
cout << " Owner: '" << ref.mOwner << "'\n";
cout << " INTV: " << ref.mIntv << " NAM9: " << ref.mIntv << endl;
}
}
void printRaw(ESMReader &esm)
{
while(esm.hasMoreRecs())
{
NAME n = esm.getRecName();
cout << "Record: " << n.toString() << endl;
esm.getRecHeader();
while(esm.hasMoreSubs())
{
uint64_t offs = esm.getOffset();
esm.getSubName();
esm.skipHSub();
n = esm.retSubName();
cout << " " << n.toString() << " - " << esm.getSubSize()
<< " bytes @ 0x" << hex << offs << "\n";
}
}
}
int load(Arguments& info)
{
ESMReader& esm = info.reader;
esm.setEncoding(info.encoding);
string filename = info.filename;
cout << "Loading file: " << filename << endl;
list<int> skipped;
try {
if(info.raw_given && info.mode == "dump")
{
cout << "RAW file listing:\n";
esm.openRaw(filename);
printRaw(esm);
return 0;
}
bool quiet = (info.quiet_given || info.mode == "clone");
bool loadCells = (info.loadcells_given || info.mode == "clone");
bool save = (info.mode == "clone");
esm.open(filename);
info.data.author = esm.getAuthor();
info.data.description = esm.getDesc();
info.data.masters = esm.getMasters();
info.data.version = esm.getVer();
info.data.type = esm.getType();
if (!quiet)
{
cout << "Author: " << esm.getAuthor() << endl
<< "Description: " << esm.getDesc() << endl
<< "File format version: " << esm.getFVer() << endl
<< "Special flag: " << esm.getSpecial() << endl;
ESMReader::MasterList m = esm.getMasters();
if (!m.empty())
{
cout << "Masters:" << endl;
for(unsigned int i=0;i<m.size();i++)
cout << " " << m[i].name << ", " << m[i].size << " bytes" << endl;
}
}
// Loop through all records
while(esm.hasMoreRecs())
{
NAME n = esm.getRecName();
uint32_t flags;
esm.getRecHeader(flags);
string id = esm.getHNOString("NAME");
if(!quiet)
cout << "\nRecord: " << n.toString()
<< " '" << id << "'\n";
Record* rec = NULL;
switch(n.val)
{
case REC_ACTI:
{
rec = new Activator();
Activator& ac = *(Activator*)rec;
ac.load(esm);
if(quiet) break;
cout << " Name: " << ac.mName << endl;
cout << " Mesh: " << ac.mModel << endl;
cout << " Script: " << ac.mScript << endl;
break;
}
case REC_ALCH:
{
rec = new Potion();
Potion& p = *(Potion*)rec;
p.load(esm);
if(quiet) break;
cout << " Name: " << p.mName << endl;
break;
}
case REC_APPA:
{
rec = new Apparatus();
Apparatus& p = *(Apparatus*)rec;
p.load(esm);
if(quiet) break;
cout << " Name: " << p.mName << endl;
break;
}
case REC_ARMO:
{
rec = new Armor();
Armor& am = *(Armor*)rec;
am.load(esm);
if(quiet) break;
cout << " Name: " << am.mName << endl;
cout << " Mesh: " << am.mModel << endl;
cout << " Icon: " << am.mIcon << endl;
cout << " Script: " << am.mScript << endl;
cout << " Enchantment: " << am.mEnchant << endl;
cout << " Type: " << am.mData.mType << endl;
cout << " Weight: " << am.mData.mWeight << endl;
break;
}
case REC_BODY:
{
rec = new BodyPart();
BodyPart& bp = *(BodyPart*)rec;
bp.load(esm);
if(quiet) break;
cout << " Name: " << bp.mName << endl;
cout << " Mesh: " << bp.mModel << endl;
break;
}
case REC_BOOK:
{
rec = new Book();
Book& b = *(Book*)rec;
b.load(esm);
if(quiet) break;
cout << " Name: " << b.mName << endl;
cout << " Mesh: " << b.mModel << endl;
break;
}
case REC_BSGN:
{
rec = new BirthSign();
BirthSign& bs = *(BirthSign*)rec;
bs.load(esm);
if(quiet) break;
cout << " Name: " << bs.mName << endl;
cout << " Texture: " << bs.mTexture << endl;
cout << " Description: " << bs.mDescription << endl;
break;
}
case REC_CELL:
{
rec = new Cell();
Cell& b = *(Cell*)rec;
b.load(esm);
if(!quiet)
{
cout << " Name: " << b.mName << endl;
cout << " Region: " << b.mRegion << endl;
}
if(loadCells)
loadCell(b, esm, info);
break;
}
case REC_CLAS:
{
rec = new Class();
Class& b = *(Class*)rec;
b.load(esm);
if(quiet) break;
cout << " Name: " << b.mName << endl;
cout << " Description: " << b.mDescription << endl;
break;
}
case REC_CLOT:
{
rec = new Clothing();
Clothing& b = *(Clothing*)rec;
b.load(esm);
if(quiet) break;
cout << " Name: " << b.mName << endl;
break;
}
case REC_CONT:
{
rec = new Container();
Container& b = *(Container*)rec;
b.load(esm);
if(quiet) break;
cout << " Name: " << b.mName << endl;
break;
}
case REC_CREA:
{
rec = new Creature();
Creature& b = *(Creature*)rec;
b.load(esm);
if(quiet) break;
cout << " Name: " << b.mName << endl;
break;
}
case REC_DIAL:
{
rec = new Dialogue();
Dialogue& b = *(Dialogue*)rec;
b.load(esm);
break;
}
case REC_DOOR:
{
rec = new Door();
Door& d = *(Door*)rec;
d.load(esm);
if(quiet) break;
cout << " Name: " << d.mName << endl;
cout << " Mesh: " << d.mModel << endl;
cout << " Script: " << d.mScript << endl;
cout << " OpenSound: " << d.mOpenSound << endl;
cout << " CloseSound: " << d.mCloseSound << endl;
break;
}
case REC_ENCH:
{
rec = new Enchantment();
Enchantment& b = *(Enchantment*)rec;
b.load(esm);
break;
}
case REC_FACT:
{
rec = new Faction();
Faction& f = *(Faction*)rec;
f.load(esm);
if (quiet) break;
cout << " Name: " << f.mName << endl
<< " Attr1: " << f.mData.mAttribute1 << endl
<< " Attr2: " << f.mData.mAttribute2 << endl
<< " Hidden: " << f.mData.mIsHidden << endl;
break;
}
case REC_GLOB:
{
rec = new Global();
Global& g = *(Global*)rec;
g.load(esm);
break;
}
case REC_GMST:
{
rec = new GameSetting();
GameSetting& b = *(GameSetting*)rec;
b.setId(id);
b.load(esm);
if(quiet) break;
cout << " Value: ";
if(b.mType == VT_String)
cout << "'" << b.mStr << "' (string)";
else if(b.mType == VT_Float)
cout << b.mF << " (float)";
else if(b.mType == VT_Int)
cout << b.mI << " (int)";
cout << "\n Dirty: " << b.mDirty << endl;
break;
}
case REC_INFO:
{
rec = new DialInfo();
DialInfo& p = *(DialInfo*)rec;
p.load(esm);
if(quiet) break;
cout << " Id: " << p.getId() << endl;
cout << " Text: " << p.mResponse << endl;
break;
}
case REC_INGR:
{
rec = new Ingredient();
Ingredient& i = *(Ingredient*)rec;
i.load(esm);
if (quiet) break;
cout << " Name: " << i.mName << endl
<< " Weight: " << i.mData.mWeight << endl
<< " Value: " << i.mData.mValue << endl;
break;
}
case REC_LAND:
{
rec = new Land();
Land& l = *(Land*)rec;
l.load(esm);
if (quiet) break;
cout << " Coords: [" << l.mX << "," << l.mY << "]" << endl;
break;
}
case REC_LEVI:
{
rec = new ItemLevList();
ItemLevList& l = *(ItemLevList*)rec;
l.load(esm);
if (quiet) break;
cout << " Number of items: " << l.mList.size() << endl;
break;
}
case REC_LEVC:
{
rec = new CreatureLevList();
CreatureLevList& l = *(CreatureLevList*)rec;
l.load(esm);
if (quiet) break;
cout << " Number of items: " << l.mList.size() << endl;
break;
}
case REC_LIGH:
{
rec = new Light();
Light& l = *(Light*)rec;
l.load(esm);
if (quiet) break;
cout << " Name: " << l.mName << endl
<< " Weight: " << l.mData.mWeight << endl
<< " Value: " << l.mData.mValue << endl;
break;
}
case REC_LOCK:
{
rec = new Tool();
Tool& l = *(Tool*)rec;
l.load(esm);
if (quiet) break;
cout << " Name: " << l.mName << endl
<< " Quality: " << l.mData.mQuality << endl;
break;
}
case REC_LTEX:
{
rec = new LandTexture();
LandTexture& t = *(LandTexture*)rec;
t.load(esm);
if (quiet) break;
cout << " Id: " << t.getId() << endl
<< " Texture: " << t.mTexture << endl;
break;
}
case REC_MISC:
{
rec = new Miscellaneous();
Miscellaneous& m = *(Miscellaneous*)rec;
m.load(esm);
if (quiet) break;
cout << " Name: " << m.mName << endl
<< " Value: " << m.mData.mValue << endl;
break;
}
case REC_MGEF:
{
rec = new MagicEffect();
MagicEffect& m = *(MagicEffect*)rec;
m.load(esm);
if (quiet) break;
cout << " Index: " << m.mIndex << endl
<< " " << (m.mData.mFlags & MagicEffect::Negative ? "Negative" : "Positive") << endl;
break;
}
case REC_NPC_:
{
rec = new NPC();
NPC& n = *(NPC*)rec;
n.load(esm);
if (quiet) break;
cout << " Name: " << n.mName << endl
<< " Race: " << n.mRace << endl;
break;
}
case REC_PGRD:
{
rec = new Pathgrid();
Pathgrid& p = *(Pathgrid*)rec;
p.load(esm);
if (quiet) break;
cout << " Cell: " << p.mCell << endl
<< " Point count: " << p.mPoints.size() << endl
<< " Edge count: " << p.mEdges.size() << endl;
break;
}
case REC_PROB:
{
rec = new Probe();
Probe& r = *(Probe*)rec;
r.load(esm);
if (quiet) break;
cout << " Name: " << r.mName << endl
<< " Quality: " << r.mData.mQuality << endl;
break;
}
case REC_RACE:
{
rec = new Race();
Race& r = *(Race*)rec;
r.load(esm);
if (quiet) break;
cout << " Name: " << r.mName << endl
<< " Length: " << r.mData.mHeight.mMale << "m " << r.mData.mHeight.mFemale << "f" << endl;
break;
}
case REC_REGN:
{
rec = new Region();
Region& r = *(Region*)rec;
r.load(esm);
if (quiet) break;
cout << " Name: " << r.mName << endl;
break;
}
case REC_REPA:
{
rec = new Repair();
Repair& r = *(Repair*)rec;
r.load(esm);
if (quiet) break;
cout << " Name: " << r.mName << endl
<< " Quality: " << r.mData.mQuality << endl;
break;
}
case REC_SCPT:
{
rec = new Script();
Script& s = *(Script*)rec;
s.load(esm);
if (quiet) break;
cout << " Name: " << s.mData.mName.toString() << endl;
break;
}
case REC_SKIL:
{
rec = new Skill();
Skill& s = *(Skill*)rec;
s.load(esm);
if (quiet) break;
cout << " ID: " << s.mIndex << endl
<< " Type: " << (s.mData.mSpecialization == 0 ? "Combat" : (s.mData.mSpecialization == 1 ? "Magic" : "Stealth")) << endl;
break;
}
case REC_SNDG:
{
rec = new SoundGenerator();
SoundGenerator& s = *(SoundGenerator*)rec;
s.load(esm);
if (quiet) break;
cout << " Creature: " << s.mCreature << endl
<< " Sound: " << s.mSound << endl;
break;
}
case REC_SOUN:
{
rec = new Sound();
Sound& d = *(Sound*)rec;
d.load(esm);
if(quiet) break;
cout << " Sound: " << d.mSound << endl;
cout << " Volume: " << (int)d.mData.mVolume << endl;
break;
}
case REC_SPEL:
{
rec = new Spell();
Spell& s = *(Spell*)rec;
s.load(esm);
if(quiet) break;
cout << " Name: " << s.mName << endl;
break;
}
case REC_STAT:
{
rec = new Static();
Static& s = *(Static*)rec;
s.load(esm);
if (quiet) break;
cout << " Model: " << s.mModel << endl;
break;
}
case REC_WEAP:
{
rec = new Weapon();
Weapon& w = *(Weapon*)rec;
w.load(esm);
if (quiet) break;
cout << " Name: " << w.mName << endl
<< " Chop: " << w.mData.mChop[0] << "-" << w.mData.mChop[1] << endl
<< " Slash: " << w.mData.mSlash[0] << "-" << w.mData.mSlash[1] << endl
<< " Thrust: " << w.mData.mThrust[0] << "-" << w.mData.mThrust[1] << endl
<< " Value: " << w.mData.mValue << endl;
break;
}
default:
if (find(skipped.begin(), skipped.end(), n.val) == skipped.end())
{
cout << "Skipping " << n.toString() << " records." << endl;
skipped.push_back(n.val);
}
esm.skipRecord();
if(quiet) break;
cout << " Skipping\n";
}
if (rec != NULL)
{
rec->setId(id);
rec->setFlags((int)flags);
if (save)
info.data.records.push_back(rec);
else
delete rec;
}
}
} catch(exception &e)
{
cout << "\nERROR:\n\n " << e.what() << endl;
for (list<Record*>::iterator it = info.data.records.begin(); it != info.data.records.end();)
{
delete *it;
info.data.records.erase(it++);
}
return 1;
}
return 0;
}
#include <iomanip>
int clone(Arguments& info)
{
if (info.outname.empty())
{
cout << "You need to specify an output name" << endl;
return 1;
}
if (load(info) != 0)
{
cout << "Failed to load, aborting." << endl;
return 1;
}
int recordCount = info.data.records.size();
int digitCount = 1; // For a nicer output
if (recordCount > 9) ++digitCount;
if (recordCount > 99) ++digitCount;
if (recordCount > 999) ++digitCount;
if (recordCount > 9999) ++digitCount;
if (recordCount > 99999) ++digitCount;
if (recordCount > 999999) ++digitCount;
cout << "Loaded " << recordCount << " records:" << endl << endl;
map<string, int> records;
for (list<Record*>::iterator it = info.data.records.begin(); it != info.data.records.end(); ++it)
{
Record* rec = *it;
NAME n;
n.val = rec->getName();
records[n.toString()]++;
}
int i = 0;
for (map<string,int>::iterator it = records.begin(); it != records.end(); ++it)
{
string n = it->first;
float amount = it->second;
cout << setw(digitCount) << amount << " " << n << " ";
if (++i % 3 == 0)
cout << endl;
}
if (i % 3 != 0)
cout << endl;
cout << endl << "Saving records to: " << info.outname << "..." << endl;
ESMWriter& esm = info.writer;
esm.setEncoding(info.encoding);
esm.setAuthor(info.data.author);
esm.setDescription(info.data.description);
esm.setVersion(info.data.version);
esm.setType(info.data.type);
for (ESMReader::MasterList::iterator it = info.data.masters.begin(); it != info.data.masters.end(); ++it)
esm.addMaster(it->name, it->size);
fstream save(info.outname.c_str(), fstream::out | fstream::binary);
esm.save(save);
int saved = 0;
for (list<Record*>::iterator it = info.data.records.begin(); it != info.data.records.end() && i > 0; ++it)
{
Record* rec = *it;
NAME n;
n.val = rec->getName();
esm.startRecord(n.toString(), rec->getFlags());
string id = rec->getId();
if (n.val == REC_GLOB || n.val == REC_CLAS || n.val == REC_FACT || n.val == REC_RACE || n.val == REC_SOUN
|| n.val == REC_REGN || n.val == REC_BSGN || n.val == REC_LTEX || n.val == REC_STAT || n.val == REC_DOOR
|| n.val == REC_MISC || n.val == REC_WEAP || n.val == REC_CONT || n.val == REC_SPEL || n.val == REC_CREA
|| n.val == REC_BODY || n.val == REC_LIGH || n.val == REC_ENCH || n.val == REC_NPC_ || n.val == REC_ARMO
|| n.val == REC_CLOT || n.val == REC_REPA || n.val == REC_ACTI || n.val == REC_APPA || n.val == REC_LOCK
|| n.val == REC_PROB || n.val == REC_INGR || n.val == REC_BOOK || n.val == REC_ALCH || n.val == REC_LEVI
|| n.val == REC_LEVC)
esm.writeHNCString("NAME", id);
else if (n.val == REC_CELL)
esm.writeHNString("NAME", id);
else
esm.writeHNOString("NAME", id);
rec->save(esm);
if (n.val == REC_CELL && !info.data.cellRefs[rec].empty())
{
list<CellRef>& refs = info.data.cellRefs[rec];
for (list<CellRef>::iterator it = refs.begin(); it != refs.end(); ++it)
{
it->save(esm);
}
}
esm.endRecord(n.toString());
saved++;
int perc = (saved / (float)recordCount)*100;
if (perc % 10 == 0)
{
cerr << "\r" << perc << "%";
}
}
cout << "\rDone!" << endl;
esm.close();
save.close();
return 0;
}
int comp(Arguments& info)
{
if (info.filename.empty() || info.outname.empty())
{
cout << "You need to specify two input files" << endl;
return 1;
}
Arguments fileOne;
Arguments fileTwo;
fileOne.raw_given = 0;
fileTwo.raw_given = 0;
fileOne.mode = "clone";
fileTwo.mode = "clone";
fileOne.encoding = info.encoding;
fileTwo.encoding = info.encoding;
fileOne.filename = info.filename;
fileTwo.filename = info.outname;
if (load(fileOne) != 0)
{
cout << "Failed to load " << info.filename << ", aborting comparison." << endl;
return 1;
}
if (load(fileTwo) != 0)
{
cout << "Failed to load " << info.outname << ", aborting comparison." << endl;
return 1;
}
if (fileOne.data.records.size() != fileTwo.data.records.size())
{
cout << "Not equal, different amount of records." << endl;
return 1;
}
return 0;
}