TDME2 1.9.121
HTTPClient.cpp
Go to the documentation of this file.
2
3#include <iomanip>
4#include <sstream>
5#include <string>
6#include <unordered_map>
7#include <vector>
8
9#include <tdme/tdme.h>
21
22using std::hex;
23using std::nouppercase;
24using std::ostringstream;
25using std::setw;
26using std::string;
27using std::stringstream;
28using std::to_string;
29using std::unordered_map;
30using std::uppercase;
31using std::vector;
32
44
46
47const string HTTPClient::HTTP_METHOD_GET = "GET";
48const string HTTPClient::HTTP_METHOD_HEAD = "HEAD";
49const string HTTPClient::HTTP_METHOD_POST = "POST";
50const string HTTPClient::HTTP_METHOD_PUT = "PUT";
51const string HTTPClient::HTTP_METHOD_DELETE = "DELETE";
52
53string HTTPClient::urlEncode(const string &value) {
54 // see: https://stackoverflow.com/questions/154536/encode-decode-urls-in-c
55 ostringstream escaped;
56 escaped.fill('0');
57 escaped << hex;
58
59 for (string::const_iterator i = value.begin(), n = value.end(); i != n; ++i) {
60 string::value_type c = (*i);
61
62 // Keep alphanumeric and other accepted characters intact
63 if (Character::isAlphaNumeric(c) == true || c == '-' || c == '_' || c == '.' || c == '~') {
64 escaped << c;
65 continue;
66 }
67
68 // Any other characters are percent-encoded
69 escaped << uppercase;
70 escaped << '%' << setw(2) << int((unsigned char) c);
71 escaped << nouppercase;
72 }
73
74 return escaped.str();
75}
76
77
78string HTTPClient::createHTTPRequestHeaders(const string& hostName, const string& method, const string& relativeUrl, const unordered_map<string, string>& getParameters, const unordered_map<string, string>& postParameters, const string& body) {
79 string query;
80 for (auto getParameterIt: getParameters) {
81 if (query.size() == 0) query+= "?"; else query+="&";
82 query+= urlEncode(getParameterIt.first) + "=" + urlEncode(getParameterIt.second);
83 }
84 auto request =
85 string(method + " " + relativeUrl + query + " HTTP/1.1\r\n") +
86 string("User-Agent: tdme2-httpclient\r\n") +
87 string("Host: " + hostName + "\r\n") +
88 string("Connection: close\r\n");
89 if (username.empty() == false || password.empty() == false) {
90 string base64Pass;
91 Base64::encode(username + ":" + password, base64Pass);
92 request+= "Authorization: Basic " + base64Pass + "\r\n";
93 }
94 if (contentType.size() > 0) {
95 request+=
96 string("Content-Type: " + contentType + "\r\n");
97 }
99 string _body;
100 if (postParameters.size() > 0) {
101 for (auto postParameterIt: postParameters) {
102 if (_body.size() >= 0) _body+="&";
103 _body+= urlEncode(postParameterIt.first) + "=" + urlEncode(postParameterIt.second);
104 }
105 } else {
106 _body = body;
107 }
108 request+= "Content-Length: " + to_string(_body.size()) + "\r\n\r\n";
109 request+= _body;
110 } else {
111 request+= "\r\n";
112 }
113 return request;
114}
115
116void HTTPClient::parseHTTPResponseHeaders(stringstream& rawResponse, int16_t& httpStatusCode, vector<string>& httpHeader) {
117 string line;
118 char lastChar = -1;
119 char currentChar;
120 while (rawResponse.eof() == false) {
121 rawResponse.get(currentChar);
122 if (lastChar == '\r' && currentChar == '\n') {
123 if (line.size() != 0) {
124 httpHeader.push_back(line);
125 }
126 if (line.size() == 0) break;
127 line.clear();
128 } else
129 if (currentChar != '\r' && currentChar != '\n') {
130 line+= currentChar;
131 }
132 lastChar = currentChar;
133 }
134 if (httpHeader.size() > 0) {
136 t.tokenize(httpHeader[0], " ");
137 for (auto i = 0; i < 3 && t.hasMoreTokens(); i++) {
138 auto token = t.nextToken();
139 if (i == 1) {
140 httpStatusCode = Integer::parse(token);
141 }
142 }
143 }
144}
145
147 url.clear();
148 method.clear();
149 getParameters.clear();
150 postParameters.clear();
151 body.clear();
152 contentType.clear();
153
154 rawResponse.clear();
155 httpStatusCode = -1;
156 httpHeader.clear();
157}
158
160 TCPSocket socket;
161 try {
162 if (StringTools::startsWith(url, "http://") == false) throw HTTPClientException("Invalid protocol");
163 auto relativeUrl = StringTools::substring(url, string("http://").size());
164 if (relativeUrl.size() == 0) throw HTTPClientException("No URL given");
165 auto slashIdx = relativeUrl.find('/');
166 auto hostName = relativeUrl;
167 if (slashIdx != -1) hostName = StringTools::substring(relativeUrl, 0, slashIdx);
168 relativeUrl = StringTools::substring(relativeUrl, hostName.size());
169
170 Console::println("HTTPClient::execute(): Hostname: " + hostName);
171 Console::println("HTTPClient::execute(): RelativeUrl: " + relativeUrl);
172
173 Console::print("HTTPClient::execute(): Resolving name to IP: " + hostName + ": ");
174 auto ip = Network::getIpByHostName(hostName);
175 if (ip.size() == 0) {
176 Console::println("HTTPClient::execute(): Failed");
177 throw HTTPClientException("Could not resolve host IP by host name");
178 }
179 Console::println(ip);
180
181 TCPSocket::create(socket, TCPSocket::determineIpVersion(ip));
182 socket.connect(ip, 80);
183 auto request = createHTTPRequestHeaders(hostName, method, relativeUrl, getParameters, postParameters, body);
184 socket.write((void*)request.data(), request.length());
185
186 char rawResponseBuf[16384];
187 auto rawResponseBytesRead = 0;
188 try {
189 for (;true;) {
190 auto rawResponseBytesRead = socket.read(rawResponseBuf, sizeof(rawResponseBuf));
191 rawResponse.write(rawResponseBuf, rawResponseBytesRead);
192 };
193 } catch (NetworkSocketClosedException& sce) {
194 // end of stream
195 }
196
198
199 Console::println("HTTPClient::execute(): performed HTTP request: HTTP status code: " + to_string(httpStatusCode));
200
201 socket.shutdown();
202 } catch (Exception& exception) {
203 socket.shutdown();
204 Console::println(string("HTTPClient::execute(): performed HTTP request: FAILED: ") + exception.what());
205 // rethrow
206 throw;
207 }
208
209}
void execute()
Execute HTTP request.
Definition: HTTPClient.cpp:159
static const string HTTP_METHOD_POST
Definition: HTTPClient.h:65
static const string HTTP_METHOD_PUT
Definition: HTTPClient.h:66
unordered_map< string, string > getParameters
Definition: HTTPClient.h:29
void parseHTTPResponseHeaders(stringstream &rawResponse, int16_t &httpStatusCode, vector< string > &httpHeader)
Parse HTTP response headers.
Definition: HTTPClient.cpp:116
string createHTTPRequestHeaders(const string &hostName, const string &method, const string &relativeUrl, const unordered_map< string, string > &getParameters, const unordered_map< string, string > &postParameters, const string &body)
Create HTTP request headers.
Definition: HTTPClient.cpp:78
unordered_map< string, string > postParameters
Definition: HTTPClient.h:30
static string urlEncode(const string &value)
Returns a URL encoded representation of value.
Definition: HTTPClient.cpp:53
void reset()
Reset this HTTP client.
Definition: HTTPClient.cpp:146
void shutdown()
shuts socket down for reading and writing
Network class.
Definition: Network.h:14
Class representing a TCP socket.
Definition: TCPSocket.h:15
size_t read(void *buf, const size_t bytes)
Reads up to "bytes" bytes from socket.
Definition: TCPSocket.cpp:46
size_t write(void *buf, const size_t bytes)
Writes up to "bytes" bytes to socket.
Definition: TCPSocket.cpp:60
void connect(const string &ip, const unsigned int port)
Connects a socket to given IP and port.
Definition: TCPSocket.cpp:99
Base64 encoding/decoding class.
Definition: Base64.h:16
Character class.
Definition: Character.h:15
Console class.
Definition: Console.h:26
Integer class.
Definition: Integer.h:26
String tokenizer class.
void tokenize(const string &str, const string &delimiters)
Tokenize.
String tools class.
Definition: StringTools.h:20
std::exception Exception
Exception base class.
Definition: Exception.h:19