/*************************************************************************** * Copyright (C) 2008 by Max-Wilhelm Bruker * * brukie@laptop * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 2 of the License, or * * (at your option) any later version. * * * * This program 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 General Public License for more details. * * * * You should have received a copy of the GNU General Public License * * along with this program; if not, write to the * * Free Software Foundation, Inc., * * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * ***************************************************************************/ #include "server.h" #include "server_game.h" #include "server_counter.h" #include "server_room.h" #include "server_protocolhandler.h" #include "pb/event_user_joined.pb.h" #include "pb/event_user_left.pb.h" #include "pb/event_list_rooms.pb.h" #include #include Server::Server(QObject *parent) : QObject(parent), serverMutex(QMutex::Recursive), nextGameId(0) { } Server::~Server() { } void Server::prepareDestroy() { QMutexLocker locker(&serverMutex); while (!clients.isEmpty()) delete clients.takeFirst(); QMapIterator roomIterator(rooms); while (roomIterator.hasNext()) delete roomIterator.next().value(); } AuthenticationResult Server::loginUser(Server_ProtocolHandler *session, QString &name, const QString &password, QString &reasonStr) { QMutexLocker locker(&serverMutex); if (name.size() > 35) name = name.left(35); AuthenticationResult authState = checkUserPassword(session, name, password, reasonStr); if ((authState == NotLoggedIn) || (authState == UserIsBanned)) return authState; ServerInfo_User data = getUserData(name); data.set_address(session->getAddress().toStdString()); name = QString::fromStdString(data.name()); // Compensate for case indifference if (authState == PasswordRight) { if (users.contains(name)) { qDebug("Login denied: would overwrite old session"); return WouldOverwriteOldSession; } } else if (authState == UnknownUser) { // Change user name so that no two users have the same names, // don't interfere with registered user names though. QString tempName = name; int i = 0; while (users.contains(tempName) || userExists(tempName)) tempName = name + "_" + QString::number(++i); name = tempName; data.set_name(name.toStdString()); } session->setUserInfo(data); users.insert(name, session); qDebug() << "Server::loginUser: name=" << name; session->setSessionId(startSession(name, session->getAddress())); qDebug() << "session id:" << session->getSessionId(); Event_UserJoined event; event.mutable_user_info()->CopyFrom(session->copyUserInfo(false)); SessionEvent *se = Server_ProtocolHandler::prepareSessionEvent(event); for (int i = 0; i < clients.size(); ++i) if (clients[i]->getAcceptsUserListChanges()) clients[i]->sendProtocolItem(*se); delete se; return authState; } void Server::addClient(Server_ProtocolHandler *client) { QMutexLocker locker(&serverMutex); clients << client; } void Server::removeClient(Server_ProtocolHandler *client) { QMutexLocker locker(&serverMutex); clients.removeAt(clients.indexOf(client)); ServerInfo_User *data = client->getUserInfo(); if (data) { Event_UserLeft event; event.set_name(data->name()); SessionEvent *se = Server_ProtocolHandler::prepareSessionEvent(event); for (int i = 0; i < clients.size(); ++i) if (clients[i]->getAcceptsUserListChanges()) clients[i]->sendProtocolItem(*se); delete se; users.remove(QString::fromStdString(data->name())); qDebug() << "Server::removeClient: name=" << QString::fromStdString(data->name()); if (client->getSessionId() != -1) endSession(client->getSessionId()); qDebug() << "closed session id:" << client->getSessionId(); } qDebug() << "Server::removeClient:" << clients.size() << "clients; " << users.size() << "users left"; } void Server::broadcastRoomUpdate() { QMutexLocker locker(&serverMutex); Server_Room *room = static_cast(sender()); Event_ListRooms event; ServerInfo_Room *roomInfo = event.add_room_list(); room->roomMutex.lock(); roomInfo->CopyFrom(room->getInfo(false)); room->roomMutex.unlock(); SessionEvent *se = Server_ProtocolHandler::prepareSessionEvent(event); for (int i = 0; i < clients.size(); ++i) if (clients[i]->getAcceptsRoomListChanges()) clients[i]->sendProtocolItem(*se); delete se; } void Server::addRoom(Server_Room *newRoom) { QMutexLocker locker(&serverMutex); rooms.insert(newRoom->getId(), newRoom); connect(newRoom, SIGNAL(roomInfoChanged()), this, SLOT(broadcastRoomUpdate())); } int Server::getUsersCount() const { QMutexLocker locker(&serverMutex); return users.size(); } int Server::getGamesCount() const { int result = 0; QMutexLocker locker(&serverMutex); QMapIterator roomIterator(rooms); while (roomIterator.hasNext()) { Server_Room *room = roomIterator.next().value(); QMutexLocker roomLocker(&room->roomMutex); result += room->getGames().size(); } return result; }