8#include <unordered_map>
9#include <unordered_set>
59using std::unordered_map;
60using std::unordered_set;
102Installer::Installer(): installThreadMutex(
"install-thread-mutex")
104 Application::setLimitFPS(
true);
105 Tools::loadSettings(
this);
106 this->
engine = Engine::getInstance();
116 vector<string> installedComponents;
118 FileSystem::getStandardFileSystem()->getContentAsStringArray(
".",
"install.components.db", installedComponents);
124 timestamp = FileSystem::getStandardFileSystem()->getContentAsString(
".",
"install.version.db");
129 string installFolder;
132 FileSystem::getStandardFileSystem()->getContentAsStringArray(
".",
"install.files.db", log);
133 if (log.size() > 0) installFolder = log[0];
144 unordered_map<string, string> parameters = {
147 {
"installfolder", installFolder.empty() ==
true?
homeFolder +
"/Applications/" +
installerProperties.
get(
"install_path",
"TDME2-based-application"):installFolder}
152 "resources/installer",
153 "installer_welcome.xml",
160 "resources/installer",
161 "installer_license.xml",
167 "installer_components",
169 "resources/installer",
170 "installer_components.xml",
174 string componentsXML =
"<space height=\"10\" />\n";
175 for (
auto componentIdx = 1;
true; componentIdx++) {
177 auto componentRequired = StringTools::trim(StringTools::toLowerCase(
installerProperties.
get(
"component" + to_string(componentIdx) +
"_required",
"false"))) ==
"true";
178 auto componentInstalled =
false;
179 for (
auto installedComponentName: installedComponents) {
180 if (installedComponentName == componentName) {
181 componentInstalled =
true;
185 if (componentName.empty() ==
true)
break;
187 string(
"<element id=\"component" + to_string(componentIdx) +
"\" width=\"100%\" height=\"25\">\n") +
188 string(
" <layout width=\"100%\" alignment=\"horizontal\">\n") +
189 string(
" <space width=\"10\" />\n") +
190 string(
" <checkbox id=\"checkbox_component" + to_string(componentIdx) +
"\" name=\"checkbox_component" + to_string(componentIdx) +
"\" value=\"1\" selected=\"" + (componentRequired ==
true || componentInstalled ==
true?
"true":
"false") +
"\" disabled=\"" + (componentRequired ==
true?
"true":
"false") +
"\" />\n") +
191 string(
" <space width=\"10\" />\n") +
192 string(
" <text width=\"*\" font=\"resources/engine/fonts/Roboto_20.fnt\" text=\"" + GUIParser::escapeQuotes(componentName) +
"\" color=\"#000000\" height=\"100%\" vertical-align=\"center\" />\n") +
193 string(
" </layout>\n") +
194 string(
"</element>\n");
203 "resources/installer",
204 "installer_folder.xml",
209 "installer_installing",
211 "resources/installer",
212 "installer_installing.xml",
217 "installer_finished",
219 "resources/installer",
220 "installer_finished.xml",
225 "installer_welcome2",
227 "resources/installer",
228 "installer_welcome2.xml",
233 "installer_uninstalling",
235 "resources/installer",
236 "installer_uninstalling.xml",
288 class CheckForUpdateThread:
public Thread {
290 CheckForUpdateThread(
Installer* installer):
Thread(
"checkforupdate-thread"), installer(installer) {
293 Console::println(
"CheckForUpdateThread::run(): init");
295 auto currentTimestamp = installer->
timestamp;
298 auto repairHaveLocalFile =
false;
299 auto completionFileName = Application::getOSName() +
"-" + Application::getCPUName() +
"-upload-";
304 vector<string> files;
305 FileSystem::getStandardFileSystem()->list(
"installer", files);
306 for (
auto file: files) {
307 if (StringTools::startsWith(file, completionFileName) ==
true) {
308 Console::println(
"CheckForUpdateThread: Have upload completion file: " + file);
309 installer->
timestamp = StringTools::substring(file, completionFileName.size());
310 repairHaveLocalFile =
true;
314 if (installer->
timestamp.empty() ==
false) {
315 Console::println(
"CheckForUpdateThread::run(): filesystem: newest timestamp: " + installer->
timestamp);
328 httpClient.
setMethod(HTTPClient::HTTP_METHOD_GET);
329 httpClient.
setURL(repository +
"?timestamp=" + to_string(Time::getCurrentMillis()));
335 while ((pos = response.find(completionFileName, pos)) != string::npos) {
336 pos+= completionFileName.size();
337 auto timestampWebNew = StringTools::substring(response, pos, pos + 14);
339 if ((installer->
timestamp.empty() ==
true && (timestampWeb.empty() ==
true || timestampWebNew > timestampWeb)) ||
340 (installer->
timestamp.empty() ==
false && ((timestampWeb.empty() ==
true || timestampWebNew > timestampWeb) && timestampWebNew > installer->
timestamp))) {
341 timestampWeb = timestampWebNew;
345 Console::println(
string(
"CheckForUpdateThread::run(): An error occurred: ") + exception.what());
353 if (timestampWeb.empty() ==
false) {
354 Console::println(
"CheckForUpdateThread::run(): repository: newest timestamp: " + timestampWeb);
361 for (
auto componentIdx = 0;
true; componentIdx++) {
362 auto componentId = componentIdx == 0?
"installer":
"component" + to_string(componentIdx);
364 if (componentName.empty() ==
true)
break;
365 Console::println(
"CheckForUpdateThread::run(): Having component: " + to_string(componentIdx) +
": " + componentName);
367 if (componentInclude.empty() ==
true) {
368 Console::println(
"CheckForUpdateThread::run(): component: " + to_string(componentIdx) +
": missing includes. Skipping.");
371 auto componentFileName = Application::getOSName() +
"-" + Application::getCPUName() +
"-" + StringTools::replace(StringTools::replace(componentName,
" - ",
"-"),
" ",
"-") +
"-" + installer->
timestamp +
".ta";
374 Console::println(
"CheckForUpdateThread::run(): Component: " + to_string(componentIdx) +
": component file name: " + componentFileName +
": Downloading");
383 if (componentIdx > 0) {
385 installer->
downloadedFiles.push_back(
"installer/" + componentFileName +
".sha256.download");
386 installer->
downloadedFiles.push_back(
"installer/" + componentFileName +
".sha256");
389 httpDownloadClient.
reset();
392 httpDownloadClient.
setFile(
"installer/" + componentFileName +
".sha256");
394 httpDownloadClient.
start();
395 while (httpDownloadClient.
isFinished() ==
false) {
399 Thread::sleep(250LL);
401 httpDownloadClient.
join();
405 Console::println(
"CheckForUpdateThread::run(): done");
411 if (componentIdx > 0) {
413 installer->
downloadedFiles.push_back(
"installer/" + componentFileName +
".download");
414 installer->
downloadedFiles.push_back(
"installer/" + componentFileName);
417 httpDownloadClient.
reset();
418 httpDownloadClient.
setFile(
"installer/" + componentFileName);
420 httpDownloadClient.
start();
421 while (httpDownloadClient.
isFinished() ==
false) {
425 Thread::sleep(250LL);
427 httpDownloadClient.
join();
431 Console::println(
"CheckForUpdateThread::run(): done");
443 Console::println(
"CheckForUpdateThread::run(): done");
456 Console::println(
"CheckForUpdateThread::run(): done");
462 CheckForUpdateThread* checkForUpdateThread =
new CheckForUpdateThread(
this);
463 checkForUpdateThread->start();
481 class InstallThread:
public Thread {
483 InstallThread(
Installer* installer):
Thread(
"install-thread"), installer(installer) {
486 Console::println(
"InstallThread::run(): init");
491 vector<string> windowsUpdateRenameFiles;
502 auto hadException =
false;
504 vector<string> components;
514 log.push_back(installFolder);
515 if (hadException ==
false) {
518 auto file =
dynamic_cast<ArchiveFileSystem*
>(FileSystem::getInstance())->getArchiveFileName();
520 vector<uint8_t> content;
521 Console::println(
"InstallThread::run(): Installer: Copy: " + file);
522 FileSystem::getStandardFileSystem()->getContent(
523 FileSystem::getStandardFileSystem()->getPathName(file),
524 FileSystem::getStandardFileSystem()->getFileName(file),
527 auto generatedFileName = installFolder +
"/" + file;
529 FileSystem::getStandardFileSystem()->getPathName(generatedFileName)
531 FileSystem::getStandardFileSystem()->setContent(
532 FileSystem::getStandardFileSystem()->getPathName(generatedFileName),
533 FileSystem::getStandardFileSystem()->getFileName(generatedFileName),
539 vector<uint8_t> content;
540 Console::println(
"InstallThread::run(): Installer: Copy: " + file);
541 FileSystem::getStandardFileSystem()->getContent(
542 FileSystem::getStandardFileSystem()->getPathName(file),
543 FileSystem::getStandardFileSystem()->getFileName(file),
546 auto generatedFileName = installFolder +
"/" + file;
548 FileSystem::getStandardFileSystem()->getPathName(generatedFileName)
550 FileSystem::getStandardFileSystem()->setContent(
551 FileSystem::getStandardFileSystem()->getPathName(generatedFileName),
552 FileSystem::getStandardFileSystem()->getFileName(generatedFileName),
558 for (
auto componentIdx = 1;
true; componentIdx++) {
561 if (componentName.empty() ==
true)
break;
564 auto componentFileName = Application::getOSName() +
"-" + Application::getCPUName() +
"-" + StringTools::replace(StringTools::replace(componentName,
" - ",
"-"),
" ",
"-") +
"-" + installer->
timestamp +
".ta";
577 FileSystem::getStandardFileSystem()->removeFile(
"installer", componentFileName);
581 FileSystem::getStandardFileSystem()->removeFile(
"installer", componentFileName +
".sha256");
591 components.push_back(componentName);
594 Console::println(
"InstallThread::run(): Having component: " + to_string(componentIdx) +
": " + componentName);
595 auto componentInclude = installer->
installerProperties.
get(
"component" + to_string(componentIdx) +
"_include",
"");
596 if (componentInclude.empty() ==
true) {
597 Console::println(
"InstallThread::run(): component: " + to_string(componentIdx) +
": missing includes. Skipping.");
601 Console::println(
"InstallThread::run(): Component: " + to_string(componentIdx) +
": component file name: " + componentFileName);
614 if (archiveFileSystem->
computeSHA256Hash() != FileSystem::getStandardFileSystem()->getContentAsString(
"installer", componentFileName +
".sha256")) {
615 throw ExceptionBase(
"Failed to verify: " + componentFileName +
", remove files in ./installer/ and try again");
620 vector<string> files;
622 uint64_t totalSize = 0LL;
623 uint64_t doneSize = 0LL;
624 for (
auto file: files) {
630 for (
auto file: files) {
631 vector<uint8_t> content;
632 Console::println(
"InstallThread::run(): Component: " + to_string(componentIdx) +
": " + file);
638 auto generatedFileName = installFolder +
"/" + file;
640 FileSystem::getStandardFileSystem()->getPathName(generatedFileName)
643 auto windowsGeneratedFile = generatedFileName;
647 (StringTools::endsWith(StringTools::toLowerCase(generatedFileName),
".exe") ==
true ||
648 StringTools::endsWith(StringTools::toLowerCase(generatedFileName),
".dll") ==
true)) {
649 windowsUpdateRenameFiles.push_back(FileSystem::getStandardFileSystem()->getFileName(generatedFileName));
650 windowsGeneratedFile+=
".update";
655 #if defined(__FreeBSD__) || defined(__linux__) || defined(__NetBSD__)
656 FileSystem::getStandardFileSystem()->setContent(
657 FileSystem::getStandardFileSystem()->getPathName(generatedFileName),
658 FileSystem::getStandardFileSystem()->getFileName(generatedFileName),
661 log.push_back(generatedFileName);
662 FileSystem::getStandardFileSystem()->setExecutable(
663 FileSystem::getStandardFileSystem()->getPathName(generatedFileName),
664 FileSystem::getStandardFileSystem()->getFileName(generatedFileName)
666 auto launcherName = installer->
installerProperties.
get(
"launcher_" + StringTools::toLowerCase(FileSystem::getStandardFileSystem()->getFileName(generatedFileName)),
"");
667 if (launcherName.empty() ==
false) {
669 FileSystem::getStandardFileSystem()->setContentFromString(
670 FileSystem::getStandardFileSystem()->getPathName(generatedFileName),
671 FileSystem::getStandardFileSystem()->getFileName(generatedFileName) +
".sh",
674 "cd " + installFolder +
"\n" +
676 FileSystem::getStandardFileSystem()->getPathName(generatedFileName) +
"/" + FileSystem::getStandardFileSystem()->getFileName(generatedFileName) +
" " +
677 "</dev/null &>/dev/null &\n"
679 FileSystem::getStandardFileSystem()->setExecutable(
680 FileSystem::getStandardFileSystem()->getPathName(generatedFileName),
681 FileSystem::getStandardFileSystem()->getFileName(generatedFileName) +
".sh"
683 auto executablePathName = FileSystem::getInstance()->getPathName(generatedFileName);
686 if (archiveFileSystem->
fileExists(
"resources/platforms/icons/" + iconFileName) ==
false &&
687 FileSystem::getInstance()->fileExists(executablePathName +
"/resources/platforms/icons/" + iconFileName) ==
false) iconFileName =
"default-icon.png";
688 FileSystem::getStandardFileSystem()->setContentFromString(
689 installer->
homeFolder +
"/" +
".local/share/applications",
690 launcherName +
".desktop",
692 "[Desktop Entry]\n" +
693 "Name=" + launcherName +
"\n" +
694 "Exec=" + FileSystem::getStandardFileSystem()->getPathName(generatedFileName) +
"/" + FileSystem::getStandardFileSystem()->getFileName(generatedFileName) +
".sh\n" +
696 "Type=Application\n" +
697 "Icon=" + installFolder +
"/resources/platforms/icons/" + iconFileName +
"\n"
699 log.push_back(generatedFileName +
".sh");
700 log.push_back(installer->
homeFolder +
"/" +
".local/share/applications/" + launcherName +
".desktop");
702 #elif defined(_WIN32)
703 FileSystem::getStandardFileSystem()->setContent(
704 FileSystem::getStandardFileSystem()->getPathName(generatedFileName),
705 FileSystem::getStandardFileSystem()->getFileName(windowsGeneratedFile),
708 log.push_back(generatedFileName);
709 auto executable = FileSystem::getStandardFileSystem()->getFileName(generatedFileName);
712 StringTools::substring(
713 StringTools::toLowerCase(executable),
715 executable.rfind(
'.')
719 if (launcherName.empty() ==
false) {
720 FileSystem::getStandardFileSystem()->setContentFromString(
722 "windows-create-shortcut.bat",
723 FileSystem::getInstance()->getContentAsString(
".",
"windows-create-shortcut.bat")
725 auto startMenuFolder = string(installer->
homeFolder) +
"/AppData/Roaming/Microsoft/Windows/Start Menu/Programs/" + installer->
installerProperties.
get(
"name",
"TDME2 based application");
727 auto linkFile = startMenuFolder +
"/" + launcherName +
".lnk";
728 auto executablePathName = FileSystem::getInstance()->getPathName(generatedFileName);
730 auto iconFileName = StringTools::replace(StringTools::toLowerCase(
executableFileName),
".exe",
"") +
"-icon.ico";
731 if (FileSystem::getInstance()->fileExists(
"resources/platforms/win32/" + iconFileName) ==
false &&
732 FileSystem::getInstance()->fileExists(executablePathName +
"/resources/platforms/win32/" + iconFileName) ==
false) iconFileName =
"default-icon.ico";
734 StringTools::replace(StringTools::replace(installFolder,
"/",
"\\"),
" ",
"^ ") +
"\\windows-create-shortcut.bat " +
735 "\"" + StringTools::replace(generatedFileName,
"/",
"\\") +
"\" " +
736 "\"" + StringTools::replace(linkFile,
"/",
"\\") +
"\" " +
737 "\"resources\\platforms\\win32\\" + iconFileName +
"\" "
740 Application::execute(
741 StringTools::replace(StringTools::replace(installFolder,
"/",
"\\"),
" ",
"^ ") +
"\\windows-create-shortcut.bat " +
742 "\"" + StringTools::replace(generatedFileName,
"/",
"\\") +
"\" " +
743 "\"" + StringTools::replace(linkFile,
"/",
"\\") +
"\" " +
744 "\"resources\\platforms\\win32\\" + iconFileName +
"\" "
747 log.push_back(linkFile);
748 FileSystem::getStandardFileSystem()->removeFile(installFolder,
"windows-create-shortcut.bat");
751 FileSystem::getStandardFileSystem()->setContent(
752 FileSystem::getStandardFileSystem()->getPathName(generatedFileName),
753 FileSystem::getStandardFileSystem()->getFileName(generatedFileName),
756 log.push_back(generatedFileName);
757 FileSystem::getStandardFileSystem()->setExecutable(
758 FileSystem::getStandardFileSystem()->getPathName(generatedFileName),
759 FileSystem::getStandardFileSystem()->getFileName(generatedFileName)
764 FileSystem::getStandardFileSystem()->setContent(
765 FileSystem::getStandardFileSystem()->getPathName(generatedFileName),
767 FileSystem::getStandardFileSystem()->getFileName(windowsGeneratedFile),
769 FileSystem::getStandardFileSystem()->getFileName(generatedFileName),
773 log.push_back(generatedFileName);
775 doneSize+= content.size();
781 delete archiveFileSystem;
782 archiveFileSystem =
nullptr;
791 FileSystem::getStandardFileSystem()->removeFile(
"installer", componentFileName);
795 FileSystem::getStandardFileSystem()->removeFile(
"installer", componentFileName +
".sha256");
801 if (archiveFileSystem !=
nullptr)
delete archiveFileSystem;
812 string updateFinishBatch;
813 updateFinishBatch+=
"@ECHO OFF\r\n";
814 updateFinishBatch+=
"ECHO FINISHING UPDATE. PLEASE DO NOT CLOSE.\r\n";
815 updateFinishBatch+=
"setlocal EnableDelayedExpansion\r\n";
817 for (
auto file: windowsUpdateRenameFiles) {
818 auto updateFile = file +
".update";
820 ":loop" + to_string(loopIdx) +
"\r\n" +
821 "if exist \"" + updateFile +
"\" (\r\n" +
822 " if exist \"" + file +
"\" (\r\n" +
823 " del \"" + file +
"\" > nul 2>&1\r\n" +
824 " if exist \"" + file +
"\" goto loop" + to_string(loopIdx) +
"\r\n" +
825 " rename \"" + updateFile +
"\" \"" + file +
"\" > nul 2>&1\r\n" +
827 " rename \"" + updateFile +
"\" \"" + file +
"\" > nul 2>&1\r\n" +
833 FileSystem::getStandardFileSystem()->setContentFromString(installFolder,
"update-finish.bat", updateFinishBatch);
842 FileSystem::getStandardFileSystem()->setContentFromStringArray(installFolder,
"install.files.db", log);
843 FileSystem::getStandardFileSystem()->setContentFromStringArray(installFolder,
"install.components.db", components);
844 FileSystem::getStandardFileSystem()->setContentFromString(installFolder,
"install.version.db", installer->
timestamp);
851 if (hadException ==
false) {
861 Console::println(
"InstallThread::run(): done");
867 InstallThread* installThread =
new InstallThread(
this);
868 installThread->start();
886 class UninstallThread:
public Thread {
888 UninstallThread(
Installer* installer):
Thread(
"install-thread"), installer(installer) {
891 Console::println(
"UninstallThread::run(): init");
901 auto hadException =
false;
904 FileSystem::getStandardFileSystem()->getContentAsStringArray(
".",
"install.files.db", log);
911 auto installFolder = log.size() > 0?log[0] +
"/":
"";
913 if (hadException ==
false) {
915 auto uninstallFinishBatchLoopIdx = 0;
916 string uninstallFinishBatch;
917 uninstallFinishBatch+=
"@ECHO OFF\r\n";
918 uninstallFinishBatch+=
"ECHO FINISHING UNINSTALL. PLEASE DO NOT CLOSE.\r\n";
919 uninstallFinishBatch+=
"setlocal EnableDelayedExpansion\r\n";
921 for (
auto i = 1; i < log.size(); i++) {
928 if (FileSystem::getStandardFileSystem()->fileExists(log[i]) ==
true) {
929 FileSystem::getStandardFileSystem()->removeFile(
930 FileSystem::getStandardFileSystem()->getPathName(log[i]),
931 FileSystem::getStandardFileSystem()->getFileName(log[i])
935 Console::println(
string(
"UninstallThread::run(): An error occurred: ") + innerException.what());
938 (StringTools::endsWith(log[i],
".dll") ==
true ||
939 StringTools::endsWith(log[i],
".exe") ==
true)) {
940 auto file = FileSystem::getStandardFileSystem()->getFileName(log[i]);
941 uninstallFinishBatch+=
942 ":loop" + to_string(uninstallFinishBatchLoopIdx) +
"\r\n" +
943 "if exist \"" + file +
"\" (\r\n" +
944 " del \"" + file +
"\" > nul 2>&1\r\n" +
945 " if exist \"" + file +
"\" goto loop" + to_string(uninstallFinishBatchLoopIdx) +
"\r\n" +
947 uninstallFinishBatchLoopIdx++;
955 auto file =
"console.log";
956 uninstallFinishBatch+=
957 ":loop" + to_string(uninstallFinishBatchLoopIdx) +
"\r\n" +
958 "if exist \"" + file +
"\" (\r\n" +
959 " del \"" + file +
"\" > nul 2>&1\r\n" +
960 " if exist \"" + file +
"\" goto loop" + to_string(uninstallFinishBatchLoopIdx) +
"\r\n" +
962 uninstallFinishBatchLoopIdx++;
965 FileSystem::getStandardFileSystem()->setContentFromString(installFolder,
"uninstall-finish.bat", uninstallFinishBatch);
975 if (hadException ==
false) {
976 vector<string> folders;
977 for (
auto i = 1; i < log.size(); i++) {
978 auto folderCandidate = FileSystem::getStandardFileSystem()->getPathName(log[i]);
979 if (FileSystem::getStandardFileSystem()->isPath(folderCandidate) ==
true) {
980 while (StringTools::startsWith(folderCandidate, installFolder) ==
true) {
981 folders.push_back(folderCandidate);
982 folderCandidate = FileSystem::getStandardFileSystem()->getPathName(folderCandidate);
986 sort(folders.begin(), folders.end());
987 reverse(folders.begin(), folders.end());
988 auto newEnd = unique(folders.begin(), folders.end());
989 folders.resize(distance(folders.begin(), newEnd));
990 for (
auto folder: folders) {
992 FileSystem::getStandardFileSystem()->removePath(
997 Console::println(
string(
"UninstallThread::run(): An error occurred: ") + exception.what());
1003 FileSystem::unsetFileSystem();
1004 vector<string> installerFiles;
1006 FileSystem::getStandardFileSystem()->list(
1007 installFolder +
"installer",
1011 Console::println(
string(
"UninstallThread::run(): An error occurred: ") + exception.what());
1013 for (
auto installerFile: installerFiles) {
1014 if (StringTools::endsWith(installerFile,
".ta") ==
true ||
1015 StringTools::endsWith(installerFile,
".ta.sha256") ==
true) {
1017 FileSystem::getStandardFileSystem()->removeFile(
1018 installFolder +
"installer",
1022 Console::println(
string(
"UninstallThread::run(): An error occurred: ") + exception.what());
1027 FileSystem::getStandardFileSystem()->removePath(installFolder +
"installer",
false);
1029 Console::println(
string(
"UninstallThread::run(): An error occurred: ") + exception.what());
1035 if (hadException ==
false) {
1037 FileSystem::getStandardFileSystem()->removeFile(
".",
"install.files.db");
1038 FileSystem::getStandardFileSystem()->removeFile(
".",
"install.components.db");
1039 FileSystem::getStandardFileSystem()->removeFile(
".",
"install.version.db");
1040 FileSystem::getStandardFileSystem()->removeFile(
".",
"console.log");
1043 hadException =
true;
1048 if (hadException ==
false) {
1050 FileSystem::getStandardFileSystem()->removePath(installFolder,
false);
1052 Console::println(
string(
"UninstallThread::run(): An error occurred: ") + exception.what());
1057 if (hadException ==
false) {
1066 if (installFolder[1] ==
':') drive = StringTools::substring(installFolder, 0, 2) +
" && ";
1072 "\"" + installFolder +
"/" +
"\"" +
1073 " && start cmd /c \"uninstall-finish.bat && del uninstall-finish.bat && cd .. && rmdir \"\"\"" + StringTools::replace(installFolder,
"/",
"\\") +
"\"\"\" > nul 2>&1\"").c_str()
1075 Application::exit(0);
1079 Console::println(
"UninstallThread::run(): done");
1085 UninstallThread* uninstallThread =
new UninstallThread(
this);
1086 uninstallThread->start();
1090 Console::println(
"Installer::performScreenAction(): Unhandled screen: " + to_string(
screen));
1101 FileSystem::getStandardFileSystem()->fileExists(
"install.files.db") ==
true &&
1102 FileSystem::getStandardFileSystem()->fileExists(
"install.components.db") ==
true &&
1103 FileSystem::getStandardFileSystem()->fileExists(
"install.version.db") ==
true;
1110 homeFolder = StringTools::replace(
string(getenv(
"USERPROFILE")),
'\\',
'/');
1135 if (node->
getId() ==
"button_next") {
1143 if (node->
getId() ==
"button_back") {
1146 timestamp = FileSystem::getStandardFileSystem()->getContentAsString(
".",
"install.version.db");
1148 Console::println(
string(
"Installer::onActionPerformed(): An error occurred: ") + exception.what());
1159 if (node->
getId() ==
"button_agree") {
1164 if (node->
getId() ==
"button_install") {
1169 if (node->
getId() ==
"button_cancel") {
1170 FileSystem::unsetFileSystem();
1174 FileSystem::getStandardFileSystem()->removeFile(
".", downloadedFile);
1176 Console::println(
"Installer::onActionPerformed(): Removing downloaded file failed: " + downloadedFile);
1179 Application::exit(0);
1181 if (node->
getId() ==
"button_finish") {
1187 if (installFolder[1] ==
':') drive = StringTools::substring(installFolder, 0, 2) +
" && ";
1188 string finishCommand;
1191 finishCommand+=
" && start cmd /c \"update-finish.bat && del update-finish.bat && " +
installerProperties.
get(
"launch",
"") +
".exe" +
"\"";
1200 "\"" + installFolder +
"/" +
"\"" +
1204 #elif defined(__APPLE__)
1205 Application::executeBackground(
1211 Application::executeBackground(
1220 if (installFolder[1] ==
':') drive = StringTools::substring(installFolder, 0, 2) +
" && ";
1228 "\"" + installFolder +
"/" +
"\"" +
1229 " && start cmd /c \"update-finish.bat && del update-finish.bat\"").c_str()
1234 FileSystem::unsetFileSystem();
1235 Application::exit(0);
1237 if (node->
getId() ==
"button_browse") {
1238 class OnBrowseAction:
public virtual Action
1241 void performAction()
override {
1242 installer->popUps->getFileDialogScreenController()->close();
1243 auto pathToShow = installer->popUps->getFileDialogScreenController()->getPathName() +
"/" + installer->installerProperties.get(
"name",
"TDME2 based application");
1244 dynamic_cast<GUIElementNode*
>(installer->engine->getGUI()->getScreen(
"installer_folder")->getNodeById(
"install_folder"))->getController()->setValue(
MutableString(pathToShow));
1251 OnBrowseAction(
Installer* installer): installer(installer) {
1258 vector<string> extensions;
1260 pathToShow = StringTools::replace(pathToShow,
"\\",
"/");
1261 while (FileSystem::getStandardFileSystem()->fileExists(pathToShow) ==
false || FileSystem::getStandardFileSystem()->isPath(pathToShow) ==
false) {
1262 pathToShow = FileSystem::getStandardFileSystem()->getPathName(pathToShow);
1264 if (StringTools::startsWith(pathToShow,
"/") ==
false) pathToShow =
homeFolder;
1271 new OnBrowseAction(
this)
1274 if (node->
getId() ==
"button_uninstall") {
1279 if (node->
getId() ==
"button_update") {
1284 if (node->
getId() ==
"button_repair") {
1289 if (StringTools::startsWith(node->
getId(),
"component") ==
true) {
1290 auto componentIdx = Integer::parse(StringTools::substring(node->
getId(),
string(
"component").size()));
1316 Console::println(
"Installer::mountInstallerFileSystem(): timestamp: " + (
timestamp.empty() ==
false?
timestamp:
"no timestamp"));
1319 auto installerArchiveFileNameStart = Application::getOSName() +
"-" + Application::getCPUName() +
"-Installer-" + (
timestamp.empty() ==
false?
timestamp +
".ta":
"");
1320 string installerArchiveFileName;
1322 vector<string> files;
1323 FileSystem::getStandardFileSystem()->list(
"installer", files);
1324 for (
auto file: files) {
1325 if (StringTools::startsWith(file, installerArchiveFileNameStart) ==
true &&
1326 StringTools::endsWith(file,
".ta") ==
true) {
1327 Console::println(
"Installer::main(): Have installer tdme archive file: " + file);
1328 installerArchiveFileName = file;
1331 if (installerArchiveFileName.empty() ==
true) {
1332 Console::println(
"Installer::main(): No installer TDME archive found. Exiting.");
1333 Application::exit(1);
1336 auto installerFileSystem =
new ArchiveFileSystem(
"installer/" + installerArchiveFileName);
1337 if (installerFileSystem->computeSHA256Hash() != FileSystem::getStandardFileSystem()->getContentAsString(
"installer", installerArchiveFileName +
".sha256")) {
1338 throw ExceptionBase(
"Installer::main(): Failed to verify: " + installerArchiveFileName +
", get new installer and try again");
1340 Console::println(
"Installer::mountInstallerFileSystem(): unmounting");
1343 FileSystem::unsetFileSystem();
1345 if (lastInstallerArchiveFileName.empty() ==
false && lastInstallerArchiveFileName !=
"installer/" + installerArchiveFileName) {
1347 Console::println(
"Installer::mountInstallerFileSystem(): deleting installer tdme archive file: " + lastInstallerArchiveFileName);
1349 FileSystem::getStandardFileSystem()->removeFile(
1350 FileSystem::getStandardFileSystem()->getPathName(lastInstallerArchiveFileName),
1351 FileSystem::getStandardFileSystem()->getFileName(lastInstallerArchiveFileName)
1353 FileSystem::getStandardFileSystem()->removeFile(
1354 FileSystem::getStandardFileSystem()->getPathName(lastInstallerArchiveFileName),
1355 FileSystem::getStandardFileSystem()->getFileName(lastInstallerArchiveFileName) +
".sha256"
1358 Console::println(
string(
"Installer::mountInstallerFileSystem(): An error occurred: ") + exception.what());
1361 Console::println(
"Installer::mountInstallerFileSystem(): mounting: " + installerArchiveFileName);
1362 FileSystem::setupFileSystem(installerFileSystem);
1364 Console::println(
string(
"Installer::mountInstallerFileSystem(): ") + exception.what());
1365 Application::exit(1);
1371 Console::println(
string(
"Installer ") + Version::getVersion());
1372 Console::println(Version::getCopyright());
1375 Console::println(
"Usage: Installer");
1376 Application::exit(1);
1378 #if defined(__APPLE__)
1381 auto executablePathName = string(argv[0]);
1382 if (executablePathName.find(
".app/Contents/MacOS/") != string::npos) {
1383 auto appBundleName = StringTools::substring(executablePathName, 0, executablePathName.rfind(
".app") +
string(
".app").size());
1384 auto workingPathName = StringTools::substring(appBundleName, 0, appBundleName.rfind(
'/'));
1385 FileSystem::getStandardFileSystem()->changePath(workingPathName);
1390 installer->run(argc, argv,
"Installer",
nullptr, Application::WINDOW_HINT_NOTRESIZEABLE);
1394 vector<string> files;
1395 archiveFileSystem->
list(pathName, files);
1396 for (
auto fileName: files) {
1397 if (archiveFileSystem->
isPath(pathName +
"/" + fileName) ==
false) {
1398 totalFiles.push_back((pathName.empty() ==
true?
"":pathName +
"/") + fileName);
1400 scanArchive(archiveFileSystem, totalFiles, pathName +
"/" + fileName);
1409 string pathCreating;
1413 pathCreating+= pathCreating.empty() ==
true?pathComponent:
"/" + pathComponent;
1415 pathCreating+=
"/" + pathComponent;
1417 if (FileSystem::getStandardFileSystem()->isDrive(pathCreating) ==
false && FileSystem::getStandardFileSystem()->fileExists(pathCreating) ==
false) {
1418 Console::println(
"Creating: " + pathCreating);
1419 FileSystem::getStandardFileSystem()->createPath(pathCreating);
1425 Application::cancelExit();
Application base class, please make sure to allocate application on heap to have correct application ...
string executableFileName
void setInputEventHandler(InputEventHandler *inputEventHandler)
Set input event handler.
void run(int argc, char **argv, const string &title, InputEventHandler *inputEventHandler=nullptr, int windowHints=WINDOW_HINT_NONE)
Run this application.
void reshape(int32_t width, int32_t height)
Reshape.
void display()
Renders the scene.
void initialize()
Initialize render engine.
void dispose()
Shutdown the engine.
void setSceneColor(const Color4 &sceneColor)
Set scene color.
void handleEvents()
Handle screen events.
void render()
Render GUIs.
GUIScreenNode * getScreen(const string &id)
Get screen.
void addRenderScreen(const string &screenId)
Add render screen.
void addScreen(const string &id, GUIScreenNode *screen)
Add screen.
void reset()
Removes all screens and caches.
void resetRenderScreens()
Reset render screens.
const string & getValue()
GUI node controller base class.
virtual const MutableString & getValue()=0
GUINodeController * getController()
GUI parent node base class thats supporting child nodes.
GUI screen node that represents a screen that can be rendered via GUI system.
void addChangeListener(GUIChangeListener *listener)
Add change listener.
GUINode * getNodeById(const string &nodeId)
Get GUI node by id.
void addActionListener(GUIActionListener *listener)
Add action listener.
void setPassword(const string &password)
Set password.
void execute()
Execute HTTP request.
void setUsername(const string &username)
Set username.
void setMethod(const string &method)
Set method.
stringstream & getResponse()
void setURL(const string &url)
Set URL.
void setPassword(const string &password)
Set password.
void start()
Starts the HTTP download to file.
void join()
Wait until underlying thread has finished.
void setUsername(const string &username)
Set username.
void reset()
Reset this HTTP client.
void setFile(const string &file)
Set file to download to.
void setURL(const string &url)
Set URL.
Archive file system implementation.
const string getPathName(const string &fileName) override
Get path name.
bool isPath(const string &pathName) override
Check if file is a path.
uint64_t getFileSize(const string &pathName, const string &fileName) override
Return file size of given file.
const string computeSHA256Hash()
Compute SHA256 hash.
const string getFileName(const string &path, const string &fileName) override
Get file name.
bool fileExists(const string &fileName) override
Check if file exists.
bool isExecutable(const string &pathName, const string &fileName) override
Returns if file is a executable file.
void list(const string &pathName, vector< string > &files, FileNameFilter *filter=nullptr, bool addDrives=false) override
List files for given path and filter by a file name filter if not null.
void getContent(const string &pathName, const string &fileName, vector< uint8_t > &content) override
Get file content.
File system singleton class.
void unlock()
Unlocks this mutex.
void lock()
Locks the mutex, additionally mutex locks will block until other locks have been unlocked.
bool equals(const string &s2) const
Equals.
const string & getString() const
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.
std::exception Exception
Exception base class.
GUI action listener interface.
GUI change listener interface.