Files
mpd/src/lib/curl/Easy.hxx
2023-09-17 14:26:47 +02:00

217 lines
4.7 KiB
C++

// SPDX-License-Identifier: BSD-2-Clause
// author: Max Kellermann <max.kellermann@gmail.com>
#pragma once
#include "Error.hxx"
#include "String.hxx"
#include <curl/curl.h>
#include <cstddef>
#include <stdexcept>
#include <string_view>
#include <utility>
/**
* An OO wrapper for a "CURL*" (a libCURL "easy" handle).
*/
class CurlEasy {
CURL *handle = nullptr;
public:
/**
* Allocate a new CURL*.
*
* Throws std::runtime_error on error.
*/
CurlEasy()
:handle(curl_easy_init())
{
if (handle == nullptr)
throw std::runtime_error("curl_easy_init() failed");
}
explicit CurlEasy(const char *url)
:CurlEasy() {
SetURL(url);
}
/**
* Create an empty instance.
*/
CurlEasy(std::nullptr_t) noexcept:handle(nullptr) {}
CurlEasy(CurlEasy &&src) noexcept
:handle(std::exchange(src.handle, nullptr)) {}
~CurlEasy() noexcept {
if (handle != nullptr)
curl_easy_cleanup(handle);
}
operator bool() const noexcept {
return handle != nullptr;
}
CurlEasy &operator=(CurlEasy &&src) noexcept {
std::swap(handle, src.handle);
return *this;
}
CURL *Get() noexcept {
return handle;
}
template<typename T>
void SetOption(CURLoption option, T value) {
CURLcode code = curl_easy_setopt(handle, option, value);
if (code != CURLE_OK)
throw Curl::MakeError(code, "Failed to set option");
}
void SetPrivate(void *pointer) {
SetOption(CURLOPT_PRIVATE, pointer);
}
void SetErrorBuffer(char *buf) {
SetOption(CURLOPT_ERRORBUFFER, buf);
}
void SetURL(const char *value) {
SetOption(CURLOPT_URL, value);
}
void SetUserAgent(const char *value) {
SetOption(CURLOPT_USERAGENT, value);
}
void SetRequestHeaders(struct curl_slist *headers) {
SetOption(CURLOPT_HTTPHEADER, headers);
}
void SetBasicAuth(const char *userpwd) {
SetOption(CURLOPT_USERPWD, userpwd);
}
void SetUpload(bool value=true) {
SetOption(CURLOPT_UPLOAD, (long)value);
}
void SetNoProgress(bool value=true) {
SetOption(CURLOPT_NOPROGRESS, (long)value);
}
void SetXferInfoFunction(curl_xferinfo_callback function,
void *data) {
SetOption(CURLOPT_XFERINFOFUNCTION, function);
SetOption(CURLOPT_XFERINFODATA, data);
SetNoProgress(false);
}
void SetNoSignal(bool value=true) {
SetOption(CURLOPT_NOSIGNAL, (long)value);
}
void SetFailOnError(bool value=true) {
SetOption(CURLOPT_FAILONERROR, (long)value);
}
void SetVerifyHost(bool value) {
SetOption(CURLOPT_SSL_VERIFYHOST, value ? 2L : 0L);
}
void SetVerifyPeer(bool value) {
SetOption(CURLOPT_SSL_VERIFYPEER, (long)value);
}
void SetProxyVerifyHost(bool value) {
SetOption(CURLOPT_PROXY_SSL_VERIFYHOST, value ? 2L : 0L);
}
void SetProxyVerifyPeer(bool value) {
SetOption(CURLOPT_PROXY_SSL_VERIFYPEER, value);
}
void SetConnectTimeout(long timeout) {
SetOption(CURLOPT_CONNECTTIMEOUT, timeout);
}
void SetTimeout(long timeout) {
SetOption(CURLOPT_TIMEOUT, timeout);
}
void SetHeaderFunction(size_t (*function)(char *buffer, size_t size,
size_t nitems,
void *userdata),
void *userdata) {
SetOption(CURLOPT_HEADERFUNCTION, function);
SetOption(CURLOPT_HEADERDATA, userdata);
}
void SetWriteFunction(size_t (*function)(char *ptr, size_t size,
size_t nmemb, void *userdata),
void *userdata) {
SetOption(CURLOPT_WRITEFUNCTION, function);
SetOption(CURLOPT_WRITEDATA, userdata);
}
void SetReadFunction(size_t (*function)(char *ptr, size_t size,
size_t nmemb, void *userdata),
void *userdata) {
SetOption(CURLOPT_READFUNCTION, function);
SetOption(CURLOPT_READDATA, userdata);
}
void SetNoBody(bool value=true) {
SetOption(CURLOPT_NOBODY, (long)value);
}
void SetPost(bool value=true) {
SetOption(CURLOPT_POST, (long)value);
}
void SetRequestBody(const void *data, size_t size) {
SetOption(CURLOPT_POSTFIELDS, data);
SetOption(CURLOPT_POSTFIELDSIZE, (long)size);
}
void SetRequestBody(std::string_view s) {
SetRequestBody(s.data(), s.size());
}
void SetMimePost(const curl_mime *mime) {
SetOption(CURLOPT_MIMEPOST, mime);
}
template<typename T>
bool GetInfo(CURLINFO info, T value_r) const noexcept {
return ::curl_easy_getinfo(handle, info, value_r) == CURLE_OK;
}
/**
* Returns the response body's size, or -1 if that is unknown.
*/
[[gnu::pure]]
curl_off_t GetContentLength() const noexcept {
curl_off_t value;
return GetInfo(CURLINFO_CONTENT_LENGTH_DOWNLOAD_T, &value)
? value
: -1;
}
void Perform() {
CURLcode code = curl_easy_perform(handle);
if (code != CURLE_OK)
throw Curl::MakeError(code, "CURL failed");
}
bool Unpause() noexcept {
return ::curl_easy_pause(handle, CURLPAUSE_CONT) == CURLE_OK;
}
CurlString Escape(const char *string, int length=0) const noexcept {
return CurlString(curl_easy_escape(handle, string, length));
}
};