mirror of
https://github.com/Icinga/icinga2.git
synced 2026-02-03 20:40:17 -05:00
248 lines
4.3 KiB
C++
248 lines
4.3 KiB
C++
#include "i2-base.h"
|
|
|
|
using namespace icinga;
|
|
|
|
/**
|
|
* Socket::Sockets
|
|
*
|
|
* A collection of weak pointers to Socket objects which have been
|
|
* registered with the socket sub-system.
|
|
*/
|
|
Socket::CollectionType Socket::Sockets;
|
|
|
|
/**
|
|
* Socket
|
|
*
|
|
* Constructor for the Socket class.
|
|
*/
|
|
Socket::Socket(void)
|
|
{
|
|
m_FD = INVALID_SOCKET;
|
|
}
|
|
|
|
/**
|
|
* ~Socket
|
|
*
|
|
* Destructor for the Socket class.
|
|
*/
|
|
Socket::~Socket(void)
|
|
{
|
|
CloseInternal(true);
|
|
}
|
|
|
|
/**
|
|
* Start
|
|
*
|
|
* Registers the socket and starts handling events for it.
|
|
*/
|
|
void Socket::Start(void)
|
|
{
|
|
assert(m_FD != INVALID_SOCKET);
|
|
|
|
OnException += bind_weak(&Socket::ExceptionEventHandler, shared_from_this());
|
|
|
|
Sockets.push_back(static_pointer_cast<Socket>(shared_from_this()));
|
|
}
|
|
|
|
/**
|
|
* Stop
|
|
*
|
|
* Unregisters the sockets and stops handling events for it.
|
|
*/
|
|
void Socket::Stop(void)
|
|
{
|
|
Sockets.remove_if(weak_ptr_eq_raw<Socket>(this));
|
|
}
|
|
|
|
/**
|
|
* SetFD
|
|
*
|
|
* Sets the file descriptor for this socket object.
|
|
*
|
|
* @param fd The file descriptor.
|
|
*/
|
|
void Socket::SetFD(SOCKET fd)
|
|
{
|
|
unsigned long lTrue = 1;
|
|
|
|
if (fd != INVALID_SOCKET)
|
|
ioctlsocket(fd, FIONBIO, &lTrue);
|
|
|
|
m_FD = fd;
|
|
}
|
|
|
|
/**
|
|
* GetFD
|
|
*
|
|
* Retrieves the file descriptor for this socket object.
|
|
*
|
|
* @returns The file descriptor.
|
|
*/
|
|
SOCKET Socket::GetFD(void) const
|
|
{
|
|
return m_FD;
|
|
}
|
|
|
|
/**
|
|
* Close
|
|
*
|
|
* Closes the socket.
|
|
*/
|
|
void Socket::Close(void)
|
|
{
|
|
CloseInternal(false);
|
|
}
|
|
|
|
/**
|
|
* CloseInternal
|
|
*
|
|
* Closes the socket.
|
|
*
|
|
* @param from_dtor Whether this method was called from the destructor.
|
|
*/
|
|
void Socket::CloseInternal(bool from_dtor)
|
|
{
|
|
if (m_FD == INVALID_SOCKET)
|
|
return;
|
|
|
|
closesocket(m_FD);
|
|
m_FD = INVALID_SOCKET;
|
|
|
|
/* nobody can possibly have a valid event subscription when the
|
|
destructor has been called */
|
|
if (!from_dtor) {
|
|
Stop();
|
|
|
|
EventArgs ea;
|
|
ea.Source = shared_from_this();
|
|
OnClosed(ea);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* HandleSocketError
|
|
*
|
|
* Handles a socket error by calling the OnError event.
|
|
*/
|
|
void Socket::HandleSocketError(void)
|
|
{
|
|
int opt;
|
|
socklen_t optlen = sizeof(opt);
|
|
|
|
int rc = getsockopt(GetFD(), SOL_SOCKET, SO_ERROR, (char *)&opt, &optlen);
|
|
|
|
if (rc >= 0 && opt != 0) {
|
|
SocketErrorEventArgs sea;
|
|
sea.Code = opt;
|
|
#ifdef _WIN32
|
|
sea.Message = Win32Exception::FormatErrorCode(sea.Code);
|
|
#else /* _WIN32 */
|
|
sea.Message = PosixException::FormatErrorCode(sea.Code);
|
|
#endif /* _WIN32 */
|
|
OnError(sea);
|
|
}
|
|
|
|
Close();
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* ExceptionEventHandler
|
|
*
|
|
* Processes errors that have occured for the socket.
|
|
*
|
|
* @param ea Event arguments for the socket error.
|
|
* @returns 0
|
|
*/
|
|
int Socket::ExceptionEventHandler(const EventArgs& ea)
|
|
{
|
|
HandleSocketError();
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* WantsToRead
|
|
*
|
|
* Checks whether data should be read for this socket object.
|
|
*
|
|
* @returns true if the socket should be registered for reading, false otherwise.
|
|
*/
|
|
bool Socket::WantsToRead(void) const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* WantsToWrite
|
|
*
|
|
* Checks whether data should be written for this socket object.
|
|
*
|
|
* @returns true if the socket should be registered for writing, false otherwise.
|
|
*/
|
|
bool Socket::WantsToWrite(void) const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* GetAddressFromSockaddr
|
|
*
|
|
* Formats a sockaddr in a human-readable way.
|
|
*
|
|
* @returns A string describing the sockaddr.
|
|
*/
|
|
string Socket::GetAddressFromSockaddr(sockaddr *address, socklen_t len)
|
|
{
|
|
char host[NI_MAXHOST];
|
|
char service[NI_MAXSERV];
|
|
|
|
if (getnameinfo(address, len, host, sizeof(host), service, sizeof(service), NI_NUMERICHOST | NI_NUMERICSERV) < 0)
|
|
throw InvalidArgumentException(); /* TODO: throw proper exception */
|
|
|
|
stringstream s;
|
|
s << "[" << host << "]:" << service;
|
|
return s.str();
|
|
}
|
|
|
|
/**
|
|
* GetClientAddress
|
|
*
|
|
* Returns a string describing the local address of the socket.
|
|
*
|
|
* @returns A string describing the local address.
|
|
*/
|
|
string Socket::GetClientAddress(void)
|
|
{
|
|
sockaddr_storage sin;
|
|
socklen_t len = sizeof(sin);
|
|
|
|
if (getsockname(GetFD(), (sockaddr *)&sin, &len) < 0) {
|
|
HandleSocketError();
|
|
|
|
return string();
|
|
}
|
|
|
|
return GetAddressFromSockaddr((sockaddr *)&sin, len);
|
|
}
|
|
|
|
/**
|
|
* GetPeerAddress
|
|
*
|
|
* Returns a string describing the peer address of the socket.
|
|
*
|
|
* @returns A string describing the peer address.
|
|
*/
|
|
string Socket::GetPeerAddress(void)
|
|
{
|
|
sockaddr_storage sin;
|
|
socklen_t len = sizeof(sin);
|
|
|
|
if (getpeername(GetFD(), (sockaddr *)&sin, &len) < 0) {
|
|
HandleSocketError();
|
|
|
|
return string();
|
|
}
|
|
|
|
return GetAddressFromSockaddr((sockaddr *)&sin, len);
|
|
}
|