TDME2 1.9.121
HTTPDownloadClient.cpp
Go to the documentation of this file.
2
3#include <fstream>
4#include <string>
5#include <vector>
6
7#include <tdme/tdme.h>
8#include <tdme/math/Math.h>
24
25using std::ifstream;
26using std::ios;
27using std::ofstream;
28using std::string;
29using std::to_string;
30using std::vector;
31
48
50
51HTTPDownloadClient::HTTPDownloadClient(): downloadThreadMutex("downloadthread-mutex") {
52}
53
54string HTTPDownloadClient::createHTTPRequestHeaders(const string& hostName, const string& relativeUrl) {
55 auto request =
56 string("GET " + relativeUrl + " HTTP/1.1\r\n") +
57 string("User-Agent: tdme2-httpdownloadclient\r\n") +
58 string("Host: " + hostName + "\r\n") +
59 string("Connection: close\r\n");
60 if (username.empty() == false || password.empty() == false) {
61 string base64Pass;
62 Base64::encode(username + ":" + password, base64Pass);
63 request+= "Authorization: Basic " + base64Pass + "\r\n";
64 }
65 request+=
66 string("\r\n");
67 return request;
68}
69
70uint64_t HTTPDownloadClient::parseHTTPResponseHeaders(ifstream& rawResponse, int16_t& httpStatusCode, vector<string>& httpHeader) {
71 httpHeader.clear();
72 string line;
73 uint64_t headerSize = 0;
74 uint64_t returnHeaderSize = 0;
75 char lastChar = -1;
76 char currentChar;
77 while (rawResponse.eof() == false) {
78 rawResponse.get(currentChar);
79 headerSize++;
80 if (lastChar == '\r' && currentChar == '\n') {
81 if (line.size() != 0) {
82 httpHeader.push_back(line);
83 }
84 if (line.size() == 0) {
85 returnHeaderSize = headerSize;
86 break;
87 }
88 line.clear();
89 } else
90 if (currentChar != '\r' && currentChar != '\n') {
91 line+= currentChar;
92 }
93 lastChar = currentChar;
94 }
95 if (httpHeader.size() > 0) {
97 t.tokenize(httpHeader[0], " ");
98 for (auto i = 0; i < 3 && t.hasMoreTokens(); i++) {
99 auto token = t.nextToken();
100 if (i == 1) {
101 httpStatusCode = Integer::parse(token);
102 }
103 }
104 }
105 return returnHeaderSize;
106}
107
109 url.clear();
110 file.clear();
111 httpStatusCode = -1;
112 httpHeader.clear();
113 haveHeaders = false;
114 haveContentSize = false;
115 headerSize = 0LL;
116 contentSize = 0LL;
117 finished = true;
118 progress = 0.0f;
119}
120
122 class DownloadThread: public Thread {
123 public:
124 DownloadThread(HTTPDownloadClient* downloadClient): Thread("download-thread"), downloadClient(downloadClient) {
125 }
126 void run() {
127 downloadClient->finished = false;
128 downloadClient->progress = 0.0f;
129 TCPSocket socket;
130 try {
131 if (StringTools::startsWith(downloadClient->url, "http://") == false) throw HTTPClientException("Invalid protocol");
132 auto relativeUrl = StringTools::substring(downloadClient->url, string("http://").size());
133 if (relativeUrl.size() == 0) throw HTTPClientException("No URL given");
134 auto slashIdx = relativeUrl.find('/');
135 auto hostName = relativeUrl;
136 if (slashIdx != -1) hostName = StringTools::substring(relativeUrl, 0, slashIdx);
137 relativeUrl = StringTools::substring(relativeUrl, hostName.size());
138
139 Console::println("HTTPDownloadClient::execute(): Hostname: " + hostName);
140 Console::println("HTTPDownloadClient::execute(): RelativeUrl: " + relativeUrl);
141 Console::print("HTTPDownloadClient::execute(): Resolving name to IP: " + hostName + ": ");
142 auto ip = Network::getIpByHostName(hostName);
143 if (ip.size() == 0) {
144 Console::println("HTTPDownloadClient::execute(): Failed");
145 throw HTTPClientException("Could not resolve host IP by host name");
146 }
147 Console::println(ip);
148
149 // socket
150 TCPSocket::create(socket, TCPSocket::determineIpVersion(ip));
151 socket.connect(ip, 80);
152 auto request = downloadClient->createHTTPRequestHeaders(hostName, relativeUrl);
153 socket.write((void*)request.data(), request.length());
154
155 {
156 // output file stream
157 ofstream ofs((downloadClient->file + ".download").c_str(), ofstream::binary);
158 if (ofs.is_open() == false) {
159 throw HTTPClientException("Unable to open file for writing(" + to_string(errno) + "): " + (downloadClient->file + ".download"));
160 }
161
162 // download
163 char rawResponseBuf[16384];
164 auto rawResponseBytesRead = 0;
165 uint64_t bytesRead = 0;
166 try {
167 for (;isStopRequested() == false;) {
168 auto rawResponseBytesRead = socket.read(rawResponseBuf, sizeof(rawResponseBuf));
169 ofs.write(rawResponseBuf, rawResponseBytesRead);
170 if (downloadClient->haveHeaders == false) {
171 // flush download file to disk
172 ofs.flush();
173 // input file stream
174 ifstream ifs((downloadClient->file + ".download").c_str(), ofstream::binary);
175 if (ifs.is_open() == false) {
176 throw HTTPClientException("Unable to open file for reading(" + to_string(errno) + "): " + (downloadClient->file + ".download"));
177 }
178 // try to read headers
179 downloadClient->httpHeader.clear();
180 if ((downloadClient->headerSize = downloadClient->parseHTTPResponseHeaders(ifs, downloadClient->httpStatusCode, downloadClient->httpHeader)) > 0) {
181 downloadClient->haveHeaders = true;
182 for (auto header: downloadClient->httpHeader) {
183 if (StringTools::startsWith(header, "Content-Length: ") == true) {
184 downloadClient->haveContentSize = true;
185 downloadClient->contentSize = Integer::parse(StringTools::substring(header, string("Content-Length: ").size()));
186 }
187 }
188 }
189 ifs.close();
190 }
191 bytesRead+= rawResponseBytesRead;
192 if (downloadClient->haveHeaders == true && downloadClient->haveContentSize == true) {
193 downloadClient->progress = static_cast<float>(bytesRead - downloadClient->headerSize) / static_cast<float>(downloadClient->contentSize);
194 }
195 };
196 } catch (NetworkSocketClosedException& sce) {
197 // end of stream
198 }
199
200 // close download file
201 ofs.close();
202 }
203
204 // transfer to real file
205 if (downloadClient->httpStatusCode == 200 && isStopRequested() == false) {
206 // input file stream
207 ifstream ifs((downloadClient->file + ".download").c_str(), ofstream::binary);
208 if (ifs.is_open() == false) {
209 throw HTTPClientException("Unable to open file for reading(" + to_string(errno) + "): " + (downloadClient->file + ".download"));
210 }
211
212 ifs.seekg(downloadClient->headerSize, ios::beg);
213 auto ifsHeaderSize = ifs.tellg();
214 ifs.seekg(0, ios::end);
215 auto ifsSizeTotal = ifs.tellg();
216 auto ifsSize = ifsSizeTotal - ifsHeaderSize;
217 ifs.seekg(ifsHeaderSize, ios::beg);
218
219 // output file stream
220 ofstream ofs(downloadClient->file.c_str(), ofstream::binary);
221 if (ofs.is_open() == false) {
222 throw HTTPClientException("Unable to open file for writing(" + to_string(errno) + "): " + downloadClient->file);
223 }
224
225 //
226 char buf[16384];
227 auto ifsBytesToRead = 0;
228 auto ifsBytesRead = 0;
229 do {
230 auto ifsBytesToRead = Math::min(static_cast<int64_t>(ifsSize - ifsBytesRead), sizeof(buf));
231 ifs.read(buf, ifsBytesToRead);
232 ofs.write(buf, ifsBytesToRead);
233 ifsBytesRead+= ifsBytesToRead;
234 } while (ifsBytesRead < ifsSize);
235
236 // close target file
237 ofs.close();
238
239 // close download file
240 ifs.close();
241 }
242
243 //
244 FileSystem::getStandardFileSystem()->removeFile(".", downloadClient->file + ".download");
245
246 //
247 socket.shutdown();
248
249 //
250 downloadClient->finished = true;
251 downloadClient->progress = 1.0f;
252 } catch (Exception& exception) {
253 socket.shutdown();
254 downloadClient->finished = true;
255 Console::println(string("HTTPDownloadClient::execute(): performed HTTP request: FAILED: ") + exception.what());
256 }
257 }
258 private:
259 HTTPDownloadClient* downloadClient;
260 };
262 finished = false;
263 this->downloadThread = new DownloadThread(this);
264 this->downloadThread->start();
266}
267
270 if (downloadThread != nullptr) downloadThread->stop();
272}
273
276 if (downloadThread != nullptr) {
278 delete this->downloadThread;
279 this->downloadThread = nullptr;
280 }
282}
Standard math functions.
Definition: Math.h:21
void start()
Starts the HTTP download to file.
void join()
Wait until underlying thread has finished.
uint64_t parseHTTPResponseHeaders(ifstream &rawResponse, int16_t &httpStatusCode, vector< string > &httpHeader)
Parse HTTP response headers.
string createHTTPRequestHeaders(const string &hostName, const string &relativeUrl)
Create HTTP request headers.
File system singleton class.
Definition: FileSystem.h:14
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
Mutex implementation.
Definition: Mutex.h:27
void unlock()
Unlocks this mutex.
Definition: Mutex.cpp:48
void lock()
Locks the mutex, additionally mutex locks will block until other locks have been unlocked.
Definition: Mutex.cpp:39
Base class for threads.
Definition: Thread.h:26
void start()
Starts this objects thread.
Definition: Thread.cpp:65
void join()
Blocks caller thread until this thread has been terminated.
Definition: Thread.cpp:55
void stop()
Requests that this thread should be stopped.
Definition: Thread.cpp:94
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