| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574 |
- //
- // Copyright (c) 2008-2018 the Urho3D project.
- //
- // Permission is hereby granted, free of charge, to any person obtaining a copy
- // of this software and associated documentation files (the "Software"), to deal
- // in the Software without restriction, including without limitation the rights
- // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- // copies of the Software, and to permit persons to whom the Software is
- // furnished to do so, subject to the following conditions:
- //
- // The above copyright notice and this permission notice shall be included in
- // all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- // THE SOFTWARE.
- //
- #include "../Precompiled.h"
- #include "../Core/Context.h"
- #include "../Core/CoreEvents.h"
- #include "../Core/Profiler.h"
- #include "../Engine/EngineEvents.h"
- #include "../IO/FileSystem.h"
- #include "../Input/InputEvents.h"
- #include "../IO/IOEvents.h"
- #include "../IO/Log.h"
- #include "../IO/MemoryBuffer.h"
- #include "../Network/HttpRequest.h"
- #include "../Network/Network.h"
- #include "../Network/NetworkEvents.h"
- #include "../Network/NetworkPriority.h"
- #include "../Network/Protocol.h"
- #include "../Scene/Scene.h"
- #include <SLikeNet/MessageIdentifiers.h>
- #include <SLikeNet/NatPunchthroughClient.h>
- #include <SLikeNet/peerinterface.h>
- #include <SLikeNet/statistics.h>
- #include <SLikeNet/FullyConnectedMesh2.h>
- #include <SLikeNet/DS_List.h>
- #include <SLikeNet/BitStream.h>
- #include <SLikeNet/ReadyEvent.h>
- #include <SLikeNet/ConnectionGraph2.h>
- #ifdef SendMessage
- #undef SendMessage
- #endif
- #include "../DebugNew.h"
- namespace Urho3D
- {
- static const char* RAKNET_MESSAGEID_STRINGS[] = {
- "ID_CONNECTED_PING",
- "ID_UNCONNECTED_PING",
- "ID_UNCONNECTED_PING_OPEN_CONNECTIONS",
- "ID_CONNECTED_PONG",
- "ID_DETECT_LOST_CONNECTIONS",
- "ID_OPEN_CONNECTION_REQUEST_1",
- "ID_OPEN_CONNECTION_REPLY_1",
- "ID_OPEN_CONNECTION_REQUEST_2",
- "ID_OPEN_CONNECTION_REPLY_2",
- "ID_CONNECTION_REQUEST",
- "ID_REMOTE_SYSTEM_REQUIRES_PUBLIC_KEY",
- "ID_OUR_SYSTEM_REQUIRES_SECURITY",
- "ID_PUBLIC_KEY_MISMATCH",
- "ID_OUT_OF_BAND_INTERNAL",
- "ID_SND_RECEIPT_ACKED",
- "ID_SND_RECEIPT_LOSS",
- "ID_CONNECTION_REQUEST_ACCEPTED",
- "ID_CONNECTION_ATTEMPT_FAILED",
- "ID_ALREADY_CONNECTED",
- "ID_NEW_INCOMING_CONNECTION",
- "ID_NO_FREE_INCOMING_CONNECTIONS",
- "ID_DISCONNECTION_NOTIFICATION",
- "ID_CONNECTION_LOST",
- "ID_CONNECTION_BANNED",
- "ID_INVALID_PASSWORD",
- "ID_INCOMPATIBLE_PROTOCOL_VERSION",
- "ID_IP_RECENTLY_CONNECTED",
- "ID_TIMESTAMP",
- "ID_UNCONNECTED_PONG",
- "ID_ADVERTISE_SYSTEM",
- "ID_DOWNLOAD_PROGRESS",
- "ID_REMOTE_DISCONNECTION_NOTIFICATION",
- "ID_REMOTE_CONNECTION_LOST",
- "ID_REMOTE_NEW_INCOMING_CONNECTION",
- "ID_FILE_LIST_TRANSFER_HEADER",
- "ID_FILE_LIST_TRANSFER_FILE",
- "ID_FILE_LIST_REFERENCE_PUSH_ACK",
- "ID_DDT_DOWNLOAD_REQUEST",
- "ID_TRANSPORT_STRING",
- "ID_REPLICA_MANAGER_CONSTRUCTION",
- "ID_REPLICA_MANAGER_SCOPE_CHANGE",
- "ID_REPLICA_MANAGER_SERIALIZE",
- "ID_REPLICA_MANAGER_DOWNLOAD_STARTED",
- "ID_REPLICA_MANAGER_DOWNLOAD_COMPLETE",
- "ID_RAKVOICE_OPEN_CHANNEL_REQUEST",
- "ID_RAKVOICE_OPEN_CHANNEL_REPLY",
- "ID_RAKVOICE_CLOSE_CHANNEL",
- "ID_RAKVOICE_DATA",
- "ID_AUTOPATCHER_GET_CHANGELIST_SINCE_DATE",
- "ID_AUTOPATCHER_CREATION_LIST",
- "ID_AUTOPATCHER_DELETION_LIST",
- "ID_AUTOPATCHER_GET_PATCH",
- "ID_AUTOPATCHER_PATCH_LIST",
- "ID_AUTOPATCHER_REPOSITORY_FATAL_ERROR",
- "ID_AUTOPATCHER_CANNOT_DOWNLOAD_ORIGINAL_UNMODIFIED_FILES",
- "ID_AUTOPATCHER_FINISHED_INTERNAL",
- "ID_AUTOPATCHER_FINISHED",
- "ID_AUTOPATCHER_RESTART_APPLICATION",
- "ID_NAT_PUNCHTHROUGH_REQUEST",
- "ID_NAT_CONNECT_AT_TIME",
- "ID_NAT_GET_MOST_RECENT_PORT",
- "ID_NAT_CLIENT_READY",
- "ID_NAT_TARGET_NOT_CONNECTED",
- "ID_NAT_TARGET_UNRESPONSIVE",
- "ID_NAT_CONNECTION_TO_TARGET_LOST",
- "ID_NAT_ALREADY_IN_PROGRESS",
- "ID_NAT_PUNCHTHROUGH_FAILED",
- "ID_NAT_PUNCHTHROUGH_SUCCEEDED",
- "ID_READY_EVENT_SET",
- "ID_READY_EVENT_UNSET",
- "ID_READY_EVENT_ALL_SET",
- "ID_READY_EVENT_QUERY",
- "ID_LOBBY_GENERAL",
- "ID_RPC_REMOTE_ERROR",
- "ID_RPC_PLUGIN",
- "ID_FILE_LIST_REFERENCE_PUSH",
- "ID_READY_EVENT_FORCE_ALL_SET",
- "ID_ROOMS_EXECUTE_FUNC",
- "ID_ROOMS_LOGON_STATUS",
- "ID_ROOMS_HANDLE_CHANGE",
- "ID_LOBBY2_SEND_MESSAGE",
- "ID_LOBBY2_SERVER_ERROR",
- "ID_FCM2_NEW_HOST",
- "ID_FCM2_REQUEST_FCMGUID",
- "ID_FCM2_RESPOND_CONNECTION_COUNT",
- "ID_FCM2_INFORM_FCMGUID",
- "ID_FCM2_UPDATE_MIN_TOTAL_CONNECTION_COUNT",
- "ID_FCM2_VERIFIED_JOIN_START",
- "ID_FCM2_VERIFIED_JOIN_CAPABLE",
- "ID_FCM2_VERIFIED_JOIN_FAILED",
- "ID_FCM2_VERIFIED_JOIN_ACCEPTED",
- "ID_FCM2_VERIFIED_JOIN_REJECTED",
- "ID_UDP_PROXY_GENERAL",
- "ID_SQLite3_EXEC",
- "ID_SQLite3_UNKNOWN_DB",
- "ID_SQLLITE_LOGGER",
- "ID_NAT_TYPE_DETECTION_REQUEST",
- "ID_NAT_TYPE_DETECTION_RESULT",
- "ID_ROUTER_2_INTERNAL",
- "ID_ROUTER_2_FORWARDING_NO_PATH",
- "ID_ROUTER_2_FORWARDING_ESTABLISHED",
- "ID_ROUTER_2_REROUTED",
- "ID_TEAM_BALANCER_INTERNAL",
- "ID_TEAM_BALANCER_REQUESTED_TEAM_FULL",
- "ID_TEAM_BALANCER_REQUESTED_TEAM_LOCKED",
- "ID_TEAM_BALANCER_TEAM_REQUESTED_CANCELLED",
- "ID_TEAM_BALANCER_TEAM_ASSIGNED",
- "ID_LIGHTSPEED_INTEGRATION",
- "ID_XBOX_LOBBY",
- "ID_TWO_WAY_AUTHENTICATION_INCOMING_CHALLENGE_SUCCESS",
- "ID_TWO_WAY_AUTHENTICATION_OUTGOING_CHALLENGE_SUCCESS",
- "ID_TWO_WAY_AUTHENTICATION_INCOMING_CHALLENGE_FAILURE",
- "ID_TWO_WAY_AUTHENTICATION_OUTGOING_CHALLENGE_FAILURE",
- "ID_TWO_WAY_AUTHENTICATION_OUTGOING_CHALLENGE_TIMEOUT",
- "ID_TWO_WAY_AUTHENTICATION_NEGOTIATION",
- "ID_CLOUD_POST_REQUEST",
- "ID_CLOUD_RELEASE_REQUEST",
- "ID_CLOUD_GET_REQUEST",
- "ID_CLOUD_GET_RESPONSE",
- "ID_CLOUD_UNSUBSCRIBE_REQUEST",
- "ID_CLOUD_SERVER_TO_SERVER_COMMAND",
- "ID_CLOUD_SUBSCRIPTION_NOTIFICATION",
- "ID_LIB_VOICE",
- "ID_RELAY_PLUGIN",
- "ID_NAT_REQUEST_BOUND_ADDRESSES",
- "ID_NAT_RESPOND_BOUND_ADDRESSES",
- "ID_FCM2_UPDATE_USER_CONTEXT",
- "ID_RESERVED_3",
- "ID_RESERVED_4",
- "ID_RESERVED_5",
- "ID_RESERVED_6",
- "ID_RESERVED_7",
- "ID_RESERVED_8",
- "ID_RESERVED_9",
- "ID_USER_PACKET_ENUM"
- };
- static const int DEFAULT_UPDATE_FPS = 30;
- static const int SERVER_TIMEOUT_TIME = 5000;
- Network::Network(Context* context) :
- Object(context),
- updateFps_(DEFAULT_UPDATE_FPS),
- simulatedLatency_(0),
- simulatedPacketLoss_(0.0f),
- updateInterval_(1.0f / (float)DEFAULT_UPDATE_FPS),
- updateAcc_(0.0f),
- isServer_(false),
- scene_(nullptr),
- natPunchServerAddress_(nullptr),
- remoteGUID_(nullptr),
- networkMode_(SERVER_CLIENT)
- {
- rakPeer_ = SLNet::RakPeerInterface::GetInstance();
- rakPeerClient_ = SLNet::RakPeerInterface::GetInstance();
- fullyConnectedMesh2_ = SLNet::FullyConnectedMesh2::GetInstance();
- rakPeer_->AttachPlugin(fullyConnectedMesh2_);
- fullyConnectedMesh2_->SetAutoparticipateConnections(false);
- readyEvent_ = SLNet::ReadyEvent::GetInstance();
- rakPeer_->AttachPlugin(readyEvent_);
- connectionGraph2_ = SLNet::ConnectionGraph2::GetInstance();
- rakPeer_->AttachPlugin(connectionGraph2_);
- rakPeer_->SetTimeoutTime(SERVER_TIMEOUT_TIME, SLNet::UNASSIGNED_SYSTEM_ADDRESS);
- rakPeerClient_->SetTimeoutTime(SERVER_TIMEOUT_TIME, SLNet::UNASSIGNED_SYSTEM_ADDRESS);
- SetPassword("");
- SetDiscoveryBeacon(VariantMap());
- natPunchthroughClient_ = new SLNet::NatPunchthroughClient;
- natPunchthroughServerClient_ = new SLNet::NatPunchthroughClient;
- SetNATServerInfo("127.0.0.1", 61111);
- // Register Network library object factories
- RegisterNetworkLibrary(context_);
- SubscribeToEvent(E_BEGINFRAME, URHO3D_HANDLER(Network, HandleBeginFrame));
- SubscribeToEvent(E_RENDERUPDATE, URHO3D_HANDLER(Network, HandleRenderUpdate));
- // Blacklist remote events which are not to be allowed to be registered in any case
- blacklistedRemoteEvents_.Insert(E_CONSOLECOMMAND);
- blacklistedRemoteEvents_.Insert(E_LOGMESSAGE);
- blacklistedRemoteEvents_.Insert(E_BEGINFRAME);
- blacklistedRemoteEvents_.Insert(E_UPDATE);
- blacklistedRemoteEvents_.Insert(E_POSTUPDATE);
- blacklistedRemoteEvents_.Insert(E_RENDERUPDATE);
- blacklistedRemoteEvents_.Insert(E_ENDFRAME);
- blacklistedRemoteEvents_.Insert(E_MOUSEBUTTONDOWN);
- blacklistedRemoteEvents_.Insert(E_MOUSEBUTTONUP);
- blacklistedRemoteEvents_.Insert(E_MOUSEMOVE);
- blacklistedRemoteEvents_.Insert(E_MOUSEWHEEL);
- blacklistedRemoteEvents_.Insert(E_KEYDOWN);
- blacklistedRemoteEvents_.Insert(E_KEYUP);
- blacklistedRemoteEvents_.Insert(E_TEXTINPUT);
- blacklistedRemoteEvents_.Insert(E_JOYSTICKCONNECTED);
- blacklistedRemoteEvents_.Insert(E_JOYSTICKDISCONNECTED);
- blacklistedRemoteEvents_.Insert(E_JOYSTICKBUTTONDOWN);
- blacklistedRemoteEvents_.Insert(E_JOYSTICKBUTTONUP);
- blacklistedRemoteEvents_.Insert(E_JOYSTICKAXISMOVE);
- blacklistedRemoteEvents_.Insert(E_JOYSTICKHATMOVE);
- blacklistedRemoteEvents_.Insert(E_TOUCHBEGIN);
- blacklistedRemoteEvents_.Insert(E_TOUCHEND);
- blacklistedRemoteEvents_.Insert(E_TOUCHMOVE);
- blacklistedRemoteEvents_.Insert(E_GESTURERECORDED);
- blacklistedRemoteEvents_.Insert(E_GESTUREINPUT);
- blacklistedRemoteEvents_.Insert(E_MULTIGESTURE);
- blacklistedRemoteEvents_.Insert(E_DROPFILE);
- blacklistedRemoteEvents_.Insert(E_INPUTFOCUS);
- blacklistedRemoteEvents_.Insert(E_MOUSEVISIBLECHANGED);
- blacklistedRemoteEvents_.Insert(E_EXITREQUESTED);
- blacklistedRemoteEvents_.Insert(E_SERVERCONNECTED);
- blacklistedRemoteEvents_.Insert(E_SERVERDISCONNECTED);
- blacklistedRemoteEvents_.Insert(E_CONNECTFAILED);
- blacklistedRemoteEvents_.Insert(E_CLIENTCONNECTED);
- blacklistedRemoteEvents_.Insert(E_CLIENTDISCONNECTED);
- blacklistedRemoteEvents_.Insert(E_CLIENTIDENTITY);
- blacklistedRemoteEvents_.Insert(E_CLIENTSCENELOADED);
- blacklistedRemoteEvents_.Insert(E_NETWORKMESSAGE);
- blacklistedRemoteEvents_.Insert(E_NETWORKUPDATE);
- blacklistedRemoteEvents_.Insert(E_NETWORKUPDATESENT);
- blacklistedRemoteEvents_.Insert(E_NETWORKSCENELOADFAILED);
- blacklistedRemoteEvents_.Insert(E_NATMASTERCONNECTIONSUCCEEDED);
- blacklistedRemoteEvents_.Insert(E_NATMASTERCONNECTIONFAILED);
- blacklistedRemoteEvents_.Insert(E_NETWORKNATPUNCHTROUGHSUCCEEDED);
- blacklistedRemoteEvents_.Insert(E_NETWORKNATPUNCHTROUGHFAILED);
- blacklistedRemoteEvents_.Insert(E_P2PJOINREQUESTDENIED);
- blacklistedRemoteEvents_.Insert(E_P2PALLREADYCHANGED);
- blacklistedRemoteEvents_.Insert(E_NETWORKHOSTDISCOVERED);
- blacklistedRemoteEvents_.Insert(E_NETWORKINVALIDPASSWORD);
- blacklistedRemoteEvents_.Insert(E_NETWORKBANNED);
- blacklistedRemoteEvents_.Insert(E_P2PNEWHOST);
- blacklistedRemoteEvents_.Insert(E_P2PSESSIONSTARTED);
- }
- Network::~Network()
- {
- // If server connection exists, disconnect, but do not send an event because we are shutting down
- Disconnect(1000);
- fullyConnectedMesh2_->ResetHostCalculation();
- rakPeer_->DetachPlugin(natPunchthroughServerClient_);
- rakPeerClient_->DetachPlugin(natPunchthroughClient_);
- rakPeer_->DetachPlugin(fullyConnectedMesh2_);
- rakPeer_->DetachPlugin(connectionGraph2_);
- rakPeer_->DetachPlugin(readyEvent_);
- serverConnection_.Reset();
- clientConnections_.Clear();
- delete natPunchthroughServerClient_;
- natPunchthroughServerClient_ = nullptr;
- delete natPunchthroughClient_;
- natPunchthroughClient_ = nullptr;
- delete remoteGUID_;
- remoteGUID_ = nullptr;
- delete natPunchServerAddress_;
- natPunchServerAddress_ = nullptr;
- SLNet::RakPeerInterface::DestroyInstance(rakPeer_);
- SLNet::RakPeerInterface::DestroyInstance(rakPeerClient_);
- SLNet::FullyConnectedMesh2::DestroyInstance(fullyConnectedMesh2_);
- SLNet::ReadyEvent::DestroyInstance(readyEvent_);
- SLNet::ConnectionGraph2::DestroyInstance(connectionGraph2_);
- rakPeer_ = nullptr;
- rakPeerClient_ = nullptr;
- }
- void Network::HandleMessage(const SLNet::AddressOrGUID& source, int packetID, int msgID, const char* data, size_t numBytes)
- {
- // Only process messages from known sources
- Connection* connection = GetConnection(source);
- if (connection)
- {
- MemoryBuffer msg(data, (unsigned)numBytes);
- if (connection->ProcessMessage((int)msgID, msg))
- return;
- // If message was not handled internally, forward as an event
- using namespace NetworkMessage;
- VariantMap& eventData = GetEventDataMap();
- eventData[P_CONNECTION] = connection;
- eventData[P_MESSAGEID] = (int)msgID;
- eventData[P_DATA].SetBuffer(msg.GetData(), msg.GetSize());
- connection->SendEvent(E_NETWORKMESSAGE, eventData);
- }
- else {
- URHO3D_LOGWARNING("Discarding message from unknown MessageConnection " + String(source.ToString()) + " => " + source.rakNetGuid.ToString());
- }
- }
- void Network::NewConnectionEstablished(const SLNet::AddressOrGUID& connection)
- {
- P2PReadyStatusChanged();
- if (networkMode_ == PEER_TO_PEER && clientConnections_[connection]) {
- // URHO3D_LOGWARNINGF("Client already in the client list.", connection.rakNetGuid.ToString());
- //TODO proper scene state management
- clientConnections_[connection]->SetSceneLoaded(true);
- return;
- }
- // URHO3D_LOGINFOF("NewConnectionEstablished ---------------------------", connection.rakNetGuid.ToString());
- // Create a new client connection corresponding to this MessageConnection
- SharedPtr<Connection> newConnection(new Connection(context_, true, connection, rakPeer_));
- newConnection->ConfigureNetworkSimulator(simulatedLatency_, simulatedPacketLoss_);
- if (networkMode_ == PEER_TO_PEER && serverConnection_) {
- newConnection->SetScene(serverConnection_->GetScene());
- newConnection->SetSceneLoaded(true);
- }
- clientConnections_[connection] = newConnection;
- URHO3D_LOGINFO("Client " + newConnection->ToString() + " connected");
- using namespace ClientConnected;
- VariantMap& eventData = GetEventDataMap();
- eventData[P_CONNECTION] = newConnection;
- newConnection->SendEvent(E_CLIENTCONNECTED, eventData);
- }
- void Network::ClientDisconnected(const SLNet::AddressOrGUID& connection)
- {
- // Remove the client connection that corresponds to this MessageConnection
- HashMap<SLNet::AddressOrGUID, SharedPtr<Connection> >::Iterator i = clientConnections_.Find(connection);
- if (i != clientConnections_.End())
- {
- Connection* connection = i->second_;
- URHO3D_LOGINFO("Client " + connection->ToString() + " disconnected");
- using namespace ClientDisconnected;
- VariantMap& eventData = GetEventDataMap();
- eventData[P_CONNECTION] = connection;
- connection->SendEvent(E_CLIENTDISCONNECTED, eventData);
- clientConnections_.Erase(i);
- }
- P2PReadyStatusChanged();
- }
- void Network::SetDiscoveryBeacon(const VariantMap& data)
- {
- VectorBuffer buffer;
- buffer.WriteVariantMap(data);
- if (buffer.GetSize() > 400)
- URHO3D_LOGERROR("Discovery beacon of size: " + String(buffer.GetSize()) + " bytes is too large, modify MAX_OFFLINE_DATA_LENGTH in RakNet or reduce size");
- rakPeer_->SetOfflinePingResponse((const char*)buffer.GetData(), buffer.GetSize());
- }
- void Network::DiscoverHosts(unsigned port)
- {
- // JSandusky: Contrary to the manual, we actually do have to perform Startup first before we can Ping
- if (!rakPeerClient_->IsActive())
- {
- SLNet::SocketDescriptor socket;
- // Startup local connection with max 1 incoming connection(first param) and 1 socket description (third param)
- rakPeerClient_->Startup(32, &socket, 1);
- rakPeerClient_->SetMaximumIncomingConnections(32);
- }
- rakPeerClient_->Ping("255.255.255.255", port, false);
- }
- void Network::SetPassword(const String& password)
- {
- rakPeer_->SetIncomingPassword(password.CString(), password.Length());
- password_ = password;
- }
- bool Network::Connect(const String& address, unsigned short port, Scene* scene, const VariantMap& identity)
- {
- URHO3D_PROFILE(Connect);
- if (!rakPeerClient_->IsActive())
- {
- URHO3D_LOGINFO("Initializing client connection...");
- SLNet::SocketDescriptor socket;
- // Startup local connection with max 2 incoming connections(first param) and 1 socket description (third param)
- rakPeerClient_->Startup(32, &socket, 1);
- rakPeerClient_->SetMaximumIncomingConnections(32);
- } else {
- OnServerDisconnected();
- }
- //isServer_ = false;
- SLNet::ConnectionAttemptResult connectResult = rakPeerClient_->Connect(address.CString(), port, password_.CString(), password_.Length());
- if (connectResult != SLNet::CONNECTION_ATTEMPT_STARTED)
- {
- URHO3D_LOGERROR("Failed to connect to server " + address + ":" + String(port) + ", error code: " + String((int)connectResult));
- SendEvent(E_CONNECTFAILED);
- return false;
- }
- else
- {
- serverConnection_ = new Connection(context_, false, rakPeerClient_->GetMyGUID(), rakPeerClient_);
- serverConnection_->SetScene(scene);
- serverConnection_->SetIdentity(identity);
- serverConnection_->SetConnectPending(true);
- serverConnection_->ConfigureNetworkSimulator(simulatedLatency_, simulatedPacketLoss_);
- URHO3D_LOGINFO("Connecting to server " + address + ":" + String(port) + ", Client: " + serverConnection_->ToString());
- return true;
- }
- }
- bool Network::P2PConnectNAT(const String& address, unsigned short port)
- {
- URHO3D_PROFILE(P2PConnectNAT);
- if (!rakPeer_->IsActive())
- {
- URHO3D_LOGINFO("Initializing client connection...");
- SLNet::SocketDescriptor socket;
- // Startup local connection with max 2 incoming connections(first param) and 1 socket description (third param)
- rakPeer_->Startup(128, &socket, 1);
- rakPeer_->SetMaximumIncomingConnections(128);
- rakPeer_->AttachPlugin(natPunchthroughServerClient_);
- }
- else {
- OnServerDisconnected();
- }
- //isServer_ = false;
- SLNet::ConnectionAttemptResult connectResult = rakPeer_->Connect(address.CString(), port, nullptr, 0);
- if (connectResult == SLNet::ALREADY_CONNECTED_TO_ENDPOINT) {
- URHO3D_LOGWARNING("Already connected to server " + address + ":" + String(port) + ", error code: " + String((int)connectResult));
- return false;
- }
- if (connectResult != SLNet::CONNECTION_ATTEMPT_STARTED)
- {
- URHO3D_LOGERROR("Failed to connect to server " + address + ":" + String(port) + ", error code: " + String((int)connectResult));
- SendEvent(E_CONNECTFAILED);
- return false;
- }
- else
- {
- URHO3D_LOGINFO("Connecting to server " + address + ":" + String(port));
- return true;
- }
- }
- void Network::Disconnect(int waitMSec)
- {
- if (!serverConnection_)
- return;
- if (networkMode_ == PEER_TO_PEER) {
- rakPeer_->Shutdown(1000);
- clientConnections_.Clear();
- }
- URHO3D_PROFILE(Disconnect);
- serverConnection_->Disconnect(waitMSec);
- serverConnection_.Reset();
- }
- bool Network::StartServer(unsigned short port)
- {
- if (IsServerRunning())
- return true;
- URHO3D_PROFILE(StartServer);
- SLNet::SocketDescriptor socket;//(port, AF_INET);
- socket.port = port;
- socket.socketFamily = AF_INET;
- // Startup local connection with max 128 incoming connection(first param) and 1 socket description (third param)
- SLNet::StartupResult startResult = rakPeer_->Startup(128, &socket, 1);
- if (startResult == SLNet::RAKNET_STARTED)
- {
- URHO3D_LOGINFO("Started server on port " + String(port));
- rakPeer_->SetMaximumIncomingConnections(128);
- isServer_ = true;
- rakPeer_->SetOccasionalPing(true);
- rakPeer_->SetUnreliableTimeout(1000);
- //rakPeer_->SetIncomingPassword("Parole", (int)strlen("Parole"));
- return true;
- }
- else
- {
- URHO3D_LOGINFO("Failed to start server on port " + String(port) + ", error code: " + String((int)startResult));
- return false;
- }
- }
- void Network::StopServer()
- {
- clientConnections_.Clear();
- if (!rakPeer_)
- return;
- if (!IsServerRunning())
- return;
- // Provide 300 ms to notify
- rakPeer_->Shutdown(300);
- URHO3D_PROFILE(StopServer);
- URHO3D_LOGINFO("Stopped server");
- }
- void Network::SetNATServerInfo(const String& address, unsigned short port)
- {
- if (!natPunchServerAddress_)
- natPunchServerAddress_ = new SLNet::SystemAddress;
- natPunchServerAddress_->FromStringExplicitPort(address.CString(), port);
- }
- void Network::StartNATClient()
- {
- if (!rakPeer_) {
- URHO3D_LOGERROR("Unable to start NAT client, client not initialized!");
- return;
- }
- if (natPunchServerAddress_->GetPort() == 0) {
- URHO3D_LOGERROR("NAT master server address incorrect!");
- return;
- }
- rakPeer_->AttachPlugin(natPunchthroughServerClient_);
- guid_ = String(rakPeer_->GetGuidFromSystemAddress(SLNet::UNASSIGNED_SYSTEM_ADDRESS).ToString());
- URHO3D_LOGINFO("GUID: " + guid_);
- rakPeer_->Connect(natPunchServerAddress_->ToString(false), natPunchServerAddress_->GetPort(), nullptr, 0);
- }
- void Network::AttemptNATPunchtrough(const String& guid, Scene* scene, const VariantMap& identity)
- {
- scene_ = scene;
- identity_ = identity;
- if (!remoteGUID_)
- remoteGUID_ = new SLNet::RakNetGUID;
- remoteGUID_->FromString(guid.CString());
- rakPeerClient_->AttachPlugin(natPunchthroughClient_);
- if (rakPeerClient_->IsActive()) {
- natPunchthroughClient_->OpenNAT(*remoteGUID_, *natPunchServerAddress_);
- }
- else {
- SLNet::SocketDescriptor socket;
- // Startup local connection with max 2 incoming connections(first param) and 1 socket description (third param)
- rakPeerClient_->Startup(32, &socket, 1);
- rakPeerClient_->SetMaximumIncomingConnections(32);
- }
- rakPeerClient_->Connect(natPunchServerAddress_->ToString(false), natPunchServerAddress_->GetPort(), nullptr, 0);
- }
- void Network::BroadcastMessage(int msgID, bool reliable, bool inOrder, const VectorBuffer& msg, unsigned contentID)
- {
- BroadcastMessage(msgID, reliable, inOrder, msg.GetData(), msg.GetSize(), contentID);
- }
- void Network::BroadcastMessage(int msgID, bool reliable, bool inOrder, const unsigned char* data, unsigned numBytes,
- unsigned contentID)
- {
- if (!rakPeer_)
- return;
- /* Make sure not to use SLikeNet(RakNet) internal message ID's
- and since RakNet uses 1 byte message ID's, they cannot exceed 255 limit */
- if (msgID < ID_USER_PACKET_ENUM || msgID >= 255)
- {
- URHO3D_LOGERROR("Can not send message with reserved ID");
- return;
- }
- VectorBuffer msgData;
- msgData.WriteUByte((unsigned char)msgID);
- msgData.Write(data, numBytes);
- if (isServer_) {
- rakPeer_->Send((const char *) msgData.GetData(), (int) msgData.GetSize(), HIGH_PRIORITY, RELIABLE, (char) 0, SLNet::UNASSIGNED_RAKNET_GUID, true);
- }
- else
- URHO3D_LOGERROR("Server not running, can not broadcast messages");
- }
- void Network::BroadcastRemoteEvent(StringHash eventType, bool inOrder, const VariantMap& eventData)
- {
- for (HashMap<SLNet::AddressOrGUID, SharedPtr<Connection> >::Iterator i = clientConnections_.Begin(); i != clientConnections_.End(); ++i)
- i->second_->SendRemoteEvent(eventType, inOrder, eventData);
- }
- void Network::BroadcastRemoteEvent(Scene* scene, StringHash eventType, bool inOrder, const VariantMap& eventData)
- {
- for (HashMap<SLNet::AddressOrGUID, SharedPtr<Connection> >::Iterator i = clientConnections_.Begin();
- i != clientConnections_.End(); ++i)
- {
- if (i->second_->GetScene() == scene)
- i->second_->SendRemoteEvent(eventType, inOrder, eventData);
- }
- }
- void Network::BroadcastRemoteEvent(Node* node, StringHash eventType, bool inOrder, const VariantMap& eventData)
- {
- if (!node)
- {
- URHO3D_LOGERROR("Null sender node for remote node event");
- return;
- }
- if (!node->IsReplicated())
- {
- URHO3D_LOGERROR("Sender node has a local ID, can not send remote node event");
- return;
- }
- Scene* scene = node->GetScene();
- for (HashMap<SLNet::AddressOrGUID, SharedPtr<Connection> >::Iterator i = clientConnections_.Begin();
- i != clientConnections_.End(); ++i)
- {
- if (i->second_->GetScene() == scene)
- i->second_->SendRemoteEvent(node, eventType, inOrder, eventData);
- }
- }
- void Network::SetUpdateFps(int fps)
- {
- updateFps_ = Max(fps, 1);
- updateInterval_ = 1.0f / (float)updateFps_;
- updateAcc_ = 0.0f;
- }
- void Network::SetSimulatedLatency(int ms)
- {
- simulatedLatency_ = Max(ms, 0);
- ConfigureNetworkSimulator();
- }
- void Network::SetSimulatedPacketLoss(float probability)
- {
- simulatedPacketLoss_ = Clamp(probability, 0.0f, 1.0f);
- ConfigureNetworkSimulator();
- }
- void Network::RegisterRemoteEvent(StringHash eventType)
- {
- if (blacklistedRemoteEvents_.Find(eventType) != blacklistedRemoteEvents_.End())
- {
- URHO3D_LOGERROR("Attempted to register blacklisted remote event type " + String(eventType));
- return;
- }
- allowedRemoteEvents_.Insert(eventType);
- }
- void Network::UnregisterRemoteEvent(StringHash eventType)
- {
- allowedRemoteEvents_.Erase(eventType);
- }
- void Network::UnregisterAllRemoteEvents()
- {
- allowedRemoteEvents_.Clear();
- }
- void Network::SetPackageCacheDir(const String& path)
- {
- packageCacheDir_ = AddTrailingSlash(path);
- }
- void Network::SendPackageToClients(Scene* scene, PackageFile* package)
- {
- if (!scene)
- {
- URHO3D_LOGERROR("Null scene specified for SendPackageToClients");
- return;
- }
- if (!package)
- {
- URHO3D_LOGERROR("Null package specified for SendPackageToClients");
- return;
- }
- for (HashMap<SLNet::AddressOrGUID, SharedPtr<Connection> >::Iterator i = clientConnections_.Begin();
- i != clientConnections_.End(); ++i)
- {
- if (i->second_->GetScene() == scene)
- i->second_->SendPackageToClient(package);
- }
- }
- SharedPtr<HttpRequest> Network::MakeHttpRequest(const String& url, const String& verb, const Vector<String>& headers,
- const String& postData)
- {
- URHO3D_PROFILE(MakeHttpRequest);
- // The initialization of the request will take time, can not know at this point if it has an error or not
- SharedPtr<HttpRequest> request(new HttpRequest(url, verb, headers, postData));
- return request;
- }
- void Network::BanAddress(const String& address)
- {
- rakPeer_->AddToBanList(address.CString(), 0);
- }
- Connection* Network::GetConnection(const SLNet::AddressOrGUID& connection) const
- {
- if (serverConnection_ && serverConnection_->GetAddressOrGUID() == connection) {
- return serverConnection_;
- }
- else
- {
- HashMap<SLNet::AddressOrGUID, SharedPtr<Connection> >::ConstIterator i = clientConnections_.Find(connection);
- if (i != clientConnections_.End())
- return i->second_;
- else
- return nullptr;
- }
- }
- Connection* Network::GetServerConnection() const
- {
- return serverConnection_;
- }
- Vector<SharedPtr<Connection> > Network::GetClientConnections() const
- {
- Vector<SharedPtr<Connection> > ret;
- for (HashMap<SLNet::AddressOrGUID, SharedPtr<Connection> >::ConstIterator i = clientConnections_.Begin();
- i != clientConnections_.End(); ++i)
- ret.Push(i->second_);
- return ret;
- }
- bool Network::IsServerRunning() const
- {
- if (!rakPeer_)
- return false;
- if (networkMode_ == PEER_TO_PEER && !isServer_) {
- return false;
- }
- return rakPeer_->IsActive() && isServer_;
- }
- bool Network::CheckRemoteEvent(StringHash eventType) const
- {
- return allowedRemoteEvents_.Contains(eventType);
- }
- void Network::HandleIncomingPacket(SLNet::Packet* packet, bool isServer)
- {
- unsigned char packetID = packet->data[0];
- // if (packetID < ID_USER_PACKET_ENUM) {
- // URHO3D_LOGERROR(">> " + String((int)packetID) + " HandleIncomingPacket: " + String(RAKNET_MESSAGEID_STRINGS[packetID]));
- // }
- bool packetHandled = false;
- // Deal with timestamped backents
- unsigned dataStart = sizeof(char);
- if (packetID == ID_TIMESTAMP)
- {
- dataStart += sizeof(SLNet::Time);
- packetID = packet->data[dataStart];
- dataStart += sizeof(char);
- }
- if (packetID == ID_NEW_INCOMING_CONNECTION)
- {
- URHO3D_LOGINFOF("ID_NEW_INCOMING_CONNECTION from %s. guid=%s.", packet->systemAddress.ToString(true), packet->guid.ToString());
- if (isServer)
- {
- NewConnectionEstablished(packet->guid);
- }
- packetHandled = true;
- }
- if (packetID == ID_REMOTE_NEW_INCOMING_CONNECTION)
- {
- URHO3D_LOGINFOF("ID_REMOTE_NEW_INCOMING_CONNECTION from %s. guid=%s.", packet->systemAddress.ToString(true), packet->guid.ToString());
- if (networkMode_ == PEER_TO_PEER)
- {
- unsigned int count;
- SLNet::BitStream bsIn(packet->data, packet->length, false);
- bsIn.IgnoreBytes(sizeof(SLNet::MessageID));
- bsIn.Read(count);
- SLNet::SystemAddress remoteAddress;
- SLNet::RakNetGUID remoteGuid;
- NewConnectionEstablished(packet->guid);
- for (unsigned int i=0; i < count; i++)
- {
- bsIn.Read(remoteAddress);
- bsIn.Read(remoteGuid);
- URHO3D_LOGINFO("Remote connection " + String(remoteGuid.ToString()) + " / " + String(packet->guid));
- NewConnectionEstablished(packet->guid);
- }
- } else {
- NewConnectionEstablished(packet->systemAddress);
- }
- packetHandled = true;
- }
- else if (packetID == ID_REMOTE_CONNECTION_LOST || packetID == ID_REMOTE_DISCONNECTION_NOTIFICATION)
- {
- //TODO find out who's really sending out this message
- packetHandled = true;
- }
- else if (packetID == ID_ALREADY_CONNECTED)
- {
- if (natPunchServerAddress_ && packet->systemAddress == *natPunchServerAddress_) {
- URHO3D_LOGINFO("Already connected to NAT server! ");
- if (!isServer && networkMode_ == SERVER_CLIENT)
- {
- natPunchthroughClient_->OpenNAT(*remoteGUID_, *natPunchServerAddress_);
- }
- }
- packetHandled = true;
- }
- else if (packetID == ID_CONNECTION_REQUEST_ACCEPTED) // We're a client, our connection has been accepted
- {
- if(natPunchServerAddress_ && packet->systemAddress == *natPunchServerAddress_) {
- URHO3D_LOGINFO("Succesfully connected to NAT punchtrough server! ");
- SendEvent(E_NATMASTERCONNECTIONSUCCEEDED);
- if (!isServer && remoteGUID_&& networkMode_ == SERVER_CLIENT)
- {
- natPunchthroughClient_->OpenNAT(*remoteGUID_, *natPunchServerAddress_);
- }
- } else {
- if (networkMode_ == SERVER_CLIENT) {
- OnServerConnected(packet->guid);
- } else {
- URHO3D_LOGERROR(">>>>>>>>>>>>>>>>>>>.");
- URHO3D_LOGINFOF("ID_CONNECTION_REQUEST_ACCEPTED from %s,guid=%s", packet->systemAddress.ToString(true), packet->guid.ToString());
- // Assume that we're connecting to the P2P host
- // serverConnection_->SetAddressOrGUID(packet->guid);
- SLNet::BitStream bsOut;
- bsOut.Write((unsigned char) MSG_P2P_JOIN_REQUEST);
- rakPeer_->Send(&bsOut, HIGH_PRIORITY, RELIABLE_ORDERED, 0, packet->guid, false);
- }
- }
- packetHandled = true;
- }
- else if (packetID == ID_NAT_TARGET_NOT_CONNECTED)
- {
- URHO3D_LOGERROR("Target server not connected to NAT master server!");
- packetHandled = true;
- }
- else if (packetID == ID_CONNECTION_LOST || packetID == ID_DISCONNECTION_NOTIFICATION) // We've lost connectivity with the packet source
- {
- if(natPunchServerAddress_ && packet->systemAddress == *natPunchServerAddress_) {
- URHO3D_LOGERROR("Connection to NAT server lost!");
- P2PConnectNAT(natPunchServerAddress_->ToString(false), natPunchServerAddress_->GetPort());
- } else if (isServer)
- {
- ClientDisconnected(packet->guid);
- }
- else
- {
- if (networkMode_ == SERVER_CLIENT) {
- OnServerDisconnected();
- } else {
- ClientDisconnected(packet->guid);
- }
- }
- packetHandled = true;
- }
- else if (packetID == ID_CONNECTION_ATTEMPT_FAILED) // We've failed to connect to the server/peer
- {
- if (natPunchServerAddress_ && packet->systemAddress == *natPunchServerAddress_) {
- URHO3D_LOGERROR("Connection to NAT punchtrough server failed!");
- SendEvent(E_NATMASTERCONNECTIONFAILED);
- } else {
- if (!isServer)
- {
- OnServerDisconnected();
- }
- }
- packetHandled = true;
- }
- else if (packetID == ID_NAT_PUNCHTHROUGH_SUCCEEDED)
- {
- SLNet::SystemAddress remotePeer = packet->systemAddress;
- URHO3D_LOGINFO("NAT punchtrough succeeded! Remote peer: " + String(remotePeer.ToString()));
- if (!isServer && networkMode_ == SERVER_CLIENT)
- {
- using namespace NetworkNatPunchtroughSucceeded;
- VariantMap eventMap;
- eventMap[P_ADDRESS] = remotePeer.ToString(false);
- eventMap[P_PORT] = remotePeer.GetPort();
- SendEvent(E_NETWORKNATPUNCHTROUGHSUCCEEDED, eventMap);
- URHO3D_LOGINFO("Connecting to server behind NAT: " + String(remotePeer.ToString()));
- Connect(String(remotePeer.ToString(false)), remotePeer.GetPort(), scene_, identity_);
- } else if (networkMode_ == PEER_TO_PEER){
- SLNet::ConnectionAttemptResult car = rakPeer_->Connect(packet->systemAddress.ToString(false), packet->systemAddress.GetPort(), password_.CString(), password_.Length());
- }
- packetHandled = true;
- }
- else if (packetID == ID_NAT_PUNCHTHROUGH_FAILED || packetID == ID_NAT_TARGET_NOT_CONNECTED || packetID == ID_NAT_TARGET_UNRESPONSIVE || packetID == ID_NAT_CONNECTION_TO_TARGET_LOST)
- {
- URHO3D_LOGERROR("NAT punchtrough failed!");
- SLNet::SystemAddress remotePeer = packet->systemAddress;
- using namespace NetworkNatPunchtroughFailed;
- VariantMap eventMap;
- eventMap[P_ADDRESS] = remotePeer.ToString(false);
- eventMap[P_PORT] = remotePeer.GetPort();
- SendEvent(E_NETWORKNATPUNCHTROUGHFAILED, eventMap);
- packetHandled = true;
- }
- else if (packetID == ID_CONNECTION_BANNED) // We're a client and we're on the ban list
- {
- URHO3D_LOGERROR("Connection failed, you're banned!");
- SendEvent(E_NETWORKBANNED);
- packetHandled = true;
- }
- else if (packetID == ID_INVALID_PASSWORD) // We're a client, and we gave an invalid password
- {
- URHO3D_LOGERROR("Invalid password provided for connection!");
- SendEvent(E_NETWORKINVALIDPASSWORD);
- packetHandled = true;
- }
- else if (packetID == ID_DOWNLOAD_PROGRESS) // Part of a file transfer
- {
- //URHO3D_LOGINFO("101010");
- }
- else if (packetID == ID_UNCONNECTED_PING)
- {
- packetHandled = true;
- }
- else if (packetID == ID_READY_EVENT_SET)
- {
- URHO3D_LOGWARNINGF("Got ID_READY_EVENT_SET from %s", packet->guid.ToString());
- P2PReadyStatusChanged();
- }
- else if (packetID == ID_READY_EVENT_UNSET)
- {
- P2PReadyStatusChanged();
- URHO3D_LOGWARNINGF("Got ID_READY_EVENT_UNSET from %s", packet->guid.ToString());
- }
- else if (packetID == ID_READY_EVENT_ALL_SET)
- {
- P2PReadyStatusChanged();
- URHO3D_LOGWARNINGF("ID_READY_EVENT_ALL_SET from %s", packet->guid.ToString());
- }
- else if (packetID == ID_READY_EVENT_QUERY)
- {
- P2PReadyStatusChanged();
- URHO3D_LOGINFOF("`````````````````````````` Got ID_READY_EVENT_QUERY from %s", packet->guid.ToString());
- }
- else if (packetID == ID_READY_EVENT_FORCE_ALL_SET)
- {
- P2PReadyStatusChanged();
- URHO3D_LOGINFOF("`````````````````````````` Got ID_READY_EVENT_FORCE_ALL_SET from %s", packet->guid.ToString());
- }
- else if (packetID == ID_UNCONNECTED_PONG) // Host discovery response
- {
- if (!isServer)
- {
- using namespace NetworkHostDiscovered;
- dataStart += sizeof(SLNet::TimeMS);
- VariantMap& eventMap = context_->GetEventDataMap();
- if (packet->length > packet->length - dataStart) {
- VectorBuffer buffer(packet->data + dataStart, packet->length - dataStart);
- VariantMap srcData = buffer.ReadVariantMap();
- eventMap[P_BEACON] = srcData;
- }
- else {
- eventMap[P_BEACON] = VariantMap();
- }
- eventMap[P_ADDRESS] = String(packet->systemAddress.ToString(false));
- eventMap[P_PORT] = (int)packet->systemAddress.GetPort();
- SendEvent(E_NETWORKHOSTDISCOVERED, eventMap);
- }
- packetHandled = true;
- }
- else if (packetID == ID_FCM2_NEW_HOST)
- {
- URHO3D_LOGINFO("");
- // URHO3D_LOGINFOF("ID_FCM2_NEW_HOST: System %s is host, GUID=%s", packet->systemAddress.ToString(true), packet->guid.ToString());
- SLNet::BitStream bs(packet->data,packet->length,false);
- bs.IgnoreBytes(1);
- SLNet::RakNetGUID oldHost;
- bs.Read(oldHost);
- if (serverConnection_) {
- serverConnection_->SetAddressOrGUID(packet->guid);
- }
- hostGuid_ = packet->guid.ToString();
- if (packet->guid == rakPeer_->GetMyGUID())
- {
- isServer_ = true;
- if (oldHost != SLNet::UNASSIGNED_RAKNET_GUID)
- {
- URHO3D_LOGINFOF("ID_FCM2_NEW_HOST: Taking over as host from the old host [%s].", oldHost.ToString());
- }
- else
- {
- // Room not hosted if we become host the first time since this was done in CreateRoom() already
- URHO3D_LOGINFO("ID_FCM2_NEW_HOST: We have become host for the first time");
- }
- for (auto it = clientConnections_.Begin(); it != clientConnections_.End(); ++it) {
- URHO3D_LOGINFO("Setting new scene for clients");
- // (*it).second_->SetScene(serverConnection_->GetScene());
- //TODO decide what to do when we take ownership as the host, should the scene needs to be reloaded?
- (*it).second_->SetSceneLoaded(true);
- }
- }
- else
- {
- isServer_ = false;
- if (oldHost != SLNet::UNASSIGNED_RAKNET_GUID) {
- URHO3D_LOGINFOF("ID_FCM2_NEW_HOST: A new system %s has become host, GUID=%s", packet->systemAddress.ToString(true), packet->guid.ToString());
- }
- else {
- URHO3D_LOGINFOF("ID_FCM2_NEW_HOST: System %s is host, GUID=%s", packet->systemAddress.ToString(true), packet->guid.ToString());
- }
- // Send the identity map now
- VectorBuffer msg;
- msg.WriteVariantMap(serverConnection_->GetIdentity());
- serverConnection_->SendMessage(MSG_IDENTITY, true, true, msg);
- }
- using namespace P2PNewHost;
- VariantMap data = GetEventDataMap();
- data[P_ADDRESS] = packet->systemAddress.ToString(false);
- data[P_PORT] = packet->systemAddress.GetPort();
- data[P_GUID] = packet->guid.ToString();
- SendEvent(E_P2PNEWHOST, data);
- P2PReadyStatusChanged();
- URHO3D_LOGINFO("");
- packetHandled = true;
- }
- else if (packetID == ID_FCM2_VERIFIED_JOIN_START) {
- URHO3D_LOGINFO("ID_FCM2_VERIFIED_JOIN_START");
- // DataStructures::List <SLNet::SystemAddress> addresses;
- // DataStructures::List <SLNet::RakNetGUID> guids;
- // DataStructures::List < SLNet::BitStream * > userData;
- // fullyConnectedMesh2_->GetVerifiedJoinRequiredProcessingList(packet->guid, addresses, guids, userData);
- // for (unsigned int i = 0; i < guids.Size(); i++)
- // {
- // natPunchthroughServerClient_->OpenNAT(guids[i], *natPunchServerAddress_);
- // }
- }
- else if (packetID == ID_FCM2_VERIFIED_JOIN_CAPABLE)
- {
- URHO3D_LOGINFO("ID_FCM2_VERIFIED_JOIN_CAPABLE");
- fullyConnectedMesh2_->RespondOnVerifiedJoinCapable(packet, true, nullptr);
- packetHandled = true;
- }
- else if (packetID == ID_FCM2_VERIFIED_JOIN_ACCEPTED)
- {
- DataStructures::List<SLNet::RakNetGUID> systemsAccepted;
- bool thisSystemAccepted;
- fullyConnectedMesh2_->GetVerifiedJoinAcceptedAdditionalData(packet, &thisSystemAccepted, systemsAccepted, nullptr);
- if (thisSystemAccepted) {
- URHO3D_LOGINFO("Game join request accepted");
- }
- URHO3D_LOGINFO("ID_FCM2_VERIFIED_JOIN_ACCEPTED");
- packetHandled = true;
- }
- else if (packetID == ID_FCM2_VERIFIED_JOIN_REJECTED)
- {
- URHO3D_LOGINFO("ID_FCM2_VERIFIED_JOIN_REJECTED");
- }
- else if (packetID == ID_FCM2_REQUEST_FCMGUID)
- {
- URHO3D_LOGINFO("ID_FCM2_REQUEST_FCMGUID");
- }
- else if (packetID == ID_FCM2_RESPOND_CONNECTION_COUNT)
- {
- URHO3D_LOGINFO("ID_FCM2_RESPOND_CONNECTION_COUNT");
- }
- else if (packetID == ID_FCM2_INFORM_FCMGUID)
- {
- URHO3D_LOGINFO("ID_FCM2_INFORM_FCMGUID");
- }
- else if (packetID == ID_FCM2_UPDATE_MIN_TOTAL_CONNECTION_COUNT)
- {
- URHO3D_LOGINFO("ID_FCM2_UPDATE_MIN_TOTAL_CONNECTION_COUNT");
- }
- else if (packetID == ID_FCM2_UPDATE_USER_CONTEXT)
- {
- URHO3D_LOGINFO("ID_FCM2_UPDATE_USER_CONTEXT");
- }
- // Urho3D messages
- if (packetID >= ID_USER_PACKET_ENUM)
- {
- //URHO3D_LOGINFOF("ID_USER_PACKET_ENUM %i", packetID);
- if (packetID == MSG_P2P_JOIN_REQUEST) {
- URHO3D_LOGINFO("MSG_P2P_JOIN_REQUEST");
- //TODO decide if the client is able to join
- fullyConnectedMesh2_->StartVerifiedJoin(packet->guid);
- //TODO or deny the join request
- // VectorBuffer msg;
- // GetConnection(packet->guid)->SendMessage(MSG_P2P_JOIN_REQUEST_DENIED, true, true, msg);
- // ClientDisconnected(packet->guid);
- } else if (networkMode_ == PEER_TO_PEER && P2PIsHostSystem())
- {
- // We are the host in the P2P server, parse the message accordingly
- HandleMessage(packet->guid, 0, packetID, (const char*)(packet->data + dataStart), packet->length - dataStart);
- }
- else if (networkMode_ == SERVER_CLIENT && isServer_) {
- // We are the server in the server-client connection
- HandleMessage(packet->guid, 0, packetID, (const char*)(packet->data + dataStart), packet->length - dataStart);
- }
- else {
- // we are client in either P2P or server-client mode
- MemoryBuffer buffer(packet->data + dataStart, packet->length - dataStart);
- bool processed = serverConnection_->ProcessMessage(packetID, buffer);
- if (!processed)
- {
- HandleMessage(packet->guid, 0, packetID, (const char*)(packet->data + dataStart), packet->length - dataStart);
- }
- }
- packetHandled = true;
- }
- if (!packetHandled && packetID < sizeof(RAKNET_MESSAGEID_STRINGS))
- URHO3D_LOGERROR("Unhandled network packet: " + String(RAKNET_MESSAGEID_STRINGS[packetID]));
- else if (!packetHandled)
- URHO3D_LOGERRORF("Unhandled network packet: %i", packetID);
- }
- void Network::Update(float timeStep)
- {
- URHO3D_PROFILE(UpdateNetwork);
- //Process all incoming messages for the server
- if (rakPeer_->IsActive())
- {
- while (SLNet::Packet* packet = rakPeer_->Receive())
- {
- if (P2PIsHostSystem() || networkMode_ == SERVER_CLIENT) {
- HandleIncomingPacket(packet, true);
- }
- else {
- HandleIncomingPacket(packet, false);
- }
- rakPeer_->DeallocatePacket(packet);
- }
- }
- // Process all incoming messages for the client
- if (rakPeerClient_->IsActive())
- {
- //bool isHost = P2PIsHostSystem();
- while (SLNet::Packet* packet = rakPeerClient_->Receive())
- {
- HandleIncomingPacket(packet, false);
- rakPeerClient_->DeallocatePacket(packet);
- }
- }
- }
- void Network::PostUpdate(float timeStep)
- {
- URHO3D_PROFILE(PostUpdateNetwork);
- // Check if periodic update should happen now
- updateAcc_ += timeStep;
- bool updateNow = updateAcc_ >= updateInterval_;
- if (updateNow)
- {
- // Notify of the impending update to allow for example updated client controls to be set
- SendEvent(E_NETWORKUPDATE);
- updateAcc_ = fmodf(updateAcc_, updateInterval_);
- if (IsServerRunning())
- {
- // Collect and prepare all networked scenes
- {
- URHO3D_PROFILE(PrepareServerUpdate);
- networkScenes_.Clear();
- for (HashMap<SLNet::AddressOrGUID, SharedPtr<Connection> >::Iterator i = clientConnections_.Begin();
- i != clientConnections_.End(); ++i)
- {
- Scene* scene = i->second_->GetScene();
- if (scene)
- networkScenes_.Insert(scene);
- }
- for (HashSet<Scene*>::ConstIterator i = networkScenes_.Begin(); i != networkScenes_.End(); ++i)
- (*i)->PrepareNetworkUpdate();
- }
- {
- URHO3D_PROFILE(SendServerUpdate);
- // Then send server updates for each client connection
- for (HashMap<SLNet::AddressOrGUID, SharedPtr<Connection> >::Iterator i = clientConnections_.Begin();
- i != clientConnections_.End(); ++i)
- {
- i->second_->SendServerUpdate();
- i->second_->SendRemoteEvents();
- i->second_->SendPackages();
- }
- }
- }
- if (serverConnection_)
- {
- if (networkMode_ == PEER_TO_PEER && !isServer_ && serverConnection_->GetGUID() != P2PGetGUID()) {
- // Send the client update
- serverConnection_->SendClientUpdate();
- serverConnection_->SendRemoteEvents();
- } else if (networkMode_ == SERVER_CLIENT) {
- // Send the client update
- serverConnection_->SendClientUpdate();
- serverConnection_->SendRemoteEvents();
- }
- }
- // Notify that the update was sent
- SendEvent(E_NETWORKUPDATESENT);
- }
- }
- void Network::HandleBeginFrame(StringHash eventType, VariantMap& eventData)
- {
- using namespace BeginFrame;
- Update(eventData[P_TIMESTEP].GetFloat());
- }
- void Network::HandleRenderUpdate(StringHash eventType, VariantMap& eventData)
- {
- using namespace RenderUpdate;
- PostUpdate(eventData[P_TIMESTEP].GetFloat());
- }
- void Network::OnServerConnected(const SLNet::AddressOrGUID& address)
- {
- serverConnection_->SetConnectPending(false);
- serverConnection_->SetAddressOrGUID(address);
- URHO3D_LOGINFO("Connected to server!");
- // Send the identity map now
- VectorBuffer msg;
- msg.WriteVariantMap(serverConnection_->GetIdentity());
- serverConnection_->SendMessage(MSG_IDENTITY, true, true, msg);
- SendEvent(E_SERVERCONNECTED);
- }
- void Network::OnServerDisconnected()
- {
- if (networkMode_ == PEER_TO_PEER) {
- return;
- }
- // Differentiate between failed connection, and disconnection
- bool failedConnect = serverConnection_ && serverConnection_->IsConnectPending();
- serverConnection_.Reset();
- if (!failedConnect)
- {
- URHO3D_LOGINFO("Disconnected from server");
- SendEvent(E_SERVERDISCONNECTED);
- }
- else
- {
- URHO3D_LOGERROR("Failed to connect to server");
- SendEvent(E_CONNECTFAILED);
- }
- }
- void Network::ConfigureNetworkSimulator()
- {
- if (serverConnection_)
- serverConnection_->ConfigureNetworkSimulator(simulatedLatency_, simulatedPacketLoss_);
- for (HashMap<SLNet::AddressOrGUID, SharedPtr<Connection> >::Iterator i = clientConnections_.Begin();
- i != clientConnections_.End(); ++i)
- i->second_->ConfigureNetworkSimulator(simulatedLatency_, simulatedPacketLoss_);
- }
- bool Network::P2PStartSession(Scene* scene, const VariantMap& identity)
- {
- Disconnect(1000);
- if (!natPunchServerAddress_) {
- URHO3D_LOGERROR("Set the NAT server info first!");
- return false;
- }
- P2PConnectNAT(natPunchServerAddress_->ToString(false), natPunchServerAddress_->GetPort());
- if (networkMode_ == SERVER_CLIENT) {
- URHO3D_LOGERROR("P2P sessions are not available for SERVER_CLIENT mode!");
- return false;
- }
- UnsubscribeFromEvent(E_NATMASTERCONNECTIONSUCCEEDED);
- SubscribeToEvent(E_NATMASTERCONNECTIONSUCCEEDED, URHO3D_HANDLER(Network, HandleNATStartP2PSession));
- scene_ = scene;
- identity_ = identity;
- return true;
- }
- void Network::HandleNATStartP2PSession(StringHash eventType, VariantMap& eventData)
- {
- UnsubscribeFromEvent(E_NATMASTERCONNECTIONSUCCEEDED);
- isServer_ = false;
- if (!serverConnection_) {
- serverConnection_ = new Connection(context_, false, rakPeer_->GetMyGUID(), rakPeer_);
- serverConnection_->SetScene(scene_);
- serverConnection_->SetSceneLoaded(true);
- serverConnection_->SetIdentity(identity_);
- serverConnection_->SetConnectPending(true);
- serverConnection_->ConfigureNetworkSimulator(simulatedLatency_, simulatedPacketLoss_);
- }
- rakPeer_->SetOccasionalPing(true);
- fullyConnectedMesh2_->Clear();
- fullyConnectedMesh2_->ResetHostCalculation();
- hostGuid_ = P2PGetGUID();
- isServer_ = true;
- P2PSetReady(false);
- SendEvent(E_P2PSESSIONSTARTED);
- }
- void Network::P2PJoinSession(const String& guid, Scene* scene, const VariantMap& identity)
- {
- Disconnect(1000);
- if (!natPunchServerAddress_) {
- URHO3D_LOGERROR("Set the NAT server info first!");
- return;
- }
- P2PConnectNAT(natPunchServerAddress_->ToString(false), natPunchServerAddress_->GetPort());
- if (networkMode_ == SERVER_CLIENT) {
- URHO3D_LOGERROR("P2P sessions are not available for SERVER_CLIENT mode!");
- return;
- }
- if (remoteGUID_) {
- delete remoteGUID_;
- remoteGUID_ = nullptr;
- }
- remoteGUID_ = new SLNet::RakNetGUID;
- remoteGUID_->FromString(guid.CString());
- scene_ = scene;
- identity_ = identity;
- UnsubscribeFromEvent(E_NATMASTERCONNECTIONSUCCEEDED);
- SubscribeToEvent(E_NATMASTERCONNECTIONSUCCEEDED, URHO3D_HANDLER(Network, HandleNATJoinP2PSession));
- }
- void Network::HandleNATJoinP2PSession(StringHash eventType, VariantMap& eventData)
- {
- UnsubscribeFromEvent(E_NATMASTERCONNECTIONSUCCEEDED);
- P2PSetReady(false);
- if (!serverConnection_) {
- serverConnection_ = new Connection(context_, false, rakPeer_->GetMyBoundAddress(), rakPeer_);
- serverConnection_->SetScene(scene_);
- serverConnection_->SetSceneLoaded(true);
- serverConnection_->SetIdentity(identity_);
- serverConnection_->SetConnectPending(true);
- serverConnection_->SetAddressOrGUID(*remoteGUID_);
- serverConnection_->ConfigureNetworkSimulator(simulatedLatency_, simulatedPacketLoss_);
- }
- rakPeer_->SetOccasionalPing(true);
- fullyConnectedMesh2_->ResetHostCalculation();
- fullyConnectedMesh2_->Clear();
- URHO3D_LOGINFO("Attempting to Join P2P Session : " + String(remoteGUID_->ToString()));
- natPunchthroughServerClient_->OpenNAT(*remoteGUID_, *natPunchServerAddress_);
- }
- int Network::GetP2PParticipantCount()
- {
- if (networkMode_ == SERVER_CLIENT) {
- return 0;
- }
- return fullyConnectedMesh2_->GetParticipantCount();
- }
- bool Network::P2PIsConnectedHost()
- {
- if (networkMode_ == SERVER_CLIENT) {
- return false;
- }
- return fullyConnectedMesh2_->IsConnectedHost();
- }
- bool Network::P2PIsHostSystem()
- {
- return fullyConnectedMesh2_->IsHostSystem();
- }
- String Network::P2PGetHostAddress()
- {
- if (networkMode_ == SERVER_CLIENT) {
- return "";
- }
- return hostGuid_;
- }
- void Network::P2PSetReady(bool value)
- {
- if (networkMode_ == SERVER_CLIENT) {
- return;
- }
- // readyEvent_->DeleteEvent(0);
- readyEvent_->SetEvent(0, value);
- }
- bool Network::P2PGetReady()
- {
- if (networkMode_ == SERVER_CLIENT) {
- return false;
- }
- return readyEvent_->IsEventSet(0);
- }
- String Network::P2PGetGUID()
- {
- return String(rakPeer_->GetGuidFromSystemAddress(SLNet::UNASSIGNED_SYSTEM_ADDRESS).ToString());
- }
- void Network::P2PReadyStatusChanged()
- {
- if (networkMode_ == SERVER_CLIENT) {
- return;
- }
- DataStructures::List<SLNet::RakNetGUID> participantList;
- fullyConnectedMesh2_->GetParticipantList(participantList);
- for (unsigned int i = 0; i < participantList.Size(); i++) {
- if (participantList[i] != rakPeerClient_->GetMyGUID()) {
- readyEvent_->AddToWaitList(0, participantList[i]);
- }
- }
- static const char* STATUS_MESSAGES[] = {
- "RES_NOT_WAITING",
- "RES_WAITING",
- "RES_READY",
- "RES_ALL_READY",
- "RES_UNKNOWN_EVENT"
- };
- fullyConnectedMesh2_->GetParticipantList(participantList);
- bool allValid = true;
- for (unsigned int i = 0; i < participantList.Size(); i++) {
- if (participantList[i] != rakPeer_->GetMyGUID()) {
- int ready = readyEvent_->GetReadyStatus(0, participantList[i]);// == SLNet::RES_READY;
- if (ready != SLNet::RES_ALL_READY && ready != SLNet::RES_READY) {
- allValid = false;
- }
- // URHO3D_LOGINFO( String(participantList[i].ToString()) + " Ready: " + String(STATUS_MESSAGES[ready]));
- }
- }
- // URHO3D_LOGINFO(P2PGetGUID() + " Ready: " + String(STATUS_MESSAGES[readyEvent_->GetEventAtIndex(0)]));
- VariantMap data = GetEventDataMap();
- if (allValid && readyEvent_->IsEventSet(0)) {
- data[P2PAllReadyChanged::P_READY] = true;
- } else {
- data[P2PAllReadyChanged::P_READY] = false;
- }
- // URHO3D_LOGWARNING("All ready " + String(data[P2PAllReadyChanged::P_READY].GetBool()));
- SendEvent(E_P2PALLREADYCHANGED, data);
- }
- void Network::P2PResetHost()
- {
- if (networkMode_ == SERVER_CLIENT) {
- return;
- }
- fullyConnectedMesh2_->ResetHostCalculation();
- }
- void Network::SetMode(NetworkMode mode, bool force)
- {
- if (rakPeer_->IsActive() || rakPeerClient_->IsActive()) {
- URHO3D_LOGERROR("Failed to change network mode! Shutdown networking system first or use forced mode changing!");
- return;
- }
- if (force) {
- if (rakPeer_->IsActive()) {
- rakPeer_->Shutdown(100);
- }
- if (rakPeerClient_->IsActive()) {
- rakPeerClient_->Shutdown(100);
- }
- clientConnections_.Clear();
- serverConnection_.Reset();
- }
- networkMode_ = mode;
- }
- const NetworkMode Network::GetMode() const
- {
- return networkMode_;
- }
- void RegisterNetworkLibrary(Context* context)
- {
- NetworkPriority::RegisterObject(context);
- }
- }
|