/*
* JsonRpc-Cpp - JSON-RPC implementation.
* Copyright (C) 2008 Sebastien Vincent <sebastien.vincent@cppextrem.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 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
/**
* \file jsonrpc_tcpserver.cpp
* \brief JSON-RPC TCP server.
* \author Sebastien Vincent
*/
#include "jsonrpc_tcpserver.h"
#include <stdexcept>
#include "netstring.h"
namespace Json
{
namespace Rpc
{
TcpServer::TcpServer(const std::string& address, uint16_t port) : Server(address, port)
{
m_protocol = TCP;
}
TcpServer::~TcpServer()
{
for(std::list<int>::iterator it = m_clients.begin() ; it != m_clients.end() ; it++)
{
::close((*it));
}
m_clients.erase(m_clients.begin(), m_clients.end());
}
ssize_t TcpServer::Recv(int fd)
{
Json::Value response;
ssize_t nb = -1;
char buf[1500];
nb = recv(fd, buf, sizeof(buf), 0);
/* give the message to JsonHandler */
if(nb > 0)
{
std::string msg = std::string(buf, nb);
if(GetEncapsulatedFormat() == Json::Rpc::NETSTRING)
{
try
{
msg = netstring::decode(msg);
}
catch(std::runtime_error e)
{
/* error parsing NetString */
return -1;
}
}
m_jsonHandler.Process(buf, response);
if(response != Json::Value::null)
{
std::string rep = m_jsonHandler.GetString(response);
/* encoding */
if(GetEncapsulatedFormat() == Json::Rpc::NETSTRING)
{
rep = netstring::encode(rep);
}
if(send(fd, rep.c_str(), rep.length(), 0) == -1)
{
/* error */
}
}
}
else
{
m_purge.push_back(fd);
}
return nb;
}
void TcpServer::WaitMessage(uint32_t ms)
{
fd_set fdsr;
struct timeval tv;
int max_sock = m_sock;
tv.tv_sec = ms / 1000;
tv.tv_usec = (ms % 1000 ) / 1000;
FD_ZERO(&fdsr);
FD_SET(m_sock, &fdsr);
for(std::list<int>::iterator it = m_clients.begin() ; it != m_clients.end() ; it++)
{
FD_SET((*it), &fdsr);
if((*it) > max_sock)
{
max_sock = (*it);
}
}
max_sock++;
if(select(max_sock, &fdsr, NULL, NULL, ms ? &tv : NULL) > 0)
{
if(FD_ISSET(m_sock, &fdsr))
{
Accept();
}
for(std::list<int>::iterator it = m_clients.begin() ; it != m_clients.end() ; it++)
{
if(FD_ISSET((*it), &fdsr))
{
Recv((*it));
}
}
/* purge disconnected list */
m_purge.erase(m_purge.begin(), m_purge.end());
}
else
{
/* error */
}
}
bool TcpServer::Listen()
{
if(m_sock == -1)
{
return false;
}
if(listen(m_sock, 5) == -1)
{
return false;
}
return true;
}
bool TcpServer::Accept()
{
int client = -1;
socklen_t addrlen = sizeof(struct sockaddr_storage);
if(m_sock == -1)
{
return false;
}
client = accept(m_sock, 0, &addrlen);
if(client == -1)
{
return false;
}
m_clients.push_back(client);
return true;
}
} /* namespace Rpc */
} /* namespace Json */