mirror of https://github.com/oxen-io/lokinet
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
360 lines
9.0 KiB
C++
360 lines
9.0 KiB
C++
#include <abyss/server.hpp>
|
|
#include <llarp/time.h>
|
|
#include <sstream>
|
|
#include <unordered_map>
|
|
#include <string>
|
|
#include <llarp/logger.hpp>
|
|
#include <algorithm>
|
|
|
|
namespace abyss
|
|
{
|
|
namespace http
|
|
{
|
|
struct RequestHeader
|
|
{
|
|
typedef std::unordered_multimap< std::string, std::string > Headers_t;
|
|
Headers_t Headers;
|
|
std::string Method;
|
|
std::string Path;
|
|
};
|
|
|
|
struct ConnImpl
|
|
{
|
|
llarp_tcp_conn* _conn;
|
|
IRPCHandler* handler;
|
|
BaseReqHandler* _parent;
|
|
llarp_time_t m_LastActive;
|
|
llarp_time_t m_ReadTimeout;
|
|
bool m_Bad;
|
|
RequestHeader m_Header;
|
|
std::stringstream m_ReadBuf;
|
|
|
|
enum HTTPState
|
|
{
|
|
eReadHTTPMethodLine,
|
|
eReadHTTPHeaders,
|
|
eReadHTTPBody,
|
|
eWriteHTTPStatusLine,
|
|
eWriteHTTPHeaders,
|
|
eWriteHTTPBody,
|
|
eCloseMe
|
|
};
|
|
|
|
HTTPState m_State;
|
|
|
|
ConnImpl(BaseReqHandler* p, llarp_tcp_conn* c, llarp_time_t readtimeout)
|
|
: _conn(c), _parent(p)
|
|
{
|
|
handler = nullptr;
|
|
m_LastActive = llarp_time_now_ms();
|
|
m_ReadTimeout = readtimeout;
|
|
// set up tcp members
|
|
_conn->user = this;
|
|
_conn->read = &ConnImpl::OnRead;
|
|
_conn->tick = &ConnImpl::OnTick;
|
|
_conn->closed = &ConnImpl::OnClosed;
|
|
m_Bad = false;
|
|
m_State = eReadHTTPMethodLine;
|
|
}
|
|
|
|
~ConnImpl()
|
|
{
|
|
}
|
|
|
|
bool
|
|
FeedLine(std::string& line)
|
|
{
|
|
switch(m_State)
|
|
{
|
|
case eReadHTTPMethodLine:
|
|
return ProcessMethodLine(line);
|
|
case eReadHTTPHeaders:
|
|
return ProcessHeaderLine(line);
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool
|
|
ProcessMethodLine(std::string& line)
|
|
{
|
|
// TODO: implement me
|
|
auto idx = line.find_first_of(' ');
|
|
if(idx == std::string::npos)
|
|
return false;
|
|
m_Header.Method = line.substr(0, idx);
|
|
line = line.substr(idx + 1);
|
|
idx = line.find_first_of(' ');
|
|
if(idx == std::string::npos)
|
|
return false;
|
|
m_Header.Path = line.substr(0, idx);
|
|
m_State = eReadHTTPHeaders;
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
ShouldProcessHeader(const std::string& name) const
|
|
{
|
|
// TODO: header whitelist
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
ProcessHeaderLine(std::string& line)
|
|
{
|
|
// TODO: implement me
|
|
auto idx = line.find_first_of(':');
|
|
if(idx == std::string::npos)
|
|
return false;
|
|
std::string header = line.substr(0, idx);
|
|
std::string val = line.substr(idx);
|
|
// to lowercase
|
|
std::transform(header.begin(), header.end(), header.begin(),
|
|
[](char ch) -> char { return ::tolower(ch); });
|
|
if(ShouldProcessHeader(header))
|
|
{
|
|
val = val.substr(val.find_first_not_of(' '));
|
|
m_Header.Headers.insert(std::make_pair(header, val));
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
WriteStatusLine(int code, const std::string& message)
|
|
{
|
|
char buf[128] = {0};
|
|
int sz = snprintf(buf, sizeof(buf), "HTTP/1.0 %d %s\r\n", code,
|
|
message.c_str());
|
|
if(sz > 0)
|
|
{
|
|
llarp::LogInfo("HTTP ", code, " ", message);
|
|
return llarp_tcp_conn_async_write(_conn, buf, sz);
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
bool
|
|
WriteResponseSimple(int code, const std::string& msg,
|
|
const char* contentType, const char* content)
|
|
{
|
|
if(!WriteStatusLine(code, msg))
|
|
return false;
|
|
char buf[128] = {0};
|
|
int sz =
|
|
snprintf(buf, sizeof(buf), "Content-Type: %s\r\n", contentType);
|
|
if(sz <= 0)
|
|
return false;
|
|
if(!llarp_tcp_conn_async_write(_conn, buf, sz))
|
|
return false;
|
|
size_t contentLength = strlen(content);
|
|
sz = snprintf(buf, sizeof(buf), "Content-Length: %zu\r\n\r\n",
|
|
contentLength);
|
|
if(sz <= 0)
|
|
return false;
|
|
if(!llarp_tcp_conn_async_write(_conn, buf, sz))
|
|
return false;
|
|
if(!llarp_tcp_conn_async_write(_conn, content, contentLength))
|
|
return false;
|
|
m_State = eWriteHTTPBody;
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
FeedBody(const char* buf, size_t sz)
|
|
{
|
|
llarp::LogInfo("HTTP ", m_Header.Method, " ", m_Header.Path, " ", sz);
|
|
if(sz == 0)
|
|
{
|
|
return WriteResponseSimple(400, "Bad Request", "text/plain", "nope");
|
|
}
|
|
if(m_Header.Method != "POST")
|
|
{
|
|
return WriteResponseSimple(400, "Bad Request", "text/plain", "nope");
|
|
}
|
|
return WriteResponseSimple(200, "OK", "text/json", "{}");
|
|
}
|
|
|
|
bool
|
|
ProcessRead(const char* buf, size_t sz)
|
|
{
|
|
if(m_Bad)
|
|
{
|
|
llarp::LogInfo("we bad");
|
|
return false;
|
|
}
|
|
|
|
m_LastActive = llarp_time_now_ms();
|
|
if(m_State < eReadHTTPBody)
|
|
{
|
|
if(strstr(buf, "\r\n") == nullptr)
|
|
{
|
|
// probably too big or small
|
|
return false;
|
|
}
|
|
m_ReadBuf << std::string(buf, sz);
|
|
std::string line;
|
|
while(std::getline(m_ReadBuf, line, '\n'))
|
|
{
|
|
if(line[0] == '\r')
|
|
{
|
|
m_State = eReadHTTPBody;
|
|
line = m_ReadBuf.str();
|
|
const char* ptr = strstr(line.c_str(), "\r\n\r\n");
|
|
if(ptr == nullptr)
|
|
return false;
|
|
line = std::string(ptr + 4);
|
|
m_ReadBuf.clear();
|
|
return FeedBody(line.c_str(), line.size());
|
|
}
|
|
auto pos = line.find_first_of('\r');
|
|
if(pos == std::string::npos)
|
|
{
|
|
return false;
|
|
}
|
|
line = line.substr(0, pos);
|
|
|
|
if(!FeedLine(line))
|
|
return false;
|
|
}
|
|
m_ReadBuf.str(line);
|
|
}
|
|
else
|
|
return FeedBody(buf, sz);
|
|
|
|
return true;
|
|
}
|
|
|
|
static void
|
|
OnRead(llarp_tcp_conn* conn, const void* buf, size_t sz)
|
|
{
|
|
ConnImpl* self = static_cast< ConnImpl* >(conn->user);
|
|
if(!self->ProcessRead((const char*)buf, sz))
|
|
self->MarkBad();
|
|
}
|
|
|
|
static void
|
|
OnClosed(llarp_tcp_conn* conn)
|
|
{
|
|
ConnImpl* self = static_cast< ConnImpl* >(conn->user);
|
|
self->_conn = nullptr;
|
|
}
|
|
|
|
static void
|
|
OnTick(llarp_tcp_conn* conn)
|
|
{
|
|
ConnImpl* self = static_cast< ConnImpl* >(conn->user);
|
|
self->Tick();
|
|
}
|
|
|
|
void
|
|
Tick()
|
|
{
|
|
}
|
|
|
|
/// mark bad so next tick we are closed
|
|
void
|
|
MarkBad()
|
|
{
|
|
m_Bad = true;
|
|
}
|
|
|
|
bool
|
|
ShouldClose(llarp_time_t now) const
|
|
{
|
|
return now - m_LastActive > m_ReadTimeout || m_Bad
|
|
|| m_State == eCloseMe;
|
|
}
|
|
|
|
void
|
|
Close()
|
|
{
|
|
if(_conn)
|
|
{
|
|
llarp_tcp_conn_close(_conn);
|
|
_conn = nullptr;
|
|
}
|
|
}
|
|
};
|
|
|
|
IRPCHandler::IRPCHandler(ConnImpl* conn) : m_Impl(conn)
|
|
{
|
|
}
|
|
|
|
IRPCHandler::~IRPCHandler()
|
|
{
|
|
m_Impl->Close();
|
|
delete m_Impl;
|
|
}
|
|
|
|
bool
|
|
IRPCHandler::ShouldClose(llarp_time_t now) const
|
|
{
|
|
return m_Impl->ShouldClose(now);
|
|
}
|
|
|
|
BaseReqHandler::BaseReqHandler(llarp_time_t reqtimeout)
|
|
: m_ReqTimeout(reqtimeout)
|
|
{
|
|
m_loop = nullptr;
|
|
m_Logic = nullptr;
|
|
m_acceptor.accepted = &BaseReqHandler::OnAccept;
|
|
m_acceptor.user = this;
|
|
m_acceptor.tick = &OnTick;
|
|
m_acceptor.closed = nullptr;
|
|
}
|
|
|
|
bool
|
|
BaseReqHandler::ServeAsync(llarp_ev_loop* loop, llarp_logic* logic,
|
|
const sockaddr* bindaddr)
|
|
{
|
|
m_loop = loop;
|
|
m_Logic = logic;
|
|
return llarp_tcp_serve(m_loop, &m_acceptor, bindaddr);
|
|
}
|
|
|
|
void
|
|
BaseReqHandler::OnTick(llarp_tcp_acceptor* tcp)
|
|
{
|
|
BaseReqHandler* self = static_cast< BaseReqHandler* >(tcp->user);
|
|
|
|
self->Tick();
|
|
}
|
|
|
|
void
|
|
BaseReqHandler::Tick()
|
|
{
|
|
auto now = llarp_time_now_ms();
|
|
auto itr = m_Conns.begin();
|
|
while(itr != m_Conns.end())
|
|
{
|
|
if((*itr)->ShouldClose(now))
|
|
itr = m_Conns.erase(itr);
|
|
else
|
|
++itr;
|
|
}
|
|
}
|
|
|
|
BaseReqHandler::~BaseReqHandler()
|
|
{
|
|
llarp_tcp_acceptor_close(&m_acceptor);
|
|
}
|
|
|
|
void
|
|
BaseReqHandler::OnAccept(llarp_tcp_acceptor* acceptor, llarp_tcp_conn* conn)
|
|
{
|
|
BaseReqHandler* self = static_cast< BaseReqHandler* >(acceptor->user);
|
|
ConnImpl* connimpl = new ConnImpl(self, conn, self->m_ReqTimeout);
|
|
IRPCHandler* rpcHandler = self->CreateHandler(connimpl);
|
|
if(rpcHandler == nullptr)
|
|
{
|
|
connimpl->Close();
|
|
delete connimpl;
|
|
return;
|
|
}
|
|
connimpl->handler = rpcHandler;
|
|
self->m_Conns.emplace_back(rpcHandler);
|
|
}
|
|
} // namespace http
|
|
} // namespace abyss
|