/* XMRig * Copyright (c) 2014-2019 heapwolf * Copyright (c) 2018-2021 SChernykh * Copyright (c) 2016-2021 XMRig , * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "base/net/http/HttpContext.h" #include "3rdparty/llhttp/llhttp.h" #include "base/kernel/interfaces/IHttpListener.h" #include "base/tools/Baton.h" #include "base/tools/Chrono.h" #include #include namespace xmrig { static llhttp_settings_t http_settings; static std::map storage; static uint64_t SEQUENCE = 0; class HttpWriteBaton : public Baton { public: XMRIG_DISABLE_COPY_MOVE_DEFAULT(HttpWriteBaton) inline HttpWriteBaton(std::string &&body, HttpContext *ctx) : m_ctx(ctx), m_body(std::move(body)) { m_buf = uv_buf_init(&m_body.front(), m_body.size()); } inline ~HttpWriteBaton() { if (m_ctx) { m_ctx->close(); } } void write(uv_stream_t *stream) { uv_write(&req, stream, &m_buf, 1, [](uv_write_t *req, int) { delete reinterpret_cast(req->data); }); } private: HttpContext *m_ctx; std::string m_body; uv_buf_t m_buf{}; }; } // namespace xmrig xmrig::HttpContext::HttpContext(int parser_type, const std::weak_ptr &listener) : HttpData(SEQUENCE++), m_timestamp(Chrono::steadyMSecs()), m_listener(listener) { storage[id()] = this; m_parser = new llhttp_t; m_tcp = new uv_tcp_t; uv_tcp_init(uv_default_loop(), m_tcp); uv_tcp_nodelay(m_tcp, 1); llhttp_init(m_parser, static_cast(parser_type), &http_settings); m_parser->data = m_tcp->data = this; if (http_settings.on_message_complete == nullptr) { attach(&http_settings); } } xmrig::HttpContext::~HttpContext() { delete m_tcp; delete m_parser; } void xmrig::HttpContext::write(std::string &&data, bool close) { if (uv_is_writable(stream()) != 1) { return; } auto baton = new HttpWriteBaton(std::move(data), close ? this : nullptr); baton->write(stream()); } bool xmrig::HttpContext::isRequest() const { return m_parser->type == HTTP_REQUEST; } bool xmrig::HttpContext::parse(const char *data, size_t size) { if (size == 0) { return true; } return llhttp_execute(m_parser, data, size) == HPE_OK; } std::string xmrig::HttpContext::ip() const { char ip[46] = {}; sockaddr_storage addr = {}; int size = sizeof(addr); uv_tcp_getpeername(m_tcp, reinterpret_cast(&addr), &size); if (reinterpret_cast(&addr)->sin_family == AF_INET6) { uv_ip6_name(reinterpret_cast(&addr), ip, 45); } else { uv_ip4_name(reinterpret_cast(&addr), ip, 16); } return ip; } uint64_t xmrig::HttpContext::elapsed() const { return Chrono::steadyMSecs() - m_timestamp; } void xmrig::HttpContext::close(int status) { if (!get(id())) { return; } auto listener = httpListener(); if (status < 0 && listener) { this->status = status; listener->onHttpData(*this); } storage.erase(id()); if (!uv_is_closing(handle())) { uv_close(handle(), [](uv_handle_t *handle) -> void { delete reinterpret_cast(handle->data); }); } } xmrig::HttpContext *xmrig::HttpContext::get(uint64_t id) { const auto it = storage.find(id); return it == storage.end() ? nullptr : it->second; } void xmrig::HttpContext::closeAll() { for (auto &kv : storage) { if (!uv_is_closing(kv.second->handle())) { uv_close(kv.second->handle(), [](uv_handle_t *handle) -> void { delete reinterpret_cast(handle->data); }); } } } int xmrig::HttpContext::onHeaderField(llhttp_t *parser, const char *at, size_t length) { auto ctx = static_cast(parser->data); if (ctx->m_wasHeaderValue) { if (!ctx->m_lastHeaderField.empty()) { ctx->setHeader(); } ctx->m_lastHeaderField = std::string(at, length); ctx->m_wasHeaderValue = false; } else { ctx->m_lastHeaderField += std::string(at, length); } return 0; } int xmrig::HttpContext::onHeaderValue(llhttp_t *parser, const char *at, size_t length) { auto ctx = static_cast(parser->data); if (!ctx->m_wasHeaderValue) { ctx->m_lastHeaderValue = std::string(at, length); ctx->m_wasHeaderValue = true; } else { ctx->m_lastHeaderValue += std::string(at, length); } return 0; } void xmrig::HttpContext::attach(llhttp_settings_t *settings) { settings->on_message_begin = nullptr; settings->on_status = nullptr; settings->on_chunk_header = nullptr; settings->on_chunk_complete = nullptr; settings->on_url = [](llhttp_t *parser, const char *at, size_t length) -> int { static_cast(parser->data)->url = std::string(at, length); return 0; }; settings->on_header_field = onHeaderField; settings->on_header_value = onHeaderValue; settings->on_headers_complete = [](llhttp_t *parser) -> int { auto ctx = static_cast(parser->data); ctx->status = parser->status_code; if (parser->type == HTTP_REQUEST) { ctx->method = parser->method; } if (!ctx->m_lastHeaderField.empty()) { ctx->setHeader(); } return 0; }; settings->on_body = [](llhttp_t *parser, const char *at, size_t len) -> int { static_cast(parser->data)->body.append(at, len); return 0; }; settings->on_message_complete = [](llhttp_t *parser) -> int { auto ctx = static_cast(parser->data); auto listener = ctx->httpListener(); if (listener) { listener->onHttpData(*ctx); ctx->m_listener.reset(); } return 0; }; } void xmrig::HttpContext::setHeader() { std::transform(m_lastHeaderField.begin(), m_lastHeaderField.end(), m_lastHeaderField.begin(), ::tolower); headers.insert({ m_lastHeaderField, m_lastHeaderValue }); m_lastHeaderField.clear(); m_lastHeaderValue.clear(); }