khanat-opennel-code/code/nel/include/nel/net/naming_client.h
acemtp@users.sourceforge.net d5c601ffa5 initial version
2010-05-06 02:08:41 +02:00

231 lines
8.5 KiB
C++

// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
#ifndef NL_NAMING_CLIENT_H
#define NL_NAMING_CLIENT_H
#include "nel/misc/types_nl.h"
#include <string>
#include "nel/misc/mutex.h"
#include "inet_address.h"
#include "callback_client.h"
#include "service.h"
namespace NLNET {
//typedef uint16 TServiceId;
typedef void (*TBroadcastCallback)(const std::string &name, TServiceId sid, const std::vector<CInetAddress> &addr);
/**
* Client side of Naming Service. Allows to register/unregister services, and to lookup for
* a registered service.
*
* \warning the Naming Service can be down, it will reconnect when up but if other services try to register during the
* NS is offline, it will cause bug
*
* \author Olivier Cado
* \author Vianney Lecroart
* \author Nevrax France
* \date 2001
*/
class CNamingClient
{
public:
struct CServiceEntry
{
CServiceEntry (std::string n, TServiceId s, std::vector<CInetAddress> a) : Name(n), SId (s), Addr(a) { }
// name of the service
std::string Name;
// id of the service
TServiceId SId;
// address to send to the service who wants to lookup this service (could have more than one)
std::vector<CInetAddress> Addr;
};
public:
/// Connect to the naming service.
static void connect( const CInetAddress& addr, CCallbackNetBase::TRecordingState rec, const std::vector<CInetAddress> &addresses );
/// Return true if the connection to the Naming Service was done.
static bool connected () { return _Connection != NULL && _Connection->connected (); }
/// Close the connection to the naming service.
static void disconnect ();
/// Returns information about the naming connection (connected or not, which naming service and so on)
static std::string info ();
/** Register a service within the naming service.
* Returns false if the registration failed.
*/
static bool registerService (const std::string &name, const std::vector<CInetAddress> &addr, TServiceId &sid);
/** Register a service within the naming service, using a specified service identifier.
* Returns false if the service identifier is unavailable i.e. the registration failed.
*/
static bool registerServiceWithSId (const std::string &name, const std::vector<CInetAddress> &addr, TServiceId sid);
/** If the NS is down and goes up, we have to send it again the registration. But in this case, the NS
* must not send a registration broadcast, so we have a special message
*/
static void resendRegisteration (const std::string &name, const std::vector<CInetAddress> &addr, TServiceId sid);
/// Unregister a service from the naming service, service identifier.
static void unregisterService (TServiceId sid);
/// Unregister all services registered by this client. You don't have to
static void unregisterAllServices ();
/** Requests the naming service to choose a port for the service
* \return An empty port number
*/
static uint16 queryServicePort ();
/** Returns true and the address of the specified service if it is found, otherwise returns false
* \param name [in] Short name of the service to find
* \param addr [out] Address of the service
* \param validitytime [out] After this number of seconds are elapsed, another lookup will be necessary
* before sending a message to the service
* \return true if all worked fine
*/
static bool lookup (const std::string &name, CInetAddress &addr);
/// Same as lookup(const string&, CInetAddress&, uint16&)
static bool lookup (TServiceId sid, CInetAddress &addr);
/** Tells the Naming Service the specified address does not respond for the specified service,
* and returns true and another address for the service if available, otherwise returns false
* \param name [in] Short name of the service to find
* \param addr [in/out] In: Address of the service that does not respond. Out: Alternative address
* \param validitytime [out] After this number of seconds are elapsed, another lookup will be necessary
* before sending a message to the service
* \return true if all worked fine.
*/
static bool lookupAlternate (const std::string& name, CInetAddress& addr);
/**
* Returns all services corresponding to the specified short name.
* Ex: lookupAll ("AS", addresses);
*/
static void lookupAll (const std::string &name, std::vector<CInetAddress> &addresses);
/**
* Returns all registered services.
*/
static const std::list<CServiceEntry> &getRegisteredServices() { return RegisteredServices; }
/** Obtains a socket connected to a service providing the service \e name.
* In case of failure to connect, the method informs the Naming Service and tries to get another service.
* \param name [in] Short name of the service to find and connected
* \param sock [out] The connected socket.
* \param validitytime [out] After this number of seconds are elapsed, another lookup will be necessary
* before sending a message to the service.
* \return false if the service was not found.
*/
static bool lookupAndConnect (const std::string &name, CCallbackClient &sock);
/// Call it evenly
static void update ();
/// You can link a callback if you want to know when a new service is registered (NULL to disable callback)
static void setRegistrationBroadcastCallback (TBroadcastCallback cb);
/// You can link a callback if you want to know when a new service is unregistered (NULL to disable callback)
static void setUnregistrationBroadcastCallback (TBroadcastCallback cb);
static void displayRegisteredServices (NLMISC::CLog *log = NLMISC::DebugLog)
{
RegisteredServicesMutex.enter ();
log->displayNL ("Display the %d registered services :", RegisteredServices.size());
for (std::list<CServiceEntry>::iterator it = RegisteredServices.begin(); it != RegisteredServices.end (); it++)
{
log->displayNL (" > %s-%hu %d addr", (*it).Name.c_str(), (*it).SId.get(), (*it).Addr.size());
for(uint i = 0; i < (*it).Addr.size(); i++)
log->displayNL (" '%s'", (*it).Addr[i].asString().c_str());
}
log->displayNL ("End of the list");
RegisteredServicesMutex.leave ();
}
private:
static CCallbackClient *_Connection;
// The service Id of this service
static TServiceId _MySId;
/// Type of map of registered services
typedef std::map<TServiceId, std::string> TRegServices;
// this container contains the server that *this* service have registered (often, there's only one)
static TRegServices _RegisteredServices;
/// Constructor
CNamingClient() {}
/// Destructor
~CNamingClient() {}
static void doReceiveLookupAnswer (const std::string &name, std::vector<CInetAddress> &addrs);
// This container contains the list of other registered services on the shard
static std::list<CServiceEntry> RegisteredServices;
static NLMISC::CMutex RegisteredServicesMutex;
static void find (const std::string &name, std::vector<CInetAddress> &addrs)
{
RegisteredServicesMutex.enter ();
for (std::list<CServiceEntry>::iterator it = RegisteredServices.begin(); it != RegisteredServices.end (); it++)
if (name == (*it).Name)
addrs.push_back ((*it).Addr[0]);
RegisteredServicesMutex.leave ();
}
static void find (TServiceId sid, std::vector<CInetAddress> &addrs)
{
RegisteredServicesMutex.enter ();
for (std::list<CServiceEntry>::iterator it = RegisteredServices.begin(); it != RegisteredServices.end (); it++)
if (sid == (*it).SId)
addrs.push_back ((*it).Addr[0]);
RegisteredServicesMutex.leave ();
}
friend void cbRegisterBroadcast (CMessage &msgin, TSockId from, CCallbackNetBase &netbase);
friend void cbUnregisterBroadcast (CMessage &msgin, TSockId from, CCallbackNetBase &netbase);
};
} // NLNET
#endif // NL_NAMING_CLIENT_H
/* End of naming_client.h */