1 //
2 // sync_client.cpp
3 // ~~~~~~~~~~~~~~~
4 //
5 // Copyright (c) 2003-2020 Christopher M. Kohlhoff (chris at kohlhoff dot com)
6 //
7 // Distributed under the Boost Software License, Version 1.0. (See accompanying
8 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
9 //
10
11 #include <array>
12 #include <iostream>
13 #include <iomanip>
14 #include <ostream>
15 #include <string>
16 #include <boost/asio.hpp>
17 #include "socks4.hpp"
18
19 using boost::asio::ip::tcp;
20
main(int argc,char * argv[])21 int main(int argc, char* argv[])
22 {
23 try
24 {
25 if (argc != 4)
26 {
27 std::cout << "Usage: sync_client <socks4server> <socks4port> <user>\n";
28 std::cout << "Examples:\n";
29 std::cout << " sync_client 127.0.0.1 1080 chris\n";
30 std::cout << " sync_client localhost socks chris\n";
31 return 1;
32 }
33
34 boost::asio::io_context io_context;
35
36 // Get a list of endpoints corresponding to the SOCKS 4 server name.
37 tcp::resolver resolver(io_context);
38 auto endpoints = resolver.resolve(argv[1], argv[2]);
39
40 // Try each endpoint until we successfully establish a connection to the
41 // SOCKS 4 server.
42 tcp::socket socket(io_context);
43 boost::asio::connect(socket, endpoints);
44
45 // Get an endpoint for the Boost website. This will be passed to the SOCKS
46 // 4 server. Explicitly specify IPv4 since SOCKS 4 does not support IPv6.
47 auto http_endpoint =
48 *resolver.resolve(tcp::v4(), "www.boost.org", "http").begin();
49
50 // Send the request to the SOCKS 4 server.
51 socks4::request socks_request(
52 socks4::request::connect, http_endpoint, argv[3]);
53 boost::asio::write(socket, socks_request.buffers());
54
55 // Receive a response from the SOCKS 4 server.
56 socks4::reply socks_reply;
57 boost::asio::read(socket, socks_reply.buffers());
58
59 // Check whether we successfully negotiated with the SOCKS 4 server.
60 if (!socks_reply.success())
61 {
62 std::cout << "Connection failed.\n";
63 std::cout << "status = 0x" << std::hex << socks_reply.status();
64 return 1;
65 }
66
67 // Form the HTTP request. We specify the "Connection: close" header so that
68 // the server will close the socket after transmitting the response. This
69 // will allow us to treat all data up until the EOF as the response.
70 std::string request =
71 "GET / HTTP/1.0\r\n"
72 "Host: www.boost.org\r\n"
73 "Accept: */*\r\n"
74 "Connection: close\r\n\r\n";
75
76 // Send the HTTP request.
77 boost::asio::write(socket, boost::asio::buffer(request));
78
79 // Read until EOF, writing data to output as we go.
80 std::array<char, 512> response;
81 boost::system::error_code error;
82 while (std::size_t s = socket.read_some(
83 boost::asio::buffer(response), error))
84 std::cout.write(response.data(), s);
85 if (error != boost::asio::error::eof)
86 throw std::system_error(error);
87 }
88 catch (std::exception& e)
89 {
90 std::cout << "Exception: " << e.what() << "\n";
91 }
92
93 return 0;
94 }
95