1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197
|
/*
* Copyright (C) 2001-2006 Jacek Sieka, arnetheduck on gmail point com
*
* 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.
*/
#if !defined(CONNECTION_MANAGER_H)
#define CONNECTION_MANAGER_H
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#include "TimerManager.h"
#include "UserConnection.h"
#include "User.h"
#include "CriticalSection.h"
#include "Singleton.h"
#include "Util.h"
#include "ConnectionManagerListener.h"
class SocketException;
class ConnectionQueueItem {
public:
typedef ConnectionQueueItem* Ptr;
typedef vector<Ptr> List;
typedef List::iterator Iter;
enum State {
CONNECTING, // Recently sent request to connect
WAITING, // Waiting to send request to connect
NO_DOWNLOAD_SLOTS, // Not needed right now
ACTIVE // In one up/downmanager
};
ConnectionQueueItem(const User::Ptr& aUser, bool aDownload) : state(WAITING), lastAttempt(0), download(aDownload), user(aUser) { }
User::Ptr& getUser() { return user; }
const User::Ptr& getUser() const { return user; }
GETSET(State, state, State);
GETSET(uint32_t, lastAttempt, LastAttempt);
GETSET(bool, download, Download);
private:
ConnectionQueueItem(const ConnectionQueueItem&);
ConnectionQueueItem& operator=(const ConnectionQueueItem&);
User::Ptr user;
};
class ExpectedMap {
public:
void add(const string& aNick, const string& aMyNick, const string& aHubUrl) {
Lock l(cs);
expectedConnections.insert(make_pair(aNick, make_pair(aMyNick, aHubUrl)));
}
StringPair remove(const string& aNick) {
Lock l(cs);
ExpectMap::iterator i = expectedConnections.find(aNick);
if(i == expectedConnections.end())
return make_pair(Util::emptyString, Util::emptyString);
StringPair tmp = i->second;
expectedConnections.erase(i);
return tmp;
}
private:
/** Nick -> myNick, hubUrl for expected NMDC incoming connections */
typedef map<string, StringPair> ExpectMap;
ExpectMap expectedConnections;
CriticalSection cs;
};
// Comparing with a user...
inline bool operator==(ConnectionQueueItem::Ptr ptr, const User::Ptr& aUser) { return ptr->getUser() == aUser; }
class ConnectionManager : public Speaker<ConnectionManagerListener>,
public UserConnectionListener, TimerManagerListener,
public Singleton<ConnectionManager>
{
public:
void nmdcExpect(const string& aNick, const string& aMyNick, const string& aHubUrl) {
expectedConnections.add(aNick, aMyNick, aHubUrl);
}
void nmdcConnect(const string& aServer, uint16_t aPort, const string& aMyNick, const string& hubUrl, const string& encoding);
void adcConnect(const OnlineUser& aUser, uint16_t aPort, const string& aToken, bool secure);
void getDownloadConnection(const User::Ptr& aUser);
void disconnect(const User::Ptr& aUser, int isDownload);
void shutdown();
/** Find a suitable port to listen on, and start doing it */
void listen() throw(SocketException);
void disconnect() throw();
uint16_t getPort() { return server ? static_cast<uint16_t>(server->getPort()) : 0; }
uint16_t getSecurePort() { return secureServer ? static_cast<uint16_t>(secureServer->getPort()) : 0; }
private:
class Server : public Thread {
public:
Server(bool secure_, uint16_t port, const string& ip = "0.0.0.0");
uint16_t getPort() { return port; }
virtual ~Server() { die = true; join(); }
private:
virtual int run() throw();
Socket sock;
uint16_t port;
bool secure;
bool die;
};
friend class Server;
CriticalSection cs;
/** All ConnectionQueueItems */
ConnectionQueueItem::List downloads;
ConnectionQueueItem::List uploads;
/** All active connections */
UserConnection::List userConnections;
User::List checkIdle;
StringList features;
StringList adcFeatures;
ExpectedMap expectedConnections;
uint32_t floodCounter;
Server* server;
Server* secureServer;
bool shuttingDown;
friend class Singleton<ConnectionManager>;
ConnectionManager();
virtual ~ConnectionManager() throw() { shutdown(); }
UserConnection* getConnection(bool aNmdc, bool secure) throw();
void putConnection(UserConnection* aConn);
void addUploadConnection(UserConnection* uc);
void addDownloadConnection(UserConnection* uc);
ConnectionQueueItem* getCQI(const User::Ptr& aUser, bool download);
void putCQI(ConnectionQueueItem* cqi);
void accept(const Socket& sock, bool secure) throw();
// UserConnectionListener
virtual void on(Connected, UserConnection*) throw();
virtual void on(Failed, UserConnection*, const string&) throw();
virtual void on(CLock, UserConnection*, const string&, const string&) throw();
virtual void on(Key, UserConnection*, const string&) throw();
virtual void on(Direction, UserConnection*, const string&, const string&) throw();
virtual void on(MyNick, UserConnection*, const string&) throw();
virtual void on(Supports, UserConnection*, const StringList&) throw();
virtual void on(AdcCommand::SUP, UserConnection*, const AdcCommand&) throw();
virtual void on(AdcCommand::INF, UserConnection*, const AdcCommand&) throw();
virtual void on(AdcCommand::STA, UserConnection*, const AdcCommand&) throw();
// TimerManagerListener
virtual void on(TimerManagerListener::Second, uint32_t aTick) throw();
virtual void on(TimerManagerListener::Minute, uint32_t aTick) throw();
};
#endif // !defined(CONNECTION_MANAGER_H)
|