/* * Original work: Copyright (c) 2014, Oculus VR, Inc. * All rights reserved. * * This source code is licensed under the BSD-style license found in the * RakNet License.txt file in the licenses directory of this source tree. An additional grant * of patent rights can be found in the RakNet Patents.txt file in the same directory. * * * Modified work: Copyright (c) 2016-2021, SLikeSoft UG (haftungsbeschränkt) * * This source code was modified by SLikeSoft. Modifications are licensed under the MIT-style * license found in the license.txt file in the root directory of this source tree. */ // Common includes #include #include #include "slikenet/Kbhit.h" #include "slikenet/GetTime.h" #include "slikenet/peerinterface.h" #include "slikenet/MessageIdentifiers.h" #include "slikenet/BitStream.h" #include "slikenet/StringCompressor.h" #include "slikenet/FileListTransfer.h" #include "slikenet/FileList.h" // FLP_Printf #include "AutopatcherServer.h" #include "AutopatcherMySQLRepository.h" #include "slikenet/PacketizedTCP.h" #include "slikenet/Gets.h" #include "slikenet/linux_adapter.h" #include "slikenet/osx_adapter.h" #ifdef _WIN32 #include "slikenet/WindowsIncludes.h" // Sleep #else #include // usleep #endif #define USE_TCP #define LISTEN_PORT 60000 #define MAX_INCOMING_CONNECTIONS 128 int main(int, char **) { // Avoids the Error: Got a packet bigger than 'max_allowed_packet' bytes printf("Important: Requires that you first set the DB schema and the max packet size on the server.\n"); printf("See DependentExtensions/AutopatcherMySQLRepository/readme.txt\n"); // // MySQL is extremely slow in AutopatcherMySQLRepository::GetFilePart printf("WARNING: MySQL is an order of magnitude slower than PostgreSQL.\nRecommended you use AutopatcherServer_PostgreSQL instead."); printf("Server starting... "); SLNet::AutopatcherServer autopatcherServer; // SLNet::FLP_Printf progressIndicator; SLNet::FileListTransfer fileListTransfer; // So only one thread runs per connection, we create an array of connection objects, and tell the autopatcher server to use one thread per item static const int workerThreadCount=4; // Used for checking patches only static const int sqlConnectionObjectCount=32; // Used for both checking patches and downloading SLNet::AutopatcherMySQLRepository connectionObject[sqlConnectionObjectCount]; SLNet::AutopatcherRepositoryInterface *connectionObjectAddresses[sqlConnectionObjectCount]; for (int i=0; i < sqlConnectionObjectCount; i++) connectionObjectAddresses[i]=&connectionObject[i]; // fileListTransfer.AddCallback(&progressIndicator); autopatcherServer.SetFileListTransferPlugin(&fileListTransfer); // MySQL is extremely slow in AutopatcherMySQLRepository::GetFilePart so running tho incremental reads in a thread allows multiple concurrent users to read // Without this, only one user would be sent files at a time basically fileListTransfer.StartIncrementalReadThreads(sqlConnectionObjectCount); autopatcherServer.SetMaxConurrentUsers(MAX_INCOMING_CONNECTIONS); // More users than this get queued up SLNet::AutopatcherServerLoadNotifier_Printf loadNotifier; autopatcherServer.SetLoadManagementCallback(&loadNotifier); #ifdef USE_TCP SLNet::PacketizedTCP packetizedTCP; if (!packetizedTCP.Start(LISTEN_PORT,MAX_INCOMING_CONNECTIONS)) { printf("Failed to start TCP. Is the port already in use?"); return 1; } packetizedTCP.AttachPlugin(&autopatcherServer); packetizedTCP.AttachPlugin(&fileListTransfer); #else SLNet::RakPeerInterface *rakPeer; rakPeer = SLNet::RakPeerInterface::GetInstance(); SLNet::SocketDescriptor socketDescriptor(LISTEN_PORT,0); rakPeer->Startup(MAX_INCOMING_CONNECTIONS,&socketDescriptor, 1); rakPeer->SetMaximumIncomingConnections(MAX_INCOMING_CONNECTIONS); rakPeer->AttachPlugin(&autopatcherServer); rakPeer->AttachPlugin(&fileListTransfer); #endif printf("started.\n"); printf("Enter database password:\n"); char password[128]; char username[256]; strcpy_s(username, "root"); Gets(password,sizeof(password)); if (password[0]==0) strcpy_s(password,"aaaa"); char db[256]; printf("Enter DB schema: "); // To create the schema, go to the command line client and type create schema autopatcher; // You also have to add // max_allowed_packet=128M // Where 128 is the maximum size file in megabytes you'll ever add // to MySQL\MySQL Server 5.1\my.ini in the [mysqld] section // Be sure to restart the service after doing so Gets(db,sizeof(db)); if (db[0]==0) strcpy_s(db,"autopatcher"); for (int conIdx=0; conIdx < sqlConnectionObjectCount; conIdx++) { if (!connectionObject[conIdx].Connect("localhost", username, password, db, 0, nullptr, 0)) { printf("Database connection failed.\n"); return 1; } } printf("Database connection suceeded.\n"); printf("Starting threads\n"); autopatcherServer.StartThreads(workerThreadCount, sqlConnectionObjectCount, connectionObjectAddresses); printf("System ready for connections\n"); printf("(D)rop database\n(C)reate database.\n(A)dd application\n(U)pdate revision.\n(R)emove application\n(Q)uit\n"); int ch; SLNet::Packet *p; for(;;) { #ifdef USE_TCP SLNet::SystemAddress notificationAddress; notificationAddress=packetizedTCP.HasCompletedConnectionAttempt(); if (notificationAddress!= SLNet::UNASSIGNED_SYSTEM_ADDRESS) printf("ID_CONNECTION_REQUEST_ACCEPTED\n"); notificationAddress=packetizedTCP.HasNewIncomingConnection(); if (notificationAddress!= SLNet::UNASSIGNED_SYSTEM_ADDRESS) printf("ID_NEW_INCOMING_CONNECTION\n"); notificationAddress=packetizedTCP.HasLostConnection(); if (notificationAddress!= SLNet::UNASSIGNED_SYSTEM_ADDRESS) printf("ID_CONNECTION_LOST\n"); p=packetizedTCP.Receive(); while (p) { packetizedTCP.DeallocatePacket(p); p=packetizedTCP.Receive(); } #else p=rakPeer->Receive(); while (p) { if (p->data[0]==ID_NEW_INCOMING_CONNECTION) printf("ID_NEW_INCOMING_CONNECTION\n"); else if (p->data[0]==ID_DISCONNECTION_NOTIFICATION) printf("ID_DISCONNECTION_NOTIFICATION\n"); else if (p->data[0]==ID_CONNECTION_LOST) printf("ID_CONNECTION_LOST\n"); rakPeer->DeallocatePacket(p); p=rakPeer->Receive(); } #endif if (_kbhit()) { ch=_getch(); if (ch=='q') break; else if (ch=='c') { if (!connectionObject[0].CreateAutopatcherTables()) printf("Error: %s\n", connectionObject[0].GetLastError()); else printf("Created\n"); } else if (ch=='d') { if (!connectionObject[0].DestroyAutopatcherTables()) printf("Error: %s\n", connectionObject[0].GetLastError()); else printf("Destroyed\n"); } else if (ch=='a') { printf("Enter application name to add: "); char appName[512]; Gets(appName,sizeof(appName)); if (appName[0]==0) strcpy_s(appName, "TestApp"); if (!connectionObject[0].AddApplication(appName, username)) printf("Error: %s\n", connectionObject[0].GetLastError()); else printf("Done\n"); } else if (ch=='r') { printf("Enter application name to remove: "); char appName[512]; Gets(appName,sizeof(appName)); if (appName[0]==0) strcpy_s(appName, "TestApp"); if (!connectionObject[0].RemoveApplication(appName)) printf("Error: %s\n", connectionObject[0].GetLastError()); else printf("Done\n"); } else if (ch=='u') { printf("Enter application name: "); char appName[512]; Gets(appName,sizeof(appName)); if (appName[0]==0) strcpy_s(appName, "TestApp"); printf("Enter application directory: "); char appDir[512]; Gets(appDir,sizeof(appName)); if (appDir[0]==0) strcpy_s(appDir, "C:/temp"); if (!connectionObject[0].UpdateApplicationFiles(appName, appDir, username, 0)) { printf("Error: %s\n", connectionObject[0].GetLastError()); } else { printf("Update success.\n"); } } } RakSleep(30); } #ifdef USE_TCP packetizedTCP.Stop(); #else SLNet::RakPeerInterface::DestroyInstance(rakPeer); #endif }