/* * Copyright (C) 2008-2017 Max Kellermann * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * - Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * - Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the * distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED * OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef CURL_REQUEST_HXX #define CURL_REQUEST_HXX #include "Easy.hxx" #include #include struct StringView; class CurlGlobal; class CurlResponseHandler; class CurlRequest { CurlGlobal &global; CurlResponseHandler &handler; /** the curl handle */ CurlEasy easy; enum class State { HEADERS, BODY, CLOSED, } state = State::HEADERS; std::multimap headers; /** error message provided by libcurl */ char error_buffer[CURL_ERROR_SIZE]; public: CurlRequest(CurlGlobal &_global, const char *url, CurlResponseHandler &_handler); ~CurlRequest(); CurlRequest(const CurlRequest &) = delete; CurlRequest &operator=(const CurlRequest &) = delete; CURL *Get() { return easy.Get(); } template void SetOption(CURLoption option, T value) { easy.SetOption(option, value); } /** * CurlResponseHandler::OnData() shall throw this to pause the * stream. Call Resume() to resume the transfer. */ struct Pause {}; void Resume(); /** * A HTTP request is finished. Called by #CurlGlobal. */ void Done(CURLcode result); private: /** * Frees the current "libcurl easy" handle, and everything * associated with it. */ void FreeEasy(); bool FinishHeaders(); void FinishBody(); size_t DataReceived(const void *ptr, size_t size); void HeaderFunction(StringView s); /** called by curl when new data is available */ static size_t _HeaderFunction(void *ptr, size_t size, size_t nmemb, void *stream); /** called by curl when new data is available */ static size_t WriteFunction(void *ptr, size_t size, size_t nmemb, void *stream); }; #endif