6 #ifndef ASIOEXT_SOCKS_DETAIL_CLIENT_HPP 7 #define ASIOEXT_SOCKS_DETAIL_CLIENT_HPP 9 #include "asioext/detail/config.hpp" 11 #if ASIOEXT_HAS_PRAGMA_ONCE 15 #include "asioext/socks/error.hpp" 16 #include "asioext/composed_operation.hpp" 17 #include "asioext/bind_handler.hpp" 19 #include "asioext/socks/detail/protocol.hpp" 20 #include "asioext/detail/coroutine.hpp" 21 #include "asioext/detail/move_support.hpp" 23 #if defined(ASIOEXT_USE_BOOST_ASIO) 24 # include <boost/asio/read.hpp> 25 # include <boost/asio/write.hpp> 27 # include <asio/read.hpp> 28 # include <asio/write.hpp> 36 template <
typename Socket,
typename DynamicBuffer,
typename Handler>
55 this->handler_, asio::error::invalid_argument,
56 auth_method::no_acceptable));
60 asio::mutable_buffer buf = buffer_.prepare(size);
62 asio::buffer_cast<uint8_t*>(buf));
65 asio::async_write(socket, buffer_.data(),
76 template <
typename Socket,
typename DynamicBuffer,
typename Handler>
81 this->handler_(ec, auth_method::no_acceptable);
85 ASIOEXT_CORO_REENTER (
this) {
86 buffer_.consume(bytes_transferred);
87 ASIOEXT_CORO_YIELD asio::async_read(
88 socket_, buffer_.prepare(2),
92 this->handler_(ec, auth_method::no_acceptable);
98 const uint8_t* data = asio::buffer_cast<
const uint8_t*>(buffer_.data());
100 const uint8_t
version = data[0];
106 this->handler_(error::invalid_version,
107 auth_method::no_acceptable);
111 if (chosen_method == auth_method::no_acceptable) {
112 this->handler_(error::no_acceptable_auth_method,
113 auth_method::no_acceptable);
117 this->handler_(ec, chosen_method);
121 template <
typename Socket,
typename DynamicBuffer,
typename Handler>
140 this->handler_, asio::error::invalid_argument));
144 asio::mutable_buffer buf = buffer_.prepare(size);
146 asio::buffer_cast<uint8_t*>(buf));
147 buffer_.commit(size);
149 asio::async_write(socket, buffer_.data(),
160 template <
typename Socket,
typename DynamicBuffer,
typename Handler>
169 ASIOEXT_CORO_REENTER (
this) {
170 buffer_.consume(bytes_transferred);
171 ASIOEXT_CORO_YIELD asio::async_read(
172 socket_, buffer_.prepare(2),
182 const uint8_t* data = asio::buffer_cast<
const uint8_t*>(buffer_.data());
184 const uint8_t
version = data[0];
185 const uint8_t status_code = data[1];
190 this->handler_(error::invalid_auth_version);
194 if (status_code != 0) {
195 this->handler_(error::login_failed);
203 template <
typename Socket,
typename DynamicBuffer,
typename Handler>
211 const asio::ip::tcp::endpoint& remote,
224 this->handler_, asio::error::invalid_argument));
228 asio::mutable_buffer buf = buffer_.prepare(size);
230 asio::buffer_cast<uint8_t*>(buf));
231 buffer_.commit(size);
233 asio::async_write(socket, buffer_.data(),
242 uint8_t address_type_;
243 uint8_t first_address_byte_;
246 template <
typename Socket,
typename DynamicBuffer,
typename Handler>
256 ASIOEXT_CORO_REENTER (
this) {
257 buffer_.consume(bytes_transferred);
258 ASIOEXT_CORO_YIELD asio::async_read(
259 socket_, buffer_.prepare(5),
270 const uint8_t* data = asio::buffer_cast<
const uint8_t*>(buffer_.data());
272 const uint8_t
version = data[0];
273 const uint8_t status_code = data[1];
275 address_type_ = data[3];
276 first_address_byte_ = data[4];
281 this->handler_(error::invalid_version);
285 if (status_code != 0) {
286 switch (status_code) {
287 case 2: ec = asio::error::no_permission;
break;
288 case 3: ec = asio::error::network_unreachable;
break;
289 case 4: ec = asio::error::host_unreachable;
break;
290 case 5: ec = asio::error::connection_refused;
break;
291 case 6: ec = asio::error::timed_out;
break;
292 case 7: ec = error::command_not_supported;
break;
293 case 8: ec = asio::error::address_family_not_supported;
break;
300 switch (address_type_) {
308 size = first_address_byte_;
318 ASIOEXT_CORO_YIELD asio::async_read(
319 socket_, buffer_.prepare(size + 2),
327 buffer_.commit(bytes_transferred);
328 buffer_.consume(bytes_transferred);
command
SOCKS commands.
Definition: constants.hpp:32
Definition: detail/client.hpp:204
auth_method
SOCKS login authentication methods.
Definition: constants.hpp:58
void encode_sgreet_packet(const auth_method *auth_methods, std::size_t num_auth_methods, uint8_t *out)
void encode_sexec_packet(command cmd, const asio::ip::tcp::endpoint &remote, const std::string &remote_host, uint16_t port, uint8_t *out)
socks_slogin_op(Handler &handler, Socket &socket, const std::string &username, const std::string &password, DynamicBuffer &buffer)
Definition: detail/client.hpp:127
implementation_defined bind_handler(Handler &&handler, Args &&... args)
Bind values to a Handler's arguments to create a CompletionHandler.
asio::const_buffers_1 buffer(const basic_linear_buffer< Allocator > &b) noexcept
Definition: linear_buffer.hpp:385
std::size_t get_sexec_packet_size(command cmd, const asio::ip::tcp::endpoint &remote, const std::string &remote_host, uint16_t port)
std::size_t get_slogin_packet_size(const std::string &username, const std::string &password)
void encode_slogin_packet(const std::string &username, const std::string &password, uint8_t *out)
Base class for composed operations.
Definition: composed_operation.hpp:111
socks_sgreet_op(Handler &handler, Socket &socket, const auth_method *auth_methods, std::size_t num_auth_methods, DynamicBuffer &buffer)
Definition: detail/client.hpp:42
version
SOCKS versions.
Definition: constants.hpp:48
socks_sexec_op(Handler &handler, Socket &socket, command cmd, const asio::ip::tcp::endpoint &remote, const std::string &remote_host, uint16_t port, DynamicBuffer &buffer)
Definition: detail/client.hpp:209
automatically_chosen error_code
Typedef for the error_code class used by this library.
Definition: error_code.hpp:37
std::size_t get_sgreet_packet_size(const auth_method *auth_methods, std::size_t num_auth_methods)
Definition: detail/client.hpp:37
Definition: detail/client.hpp:122