#include "http.hpp" #include #include "finally.hpp" #pragma comment(lib, "ws2_32.lib") namespace utils::http { namespace { struct progress_helper { const std::function* callback{}; std::exception_ptr exception{}; }; int progress_callback(void* clientp, const curl_off_t /*dltotal*/, const curl_off_t dlnow, const curl_off_t /*ultotal*/, const curl_off_t /*ulnow*/) { auto* helper = static_cast(clientp); try { if (*helper->callback) { (*helper->callback)(dlnow); } } catch (...) { helper->exception = std::current_exception(); return -1; } return 0; } size_t write_callback(void* contents, const size_t size, const size_t nmemb, void* userp) { auto* buffer = static_cast(userp); const auto total_size = size * nmemb; buffer->append(static_cast(contents), total_size); return total_size; } } std::optional get_data(const std::string& url, const headers& headers, const std::function& callback, const uint32_t retries) { curl_slist* header_list = nullptr; auto* curl = curl_easy_init(); if (!curl) { return {}; } auto _ = utils::finally([&]() { curl_slist_free_all(header_list); curl_easy_cleanup(curl); }); for (const auto& header : headers) { auto data = header.first + ": " + header.second; header_list = curl_slist_append(header_list, data.data()); } std::string buffer{}; progress_helper helper{}; helper.callback = &callback; curl_easy_setopt(curl, CURLOPT_HTTPHEADER, header_list); curl_easy_setopt(curl, CURLOPT_URL, url.data()); curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_callback); curl_easy_setopt(curl, CURLOPT_WRITEDATA, &buffer); curl_easy_setopt(curl, CURLOPT_XFERINFOFUNCTION, progress_callback); curl_easy_setopt(curl, CURLOPT_XFERINFODATA, &helper); curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 0L); curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1L); curl_easy_setopt(curl, CURLOPT_USERAGENT, "xlabs-updater/1.0"); curl_easy_setopt(curl, CURLOPT_FAILONERROR, 1L); curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L); curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L); for (auto i = 0u; i < retries + 1; ++i) { // Due to CURLOPT_FAILONERROR, CURLE_OK will not be met when the server returns 400 or 500 if (curl_easy_perform(curl) == CURLE_OK) { long http_code = 0; curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &http_code); if (http_code >= 200) { return { std::move(buffer) }; } throw std::runtime_error( "Bad status code " + std::to_string(http_code) + " met while trying to download file " + url); } if (helper.exception) { std::rethrow_exception(helper.exception); } long http_code = 0; curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &http_code); if (http_code > 0) { break; } } return {}; } std::future> get_data_async(const std::string& url, const headers& headers) { return std::async(std::launch::async, [url, headers]() { return get_data(url, headers); }); } }