/* 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-2020 SChernykh * Copyright 2016-2020 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 "3rdparty/http-parser/http_parser.h" #include "base/io/log/Log.h" #include "base/kernel/Platform.h" #include "base/net/dns/Dns.h" #include "base/net/http/HttpClient.h" #include "base/tools/Baton.h" namespace xmrig { static const char *kCRLF = "\r\n"; class ClientWriteBaton : public Baton { public: inline ClientWriteBaton(const std::string &header, std::string &&body) : m_body(std::move(body)), m_header(header) { bufs[0].len = m_header.size(); bufs[0].base = const_cast(m_header.c_str()); if (!m_body.empty()) { bufs[1].len = m_body.size(); bufs[1].base = const_cast(m_body.c_str()); } else { bufs[1].base = nullptr; bufs[1].len = 0; } } inline size_t count() const { return bufs[1].base == nullptr ? 1 : 2; } inline size_t size() const { return bufs[0].len + bufs[1].len; } inline static void onWrite(uv_write_t *req, int) { delete reinterpret_cast(req->data); } uv_buf_t bufs[2]{}; private: std::string m_body; std::string m_header; }; } // namespace xmrig xmrig::HttpClient::HttpClient(int method, const String &url, const std::weak_ptr &listener, const char *data, size_t size) : HttpContext(HTTP_RESPONSE, listener) { this->method = method; this->url = url; if (data) { body = size ? std::string(data, size) : data; } m_dns = new Dns(this); } xmrig::HttpClient::~HttpClient() { delete m_dns; } bool xmrig::HttpClient::connect(const String &host, uint16_t port) { m_port = port; return m_dns->resolve(host); } const xmrig::String &xmrig::HttpClient::host() const { return m_dns->host(); } void xmrig::HttpClient::onResolved(const Dns &dns, int status) { this->status = status; if (status < 0 && dns.isEmpty()) { if (!m_quiet) { LOG_ERR("[%s:%d] DNS error: \"%s\"", dns.host().data(), m_port, uv_strerror(status)); } return; } sockaddr *addr = dns.get().addr(m_port); auto req = new uv_connect_t; req->data = this; uv_tcp_connect(req, m_tcp, addr, onConnect); } void xmrig::HttpClient::handshake() { headers.insert({ "Host", m_dns->host().data() }); headers.insert({ "Connection", "close" }); headers.insert({ "User-Agent", Platform::userAgent() }); if (!body.empty()) { headers.insert({ "Content-Length", std::to_string(body.size()) }); headers.insert({ "Content-Type", "application/json" }); } std::stringstream ss; ss << http_method_str(static_cast(method)) << " " << url << " HTTP/1.1" << kCRLF; for (auto &header : headers) { ss << header.first << ": " << header.second << kCRLF; } ss << kCRLF; headers.clear(); write(ss.str()); } void xmrig::HttpClient::read(const char *data, size_t size) { if (parse(data, size) < size) { close(UV_EPROTO); } } void xmrig::HttpClient::write(const std::string &header) { auto baton = new ClientWriteBaton(header, std::move(body)); uv_write(&baton->req, stream(), baton->bufs, baton->count(), ClientWriteBaton::onWrite); } void xmrig::HttpClient::onConnect(uv_connect_t *req, int status) { auto client = static_cast(req->data); if (!client) { delete req; return; } if (status < 0) { if (!client->m_quiet) { LOG_ERR("[%s:%d] connect error: \"%s\"", client->m_dns->host().data(), client->m_port, uv_strerror(status)); } delete req; client->close(status); return; } uv_read_start(client->stream(), [](uv_handle_t *, size_t suggested_size, uv_buf_t *buf) { buf->base = new char[suggested_size]; # ifdef _WIN32 buf->len = static_cast(suggested_size); # else buf->len = suggested_size; # endif }, [](uv_stream_t *tcp, ssize_t nread, const uv_buf_t *buf) { auto client = static_cast(tcp->data); if (nread >= 0) { client->read(buf->base, static_cast(nread)); } else { if (!client->m_quiet && nread != UV_EOF) { LOG_ERR("[%s:%d] read error: \"%s\"", client->m_dns->host().data(), client->m_port, uv_strerror(static_cast(nread))); } client->close(static_cast(nread)); } delete [] buf->base; }); client->handshake(); }