123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165 |
- /*
- * GlobalLobbyProcessor.cpp, part of VCMI engine
- *
- * Authors: listed in file AUTHORS in main folder
- *
- * License: GNU General Public License v2.0 or later
- * Full text of license available in license.txt file, in main folder
- *
- */
- #include "StdInc.h"
- #include "GlobalLobbyProcessor.h"
- #include "CVCMIServer.h"
- #include "../lib/CConfigHandler.h"
- #include "../lib/json/JsonUtils.h"
- GlobalLobbyProcessor::GlobalLobbyProcessor(CVCMIServer & owner)
- : owner(owner)
- {
- logGlobal->info("Connecting to lobby server");
- establishNewConnection();
- }
- void GlobalLobbyProcessor::establishNewConnection()
- {
- std::string hostname = settings["lobby"]["hostname"].String();
- int16_t port = settings["lobby"]["port"].Integer();
- owner.getNetworkHandler().connectToRemote(*this, hostname, port);
- }
- void GlobalLobbyProcessor::onDisconnected(const std::shared_ptr<INetworkConnection> & connection, const std::string & errorMessage)
- {
- if (connection == controlConnection)
- {
- owner.setState(EServerState::SHUTDOWN);
- return;
- }
- else
- {
- if (owner.getState() == EServerState::LOBBY)
- {
- for (auto const & proxy : proxyConnections)
- {
- if (proxy.second == connection)
- {
- JsonNode message;
- message["type"].String() = "leaveGameRoom";
- message["accountID"].String() = proxy.first;
- sendMessage(controlConnection, message);
- break;
- }
- }
- }
- // player disconnected
- owner.onDisconnected(connection, errorMessage);
- }
- }
- void GlobalLobbyProcessor::onPacketReceived(const std::shared_ptr<INetworkConnection> & connection, const std::vector<std::byte> & message)
- {
- if (connection == controlConnection)
- {
- JsonNode json(message.data(), message.size());
- if(json["type"].String() == "operationFailed")
- return receiveOperationFailed(json);
- if(json["type"].String() == "serverLoginSuccess")
- return receiveServerLoginSuccess(json);
- if(json["type"].String() == "accountJoinsRoom")
- return receiveAccountJoinsRoom(json);
- logGlobal->error("Received unexpected message from lobby server of type '%s' ", json["type"].String());
- }
- else
- {
- // received game message via proxy connection
- owner.onPacketReceived(connection, message);
- }
- }
- void GlobalLobbyProcessor::receiveOperationFailed(const JsonNode & json)
- {
- logGlobal->info("Lobby: Failed to login into a lobby server!");
- owner.setState(EServerState::SHUTDOWN);
- }
- void GlobalLobbyProcessor::receiveServerLoginSuccess(const JsonNode & json)
- {
- // no-op, wait just for any new commands from lobby
- logGlobal->info("Lobby: Succesfully connected to lobby server");
- owner.startAcceptingIncomingConnections();
- }
- void GlobalLobbyProcessor::receiveAccountJoinsRoom(const JsonNode & json)
- {
- std::string accountID = json["accountID"].String();
- logGlobal->info("Lobby: Account %s will join our room!", accountID);
- assert(proxyConnections.count(accountID) == 0);
- proxyConnections[accountID] = nullptr;
- establishNewConnection();
- }
- void GlobalLobbyProcessor::onConnectionFailed(const std::string & errorMessage)
- {
- owner.setState(EServerState::SHUTDOWN);
- }
- void GlobalLobbyProcessor::onConnectionEstablished(const std::shared_ptr<INetworkConnection> & connection)
- {
- if (controlConnection == nullptr)
- {
- controlConnection = connection;
- logGlobal->info("Connection to lobby server established");
- JsonNode toSend;
- toSend["type"].String() = "serverLogin";
- toSend["gameRoomID"].String() = owner.uuid;
- toSend["accountID"] = settings["lobby"]["accountID"];
- toSend["accountCookie"] = settings["lobby"]["accountCookie"];
- toSend["version"].String() = VCMI_VERSION_STRING;
- sendMessage(connection, toSend);
- }
- else
- {
- // Proxy connection for a player
- std::string guestAccountID;
- for (auto const & proxies : proxyConnections)
- if (proxies.second == nullptr)
- guestAccountID = proxies.first;
- JsonNode toSend;
- toSend["type"].String() = "serverProxyLogin";
- toSend["gameRoomID"].String() = owner.uuid;
- toSend["guestAccountID"].String() = guestAccountID;
- toSend["accountCookie"] = settings["lobby"]["accountCookie"];
- sendMessage(connection, toSend);
- proxyConnections[guestAccountID] = connection;
- owner.onNewConnection(connection);
- }
- }
- void GlobalLobbyProcessor::sendChangeRoomDescription(const std::string & description)
- {
- JsonNode toSend;
- toSend["type"].String() = "changeRoomDescription";
- toSend["description"].String() = description;
- sendMessage(controlConnection, toSend);
- }
- void GlobalLobbyProcessor::sendMessage(const NetworkConnectionPtr & targetConnection, const JsonNode & toSend)
- {
- assert(JsonUtils::validate(toSend, "vcmi:lobbyProtocol/" + toSend["type"].String(), toSend["type"].String() + " pack"));
- targetConnection->sendPacket(toSend.toBytes());
- }
|