blob: 24f62616fb4666d8261b9f11fb9e6eb00da7c897 [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>
Gilad Arnoldb6c562a2013-07-01 02:19:26 -070026#include <fcntl.h>
Andrew de los Reyes08c4e272010-04-15 14:02:17 -070027#include <inttypes.h>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070028#include <netinet/in.h>
29#include <signal.h>
adlr@google.com3defe6a2009-12-04 20:57:17 +000030#include <stdio.h>
31#include <stdlib.h>
32#include <string.h>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070033#include <sys/socket.h>
Gilad Arnoldb6c562a2013-07-01 02:19:26 -070034#include <sys/stat.h>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070035#include <sys/types.h>
adlr@google.com3defe6a2009-12-04 20:57:17 +000036#include <unistd.h>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070037
adlr@google.com3defe6a2009-12-04 20:57:17 +000038#include <string>
39#include <vector>
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -070040
41#include <base/logging.h>
Alex Deymof123ae22015-09-24 14:59:43 -070042#include <base/posix/eintr_wrapper.h>
Alex Vakulenko75039d72014-03-25 12:36:28 -070043#include <base/strings/string_split.h>
44#include <base/strings/string_util.h>
Kelvin Zhangb9a9aa22024-10-15 10:38:35 -070045#include <android-base/stringprintf.h>
Gilad Arnold9bedeb52011-11-17 16:19:57 -080046
Alex Deymo39910dc2015-11-09 17:04:30 -080047#include "update_engine/common/http_common.h"
Gilad Arnold9bedeb52011-11-17 16:19:57 -080048
Gilad Arnolde4ad2502011-12-29 17:08:54 -080049// HTTP end-of-line delimiter; sorry, this needs to be a macro.
50#define EOL "\r\n"
51
adlr@google.com3defe6a2009-12-04 20:57:17 +000052using std::string;
53using std::vector;
54
55namespace chromeos_update_engine {
56
Gilad Arnoldb6c562a2013-07-01 02:19:26 -070057static const char* kListeningMsgPrefix = "listening on port ";
Gilad Arnold97bdb3f2013-06-30 23:05:50 -070058
59enum {
60 RC_OK = 0,
61 RC_BAD_ARGS,
62 RC_ERR_READ,
63 RC_ERR_SETSOCKOPT,
64 RC_ERR_BIND,
Gilad Arnoldb6c562a2013-07-01 02:19:26 -070065 RC_ERR_LISTEN,
66 RC_ERR_GETSOCKNAME,
67 RC_ERR_REPORT,
Gilad Arnold97bdb3f2013-06-30 23:05:50 -070068};
69
adlr@google.com3defe6a2009-12-04 20:57:17 +000070struct HttpRequest {
Alex Deymo6f10c5f2016-03-03 22:35:43 -080071 string raw_headers;
Darin Petkov41c2fcf2010-08-25 13:14:48 -070072 string host;
adlr@google.com3defe6a2009-12-04 20:57:17 +000073 string url;
Alex Deymo6f10c5f2016-03-03 22:35:43 -080074 off_t start_offset{0};
75 off_t end_offset{0}; // non-inclusive, zero indicates unspecified.
76 HttpResponseCode return_code{kHttpResponseOk};
adlr@google.com3defe6a2009-12-04 20:57:17 +000077};
78
adlr@google.com3defe6a2009-12-04 20:57:17 +000079bool ParseRequest(int fd, HttpRequest* request) {
80 string headers;
Gilad Arnold9bedeb52011-11-17 16:19:57 -080081 do {
82 char buf[1024];
83 ssize_t r = read(fd, buf, sizeof(buf));
adlr@google.com3defe6a2009-12-04 20:57:17 +000084 if (r < 0) {
85 perror("read");
Gilad Arnold97bdb3f2013-06-30 23:05:50 -070086 exit(RC_ERR_READ);
adlr@google.com3defe6a2009-12-04 20:57:17 +000087 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -080088 headers.append(buf, r);
Alex Vakulenkoa3cf75a2016-01-20 07:56:15 -080089 } while (!base::EndsWith(headers, EOL EOL, base::CompareCase::SENSITIVE));
adlr@google.com3defe6a2009-12-04 20:57:17 +000090
Gilad Arnold9bedeb52011-11-17 16:19:57 -080091 LOG(INFO) << "got headers:\n--8<------8<------8<------8<----\n"
Amin Hassani7cc8bb02019-01-14 16:29:47 -080092 << headers << "\n--8<------8<------8<------8<----";
Alex Deymo6f10c5f2016-03-03 22:35:43 -080093 request->raw_headers = headers;
adlr@google.com3defe6a2009-12-04 20:57:17 +000094
Gilad Arnold9bedeb52011-11-17 16:19:57 -080095 // Break header into lines.
Hidehiko Abeeab915e2018-09-23 16:54:06 +090096 vector<string> lines = base::SplitStringUsingSubstr(
97 headers.substr(0, headers.length() - strlen(EOL EOL)),
98 EOL,
99 base::TRIM_WHITESPACE,
100 base::SPLIT_WANT_ALL);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000101
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800102 // Decode URL line.
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800103 vector<string> terms = base::SplitString(lines[0],
104 base::kWhitespaceASCII,
Alex Vakulenkoa3cf75a2016-01-20 07:56:15 -0800105 base::KEEP_WHITESPACE,
106 base::SPLIT_WANT_NONEMPTY);
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500107 CHECK_EQ(terms.size(), static_cast<vector<string>::size_type>(3));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800108 CHECK_EQ(terms[0], "GET");
109 request->url = terms[1];
110 LOG(INFO) << "URL: " << request->url;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000111
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800112 // Decode remaining lines.
Daniel Zheng3e881aa2022-09-07 22:10:29 +0000113 size_t i{};
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800114 for (i = 1; i < lines.size(); i++) {
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800115 terms = base::SplitString(lines[i],
116 base::kWhitespaceASCII,
117 base::KEEP_WHITESPACE,
118 base::SPLIT_WANT_NONEMPTY);
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700119
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800120 if (terms[0] == "Range:") {
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500121 CHECK_EQ(terms.size(), static_cast<vector<string>::size_type>(2));
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800122 string& range = terms[1];
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800123 LOG(INFO) << "range attribute: " << range;
Alex Vakulenkoa3cf75a2016-01-20 07:56:15 -0800124 CHECK(base::StartsWith(range, "bytes=", base::CompareCase::SENSITIVE) &&
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800125 range.find('-') != string::npos);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800126 request->start_offset = atoll(range.c_str() + strlen("bytes="));
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800127 // Decode end offset and increment it by one (so it is non-inclusive).
128 if (range.find('-') < range.length() - 1)
129 request->end_offset = atoll(range.c_str() + range.find('-') + 1) + 1;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800130 request->return_code = kHttpResponsePartialContent;
Kelvin Zhangb9a9aa22024-10-15 10:38:35 -0700131 string tmp_str = android::base::StringPrintf(
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800132 "decoded range offsets: "
133 "start=%jd end=",
134 (intmax_t)request->start_offset);
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800135 if (request->end_offset > 0)
Kelvin Zhangb9a9aa22024-10-15 10:38:35 -0700136 android::base::StringAppendF(
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800137 &tmp_str, "%jd (non-inclusive)", (intmax_t)request->end_offset);
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800138 else
Kelvin Zhangb9a9aa22024-10-15 10:38:35 -0700139 android::base::StringAppendF(&tmp_str, "unspecified");
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800140 LOG(INFO) << tmp_str;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800141 } else if (terms[0] == "Host:") {
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500142 CHECK_EQ(terms.size(), static_cast<vector<string>::size_type>(2));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800143 request->host = terms[1];
144 LOG(INFO) << "host attribute: " << request->host;
145 } else {
146 LOG(WARNING) << "ignoring HTTP attribute: `" << lines[i] << "'";
adlr@google.com3defe6a2009-12-04 20:57:17 +0000147 }
adlr@google.com3defe6a2009-12-04 20:57:17 +0000148 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800149
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -0700150 return true;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000151}
152
153string Itoa(off_t num) {
154 char buf[100] = {0};
Andrew de los Reyes08c4e272010-04-15 14:02:17 -0700155 snprintf(buf, sizeof(buf), "%" PRIi64, num);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000156 return buf;
157}
158
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800159// Writes a string into a file. Returns total number of bytes written or -1 if a
160// write error occurred.
161ssize_t WriteString(int fd, const string& str) {
162 const size_t total_size = str.size();
163 size_t remaining_size = total_size;
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800164 char const* data = str.data();
Gilad Arnold48085ba2011-11-16 09:36:08 -0800165
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800166 while (remaining_size) {
167 ssize_t written = write(fd, data, remaining_size);
168 if (written < 0) {
169 perror("write");
170 LOG(INFO) << "write failed";
171 return -1;
172 }
173 data += written;
174 remaining_size -= written;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000175 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800176
177 return total_size;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000178}
179
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800180// Writes the headers of an HTTP response into a file.
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800181ssize_t WriteHeaders(int fd,
182 const off_t start_offset,
183 const off_t end_offset,
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800184 HttpResponseCode return_code) {
Daniel Zheng3e881aa2022-09-07 22:10:29 +0000185 ssize_t written = 0, ret{};
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800186
187 ret = WriteString(fd,
188 string("HTTP/1.1 ") + Itoa(return_code) + " " +
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800189 GetHttpResponseDescription(return_code) +
Haibo Huang59ad2732020-09-09 15:23:59 -0700190 EOL "Content-Type: application/octet-stream" EOL
Vyshu852f57d2020-10-09 17:35:14 +0000191 "Connection: close" EOL);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800192 if (ret < 0)
193 return -1;
194 written += ret;
195
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800196 // Compute content legnth.
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800197 const off_t content_length = end_offset - start_offset;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800198
199 // A start offset that equals the end offset indicates that the response
200 // should contain the full range of bytes in the requested resource.
201 if (start_offset || start_offset == end_offset) {
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800202 ret = WriteString(
203 fd,
204 string("Accept-Ranges: bytes" EOL "Content-Range: bytes ") +
205 Itoa(start_offset == end_offset ? 0 : start_offset) + "-" +
206 Itoa(end_offset - 1) + "/" + Itoa(end_offset) + EOL);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800207 if (ret < 0)
208 return -1;
209 written += ret;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800210 }
211
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800212 ret = WriteString(
213 fd, string("Content-Length: ") + Itoa(content_length) + EOL EOL);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800214 if (ret < 0)
215 return -1;
216 written += ret;
217
218 return written;
219}
220
221// Writes a predetermined payload of lines of ascending bytes to a file. The
222// first byte of output is appropriately offset with respect to the request line
223// length. Returns the number of successfully written bytes.
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800224size_t WritePayload(int fd,
225 const off_t start_offset,
226 const off_t end_offset,
227 const char first_byte,
228 const size_t line_len) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800229 CHECK_LE(start_offset, end_offset);
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500230 CHECK_GT(line_len, static_cast<size_t>(0));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800231
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800232 LOG(INFO) << "writing payload: " << line_len << "-byte lines starting with `"
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800233 << first_byte << "', offset range " << start_offset << " -> "
234 << end_offset;
235
236 // Populate line of ascending characters.
237 string line;
238 line.reserve(line_len);
239 char byte = first_byte;
Daniel Zheng3e881aa2022-09-07 22:10:29 +0000240 size_t i{};
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800241 for (i = 0; i < line_len; i++)
242 line += byte++;
243
244 const size_t total_len = end_offset - start_offset;
245 size_t remaining_len = total_len;
246 bool success = true;
247
248 // If start offset is not aligned with line boundary, output partial line up
249 // to the first line boundary.
250 size_t start_modulo = start_offset % line_len;
251 if (start_modulo) {
252 string partial = line.substr(start_modulo, remaining_len);
253 ssize_t ret = WriteString(fd, partial);
Colin Cross26b82b12021-12-22 10:09:19 -0800254 if ((success = (ret >= 0 && static_cast<size_t>(ret) == partial.length())))
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800255 remaining_len -= partial.length();
256 }
257
258 // Output full lines up to the maximal line boundary below the end offset.
259 while (success && remaining_len >= line_len) {
260 ssize_t ret = WriteString(fd, line);
Colin Cross26b82b12021-12-22 10:09:19 -0800261 if ((success = (ret >= 0 && static_cast<size_t>(ret) == line_len)))
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800262 remaining_len -= line_len;
263 }
264
265 // Output a partial line up to the end offset.
266 if (success && remaining_len) {
267 string partial = line.substr(0, remaining_len);
268 ssize_t ret = WriteString(fd, partial);
Colin Cross26b82b12021-12-22 10:09:19 -0800269 if ((success = (ret >= 0 && static_cast<size_t>(ret) == partial.length())))
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800270 remaining_len -= partial.length();
271 }
272
273 return (total_len - remaining_len);
274}
275
276// Write default payload lines of the form 'abcdefghij'.
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800277inline size_t WritePayload(int fd,
278 const off_t start_offset,
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800279 const off_t end_offset) {
280 return WritePayload(fd, start_offset, end_offset, 'a', 10);
281}
282
283// Send an empty response, then kill the server.
284void HandleQuit(int fd) {
285 WriteHeaders(fd, 0, 0, kHttpResponseOk);
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800286 LOG(INFO) << "pid(" << getpid() << "): HTTP server exiting ...";
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700287 exit(RC_OK);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000288}
289
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800290// Generates an HTTP response with payload corresponding to requested offsets
291// and length. Optionally, truncate the payload at a given length and add a
292// pause midway through the transfer. Returns the total number of bytes
293// delivered or -1 for error.
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800294ssize_t HandleGet(int fd,
295 const HttpRequest& request,
296 const size_t total_length,
297 const size_t truncate_length,
298 const int sleep_every,
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800299 const int sleep_secs) {
Daniel Zheng3e881aa2022-09-07 22:10:29 +0000300 ssize_t ret{};
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800301 size_t written = 0;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000302
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800303 // Obtain start offset, make sure it is within total payload length.
304 const size_t start_offset = request.start_offset;
305 if (start_offset >= total_length) {
306 LOG(WARNING) << "start offset (" << start_offset
307 << ") exceeds total length (" << total_length
308 << "), generating error response ("
309 << kHttpResponseReqRangeNotSat << ")";
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800310 return WriteHeaders(
311 fd, total_length, total_length, kHttpResponseReqRangeNotSat);
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800312 }
313
314 // Obtain end offset, adjust to fit in total payload length and ensure it does
315 // not preceded the start offset.
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800316 size_t end_offset =
317 (request.end_offset > 0 ? request.end_offset : total_length);
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800318 if (end_offset < start_offset) {
319 LOG(WARNING) << "end offset (" << end_offset << ") precedes start offset ("
320 << start_offset << "), generating error response";
321 return WriteHeaders(fd, 0, 0, kHttpResponseBadRequest);
322 }
323 if (end_offset > total_length) {
324 LOG(INFO) << "requested end offset (" << end_offset
325 << ") exceeds total length (" << total_length << "), adjusting";
326 end_offset = total_length;
327 }
328
329 // Generate headers
330 LOG(INFO) << "generating response header: range=" << start_offset << "-"
331 << (end_offset - 1) << "/" << (end_offset - start_offset)
332 << ", return code=" << request.return_code;
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800333 if ((ret = WriteHeaders(fd, start_offset, end_offset, request.return_code)) <
334 0)
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800335 return -1;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800336 LOG(INFO) << ret << " header bytes written";
337 written += ret;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800338
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800339 // Compute payload length, truncate as necessary.
340 size_t payload_length = end_offset - start_offset;
341 if (truncate_length > 0 && truncate_length < payload_length) {
342 LOG(INFO) << "truncating request payload length (" << payload_length
343 << ") at " << truncate_length;
344 payload_length = truncate_length;
345 end_offset = start_offset + payload_length;
346 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800347
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800348 LOG(INFO) << "generating response payload: range=" << start_offset << "-"
349 << (end_offset - 1) << "/" << (end_offset - start_offset);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800350
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800351 // Decide about optional midway delay.
352 if (truncate_length > 0 && sleep_every > 0 && sleep_secs >= 0 &&
353 start_offset % (truncate_length * sleep_every) == 0) {
354 const off_t midway_offset = start_offset + payload_length / 2;
355
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800356 if ((ret = WritePayload(fd, start_offset, midway_offset)) < 0)
357 return -1;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800358 LOG(INFO) << ret << " payload bytes written (first chunk)";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800359 written += ret;
360
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800361 LOG(INFO) << "sleeping for " << sleep_secs << " seconds...";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800362 sleep(sleep_secs);
363
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800364 if ((ret = WritePayload(fd, midway_offset, end_offset)) < 0)
365 return -1;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800366 LOG(INFO) << ret << " payload bytes written (second chunk)";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800367 written += ret;
368 } else {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800369 if ((ret = WritePayload(fd, start_offset, end_offset)) < 0)
370 return -1;
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800371 LOG(INFO) << ret << " payload bytes written";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800372 written += ret;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000373 }
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800374
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800375 LOG(INFO) << "response generation complete, " << written
376 << " total bytes written";
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800377 return written;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000378}
379
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800380ssize_t HandleGet(int fd,
381 const HttpRequest& request,
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800382 const size_t total_length) {
383 return HandleGet(fd, request, total_length, 0, 0, 0);
384}
385
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700386// Handles /redirect/<code>/<url> requests by returning the specified
387// redirect <code> with a location pointing to /<url>.
388void HandleRedirect(int fd, const HttpRequest& request) {
389 LOG(INFO) << "Redirecting...";
390 string url = request.url;
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500391 CHECK_EQ(static_cast<size_t>(0), url.find("/redirect/"));
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700392 url.erase(0, strlen("/redirect/"));
393 string::size_type url_start = url.find('/');
394 CHECK_NE(url_start, string::npos);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800395 HttpResponseCode code = StringToHttpResponseCode(url.c_str());
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700396 url.erase(0, url_start);
397 url = "http://" + request.host + url;
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800398 const char* status = GetHttpResponseDescription(code);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800399 if (!status)
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700400 CHECK(false) << "Unrecognized redirection code: " << code;
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700401 LOG(INFO) << "Code: " << code << " " << status;
402 LOG(INFO) << "New URL: " << url;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800403
Daniel Zheng3e881aa2022-09-07 22:10:29 +0000404 ssize_t ret{};
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800405 if ((ret = WriteString(fd, "HTTP/1.1 " + Itoa(code) + " " + status + EOL)) <
406 0)
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800407 return;
Haibo Huang59ad2732020-09-09 15:23:59 -0700408 WriteString(fd, "Connection: close" EOL);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800409 WriteString(fd, "Location: " + url + EOL);
410}
411
412// Generate a page not found error response with actual text payload. Return
413// number of bytes written or -1 for error.
414ssize_t HandleError(int fd, const HttpRequest& request) {
415 LOG(INFO) << "Generating error HTTP response";
416
Daniel Zheng3e881aa2022-09-07 22:10:29 +0000417 ssize_t ret{};
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800418 size_t written = 0;
419
420 const string data("This is an error page.");
421
422 if ((ret = WriteHeaders(fd, 0, data.size(), kHttpResponseNotFound)) < 0)
423 return -1;
424 written += ret;
425
426 if ((ret = WriteString(fd, data)) < 0)
427 return -1;
428 written += ret;
429
430 return written;
431}
432
433// Generate an error response if the requested offset is nonzero, up to a given
434// maximal number of successive failures. The error generated is an "Internal
435// Server Error" (500).
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800436ssize_t HandleErrorIfOffset(int fd,
437 const HttpRequest& request,
438 size_t end_offset,
439 int max_fails) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800440 static int num_fails = 0;
441
442 if (request.start_offset > 0 && num_fails < max_fails) {
443 LOG(INFO) << "Generating error HTTP response";
444
Daniel Zheng3e881aa2022-09-07 22:10:29 +0000445 ssize_t ret{};
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800446 size_t written = 0;
447
448 const string data("This is an error page.");
449
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800450 if ((ret = WriteHeaders(
451 fd, 0, data.size(), kHttpResponseInternalServerError)) < 0)
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800452 return -1;
453 written += ret;
454
455 if ((ret = WriteString(fd, data)) < 0)
456 return -1;
457 written += ret;
458
459 num_fails++;
460 return written;
461 } else {
462 num_fails = 0;
463 return HandleGet(fd, request, end_offset);
464 }
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700465}
466
Alex Deymo6f10c5f2016-03-03 22:35:43 -0800467// Returns a valid response echoing in the body of the response all the headers
468// sent by the client.
469void HandleEchoHeaders(int fd, const HttpRequest& request) {
470 WriteHeaders(fd, 0, request.raw_headers.size(), kHttpResponseOk);
471 WriteString(fd, request.raw_headers);
472}
473
Alex Deymof123ae22015-09-24 14:59:43 -0700474void HandleHang(int fd) {
475 LOG(INFO) << "Hanging until the other side of the connection is closed.";
Daniel Zheng3e881aa2022-09-07 22:10:29 +0000476 char c{};
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800477 while (HANDLE_EINTR(read(fd, &c, 1)) > 0) {
478 }
Alex Deymof123ae22015-09-24 14:59:43 -0700479}
480
adlr@google.com3defe6a2009-12-04 20:57:17 +0000481void HandleDefault(int fd, const HttpRequest& request) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800482 const off_t start_offset = request.start_offset;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000483 const string data("unhandled path");
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800484 const size_t size = data.size();
Daniel Zheng3e881aa2022-09-07 22:10:29 +0000485 ssize_t ret{};
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800486
487 if ((ret = WriteHeaders(fd, start_offset, size, request.return_code)) < 0)
488 return;
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800489 WriteString(
490 fd,
491 (start_offset < static_cast<off_t>(size) ? data.substr(start_offset)
492 : ""));
adlr@google.com3defe6a2009-12-04 20:57:17 +0000493}
494
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800495// Break a URL into terms delimited by slashes.
496class UrlTerms {
497 public:
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800498 UrlTerms(const string& url, size_t num_terms) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800499 // URL must be non-empty and start with a slash.
Mike Frysinger0f9547d2012-02-16 12:11:37 -0500500 CHECK_GT(url.size(), static_cast<size_t>(0));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800501 CHECK_EQ(url[0], '/');
502
Alex Vakulenkoa3cf75a2016-01-20 07:56:15 -0800503 // Split it into terms delimited by slashes, omitting the preceding slash.
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800504 terms = base::SplitString(
505 url.substr(1), "/", base::KEEP_WHITESPACE, base::SPLIT_WANT_ALL);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800506
507 // Ensure expected length.
508 CHECK_EQ(terms.size(), num_terms);
509 }
510
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800511 inline const string& Get(const off_t index) const { return terms[index]; }
512 inline const char* GetCStr(const off_t index) const {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800513 return Get(index).c_str();
514 }
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800515 inline int GetInt(const off_t index) const { return atoi(GetCStr(index)); }
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700516 inline size_t GetSizeT(const off_t index) const {
517 return static_cast<size_t>(atol(GetCStr(index)));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800518 }
519
520 private:
Alex Deymof329b932014-10-30 01:37:48 -0700521 vector<string> terms;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800522};
Gilad Arnold48085ba2011-11-16 09:36:08 -0800523
adlr@google.com3defe6a2009-12-04 20:57:17 +0000524void HandleConnection(int fd) {
525 HttpRequest request;
526 ParseRequest(fd, &request);
527
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800528 string& url = request.url;
529 LOG(INFO) << "pid(" << getpid() << "): handling url " << url;
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800530 if (url == "/quitquitquit") {
531 HandleQuit(fd);
Alex Deymo6f10c5f2016-03-03 22:35:43 -0800532 } else if (base::StartsWith(
533 url, "/download/", base::CompareCase::SENSITIVE)) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800534 const UrlTerms terms(url, 2);
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700535 HandleGet(fd, request, terms.GetSizeT(1));
Alex Vakulenkoa3cf75a2016-01-20 07:56:15 -0800536 } else if (base::StartsWith(url, "/flaky/", base::CompareCase::SENSITIVE)) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800537 const UrlTerms terms(url, 5);
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800538 HandleGet(fd,
539 request,
540 terms.GetSizeT(1),
541 terms.GetSizeT(2),
542 terms.GetInt(3),
543 terms.GetInt(4));
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800544 } else if (url.find("/redirect/") == 0) {
Darin Petkov41c2fcf2010-08-25 13:14:48 -0700545 HandleRedirect(fd, request);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800546 } else if (url == "/error") {
Gilad Arnold48085ba2011-11-16 09:36:08 -0800547 HandleError(fd, request);
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800548 } else if (base::StartsWith(
549 url, "/error-if-offset/", base::CompareCase::SENSITIVE)) {
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800550 const UrlTerms terms(url, 3);
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700551 HandleErrorIfOffset(fd, request, terms.GetSizeT(1), terms.GetInt(2));
Alex Deymo6f10c5f2016-03-03 22:35:43 -0800552 } else if (url == "/echo-headers") {
553 HandleEchoHeaders(fd, request);
Alex Deymof123ae22015-09-24 14:59:43 -0700554 } else if (url == "/hang") {
555 HandleHang(fd);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800556 } else {
adlr@google.com3defe6a2009-12-04 20:57:17 +0000557 HandleDefault(fd, request);
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800558 }
adlr@google.com3defe6a2009-12-04 20:57:17 +0000559
560 close(fd);
561}
562
563} // namespace chromeos_update_engine
564
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700565using namespace chromeos_update_engine; // NOLINT(build/namespaces)
adlr@google.com3defe6a2009-12-04 20:57:17 +0000566
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800567void usage(const char* prog_arg) {
568 fprintf(stderr,
569 "Usage: %s [ FILE ]\n"
570 "Once accepting connections, the following is written to FILE (or "
571 "stdout):\n"
572 "\"%sN\" (where N is an integer port number)\n",
573 basename(prog_arg),
574 kListeningMsgPrefix);
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700575}
576
adlr@google.com3defe6a2009-12-04 20:57:17 +0000577int main(int argc, char** argv) {
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700578 // Check invocation.
579 if (argc > 2)
580 errx(RC_BAD_ARGS, "unexpected number of arguments (use -h for usage)");
581
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700582 // Parse (optional) argument.
583 int report_fd = STDOUT_FILENO;
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700584 if (argc == 2) {
585 if (!strcmp(argv[1], "-h")) {
586 usage(argv[0]);
587 exit(RC_OK);
588 }
589
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700590 report_fd = open(argv[1], O_WRONLY | O_CREAT, 00644);
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700591 }
592
Andrew de los Reyes3fd5d302010-10-07 20:07:18 -0700593 // Ignore SIGPIPE on write() to sockets.
594 signal(SIGPIPE, SIG_IGN);
Darin Petkovf67bb1f2010-11-08 16:10:26 -0800595
adlr@google.com3defe6a2009-12-04 20:57:17 +0000596 int listen_fd = socket(AF_INET, SOCK_STREAM, 0);
597 if (listen_fd < 0)
598 LOG(FATAL) << "socket() failed";
599
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700600 struct sockaddr_in server_addr = sockaddr_in();
adlr@google.com3defe6a2009-12-04 20:57:17 +0000601 server_addr.sin_family = AF_INET;
602 server_addr.sin_addr.s_addr = INADDR_ANY;
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700603 server_addr.sin_port = 0;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000604
605 {
606 // Get rid of "Address in use" error
607 int tr = 1;
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800608 if (setsockopt(listen_fd, SOL_SOCKET, SO_REUSEADDR, &tr, sizeof(int)) ==
609 -1) {
adlr@google.com3defe6a2009-12-04 20:57:17 +0000610 perror("setsockopt");
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700611 exit(RC_ERR_SETSOCKOPT);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000612 }
613 }
614
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700615 // Bind the socket and set for listening.
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800616 if (bind(listen_fd,
617 reinterpret_cast<struct sockaddr*>(&server_addr),
adlr@google.com3defe6a2009-12-04 20:57:17 +0000618 sizeof(server_addr)) < 0) {
619 perror("bind");
Gilad Arnold97bdb3f2013-06-30 23:05:50 -0700620 exit(RC_ERR_BIND);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000621 }
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700622 if (listen(listen_fd, 5) < 0) {
623 perror("listen");
624 exit(RC_ERR_LISTEN);
625 }
626
627 // Check the actual port.
628 struct sockaddr_in bound_addr = sockaddr_in();
629 socklen_t bound_addr_len = sizeof(bound_addr);
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800630 if (getsockname(listen_fd,
631 reinterpret_cast<struct sockaddr*>(&bound_addr),
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700632 &bound_addr_len) < 0) {
633 perror("getsockname");
634 exit(RC_ERR_GETSOCKNAME);
635 }
636 in_port_t port = ntohs(bound_addr.sin_port);
637
638 // Output the listening port, indicating that the server is processing
639 // requests. IMPORTANT! (a) the format of this message is as expected by some
640 // unit tests, avoid unilateral changes; (b) it is necessary to flush/sync the
641 // file to prevent the spawning process from waiting indefinitely for this
642 // message.
Kelvin Zhangb9a9aa22024-10-15 10:38:35 -0700643 string listening_msg =
644 android::base::StringPrintf("%s%hu", kListeningMsgPrefix, port);
Gilad Arnoldb6c562a2013-07-01 02:19:26 -0700645 LOG(INFO) << listening_msg;
646 CHECK_EQ(write(report_fd, listening_msg.c_str(), listening_msg.length()),
647 static_cast<int>(listening_msg.length()));
648 CHECK_EQ(write(report_fd, "\n", 1), 1);
649 if (report_fd == STDOUT_FILENO)
650 fsync(report_fd);
651 else
652 close(report_fd);
653
adlr@google.com3defe6a2009-12-04 20:57:17 +0000654 while (1) {
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800655 LOG(INFO) << "pid(" << getpid() << "): waiting to accept new connection";
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700656 int client_fd = accept(listen_fd, nullptr, nullptr);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000657 LOG(INFO) << "got past accept";
658 if (client_fd < 0)
659 LOG(FATAL) << "ERROR on accept";
660 HandleConnection(client_fd);
661 }
adlr@google.com3defe6a2009-12-04 20:57:17 +0000662}