/* XMRig * Copyright 2010 Jeff Garzik * Copyright 2012-2014 pooler * Copyright 2014 Lucas Jones * Copyright 2014-2016 Wolf9466 * Copyright 2016 Jay D Dee * Copyright 2017-2018 XMR-Stak , * Copyright 2014-2019 heapwolf * Copyright 2018-2019 SChernykh * Copyright 2016-2019 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 #include #include "3rdparty/http-parser/http_parser.h" #include "base/kernel/interfaces/IHttpListener.h" #include "base/net/http/HttpContext.h" namespace xmrig { static uint64_t SEQUENCE = 0; std::map HttpContext::m_storage; } // namespace xmrig xmrig::HttpContext::HttpContext(int parser_type, IHttpListener *listener) : HttpRequest(SEQUENCE++), listener(listener), connect(nullptr), m_wasHeaderValue(false) { m_storage[id()] = this; parser = new http_parser; tcp = new uv_tcp_t; uv_tcp_init(uv_default_loop(), tcp); http_parser_init(parser, static_cast(parser_type)); parser->data = tcp->data = this; } xmrig::HttpContext::~HttpContext() { delete connect; delete tcp; delete parser; } void xmrig::HttpContext::close() { auto it = m_storage.find(id()); if (it != m_storage.end()) { m_storage.erase(it); } 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) { if (m_storage.count(id) == 0) { return nullptr; } return m_storage[id]; } void xmrig::HttpContext::attach(http_parser_settings *settings) { if (settings->on_message_complete != nullptr) { return; } settings->on_message_begin = nullptr; settings->on_status = nullptr; settings->on_chunk_header = nullptr; settings->on_chunk_complete = nullptr; settings->on_url = [](http_parser *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 = [](http_parser* parser) -> int { HttpContext *ctx = static_cast(parser->data); ctx->method = parser->method; if (!ctx->m_lastHeaderField.empty()) { ctx->setHeader(); } return 0; }; settings->on_body = [](http_parser *parser, const char *at, size_t len) -> int { static_cast(parser->data)->body += std::string(at, len); return 0; }; settings->on_message_complete = [](http_parser *parser) -> int { const HttpContext *ctx = reinterpret_cast(parser->data); ctx->listener->onHttpRequest(*ctx); return 0; }; } void xmrig::HttpContext::closeAll() { for (auto kv : m_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(http_parser *parser, const char *at, size_t length) { HttpContext *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(http_parser *parser, const char *at, size_t length) { HttpContext *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::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(); }