blob: eaffba8eeb697e441ab6bf017a41367daff8875f [file] [log] [blame]
Alex Deymoaea4c1c2015-08-19 20:24:43 -07001//
2// Copyright (C) 2012 The Android Open Source Project
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15//
adlr@google.com3defe6a2009-12-04 20:57:17 +000016
17// This file implements a simple HTTP server. It can exhibit odd behavior
18// that's useful for testing. For example, it's useful to test that
19// the updater can continue a connection if it's dropped, or that it
20// handles very slow data transfers.
21
22// To use this, simply make an HTTP connection to localhost:port and
23// GET a url.
24
Gilad Arnold97bdb3f2013-06-30 23:05:50 -070025#include <err.h>
adlr@google.com3defe6a2009-12-04 20:57:17 +000026#include <errno.h>
Gilad Arnoldb6c562a2013-07-01 02:19:26 -070027#include <fcntl.h>
Andrew de los Reyes08c4e272010-04-15 14:02:17 -070028#include <inttypes.h>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070029#include <netinet/in.h>
30#include <signal.h>
adlr@google.com3defe6a2009-12-04 20:57:17 +000031#include <stdio.h>
32#include <stdlib.h>
33#include <string.h>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070034#include <sys/socket.h>
Gilad Arnoldb6c562a2013-07-01 02:19:26 -070035#include <sys/stat.h>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070036#include <sys/types.h>
adlr@google.com3defe6a2009-12-04 20:57:17 +000037#include <unistd.h>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070038
adlr@google.com3defe6a2009-12-04 20:57:17 +000039#include <algorithm>
40#include <string>
41#include <vector>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070042
43#include <base/logging.h>
Alex Deymof123ae22015-09-24 14:59:43 -070044#include <base/posix/eintr_wrapper.h>
Alex Vakulenko75039d72014-03-25 12:36:28 -070045#include <base/strings/string_split.h>
46#include <base/strings/string_util.h>
47#include <base/strings/stringprintf.h>
Gilad Arnold9bedeb52011-11-17 16:19:57 -080048
Alex Deymo39910dc2015-11-09 17:04:30 -080049#include "update_engine/common/http_common.h"
Gilad Arnold9bedeb52011-11-17 16:19:57 -080050
adlr@google.com3defe6a2009-12-04 20:57:17 +000051
Gilad Arnolde4ad2502011-12-29 17:08:54 -080052// HTTP end-of-line delimiter; sorry, this needs to be a macro.
53#define EOL "\r\n"
54
adlr@google.com3defe6a2009-12-04 20:57:17 +000055using std::string;
56using std::vector;
57
Gilad Arnold9bedeb52011-11-17 16:19:57 -080058
adlr@google.com3defe6a2009-12-04 20:57:17 +000059namespace chromeos_update_engine {
60
Gilad Arnoldb6c562a2013-07-01 02:19:26 -070061static const char* kListeningMsgPrefix = "listening on port ";
Gilad Arnold97bdb3f2013-06-30 23:05:50 -070062
63enum {
64 RC_OK = 0,
65 RC_BAD_ARGS,
66 RC_ERR_READ,
67 RC_ERR_SETSOCKOPT,
68 RC_ERR_BIND,
Gilad Arnoldb6c562a2013-07-01 02:19:26 -070069 RC_ERR_LISTEN,
70 RC_ERR_GETSOCKNAME,
71 RC_ERR_REPORT,
Gilad Arnold97bdb3f2013-06-30 23:05:50 -070072};
73
adlr@google.com3defe6a2009-12-04 20:57:17 +000074struct HttpRequest {
Gilad Arnolde4ad2502011-12-29 17:08:54 -080075 HttpRequest()
76 : start_offset(0), end_offset(0), return_code(kHttpResponseOk) {}
Darin Petkov41c2fcf2010-08-25 13:14:48 -070077 string host;
adlr@google.com3defe6a2009-12-04 20:57:17 +000078 string url;
Gilad Arnold9bedeb52011-11-17 16:19:57 -080079 off_t start_offset;
Gilad Arnolde4ad2502011-12-29 17:08:54 -080080 off_t end_offset; // non-inclusive, zero indicates unspecified.
Gilad Arnold9bedeb52011-11-17 16:19:57 -080081 HttpResponseCode return_code;
adlr@google.com3defe6a2009-12-04 20:57:17 +000082};
83
adlr@google.com3defe6a2009-12-04 20:57:17 +000084bool ParseRequest(int fd, HttpRequest* request) {
85 string headers;
Gilad Arnold9bedeb52011-11-17 16:19:57 -080086 do {
87 char buf[1024];
88 ssize_t r = read(fd, buf, sizeof(buf));
adlr@google.com3defe6a2009-12-04 20:57:17 +000089 if (r < 0) {
90 perror("read");
Gilad Arnold97bdb3f2013-06-30 23:05:50 -070091 exit(RC_ERR_READ);
adlr@google.com3defe6a2009-12-04 20:57:17 +000092 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -080093 headers.append(buf, r);
Alex Vakulenko6a9d3492015-06-15 12:53:22 -070094 } while (!base::EndsWith(headers, EOL EOL, true));
adlr@google.com3defe6a2009-12-04 20:57:17 +000095
Gilad Arnold9bedeb52011-11-17 16:19:57 -080096 LOG(INFO) << "got headers:\n--8<------8<------8<------8<----\n"
97 << headers
98 << "\n--8<------8<------8<------8<----";
adlr@google.com3defe6a2009-12-04 20:57:17 +000099
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800100 // Break header into lines.
Alex Deymof329b932014-10-30 01:37:48 -0700101 vector<string> lines;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800102 base::SplitStringUsingSubstr(
103 headers.substr(0, headers.length() - strlen(EOL EOL)), EOL, &lines);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000104
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800105 // Decode URL line.
Alex Deymof329b932014-10-30 01:37:48 -0700106 vector<string> terms;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800107 base::SplitStringAlongWhitespace(lines[0], &terms);
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500108 CHECK_EQ(terms.size(), static_cast<vector<string>::size_type>(3));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800109 CHECK_EQ(terms[0], "GET");
110 request->url = terms[1];
111 LOG(INFO) << "URL: " << request->url;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000112
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800113 // Decode remaining lines.
114 size_t i;
115 for (i = 1; i < lines.size(); i++) {
Alex Deymof329b932014-10-30 01:37:48 -0700116 vector<string> terms;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800117 base::SplitStringAlongWhitespace(lines[i], &terms);
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700118
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800119 if (terms[0] == "Range:") {
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500120 CHECK_EQ(terms.size(), static_cast<vector<string>::size_type>(2));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800121 string &range = terms[1];
122 LOG(INFO) << "range attribute: " << range;
Alex Vakulenko6a9d3492015-06-15 12:53:22 -0700123 CHECK(base::StartsWithASCII(range, "bytes=", true) &&
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800124 range.find('-') != string::npos);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800125 request->start_offset = atoll(range.c_str() + strlen("bytes="));
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800126 // Decode end offset and increment it by one (so it is non-inclusive).
127 if (range.find('-') < range.length() - 1)
128 request->end_offset = atoll(range.c_str() + range.find('-') + 1) + 1;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800129 request->return_code = kHttpResponsePartialContent;
Alex Deymof329b932014-10-30 01:37:48 -0700130 string tmp_str = base::StringPrintf("decoded range offsets: "
Alex Vakulenko75039d72014-03-25 12:36:28 -0700131 "start=%jd end=",
132 (intmax_t)request->start_offset);
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800133 if (request->end_offset > 0)
134 base::StringAppendF(&tmp_str, "%jd (non-inclusive)",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700135 (intmax_t)request->end_offset);
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800136 else
137 base::StringAppendF(&tmp_str, "unspecified");
138 LOG(INFO) << tmp_str;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800139 } else if (terms[0] == "Host:") {
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500140 CHECK_EQ(terms.size(), static_cast<vector<string>::size_type>(2));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800141 request->host = terms[1];
142 LOG(INFO) << "host attribute: " << request->host;
143 } else {
144 LOG(WARNING) << "ignoring HTTP attribute: `" << lines[i] << "'";
adlr@google.com3defe6a2009-12-04 20:57:17 +0000145 }
adlr@google.com3defe6a2009-12-04 20:57:17 +0000146 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800147
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -0700148 return true;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000149}
150
151string Itoa(off_t num) {
152 char buf[100] = {0};
Andrew de los Reyes08c4e272010-04-15 14:02:17 -0700153 snprintf(buf, sizeof(buf), "%" PRIi64, num);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000154 return buf;
155}
156
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800157// Writes a string into a file. Returns total number of bytes written or -1 if a
158// write error occurred.
159ssize_t WriteString(int fd, const string& str) {
160 const size_t total_size = str.size();
161 size_t remaining_size = total_size;
162 char const *data = str.data();
Gilad Arnold48085ba2011-11-16 09:36:08 -0800163
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800164 while (remaining_size) {
165 ssize_t written = write(fd, data, remaining_size);
166 if (written < 0) {
167 perror("write");
168 LOG(INFO) << "write failed";
169 return -1;
170 }
171 data += written;
172 remaining_size -= written;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000173 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800174
175 return total_size;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000176}
177
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800178// Writes the headers of an HTTP response into a file.
179ssize_t WriteHeaders(int fd, const off_t start_offset, const off_t end_offset,
180 HttpResponseCode return_code) {
181 ssize_t written = 0, ret;
182
183 ret = WriteString(fd,
184 string("HTTP/1.1 ") + Itoa(return_code) + " " +
185 GetHttpResponseDescription(return_code) +
186 EOL
187 "Content-Type: application/octet-stream" EOL);
188 if (ret < 0)
189 return -1;
190 written += ret;
191
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800192 // Compute content legnth.
193 const off_t content_length = end_offset - start_offset;;
194
195 // A start offset that equals the end offset indicates that the response
196 // should contain the full range of bytes in the requested resource.
197 if (start_offset || start_offset == end_offset) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800198 ret = WriteString(fd,
199 string("Accept-Ranges: bytes" EOL
200 "Content-Range: bytes ") +
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800201 Itoa(start_offset == end_offset ? 0 : start_offset) +
202 "-" + Itoa(end_offset - 1) + "/" + Itoa(end_offset) +
203 EOL);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800204 if (ret < 0)
205 return -1;
206 written += ret;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800207 }
208
209 ret = WriteString(fd, string("Content-Length: ") + Itoa(content_length) +
210 EOL EOL);
211 if (ret < 0)
212 return -1;
213 written += ret;
214
215 return written;
216}
217
218// Writes a predetermined payload of lines of ascending bytes to a file. The
219// first byte of output is appropriately offset with respect to the request line
220// length. Returns the number of successfully written bytes.
221size_t WritePayload(int fd, const off_t start_offset, const off_t end_offset,
222 const char first_byte, const size_t line_len) {
223 CHECK_LE(start_offset, end_offset);
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500224 CHECK_GT(line_len, static_cast<size_t>(0));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800225
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800226 LOG(INFO) << "writing payload: " << line_len << "-byte lines starting with `"
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800227 << first_byte << "', offset range " << start_offset << " -> "
228 << end_offset;
229
230 // Populate line of ascending characters.
231 string line;
232 line.reserve(line_len);
233 char byte = first_byte;
234 size_t i;
235 for (i = 0; i < line_len; i++)
236 line += byte++;
237
238 const size_t total_len = end_offset - start_offset;
239 size_t remaining_len = total_len;
240 bool success = true;
241
242 // If start offset is not aligned with line boundary, output partial line up
243 // to the first line boundary.
244 size_t start_modulo = start_offset % line_len;
245 if (start_modulo) {
246 string partial = line.substr(start_modulo, remaining_len);
247 ssize_t ret = WriteString(fd, partial);
248 if ((success = (ret >= 0 && (size_t) ret == partial.length())))
249 remaining_len -= partial.length();
250 }
251
252 // Output full lines up to the maximal line boundary below the end offset.
253 while (success && remaining_len >= line_len) {
254 ssize_t ret = WriteString(fd, line);
255 if ((success = (ret >= 0 && (size_t) ret == line_len)))
256 remaining_len -= line_len;
257 }
258
259 // Output a partial line up to the end offset.
260 if (success && remaining_len) {
261 string partial = line.substr(0, remaining_len);
262 ssize_t ret = WriteString(fd, partial);
263 if ((success = (ret >= 0 && (size_t) ret == partial.length())))
264 remaining_len -= partial.length();
265 }
266
267 return (total_len - remaining_len);
268}
269
270// Write default payload lines of the form 'abcdefghij'.
271inline size_t WritePayload(int fd, const off_t start_offset,
272 const off_t end_offset) {
273 return WritePayload(fd, start_offset, end_offset, 'a', 10);
274}
275
276// Send an empty response, then kill the server.
277void HandleQuit(int fd) {
278 WriteHeaders(fd, 0, 0, kHttpResponseOk);
Jay Srinivasan135a58b2012-07-13 12:46:49 -0700279 LOG(INFO) << "pid(" << getpid() << "): HTTP server exiting ...";
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700280 exit(RC_OK);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000281}
282
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800283
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800284// Generates an HTTP response with payload corresponding to requested offsets
285// and length. Optionally, truncate the payload at a given length and add a
286// pause midway through the transfer. Returns the total number of bytes
287// delivered or -1 for error.
288ssize_t HandleGet(int fd, const HttpRequest& request, const size_t total_length,
289 const size_t truncate_length, const int sleep_every,
290 const int sleep_secs) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800291 ssize_t ret;
292 size_t written = 0;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000293
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800294 // Obtain start offset, make sure it is within total payload length.
295 const size_t start_offset = request.start_offset;
296 if (start_offset >= total_length) {
297 LOG(WARNING) << "start offset (" << start_offset
298 << ") exceeds total length (" << total_length
299 << "), generating error response ("
300 << kHttpResponseReqRangeNotSat << ")";
301 return WriteHeaders(fd, total_length, total_length,
302 kHttpResponseReqRangeNotSat);
303 }
304
305 // Obtain end offset, adjust to fit in total payload length and ensure it does
306 // not preceded the start offset.
307 size_t end_offset = (request.end_offset > 0 ?
308 request.end_offset : total_length);
309 if (end_offset < start_offset) {
310 LOG(WARNING) << "end offset (" << end_offset << ") precedes start offset ("
311 << start_offset << "), generating error response";
312 return WriteHeaders(fd, 0, 0, kHttpResponseBadRequest);
313 }
314 if (end_offset > total_length) {
315 LOG(INFO) << "requested end offset (" << end_offset
316 << ") exceeds total length (" << total_length << "), adjusting";
317 end_offset = total_length;
318 }
319
320 // Generate headers
321 LOG(INFO) << "generating response header: range=" << start_offset << "-"
322 << (end_offset - 1) << "/" << (end_offset - start_offset)
323 << ", return code=" << request.return_code;
324 if ((ret = WriteHeaders(fd, start_offset, end_offset,
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800325 request.return_code)) < 0)
326 return -1;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800327 LOG(INFO) << ret << " header bytes written";
328 written += ret;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800329
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800330 // Compute payload length, truncate as necessary.
331 size_t payload_length = end_offset - start_offset;
332 if (truncate_length > 0 && truncate_length < payload_length) {
333 LOG(INFO) << "truncating request payload length (" << payload_length
334 << ") at " << truncate_length;
335 payload_length = truncate_length;
336 end_offset = start_offset + payload_length;
337 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800338
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800339 LOG(INFO) << "generating response payload: range=" << start_offset << "-"
340 << (end_offset - 1) << "/" << (end_offset - start_offset);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800341
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800342 // Decide about optional midway delay.
343 if (truncate_length > 0 && sleep_every > 0 && sleep_secs >= 0 &&
344 start_offset % (truncate_length * sleep_every) == 0) {
345 const off_t midway_offset = start_offset + payload_length / 2;
346
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800347 if ((ret = WritePayload(fd, start_offset, midway_offset)) < 0)
348 return -1;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800349 LOG(INFO) << ret << " payload bytes written (first chunk)";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800350 written += ret;
351
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800352 LOG(INFO) << "sleeping for " << sleep_secs << " seconds...";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800353 sleep(sleep_secs);
354
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800355 if ((ret = WritePayload(fd, midway_offset, end_offset)) < 0)
356 return -1;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800357 LOG(INFO) << ret << " payload bytes written (second chunk)";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800358 written += ret;
359 } else {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800360 if ((ret = WritePayload(fd, start_offset, end_offset)) < 0)
361 return -1;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800362 LOG(INFO) << ret << " payload bytes written";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800363 written += ret;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000364 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800365
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800366 LOG(INFO) << "response generation complete, " << written
367 << " total bytes written";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800368 return written;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000369}
370
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800371ssize_t HandleGet(int fd, const HttpRequest& request,
372 const size_t total_length) {
373 return HandleGet(fd, request, total_length, 0, 0, 0);
374}
375
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700376// Handles /redirect/<code>/<url> requests by returning the specified
377// redirect <code> with a location pointing to /<url>.
378void HandleRedirect(int fd, const HttpRequest& request) {
379 LOG(INFO) << "Redirecting...";
380 string url = request.url;
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500381 CHECK_EQ(static_cast<size_t>(0), url.find("/redirect/"));
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700382 url.erase(0, strlen("/redirect/"));
383 string::size_type url_start = url.find('/');
384 CHECK_NE(url_start, string::npos);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800385 HttpResponseCode code = StringToHttpResponseCode(url.c_str());
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700386 url.erase(0, url_start);
387 url = "http://" + request.host + url;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800388 const char *status = GetHttpResponseDescription(code);
389 if (!status)
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700390 CHECK(false) << "Unrecognized redirection code: " << code;
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700391 LOG(INFO) << "Code: " << code << " " << status;
392 LOG(INFO) << "New URL: " << url;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800393
394 ssize_t ret;
395 if ((ret = WriteString(fd, "HTTP/1.1 " + Itoa(code) + " " +
396 status + EOL)) < 0)
397 return;
398 WriteString(fd, "Location: " + url + EOL);
399}
400
401// Generate a page not found error response with actual text payload. Return
402// number of bytes written or -1 for error.
403ssize_t HandleError(int fd, const HttpRequest& request) {
404 LOG(INFO) << "Generating error HTTP response";
405
406 ssize_t ret;
407 size_t written = 0;
408
409 const string data("This is an error page.");
410
411 if ((ret = WriteHeaders(fd, 0, data.size(), kHttpResponseNotFound)) < 0)
412 return -1;
413 written += ret;
414
415 if ((ret = WriteString(fd, data)) < 0)
416 return -1;
417 written += ret;
418
419 return written;
420}
421
422// Generate an error response if the requested offset is nonzero, up to a given
423// maximal number of successive failures. The error generated is an "Internal
424// Server Error" (500).
425ssize_t HandleErrorIfOffset(int fd, const HttpRequest& request,
426 size_t end_offset, int max_fails) {
427 static int num_fails = 0;
428
429 if (request.start_offset > 0 && num_fails < max_fails) {
430 LOG(INFO) << "Generating error HTTP response";
431
432 ssize_t ret;
433 size_t written = 0;
434
435 const string data("This is an error page.");
436
437 if ((ret = WriteHeaders(fd, 0, data.size(),
438 kHttpResponseInternalServerError)) < 0)
439 return -1;
440 written += ret;
441
442 if ((ret = WriteString(fd, data)) < 0)
443 return -1;
444 written += ret;
445
446 num_fails++;
447 return written;
448 } else {
449 num_fails = 0;
450 return HandleGet(fd, request, end_offset);
451 }
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700452}
453
Alex Deymof123ae22015-09-24 14:59:43 -0700454void HandleHang(int fd) {
455 LOG(INFO) << "Hanging until the other side of the connection is closed.";
456 char c;
457 while (HANDLE_EINTR(read(fd, &c, 1)) > 0) {}
458}
459
adlr@google.com3defe6a2009-12-04 20:57:17 +0000460void HandleDefault(int fd, const HttpRequest& request) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800461 const off_t start_offset = request.start_offset;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000462 const string data("unhandled path");
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800463 const size_t size = data.size();
464 ssize_t ret;
465
466 if ((ret = WriteHeaders(fd, start_offset, size, request.return_code)) < 0)
467 return;
468 WriteString(fd, (start_offset < static_cast<off_t>(size) ?
469 data.substr(start_offset) : ""));
adlr@google.com3defe6a2009-12-04 20:57:17 +0000470}
471
Gilad Arnold48085ba2011-11-16 09:36:08 -0800472
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800473// Break a URL into terms delimited by slashes.
474class UrlTerms {
475 public:
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700476 UrlTerms(const string &url, size_t num_terms) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800477 // URL must be non-empty and start with a slash.
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500478 CHECK_GT(url.size(), static_cast<size_t>(0));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800479 CHECK_EQ(url[0], '/');
480
481 // Split it into terms delimited by slashes, omitting the preceeding slash.
482 base::SplitStringDontTrim(url.substr(1), '/', &terms);
483
484 // Ensure expected length.
485 CHECK_EQ(terms.size(), num_terms);
486 }
487
488 inline string Get(const off_t index) const {
489 return terms[index];
490 }
491 inline const char *GetCStr(const off_t index) const {
492 return Get(index).c_str();
493 }
494 inline int GetInt(const off_t index) const {
495 return atoi(GetCStr(index));
496 }
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700497 inline size_t GetSizeT(const off_t index) const {
498 return static_cast<size_t>(atol(GetCStr(index)));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800499 }
500
501 private:
Alex Deymof329b932014-10-30 01:37:48 -0700502 vector<string> terms;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800503};
Gilad Arnold48085ba2011-11-16 09:36:08 -0800504
adlr@google.com3defe6a2009-12-04 20:57:17 +0000505void HandleConnection(int fd) {
506 HttpRequest request;
507 ParseRequest(fd, &request);
508
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800509 string &url = request.url;
Jay Srinivasan135a58b2012-07-13 12:46:49 -0700510 LOG(INFO) << "pid(" << getpid() << "): handling url " << url;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800511 if (url == "/quitquitquit") {
512 HandleQuit(fd);
Alex Vakulenko6a9d3492015-06-15 12:53:22 -0700513 } else if (base::StartsWithASCII(url, "/download/", true)) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800514 const UrlTerms terms(url, 2);
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700515 HandleGet(fd, request, terms.GetSizeT(1));
Alex Vakulenko6a9d3492015-06-15 12:53:22 -0700516 } else if (base::StartsWithASCII(url, "/flaky/", true)) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800517 const UrlTerms terms(url, 5);
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700518 HandleGet(fd, request, terms.GetSizeT(1), terms.GetSizeT(2),
519 terms.GetInt(3), terms.GetInt(4));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800520 } else if (url.find("/redirect/") == 0) {
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700521 HandleRedirect(fd, request);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800522 } else if (url == "/error") {
Gilad Arnold48085ba2011-11-16 09:36:08 -0800523 HandleError(fd, request);
Alex Vakulenko6a9d3492015-06-15 12:53:22 -0700524 } else if (base::StartsWithASCII(url, "/error-if-offset/", true)) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800525 const UrlTerms terms(url, 3);
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700526 HandleErrorIfOffset(fd, request, terms.GetSizeT(1), terms.GetInt(2));
Alex Deymof123ae22015-09-24 14:59:43 -0700527 } else if (url == "/hang") {
528 HandleHang(fd);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800529 } else {
adlr@google.com3defe6a2009-12-04 20:57:17 +0000530 HandleDefault(fd, request);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800531 }
adlr@google.com3defe6a2009-12-04 20:57:17 +0000532
533 close(fd);
534}
535
536} // namespace chromeos_update_engine
537
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700538using namespace chromeos_update_engine; // NOLINT(build/namespaces)
adlr@google.com3defe6a2009-12-04 20:57:17 +0000539
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700540
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700541void usage(const char *prog_arg) {
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700542 fprintf(
543 stderr,
544 "Usage: %s [ FILE ]\n"
545 "Once accepting connections, the following is written to FILE (or "
546 "stdout):\n"
547 "\"%sN\" (where N is an integer port number)\n",
548 basename(prog_arg), kListeningMsgPrefix);
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700549}
550
adlr@google.com3defe6a2009-12-04 20:57:17 +0000551int main(int argc, char** argv) {
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700552 // Check invocation.
553 if (argc > 2)
554 errx(RC_BAD_ARGS, "unexpected number of arguments (use -h for usage)");
555
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700556 // Parse (optional) argument.
557 int report_fd = STDOUT_FILENO;
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700558 if (argc == 2) {
559 if (!strcmp(argv[1], "-h")) {
560 usage(argv[0]);
561 exit(RC_OK);
562 }
563
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700564 report_fd = open(argv[1], O_WRONLY | O_CREAT, 00644);
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700565 }
566
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -0700567 // Ignore SIGPIPE on write() to sockets.
568 signal(SIGPIPE, SIG_IGN);
Darin Petkovf67bb1f2010-11-08 16:10:26 -0800569
adlr@google.com3defe6a2009-12-04 20:57:17 +0000570 int listen_fd = socket(AF_INET, SOCK_STREAM, 0);
571 if (listen_fd < 0)
572 LOG(FATAL) << "socket() failed";
573
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700574 struct sockaddr_in server_addr = sockaddr_in();
adlr@google.com3defe6a2009-12-04 20:57:17 +0000575 server_addr.sin_family = AF_INET;
576 server_addr.sin_addr.s_addr = INADDR_ANY;
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700577 server_addr.sin_port = 0;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000578
579 {
580 // Get rid of "Address in use" error
581 int tr = 1;
582 if (setsockopt(listen_fd, SOL_SOCKET, SO_REUSEADDR, &tr,
583 sizeof(int)) == -1) {
584 perror("setsockopt");
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700585 exit(RC_ERR_SETSOCKOPT);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000586 }
587 }
588
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700589 // Bind the socket and set for listening.
adlr@google.com3defe6a2009-12-04 20:57:17 +0000590 if (bind(listen_fd, reinterpret_cast<struct sockaddr *>(&server_addr),
591 sizeof(server_addr)) < 0) {
592 perror("bind");
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700593 exit(RC_ERR_BIND);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000594 }
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700595 if (listen(listen_fd, 5) < 0) {
596 perror("listen");
597 exit(RC_ERR_LISTEN);
598 }
599
600 // Check the actual port.
601 struct sockaddr_in bound_addr = sockaddr_in();
602 socklen_t bound_addr_len = sizeof(bound_addr);
603 if (getsockname(listen_fd, reinterpret_cast<struct sockaddr*>(&bound_addr),
604 &bound_addr_len) < 0) {
605 perror("getsockname");
606 exit(RC_ERR_GETSOCKNAME);
607 }
608 in_port_t port = ntohs(bound_addr.sin_port);
609
610 // Output the listening port, indicating that the server is processing
611 // requests. IMPORTANT! (a) the format of this message is as expected by some
612 // unit tests, avoid unilateral changes; (b) it is necessary to flush/sync the
613 // file to prevent the spawning process from waiting indefinitely for this
614 // message.
Alex Vakulenko75039d72014-03-25 12:36:28 -0700615 string listening_msg = base::StringPrintf("%s%hu", kListeningMsgPrefix, port);
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700616 LOG(INFO) << listening_msg;
617 CHECK_EQ(write(report_fd, listening_msg.c_str(), listening_msg.length()),
618 static_cast<int>(listening_msg.length()));
619 CHECK_EQ(write(report_fd, "\n", 1), 1);
620 if (report_fd == STDOUT_FILENO)
621 fsync(report_fd);
622 else
623 close(report_fd);
624
adlr@google.com3defe6a2009-12-04 20:57:17 +0000625 while (1) {
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700626 LOG(INFO) << "pid(" << getpid() << "): waiting to accept new connection";
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700627 int client_fd = accept(listen_fd, nullptr, nullptr);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000628 LOG(INFO) << "got past accept";
629 if (client_fd < 0)
630 LOG(FATAL) << "ERROR on accept";
631 HandleConnection(client_fd);
632 }
633 return 0;
634}