Accueil > > > ENCAPSULATION D'UNE PARTIE DE L'API SOCKET PORTABLE
ENCAPSULATION D'UNE PARTIE DE L'API SOCKET PORTABLE
Information sur la source
Description
voila, ceci est une tentative d'encapsulation dans des classes d'une partie des api sockets. ce code nécéssite une partie de boost : boost::noncopyable (www.boost.org) qui empeche la copie d'une instance d'un objet.
Source
- #ifndef sock_h
- #define sock_h
-
- #include <boost/utility.hpp>
- #include <string>
-
- #ifdef WIN32
- #include <winsock.h>
- #define wsastart(); {WSADATA WsaData; WSAStartup(MAKEWORD(1,1), &WsaData);}
- #define wsaclean(); WSACleanup();
- #pragma comment(lib, "ws2_32.lib")
- #else
- #define wsastart();
- #define wsaclean();
- #include <unistd.h>
- #include <sys/socket.h>
- #include <netinet/in.h>
- #include <netdb.h>
- #include <sys/types.h>
- #define SOCKET_ERROR (-1) /* défini dans winsock.h */
- typedef int SOCKET;
- #define INVALID_SOCKET (SOCKET)(~0) /* défini dans winsock.h */
- #define closesocket close
- #endif
-
- struct socket_error : std::exception {};
-
- struct base_sock : boost::noncopyable
- {
- struct address
- {
- protected:
- std::string _name;
- unsigned short _port;
-
- public:
- address(const address & other)
- : _name(other._name), _port(other._port)
- {
- }
-
- address(const std::string & __name, const unsigned short & __port)
- : _name(__name), _port(__port)
- {
- }
-
- address(const std::string & __name)
- : _name(__name), _port(0)
- {
- }
-
- address(const unsigned short & __port)
- : _port(__port)
- {
- }
-
- address()
- : _port(0)
- {
- }
-
- address & operator = (const address & other)
- {
- _name = other._name;
- _port = other._port;
- return *this;
- }
-
- void clear()
- {
- _name = "";
- _port = 0;
- }
-
- bool is_empty() const
- {
- return _name == "" && _port == 0;
- }
-
- const unsigned short & port() const
- {
- return _port;
- }
-
- const std::string & name() const
- {
- return _name;
- }
-
- unsigned short & set_port(const unsigned short & port)
- {
- return _port = port;
- }
-
- std::string & set_name(const std::string & name)
- {
- return _name = name;
- }
-
- bool operator == (const address & other) const
- {
- return (_name == other._name) && (_port == other._port);
- }
- };
-
- protected:
- address sock_address;
- SOCKET s;
-
- private:
- void create()
- {
- close();
- s = socket(AF_INET, SOCK_STREAM, 0);
- if(s == INVALID_SOCKET) throw socket_error();
- }
-
- public:
-
- base_sock()
- : s(INVALID_SOCKET)
- {
- wsastart();
- }
-
- virtual ~base_sock()
- {
- close();
- wsaclean();
- }
-
- bool good() const
- {
- return s != INVALID_SOCKET;
- }
-
- virtual void write(const std::string & data) const
- {
- if(!good()) throw socket_error();
- if(::send(s, data.data(), (int)data.size(), 0) == SOCKET_ERROR) throw socket_error();
- }
-
- virtual std::string read() const
- {
- if(!good()) throw socket_error();
-
- // on crée un buf avec la taille max de data qu'il peut recevoir
- unsigned long cbdata;
- int cbopt = sizeof(cbdata);
- if(getsockopt(s, SOL_SOCKET, SO_RCVBUF, (char*)&(cbdata), &cbopt) == SOCKET_ERROR) throw socket_error();
- char* data = new char[cbdata];
-
- // on recoit
- int len;
- if((len = recv(s, data, cbdata, 0)) == SOCKET_ERROR)
- {
- delete[] data;
- throw socket_error();
- } else {
- std::string ret(data, len);
- delete[] data;
- return ret;
- }
- }
-
- void close()
- {
- if(!good()) return;
- ::closesocket(s);
- sock_address.clear();
- s = INVALID_SOCKET;
- }
-
- void listen(const address & addr)
- {
- create();
-
- sockaddr_in sain;
- sain.sin_family = AF_INET;
- sain.sin_port = ::htons(addr.port());
-
- if(addr.name() == "")
- sain.sin_addr.s_addr = INADDR_ANY;
- else
- {
- hostent* host = ::gethostbyname(addr.name().c_str());
- if(host == NULL) throw socket_error();
- memcpy((void*)&sain.sin_addr, (void*)host->h_addr, 4);
- }
-
- if(::bind(s, (struct sockaddr *) &sain, sizeof(struct sockaddr_in)) == SOCKET_ERROR) throw socket_error();
- sock_address = addr;
-
- if(::listen(s, SOMAXCONN) == SOCKET_ERROR) throw socket_error();
- }
-
- void accept(const base_sock & listen_sock)
- {
- if(!listen_sock.good()) throw socket_error();
-
- create();
-
- sockaddr_in addr;
- int len = sizeof(sockaddr_in);
- s = ::accept(listen_sock.s, (sockaddr*)&addr, &len);
- if(s == INVALID_SOCKET) throw socket_error();
-
- sock_address.set_port(addr.sin_port);
- sock_address.set_name(inet_ntoa(addr.sin_addr));
- }
-
- void connect(const address & addr)
- {
- create();
-
- sockaddr_in sain;
- int len = sizeof(sockaddr_in);
- sain.sin_family = AF_INET;
- sain.sin_port = ::htons(addr.port());
-
- hostent* host = ::gethostbyname(addr.name().c_str());
- if(host == NULL) throw socket_error();
- memcpy((void*)&sain.sin_addr, (void*)host->h_addr, 4);
-
- if(::connect(s, (struct sockaddr *) &sain, sizeof(struct sockaddr_in)) == SOCKET_ERROR) throw socket_error();
- sock_address = addr;
- }
-
- const address & get_address()
- {
- return sock_address;
- }
-
- class sock_set : boost::noncopyable
- {
- // initialisation d'une instance nulle
- sock_set(const bool & is_null)
- : bnull(is_null)
- {
- }
-
- protected:
- fd_set fds;
- bool bnull;
-
- public:
-
- // renvoit un sock_set nul
- static sock_set & null()
- {
- static sock_set ret(true);
- return ret;
- }
-
- sock_set()
- : bnull(false)
- {
- FD_ZERO(&fds);
- }
-
- inline void add(const base_sock & _s)
- {
- if(bnull) throw socket_error();
- if(!_s.good()) throw socket_error();
- FD_SET(_s.s, &fds);
- }
-
- inline void del(const base_sock & _s)
- {
- if(bnull) throw socket_error();
- if(!_s.good()) throw socket_error();
- FD_CLR(_s.s, &fds);
- }
-
- inline bool is_set(const base_sock & _s) const
- {
- if(bnull) throw socket_error();
- if(!_s.good()) throw socket_error();
- return FD_ISSET(_s.s, &fds) != 0;
- }
-
- static int select(sock_set & read_set, sock_set & write_set, sock_set & error_set, const timeval & timeout)
- {
- if(read_set.bnull && write_set.bnull && error_set.bnull) throw socket_error();
-
- fd_set * r = read_set.bnull ? NULL : &(read_set.fds);
- fd_set * w = write_set.bnull ? NULL : &(write_set.fds);
- fd_set * e = error_set.bnull ? NULL : &(error_set.fds);
-
- int ret = ::select(0, r, w, e, &timeout);
- if(ret == SOCKET_ERROR) throw socket_error();
- return ret;
- }
-
- };
- };
-
- #endif
#ifndef sock_h
#define sock_h
#include <boost/utility.hpp>
#include <string>
#ifdef WIN32
#include <winsock.h>
#define wsastart(); {WSADATA WsaData; WSAStartup(MAKEWORD(1,1), &WsaData);}
#define wsaclean(); WSACleanup();
#pragma comment(lib, "ws2_32.lib")
#else
#define wsastart();
#define wsaclean();
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h>
#include <sys/types.h>
#define SOCKET_ERROR (-1) /* défini dans winsock.h */
typedef int SOCKET;
#define INVALID_SOCKET (SOCKET)(~0) /* défini dans winsock.h */
#define closesocket close
#endif
struct socket_error : std::exception {};
struct base_sock : boost::noncopyable
{
struct address
{
protected:
std::string _name;
unsigned short _port;
public:
address(const address & other)
: _name(other._name), _port(other._port)
{
}
address(const std::string & __name, const unsigned short & __port)
: _name(__name), _port(__port)
{
}
address(const std::string & __name)
: _name(__name), _port(0)
{
}
address(const unsigned short & __port)
: _port(__port)
{
}
address()
: _port(0)
{
}
address & operator = (const address & other)
{
_name = other._name;
_port = other._port;
return *this;
}
void clear()
{
_name = "";
_port = 0;
}
bool is_empty() const
{
return _name == "" && _port == 0;
}
const unsigned short & port() const
{
return _port;
}
const std::string & name() const
{
return _name;
}
unsigned short & set_port(const unsigned short & port)
{
return _port = port;
}
std::string & set_name(const std::string & name)
{
return _name = name;
}
bool operator == (const address & other) const
{
return (_name == other._name) && (_port == other._port);
}
};
protected:
address sock_address;
SOCKET s;
private:
void create()
{
close();
s = socket(AF_INET, SOCK_STREAM, 0);
if(s == INVALID_SOCKET) throw socket_error();
}
public:
base_sock()
: s(INVALID_SOCKET)
{
wsastart();
}
virtual ~base_sock()
{
close();
wsaclean();
}
bool good() const
{
return s != INVALID_SOCKET;
}
virtual void write(const std::string & data) const
{
if(!good()) throw socket_error();
if(::send(s, data.data(), (int)data.size(), 0) == SOCKET_ERROR) throw socket_error();
}
virtual std::string read() const
{
if(!good()) throw socket_error();
// on crée un buf avec la taille max de data qu'il peut recevoir
unsigned long cbdata;
int cbopt = sizeof(cbdata);
if(getsockopt(s, SOL_SOCKET, SO_RCVBUF, (char*)&(cbdata), &cbopt) == SOCKET_ERROR) throw socket_error();
char* data = new char[cbdata];
// on recoit
int len;
if((len = recv(s, data, cbdata, 0)) == SOCKET_ERROR)
{
delete[] data;
throw socket_error();
} else {
std::string ret(data, len);
delete[] data;
return ret;
}
}
void close()
{
if(!good()) return;
::closesocket(s);
sock_address.clear();
s = INVALID_SOCKET;
}
void listen(const address & addr)
{
create();
sockaddr_in sain;
sain.sin_family = AF_INET;
sain.sin_port = ::htons(addr.port());
if(addr.name() == "")
sain.sin_addr.s_addr = INADDR_ANY;
else
{
hostent* host = ::gethostbyname(addr.name().c_str());
if(host == NULL) throw socket_error();
memcpy((void*)&sain.sin_addr, (void*)host->h_addr, 4);
}
if(::bind(s, (struct sockaddr *) &sain, sizeof(struct sockaddr_in)) == SOCKET_ERROR) throw socket_error();
sock_address = addr;
if(::listen(s, SOMAXCONN) == SOCKET_ERROR) throw socket_error();
}
void accept(const base_sock & listen_sock)
{
if(!listen_sock.good()) throw socket_error();
create();
sockaddr_in addr;
int len = sizeof(sockaddr_in);
s = ::accept(listen_sock.s, (sockaddr*)&addr, &len);
if(s == INVALID_SOCKET) throw socket_error();
sock_address.set_port(addr.sin_port);
sock_address.set_name(inet_ntoa(addr.sin_addr));
}
void connect(const address & addr)
{
create();
sockaddr_in sain;
int len = sizeof(sockaddr_in);
sain.sin_family = AF_INET;
sain.sin_port = ::htons(addr.port());
hostent* host = ::gethostbyname(addr.name().c_str());
if(host == NULL) throw socket_error();
memcpy((void*)&sain.sin_addr, (void*)host->h_addr, 4);
if(::connect(s, (struct sockaddr *) &sain, sizeof(struct sockaddr_in)) == SOCKET_ERROR) throw socket_error();
sock_address = addr;
}
const address & get_address()
{
return sock_address;
}
class sock_set : boost::noncopyable
{
// initialisation d'une instance nulle
sock_set(const bool & is_null)
: bnull(is_null)
{
}
protected:
fd_set fds;
bool bnull;
public:
// renvoit un sock_set nul
static sock_set & null()
{
static sock_set ret(true);
return ret;
}
sock_set()
: bnull(false)
{
FD_ZERO(&fds);
}
inline void add(const base_sock & _s)
{
if(bnull) throw socket_error();
if(!_s.good()) throw socket_error();
FD_SET(_s.s, &fds);
}
inline void del(const base_sock & _s)
{
if(bnull) throw socket_error();
if(!_s.good()) throw socket_error();
FD_CLR(_s.s, &fds);
}
inline bool is_set(const base_sock & _s) const
{
if(bnull) throw socket_error();
if(!_s.good()) throw socket_error();
return FD_ISSET(_s.s, &fds) != 0;
}
static int select(sock_set & read_set, sock_set & write_set, sock_set & error_set, const timeval & timeout)
{
if(read_set.bnull && write_set.bnull && error_set.bnull) throw socket_error();
fd_set * r = read_set.bnull ? NULL : &(read_set.fds);
fd_set * w = write_set.bnull ? NULL : &(write_set.fds);
fd_set * e = error_set.bnull ? NULL : &(error_set.fds);
int ret = ::select(0, r, w, e, &timeout);
if(ret == SOCKET_ERROR) throw socket_error();
return ret;
}
};
};
#endif
Conclusion
je l'ai testé sous windows (vc++7) ca a l'air de bien fonctionner. normalement il devrait compiler/fonctionner sous linux, si qq1 pouvait tester (je n'en ai plus sous la main)
j'essaierais rapidement de dériver base_sock pour supporter la cryptographie, et pour respecter l'interface des i/o streams standards du c++, ainsi que d'améliorer la gestion d'erreur.
merci de me signaler tout bug, suggestion ou demande d'explication :)
14/01 : maj
correction de 2-3 petites fautes petites modifications pour faciliter la dérivation suppression de la lourdeur d'utilisation du template select par la mise en place d'un sock_set null
Sources du même auteur
Sources de la même categorie
Commentaires et avis
|
Derniers Blogs
TECHDAYS PARIS 2012 : NOUVELLES TENDANCES DU POSTE DE TRAVAIL - BRING YOUR OWN PCTECHDAYS PARIS 2012 : NOUVELLES TENDANCES DU POSTE DE TRAVAIL - BRING YOUR OWN PC par ROMELARD Fabrice
Speakers: Thierry Rapatout, Antoine Petit et Xavier Trebbia Cette session entre dans le cadre des RDV Décideurs des TechDays 2012, elle est liée à la consumérisation de l'IT et la mise en place du "DeskTop as a Service" dans de plus en ...
Cliquez pour lire la suite de l'article par ROMELARD Fabrice TECHDAYS PARIS 2012 : SYSTEM CENTER SERVICE MANAGER 2012 VUE D'ENSEMBLETECHDAYS PARIS 2012 : SYSTEM CENTER SERVICE MANAGER 2012 VUE D'ENSEMBLE par ROMELARD Fabrice
Speakers: Julien Marechal, Gautier Confiant, Sébastien MEYER La session débute par le positionnement de la solution System Center par rapport aux concepts d'organisation ITIL. Le portail du catalogue de se...
Cliquez pour lire la suite de l'article par ROMELARD Fabrice TECHDAYS PARIS 2012 : PLEINIèRE SECOND JOURTECHDAYS PARIS 2012 : PLEINIèRE SECOND JOUR par ROMELARD Fabrice
Après une première journée dédiée aux développeurs, cette seconde journée est dédiée au monde des entreprises et de ses applications. Ainsi, cette pleinière est dédiée à faire un 360 de l'évolution des applications Business aux demandes ac...
Cliquez pour lire la suite de l'article par ROMELARD Fabrice TECHDAYS PARIS 2012 : RETOUR D'EXPéRIENCE SUR LA MISE EN PLACE D'UN CLOUD PRIVéTECHDAYS PARIS 2012 : RETOUR D'EXPéRIENCE SUR LA MISE EN PLACE D'UN CLOUD PRIVé par ROMELARD Fabrice
Speaker : Guillaume Rochette Cette session est dédiée à fournir le retour sur la mise en place d'un cloud privé (IaaS) par Osiatis pour son compte ou celui de ses clients. Ce projet s'est déroulé sur 4 mois et a permis de faire évoluer...
Cliquez pour lire la suite de l'article par ROMELARD Fabrice TECHDAYS PARIS 2012 : COMMENT SHAREPOINT A SAUVé MES TECHDAYSTECHDAYS PARIS 2012 : COMMENT SHAREPOINT A SAUVé MES TECHDAYS par ROMELARD Fabrice
Speakers : Lionel Limozin et Alain Marty La session commence par une découverte de SharePoint à travers la mise en place d'un environnement SharePoint pour la gestion des Sessions animées par BeWise. Le besoin est très ba...
Cliquez pour lire la suite de l'article par ROMELARD Fabrice
Logiciels
Academy System (17.2.1.0)ACADEMY SYSTEM (17.2.1.0)Logiciel de gestion des établissements.
- élèves/étudiants (inscription, dossier, absence...)
-... Cliquez pour télécharger Academy System Easy-Planning (1.0.0.1)EASY-PLANNING (1.0.0.1)Basé sur les mêmes principes que MyPlanning, Easy-Planning permet de créer des plannings sous la ... Cliquez pour télécharger Easy-Planning COLLECTOR PLUS (3.00B)COLLECTOR PLUS (3.00B)COLLECTOR PLUS version 3.00B est un logiciel utilisant une base de données alimentée par :
- L... Cliquez pour télécharger COLLECTOR PLUS PONAMEDIA PREMIUM - HELLLOOO FLASH DEMO (V7.4)PONAMEDIA PREMIUM - HELLLOOO FLASH DEMO (V7.4)PONAMEDIA TV DEVIENS HELLLOOO FLASH
LA TV SUR VOTRE ORDINATEUR.
Toute une plateforme Multi... Cliquez pour télécharger PONAMEDIA PREMIUM - HELLLOOO FLASH DEMO LettresFaciles 2011 (8.0.0.1)LETTRESFACILES 2011 (8.0.0.1)LettresFaciles est un logiciel facilitant la création et la rédaction de lettres types.
Son inte... Cliquez pour télécharger LettresFaciles 2011
|