Archived
1
0
This repository has been archived on 2023-08-12. You can view files and clone it, but cannot push or open issues or pull requests.

235 lines
4.7 KiB
C++
Raw Normal View History

2016-10-15 12:37:59 -04:00
#include <algorithm>
#include <cctype>
2017-05-03 13:33:04 -04:00
#include <cstring>
2016-10-15 12:37:59 -04:00
#include <iostream>
2017-10-17 10:29:56 -04:00
#include <nntpchan/nntp_handler.hpp>
#include <nntpchan/sanitize.hpp>
#include <sstream>
#include <string>
2016-10-15 12:37:59 -04:00
namespace nntpchan
{
2017-10-17 10:29:56 -04:00
NNTPServerHandler::NNTPServerHandler(const fs::path &storage)
: LineReader(1024), m_article(nullptr), m_auth(nullptr), m_store(std::make_unique<ArticleStorage>(storage)),
m_authed(false), m_state(eStateReadCommand)
{
}
2016-10-15 13:53:35 -04:00
2017-10-17 10:29:56 -04:00
NNTPServerHandler::~NNTPServerHandler() {}
2017-04-14 06:24:53 -04:00
2017-10-17 10:29:56 -04:00
void NNTPServerHandler::HandleLine(const std::string &line)
{
if (m_state == eStateReadCommand)
2016-10-15 12:37:59 -04:00
{
2017-10-17 10:29:56 -04:00
std::deque<std::string> command;
std::istringstream s;
s.str(line);
for (std::string part; std::getline(s, part, ' ');)
2017-05-03 13:33:04 -04:00
{
2017-10-17 10:29:56 -04:00
if (part.size())
command.push_back(part);
2017-05-03 13:33:04 -04:00
}
2017-10-17 10:29:56 -04:00
if (command.size())
HandleCommand(command);
2017-05-03 13:33:04 -04:00
else
2017-10-17 10:29:56 -04:00
QueueLine("501 Syntax error");
}
else if (m_state == eStateStoreArticle)
{
std::string l = line + "\r\n";
OnData(l.c_str(), l.size());
2016-10-15 12:37:59 -04:00
}
2017-10-17 10:29:56 -04:00
else
{
std::cerr << "invalid state" << std::endl;
}
}
2016-10-15 12:37:59 -04:00
2017-10-17 10:29:56 -04:00
void NNTPServerHandler::OnData(const char *data, ssize_t l)
{
if (l <= 0)
return;
if (m_state == eStateStoreArticle)
2017-05-03 09:44:42 -04:00
{
2017-10-17 10:29:56 -04:00
const char *end = strstr(data, "\r\n.\r\n");
if (end)
2017-05-03 13:33:04 -04:00
{
2017-10-17 10:29:56 -04:00
std::size_t diff = end - data;
if (m_article)
2017-10-09 12:51:49 -04:00
{
2017-10-17 10:29:56 -04:00
m_article->write(data, diff + 2);
2017-10-09 12:51:49 -04:00
m_article->flush();
}
2017-10-17 10:29:56 -04:00
ArticleObtained();
diff += 5;
Data(end + 5, l - diff);
return;
}
if (m_article)
{
m_article->write(data, l);
m_article->flush();
2017-05-03 13:33:04 -04:00
}
2017-05-03 09:44:42 -04:00
}
2017-10-17 10:29:56 -04:00
else
Data(data, l);
}
2017-05-03 09:44:42 -04:00
2017-10-17 10:29:56 -04:00
void NNTPServerHandler::HandleCommand(const std::deque<std::string> &command)
{
auto cmd = command[0];
std::transform(cmd.begin(), cmd.end(), cmd.begin(), ::toupper);
std::size_t cmdlen = command.size();
for (const auto &part : command)
std::cerr << " " << part;
std::cerr << std::endl;
if (cmd == "QUIT")
2016-10-15 12:37:59 -04:00
{
2017-10-17 10:29:56 -04:00
Quit();
return;
}
else if (cmd[0] == '5')
{
return;
}
else if (cmd == "MODE")
{
if (cmdlen == 2)
{
// set mode
SwitchMode(command[1]);
2017-05-03 13:33:04 -04:00
}
2017-10-17 10:29:56 -04:00
else if (cmdlen)
2017-05-03 13:33:04 -04:00
{
2017-10-17 10:29:56 -04:00
// too many arguments
QueueLine("500 too many arguments");
2017-05-03 13:33:04 -04:00
}
2017-10-17 10:29:56 -04:00
else
{
// get mode
QueueLine("500 wrong arguments");
2016-10-15 12:37:59 -04:00
}
}
2017-10-17 10:29:56 -04:00
else if (cmd == "CAPABILITIES")
2017-05-03 13:33:04 -04:00
{
2017-10-17 10:29:56 -04:00
QueueLine("101 I support the following:");
QueueLine("READER");
QueueLine("IMPLEMENTATION nntpchan-daemon");
QueueLine("VERSION 2");
QueueLine("STREAMING");
QueueLine(".");
}
else if (cmd == "CHECK")
{
if (cmdlen >= 2)
2017-05-03 13:33:04 -04:00
{
2017-10-17 10:29:56 -04:00
const std::string &msgid = command[1];
if (IsValidMessageID(msgid) && m_store->Accept(msgid))
{
QueueLine("238 " + msgid);
}
else
QueueLine("438 " + msgid);
2017-05-03 13:33:04 -04:00
}
else
2017-10-17 10:29:56 -04:00
QueueLine("501 syntax error");
2017-05-03 13:33:04 -04:00
}
2017-10-17 10:29:56 -04:00
else if (cmd == "TAKETHIS")
2016-10-15 12:37:59 -04:00
{
2017-10-17 10:29:56 -04:00
if (cmdlen >= 2)
{
const std::string &msgid = command[1];
if (m_store->Accept(msgid))
{
m_article = m_store->OpenWrite(msgid);
2017-04-14 06:24:53 -04:00
}
2017-10-17 10:29:56 -04:00
m_articleName = msgid;
EnterState(eStateStoreArticle);
return;
2017-04-14 06:24:53 -04:00
}
2017-10-17 10:29:56 -04:00
QueueLine("501 invalid syntax");
2016-10-15 12:37:59 -04:00
}
2017-10-17 10:29:56 -04:00
else
2016-10-15 12:37:59 -04:00
{
2017-10-17 10:29:56 -04:00
// unknown command
QueueLine("500 Unknown Command");
2016-10-15 12:37:59 -04:00
}
2017-10-17 10:29:56 -04:00
}
2016-10-15 12:37:59 -04:00
2017-10-17 10:29:56 -04:00
void NNTPServerHandler::ArticleObtained()
{
if (m_article)
2016-10-15 12:37:59 -04:00
{
2017-10-17 10:29:56 -04:00
m_article->close();
m_article = nullptr;
QueueLine("239 " + m_articleName);
std::cerr << "stored " << m_articleName << std::endl;
2016-10-15 12:37:59 -04:00
}
2017-10-17 10:29:56 -04:00
else
QueueLine("439 " + m_articleName);
m_articleName = "";
EnterState(eStateReadCommand);
}
2016-10-15 13:53:35 -04:00
2017-10-17 10:29:56 -04:00
void NNTPServerHandler::SwitchMode(const std::string &mode)
{
std::string m = mode;
std::transform(m.begin(), m.end(), m.begin(), ::toupper);
if (m == "READER")
2016-10-15 13:53:35 -04:00
{
2017-10-17 10:29:56 -04:00
m_mode = m;
if (PostingAllowed())
{
QueueLine("200 Posting is permitted yo");
}
else
{
QueueLine("201 Posting is not permitted yo");
}
2016-10-15 13:53:35 -04:00
}
2017-10-17 10:29:56 -04:00
else if (m == "STREAM")
2016-10-15 13:53:35 -04:00
{
2017-10-17 10:29:56 -04:00
m_mode = m;
if (PostingAllowed())
{
QueueLine("203 Streaming enabled");
}
2016-10-15 13:53:35 -04:00
else
2017-10-17 10:29:56 -04:00
{
QueueLine("483 Streaming Denied");
}
2016-10-15 13:53:35 -04:00
}
2017-10-17 10:29:56 -04:00
else
2016-10-15 13:53:35 -04:00
{
2017-10-17 10:29:56 -04:00
// unknown mode
QueueLine("500 Unknown mode");
2016-10-15 13:53:35 -04:00
}
2016-10-15 12:37:59 -04:00
}
2017-10-17 10:29:56 -04:00
void NNTPServerHandler::EnterState(State st)
{
std::cerr << "enter state " << st << std::endl;
m_state = st;
}
void NNTPServerHandler::Quit()
{
EnterState(eStateQuit);
QueueLine("205 quitting");
}
bool NNTPServerHandler::ShouldClose() { return m_state == eStateQuit; }
bool NNTPServerHandler::PostingAllowed() { return m_authed || m_auth == nullptr; }
void NNTPServerHandler::Greet()
{
if (PostingAllowed())
QueueLine("200 Posting allowed");
else
QueueLine("201 Posting not allowed");
}
void NNTPServerHandler::SetAuth(CredDB_ptr creds) { m_auth = creds; }
}