Browse Source

Started rewrite of HTTPServer.

pull/271/head
EinMByte 9 years ago
parent
commit
e7350a3af4
  1. 1103
      client/HTTPServer.cpp
  2. 176
      client/HTTPServer.h
  3. 12
      core/util/HTTP.cpp
  4. 10
      core/util/HTTP.h
  5. 12
      tests/Utility.cpp
  6. 12
      webui/index.html

1103
client/HTTPServer.cpp

File diff suppressed because it is too large Load Diff

176
client/HTTPServer.h

@ -6,119 +6,71 @@
#include <memory> #include <memory>
#include <boost/asio.hpp> #include <boost/asio.hpp>
#include <boost/array.hpp> #include <boost/array.hpp>
#include "util/HTTP.h"
namespace i2p namespace i2p {
{ namespace util {
namespace util
{ const size_t HTTP_CONNECTION_BUFFER_SIZE = 8192;
const size_t HTTP_CONNECTION_BUFFER_SIZE = 8192; const int HTTP_DESTINATION_REQUEST_TIMEOUT = 10; // in seconds
const int HTTP_DESTINATION_REQUEST_TIMEOUT = 10; // in seconds
class HTTPConnection: public std::enable_shared_from_this<HTTPConnection> class HTTPConnection: public std::enable_shared_from_this<HTTPConnection> {
{ public:
protected:
HTTPConnection(boost::asio::ip::tcp::socket * socket)
struct header : m_Socket(socket), m_Timer(socket->get_io_service()),
{ m_BufferLen(0) {};
std::string name; ~HTTPConnection() { delete m_Socket; }
std::string value; void Receive();
};
private:
struct request
{ void Terminate();
std::string method; void HandleReceive(const boost::system::error_code& ecode, std::size_t bytes_transferred);
std::string uri; void RunRequest();
std::string host; void HandleWriteReply(const boost::system::error_code& ecode);
int port; void SendReply();
int http_version_major;
int http_version_minor; void HandleRequest();
std::vector<header> headers; void ExtractParams(const std::string& str, std::map<std::string, std::string>& params);
};
bool isAllowed(const std::string& address);
struct reply private:
{ boost::asio::ip::tcp::socket* m_Socket;
std::vector<header> headers; boost::asio::deadline_timer m_Timer;
std::string content; char m_Buffer[HTTP_CONNECTION_BUFFER_SIZE + 1];
char m_StreamBuffer[HTTP_CONNECTION_BUFFER_SIZE + 1];
std::vector<boost::asio::const_buffer> to_buffers (int status); size_t m_BufferLen;
}; util::http::Request m_Request;
util::http::Response m_Reply;
public: };
HTTPConnection (boost::asio::ip::tcp::socket * socket): class HTTPServer {
m_Socket (socket), m_Timer (socket->get_io_service ()), public:
m_BufferLen (0) {};
~HTTPConnection() { delete m_Socket; } HTTPServer(const std::string& address, int port);
void Receive (); virtual ~HTTPServer();
private: void Start();
void Stop();
void Terminate ();
void HandleReceive (const boost::system::error_code& ecode, std::size_t bytes_transferred); private:
void HandleWriteReply(const boost::system::error_code& ecode);
void SendReply (const std::string& content, int status = 200); void Run();
void Accept();
void HandleRequest (const std::string& address); void HandleAccept(const boost::system::error_code& ecode);
void HandleCommand (const std::string& command, std::stringstream& s);
void ShowTransports (std::stringstream& s); private:
void ShowTunnels (std::stringstream& s);
void ShowTransitTunnels (std::stringstream& s); std::thread * m_Thread;
void ShowLocalDestinations (std::stringstream& s); boost::asio::io_service m_Service;
void ShowLocalDestination (const std::string& b32, std::stringstream& s); boost::asio::io_service::work m_Work;
void ShowSAMSessions (std::stringstream& s); boost::asio::ip::tcp::acceptor m_Acceptor;
void ShowSAMSession (const std::string& id, std::stringstream& s); boost::asio::ip::tcp::socket * m_NewSocket;
void StartAcceptingTunnels (std::stringstream& s);
void StopAcceptingTunnels (std::stringstream& s); protected:
void FillContent (std::stringstream& s); virtual void CreateConnection(boost::asio::ip::tcp::socket * m_NewSocket);
std::string ExtractAddress (); };
void ExtractParams (const std::string& str, std::map<std::string, std::string>& params);
protected:
boost::asio::ip::tcp::socket * m_Socket;
boost::asio::deadline_timer m_Timer;
char m_Buffer[HTTP_CONNECTION_BUFFER_SIZE + 1], m_StreamBuffer[HTTP_CONNECTION_BUFFER_SIZE + 1];
size_t m_BufferLen;
request m_Request;
reply m_Reply;
protected:
virtual void RunRequest ();
public:
static const std::string itoopieImage;
static const std::string itoopieFavicon;
};
class HTTPServer
{
public:
HTTPServer (const std::string& address, int port);
virtual ~HTTPServer ();
void Start ();
void Stop ();
private:
void Run ();
void Accept ();
void HandleAccept(const boost::system::error_code& ecode);
private:
std::thread * m_Thread;
boost::asio::io_service m_Service;
boost::asio::io_service::work m_Work;
boost::asio::ip::tcp::acceptor m_Acceptor;
boost::asio::ip::tcp::socket * m_NewSocket;
protected:
virtual void CreateConnection(boost::asio::ip::tcp::socket * m_NewSocket);
};
} }
} }

12
core/util/HTTP.cpp

@ -55,8 +55,8 @@ std::string Request::getHeader(const std::string& name) const
return headers.at(name); return headers.at(name);
} }
Response::Response(int status) Response::Response(int status, const std::string& content)
: status(status), headers() : status(status), content(content), headers()
{ {
} }
@ -72,7 +72,7 @@ std::string Response::toString() const
ss << "HTTP/1.1 " << status << ' ' << getStatusMessage() << "\r\n"; ss << "HTTP/1.1 " << status << ' ' << getStatusMessage() << "\r\n";
for(auto& pair : headers) for(auto& pair : headers)
ss << pair.first << ": " << pair.second << "\r\n"; ss << pair.first << ": " << pair.second << "\r\n";
ss << "\r\n"; ss << "\r\n" << content;
return ss.str(); return ss.str();
} }
@ -99,6 +99,12 @@ std::string Response::getStatusMessage() const
return std::string(); return std::string();
} }
} }
void Response::setContentLength()
{
setHeader("Content-Length", std::to_string(content.size()));
}
} }
} }
} }

10
core/util/HTTP.h

@ -14,9 +14,9 @@ class Request {
void parseHeaderLine(const std::string& line); void parseHeaderLine(const std::string& line);
public: public:
Request(const std::string& data); Request() = default;
Request(); Request(const std::string& data);
std::string getMethod() const; std::string getMethod() const;
@ -41,8 +41,9 @@ private:
class Response { class Response {
public: public:
Response() = default;
Response(int status); Response(int status, const std::string& content = "");
/** /**
* @note overrides existing header values with the same name * @note overrides existing header values with the same name
@ -57,8 +58,11 @@ public:
*/ */
std::string getStatusMessage() const; std::string getStatusMessage() const;
void setContentLength();
private: private:
int status; int status;
std::string content;
std::map<std::string, std::string> headers; std::map<std::string, std::string> headers;
}; };

12
tests/Utility.cpp

@ -125,6 +125,7 @@ BOOST_AUTO_TEST_CASE(HTTPResponseStatusMessage)
BOOST_CHECK_EQUAL(Response(502).getStatusMessage(), "Not Implemented"); BOOST_CHECK_EQUAL(Response(502).getStatusMessage(), "Not Implemented");
BOOST_CHECK_EQUAL(Response(504).getStatusMessage(), "Gateway Timeout"); BOOST_CHECK_EQUAL(Response(504).getStatusMessage(), "Gateway Timeout");
} }
BOOST_AUTO_TEST_CASE(WriteHTTPResponse) BOOST_AUTO_TEST_CASE(WriteHTTPResponse)
{ {
Response rsp(200); Response rsp(200);
@ -136,5 +137,16 @@ BOOST_AUTO_TEST_CASE(WriteHTTPResponse)
); );
} }
BOOST_AUTO_TEST_CASE(WriteHTTPResponseWithContent)
{
Response rsp(200, "Test content.");
rsp.setHeader("Connection", "close");
BOOST_CHECK_EQUAL(
rsp.toString(),
"HTTP/1.1 200 OK\r\n"
"Connection: close\r\n\r\n"
"Test content."
);
}
BOOST_AUTO_TEST_SUITE_END() BOOST_AUTO_TEST_SUITE_END()

12
webui/index.html

@ -1 +1,11 @@
Nothing here yet. <!DOCTYPE html>
<html lang="en">
<head>
<title>Purple I2P 0.10.0 Webconsole</title>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
</head>
<body>
<p>Nothing here yet.</p>
</body>
</html>

Loading…
Cancel
Save