2013-09-27 12:14:26 -04:00
|
|
|
|
|
|
|
// HTTPServer.cpp
|
|
|
|
|
|
|
|
// Implements the cHTTPServer class representing a HTTP webserver that uses cListenThread and cSocketThreads for processing
|
|
|
|
|
|
|
|
#include "Globals.h"
|
|
|
|
#include "HTTPServer.h"
|
|
|
|
#include "HTTPMessage.h"
|
2013-09-27 13:34:46 -04:00
|
|
|
#include "HTTPConnection.h"
|
2013-09-28 13:30:25 -04:00
|
|
|
#include "HTTPFormParser.h"
|
2013-09-27 12:14:26 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Disable MSVC warnings:
|
|
|
|
#if defined(_MSC_VER)
|
|
|
|
#pragma warning(push)
|
|
|
|
#pragma warning(disable:4355) // 'this' : used in base member initializer list
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-09-27 15:28:41 -04:00
|
|
|
class cDebugCallbacks :
|
2013-10-04 07:07:57 -04:00
|
|
|
public cHTTPServer::cCallbacks,
|
|
|
|
protected cHTTPFormParser::cCallbacks
|
2013-09-27 15:28:41 -04:00
|
|
|
{
|
|
|
|
virtual void OnRequestBegun(cHTTPConnection & a_Connection, cHTTPRequest & a_Request) override
|
|
|
|
{
|
2013-09-28 13:30:25 -04:00
|
|
|
if (cHTTPFormParser::HasFormData(a_Request))
|
|
|
|
{
|
2013-10-04 07:07:57 -04:00
|
|
|
a_Request.SetUserData(new cHTTPFormParser(a_Request, *this));
|
2013-09-28 13:30:25 -04:00
|
|
|
}
|
2013-09-27 15:28:41 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virtual void OnRequestBody(cHTTPConnection & a_Connection, cHTTPRequest & a_Request, const char * a_Data, int a_Size) override
|
|
|
|
{
|
2013-09-28 13:30:25 -04:00
|
|
|
cHTTPFormParser * FormParser = (cHTTPFormParser *)(a_Request.GetUserData());
|
|
|
|
if (FormParser != NULL)
|
|
|
|
{
|
|
|
|
FormParser->Parse(a_Data, a_Size);
|
|
|
|
}
|
2013-09-27 15:28:41 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virtual void OnRequestFinished(cHTTPConnection & a_Connection, cHTTPRequest & a_Request) override
|
|
|
|
{
|
2013-09-28 13:30:25 -04:00
|
|
|
cHTTPFormParser * FormParser = (cHTTPFormParser *)(a_Request.GetUserData());
|
|
|
|
if (FormParser != NULL)
|
|
|
|
{
|
|
|
|
if (FormParser->Finish())
|
|
|
|
{
|
|
|
|
cHTTPResponse Resp;
|
|
|
|
Resp.SetContentType("text/html");
|
|
|
|
a_Connection.Send(Resp);
|
|
|
|
a_Connection.Send("<html><body><table border=1 cellspacing=0><tr><th>Name</th><th>Value</th></tr>\r\n");
|
|
|
|
for (cHTTPFormParser::iterator itr = FormParser->begin(), end = FormParser->end(); itr != end; ++itr)
|
|
|
|
{
|
|
|
|
a_Connection.Send(Printf("<tr><td valign=\"top\"><pre>%s</pre></td><td valign=\"top\"><pre>%s</pre></td></tr>\r\n", itr->first.c_str(), itr->second.c_str()));
|
|
|
|
} // for itr - FormParser[]
|
|
|
|
a_Connection.Send("</table></body></html>");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parsing failed:
|
|
|
|
cHTTPResponse Resp;
|
|
|
|
Resp.SetContentType("text/plain");
|
|
|
|
a_Connection.Send(Resp);
|
|
|
|
a_Connection.Send("Form parsing failed");
|
2013-09-28 14:06:35 -04:00
|
|
|
return;
|
2013-09-28 13:30:25 -04:00
|
|
|
}
|
|
|
|
|
2013-09-27 15:28:41 -04:00
|
|
|
cHTTPResponse Resp;
|
|
|
|
Resp.SetContentType("text/plain");
|
|
|
|
a_Connection.Send(Resp);
|
|
|
|
a_Connection.Send("Hello, world");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-10-04 07:07:57 -04:00
|
|
|
virtual void OnFileStart(cHTTPFormParser & a_Parser, const AString & a_FileName) override
|
|
|
|
{
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virtual void OnFileData(cHTTPFormParser & a_Parser, const char * a_Data, int a_Size) override
|
|
|
|
{
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virtual void OnFileEnd(cHTTPFormParser & a_Parser) override
|
|
|
|
{
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
|
2013-09-27 15:28:41 -04:00
|
|
|
} g_DebugCallbacks;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// cHTTPServer:
|
|
|
|
|
2013-09-27 12:14:26 -04:00
|
|
|
cHTTPServer::cHTTPServer(void) :
|
|
|
|
m_ListenThreadIPv4(*this, cSocket::IPv4, "WebServer IPv4"),
|
|
|
|
m_ListenThreadIPv6(*this, cSocket::IPv6, "WebServer IPv6"),
|
2013-09-27 15:28:41 -04:00
|
|
|
m_Callbacks(NULL)
|
2013-09-27 12:14:26 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool cHTTPServer::Initialize(cIniFile & a_IniFile)
|
|
|
|
{
|
|
|
|
if (!a_IniFile.GetValueSetB("WebAdmin", "Enabled", false))
|
|
|
|
{
|
|
|
|
// The WebAdmin is disabled
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool HasAnyPort;
|
|
|
|
HasAnyPort = m_ListenThreadIPv4.Initialize(a_IniFile.GetValueSet("WebAdmin", "Port", "8081"));
|
|
|
|
HasAnyPort = m_ListenThreadIPv6.Initialize(a_IniFile.GetValueSet("WebAdmin", "PortsIPv6", "8082, 3300")) || HasAnyPort;
|
|
|
|
if (!HasAnyPort)
|
|
|
|
{
|
|
|
|
LOG("WebAdmin is disabled");
|
|
|
|
return false;
|
|
|
|
}
|
2013-09-27 15:28:41 -04:00
|
|
|
|
|
|
|
// DEBUG:
|
|
|
|
return Start(g_DebugCallbacks);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool cHTTPServer::Start(cCallbacks & a_Callbacks)
|
|
|
|
{
|
|
|
|
m_Callbacks = &a_Callbacks;
|
2013-09-27 12:14:26 -04:00
|
|
|
if (!m_ListenThreadIPv4.Start())
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!m_ListenThreadIPv6.Start())
|
|
|
|
{
|
|
|
|
m_ListenThreadIPv4.Stop();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-09-27 15:28:41 -04:00
|
|
|
void cHTTPServer::Stop(void)
|
|
|
|
{
|
|
|
|
m_ListenThreadIPv4.Stop();
|
|
|
|
m_ListenThreadIPv6.Stop();
|
|
|
|
|
|
|
|
// Drop all current connections:
|
|
|
|
cCSLock Lock(m_CSConnections);
|
|
|
|
for (cHTTPConnections::iterator itr = m_Connections.begin(), end = m_Connections.end(); itr != end; ++itr)
|
|
|
|
{
|
|
|
|
m_SocketThreads.RemoveClient(*itr);
|
|
|
|
} // for itr - m_Connections[]
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-09-27 12:14:26 -04:00
|
|
|
void cHTTPServer::OnConnectionAccepted(cSocket & a_Socket)
|
|
|
|
{
|
|
|
|
cHTTPConnection * Connection = new cHTTPConnection(*this);
|
|
|
|
m_SocketThreads.AddClient(a_Socket, Connection);
|
|
|
|
cCSLock Lock(m_CSConnections);
|
|
|
|
m_Connections.push_back(Connection);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void cHTTPServer::CloseConnection(cHTTPConnection & a_Connection)
|
|
|
|
{
|
|
|
|
m_SocketThreads.RemoveClient(&a_Connection);
|
|
|
|
cCSLock Lock(m_CSConnections);
|
|
|
|
for (cHTTPConnections::iterator itr = m_Connections.begin(), end = m_Connections.end(); itr != end; ++itr)
|
|
|
|
{
|
|
|
|
if (*itr == &a_Connection)
|
|
|
|
{
|
|
|
|
m_Connections.erase(itr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-09-27 14:33:18 -04:00
|
|
|
void cHTTPServer::NotifyConnectionWrite(cHTTPConnection & a_Connection)
|
|
|
|
{
|
|
|
|
m_SocketThreads.NotifyWrite(&a_Connection);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-09-27 12:14:26 -04:00
|
|
|
void cHTTPServer::NewRequest(cHTTPConnection & a_Connection, cHTTPRequest & a_Request)
|
|
|
|
{
|
2013-09-27 15:28:41 -04:00
|
|
|
m_Callbacks->OnRequestBegun(a_Connection, a_Request);
|
2013-09-27 12:14:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-09-27 14:33:18 -04:00
|
|
|
void cHTTPServer::RequestBody(cHTTPConnection & a_Connection, cHTTPRequest & a_Request, const char * a_Data, int a_Size)
|
2013-09-27 12:14:26 -04:00
|
|
|
{
|
2013-09-27 15:28:41 -04:00
|
|
|
m_Callbacks->OnRequestBody(a_Connection, a_Request, a_Data, a_Size);
|
2013-09-27 12:14:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void cHTTPServer::RequestFinished(cHTTPConnection & a_Connection, cHTTPRequest & a_Request)
|
|
|
|
{
|
2013-09-27 15:28:41 -04:00
|
|
|
m_Callbacks->OnRequestFinished(a_Connection, a_Request);
|
|
|
|
a_Connection.AwaitNextRequest();
|
2013-09-27 12:14:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|