diff --git a/source/client/main.cpp b/source/client/main.cpp index b26bb0217..4b5171ed5 100644 --- a/source/client/main.cpp +++ b/source/client/main.cpp @@ -43,8 +43,14 @@ extern "C" { int main(int argc, char **argv) { ClientApplication app(argc, argv); - app.run(); - return 0; + int exitCode = app.run(); + +#ifdef _WIN32 + if (exitCode != 0) + system("pause"); +#endif + + return exitCode; } diff --git a/source/server/lua/ServerModLoader.cpp b/source/server/lua/ServerModLoader.cpp index 31c2d1306..73ec7b000 100644 --- a/source/server/lua/ServerModLoader.cpp +++ b/source/server/lua/ServerModLoader.cpp @@ -58,23 +58,31 @@ void ServerModLoader::loadMods() { mod.path = entry.path().string(); mod.id = config["id"]; - if (mod.id != entry.path().filename().string()) - gkWarning() << ("For mod '" + mod.id + "': Folder name '" + entry.path().filename().string() + "' is different from mod id").c_str(); - - auto it = mods.find(mod.id); - if (it == mods.end()) { - sol::table dependencies = config["dependencies"]; - for (auto &it : dependencies) { - mod.dependencies.emplace_back(it.second.as()); + if (!mod.id.empty()) { + if (mod.id != entry.path().filename().string()) + gkWarning() << ("Mod ID '" + mod.id + "' is different from folder name '" + entry.path().filename().string() + "'").c_str(); + + auto it = mods.find(mod.id); + if (it == mods.end()) { + sol::object dependencies = config["dependencies"]; + if (dependencies.get_type() == sol::type::table) { + for (auto &it : dependencies.as()) { + mod.dependencies.emplace_back(it.second.as()); + } + } + else + gkWarning() << ("Failed to load dependencies for mod '" + mod.id + "'").c_str(); + + mods.emplace(mod.id, mod); + } + else { + gkError() << ("Trying to load mod '" + mod.id + "' twice").c_str(); + gkError() << "Loaded path: " << it->second.path.string(); + gkError() << "Current path:" << mod.path.string(); } - - mods.emplace(mod.id, mod); - } - else { - gkWarning() << ("Trying to load mod '" + mod.id + "' twice").c_str(); - gkWarning() << "Loaded path: " << it->second.path.string(); - gkWarning() << "Current path:" << mod.path.string(); } + else + gkError() << ("Failed to load mod '" + mod.path.filename().string() + "': Mod ID required in 'config.lua'").c_str(); } catch (sol::error &e) { gkError() << e.what(); @@ -82,11 +90,38 @@ void ServerModLoader::loadMods() { } else - gkWarning() << ("The mod at '" + entry.path().string() + "' doesn't contain a 'config.lua' file.").c_str(); + gkError() << ("The mod at '" + entry.path().string() + "' doesn't contain a 'config.lua' file.").c_str(); + } + + { + // Small BFS to check cyclic dependencies + for (auto &modit : mods) { + std::queue queue; + queue.emplace(modit.first); + while (!queue.empty()) { + std::string modID = queue.front(); + queue.pop(); + + auto it = mods.find(modID); + if (it == mods.end()) + break; + + for (auto &dep : it->second.dependencies) { + if (dep == modit.first) + throw EXCEPTION("Cyclic dependency detected for mod '" + + modit.second.id + "' in mod '" + it->second.id + "'"); + + queue.emplace(dep); + } + } + } } // TODO: Build a dependency graph - // TODO: Make sure to handle cyclic dependencies + // Two types of dependencies should be handled + // - Required (cyclic depedencies not handled) + // - Optional (cyclic dependencies handled) + // TODO: Load 'mods/config.lua' to let user define a preferred mod path (see #82) m_scriptEngine.init(); @@ -100,7 +135,7 @@ void ServerModLoader::loadMods() { m_scriptEngine.lua().safe_script_file("init.lua"); } catch (const sol::error &e) { - gkError() << "Error: Failed eo load mod at" << it.second.path.string(); + gkError() << "Error: Failed to load mod at" << it.second.path.string(); gkError() << e.what(); } @@ -109,7 +144,7 @@ void ServerModLoader::loadMods() { gkInfo() << "Mod" << it.first << "loaded"; } else - gkWarning() << ("The mod at '" + it.second.path.string() + "' doesn't contain an 'init.lua' file.").c_str(); + gkError() << ("The mod at '" + it.second.path.string() + "' doesn't contain an 'init.lua' file.").c_str(); } for (auto &it : m_mods) { diff --git a/source/server/main.cpp b/source/server/main.cpp index ec48c613a..074af02ff 100644 --- a/source/server/main.cpp +++ b/source/server/main.cpp @@ -28,8 +28,14 @@ int main(int argc, char **argv) { ServerApplication app(argc, argv); - app.run(); - return 0; + int exitCode = app.run(); + +#ifdef _WIN32 + if (exitCode != 0) + system("pause"); +#endif + + return exitCode; }