123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487 |
- /*
- * Copyright (c) 2017-2023 zhllxt
- *
- * author : zhllxt
- * email : 37792738@qq.com
- *
- * Distributed under the Boost Software License, Version 1.0. (See accompanying
- * file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
- */
- #if defined(ASIO2_ENABLE_SSL) || defined(ASIO2_USE_SSL)
- #ifndef __ASIO2_HTTPS_DOWNLOAD_HPP__
- #define __ASIO2_HTTPS_DOWNLOAD_HPP__
- #if defined(_MSC_VER) && (_MSC_VER >= 1200)
- #pragma once
- #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
- #include <asio2/base/detail/push_options.hpp>
- #include <fstream>
- #include <asio2/base/detail/function_traits.hpp>
- #include <asio2/util/string.hpp>
- #include <asio2/http/detail/http_util.hpp>
- #include <asio2/http/detail/http_make.hpp>
- #include <asio2/http/detail/http_traits.hpp>
- #include <asio2/component/socks/socks5_client_cp.hpp>
- namespace asio2::detail
- {
- template<class derived_t, class args_t, bool Enable = is_https_execute_download_enabled<args_t>::value()>
- struct https_download_impl_bridge;
- template<class derived_t, class args_t>
- struct https_download_impl_bridge<derived_t, args_t, false>
- {
- };
- template<class derived_t, class args_t>
- struct https_download_impl_bridge<derived_t, args_t, true>
- {
- protected:
- template<typename String, typename StrOrInt, class HeaderCallback, class BodyCallback, class Proxy,
- class Body, class Fields, class Buffer>
- static void _download_with_socks5(
- asio::io_context& ioc, asio::ip::tcp::resolver& resolver, asio::ip::tcp::socket& socket,
- asio::ssl::stream<asio::ip::tcp::socket&>& stream,
- Buffer& buffer,
- String&& host, StrOrInt&& port,
- http::request<Body, Fields>& req, HeaderCallback&& cbh, BodyCallback&& cbb, Proxy&& proxy)
- {
- auto sk5 = detail::to_shared_ptr(std::forward<Proxy>(proxy));
- std::string_view h{ sk5->host() };
- std::string_view p{ sk5->port() };
- if (static_cast<int>(sk5->command()) == 0)
- {
- sk5->command(socks5::command::connect);
- }
- // Look up the domain name
- resolver.async_resolve(h, p, [&, s5 = std::move(sk5)]
- (const error_code& ec1, const asio::ip::tcp::resolver::results_type& endpoints) mutable
- {
- if (ec1) { set_last_error(ec1); return; }
- // Make the connection on the IP address we get from a lookup
- asio::async_connect(socket, endpoints,
- [&, s5 = std::move(s5)](const error_code& ec2, const asio::ip::tcp::endpoint&) mutable
- {
- if (ec2) { set_last_error(ec2); return; }
- socks5_async_handshake
- (
- detail::to_string(std::forward<String >(host)),
- detail::to_string(std::forward<StrOrInt>(port)),
- socket,
- std::move(s5),
- [&](error_code ecs5, std::string, std::string) mutable
- {
- if (ecs5) { set_last_error(ecs5); return; }
- // https://github.com/djarek/certify
- if (auto it = req.find(http::field::host); it != req.end())
- {
- std::string hostname(it->value());
- SSL_set_tlsext_host_name(stream.native_handle(), hostname.data());
- }
- stream.async_handshake(asio::ssl::stream_base::client,
- [&](const error_code& ec3) mutable
- {
- if (ec3) { set_last_error(ec3); return; }
- http::async_write(stream, req, [&](const error_code& ec4, std::size_t) mutable
- {
- // can't use stream.shutdown(),in some case the shutdowm will blocking forever.
- if (ec4) { set_last_error(ec4); stream.async_shutdown([](const error_code&) {}); return; }
- http::read_large_body<false>(stream, buffer,
- std::forward<HeaderCallback>(cbh), std::forward<BodyCallback>(cbb));
- stream.async_shutdown([](const error_code&) mutable {});
- });
- });
- }
- );
- });
- });
- }
- template<typename String, typename StrOrInt, class HeaderCallback, class BodyCallback,
- class Body, class Fields, class Buffer>
- static void _download_trivially(
- asio::io_context& ioc, asio::ip::tcp::resolver& resolver, asio::ip::tcp::socket& socket,
- asio::ssl::stream<asio::ip::tcp::socket&>& stream,
- Buffer& buffer,
- String&& host, StrOrInt&& port,
- http::request<Body, Fields>& req, HeaderCallback&& cbh, BodyCallback&& cbb)
- {
- detail::ignore_unused(ioc);
- // Look up the domain name
- resolver.async_resolve(std::forward<String>(host), detail::to_string(std::forward<StrOrInt>(port)),
- [&](const error_code& ec1, const asio::ip::tcp::resolver::results_type& endpoints) mutable
- {
- if (ec1) { set_last_error(ec1); return; }
- // Make the connection on the IP address we get from a lookup
- asio::async_connect(socket, endpoints,
- [&](const error_code& ec2, const asio::ip::tcp::endpoint&) mutable
- {
- if (ec2) { set_last_error(ec2); return; }
- // https://github.com/djarek/certify
- if (auto it = req.find(http::field::host); it != req.end())
- {
- std::string hostname(it->value());
- SSL_set_tlsext_host_name(stream.native_handle(), hostname.data());
- }
- stream.async_handshake(asio::ssl::stream_base::client,
- [&](const error_code& ec3) mutable
- {
- if (ec3) { set_last_error(ec3); return; }
- http::async_write(stream, req, [&](const error_code& ec4, std::size_t) mutable
- {
- // can't use stream.shutdown(),in some case the shutdowm will blocking forever.
- if (ec4) { set_last_error(ec4); stream.async_shutdown([](const error_code&) {}); return; }
- http::read_large_body<false>(stream, buffer,
- std::forward<HeaderCallback>(cbh), std::forward<BodyCallback>(cbb));
- stream.async_shutdown([](const error_code&) mutable {});
- });
- });
- });
- });
- }
- template<typename String, typename StrOrInt, class HeaderCallback, class BodyCallback, class Proxy,
- class Body, class Fields, class Buffer>
- static void _download_impl(
- asio::io_context& ioc, asio::ip::tcp::resolver& resolver, asio::ip::tcp::socket& socket,
- asio::ssl::stream<asio::ip::tcp::socket&>& stream,
- Buffer& buffer,
- String&& host, StrOrInt&& port,
- http::request<Body, Fields>& req, HeaderCallback&& cbh, BodyCallback&& cbb, Proxy&& proxy)
- {
- // if has socks5 proxy
- if constexpr (std::is_base_of_v<asio2::socks5::option_base,
- typename detail::element_type_adapter<detail::remove_cvref_t<Proxy>>::type>)
- {
- derived_t::_download_with_socks5(ioc, resolver, socket, stream, buffer
- , std::forward<String>(host), std::forward<StrOrInt>(port)
- , req, std::forward<HeaderCallback>(cbh), std::forward<BodyCallback>(cbb)
- , std::forward<Proxy>(proxy)
- );
- }
- else
- {
- detail::ignore_unused(proxy);
- derived_t::_download_trivially(ioc, resolver, socket, stream, buffer
- , std::forward<String>(host), std::forward<StrOrInt>(port)
- , req, std::forward<HeaderCallback>(cbh), std::forward<BodyCallback>(cbb)
- );
- }
- }
- public:
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param ctx - asio ssl context.
- * @param host - The ip of the server.
- * @param port - The port of the server.
- * @param req - The http request object to send to the https server.
- * @param cbh - A function that recv the http response header message. void(auto& message)
- * @param cbb - A function that circularly receives the contents of the file in chunks. void(std::string_view data)
- * @param proxy - socks5 proxy, if you do not need a proxy, pass "nullptr" is ok.
- */
- template<typename String, typename StrOrInt, class HeaderCallback, class BodyCallback, class Proxy,
- class Body = http::string_body, class Fields = http::fields, class Buffer = beast::flat_buffer>
- typename std::enable_if_t<detail::is_character_string_v<detail::remove_cvref_t<String>>
- && detail::http_proxy_checker_v<Proxy>, bool>
- static inline download(const asio::ssl::context& ctx, String&& host, StrOrInt&& port,
- http::request<Body, Fields>& req, HeaderCallback&& cbh, BodyCallback&& cbb, Proxy&& proxy)
- {
- // First assign default value 0 to last error
- clear_last_error();
- // The io_context is required for all I/O
- asio::io_context ioc;
- // These objects perform our I/O
- asio::ip::tcp::resolver resolver{ ioc };
- asio::ip::tcp::socket socket{ ioc };
- asio::ssl::stream<asio::ip::tcp::socket&> stream(socket, const_cast<asio::ssl::context&>(ctx));
- // This buffer is used for reading and must be persisted
- Buffer buffer;
- // Some sites must set the http::field::host
- if (req.find(http::field::host) == req.end())
- {
- std::string strhost = asio2::to_string(host);
- std::string strport = asio2::to_string(port);
- if (strport != "443")
- {
- strhost += ":";
- strhost += strport;
- }
- req.set(http::field::host, strhost);
- }
- // Some sites must set the http::field::user_agent
- if (req.find(http::field::user_agent) == req.end())
- {
- req.set(http::field::user_agent,
- "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/105.0.0.0 Safari/537.36");
- }
- // do work
- derived_t::_download_impl(ioc, resolver, socket, stream, buffer
- , std::forward<String>(host), std::forward<StrOrInt>(port)
- , req, std::forward<HeaderCallback>(cbh), std::forward<BodyCallback>(cbb)
- , std::forward<Proxy>(proxy)
- );
- // timedout run
- ioc.run();
- error_code ec_ignore{};
- // Gracefully close the socket
- socket.shutdown(asio::ip::tcp::socket::shutdown_both, ec_ignore);
- socket.cancel(ec_ignore);
- socket.close(ec_ignore);
- return bool(!get_last_error());
- }
- // ----------------------------------------------------------------------------------------
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param req - The web_request which can be create by http::make_request(...)
- * @param filepath - The file path to saved the received file content.
- */
- template<class String2>
- typename std::enable_if_t<detail::can_convert_to_string_v<detail::remove_cvref_t<String2>>, bool>
- static inline download(const asio::ssl::context& ctx, http::web_request& req, String2&& filepath)
- {
- std::filesystem::path path(std::forward<String2>(filepath));
- std::filesystem::create_directories(path.parent_path(), get_last_error());
- std::fstream file(path, std::ios::out | std::ios::binary | std::ios::trunc);
- if (!file)
- {
- set_last_error(asio::error::access_denied); // Permission denied
- return false;
- }
- auto cbh = [](const auto&) {};
- auto cbb = [&file](std::string_view chunk) mutable
- {
- file.write(chunk.data(), chunk.size());
- return true;
- };
- return derived_t::download(ctx, req.host(), req.port(), req.base(), cbh, cbb, std::in_place);
- }
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param req - The web_request which can be create by http::make_request(...)
- * @param filepath - The file path to saved the received file content.
- */
- template<class String2>
- typename std::enable_if_t<detail::can_convert_to_string_v<detail::remove_cvref_t<String2>>, bool>
- static inline download(http::web_request& req, String2&& filepath)
- {
- return derived_t::download(asio::ssl::context{ ASIO2_DEFAULT_SSL_METHOD },
- req, std::forward<String2>(filepath));
- }
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param url - The url of the file to download.
- * @param filepath - The file path to saved the received file content.
- */
- template<class String1, class String2>
- typename std::enable_if_t<
- detail::can_convert_to_string_v<detail::remove_cvref_t<String1>> &&
- detail::can_convert_to_string_v<detail::remove_cvref_t<String2>>, bool>
- static inline download(const asio::ssl::context& ctx, String1&& url, String2&& filepath)
- {
- http::web_request req = http::make_request(std::forward<String1>(url));
- if (get_last_error())
- return false;
- return derived_t::download(ctx, req, std::forward<String2>(filepath));
- }
- // ----------------------------------------------------------------------------------------
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param url - The url of the file to download.
- * @param filepath - The file path to saved the received file content.
- */
- template<class String1, class String2>
- typename std::enable_if_t<
- detail::can_convert_to_string_v<detail::remove_cvref_t<String1>> &&
- detail::can_convert_to_string_v<detail::remove_cvref_t<String2>>, bool>
- static inline download(String1&& url, String2&& filepath)
- {
- return derived_t::download(asio::ssl::context{ ASIO2_DEFAULT_SSL_METHOD },
- std::forward<String1>(url), std::forward<String2>(filepath));
- }
- // ----------------------------------------------------------------------------------------
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param url - The url of the file to download.
- * @param cbb - A function that circularly receives the contents of the file in chunks. void(std::string_view data)
- */
- template<class String1, class BodyCallback>
- typename std::enable_if_t<
- detail::can_convert_to_string_v<detail::remove_cvref_t<String1>> &&
- detail::is_callable_v<BodyCallback>, bool>
- static inline download(const asio::ssl::context& ctx, String1&& url, BodyCallback&& cbb)
- {
- http::web_request req = http::make_request(std::forward<String1>(url));
- if (get_last_error())
- return false;
- auto cbh = [](const auto&) {};
- return derived_t::download(ctx, req.host(), req.port(), req.base(), cbh, cbb, std::in_place);
- }
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param req - The web_request which can be create by http::make_request(...)
- * @param cbb - A function that circularly receives the contents of the file in chunks. void(std::string_view data)
- */
- template<class BodyCallback>
- typename std::enable_if_t<detail::is_callable_v<BodyCallback>, bool>
- static inline download(const asio::ssl::context& ctx, http::web_request& req, BodyCallback&& cbb)
- {
- auto cbh = [](const auto&) {};
- return derived_t::download(ctx, req.host(), req.port(), req.base(), cbh, cbb, std::in_place);
- }
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param req - The web_request which can be create by http::make_request(...)
- * @param cbb - A function that circularly receives the contents of the file in chunks. void(std::string_view data)
- */
- template<class BodyCallback>
- typename std::enable_if_t<detail::is_callable_v<BodyCallback>, bool>
- static inline download(http::web_request& req, BodyCallback&& cbb)
- {
- return derived_t::download(asio::ssl::context{ ASIO2_DEFAULT_SSL_METHOD }, req, cbb);
- }
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param req - The web_request which can be create by http::make_request(...)
- * @param cbh - A function that recv the http response header message. void(auto& message)
- * @param cbb - A function that circularly receives the contents of the file in chunks. void(std::string_view data)
- */
- template<class HeaderCallback, class BodyCallback>
- typename std::enable_if_t<detail::is_callable_v<BodyCallback>, bool>
- static inline download(const asio::ssl::context& ctx, http::web_request& req, HeaderCallback&& cbh, BodyCallback&& cbb)
- {
- return derived_t::download(ctx, req.host(), req.port(), req.base(), cbh, cbb, std::in_place);
- }
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param req - The web_request which can be create by http::make_request(...)
- * @param cbh - A function that recv the http response header message. void(auto& message)
- * @param cbb - A function that circularly receives the contents of the file in chunks. void(std::string_view data)
- */
- template<class HeaderCallback, class BodyCallback>
- typename std::enable_if_t<detail::is_callable_v<BodyCallback>, bool>
- static inline download(http::web_request& req, HeaderCallback&& cbh, BodyCallback&& cbb)
- {
- return derived_t::download(asio::ssl::context{ ASIO2_DEFAULT_SSL_METHOD }, req, cbh, cbb);
- }
- // ----------------------------------------------------------------------------------------
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param url - The url of the file to download.
- * @param cbb - A function that circularly receives the contents of the file in chunks. void(std::string_view data)
- */
- template<class String1, class BodyCallback>
- typename std::enable_if_t<
- detail::can_convert_to_string_v<detail::remove_cvref_t<String1>> &&
- detail::is_callable_v<BodyCallback>, bool>
- static inline download(String1&& url, BodyCallback&& cbb)
- {
- return derived_t::download(asio::ssl::context{ ASIO2_DEFAULT_SSL_METHOD },
- std::forward<String1>(url), std::forward<BodyCallback>(cbb));
- }
- // ----------------------------------------------------------------------------------------
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param url - The url of the file to download.
- * @param cbh - A function that recv the http response header message. void(auto& message)
- * @param cbb - A function that circularly receives the contents of the file in chunks. void(std::string_view data)
- */
- template<class String1, class HeaderCallback, class BodyCallback>
- typename std::enable_if_t<
- detail::can_convert_to_string_v<detail::remove_cvref_t<String1>> &&
- detail::is_callable_v<BodyCallback>, bool>
- static inline download(const asio::ssl::context& ctx, String1&& url, HeaderCallback&& cbh, BodyCallback&& cbb)
- {
- http::web_request req = http::make_request(std::forward<String1>(url));
- if (get_last_error())
- return false;
- return derived_t::download(ctx, req.host(), req.port(), req.base(), cbh, cbb, std::in_place);
- }
- // ----------------------------------------------------------------------------------------
- /**
- * @brief blocking download the http file until it is returned on success or failure
- * @param url - The url of the file to download.
- * @param cbh - A function that recv the http response header message. void(auto& message)
- * @param cbb - A function that circularly receives the contents of the file in chunks. void(std::string_view data)
- */
- template<class String1, class HeaderCallback, class BodyCallback>
- typename std::enable_if_t<
- detail::can_convert_to_string_v<detail::remove_cvref_t<String1>> &&
- detail::is_callable_v<BodyCallback>, bool>
- static inline download(String1&& url, HeaderCallback&& cbh, BodyCallback&& cbb)
- {
- return derived_t::download(asio::ssl::context{ ASIO2_DEFAULT_SSL_METHOD },
- std::forward<String1>(url), std::forward<HeaderCallback>(cbh), std::forward<BodyCallback>(cbb));
- }
- };
- template<class derived_t, class args_t = void>
- struct https_download_impl : public https_download_impl_bridge<derived_t, args_t> {};
- }
- #include <asio2/base/detail/pop_options.hpp>
- #endif // !__ASIO2_HTTPS_DOWNLOAD_HPP__
- #endif
|