mirror of
https://github.com/RetroShare/RetroShare.git
synced 2025-01-19 03:41:31 -05:00
d36b15d526
* Switched off most of std::cerr output. (expect UPnP) * moved pqi/pqidebug to util/rsdebug to allow greater usage. * added Mutexes to logging. * Downgraded many log messages to DEBUG. * New Log messages for connections, dht and udp code. * Added check for bad UDP socket. * Added check for non-firewalled scenario. * Increased Storage period from 1 Month to 6 Months for Links & Blog. git-svn-id: http://svn.code.sf.net/p/retroshare/code/trunk@644 b45a01b8-16f6-495d-af2f-9b41ad6348cc
1342 lines
29 KiB
C++
1342 lines
29 KiB
C++
/*
|
|
* libretroshare/src/services: p3gamelauncher.cc
|
|
*
|
|
* Other Bits for RetroShare.
|
|
*
|
|
* Copyright 2007-2008 by Robert Fernie.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License Version 2 as published by the Free Software Foundation.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
|
* USA.
|
|
*
|
|
* Please report all bugs and problems to "retroshare@lunamutt.com".
|
|
*
|
|
*/
|
|
|
|
#include "services/p3gamelauncher.h"
|
|
#include "services/p3gameservice.h"
|
|
#include "util/rsdebug.h"
|
|
#include "pqi/p3connmgr.h"
|
|
#include <sstream>
|
|
#include <iomanip>
|
|
|
|
/* global variable for GUI */
|
|
RsGameLauncher *rsGameLauncher = NULL;
|
|
|
|
/*****
|
|
* #define GAME_DEBUG 1
|
|
*****/
|
|
|
|
#define TEST_NO_GAMES 1
|
|
|
|
/* So STATEs is always the best way to do things....
|
|
*
|
|
* CLIENT:
|
|
* -------
|
|
* state: message result
|
|
* NULL
|
|
* <------- START ----- to INVITED state
|
|
* <------- RESUME ----- to INVITED(resume) etc...
|
|
*
|
|
* INVITED
|
|
* ------ INTERESTED -----> to READY state
|
|
* ------ REJECT -----> to NULL state
|
|
*
|
|
* READY
|
|
* <------- CONFIRMED ----- to/stay READY (update)
|
|
* <------- REJECTED ----- to NULL (abort)
|
|
* <------- PLAY ----- to ACTIVE
|
|
*
|
|
* ----
|
|
* below here the game s setup (game / players / order / etc is defined)
|
|
*
|
|
* ACTIVE (game play happens).
|
|
* <------ PAUSE -----> to NULL
|
|
* <------ disconnect ----> to NULL
|
|
* <------ QUIT -----> to NULL
|
|
*
|
|
********************************************************
|
|
*
|
|
* SERVER:
|
|
* -------
|
|
* state: message result
|
|
* NULL
|
|
* ------ START -----> to SETUP state
|
|
*
|
|
* SETUP
|
|
* <----- INTERESTED ------ to SETUP state
|
|
* <----- REJECT ------ to SETUP state
|
|
* ------ CONFIRMED -----> to SETUP state
|
|
* ------ REJECTED -----> to SETUP state
|
|
* ------ PLAY -----> to ACTIVE state
|
|
*
|
|
* ----
|
|
* below here the game s setup (game / players / order / etc is defined)
|
|
*
|
|
* ACTIVE (game play happens).
|
|
* <------- PAUSE ----- to NULL
|
|
* ------- disconnect ----- to NULL
|
|
* <------- QUIT ----- to NULL
|
|
*/
|
|
|
|
/* Game Setup States ****/
|
|
const uint32_t RS_GAME_INIT_INVITED = 1; /* Client */
|
|
const uint32_t RS_GAME_INIT_READY = 2; /* Client */
|
|
const uint32_t RS_GAME_INIT_SETUP = 3; /* Server */
|
|
const uint32_t RS_GAME_INIT_CONFIRM = 4; /* Server */
|
|
const uint32_t RS_GAME_INIT_ACTIVE = 5; /* Client/Server */
|
|
|
|
|
|
/* Game Setup Messages ****/
|
|
const uint32_t RS_GAME_MSG_START = 0; /* SERVER->CLIENT */
|
|
const uint32_t RS_GAME_MSG_RESUME = 1; /* SERVER->CLIENT * TODO later */
|
|
const uint32_t RS_GAME_MSG_INTERESTED = 2; /* CLIENT->SERVER */
|
|
const uint32_t RS_GAME_MSG_CONFIRM = 3; /* SERVER->CLIENT */
|
|
const uint32_t RS_GAME_MSG_PLAY = 4; /* SERVER->CLIENT */
|
|
const uint32_t RS_GAME_MSG_PAUSE = 5; /* ANY -> ANY */
|
|
const uint32_t RS_GAME_MSG_QUIT = 6; /* ANY -> ANY */
|
|
const uint32_t RS_GAME_MSG_REJECT = 6; /* ANY -> ANY */
|
|
|
|
const int p3gamezone = 1745;
|
|
|
|
p3GameLauncher::p3GameLauncher(p3ConnectMgr *connMgr)
|
|
:p3Service(RS_SERVICE_TYPE_GAME_LAUNCHER),
|
|
mConnMgr(connMgr)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::p3GameLauncher()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
addSerialType(new RsGameSerialiser());
|
|
mOwnId = mConnMgr->getOwnId();
|
|
}
|
|
|
|
int p3GameLauncher::tick()
|
|
{
|
|
pqioutput(PQL_DEBUG_BASIC, p3gamezone,
|
|
"p3GameLauncher::tick()");
|
|
checkIncoming();
|
|
return 0;
|
|
}
|
|
|
|
int p3GameLauncher::status()
|
|
{
|
|
pqioutput(PQL_DEBUG_BASIC, p3gamezone,
|
|
"p3GameLauncher::status()");
|
|
return 1;
|
|
}
|
|
|
|
|
|
/**** Interface to GUI Game Launcher ****/
|
|
|
|
|
|
std::string generateRandomGameId()
|
|
{
|
|
std::ostringstream out;
|
|
out << std::hex;
|
|
/********************************** WINDOWS/UNIX SPECIFIC PART ******************/
|
|
#ifndef WINDOWS_SYS
|
|
/* 4 bytes per random number: 4 x 4 = 16 bytes */
|
|
for(int i = 0; i < 4; i++)
|
|
{
|
|
out << std::setw(8) << std::setfill('0');
|
|
uint32_t rint = random();
|
|
out << rint;
|
|
}
|
|
#else
|
|
srand(time(NULL));
|
|
/* 2 bytes per random number: 8 x 2 = 16 bytes */
|
|
for(int i = 0; i < 8; i++)
|
|
{
|
|
out << std::setw(4) << std::setfill('0');
|
|
uint16_t rint = rand(); /* only gives 16 bits */
|
|
out << rint;
|
|
}
|
|
#endif
|
|
/********************************** WINDOWS/UNIX SPECIFIC PART ******************/
|
|
|
|
return out.str();
|
|
}
|
|
|
|
/**** GUI Interface ****/
|
|
|
|
|
|
bool p3GameLauncher::resumeGame(std::string)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::resumeGame()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
/* get game details from p3gameService */
|
|
/* add to status reports */
|
|
/* send resume invites to peers in list */
|
|
|
|
return ""; /* TODO */
|
|
}
|
|
|
|
|
|
/******************************************************************/
|
|
/******************************************************************/
|
|
/***** EXTERNAL RsGameLauncher Interface *******/
|
|
/******************************************************************/
|
|
/******************************************************************/
|
|
|
|
/* Server commands */
|
|
|
|
std::string p3GameLauncher::createGame(uint32_t gameType, std::wstring name)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::createGame()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
/* translate Id */
|
|
uint16_t srvId = gameType;
|
|
|
|
return newGame(srvId, name);
|
|
}
|
|
|
|
std::string p3GameLauncher::newGame(uint16_t srvId, std::wstring name)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::newGame()";
|
|
std::string tmpname(name.begin(), name.end());
|
|
std::cerr << "srvId: " << srvId << " name: " << tmpname;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
/* generate GameId (random string) */
|
|
std::string gameId = generateRandomGameId();
|
|
|
|
gameStatus newGame;
|
|
newGame.gameId = gameId;
|
|
newGame.serviceId = srvId;
|
|
newGame.numPlayers = 0;
|
|
newGame.gameName = name;
|
|
|
|
newGame.interestedPeers.clear();
|
|
newGame.peerIds.clear();
|
|
newGame.state = RS_GAME_INIT_SETUP; /* Server Only */
|
|
|
|
newGame.areServer = true;
|
|
newGame.serverId = mOwnId;
|
|
newGame.allowedPeers.push_back(mOwnId);
|
|
newGame.interestedPeers.push_back(mOwnId);
|
|
newGame.peerIds.push_back(mOwnId);
|
|
|
|
/* send messages to peers inviting to game */
|
|
std::list<std::string>::const_iterator it;
|
|
|
|
/* store gameStatus in list */
|
|
gamesCurrent[gameId] = newGame;
|
|
|
|
/* return new game Id */
|
|
return gameId;
|
|
}
|
|
|
|
void p3GameLauncher::cleanupGame(std::string gameId)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::cleanupGame()";
|
|
std::cerr << " gameId: " << gameId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
deleteGame(gameId);
|
|
}
|
|
|
|
bool p3GameLauncher::deleteGame(std::string gameId)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::deleteGame()";
|
|
std::cerr << " gameId: " << gameId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
std::map<std::string, gameStatus>::iterator git;
|
|
git = gamesCurrent.find(gameId);
|
|
if (git == gamesCurrent.end())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (git->second.state != RS_GAME_INIT_SETUP)
|
|
{
|
|
/* send off quit messages */
|
|
quitGame(gameId);
|
|
}
|
|
|
|
gamesCurrent.erase(git);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool p3GameLauncher::inviteGame(std::string gameId)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::inviteGame()";
|
|
std::cerr << " gameId: " << gameId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
std::map<std::string, gameStatus>::iterator git;
|
|
git = gamesCurrent.find(gameId);
|
|
if (git == gamesCurrent.end())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (git->second.state != RS_GAME_INIT_SETUP)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
/* send messages to peers inviting to game */
|
|
std::list<std::string>::const_iterator it;
|
|
|
|
for(it = git->second.allowedPeers.begin();
|
|
it != git->second.allowedPeers.end(); it++)
|
|
{
|
|
/* for an invite we need:
|
|
* serviceId, gameId, numPlayers....
|
|
*/
|
|
if (*it == mOwnId)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
RsGameItem *rgi = new RsGameItem();
|
|
rgi->serviceId = git->second.serviceId;
|
|
rgi->gameId = git->second.gameId;
|
|
rgi->gameComment = git->second.gameName;
|
|
rgi->numPlayers = git->second.numPlayers;
|
|
rgi->players.ids = git->second.allowedPeers;
|
|
|
|
rgi->msg = RS_GAME_MSG_START;
|
|
/* destination */
|
|
rgi->PeerId(*it);
|
|
|
|
/* send Msg */
|
|
sendItem(rgi);
|
|
}
|
|
|
|
git->second.state = RS_GAME_INIT_CONFIRM;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
/* support Game */
|
|
bool p3GameLauncher::confirmGame(std::string gameId)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::confirmGame()";
|
|
std::cerr << " gameId: " << gameId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
|
|
std::map<std::string, gameStatus>::iterator git;
|
|
|
|
git = gamesCurrent.find(gameId);
|
|
if (git == gamesCurrent.end())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
/* send messages to peers inviting to game */
|
|
std::list<std::string>::const_iterator it;
|
|
|
|
for(it = git->second.peerIds.begin();
|
|
it != git->second.peerIds.end(); it++)
|
|
{
|
|
/* for an invite we need:
|
|
* serviceId, gameId, numPlayers....
|
|
*/
|
|
|
|
if (*it == mOwnId)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
RsGameItem *rgi = new RsGameItem();
|
|
rgi->serviceId = git->second.serviceId;
|
|
rgi->gameId = git->second.gameId;
|
|
rgi->gameComment = git->second.gameName;
|
|
rgi->numPlayers = git->second.numPlayers;
|
|
rgi->players.ids = git->second.peerIds;
|
|
|
|
rgi->msg = RS_GAME_MSG_CONFIRM;
|
|
|
|
/* destination */
|
|
rgi->PeerId(*it);
|
|
|
|
/* send Msg */
|
|
sendItem(rgi);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool p3GameLauncher::playGame(std::string gameId)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::playGame()";
|
|
std::cerr << " gameId: " << gameId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
std::map<std::string, gameStatus>::iterator git;
|
|
|
|
git = gamesCurrent.find(gameId);
|
|
if (git == gamesCurrent.end())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (git->second.state != RS_GAME_INIT_CONFIRM)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
/* send messages to peers inviting to game */
|
|
std::list<std::string>::const_iterator it;
|
|
|
|
for(it = git->second.peerIds.begin();
|
|
it != git->second.peerIds.end(); it++)
|
|
{
|
|
/* for an invite we need:
|
|
* serviceId, gameId, numPlayers....
|
|
*/
|
|
|
|
if (*it == mOwnId)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
RsGameItem *rgi = new RsGameItem();
|
|
rgi->serviceId = git->second.serviceId;
|
|
rgi->gameId = git->second.gameId;
|
|
rgi->gameComment = git->second.gameName;
|
|
rgi->numPlayers = git->second.numPlayers;
|
|
rgi->players.ids = git->second.peerIds;
|
|
|
|
rgi->msg = RS_GAME_MSG_PLAY;
|
|
|
|
/* destination */
|
|
rgi->PeerId(*it);
|
|
|
|
/* send Msg */
|
|
sendItem(rgi);
|
|
}
|
|
|
|
/* inform all the other peers that we've started the game */
|
|
for(it = git->second.interestedPeers.begin();
|
|
it != git->second.interestedPeers.end(); it++)
|
|
{
|
|
if (git->second.peerIds.end() == (std::find(git->second.peerIds.begin(),
|
|
git->second.peerIds.end(), *it)))
|
|
{
|
|
/* tell the them they're not needed */
|
|
|
|
RsGameItem *rgi = new RsGameItem();
|
|
rgi->serviceId = git->second.serviceId;
|
|
rgi->gameId = git->second.gameId;
|
|
rgi->gameComment= git->second.gameName;
|
|
rgi->numPlayers = 0;
|
|
rgi->players.ids.clear();
|
|
|
|
rgi->msg = RS_GAME_MSG_REJECT;
|
|
|
|
/* destination */
|
|
rgi->PeerId(*it);
|
|
|
|
/* send Msg */
|
|
sendItem(rgi);
|
|
}
|
|
}
|
|
|
|
/* Finally start the actual Game */
|
|
|
|
/* TODO */
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool p3GameLauncher::quitGame(std::string gameId)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::checkGameProperties()";
|
|
std::cerr << " gameId: " << gameId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
/* TODO */
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
|
|
bool p3GameLauncher::invitePeer(std::string gameId, std::string peerId)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::invitePeer()";
|
|
std::cerr << " gameId: " << gameId << " peerId: " << peerId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
std::map<std::string, gameStatus>::iterator git;
|
|
git = gamesCurrent.find(gameId);
|
|
if (git == gamesCurrent.end())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (git->second.state != RS_GAME_INIT_SETUP)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (peerId == mOwnId)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
/* send messages to peers inviting to game */
|
|
std::list<std::string>::const_iterator it;
|
|
if (git->second.allowedPeers.end() !=
|
|
(it = std::find(git->second.allowedPeers.begin(),
|
|
git->second.allowedPeers.end(), peerId)))
|
|
{
|
|
return true;
|
|
}
|
|
|
|
git->second.allowedPeers.push_back(peerId);
|
|
return true;
|
|
}
|
|
|
|
|
|
/* Server GUI commands */
|
|
bool p3GameLauncher::uninvitePeer(std::string gameId, std::string peerId)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::uninvitePeer()";
|
|
std::cerr << " gameId: " << gameId << " peerId: " << peerId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
std::map<std::string, gameStatus>::iterator git;
|
|
git = gamesCurrent.find(gameId);
|
|
if (git == gamesCurrent.end())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (git->second.state != RS_GAME_INIT_SETUP)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (peerId == mOwnId)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
/* send messages to peers inviting to game */
|
|
std::list<std::string>::iterator it;
|
|
if (git->second.allowedPeers.end() ==
|
|
(it = std::find(git->second.allowedPeers.begin(),
|
|
git->second.allowedPeers.end(), peerId)))
|
|
{
|
|
return true;
|
|
}
|
|
|
|
git->second.allowedPeers.erase(it);
|
|
return true;
|
|
}
|
|
|
|
|
|
bool p3GameLauncher::confirmPeer(std::string gameId, std::string peerId,
|
|
int16_t pos)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::confirmPeer()";
|
|
std::cerr << " gameId: " << gameId << " peerId: " << peerId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
std::map<std::string, gameStatus>::iterator git;
|
|
|
|
git = gamesCurrent.find(gameId);
|
|
if (git == gamesCurrent.end())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (git->second.state != RS_GAME_INIT_CONFIRM)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
std::list<std::string>::iterator it;
|
|
if (git->second.interestedPeers.end() ==
|
|
(it = std::find(git->second.interestedPeers.begin(),
|
|
git->second.interestedPeers.end(), peerId)))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
it = std::find(git->second.peerIds.begin(),
|
|
git->second.peerIds.end(), peerId);
|
|
if (it != git->second.peerIds.end())
|
|
{
|
|
git->second.peerIds.erase(it);
|
|
}
|
|
|
|
int32_t i = 0;
|
|
for(it = git->second.peerIds.begin(); (i < pos) &&
|
|
(it != git->second.peerIds.end()); it++, i++);
|
|
|
|
if ((pos < 0) || (it == git->second.peerIds.end()))
|
|
{
|
|
/* */
|
|
git->second.peerIds.push_back(peerId);
|
|
}
|
|
|
|
git->second.peerIds.insert(it, peerId);
|
|
return confirmGame(gameId);
|
|
}
|
|
|
|
|
|
bool p3GameLauncher::unconfirmPeer(std::string gameId, std::string peerId)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::unconfirmPeer()";
|
|
std::cerr << " gameId: " << gameId << " peerId: " << peerId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
std::map<std::string, gameStatus>::iterator git;
|
|
|
|
git = gamesCurrent.find(gameId);
|
|
if (git == gamesCurrent.end())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (git->second.state != RS_GAME_INIT_CONFIRM)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
std::list<std::string>::iterator it;
|
|
|
|
it = std::find(git->second.peerIds.begin(),
|
|
git->second.peerIds.end(), peerId);
|
|
if (it != git->second.peerIds.end())
|
|
{
|
|
git->second.peerIds.erase(it);
|
|
}
|
|
return confirmGame(gameId);
|
|
}
|
|
|
|
|
|
/* Client GUI Commands */
|
|
bool p3GameLauncher::interestedPeer(std::string gameId)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::interestedPeer()";
|
|
std::cerr << " gameId: " << gameId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
return inviteResponse(gameId, true);
|
|
}
|
|
|
|
bool p3GameLauncher::uninterestedPeer(std::string gameId)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::uninterestedPeer()";
|
|
std::cerr << " gameId: " << gameId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
return inviteResponse(gameId, false);
|
|
}
|
|
|
|
bool p3GameLauncher::inviteResponse(std::string gameId, bool interested)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::inviteResponse()";
|
|
std::cerr << " gameId: " << gameId << "interested: " << interested;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
std::map<std::string, gameStatus>::iterator git;
|
|
|
|
git = gamesCurrent.find(gameId);
|
|
if (git == gamesCurrent.end())
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
/* TODO */
|
|
|
|
RsGameItem *rgi = new RsGameItem();
|
|
rgi->serviceId = git->second.serviceId;
|
|
rgi->gameId = git->second.gameId;
|
|
rgi->gameComment= git->second.gameName;
|
|
rgi->numPlayers = 0;
|
|
rgi->players.ids.clear();
|
|
|
|
if (interested)
|
|
{
|
|
rgi->msg = RS_GAME_MSG_INTERESTED;
|
|
git->second.state = RS_GAME_INIT_READY;
|
|
}
|
|
else
|
|
{
|
|
rgi->msg = RS_GAME_MSG_REJECT;
|
|
git->second.state = RS_GAME_INIT_INVITED;
|
|
}
|
|
|
|
/* destination */
|
|
rgi->PeerId(git->second.serverId);
|
|
|
|
/* send Msg */
|
|
sendItem(rgi);
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
/***** Details ****/
|
|
|
|
/* get details */
|
|
bool p3GameLauncher::getGameList(std::list<RsGameInfo> &gameList)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::getGameList()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
std::map<std::string, gameStatus>::iterator git;
|
|
for(git = gamesCurrent.begin(); git != gamesCurrent.end(); git++)
|
|
{
|
|
RsGameInfo info;
|
|
|
|
info.gameId = git->first;
|
|
info.serverId = git->second.serverId;
|
|
|
|
std::ostringstream out;
|
|
out << "GameType: " << git->second.serviceId;
|
|
info.gameType = out.str();
|
|
|
|
info.serverName = "ServerName";
|
|
info.numPlayers = git->second.numPlayers;
|
|
info.gameName = git->second.gameName;
|
|
|
|
if (git->second.state == RS_GAME_INIT_SETUP)
|
|
{
|
|
info.status = "Setup";
|
|
}
|
|
else if (git->second.state == RS_GAME_INIT_INVITED)
|
|
{
|
|
info.status = "Invite";
|
|
}
|
|
else if (git->second.state == RS_GAME_INIT_CONFIRM)
|
|
{
|
|
info.status = "Confirm";
|
|
}
|
|
else if (git->second.state == RS_GAME_INIT_READY)
|
|
{
|
|
info.status = "Ready";
|
|
}
|
|
else if (git->second.state == RS_GAME_INIT_ACTIVE)
|
|
{
|
|
info.status = "Playing";
|
|
}
|
|
else
|
|
{
|
|
info.status = "Unknown";
|
|
}
|
|
|
|
gameList.push_back(info);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool p3GameLauncher::getGameDetail(std::string gameId, RsGameDetail &detail)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::getGameDetail()";
|
|
std::cerr << " gameId: " << gameId;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
std::map<std::string, gameStatus>::iterator git;
|
|
std::list<std::string>::iterator it;
|
|
git = gamesCurrent.find(gameId);
|
|
if (git == gamesCurrent.end())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
/* fill in the details */
|
|
detail.gameId = gameId;
|
|
detail.gameType = git->second.serviceId;
|
|
detail.gameName = git->second.gameName;
|
|
detail.areServer = git->second.areServer;
|
|
detail.serverId = git->second.serverId;
|
|
detail.serverName = "Server???";
|
|
detail.numPlayers = git->second.numPlayers;
|
|
|
|
if ((git->second.state == RS_GAME_INIT_SETUP) ||
|
|
(git->second.state == RS_GAME_INIT_INVITED))
|
|
{
|
|
if (git->second.state == RS_GAME_INIT_SETUP)
|
|
detail.status = "Setup";
|
|
else
|
|
detail.status = "Invite";
|
|
|
|
/* copy from invited List */
|
|
for(it = git->second.allowedPeers.begin();
|
|
it != git->second.allowedPeers.end(); it++)
|
|
{
|
|
RsGamePeer rgp;
|
|
rgp.id = *it;
|
|
rgp.invite = true;
|
|
rgp.interested = false;
|
|
rgp.play = false;
|
|
|
|
detail.gamers[*it] = rgp;
|
|
}
|
|
}
|
|
else if ((git->second.state == RS_GAME_INIT_CONFIRM) ||
|
|
(git->second.state == RS_GAME_INIT_READY))
|
|
{
|
|
if (git->second.state == RS_GAME_INIT_CONFIRM)
|
|
detail.status = "Confirm";
|
|
else
|
|
detail.status = "Ready";
|
|
|
|
/* copy from invited List */
|
|
for(it = git->second.allowedPeers.begin();
|
|
it != git->second.allowedPeers.end(); it++)
|
|
{
|
|
RsGamePeer rgp;
|
|
rgp.id = *it;
|
|
rgp.invite = true;
|
|
|
|
if (git->second.interestedPeers.end() !=
|
|
std::find(git->second.interestedPeers.begin(),
|
|
git->second.interestedPeers.end(),
|
|
*it))
|
|
{
|
|
rgp.interested = true;
|
|
}
|
|
else
|
|
{
|
|
rgp.interested = false;
|
|
}
|
|
/* if in peerIds */
|
|
if (git->second.peerIds.end() !=
|
|
std::find(git->second.peerIds.begin(),
|
|
git->second.peerIds.end(),
|
|
*it))
|
|
{
|
|
rgp.play = true;
|
|
}
|
|
else
|
|
{
|
|
rgp.play = false;
|
|
}
|
|
detail.gamers[*it] = rgp;
|
|
}
|
|
|
|
|
|
}
|
|
else if (git->second.state == RS_GAME_INIT_ACTIVE)
|
|
{
|
|
detail.status = "Playing";
|
|
|
|
/* copy from invited List */
|
|
for(it = git->second.peerIds.begin();
|
|
it != git->second.peerIds.end(); it++)
|
|
{
|
|
RsGamePeer rgp;
|
|
rgp.id = *it;
|
|
rgp.invite = true;
|
|
rgp.interested = true;
|
|
rgp.play = true;
|
|
|
|
detail.gamers[*it] = rgp;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
/**** Network interface ****/
|
|
|
|
int p3GameLauncher::checkIncoming()
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::checkIncoming()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
/* check for incoming items */
|
|
|
|
RsGameItem *gi = NULL;
|
|
|
|
while(NULL != (gi = (RsGameItem *) recvItem()))
|
|
{
|
|
handleIncoming(gi);
|
|
delete gi;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
|
|
int p3GameLauncher::handleIncoming(RsGameItem *gi)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
/* check that its a valid packet...
|
|
* and that there is a gameStatus.
|
|
*/
|
|
|
|
/* Always check the Properties */
|
|
if (!checkGameProperties(gi->serviceId, gi->numPlayers))
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
sendRejectMsg(gi);
|
|
return 0;
|
|
}
|
|
|
|
/* check if there is an existing game? */
|
|
std::map<std::string, gameStatus>::iterator it;
|
|
bool haveStatus = (gamesCurrent.end() !=
|
|
(it = gamesCurrent.find(gi->gameId)));
|
|
|
|
/* handle startup first */
|
|
if (!haveStatus)
|
|
{
|
|
if (gi->msg == RS_GAME_MSG_START)
|
|
{
|
|
/***** CLIENT HANDLING ****/
|
|
/* we are the client -> start it up! */
|
|
return handleClientStart(gi);
|
|
}
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
sendRejectMsg(gi);
|
|
return 0;
|
|
}
|
|
|
|
/* have a current status - if we get here
|
|
* switch on
|
|
* 1) server/client.
|
|
* 2) state,
|
|
* 3) msg.
|
|
*/
|
|
|
|
if (it->second.areServer)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming() AreServer for Game";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
/***** SERVER HANDLING ****/
|
|
switch(it->second.state)
|
|
{
|
|
case RS_GAME_INIT_CONFIRM:
|
|
/* only accept INTERESTED | REJECT */
|
|
if ((gi->msg == RS_GAME_MSG_INTERESTED) ||
|
|
(gi->msg == RS_GAME_MSG_REJECT))
|
|
{
|
|
handleServerSetup(gi);
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming()";
|
|
std::cerr << " INIT_CONFIRM & msg != INT | REJ - reject";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
sendRejectMsg(gi);
|
|
return 0;
|
|
}
|
|
break;
|
|
|
|
case RS_GAME_INIT_ACTIVE:
|
|
if ((gi->msg == RS_GAME_MSG_PAUSE) ||
|
|
(gi->msg == RS_GAME_MSG_QUIT))
|
|
{
|
|
handleServerActive(gi);
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming()";
|
|
std::cerr << " INIT_ACTIVE & msg != PAU | QUIT - reject ";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
sendRejectMsg(gi);
|
|
return 0;
|
|
}
|
|
break;
|
|
|
|
case RS_GAME_INIT_SETUP: /* invalid state */
|
|
case RS_GAME_INIT_INVITED: /* invalid state */
|
|
case RS_GAME_INIT_READY: /* invalid state */
|
|
default:
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming()";
|
|
std::cerr << " INIT_SETUP | INIT_INVITED | INIT_READY | default - reject ";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
sendRejectMsg(gi);
|
|
return 0;
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming() AreClient for Game";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
/***** CLIENT HANDLING ****/
|
|
switch(it->second.state)
|
|
{
|
|
case RS_GAME_INIT_INVITED:
|
|
/* only accept REJECT */
|
|
if (gi->msg == RS_GAME_MSG_REJECT)
|
|
{
|
|
handleClientInvited(gi);
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming()";
|
|
std::cerr << " INIT_INVITED & msg != REJ - reject ";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
sendRejectMsg(gi);
|
|
return 0;
|
|
}
|
|
break;
|
|
|
|
case RS_GAME_INIT_READY:
|
|
|
|
if ((gi->msg == RS_GAME_MSG_CONFIRM) ||
|
|
(gi->msg == RS_GAME_MSG_REJECT) ||
|
|
(gi->msg == RS_GAME_MSG_PLAY))
|
|
{
|
|
handleClientReady(gi);
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming()";
|
|
std::cerr << " INIT_READY & msg != CFM, REJ, PLY - reject ";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
sendRejectMsg(gi);
|
|
return 0;
|
|
}
|
|
break;
|
|
|
|
case RS_GAME_INIT_ACTIVE:
|
|
if ((gi->msg == RS_GAME_MSG_PAUSE) ||
|
|
(gi->msg == RS_GAME_MSG_QUIT))
|
|
{
|
|
handleClientActive(gi);
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming()";
|
|
std::cerr << " INIT_ACTIVE & msg != PAU, QUIT - reject ";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
sendRejectMsg(gi);
|
|
return 0;
|
|
}
|
|
break;
|
|
|
|
|
|
case RS_GAME_INIT_SETUP: /* invalid state */
|
|
default:
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming()";
|
|
std::cerr << " INIT_SETUP - invalid state - reject ";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
sendRejectMsg(gi);
|
|
return 0;
|
|
break;
|
|
}
|
|
}
|
|
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleIncoming() Never Get Here - reject";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
/* should never get here */
|
|
sendRejectMsg(gi);
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
/***** Network Msg Functions *****
|
|
*
|
|
*
|
|
*
|
|
handleClientStart(gi) * START msg *
|
|
handleClientInvited(gi); * REJECT msg *
|
|
handleClientReady(gi); * CONFIRM / REJECT / PLAY msg *
|
|
handleClientActive(gi); * PAUSE / QUIT msg *
|
|
|
|
handleServerSetup(gi); * INTERESTED / REJECT msg *
|
|
handleServerActive(gi); * PAUSE / QUIT msg *
|
|
sendRejectMsg(gi);
|
|
*
|
|
*
|
|
*
|
|
*/
|
|
|
|
/* START msg */
|
|
int p3GameLauncher::handleClientStart(RsGameItem *gi)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleClientStart()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
/* Already checked existance / Properties */
|
|
|
|
/* else -> add into the list of games */
|
|
gameStatus gs;
|
|
gs.serviceId = gi->serviceId;
|
|
gs.gameId = gi->gameId;
|
|
gs.gameName = gi->gameComment;
|
|
gs.areServer = false;
|
|
gs.serverId = gi->PeerId();
|
|
gs.state = RS_GAME_INIT_INVITED; /* Client */
|
|
gs.numPlayers = gi->numPlayers;
|
|
gs.allowedPeers = gi->players.ids;
|
|
//gs.interestedPeers = gi->players.ids;
|
|
//gs.peerIds = gi->players.ids;
|
|
|
|
gamesCurrent[gi->gameId] = gs;
|
|
|
|
return 1;
|
|
}
|
|
|
|
/* REJECT msg */
|
|
int p3GameLauncher::handleClientInvited(RsGameItem *gi)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleClientInvited()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
/* cleanup game */
|
|
cleanupGame(gi->gameId);
|
|
return 1;
|
|
}
|
|
|
|
/* CONFIRM / REJECT / PLAY msg */
|
|
int p3GameLauncher::handleClientReady(RsGameItem *gi)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleClientReady()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
/* get game */
|
|
std::map<std::string, gameStatus>::iterator it;
|
|
if (gamesCurrent.end() == (it = gamesCurrent.find(gi->gameId)))
|
|
{
|
|
/* no game exists */
|
|
return 0;
|
|
}
|
|
|
|
switch(gi->msg)
|
|
{
|
|
case RS_GAME_MSG_CONFIRM:
|
|
|
|
/* update the information
|
|
* (other info should be the same)
|
|
*/
|
|
it->second.numPlayers = gi->numPlayers;
|
|
// Which one?
|
|
it->second.interestedPeers = gi->players.ids;
|
|
//it->second.peerIds = gi->players.ids;
|
|
|
|
return 1;
|
|
break;
|
|
|
|
case RS_GAME_MSG_REJECT:
|
|
cleanupGame(gi->gameId);
|
|
return 1;
|
|
break;
|
|
|
|
case RS_GAME_MSG_PLAY:
|
|
/* TODO */
|
|
return 1;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* PAUSE / QUIT msg */
|
|
int p3GameLauncher::handleClientActive(RsGameItem *gi)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleClientActive()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
/* INTERESTED / REJECT msg */
|
|
int p3GameLauncher::handleServerSetup(RsGameItem *gi)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleServerSetup()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
/* check if there is an existing game? */
|
|
std::map<std::string, gameStatus>::iterator it;
|
|
if (gamesCurrent.end() == (it = gamesCurrent.find(gi->gameId)))
|
|
{
|
|
/* no game exists */
|
|
return 0;
|
|
}
|
|
|
|
/* we only care about this notice -> if we're the server */
|
|
if (it->second.areServer)
|
|
{
|
|
std::list<std::string>::iterator it2, it3;
|
|
it2 = std::find(it->second.allowedPeers.begin(),
|
|
it->second.allowedPeers.end(), gi->PeerId());
|
|
it3 = std::find(it->second.interestedPeers.begin(),
|
|
it->second.interestedPeers.end(), gi->PeerId());
|
|
|
|
if ((it2 != it->second.allowedPeers.end()) &&
|
|
(it3 == it->second.interestedPeers.end()))
|
|
{
|
|
it->second.interestedPeers.push_back(gi->PeerId());
|
|
return 1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* This is a setup update from server
|
|
* only updates the players...
|
|
*/
|
|
int p3GameLauncher::handleServerActive(RsGameItem *gi)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::handleServerActive()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
|
|
return 1;
|
|
}
|
|
|
|
int p3GameLauncher::sendRejectMsg(RsGameItem *gi)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::sendRejectMsg()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
/* all should be okay ... except msg */
|
|
|
|
RsGameItem *response = new RsGameItem();
|
|
response->serviceId = gi->serviceId;
|
|
response->numPlayers = gi->numPlayers;
|
|
response->msg = RS_GAME_MSG_REJECT;
|
|
response->gameId = gi->gameId;
|
|
response->gameComment = gi->gameComment;
|
|
response->players.ids = gi->players.ids;
|
|
|
|
sendItem(response);
|
|
return 1;
|
|
}
|
|
|
|
bool p3GameLauncher::checkGameProperties(uint16_t serviceId, uint16_t players)
|
|
{
|
|
#ifdef GAME_DEBUG
|
|
std::cerr << "p3GameLauncher::checkGameProperties()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
#ifdef TEST_NO_GAMES
|
|
return true;
|
|
#endif
|
|
|
|
std::map<uint16_t, p3GameService *>::iterator it;
|
|
if (gameList.end() == (it = gameList.find(serviceId)))
|
|
{
|
|
return false; /* we don't support the game */
|
|
}
|
|
|
|
if ((players <= it->second->getMaxPlayers()) &&
|
|
(players >= it->second->getMinPlayers()))
|
|
{
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|