Compare commits

...

2 Commits

Author SHA1 Message Date
4c119c6138 initial webhook infrastructure 2025-02-01 20:57:57 -08:00
1a6ec835ed make a myirc namespace 2025-02-01 11:04:33 -08:00
31 changed files with 551 additions and 50 deletions

View File

@ -10,7 +10,7 @@ find_package(OpenSSL REQUIRED)
pkg_check_modules(LIBHS libhs REQUIRED IMPORTED_TARGET)
set(BOOST_INCLUDE_LIBRARIES asio log signals2 endian)
set(BOOST_INCLUDE_LIBRARIES asio log signals2 endian beast)
set(BOOST_ENABLE_CMAKE ON)
include(FetchContent)
FetchContent_Declare(

View File

@ -1,12 +1,13 @@
add_executable(xbot
main.cpp
settings.cpp
web.cpp
)
target_link_libraries(xbot PRIVATE
myirc
OpenSSL::SSL
Boost::signals2 Boost::log Boost::asio
Boost::signals2 Boost::log Boost::asio Boost::beast
tomlplusplus_tomlplusplus
PkgConfig::LIBHS
mysocks5 mybase64)

View File

@ -1,13 +1,16 @@
#include "bot.hpp"
#include "challenge.hpp"
#include "client.hpp"
#include "connection.hpp"
#include "openssl_utils.hpp"
#include "registration.hpp"
#include "sasl_mechanism.hpp"
#include "settings.hpp"
#include "ref.hpp"
#include "irc_coroutine.hpp"
#include "web.hpp"
#include "myirc/bot.hpp"
#include "myirc/challenge.hpp"
#include "myirc/client.hpp"
#include "myirc/connection.hpp"
#include "myirc/openssl_utils.hpp"
#include "myirc/registration.hpp"
#include "myirc/sasl_mechanism.hpp"
#include "myirc/ref.hpp"
#include "myirc/irc_coroutine.hpp"
#include <boost/asio.hpp>
#include <boost/log/trivial.hpp>
@ -19,6 +22,16 @@
#include <memory>
using namespace std::literals;
using myirc::SaslMechanism;
using myirc::SaslPlain;
using myirc::SaslExternal;
using myirc::SaslEcdsa;
using myirc::Bot;
using myirc::Client;
using myirc::Connection;
using myirc::Registration;
using myirc::Challenge;
using myirc::Ref;
auto configure_sasl(const Settings &settings) -> std::unique_ptr<SaslMechanism>
{
@ -38,7 +51,7 @@ auto configure_sasl(const Settings &settings) -> std::unique_ptr<SaslMechanism>
not settings.sasl_authcid.empty() &&
not settings.sasl_key_file.empty()
) {
if (auto sasl_key = key_from_file(settings.sasl_key_file, settings.sasl_key_password))
if (auto sasl_key = myirc::key_from_file(settings.sasl_key_file, settings.sasl_key_password))
return std::make_unique<SaslEcdsa>(
settings.sasl_authcid,
settings.sasl_authzid,
@ -47,18 +60,22 @@ auto configure_sasl(const Settings &settings) -> std::unique_ptr<SaslMechanism>
return nullptr;
}
static auto start(boost::asio::io_context &io, const Settings &settings) -> void
static auto start(
boost::asio::io_context &io,
const Settings &settings,
std::shared_ptr<GithubWebhook> webhook
) -> void
{
Ref<X509> tls_cert;
if (settings.use_tls && not settings.tls_cert_file.empty())
{
tls_cert = cert_from_file(settings.tls_cert_file);
tls_cert = myirc::cert_from_file(settings.tls_cert_file);
}
Ref<EVP_PKEY> tls_key;
if (settings.use_tls && not settings.tls_key_file.empty())
{
tls_key = key_from_file(settings.tls_key_file, settings.tls_key_password);
tls_key = myirc::key_from_file(settings.tls_key_file, settings.tls_key_password);
}
const auto connection = std::make_shared<Connection>(io);
@ -75,20 +92,25 @@ static auto start(boost::asio::io_context &io, const Settings &settings) -> void
// Configure CHALLENGE on registration if applicable
if (not settings.challenge_username.empty() && not settings.challenge_key_file.empty()) {
if (auto key = key_from_file(settings.challenge_key_file, settings.challenge_key_password)) {
if (auto key = myirc::key_from_file(settings.challenge_key_file, settings.challenge_key_password)) {
client->sig_registered.connect([&settings, connection, key = std::move(key)]() {
Challenge::start(connection, settings.challenge_username, key);
});
}
}
client->sig_registered.connect([connection, webhook]() {
webhook->set_connection(connection);
});
// On disconnect reconnect in 5 seconds
// connection is captured in the disconnect handler so it can keep itself alive
connection->sig_disconnect.connect(
[&io, &settings, connection]() {
[&io, &settings, connection, webhook]() {
webhook->clear_connection();
auto timer = std::make_shared<boost::asio::steady_timer>(io);
timer->expires_after(5s);
timer->async_wait([&io, &settings, timer](auto) { start(io, settings); });
timer->async_wait([&io, &settings, timer, webhook](auto) { start(io, settings, webhook); });
}
);
@ -128,12 +150,15 @@ auto main(int argc, char *argv[]) -> int
{
//boost::log::core::get()->set_filter(boost::log::trivial::severity >= boost::log::trivial::warning);
if (argc != 2) {
if (argc != 3) {
BOOST_LOG_TRIVIAL(error) << "Bad arguments";
return 1;
}
const auto settings = get_settings(argv[1]);
auto io = boost::asio::io_context{};
start(io, settings);
auto webhooks = start_webhook(io, argv[2]);
start(io, settings, webhooks);
io.run();
}

263
driver/web.cpp Normal file
View File

@ -0,0 +1,263 @@
#include "web.hpp"
#include <boost/beast.hpp>
#include <boost/log/trivial.hpp>
#include <chrono>
#include <fstream>
#include <vector>
namespace beast = boost::beast; // from <boost/beast.hpp>
namespace http = beast::http; // from <boost/beast/http.hpp>
namespace net = boost::asio; // from <boost/asio.hpp>
namespace websocket = beast::websocket;
using tcp = net::ip::tcp; // from <boost/asio/ip/tcp.hpp>
using namespace std::literals;
namespace {
auto report_error(std::exception_ptr eptr) -> void
{
if (eptr)
{
try
{
std::rethrow_exception(eptr);
}
catch (const std::exception &e)
{
BOOST_LOG_TRIVIAL(error) << "An error occurred: " << e.what();
}
}
}
template <class Body, class Allocator>
auto handle_request(
std::shared_ptr<GithubWebhook> self,
http::request<Body, http::basic_fields<Allocator>> &&req
) -> http::message_generator
{
self->add_event({"project", "message"});
http::response<http::string_body> res{http::int_to_status(200), req.version()};
res.set(http::field::server, BOOST_BEAST_VERSION_STRING);
res.keep_alive(req.keep_alive());
std::string reply_text = "Hello, world!";
res.content_length(reply_text.size());
res.body() = std::move(reply_text);
return res;
}
auto read_loop(tcp::socket socket, std::shared_ptr<GithubWebhook> self) -> boost::asio::awaitable<void>
{
beast::tcp_stream stream{std::move(socket)};
beast::flat_buffer buffer;
http::request<http::string_body> req;
for (;;)
{
req.clear();
stream.expires_after(30s);
boost::system::error_code ec;
co_await http::async_read(stream, buffer, req, net::redirect_error(net::use_awaitable, ec));
if (ec == http::error::end_of_stream)
{
stream.socket().shutdown(tcp::socket::shutdown_send, ec);
co_return;
}
else if (ec)
{
co_return;
}
auto msg = handle_request(self, std::move(req));
const auto keep_alive = msg.keep_alive();
co_await beast::async_write(stream, std::move(msg), net::use_awaitable);
if (!keep_alive)
{
stream.socket().shutdown(tcp::socket::shutdown_send, ec);
co_return;
}
}
}
auto accept_loop(
tcp::acceptor acceptor,
std::shared_ptr<GithubWebhook> self
) -> boost::asio::awaitable<void>
{
for (;;)
{
auto socket = co_await acceptor.async_accept(net::use_awaitable);
boost::asio::co_spawn(
acceptor.get_executor(),
read_loop(std::move(socket), self),
report_error
);
}
}
auto spawn_webhook(
boost::asio::io_context &io,
const std::shared_ptr<GithubWebhook> webhook
) -> boost::asio::awaitable<void>
{
tcp::resolver resolver{io};
auto results = co_await resolver.async_resolve(webhook->settings_.host, webhook->settings_.service, tcp::resolver::passive, boost::asio::use_awaitable);
for (auto &&result : results)
{
const auto endpoint = result.endpoint();
BOOST_LOG_TRIVIAL(info) << "HTTP: Listening on " << endpoint;
tcp::acceptor acceptor{io};
acceptor.open(endpoint.protocol());
acceptor.set_option(net::socket_base::reuse_address(true));
acceptor.bind(endpoint);
acceptor.listen(net::socket_base::max_listen_connections);
boost::asio::co_spawn(io, accept_loop(std::move(acceptor), webhook), report_error);
}
}
} // namespace
auto start_webhook(
boost::asio::io_context &io,
const char * webhook_settings_filename
) -> std::shared_ptr<GithubWebhook>
{
std::ifstream webhook_settings_file{webhook_settings_filename};
if (!webhook_settings_file)
{
BOOST_LOG_TRIVIAL(error) << "Unable to open webhook settings file";
std::exit(1);
}
auto webhook_settings = toml::parse(webhook_settings_file);
WebhookSettings settings = WebhookSettings::from_toml(webhook_settings);
BOOST_LOG_TRIVIAL(info) << "Webhook settings: " << settings.to_toml();
auto webhook = std::make_shared<GithubWebhook>(std::move(settings));
boost::asio::co_spawn(io, spawn_webhook(io, webhook), report_error);
return webhook;
}
auto GithubWebhook::write_event(WebhookEvent event) -> void
{
connection_->send_notice("glguy", event.channel + ": " + event.message);
}
auto ProjectSettings::from_toml(const toml::table &v) -> ProjectSettings
{
ProjectSettings result;
result.channel = v["channel"].value_or(""s);
result.credential_name = v["credential_name"].value_or(""s);
result.enabled = v["enabled"].value_or(false);
if (const auto events = v["events"].as_array())
{
for (const auto &event : *events)
{
result.events.insert(event.value_or(""s));
}
}
if (const auto accounts = v["authorized_accounts"].as_array())
{
for (const auto &account : *accounts)
{
result.authorized_accounts.insert(account.value_or(""s));
}
}
return result;
}
auto ProjectSettings::to_toml() const -> toml::table
{
toml::array events_array;
for (const auto &event : events)
{
events_array.emplace_back(event);
}
toml::array authorized_accounts_array;
for (const auto &account : authorized_accounts)
{
authorized_accounts_array.emplace_back(account);
}
\
return toml::table{
{"channel", channel},
{"credential_name", credential_name},
{"enabled", enabled},
{"events", std::move(events_array)},
{"authorized_accounts", std::move(authorized_accounts_array)}
};
}
auto WebhookSettings::from_toml(const toml::table &v) -> WebhookSettings
{
WebhookSettings result;
result.host = v["host"].value_or(""s);
result.service = v["service"].value_or("http"s);
if (const auto credentials = v["credentials"].as_array())
{
for (const auto &credential : *credentials)
{
if (auto credential_table = credential.as_table())
{
result.credentials.emplace(
(*credential_table)["name"].value_or(""s),
(*credential_table)["key"].value_or(""s));
}
}
}
if (const auto projects = v["projects"].as_array())
{
for (const auto &project : *projects)
{
if (auto project_table = project.as_table())
{
result.projects.emplace(
(*project_table)["name"].value_or(""s),
ProjectSettings::from_toml(*project_table));
}
}
}
return result;
}
auto WebhookSettings::to_toml() const -> toml::table
{
toml::array credential_tables;
for (const auto &[name, key] : credentials)
{
credential_tables.emplace_back(toml::table {
{"name", name},
{"key", key}
});
}
toml::array project_tables;
for (const auto &[name, project] : projects)
{
auto tab = project.to_toml();
tab.emplace("name", name);
project_tables.emplace_back(std::move(tab));
}
return toml::table{
{"host", host},
{"service", service},
{"credentials", std::move(credential_tables)},
{"projects", std::move(project_tables)}
};
}

100
driver/web.hpp Normal file
View File

@ -0,0 +1,100 @@
#pragma once
#include <myirc/connection.hpp>
#include <toml++/toml.hpp>
#include <boost/asio.hpp>
#include <boost/signals2.hpp>
#include <memory>
#include <map>
#include <set>
struct ProjectSettings {
// *** Administrative settings ***
// IRC channel to announce to
std::string channel;
// name extracted from notify/$user
std::string credential_name;
// Authorized accounts can edit the event list
std::set<std::string> authorized_accounts;
// *** User settings ***
// Events to announce
std::set<std::string> events;
// Whether to announce events
bool enabled;
auto to_toml() const -> toml::table;
static auto from_toml(const toml::table &v) -> ProjectSettings;
};
struct WebhookSettings {
std::string host;
std::string service;
std::map<std::string, std::string> credentials;
std::map<std::string, ProjectSettings> projects;
auto to_toml() const -> toml::table;
static auto from_toml(const toml::table &v) -> WebhookSettings;
};
struct WebhookEvent {
std::string channel;
std::string message;
};
class GithubWebhook {
// IRC connection to announce on; could be empty
std::shared_ptr<myirc::Connection> connection_;
// Buffered events in case connection was inactive when event was received
std::vector<WebhookEvent> events_;
// Actually write the event to the connection.
// Only call when there is a connection.
auto write_event(WebhookEvent event) -> void;
public:
WebhookSettings settings_;
GithubWebhook(WebhookSettings settings)
: settings_(std::move(settings))
{
}
// Either emit the event now or save it until a connection is set
auto add_event(WebhookEvent event) -> void
{
if (connection_) {
write_event(std::move(event));
} else {
events_.emplace_back(std::move(event));
}
}
auto set_connection(std::shared_ptr<myirc::Connection> connection) -> void
{
connection_ = std::move(connection);
for (auto &&event : events_)
{
write_event(event);
}
events_.clear();
}
auto clear_connection() -> void
{
connection_.reset();
}
};
auto start_webhook(boost::asio::io_context &io, const char *) -> std::shared_ptr<GithubWebhook>;

View File

@ -1,7 +1,9 @@
#include "bot.hpp"
#include "myirc/bot.hpp"
#include <boost/log/trivial.hpp>
namespace myirc {
auto Bot::start(std::shared_ptr<Client> self) -> std::shared_ptr<Bot>
{
const auto thread = std::make_shared<Bot>(std::move(self));
@ -77,3 +79,5 @@ auto Bot::shutdown() -> void
{
sig_command.disconnect_all_slots();
}
} // namespace myirc

View File

@ -1,6 +1,6 @@
#include "challenge.hpp"
#include "myirc/challenge.hpp"
#include "openssl_utils.hpp"
#include "myirc/openssl_utils.hpp"
#include <mybase64.hpp>
@ -12,6 +12,8 @@
#include <memory>
#include <string>
namespace myirc {
Challenge::Challenge(Ref<EVP_PKEY> key, std::shared_ptr<Connection> connection)
: key_{std::move(key)}
, connection_{std::move(connection)}
@ -29,7 +31,7 @@ auto Challenge::on_ircmsg(IrcCommand cmd, const IrcMsg &msg) -> void {
break;
case IrcCommand::RPL_YOUREOPER:
slot_.disconnect();
connection_->send_ping("mitigation");
//connection_->send_ping("mitigation");
break;
case IrcCommand::RPL_ENDOFRSACHALLENGE2:
finish_challenge();
@ -89,3 +91,5 @@ auto Challenge::start(std::shared_ptr<Connection> connection, const std::string_
connection->send_challenge(user);
return self;
}
} // namespace myirc

View File

@ -1,12 +1,15 @@
#include "client.hpp"
#include "myirc/client.hpp"
#include "connection.hpp"
#include "myirc/connection.hpp"
#include "myirc/sasl_mechanism.hpp"
#include <mybase64.hpp>
#include <boost/container/flat_map.hpp>
#include <boost/log/trivial.hpp>
namespace myirc {
using namespace std::literals;
auto Client::on_welcome(const IrcMsg &irc) -> void
@ -425,3 +428,5 @@ auto Client::on_cap(const IrcMsg &msg) -> void
);
}
}
} // namespace myirc

View File

@ -1,19 +1,19 @@
#include "connection.hpp"
#include "myirc/connection.hpp"
#include "boost/asio/steady_timer.hpp"
#include "linebuffer.hpp"
#include "myirc/linebuffer.hpp"
#include "myirc/snote.hpp"
#include <mybase64.hpp>
#include <boost/asio/steady_timer.hpp>
#include <boost/log/trivial.hpp>
namespace {
namespace myirc {
#include "irc_commands.inc"
using tcp_type = boost::asio::ip::tcp::socket;
using tls_type = boost::asio::ssl::stream<tcp_type>;
} // namespace
using namespace std::literals;
Connection::Connection(boost::asio::io_context &io)
@ -518,7 +518,10 @@ auto Connection::connect(
// Name resolution
auto resolver = boost::asio::ip::tcp::resolver{stream_.get_executor()};
const auto endpoints = co_await resolver.async_resolve(settings.host, std::to_string(settings.port), boost::asio::use_awaitable);
for (auto e : endpoints) {
BOOST_LOG_TRIVIAL(debug) << "DNS: " << e.endpoint();
}
// Connect to the IRC server
auto& socket = stream_.reset();
const auto endpoint = co_await boost::asio::async_connect(socket, endpoints, boost::asio::use_awaitable);
@ -604,3 +607,5 @@ auto Connection::start(Settings settings) -> void
self->sig_disconnect.disconnect_all_slots();
});
}
} // namespace myirc

View File

@ -6,6 +6,8 @@
#include <memory>
namespace myirc {
struct Bot : std::enable_shared_from_this<Bot>
{
struct Command
@ -34,3 +36,5 @@ struct Bot : std::enable_shared_from_this<Bot>
auto shutdown() -> void;
};
} // namespace myirc

View File

@ -1,5 +1,7 @@
#pragma once
namespace myirc {
template <typename> struct CCallback_;
template <typename F, typename R, typename... Ts>
struct CCallback_<R (F::*) (Ts...) const>
@ -14,3 +16,5 @@ struct CCallback_<R (F::*) (Ts...) const>
/// @tparam F Type of the closure
template <typename F>
using CCallback = CCallback_<decltype(&F::operator())>;
} // namespace myirc

View File

@ -8,6 +8,8 @@
#include <memory>
#include <string>
namespace myirc {
/// @brief Implements the CHALLENGE command protocol to identify as an operator.
class Challenge : std::enable_shared_from_this<Challenge>
{
@ -29,3 +31,5 @@ public:
/// @return Handle to the challenge object.
static auto start(std::shared_ptr<Connection>, std::string_view user, Ref<EVP_PKEY> key) -> std::shared_ptr<Challenge>;
};
} // namespace myirc

View File

@ -1,13 +1,14 @@
#pragma once
#include "connection.hpp"
#include "sasl_mechanism.hpp"
#include "myirc/connection.hpp"
#include "myirc/sasl_mechanism.hpp"
#include <string>
#include <unordered_set>
#include <span>
struct Connection;
namespace myirc {
struct IrcMsg;
enum class Casemap
@ -101,3 +102,5 @@ public:
auto shutdown() -> void;
};
} // namespace myirc

View File

@ -4,7 +4,6 @@
#include "ircmsg.hpp"
#include "ratelimit.hpp"
#include "ref.hpp"
#include "snote.hpp"
#include "stream.hpp"
#include <boost/asio.hpp>
@ -14,6 +13,10 @@
#include <memory>
#include <string>
namespace myirc {
struct SnoteMatch;
class Connection : public std::enable_shared_from_this<Connection>
{
public:
@ -136,3 +139,5 @@ auto Connection::write_irc(std::string front, std::string_view next, Args... res
front += next;
write_irc(std::move(front), rest...);
}
} // namespace myirc

View File

@ -1,3 +1,7 @@
#pragma once
namespace myirc {
enum class IrcCommand
{
UNKNOWN,
@ -280,3 +284,5 @@ enum class IrcCommand
TOPIC,
WALLOPS,
};
} // namespace myirc

View File

@ -1,6 +1,7 @@
#pragma once
#include "connection.hpp"
#include "myirc/connection.hpp"
#include "myirc/snote.hpp"
#include <chrono>
#include <coroutine>
@ -8,6 +9,8 @@
#include <utility>
#include <vector>
namespace myirc {
struct irc_promise;
/// A coroutine that can co_await on various IRC events
@ -275,3 +278,5 @@ inline auto irc_coroutine::exception() -> std::exception_ptr
{
return promise().exception_;
}
} // namespace myirc

View File

@ -4,6 +4,8 @@
#include <string_view>
#include <vector>
namespace myirc {
struct irctag
{
std::string_view key;
@ -72,3 +74,5 @@ struct irc_parse_error : public std::exception
auto parse_irc_message(char *msg) -> IrcMsg;
auto parse_irc_tags(char *msg) -> std::vector<irctag>;
} // namespace myirc

View File

@ -16,6 +16,8 @@
#include <concepts>
#include <vector>
namespace myirc {
/**
* @brief Fixed-size buffer with line-oriented dispatch
*
@ -99,3 +101,5 @@ public:
}
}
};
} // namespace myirc

View File

@ -4,6 +4,10 @@
#include <string_view>
namespace myirc {
auto log_openssl_errors(const std::string_view prefix) -> void;
auto key_from_file(const std::string &filename, const std::string_view password) -> Ref<EVP_PKEY>;
auto cert_from_file(const std::string &filename) -> Ref<X509>;
} // namespace myirc

View File

@ -3,6 +3,8 @@
#include <chrono>
#include <utility>
namespace myirc {
struct RateLimit {
virtual ~RateLimit();
auto virtual query(size_t want_to_send) -> std::pair<std::chrono::milliseconds, size_t> = 0;
@ -18,3 +20,5 @@ struct Rfc1459RateLimit final : RateLimit
auto query(size_t want_to_send) -> std::pair<std::chrono::milliseconds, size_t> override;
};
} // namespace myirc

View File

@ -5,6 +5,8 @@
#include <memory>
namespace myirc {
// Specializations must Free to release a reference
// Specializations can implement UpRef to increase a reference count on copy
template <typename> struct RefTraits {};
@ -56,3 +58,5 @@ struct Ref : std::unique_ptr<T, RefDeleter<T>>
return *this;
}
};
} // namespace myirc

View File

@ -7,6 +7,8 @@
#include <string>
#include <unordered_map>
namespace myirc {
class Registration : public std::enable_shared_from_this<Registration>
{
public:
@ -44,3 +46,5 @@ public:
std::shared_ptr<Client>
) -> std::shared_ptr<Registration>;
};
} // namespace myirc

View File

@ -10,6 +10,8 @@
#include <string_view>
#include <variant>
namespace myirc {
class SaslMechanism
{
public:
@ -108,3 +110,5 @@ public:
return stage_ == 2;;
}
};
} // namespace myirc

View File

@ -12,6 +12,8 @@
struct hs_database;
struct hs_scratch;
namespace myirc {
enum class SnoteTag
{
ClientConnecting,
@ -95,3 +97,5 @@ struct SnoteCore
};
extern SnoteCore snoteCore;
} // namespace myirc

View File

@ -6,6 +6,8 @@
#include <cstddef>
#include <variant>
namespace myirc {
/// @brief Abstraction over plain-text and TLS streams.
class Stream : private
std::variant<
@ -112,3 +114,5 @@ public:
socket.lowest_layer().close(err);
}
};
} // namespace myirc

View File

@ -1,8 +1,8 @@
#include "myirc/ircmsg.hpp"
#include <cstring>
#include <optional>
#include "ircmsg.hpp"
namespace {
class Parser
{
@ -104,6 +104,8 @@ std::string_view unescape_tag_value(char *const val)
} // namespace
namespace myirc {
auto parse_irc_tags(char *str) -> std::vector<irctag>
{
std::vector<irctag> tags;
@ -184,3 +186,5 @@ auto operator<<(std::ostream &out, irc_error_code code) -> std::ostream &
return out;
}
}
} // namespace myirc

View File

@ -1,6 +1,6 @@
#include "openssl_utils.hpp"
#include "myirc/openssl_utils.hpp"
#include "c_callback.hpp"
#include "myirc/c_callback.hpp"
#include <openssl/err.h>
#include <openssl/pem.h>
@ -11,6 +11,8 @@
using namespace std::literals;
namespace myirc {
auto log_openssl_errors(const std::string_view prefix) -> void
{
auto err_cb = [prefix](const char *str, size_t len) -> int {
@ -65,3 +67,5 @@ auto key_from_file(const std::string &filename, const std::string_view password)
}
return key;
}
} // namespace myirc

View File

@ -1,6 +1,9 @@
#include "ratelimit.hpp"
#include "myirc/ratelimit.hpp"
#include <chrono>
namespace myirc {
using namespace std::literals;
using ms = std::chrono::milliseconds;
@ -21,3 +24,5 @@ auto Rfc1459RateLimit::query(size_t want_to_send) -> std::pair<ms, size_t>
return {cost_ - gap, 1};
}
}
} // namespace myirc

View File

@ -1,12 +1,14 @@
#include "registration.hpp"
#include "myirc/registration.hpp"
#include "connection.hpp"
#include "ircmsg.hpp"
#include "myirc/connection.hpp"
#include "myirc/ircmsg.hpp"
#include <memory>
#include <random>
#include <unordered_map>
namespace myirc {
Registration::Registration(
Settings settings,
std::shared_ptr<Client> client
@ -142,3 +144,5 @@ auto Registration::on_ircmsg(const IrcCommand cmd, const IrcMsg &msg) -> void
break;
}
}
} // namespace myirc

View File

@ -1,8 +1,11 @@
#include "sasl_mechanism.hpp"
#include "openssl_utils.hpp"
#include "myirc/sasl_mechanism.hpp"
#include "myirc/openssl_utils.hpp"
#include <openssl/evp.h>
namespace myirc {
auto SaslPlain::step(std::string_view msg) -> StepResult {
if (complete_) {
return Failure{};
@ -72,3 +75,5 @@ auto SaslEcdsa::step(std::string_view msg) -> StepResult {
return Failure{};
}
}
} // namespace myirc

View File

@ -1,6 +1,6 @@
#include "snote.hpp"
#include "myirc/snote.hpp"
#include "c_callback.hpp"
#include "myirc/c_callback.hpp"
#include <hs.h>
@ -13,6 +13,9 @@
#include <stdexcept>
#include <utility>
namespace myirc {
namespace {
struct SnotePattern
@ -278,3 +281,5 @@ auto SnoteCore::ScratchDeleter::operator()(hs_scratch_t *scratch) const -> void
}
SnoteCore snoteCore;
} // namespace myirc