diff --git a/include/Settings.h b/include/Settings.h index 016bef14..4254877f 100644 --- a/include/Settings.h +++ b/include/Settings.h @@ -79,6 +79,7 @@ struct Settings { General_Map, General_AuthKey, General_Private, + General_Offline, General_Port, General_MaxCars, General_LogChat, diff --git a/src/LuaAPI.cpp b/src/LuaAPI.cpp index fe16fe71..06368120 100644 --- a/src/LuaAPI.cpp +++ b/src/LuaAPI.cpp @@ -325,6 +325,14 @@ void LuaAPI::MP::Set(int ConfigID, sol::object NewValue) { beammp_lua_error("set invalid argument [2] expected boolean"); } break; + case 8: // Offline play + if (NewValue.is()) { + Application::Settings.set(Settings::Key::General_Offline, NewValue.as()); + beammp_info(std::string("Set `Offline` to ") + (Application::Settings.getAsBool(Settings::Key::General_Offline) ? "true" : "false")); + } else { + beammp_lua_error("set invalid argument [2] expected boolean"); + } + break; default: beammp_warn("Invalid config ID \"" + std::to_string(ConfigID) + "\". Use `MP.Settings.*` enum for this."); break; diff --git a/src/Settings.cpp b/src/Settings.cpp index a8dfdb84..3d1320e1 100644 --- a/src/Settings.cpp +++ b/src/Settings.cpp @@ -28,6 +28,7 @@ Settings::Settings() { { General_Map, std::string("/levels/gridmap_v2/info.json") }, { General_AuthKey, std::string("") }, { General_Private, true }, + { General_Offline, false }, { General_Port, 30814 }, { General_MaxCars, 1 }, { General_LogChat, true }, @@ -47,6 +48,7 @@ Settings::Settings() { { { "General", "Map" }, { General_Map, READ_WRITE } }, { { "General", "AuthKey" }, { General_AuthKey, NO_ACCESS } }, { { "General", "Private" }, { General_Private, READ_ONLY } }, + { { "General", "Offline" }, { General_Offline, READ_ONLY } }, { { "General", "Port" }, { General_Port, READ_ONLY } }, { { "General", "MaxCars" }, { General_MaxCars, READ_WRITE } }, { { "General", "LogChat" }, { General_LogChat, READ_ONLY } }, diff --git a/src/TConfig.cpp b/src/TConfig.cpp index 0c902a86..c359c971 100644 --- a/src/TConfig.cpp +++ b/src/TConfig.cpp @@ -34,6 +34,8 @@ static constexpr std::string_view StrDebug = "Debug"; static constexpr std::string_view EnvStrDebug = "BEAMMP_DEBUG"; static constexpr std::string_view StrPrivate = "Private"; static constexpr std::string_view EnvStrPrivate = "BEAMMP_PRIVATE"; +static constexpr std::string_view StrOffline = "Offline"; +static constexpr std::string_view EnvStrOffline = "BEAMMP_OFFLINE"; static constexpr std::string_view StrPort = "Port"; static constexpr std::string_view EnvStrPort = "BEAMMP_PORT"; static constexpr std::string_view StrMaxCars = "MaxCars"; @@ -134,6 +136,7 @@ void TConfig::FlushToFile() { SetComment(data["General"][StrLogChat.data()].comments(), " Whether to log chat messages in the console / log"); data["General"][StrDebug.data()] = Application::Settings.getAsBool(Settings::Key::General_Debug); data["General"][StrPrivate.data()] = Application::Settings.getAsBool(Settings::Key::General_Private); + data["General"][StrOffline.data()] = Application::Settings.getAsBool(Settings::Key::General_Offline); SetComment(data["General"][StrInformationPacket.data()].comments(), " Whether to allow unconnected clients to get the public server information without joining"); data["General"][StrInformationPacket.data()] = Application::Settings.getAsBool(Settings::Key::General_InformationPacket); data["General"][StrAllowGuests.data()] = Application::Settings.getAsBool(Settings::Key::General_AllowGuests); @@ -248,6 +251,7 @@ void TConfig::ParseFromFile(std::string_view name) { // Read into new Settings Singleton TryReadValue(data, "General", StrDebug, EnvStrDebug, Settings::Key::General_Debug); TryReadValue(data, "General", StrPrivate, EnvStrPrivate, Settings::Key::General_Private); + TryReadValue(data, "General", StrOffline, EnvStrOffline, Settings::Key::General_Offline); TryReadValue(data, "General", StrInformationPacket, EnvStrInformationPacket, Settings::Key::General_InformationPacket); if (Env::Get(Env::Key::PROVIDER_PORT_ENV).has_value()) { TryReadValue(data, "General", StrPort, Env::Get(Env::Key::PROVIDER_PORT_ENV).value(), Settings::Key::General_Port); @@ -302,6 +306,7 @@ void TConfig::PrintDebug() { } beammp_debug(std::string(StrDebug) + ": " + std::string(Application::Settings.getAsBool(Settings::Key::General_Debug) ? "true" : "false")); beammp_debug(std::string(StrPrivate) + ": " + std::string(Application::Settings.getAsBool(Settings::Key::General_Private) ? "true" : "false")); + beammp_debug(std::string(StrOffline) + ": " + std::string(Application::Settings.getAsBool(Settings::Key::General_Offline) ? "true" : "false")); beammp_debug(std::string(StrInformationPacket) + ": " + std::string(Application::Settings.getAsBool(Settings::Key::General_InformationPacket) ? "true" : "false")); beammp_debug(std::string(StrPort) + ": " + std::to_string(Application::Settings.getAsInt(Settings::Key::General_Port))); beammp_debug(std::string(StrMaxCars) + ": " + std::to_string(Application::Settings.getAsInt(Settings::Key::General_MaxCars))); diff --git a/src/THeartbeatThread.cpp b/src/THeartbeatThread.cpp index 36cdde1c..9a8a03e8 100644 --- a/src/THeartbeatThread.cpp +++ b/src/THeartbeatThread.cpp @@ -61,6 +61,11 @@ void THeartbeatThread::operator()() { nlohmann::json Doc; bool Ok = false; + + if (Application::Settings.getAsBool(Settings::Key::General_Offline)) { + beammp_warn("Server is in Offline Mode. Your server will not appear in the global server list. Direct connect will still work with LAN connections."); + return; + } for (const auto& Url : Application::GetBackendUrlsInOrder()) { T = Http::POST(Url + Target, Body, "application/json", &ResponseCode, { { "api-v", "2" } }); @@ -113,7 +118,6 @@ void THeartbeatThread::operator()() { beammp_warn("Backend failed to respond to a heartbeat. Your server may temporarily disappear from the server list. This is not an error, and will likely resolve itself soon. Direct connect will still work."); } } - if (Ok && !isAuth && !Application::Settings.getAsBool(Settings::Key::General_Private)) { if (Status == "2000") { beammp_info(("Authenticated! " + Message)); @@ -145,6 +149,7 @@ std::string THeartbeatThread::GenerateCall() { { "port", std::to_string(Application::Settings.getAsInt(Settings::Key::General_Port)) }, { "map", Application::Settings.getAsString(Settings::Key::General_Map) }, { "private", Application::Settings.getAsBool(Settings::Key::General_Private) ? "true" : "false" }, + { "offline", Application::Settings.getAsBool(Settings::Key::General_Offline) ? "true" : "false" }, { "version", Application::ServerVersionString() }, { "clientversion", Application::ClientMinimumVersion().AsString() }, { "name", Application::Settings.getAsString(Settings::Key::General_Name) }, diff --git a/src/TNetwork.cpp b/src/TNetwork.cpp index 3a40543f..24ea40d7 100644 --- a/src/TNetwork.cpp +++ b/src/TNetwork.cpp @@ -367,32 +367,41 @@ std::shared_ptr TNetwork::Authentication(TConnection&& RawConnection) { return nullptr; } - beammp_debug("Response from authentication backend: " + AuthResStr); + // If we are in "offline mode", we want to do something different for auth + if (Application::Settings.getAsBool(Settings::Key::General_Offline)) { + // skip auth altogether. + // this prevents players from getting kicked for not being logged in. + + // TODO: authenticate using a different method here. - try { - nlohmann::json AuthRes = nlohmann::json::parse(AuthResStr); - - if (AuthRes["username"].is_string() && AuthRes["roles"].is_string() - && AuthRes["guest"].is_boolean() && AuthRes["identifiers"].is_array()) { - - Client->SetName(AuthRes["username"]); - Client->SetRoles(AuthRes["roles"]); - Client->SetIsGuest(AuthRes["guest"]); - for (const auto& ID : AuthRes["identifiers"]) { - auto Raw = std::string(ID); - auto SepIndex = Raw.find(':'); - Client->SetIdentifier(Raw.substr(0, SepIndex), Raw.substr(SepIndex + 1)); + } else { + beammp_debug("Response from authentication backend: " + AuthResStr); + + try { + nlohmann::json AuthRes = nlohmann::json::parse(AuthResStr); + + if (AuthRes["username"].is_string() && AuthRes["roles"].is_string() + && AuthRes["guest"].is_boolean() && AuthRes["identifiers"].is_array()) { + + Client->SetName(AuthRes["username"]); + Client->SetRoles(AuthRes["roles"]); + Client->SetIsGuest(AuthRes["guest"]); + for (const auto& ID : AuthRes["identifiers"]) { + auto Raw = std::string(ID); + auto SepIndex = Raw.find(':'); + Client->SetIdentifier(Raw.substr(0, SepIndex), Raw.substr(SepIndex + 1)); + } + } else { + beammp_error("Invalid authentication data received from authentication backend"); + ClientKick(*Client, "Invalid authentication data!"); + return nullptr; } - } else { - beammp_error("Invalid authentication data received from authentication backend"); - ClientKick(*Client, "Invalid authentication data!"); + } catch (const std::exception& e) { + beammp_errorf("Client sent invalid key. Error was: {}", e.what()); + // TODO: we should really clarify that this was a backend response or parsing error + ClientKick(*Client, "Invalid key! Please restart your game."); return nullptr; } - } catch (const std::exception& e) { - beammp_errorf("Client sent invalid key. Error was: {}", e.what()); - // TODO: we should really clarify that this was a backend response or parsing error - ClientKick(*Client, "Invalid key! Please restart your game."); - return nullptr; } beammp_debug("Name -> " + Client->GetName() + ", Guest -> " + std::to_string(Client->IsGuest()) + ", Roles -> " + Client->GetRoles());