|
|
|
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
|
|
|
// Copyright (C) 2010 Winch Gate Property Limited
|
|
|
|
//
|
|
|
|
// This source file has been modified by the following contributors:
|
|
|
|
// Copyright (C) 2014-2019 Jan BOON (Kaetemi) <jan.boon@kaetemi.be>
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Affero General Public License as
|
|
|
|
// published by the Free Software Foundation, either version 3 of the
|
|
|
|
// License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
#include "nel/misc/types_nl.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#ifdef NL_OS_WINDOWS
|
|
|
|
# include <io.h>
|
|
|
|
# include <direct.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
#include "nel/misc/debug.h"
|
|
|
|
#include "nel/misc/file.h"
|
|
|
|
#include "nel/misc/path.h"
|
|
|
|
#include "nel/misc/algo.h"
|
|
|
|
#include "nel/misc/common.h"
|
|
|
|
#include "nel/misc/streamed_package.h"
|
|
|
|
#include "nel/misc/seven_zip.h"
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using namespace NLMISC;
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
class CWildCard
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
string Expression;
|
|
|
|
bool Not;
|
|
|
|
};
|
|
|
|
std::vector<CWildCard> WildCards;
|
|
|
|
|
|
|
|
std::string SourceDirectory;
|
|
|
|
std::string PackageFileName;
|
|
|
|
std::string StreamDirectory;
|
|
|
|
|
|
|
|
CStreamedPackage Package;
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
bool keepFile (const char *fileName)
|
|
|
|
{
|
|
|
|
uint i;
|
|
|
|
bool ifPresent = false;
|
|
|
|
bool ifTrue = false;
|
|
|
|
string file = toLowerAscii(CFile::getFilename (fileName));
|
|
|
|
for (i=0; i<WildCards.size(); i++)
|
|
|
|
{
|
|
|
|
if (WildCards[i].Not)
|
|
|
|
{
|
|
|
|
// One ifnot condition met and the file is not added
|
|
|
|
if (testWildCard(file.c_str(), WildCards[i].Expression.c_str()))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ifPresent = true;
|
|
|
|
ifTrue |= testWildCard(file.c_str(), WildCards[i].Expression.c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return !ifPresent || ifTrue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
void usage()
|
|
|
|
{
|
|
|
|
printf ("USAGE : \n");
|
|
|
|
printf (" snp_make -p <directory_name> <package_file> <stream_directory> [option] ... [option]\n");
|
|
|
|
printf (" option : \n");
|
|
|
|
printf (" -if wildcard : add the file if it matches the wilcard (at least one 'if' conditions must be met for a file to be adding)\n");
|
|
|
|
printf (" -ifnot wildcard : add the file if it doesn't match the wilcard (all the 'ifnot' conditions must be met for a file to be adding)\n");
|
|
|
|
printf (" Pack the directory to a snp file\n");
|
|
|
|
printf (" snp_make -l <package_file>\n");
|
|
|
|
printf (" List the files contained in the snp file\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
void generateLZMA(const std::string &sourceFile, const std::string &outputFile)
|
|
|
|
{
|
|
|
|
NLMISC::packLZMA(sourceFile, outputFile);
|
|
|
|
|
|
|
|
/*
|
|
|
|
std::string cmd="lzma e ";
|
|
|
|
cmd+=" "+sourceFile+" "+outputFile;
|
|
|
|
nlinfo("executing system command: %s",cmd.c_str());
|
|
|
|
#ifdef NL_OS_WINDOWS
|
|
|
|
_spawnlp(_P_WAIT, "lzma.exe","lzma.exe", "e", sourceFile.c_str(), outputFile.c_str(), NULL);
|
|
|
|
#else // NL_OS_WINDOWS
|
|
|
|
sint error = system (cmd.c_str());
|
|
|
|
if (error)
|
|
|
|
nlwarning("'%s' failed with error code %d", cmd.c_str(), error);
|
|
|
|
#endif // NL_OS_WINDOWS
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
uint readOptions (int nNbArg, char **ppArgs)
|
|
|
|
{
|
|
|
|
uint i;
|
|
|
|
uint optionCount = 0;
|
|
|
|
for (i=0; i<(uint)nNbArg; i++)
|
|
|
|
{
|
|
|
|
// If ?
|
|
|
|
if ((strcmp (ppArgs[i], "-if") == 0) && ((i+1)<(uint)nNbArg))
|
|
|
|
{
|
|
|
|
CWildCard card;
|
|
|
|
card.Expression = toLower(string(ppArgs[i+1]));
|
|
|
|
card.Not = false;
|
|
|
|
WildCards.push_back (card);
|
|
|
|
optionCount += 2;
|
|
|
|
}
|
|
|
|
// If not ?
|
|
|
|
if ((strcmp (ppArgs[i], "-ifnot") == 0) && ((i+1)<(uint)nNbArg))
|
|
|
|
{
|
|
|
|
CWildCard card;
|
|
|
|
card.Expression = toLower(string(ppArgs[i+1]));
|
|
|
|
card.Not = true;
|
|
|
|
WildCards.push_back (card);
|
|
|
|
optionCount += 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return optionCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
int main (int nNbArg, char **ppArgs)
|
|
|
|
{
|
|
|
|
NLMISC::CApplicationContext myApplicationContext;
|
|
|
|
|
|
|
|
if (nNbArg < 3)
|
|
|
|
{
|
|
|
|
usage();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((strcmp(ppArgs[1], "/p") == 0) || (strcmp(ppArgs[1], "/P") == 0) ||
|
|
|
|
(strcmp(ppArgs[1], "-p") == 0) || (strcmp(ppArgs[1], "-P") == 0))
|
|
|
|
{
|
|
|
|
if (nNbArg < 5)
|
|
|
|
{
|
|
|
|
usage();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceDirectory = ppArgs[2];
|
|
|
|
PackageFileName = ppArgs[3];
|
|
|
|
StreamDirectory = ppArgs[4];
|
|
|
|
readOptions(nNbArg, ppArgs);
|
|
|
|
|
|
|
|
nldebug("Make streamed package: '%s'", PackageFileName.c_str());
|
|
|
|
|
|
|
|
if (CFile::fileExists(PackageFileName))
|
|
|
|
{
|
|
|
|
nldebug("Update existing package");
|
|
|
|
try
|
|
|
|
{
|
|
|
|
CIFile fi;
|
|
|
|
fi.open(PackageFileName);
|
|
|
|
fi.serial(Package);
|
|
|
|
}
|
|
|
|
catch (Exception &e)
|
|
|
|
{
|
|
|
|
nlwarning("ERROR (snp_make) : serial exception: '%s'", e.what());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nldebug("New package");
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::string> pathContent; // contains full pathnames
|
|
|
|
std::vector<std::string> nameContent; // only filename
|
|
|
|
CPath::getPathContent(SourceDirectory, true, false, true, pathContent);
|
|
|
|
nameContent.reserve(pathContent.size());
|
|
|
|
for (std::vector<std::string>::size_type i = 0; i < pathContent.size(); ++i)
|
|
|
|
{
|
|
|
|
const std::string &file = pathContent[i];
|
|
|
|
if (keepFile(file.c_str()))
|
|
|
|
{
|
|
|
|
std::string fileName = NLMISC::toLower(CFile::getFilename(file));
|
|
|
|
// nldebug("File: '%s' ('%s')", file.c_str(), fileName.c_str());
|
|
|
|
nameContent.push_back(fileName);
|
|
|
|
nlassert(nameContent.size() == (i + 1));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Not included in this package
|
|
|
|
pathContent.erase(pathContent.begin() + i);
|
|
|
|
--i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<sint> packageIndex; // index of file in package
|
|
|
|
packageIndex.resize(pathContent.size(), -1);
|
|
|
|
|
|
|
|
for (CStreamedPackage::TEntries::size_type i = 0; i < Package.Entries.size(); ++i)
|
|
|
|
{
|
|
|
|
const CStreamedPackage::CEntry &entry = Package.Entries[i];
|
|
|
|
|
|
|
|
sint foundIndex = -1; // find index in found file list
|
|
|
|
for (std::vector<std::string>::size_type j = 0; j < pathContent.size(); ++j)
|
|
|
|
{
|
|
|
|
if (nameContent[j] == entry.Name)
|
|
|
|
{
|
|
|
|
foundIndex = j;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (foundIndex < 0)
|
|
|
|
{
|
|
|
|
nlinfo("File no longer exists: '%s'", entry.Name.c_str());
|
|
|
|
Package.Entries.erase(Package.Entries.begin() + i);
|
|
|
|
--i;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// File still exists, map it
|
|
|
|
packageIndex[foundIndex] = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (std::vector<std::string>::size_type i = 0; i < pathContent.size(); ++i)
|
|
|
|
{
|
|
|
|
sint pidx = packageIndex[i];
|
|
|
|
const std::string &name = nameContent[i];
|
|
|
|
const std::string &path = pathContent[i];
|
|
|
|
|
|
|
|
if (pidx < 0)
|
|
|
|
{
|
|
|
|
nlinfo("File added: '%s'", name.c_str());
|
|
|
|
pidx = Package.Entries.size();
|
|
|
|
Package.Entries.push_back(CStreamedPackage::CEntry());
|
|
|
|
Package.Entries[pidx].Name = name;
|
|
|
|
Package.Entries[pidx].LastModified = 0;
|
|
|
|
Package.Entries[pidx].Size = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nlinfo("File check for changes: '%s'", name.c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
CStreamedPackage::CEntry &entry = Package.Entries[pidx];
|
|
|
|
|
|
|
|
std::string targetLzmaOld; // in case lzma wasn't made make sure it exists a second run
|
|
|
|
CStreamedPackage::makePath(targetLzmaOld, entry.Hash);
|
|
|
|
targetLzmaOld = StreamDirectory + targetLzmaOld + ".lzma";
|
|
|
|
|
|
|
|
uint32 lastModified = CFile::getFileModificationDate(path);
|
|
|
|
uint32 fileSize = CFile::getFileSize(path);
|
|
|
|
if (lastModified > entry.LastModified || fileSize != entry.Size || !CFile::fileExists(targetLzmaOld))
|
|
|
|
{
|
|
|
|
entry.LastModified = lastModified;
|
|
|
|
|
|
|
|
nlinfo("Calculate file hash");
|
|
|
|
CHashKey hash = getSHA1(path, true);
|
|
|
|
/*nldebug("%s", hash.toString().c_str());
|
|
|
|
std::string hashPath;
|
|
|
|
CStreamedPackage::makePath(hashPath, hash);
|
|
|
|
nldebug("%s", hashPath.c_str());*/
|
|
|
|
|
|
|
|
if (hash == entry.Hash && fileSize == entry.Size)
|
|
|
|
{
|
|
|
|
// File has not changed
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nlinfo("File changed");
|
|
|
|
entry.Hash = hash;
|
|
|
|
entry.Size = fileSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string targetLzma; // in case lzma wasn't made make sure it exists a second run
|
|
|
|
CStreamedPackage::makePath(targetLzma, entry.Hash);
|
|
|
|
targetLzma = StreamDirectory + targetLzma + ".lzma";
|
|
|
|
|
|
|
|
if (!CFile::fileExists(targetLzma))
|
|
|
|
{
|
|
|
|
// make the compressed file
|
|
|
|
nlinfo("%s -> %s", path.c_str(), targetLzma.c_str());
|
|
|
|
CFile::createDirectoryTree(CFile::getPath(targetLzma));
|
|
|
|
generateLZMA(path, targetLzma);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
nldebug("Store package '%s'", PackageFileName.c_str());
|
|
|
|
COFile fo;
|
|
|
|
fo.open(PackageFileName);
|
|
|
|
fo.serial(Package);
|
|
|
|
}
|
|
|
|
catch (Exception &e)
|
|
|
|
{
|
|
|
|
nlwarning("ERROR (snp_make) : serial exception: '%s'", e.what());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((strcmp(ppArgs[1], "/l") == 0) || (strcmp(ppArgs[1], "/L") == 0) ||
|
|
|
|
(strcmp(ppArgs[1], "-l") == 0) || (strcmp(ppArgs[1], "-L") == 0))
|
|
|
|
{
|
|
|
|
PackageFileName = ppArgs[2];
|
|
|
|
if (!CFile::fileExists(PackageFileName))
|
|
|
|
{
|
|
|
|
nlwarning("ERROR (snp_make) : package doesn't exist: '%s'", PackageFileName.c_str());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
CIFile fi;
|
|
|
|
fi.open(PackageFileName);
|
|
|
|
fi.serial(Package);
|
|
|
|
}
|
|
|
|
catch (Exception &e)
|
|
|
|
{
|
|
|
|
nlwarning("ERROR (snp_make) : serial exception: '%s'", e.what());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (CStreamedPackage::TEntries::const_iterator it(Package.Entries.begin()), end(Package.Entries.end()); it != end; ++it)
|
|
|
|
{
|
|
|
|
const CStreamedPackage::CEntry &entry = (*it);
|
|
|
|
|
|
|
|
printf("List files in '%s'", PackageFileName.c_str());
|
|
|
|
printf("%s { Hash: '%s', Size: '%u', LastModified: '%u' }", entry.Name.c_str(), entry.Hash.toString().c_str(), entry.Size, entry.LastModified);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
usage ();
|
|
|
|
return -1;
|
|
|
|
}
|