6#include <ext/zlib/zlib.h>
62 virtual ~ListFilter() {}
64 bool accept(
const string& pathName,
const string& fileName)
override {
65 if (fileName ==
".")
return false;
66 if (fileName ==
"..")
return false;
67 if (FileSystem::getInstance()->isPath(pathName +
"/" + fileName) ==
true)
return true;
68 auto fileNameLowerCase = StringTools::toLowerCase(fileName);
70 if (StringTools::endsWith(fileNameLowerCase,
".ogg") ==
true)
return true;
72 if (StringTools::endsWith(fileNameLowerCase,
".fnt") ==
true)
return true;
74 if (StringTools::endsWith(fileNameLowerCase,
".icns") ==
true)
return true;
75 if (StringTools::endsWith(fileNameLowerCase,
".ico") ==
true)
return true;
76 if (StringTools::endsWith(fileNameLowerCase,
".png") ==
true)
return true;
78 if (StringTools::endsWith(fileNameLowerCase,
".dae") ==
true)
return true;
79 if (StringTools::endsWith(fileNameLowerCase,
".fbx") ==
true)
return true;
80 if (StringTools::endsWith(fileNameLowerCase,
".glb") ==
true)
return true;
81 if (StringTools::endsWith(fileNameLowerCase,
".tm") ==
true)
return true;
83 if (StringTools::endsWith(fileNameLowerCase,
".properties") ==
true)
return true;
85 if (StringTools::endsWith(fileNameLowerCase,
".cl") ==
true)
return true;
86 if (StringTools::endsWith(fileNameLowerCase,
".frag") ==
true)
return true;
87 if (StringTools::endsWith(fileNameLowerCase,
".glsl") ==
true)
return true;
88 if (StringTools::endsWith(fileNameLowerCase,
".vert") ==
true)
return true;
90 if (StringTools::endsWith(fileNameLowerCase,
".tempty") ==
true)
return true;
92 if (StringTools::endsWith(fileNameLowerCase,
".ttrigger") ==
true)
return true;
94 if (StringTools::endsWith(fileNameLowerCase,
".tenvmap") ==
true)
return true;
96 if (StringTools::endsWith(fileNameLowerCase,
".tmodel") ==
true)
return true;
98 if (StringTools::endsWith(fileNameLowerCase,
".tscene") ==
true)
return true;
100 if (StringTools::endsWith(fileNameLowerCase,
".tparticle") ==
true)
return true;
102 if (StringTools::endsWith(fileNameLowerCase,
".tterrain") ==
true)
return true;
104 if (StringTools::endsWith(fileNameLowerCase,
".tscript") ==
true)
return true;
106 if (StringTools::endsWith(fileNameLowerCase,
".xml") ==
true)
return true;
108 if (StringTools::endsWith(fileNameLowerCase,
".plist") ==
true)
return true;
110 if (fileName.rfind(
".") == string::npos ||
111 (fileName.rfind(
"/") != string::npos &&
112 fileName.rfind(
".") < fileName.rfind(
"/"))) {
120 ListFilter listFilter;
121 vector<string> files;
123 if (FileSystem::getInstance()->fileExists(path) ==
false) {
124 Console::println(
"Error: scanPathResources: file does not exist: " + path);
126 if (FileSystem::getInstance()->isPath(path) ==
false) {
127 if (listFilter.accept(
".", path) ==
true) {
128 totalFiles.push_back(path);
130 Console::println(
"Error: scanPathResources: file exist, but does not match filter: " + path);
133 FileSystem::getInstance()->list(path, files, &listFilter);
134 for (
auto fileName: files) {
135 if (FileSystem::getInstance()->isPath(path +
"/" + fileName) ==
false) {
136 totalFiles.push_back(path +
"/" + fileName);
147 virtual ~ListFilter() {}
149 bool accept(
const string& pathName,
const string& fileName)
override {
150 if (fileName ==
".")
return false;
151 if (fileName ==
"..")
return false;
153 if (FileSystem::getInstance()->isPath(pathName +
"/" + fileName) ==
true) {
155 #if defined(__APPLE__)
156 if (StringTools::endsWith(fileName,
".dSYM") ==
true)
return false;
161 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".a") ==
true)
return true;
164 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".dll") ==
true)
return true;
165 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".lib") ==
true)
return true;
166 #elif defined(__APPLE__)
167 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".dylib") ==
true)
return true;
168 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".so") ==
true)
return true;
170 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".so") ==
true)
return true;
177 ListFilter listFilter;
178 vector<string> files;
180 if (FileSystem::getInstance()->fileExists(path) ==
false) {
181 Console::println(
"Error: scanPathLibraries: file does not exist: " + path);
183 if (FileSystem::getInstance()->isPath(path) ==
false) {
184 if (listFilter.accept(
".", path) ==
true) {
185 totalFiles.push_back(path);
187 Console::println(
"Error: scanPathLibraries: file exist, but does not match filter: " + path);
190 FileSystem::getInstance()->list(path, files, &listFilter);
191 for (
auto fileName: files) {
192 if (FileSystem::getInstance()->isPath(path +
"/" + fileName) ==
false) {
193 totalFiles.push_back(path +
"/" + fileName);
204 virtual ~ListFilter() {}
206 bool accept(
const string& pathName,
const string& fileName)
override {
207 if (fileName ==
".")
return false;
208 if (fileName ==
"..")
return false;
209 if (FileSystem::getInstance()->isPath(pathName +
"/" + fileName) ==
true)
return true;
211 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".h") ==
true)
return true;
217 ListFilter listFilter;
218 vector<string> files;
220 if (FileSystem::getInstance()->fileExists(path) ==
false) {
221 Console::println(
"Error: scanPathHeaders: file does not exist: " + path);
223 if (FileSystem::getInstance()->isPath(path) ==
false) {
224 if (listFilter.accept(
".", path) ==
true) {
225 totalFiles.push_back(path);
227 Console::println(
"Error: scanPathHeaders: file exist, but does not match filter: " + path);
230 FileSystem::getInstance()->list(path, files, &listFilter);
231 for (
auto fileName: files) {
232 if (FileSystem::getInstance()->isPath(path +
"/" + fileName) ==
false) {
233 totalFiles.push_back(path +
"/" + fileName);
244 virtual ~ListFilter() {}
246 bool accept(
const string& pathName,
const string& fileName)
override {
247 if (fileName ==
".")
return false;
248 if (fileName ==
"..")
return false;
250 if (FileSystem::getInstance()->isPath(pathName +
"/" + fileName) ==
true) {
252 #if defined(__APPLE__)
253 if (StringTools::endsWith(fileName,
".dSYM") ==
true)
return false;
259 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".exe") ==
true)
return true;
260 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".dll") ==
true)
return true;
261 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".bat") ==
true)
return true;
262 #elif defined(__APPLE__)
264 if (fileName.rfind(
".") == string::npos ||
265 (fileName.rfind(
"/") != string::npos &&
266 fileName.rfind(
".") < fileName.rfind(
"/"))) {
269 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".dylib") ==
true)
return true;
270 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".so") ==
true)
return true;
271 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".sh") ==
true)
return true;
274 if (fileName.rfind(
".") == string::npos ||
275 (fileName.rfind(
"/") != string::npos &&
276 fileName.rfind(
".") < fileName.rfind(
"/"))) {
279 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".so") ==
true)
return true;
280 if (StringTools::endsWith(StringTools::toLowerCase(fileName),
".sh") ==
true)
return true;
287 ListFilter listFilter;
288 vector<string> files;
290 if (FileSystem::getInstance()->fileExists(path) ==
false) {
291 Console::println(
"Error: scanPathExecutables: file does not exist: " + path);
293 if (FileSystem::getInstance()->isPath(path) ==
false) {
294 if (listFilter.accept(
".", path) ==
true) {
295 totalFiles.push_back(path);
297 Console::println(
"Error: scanPathExecutables: file exist, but does not match filter: " + path);
300 FileSystem::getInstance()->list(path, files, &listFilter);
301 for (
auto fileName: files) {
302 if (FileSystem::getInstance()->isPath(path +
"/" + fileName) ==
false) {
303 totalFiles.push_back(path +
"/" + fileName);
311void processFile(
const string& fileName, vector<FileInformation>& fileInformations,
const string& archiveFileName,
bool executableFile,
const string& basePath,
const string& executablePath =
string()) {
313 vector<uint8_t> content;
314 FileSystem::getInstance()->getContent(
315 FileSystem::getInstance()->getPathName(fileName),
316 FileSystem::getInstance()->getFileName(fileName),
320 auto fileNameToUse = StringTools::startsWith(fileName, basePath +
"/") ==
true?StringTools::substring(fileName, (basePath +
"/").size(), fileName.size()):fileName;
322 if (executableFile ==
true && fileName.find_last_of(
'/') != string::npos) {
323 fileNameToUse = (executablePath.empty() ==
false?executablePath +
"/":
"") + StringTools::substring(fileNameToUse, fileNameToUse.find_last_of(
'/') + 1);
326 Console::print(archiveFileName +
": Processing file: " + fileNameToUse);
329 ofstream ofs(archiveFileName.c_str(), ofstream::binary | ofstream::app);
330 ofs.seekp(0, ofstream::end);
331 uint64_t fileOffset = ofs.tellp();
334 uint64_t bytesCompressed = 0;
335 uint8_t compressed = 1;
338 if (compressed == 1) {
347 unsigned char in[
CHUNK];
348 unsigned char out[
CHUNK];
351 strm.zalloc = Z_NULL;
353 strm.opaque = Z_NULL;
354 ret = deflateInit(&strm, Z_DEFAULT_COMPRESSION);
356 Console::println(
"processFile(): Error compressing file: Aborting");
361 size_t inPosition = 0;
362 size_t inBytes = content.size();
364 auto inStartPosition = inPosition;
365 for (
size_t i = 0; i <
CHUNK; i++) {
366 if (inPosition == inBytes)
break;
367 in[i] = content[inPosition];
370 strm.avail_in = inPosition - inStartPosition;
371 flush = inPosition == inBytes?Z_FINISH:Z_NO_FLUSH;
376 strm.avail_out =
CHUNK;
378 ret = deflate(&strm, flush);
379 assert(ret != Z_STREAM_ERROR);
380 have =
CHUNK - strm.avail_out;
381 ofs.write((
char*)out, have);
382 bytesCompressed+= have;
383 }
while (strm.avail_out == 0);
384 assert(strm.avail_in == 0);
387 }
while (flush != Z_FINISH);
388 assert(ret == Z_STREAM_END);
391 (void) deflateEnd(&strm);
393 ofs.write((
char*)content.data(), content.size());
399 fileInformation.
name = fileNameToUse;
400 fileInformation.
bytes = content.size();
403 fileInformation.
offset = fileOffset;
405 fileInformations.push_back(fileInformation);
408 Console::println(
", processed " + to_string(content.size()) +
" bytes" + (compressed == 1?
", " + to_string(bytesCompressed) +
" bytes compressed":
""));
411#if defined(__APPLE__)
412void createMacApplication(
const Properties& installerProperties,
const string& fileName,
const string& pathName =
string()) {
413 auto _pathName = pathName.empty() ==
true?
"":pathName +
"/";
414 auto _fileName = StringTools::substring(fileName, fileName.rfind(
'/') + 1, fileName.size());
415 auto _filePath = StringTools::substring(fileName, 0, fileName.rfind(
'/'));
416 auto startMenuName = installerProperties.
get(
"startmenu_" + StringTools::toLowerCase(_fileName),
"");
417 if (startMenuName.empty() ==
false) {
418 auto executablePathName = FileSystem::getInstance()->getPathName(fileName);
419 auto executableFileName = FileSystem::getInstance()->getFileName(fileName);
420 auto iconFileName = StringTools::toLowerCase(executableFileName) +
"-icon.icns";
421 if (FileSystem::getInstance()->fileExists(
"resources/platforms/macos/" + iconFileName) ==
false &&
422 FileSystem::getInstance()->fileExists(executablePathName +
"/resources/platforms/macos/" + iconFileName) ==
false) iconFileName =
"default-icon.icns";
423 auto infoplistFile = FileSystem::getInstance()->getContentAsString(
"resources/platforms/macos",
"Info.plist");
424 infoplistFile = StringTools::replace(infoplistFile,
"{__EXECUTABLE__}", executableFileName);
425 infoplistFile = StringTools::replace(infoplistFile,
"{__EXECUTABLE_LOWERCASE__}", StringTools::toLowerCase(executableFileName));
426 infoplistFile = StringTools::replace(infoplistFile,
"{__ICON__}",
"icon.icns");
427 infoplistFile = StringTools::replace(infoplistFile,
"{__COPYRIGHT__}", Version::getCopyright());
428 infoplistFile = StringTools::replace(infoplistFile,
"{__VERSION__}", Version::getVersion());
429 FileSystem::getInstance()->createPath(_pathName + executableFileName +
".app");
430 FileSystem::getInstance()->createPath(_pathName + executableFileName +
".app/Contents");
431 FileSystem::getInstance()->createPath(_pathName + executableFileName +
".app/Contents/MacOS");
432 FileSystem::getInstance()->createPath(_pathName + executableFileName +
".app/Contents/Resources");
433 FileSystem::getStandardFileSystem()->setContentFromString(
434 _pathName + executableFileName +
".app/Contents",
439 vector<uint8_t> content;
440 FileSystem::getInstance()->getContent(
441 "resources/platforms/macos",
445 FileSystem::getStandardFileSystem()->setContent(
446 _pathName + executableFileName +
".app/Contents/Resources",
452 vector<uint8_t> content;
453 FileSystem::getInstance()->getContent(
458 FileSystem::getInstance()->setContent(
459 _pathName + executableFileName +
".app/Contents/MacOS",
463 FileSystem::getInstance()->setExecutable(
464 _pathName + executableFileName +
".app/Contents/MacOS",
468 auto codeSignCommand =
"codesign -s \"" + installerProperties.
get(
"macos_codesign_identity",
"No identity") +
"\" \"" + _pathName + executableFileName +
".app\"";
469 Console::println(
"Signing '" + fileName +
"': " + codeSignCommand);
470 Console::println(Application::execute(codeSignCommand));
480 Console::println(
string(
"create-installer ") + Version::getVersion());
481 Console::println(Version::getCopyright());
485 string tdmePath =
"../tdme2";
486 auto cpu = Application::getCPUName();
487 auto os = Application::getOSName();
488 auto fileNameTime = StringTools::replace(StringTools::replace(StringTools::replace(Time::getAsString(),
" ",
"-" ),
":",
""),
"-",
"");
492 installerProperties.
load(
"resources/installer",
"installer.properties");
493 for (
auto componentIdx = 0;
true; componentIdx++) {
494 auto componentId = componentIdx == 0?
"installer":
"component" + to_string(componentIdx);
495 auto componentName = installerProperties.
get(componentId,
"");
496 if (componentName.empty() ==
true)
break;
497 Console::println(
"Having component: " + to_string(componentIdx) +
": " + componentName);
498 auto componentInclude = installerProperties.
get(componentId +
"_include",
"");
499 if (componentInclude.empty() ==
true) {
500 Console::println(
"component: " + to_string(componentIdx) +
": missing includes. Skipping.");
505 auto componentFileName = os +
"-" + cpu +
"-" + StringTools::replace(StringTools::replace(componentName,
" - ",
"-"),
" ",
"-") +
"-" + fileNameTime +
".ta";
507 Console::println(
"Component: " + to_string(componentIdx) +
": component file name: " + componentFileName);
509 if (FileSystem::getInstance()->fileExists(
"installer") ==
false) {
510 FileSystem::getInstance()->createPath(
"installer");
515 ofstream ofs(
"installer/" + componentFileName, ofstream::binary | ofstream::trunc);
520 vector<FileInformation> fileInformations;
521 vector<string> filesData;
522 vector<string> filesBin;
529 auto componentIncludeDefinition = t.
nextToken();
530 t2.
tokenize(componentIncludeDefinition,
":");
533 if (type.empty() ==
true || file.empty() ==
true) {
534 Console::println(
"Component: " + to_string(componentIdx) +
": type or file empty. Skipping");
537 Console::println(
"Component: " + to_string(componentIdx) +
": type = " + type +
"; file = " + file);
542 t.
tokenize(installerProperties.
get(
"exe_path",
""),
",");
547 if (type ==
"!exe") {
549 t.
tokenize(installerProperties.
get(
"exe_path",
""),
",");
557 if (type ==
"!res") {
566 Console::println(
"Component: " + to_string(componentIdx) +
": type = " + type +
" unsupported!");
570 Console::println(
"Component: " + to_string(componentIdx) +
": Processing files");
574 for (
auto fileName: filesData) {
575 processFile(fileName, fileInformations,
"installer/" + componentFileName,
false, tdmePath);
579 for (
auto fileName: filesBin) {
580 #if defined(__APPLE__)
581 auto _fileName = StringTools::substring(fileName, fileName.rfind(
'/') + 1, fileName.size());
582 auto _filePath = StringTools::substring(fileName, 0, fileName.rfind(
'/'));
583 auto startMenuName = installerProperties.
get(
"startmenu_" + StringTools::toLowerCase(_fileName),
"");
584 if (startMenuName.empty() ==
false) {
585 createMacApplication(installerProperties, fileName);
586 auto executableFileName = FileSystem::getInstance()->getFileName(fileName);
587 processFile(executableFileName +
".app/Contents/Info.plist", fileInformations,
"installer/" + componentFileName,
false, tdmePath);
588 processFile(executableFileName +
".app/Contents/Resources/icon.icns", fileInformations,
"installer/" + componentFileName,
false, tdmePath);
589 processFile(executableFileName +
".app/Contents/_CodeSignature/CodeResources", fileInformations,
"installer/" + componentFileName,
false, tdmePath);
590 processFile(executableFileName +
".app/Contents/MacOS/" + executableFileName, fileInformations,
"installer/" + componentFileName,
true, tdmePath, executableFileName +
".app/Contents/MacOS");
591 FileSystem::getInstance()->removePath(executableFileName +
".app",
true);
593 auto executablePathName = FileSystem::getInstance()->getPathName(fileName);
594 auto executableFileName = FileSystem::getInstance()->getFileName(fileName);
595 FileSystem::getInstance()->createPath(
"signed-executables");
597 vector<uint8_t> content;
598 FileSystem::getInstance()->getContent(
603 FileSystem::getInstance()->setContent(
604 "signed-executables",
608 FileSystem::getInstance()->setExecutable(
609 "signed-executables",
613 auto signedFileName =
"signed-executables/" + executableFileName;
614 auto codeSignCommand =
"codesign -s \"" + installerProperties.
get(
"macos_codesign_identity",
"No identity") +
"\" \"" + signedFileName +
"\"";
615 Console::println(
"Signing '" + fileName +
"': " + codeSignCommand);
616 Console::println(Application::execute(codeSignCommand));
617 processFile(signedFileName, fileInformations,
"installer/" + componentFileName,
true, tdmePath);
618 FileSystem::getInstance()->removePath(
"signed-executables",
true);
621 processFile(fileName, fileInformations,
"installer/" + componentFileName,
true, tdmePath);
627 ofstream ofs(
"installer/" + componentFileName, ofstream::binary | ofstream::app);
628 ofs.seekp(0, ofstream::end);
629 uint32_t fileInformationOffsetEnd = 0LL;
630 uint64_t fileInformationOffset = ofs.tellp();
631 for (
auto& fileInformation: fileInformations) {
632 uint32_t nameSize = fileInformation.name.size();
633 ofs.write((
char*)&nameSize,
sizeof(nameSize));
634 for (
auto i = 0; i < nameSize; i++) ofs.write(&fileInformation.name[i], 1);
635 ofs.write((
char*)&fileInformation.bytes,
sizeof(fileInformation.bytes));
636 ofs.write((
char*)&fileInformation.compressed,
sizeof(fileInformation.compressed));
637 ofs.write((
char*)&fileInformation.bytesCompressed,
sizeof(fileInformation.bytesCompressed));
638 ofs.write((
char*)&fileInformation.offset,
sizeof(fileInformation.offset));
639 ofs.write((
char*)&fileInformation.executable,
sizeof(fileInformation.executable));
641 ofs.write((
char*)&fileInformationOffsetEnd,
sizeof(fileInformationOffsetEnd));
642 ofs.write((
char*)&fileInformationOffset,
sizeof(fileInformationOffset));
647 auto archiveFileSystem =
new ArchiveFileSystem(
"installer/" + componentFileName);
648 auto archiveHash = archiveFileSystem->computeSHA256Hash();
649 delete archiveFileSystem;
650 FileSystem::getStandardFileSystem()->setContentFromString(
"installer", componentFileName +
".sha256", archiveHash);
651 Console::println(
"Component: " + to_string(componentIdx) +
": component file name: " + componentFileName +
": hash: " + archiveHash);
655 auto completionFileName = os +
"-" + cpu +
"-upload-" + fileNameTime;
658 ofstream ofs(
"installer/" + completionFileName, ofstream::binary | ofstream::trunc);
663 #if defined(__APPLE__)
664 if (FileSystem::getInstance()->fileExists(
"installer-package") ==
true &&
665 FileSystem::getInstance()->isPath(
"installer-package") ==
true) {
666 FileSystem::getInstance()->removePath(
"installer-package",
true);
668 FileSystem::getInstance()->createPath(
"installer-package");
669 createMacApplication(installerProperties,
"bin/tdme/tools/installer/Installer",
"installer-package");
Application base class, please make sure to allocate application on heap to have correct application ...
Archive file system implementation.
File system singleton class.
Properties class, which helps out with storeing or loading key value pairs from/to property files.
const string & get(const string &key, const string &defaultValue) const
Get property value by key.
void load(const string &pathName, const string &fileName, FileSystemInterface *fileSystem=nullptr)
Load property file.
const string & nextToken()
void tokenize(const string &str, const string &delimiters)
Tokenize.
void processFile(const string &fileName, vector< FileInformation > &fileInformations, const string &archiveFileName, bool executableFile, const string &basePath, const string &executablePath=string())
int main(int argc, char **argv)
static void scanPathResources(const string &path, vector< string > &totalFiles)
static void scanPathHeaders(const string &path, vector< string > &totalFiles)
static void scanPathLibraries(const string &path, vector< string > &totalFiles)
static void scanPathExecutables(const string &path, vector< string > &totalFiles)
std::exception Exception
Exception base class.
File system file name filter interface.
virtual bool accept(const string &path, const string &file)=0
Accept a file.