blob: 988599e428672a59e09fab2bc4fcc2558bf80d70 [file] [log] [blame]
Mike Frysinger8155d082012-04-06 15:23:18 -04001// Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
adlr@google.com3defe6a2009-12-04 20:57:17 +00002// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5// This file implements a simple HTTP server. It can exhibit odd behavior
6// that's useful for testing. For example, it's useful to test that
7// the updater can continue a connection if it's dropped, or that it
8// handles very slow data transfers.
9
10// To use this, simply make an HTTP connection to localhost:port and
11// GET a url.
12
Gilad Arnold97bdb3f2013-06-30 23:05:50 -070013#include <err.h>
adlr@google.com3defe6a2009-12-04 20:57:17 +000014#include <errno.h>
Gilad Arnoldb6c562a2013-07-01 02:19:26 -070015#include <fcntl.h>
Andrew de los Reyes08c4e272010-04-15 14:02:17 -070016#include <inttypes.h>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070017#include <netinet/in.h>
18#include <signal.h>
adlr@google.com3defe6a2009-12-04 20:57:17 +000019#include <stdio.h>
20#include <stdlib.h>
21#include <string.h>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070022#include <sys/socket.h>
Gilad Arnoldb6c562a2013-07-01 02:19:26 -070023#include <sys/stat.h>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070024#include <sys/types.h>
adlr@google.com3defe6a2009-12-04 20:57:17 +000025#include <unistd.h>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070026
adlr@google.com3defe6a2009-12-04 20:57:17 +000027#include <algorithm>
28#include <string>
29#include <vector>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070030
31#include <base/logging.h>
Alex Vakulenko75039d72014-03-25 12:36:28 -070032#include <base/strings/string_split.h>
33#include <base/strings/string_util.h>
34#include <base/strings/stringprintf.h>
Gilad Arnold9bedeb52011-11-17 16:19:57 -080035
36#include "update_engine/http_common.h"
Gilad Arnold9bedeb52011-11-17 16:19:57 -080037
adlr@google.com3defe6a2009-12-04 20:57:17 +000038
Gilad Arnolde4ad2502011-12-29 17:08:54 -080039// HTTP end-of-line delimiter; sorry, this needs to be a macro.
40#define EOL "\r\n"
41
adlr@google.com3defe6a2009-12-04 20:57:17 +000042using std::string;
43using std::vector;
44
Gilad Arnold9bedeb52011-11-17 16:19:57 -080045
adlr@google.com3defe6a2009-12-04 20:57:17 +000046namespace chromeos_update_engine {
47
Gilad Arnoldb6c562a2013-07-01 02:19:26 -070048static const char* kListeningMsgPrefix = "listening on port ";
Gilad Arnold97bdb3f2013-06-30 23:05:50 -070049
50enum {
51 RC_OK = 0,
52 RC_BAD_ARGS,
53 RC_ERR_READ,
54 RC_ERR_SETSOCKOPT,
55 RC_ERR_BIND,
Gilad Arnoldb6c562a2013-07-01 02:19:26 -070056 RC_ERR_LISTEN,
57 RC_ERR_GETSOCKNAME,
58 RC_ERR_REPORT,
Gilad Arnold97bdb3f2013-06-30 23:05:50 -070059};
60
adlr@google.com3defe6a2009-12-04 20:57:17 +000061struct HttpRequest {
Gilad Arnolde4ad2502011-12-29 17:08:54 -080062 HttpRequest()
63 : start_offset(0), end_offset(0), return_code(kHttpResponseOk) {}
Darin Petkov41c2fcf2010-08-25 13:14:48 -070064 string host;
adlr@google.com3defe6a2009-12-04 20:57:17 +000065 string url;
Gilad Arnold9bedeb52011-11-17 16:19:57 -080066 off_t start_offset;
Gilad Arnolde4ad2502011-12-29 17:08:54 -080067 off_t end_offset; // non-inclusive, zero indicates unspecified.
Gilad Arnold9bedeb52011-11-17 16:19:57 -080068 HttpResponseCode return_code;
adlr@google.com3defe6a2009-12-04 20:57:17 +000069};
70
adlr@google.com3defe6a2009-12-04 20:57:17 +000071bool ParseRequest(int fd, HttpRequest* request) {
72 string headers;
Gilad Arnold9bedeb52011-11-17 16:19:57 -080073 do {
74 char buf[1024];
75 ssize_t r = read(fd, buf, sizeof(buf));
adlr@google.com3defe6a2009-12-04 20:57:17 +000076 if (r < 0) {
77 perror("read");
Gilad Arnold97bdb3f2013-06-30 23:05:50 -070078 exit(RC_ERR_READ);
adlr@google.com3defe6a2009-12-04 20:57:17 +000079 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -080080 headers.append(buf, r);
Alex Vakulenko6a9d3492015-06-15 12:53:22 -070081 } while (!base::EndsWith(headers, EOL EOL, true));
adlr@google.com3defe6a2009-12-04 20:57:17 +000082
Gilad Arnold9bedeb52011-11-17 16:19:57 -080083 LOG(INFO) << "got headers:\n--8<------8<------8<------8<----\n"
84 << headers
85 << "\n--8<------8<------8<------8<----";
adlr@google.com3defe6a2009-12-04 20:57:17 +000086
Gilad Arnold9bedeb52011-11-17 16:19:57 -080087 // Break header into lines.
Alex Deymof329b932014-10-30 01:37:48 -070088 vector<string> lines;
Gilad Arnold9bedeb52011-11-17 16:19:57 -080089 base::SplitStringUsingSubstr(
90 headers.substr(0, headers.length() - strlen(EOL EOL)), EOL, &lines);
adlr@google.com3defe6a2009-12-04 20:57:17 +000091
Gilad Arnold9bedeb52011-11-17 16:19:57 -080092 // Decode URL line.
Alex Deymof329b932014-10-30 01:37:48 -070093 vector<string> terms;
Gilad Arnold9bedeb52011-11-17 16:19:57 -080094 base::SplitStringAlongWhitespace(lines[0], &terms);
Mike Frysinger0f9547d2012-02-16 12:11:37 -050095 CHECK_EQ(terms.size(), static_cast<vector<string>::size_type>(3));
Gilad Arnold9bedeb52011-11-17 16:19:57 -080096 CHECK_EQ(terms[0], "GET");
97 request->url = terms[1];
98 LOG(INFO) << "URL: " << request->url;
adlr@google.com3defe6a2009-12-04 20:57:17 +000099
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800100 // Decode remaining lines.
101 size_t i;
102 for (i = 1; i < lines.size(); i++) {
Alex Deymof329b932014-10-30 01:37:48 -0700103 vector<string> terms;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800104 base::SplitStringAlongWhitespace(lines[i], &terms);
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700105
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800106 if (terms[0] == "Range:") {
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500107 CHECK_EQ(terms.size(), static_cast<vector<string>::size_type>(2));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800108 string &range = terms[1];
109 LOG(INFO) << "range attribute: " << range;
Alex Vakulenko6a9d3492015-06-15 12:53:22 -0700110 CHECK(base::StartsWithASCII(range, "bytes=", true) &&
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800111 range.find('-') != string::npos);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800112 request->start_offset = atoll(range.c_str() + strlen("bytes="));
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800113 // Decode end offset and increment it by one (so it is non-inclusive).
114 if (range.find('-') < range.length() - 1)
115 request->end_offset = atoll(range.c_str() + range.find('-') + 1) + 1;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800116 request->return_code = kHttpResponsePartialContent;
Alex Deymof329b932014-10-30 01:37:48 -0700117 string tmp_str = base::StringPrintf("decoded range offsets: "
Alex Vakulenko75039d72014-03-25 12:36:28 -0700118 "start=%jd end=",
119 (intmax_t)request->start_offset);
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800120 if (request->end_offset > 0)
121 base::StringAppendF(&tmp_str, "%jd (non-inclusive)",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700122 (intmax_t)request->end_offset);
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800123 else
124 base::StringAppendF(&tmp_str, "unspecified");
125 LOG(INFO) << tmp_str;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800126 } else if (terms[0] == "Host:") {
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500127 CHECK_EQ(terms.size(), static_cast<vector<string>::size_type>(2));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800128 request->host = terms[1];
129 LOG(INFO) << "host attribute: " << request->host;
130 } else {
131 LOG(WARNING) << "ignoring HTTP attribute: `" << lines[i] << "'";
adlr@google.com3defe6a2009-12-04 20:57:17 +0000132 }
adlr@google.com3defe6a2009-12-04 20:57:17 +0000133 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800134
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -0700135 return true;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000136}
137
138string Itoa(off_t num) {
139 char buf[100] = {0};
Andrew de los Reyes08c4e272010-04-15 14:02:17 -0700140 snprintf(buf, sizeof(buf), "%" PRIi64, num);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000141 return buf;
142}
143
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800144// Writes a string into a file. Returns total number of bytes written or -1 if a
145// write error occurred.
146ssize_t WriteString(int fd, const string& str) {
147 const size_t total_size = str.size();
148 size_t remaining_size = total_size;
149 char const *data = str.data();
Gilad Arnold48085ba2011-11-16 09:36:08 -0800150
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800151 while (remaining_size) {
152 ssize_t written = write(fd, data, remaining_size);
153 if (written < 0) {
154 perror("write");
155 LOG(INFO) << "write failed";
156 return -1;
157 }
158 data += written;
159 remaining_size -= written;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000160 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800161
162 return total_size;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000163}
164
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800165// Writes the headers of an HTTP response into a file.
166ssize_t WriteHeaders(int fd, const off_t start_offset, const off_t end_offset,
167 HttpResponseCode return_code) {
168 ssize_t written = 0, ret;
169
170 ret = WriteString(fd,
171 string("HTTP/1.1 ") + Itoa(return_code) + " " +
172 GetHttpResponseDescription(return_code) +
173 EOL
174 "Content-Type: application/octet-stream" EOL);
175 if (ret < 0)
176 return -1;
177 written += ret;
178
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800179 // Compute content legnth.
180 const off_t content_length = end_offset - start_offset;;
181
182 // A start offset that equals the end offset indicates that the response
183 // should contain the full range of bytes in the requested resource.
184 if (start_offset || start_offset == end_offset) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800185 ret = WriteString(fd,
186 string("Accept-Ranges: bytes" EOL
187 "Content-Range: bytes ") +
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800188 Itoa(start_offset == end_offset ? 0 : start_offset) +
189 "-" + Itoa(end_offset - 1) + "/" + Itoa(end_offset) +
190 EOL);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800191 if (ret < 0)
192 return -1;
193 written += ret;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800194 }
195
196 ret = WriteString(fd, string("Content-Length: ") + Itoa(content_length) +
197 EOL EOL);
198 if (ret < 0)
199 return -1;
200 written += ret;
201
202 return written;
203}
204
205// Writes a predetermined payload of lines of ascending bytes to a file. The
206// first byte of output is appropriately offset with respect to the request line
207// length. Returns the number of successfully written bytes.
208size_t WritePayload(int fd, const off_t start_offset, const off_t end_offset,
209 const char first_byte, const size_t line_len) {
210 CHECK_LE(start_offset, end_offset);
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500211 CHECK_GT(line_len, static_cast<size_t>(0));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800212
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800213 LOG(INFO) << "writing payload: " << line_len << "-byte lines starting with `"
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800214 << first_byte << "', offset range " << start_offset << " -> "
215 << end_offset;
216
217 // Populate line of ascending characters.
218 string line;
219 line.reserve(line_len);
220 char byte = first_byte;
221 size_t i;
222 for (i = 0; i < line_len; i++)
223 line += byte++;
224
225 const size_t total_len = end_offset - start_offset;
226 size_t remaining_len = total_len;
227 bool success = true;
228
229 // If start offset is not aligned with line boundary, output partial line up
230 // to the first line boundary.
231 size_t start_modulo = start_offset % line_len;
232 if (start_modulo) {
233 string partial = line.substr(start_modulo, remaining_len);
234 ssize_t ret = WriteString(fd, partial);
235 if ((success = (ret >= 0 && (size_t) ret == partial.length())))
236 remaining_len -= partial.length();
237 }
238
239 // Output full lines up to the maximal line boundary below the end offset.
240 while (success && remaining_len >= line_len) {
241 ssize_t ret = WriteString(fd, line);
242 if ((success = (ret >= 0 && (size_t) ret == line_len)))
243 remaining_len -= line_len;
244 }
245
246 // Output a partial line up to the end offset.
247 if (success && remaining_len) {
248 string partial = line.substr(0, remaining_len);
249 ssize_t ret = WriteString(fd, partial);
250 if ((success = (ret >= 0 && (size_t) ret == partial.length())))
251 remaining_len -= partial.length();
252 }
253
254 return (total_len - remaining_len);
255}
256
257// Write default payload lines of the form 'abcdefghij'.
258inline size_t WritePayload(int fd, const off_t start_offset,
259 const off_t end_offset) {
260 return WritePayload(fd, start_offset, end_offset, 'a', 10);
261}
262
263// Send an empty response, then kill the server.
264void HandleQuit(int fd) {
265 WriteHeaders(fd, 0, 0, kHttpResponseOk);
Jay Srinivasan135a58b2012-07-13 12:46:49 -0700266 LOG(INFO) << "pid(" << getpid() << "): HTTP server exiting ...";
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700267 exit(RC_OK);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000268}
269
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800270
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800271// Generates an HTTP response with payload corresponding to requested offsets
272// and length. Optionally, truncate the payload at a given length and add a
273// pause midway through the transfer. Returns the total number of bytes
274// delivered or -1 for error.
275ssize_t HandleGet(int fd, const HttpRequest& request, const size_t total_length,
276 const size_t truncate_length, const int sleep_every,
277 const int sleep_secs) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800278 ssize_t ret;
279 size_t written = 0;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000280
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800281 // Obtain start offset, make sure it is within total payload length.
282 const size_t start_offset = request.start_offset;
283 if (start_offset >= total_length) {
284 LOG(WARNING) << "start offset (" << start_offset
285 << ") exceeds total length (" << total_length
286 << "), generating error response ("
287 << kHttpResponseReqRangeNotSat << ")";
288 return WriteHeaders(fd, total_length, total_length,
289 kHttpResponseReqRangeNotSat);
290 }
291
292 // Obtain end offset, adjust to fit in total payload length and ensure it does
293 // not preceded the start offset.
294 size_t end_offset = (request.end_offset > 0 ?
295 request.end_offset : total_length);
296 if (end_offset < start_offset) {
297 LOG(WARNING) << "end offset (" << end_offset << ") precedes start offset ("
298 << start_offset << "), generating error response";
299 return WriteHeaders(fd, 0, 0, kHttpResponseBadRequest);
300 }
301 if (end_offset > total_length) {
302 LOG(INFO) << "requested end offset (" << end_offset
303 << ") exceeds total length (" << total_length << "), adjusting";
304 end_offset = total_length;
305 }
306
307 // Generate headers
308 LOG(INFO) << "generating response header: range=" << start_offset << "-"
309 << (end_offset - 1) << "/" << (end_offset - start_offset)
310 << ", return code=" << request.return_code;
311 if ((ret = WriteHeaders(fd, start_offset, end_offset,
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800312 request.return_code)) < 0)
313 return -1;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800314 LOG(INFO) << ret << " header bytes written";
315 written += ret;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800316
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800317 // Compute payload length, truncate as necessary.
318 size_t payload_length = end_offset - start_offset;
319 if (truncate_length > 0 && truncate_length < payload_length) {
320 LOG(INFO) << "truncating request payload length (" << payload_length
321 << ") at " << truncate_length;
322 payload_length = truncate_length;
323 end_offset = start_offset + payload_length;
324 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800325
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800326 LOG(INFO) << "generating response payload: range=" << start_offset << "-"
327 << (end_offset - 1) << "/" << (end_offset - start_offset);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800328
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800329 // Decide about optional midway delay.
330 if (truncate_length > 0 && sleep_every > 0 && sleep_secs >= 0 &&
331 start_offset % (truncate_length * sleep_every) == 0) {
332 const off_t midway_offset = start_offset + payload_length / 2;
333
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800334 if ((ret = WritePayload(fd, start_offset, midway_offset)) < 0)
335 return -1;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800336 LOG(INFO) << ret << " payload bytes written (first chunk)";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800337 written += ret;
338
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800339 LOG(INFO) << "sleeping for " << sleep_secs << " seconds...";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800340 sleep(sleep_secs);
341
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800342 if ((ret = WritePayload(fd, midway_offset, end_offset)) < 0)
343 return -1;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800344 LOG(INFO) << ret << " payload bytes written (second chunk)";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800345 written += ret;
346 } else {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800347 if ((ret = WritePayload(fd, start_offset, end_offset)) < 0)
348 return -1;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800349 LOG(INFO) << ret << " payload bytes written";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800350 written += ret;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000351 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800352
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800353 LOG(INFO) << "response generation complete, " << written
354 << " total bytes written";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800355 return written;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000356}
357
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800358ssize_t HandleGet(int fd, const HttpRequest& request,
359 const size_t total_length) {
360 return HandleGet(fd, request, total_length, 0, 0, 0);
361}
362
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700363// Handles /redirect/<code>/<url> requests by returning the specified
364// redirect <code> with a location pointing to /<url>.
365void HandleRedirect(int fd, const HttpRequest& request) {
366 LOG(INFO) << "Redirecting...";
367 string url = request.url;
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500368 CHECK_EQ(static_cast<size_t>(0), url.find("/redirect/"));
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700369 url.erase(0, strlen("/redirect/"));
370 string::size_type url_start = url.find('/');
371 CHECK_NE(url_start, string::npos);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800372 HttpResponseCode code = StringToHttpResponseCode(url.c_str());
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700373 url.erase(0, url_start);
374 url = "http://" + request.host + url;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800375 const char *status = GetHttpResponseDescription(code);
376 if (!status)
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700377 CHECK(false) << "Unrecognized redirection code: " << code;
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700378 LOG(INFO) << "Code: " << code << " " << status;
379 LOG(INFO) << "New URL: " << url;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800380
381 ssize_t ret;
382 if ((ret = WriteString(fd, "HTTP/1.1 " + Itoa(code) + " " +
383 status + EOL)) < 0)
384 return;
385 WriteString(fd, "Location: " + url + EOL);
386}
387
388// Generate a page not found error response with actual text payload. Return
389// number of bytes written or -1 for error.
390ssize_t HandleError(int fd, const HttpRequest& request) {
391 LOG(INFO) << "Generating error HTTP response";
392
393 ssize_t ret;
394 size_t written = 0;
395
396 const string data("This is an error page.");
397
398 if ((ret = WriteHeaders(fd, 0, data.size(), kHttpResponseNotFound)) < 0)
399 return -1;
400 written += ret;
401
402 if ((ret = WriteString(fd, data)) < 0)
403 return -1;
404 written += ret;
405
406 return written;
407}
408
409// Generate an error response if the requested offset is nonzero, up to a given
410// maximal number of successive failures. The error generated is an "Internal
411// Server Error" (500).
412ssize_t HandleErrorIfOffset(int fd, const HttpRequest& request,
413 size_t end_offset, int max_fails) {
414 static int num_fails = 0;
415
416 if (request.start_offset > 0 && num_fails < max_fails) {
417 LOG(INFO) << "Generating error HTTP response";
418
419 ssize_t ret;
420 size_t written = 0;
421
422 const string data("This is an error page.");
423
424 if ((ret = WriteHeaders(fd, 0, data.size(),
425 kHttpResponseInternalServerError)) < 0)
426 return -1;
427 written += ret;
428
429 if ((ret = WriteString(fd, data)) < 0)
430 return -1;
431 written += ret;
432
433 num_fails++;
434 return written;
435 } else {
436 num_fails = 0;
437 return HandleGet(fd, request, end_offset);
438 }
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700439}
440
adlr@google.com3defe6a2009-12-04 20:57:17 +0000441void HandleDefault(int fd, const HttpRequest& request) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800442 const off_t start_offset = request.start_offset;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000443 const string data("unhandled path");
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800444 const size_t size = data.size();
445 ssize_t ret;
446
447 if ((ret = WriteHeaders(fd, start_offset, size, request.return_code)) < 0)
448 return;
449 WriteString(fd, (start_offset < static_cast<off_t>(size) ?
450 data.substr(start_offset) : ""));
adlr@google.com3defe6a2009-12-04 20:57:17 +0000451}
452
Gilad Arnold48085ba2011-11-16 09:36:08 -0800453
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800454// Break a URL into terms delimited by slashes.
455class UrlTerms {
456 public:
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700457 UrlTerms(const string &url, size_t num_terms) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800458 // URL must be non-empty and start with a slash.
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500459 CHECK_GT(url.size(), static_cast<size_t>(0));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800460 CHECK_EQ(url[0], '/');
461
462 // Split it into terms delimited by slashes, omitting the preceeding slash.
463 base::SplitStringDontTrim(url.substr(1), '/', &terms);
464
465 // Ensure expected length.
466 CHECK_EQ(terms.size(), num_terms);
467 }
468
469 inline string Get(const off_t index) const {
470 return terms[index];
471 }
472 inline const char *GetCStr(const off_t index) const {
473 return Get(index).c_str();
474 }
475 inline int GetInt(const off_t index) const {
476 return atoi(GetCStr(index));
477 }
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700478 inline size_t GetSizeT(const off_t index) const {
479 return static_cast<size_t>(atol(GetCStr(index)));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800480 }
481
482 private:
Alex Deymof329b932014-10-30 01:37:48 -0700483 vector<string> terms;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800484};
Gilad Arnold48085ba2011-11-16 09:36:08 -0800485
adlr@google.com3defe6a2009-12-04 20:57:17 +0000486void HandleConnection(int fd) {
487 HttpRequest request;
488 ParseRequest(fd, &request);
489
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800490 string &url = request.url;
Jay Srinivasan135a58b2012-07-13 12:46:49 -0700491 LOG(INFO) << "pid(" << getpid() << "): handling url " << url;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800492 if (url == "/quitquitquit") {
493 HandleQuit(fd);
Alex Vakulenko6a9d3492015-06-15 12:53:22 -0700494 } else if (base::StartsWithASCII(url, "/download/", true)) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800495 const UrlTerms terms(url, 2);
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700496 HandleGet(fd, request, terms.GetSizeT(1));
Alex Vakulenko6a9d3492015-06-15 12:53:22 -0700497 } else if (base::StartsWithASCII(url, "/flaky/", true)) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800498 const UrlTerms terms(url, 5);
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700499 HandleGet(fd, request, terms.GetSizeT(1), terms.GetSizeT(2),
500 terms.GetInt(3), terms.GetInt(4));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800501 } else if (url.find("/redirect/") == 0) {
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700502 HandleRedirect(fd, request);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800503 } else if (url == "/error") {
Gilad Arnold48085ba2011-11-16 09:36:08 -0800504 HandleError(fd, request);
Alex Vakulenko6a9d3492015-06-15 12:53:22 -0700505 } else if (base::StartsWithASCII(url, "/error-if-offset/", true)) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800506 const UrlTerms terms(url, 3);
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700507 HandleErrorIfOffset(fd, request, terms.GetSizeT(1), terms.GetInt(2));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800508 } else {
adlr@google.com3defe6a2009-12-04 20:57:17 +0000509 HandleDefault(fd, request);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800510 }
adlr@google.com3defe6a2009-12-04 20:57:17 +0000511
512 close(fd);
513}
514
515} // namespace chromeos_update_engine
516
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700517using namespace chromeos_update_engine; // NOLINT(build/namespaces)
adlr@google.com3defe6a2009-12-04 20:57:17 +0000518
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700519
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700520void usage(const char *prog_arg) {
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700521 fprintf(
522 stderr,
523 "Usage: %s [ FILE ]\n"
524 "Once accepting connections, the following is written to FILE (or "
525 "stdout):\n"
526 "\"%sN\" (where N is an integer port number)\n",
527 basename(prog_arg), kListeningMsgPrefix);
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700528}
529
adlr@google.com3defe6a2009-12-04 20:57:17 +0000530int main(int argc, char** argv) {
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700531 // Check invocation.
532 if (argc > 2)
533 errx(RC_BAD_ARGS, "unexpected number of arguments (use -h for usage)");
534
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700535 // Parse (optional) argument.
536 int report_fd = STDOUT_FILENO;
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700537 if (argc == 2) {
538 if (!strcmp(argv[1], "-h")) {
539 usage(argv[0]);
540 exit(RC_OK);
541 }
542
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700543 report_fd = open(argv[1], O_WRONLY | O_CREAT, 00644);
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700544 }
545
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -0700546 // Ignore SIGPIPE on write() to sockets.
547 signal(SIGPIPE, SIG_IGN);
Darin Petkovf67bb1f2010-11-08 16:10:26 -0800548
adlr@google.com3defe6a2009-12-04 20:57:17 +0000549 int listen_fd = socket(AF_INET, SOCK_STREAM, 0);
550 if (listen_fd < 0)
551 LOG(FATAL) << "socket() failed";
552
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700553 struct sockaddr_in server_addr = sockaddr_in();
adlr@google.com3defe6a2009-12-04 20:57:17 +0000554 server_addr.sin_family = AF_INET;
555 server_addr.sin_addr.s_addr = INADDR_ANY;
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700556 server_addr.sin_port = 0;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000557
558 {
559 // Get rid of "Address in use" error
560 int tr = 1;
561 if (setsockopt(listen_fd, SOL_SOCKET, SO_REUSEADDR, &tr,
562 sizeof(int)) == -1) {
563 perror("setsockopt");
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700564 exit(RC_ERR_SETSOCKOPT);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000565 }
566 }
567
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700568 // Bind the socket and set for listening.
adlr@google.com3defe6a2009-12-04 20:57:17 +0000569 if (bind(listen_fd, reinterpret_cast<struct sockaddr *>(&server_addr),
570 sizeof(server_addr)) < 0) {
571 perror("bind");
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700572 exit(RC_ERR_BIND);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000573 }
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700574 if (listen(listen_fd, 5) < 0) {
575 perror("listen");
576 exit(RC_ERR_LISTEN);
577 }
578
579 // Check the actual port.
580 struct sockaddr_in bound_addr = sockaddr_in();
581 socklen_t bound_addr_len = sizeof(bound_addr);
582 if (getsockname(listen_fd, reinterpret_cast<struct sockaddr*>(&bound_addr),
583 &bound_addr_len) < 0) {
584 perror("getsockname");
585 exit(RC_ERR_GETSOCKNAME);
586 }
587 in_port_t port = ntohs(bound_addr.sin_port);
588
589 // Output the listening port, indicating that the server is processing
590 // requests. IMPORTANT! (a) the format of this message is as expected by some
591 // unit tests, avoid unilateral changes; (b) it is necessary to flush/sync the
592 // file to prevent the spawning process from waiting indefinitely for this
593 // message.
Alex Vakulenko75039d72014-03-25 12:36:28 -0700594 string listening_msg = base::StringPrintf("%s%hu", kListeningMsgPrefix, port);
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700595 LOG(INFO) << listening_msg;
596 CHECK_EQ(write(report_fd, listening_msg.c_str(), listening_msg.length()),
597 static_cast<int>(listening_msg.length()));
598 CHECK_EQ(write(report_fd, "\n", 1), 1);
599 if (report_fd == STDOUT_FILENO)
600 fsync(report_fd);
601 else
602 close(report_fd);
603
adlr@google.com3defe6a2009-12-04 20:57:17 +0000604 while (1) {
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700605 LOG(INFO) << "pid(" << getpid() << "): waiting to accept new connection";
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700606 int client_fd = accept(listen_fd, nullptr, nullptr);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000607 LOG(INFO) << "got past accept";
608 if (client_fd < 0)
609 LOG(FATAL) << "ERROR on accept";
610 HandleConnection(client_fd);
611 }
612 return 0;
613}