/**
\file LinHttpHandler.cpp
Copyright Notice\n
Copyright (C) 2017 Jan Rogall - developer\n
Copyright (C) 2017 Moritz Wirger - developer\n
This file is part of hueplusplus.
hueplusplus is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
hueplusplus 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 Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with hueplusplus. If not, see .
**/
#include "hueplusplus/LinHttpHandler.h"
#include
#include
#include
#include
#include
#include
#include
#include // struct hostent, gethostbyname
#include // struct sockaddr_in, struct sockaddr
#include // printf, sprintf
#include // exit
#include // functions for C style null-terminated strings
#include // socket, connect
#include // read, write, close
namespace hueplusplus
{
class SocketCloser
{
public:
explicit SocketCloser(int sockFd) : s(sockFd) {}
~SocketCloser() { close(s); }
private:
int s;
};
std::string LinHttpHandler::send(const std::string& msg, const std::string& adr, int port) const
{
// create socket
int socketFD = socket(AF_INET, SOCK_STREAM, 0);
SocketCloser closeMySocket(socketFD);
if (socketFD < 0)
{
int errCode = errno;
std::cerr << "LinHttpHandler: Failed to open socket: " << std::strerror(errCode) << "\n";
throw(std::system_error(errCode, std::generic_category(), "LinHttpHandler: Failed to open socket"));
}
// lookup ip address
hostent* server;
server = gethostbyname(adr.c_str());
if (server == NULL)
{
int errCode = errno;
std::cerr << "LinHttpHandler: Failed to find host with address " << adr << ": " << std::strerror(errCode)
<< "\n";
throw(std::system_error(errCode, std::generic_category(), "LinHttpHandler: gethostbyname"));
}
// fill in the structure
sockaddr_in server_addr;
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(port);
memcpy(&server_addr.sin_addr.s_addr, server->h_addr, server->h_length);
// connect the socket
if (connect(socketFD, (struct sockaddr*)&server_addr, sizeof(server_addr)) < 0)
{
int errCode = errno;
std::cerr << "LinHttpHandler: Failed to connect socket: " << std::strerror(errCode) << "\n";
throw(std::system_error(errCode, std::generic_category(), "LinHttpHandler: Failed to connect socket"));
}
// send the request
size_t total = msg.length();
size_t sent = 0;
do
{
ssize_t bytes = write(socketFD, msg.c_str() + sent, total - sent);
if (bytes < 0)
{
int errCode = errno;
std::cerr << "LinHttpHandler: Failed to write message to socket: " << std::strerror(errCode) << "\n";
throw(std::system_error(
errCode, std::generic_category(), "LinHttpHandler: Failed to write message to socket"));
}
else if (bytes == 0)
{
break;
}
else
{
sent += bytes;
}
} while (sent < total);
// receive the response
std::string response;
char buffer[128] = {};
do
{
ssize_t bytes = read(socketFD, buffer, 127);
if (bytes < 0)
{
int errCode = errno;
std::cerr << "LinHttpHandler: Failed to read response from socket: " << std::strerror(errCode) << std::endl;
throw(std::system_error(
errCode, std::generic_category(), "LinHttpHandler: Failed to read response from socket"));
}
else if (bytes == 0)
{
break;
}
else
{
response.append(buffer, bytes);
}
} while (true);
return response;
}
std::vector LinHttpHandler::sendMulticast(
const std::string& msg, const std::string& adr, int port, std::chrono::steady_clock::duration timeout) const
{
hostent* server; // host information
sockaddr_in server_addr; // server address
// fill in the server's address and data
memset((char*)&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(port);
// look up the address of the server given its name
server = gethostbyname(adr.c_str());
if (!server)
{
int errCode = errno;
std::cerr << "LinHttpHandler: sendMulticast: Failed to obtain address of " << msg << ": "
<< std::strerror(errCode) << "\n";
throw(std::system_error(
errCode, std::generic_category(), "LinHttpHandler: sendMulticast: Failed to obtain address of host"));
}
// put the host's address into the server address structure
memcpy((void*)&server_addr.sin_addr, server->h_addr_list[0], server->h_length);
// create the socket
int socketFD = socket(AF_INET, SOCK_DGRAM, 0);
SocketCloser closeMySendSocket(socketFD);
if (socketFD < 0)
{
int errCode = errno;
std::cerr << "LinHttpHandler: sendMulticast: Failed to open socket: " << std::strerror(errCode) << "\n";
throw(std::system_error(
errCode, std::generic_category(), "LinHttpHandler: sendMulticast: Failed to open socket"));
}
// send a message to the server
if (sendto(socketFD, msg.c_str(), strlen(msg.c_str()), 0, (struct sockaddr*)&server_addr, sizeof(server_addr)) < 0)
{
int errCode = errno;
std::cerr << "LinHttpHandler: sendMulticast: Failed to send message: " << std::strerror(errCode) << "\n";
throw(std::system_error(
errCode, std::generic_category(), "LinHttpHandler: sendMulticast: Failed to send message"));
}
std::string response;
char buffer[2048] = {}; // receive buffer
std::chrono::steady_clock::time_point start = std::chrono::steady_clock::now();
while (std::chrono::steady_clock::now() - start < timeout)
{
ssize_t bytesReceived = recv(socketFD, &buffer, 2048, MSG_DONTWAIT);
if (bytesReceived < 0)
{
int errCode = errno;
if (errCode != EAGAIN && errCode != EWOULDBLOCK)
{
std::cerr << "LinHttpHandler: sendMulticast: Failed to read response "
"from socket: "
<< std::strerror(errCode) << "\n";
throw(std::system_error(errCode, std::generic_category(),
"LinHttpHandler: sendMulticast: Failed to read "
"response from socket"));
}
continue;
}
if (bytesReceived)
{
response.append(buffer, bytesReceived);
}
}
// construct return vector
std::vector returnString;
size_t pos = response.find("\r\n\r\n");
size_t prevpos = 0;
while (pos != std::string::npos)
{
returnString.push_back(response.substr(prevpos, pos - prevpos));
pos += 4;
prevpos = pos;
pos = response.find("\r\n\r\n", pos);
}
return returnString;
}
} // namespace hueplusplus