xref: /aosp_15_r20/system/update_engine/test_http_server.cc (revision 5a9231315b4521097b8dc3750bc806fcafe0c72f)
1*5a923131SAndroid Build Coastguard Worker //
2*5a923131SAndroid Build Coastguard Worker // Copyright (C) 2012 The Android Open Source Project
3*5a923131SAndroid Build Coastguard Worker //
4*5a923131SAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
5*5a923131SAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
6*5a923131SAndroid Build Coastguard Worker // You may obtain a copy of the License at
7*5a923131SAndroid Build Coastguard Worker //
8*5a923131SAndroid Build Coastguard Worker //      http://www.apache.org/licenses/LICENSE-2.0
9*5a923131SAndroid Build Coastguard Worker //
10*5a923131SAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
11*5a923131SAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
12*5a923131SAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*5a923131SAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
14*5a923131SAndroid Build Coastguard Worker // limitations under the License.
15*5a923131SAndroid Build Coastguard Worker //
16*5a923131SAndroid Build Coastguard Worker 
17*5a923131SAndroid Build Coastguard Worker // This file implements a simple HTTP server. It can exhibit odd behavior
18*5a923131SAndroid Build Coastguard Worker // that's useful for testing. For example, it's useful to test that
19*5a923131SAndroid Build Coastguard Worker // the updater can continue a connection if it's dropped, or that it
20*5a923131SAndroid Build Coastguard Worker // handles very slow data transfers.
21*5a923131SAndroid Build Coastguard Worker 
22*5a923131SAndroid Build Coastguard Worker // To use this, simply make an HTTP connection to localhost:port and
23*5a923131SAndroid Build Coastguard Worker // GET a url.
24*5a923131SAndroid Build Coastguard Worker 
25*5a923131SAndroid Build Coastguard Worker #include <err.h>
26*5a923131SAndroid Build Coastguard Worker #include <fcntl.h>
27*5a923131SAndroid Build Coastguard Worker #include <inttypes.h>
28*5a923131SAndroid Build Coastguard Worker #include <netinet/in.h>
29*5a923131SAndroid Build Coastguard Worker #include <signal.h>
30*5a923131SAndroid Build Coastguard Worker #include <stdio.h>
31*5a923131SAndroid Build Coastguard Worker #include <stdlib.h>
32*5a923131SAndroid Build Coastguard Worker #include <string.h>
33*5a923131SAndroid Build Coastguard Worker #include <sys/socket.h>
34*5a923131SAndroid Build Coastguard Worker #include <sys/stat.h>
35*5a923131SAndroid Build Coastguard Worker #include <sys/types.h>
36*5a923131SAndroid Build Coastguard Worker #include <unistd.h>
37*5a923131SAndroid Build Coastguard Worker 
38*5a923131SAndroid Build Coastguard Worker #include <string>
39*5a923131SAndroid Build Coastguard Worker #include <vector>
40*5a923131SAndroid Build Coastguard Worker 
41*5a923131SAndroid Build Coastguard Worker #include <base/logging.h>
42*5a923131SAndroid Build Coastguard Worker #include <base/posix/eintr_wrapper.h>
43*5a923131SAndroid Build Coastguard Worker #include <base/strings/string_split.h>
44*5a923131SAndroid Build Coastguard Worker #include <android-base/stringprintf.h>
45*5a923131SAndroid Build Coastguard Worker 
46*5a923131SAndroid Build Coastguard Worker #include "android-base/strings.h"
47*5a923131SAndroid Build Coastguard Worker #include "update_engine/common/http_common.h"
48*5a923131SAndroid Build Coastguard Worker 
49*5a923131SAndroid Build Coastguard Worker // HTTP end-of-line delimiter; sorry, this needs to be a macro.
50*5a923131SAndroid Build Coastguard Worker #define EOL "\r\n"
51*5a923131SAndroid Build Coastguard Worker 
52*5a923131SAndroid Build Coastguard Worker using std::string;
53*5a923131SAndroid Build Coastguard Worker using std::vector;
54*5a923131SAndroid Build Coastguard Worker 
55*5a923131SAndroid Build Coastguard Worker namespace chromeos_update_engine {
56*5a923131SAndroid Build Coastguard Worker 
57*5a923131SAndroid Build Coastguard Worker static const char* kListeningMsgPrefix = "listening on port ";
58*5a923131SAndroid Build Coastguard Worker 
59*5a923131SAndroid Build Coastguard Worker enum {
60*5a923131SAndroid Build Coastguard Worker   RC_OK = 0,
61*5a923131SAndroid Build Coastguard Worker   RC_BAD_ARGS,
62*5a923131SAndroid Build Coastguard Worker   RC_ERR_READ,
63*5a923131SAndroid Build Coastguard Worker   RC_ERR_SETSOCKOPT,
64*5a923131SAndroid Build Coastguard Worker   RC_ERR_BIND,
65*5a923131SAndroid Build Coastguard Worker   RC_ERR_LISTEN,
66*5a923131SAndroid Build Coastguard Worker   RC_ERR_GETSOCKNAME,
67*5a923131SAndroid Build Coastguard Worker   RC_ERR_REPORT,
68*5a923131SAndroid Build Coastguard Worker };
69*5a923131SAndroid Build Coastguard Worker 
70*5a923131SAndroid Build Coastguard Worker struct HttpRequest {
71*5a923131SAndroid Build Coastguard Worker   string raw_headers;
72*5a923131SAndroid Build Coastguard Worker   string host;
73*5a923131SAndroid Build Coastguard Worker   string url;
74*5a923131SAndroid Build Coastguard Worker   off_t start_offset{0};
75*5a923131SAndroid Build Coastguard Worker   off_t end_offset{0};  // non-inclusive, zero indicates unspecified.
76*5a923131SAndroid Build Coastguard Worker   HttpResponseCode return_code{kHttpResponseOk};
77*5a923131SAndroid Build Coastguard Worker };
78*5a923131SAndroid Build Coastguard Worker 
ParseRequest(int fd,HttpRequest * request)79*5a923131SAndroid Build Coastguard Worker bool ParseRequest(int fd, HttpRequest* request) {
80*5a923131SAndroid Build Coastguard Worker   string headers;
81*5a923131SAndroid Build Coastguard Worker   do {
82*5a923131SAndroid Build Coastguard Worker     char buf[1024];
83*5a923131SAndroid Build Coastguard Worker     ssize_t r = read(fd, buf, sizeof(buf));
84*5a923131SAndroid Build Coastguard Worker     if (r < 0) {
85*5a923131SAndroid Build Coastguard Worker       perror("read");
86*5a923131SAndroid Build Coastguard Worker       exit(RC_ERR_READ);
87*5a923131SAndroid Build Coastguard Worker     }
88*5a923131SAndroid Build Coastguard Worker     headers.append(buf, r);
89*5a923131SAndroid Build Coastguard Worker   } while (!android::base::EndsWith(headers, EOL EOL));
90*5a923131SAndroid Build Coastguard Worker 
91*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "got headers:\n--8<------8<------8<------8<----\n"
92*5a923131SAndroid Build Coastguard Worker             << headers << "\n--8<------8<------8<------8<----";
93*5a923131SAndroid Build Coastguard Worker   request->raw_headers = headers;
94*5a923131SAndroid Build Coastguard Worker 
95*5a923131SAndroid Build Coastguard Worker   // Break header into lines.
96*5a923131SAndroid Build Coastguard Worker   vector<string> lines = base::SplitStringUsingSubstr(
97*5a923131SAndroid Build Coastguard Worker       headers.substr(0, headers.length() - strlen(EOL EOL)),
98*5a923131SAndroid Build Coastguard Worker       EOL,
99*5a923131SAndroid Build Coastguard Worker       base::TRIM_WHITESPACE,
100*5a923131SAndroid Build Coastguard Worker       base::SPLIT_WANT_ALL);
101*5a923131SAndroid Build Coastguard Worker 
102*5a923131SAndroid Build Coastguard Worker   // Decode URL line.
103*5a923131SAndroid Build Coastguard Worker   vector<string> terms = android::base::Tokenize(lines[0], " ");
104*5a923131SAndroid Build Coastguard Worker   CHECK_EQ(terms.size(), static_cast<vector<string>::size_type>(3));
105*5a923131SAndroid Build Coastguard Worker   CHECK_EQ(terms[0], "GET");
106*5a923131SAndroid Build Coastguard Worker   request->url = terms[1];
107*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "URL: " << request->url;
108*5a923131SAndroid Build Coastguard Worker 
109*5a923131SAndroid Build Coastguard Worker   // Decode remaining lines.
110*5a923131SAndroid Build Coastguard Worker   size_t i{};
111*5a923131SAndroid Build Coastguard Worker   for (i = 1; i < lines.size(); i++) {
112*5a923131SAndroid Build Coastguard Worker     terms = android::base::Tokenize(lines[i], " ");
113*5a923131SAndroid Build Coastguard Worker 
114*5a923131SAndroid Build Coastguard Worker     if (terms[0] == "Range:") {
115*5a923131SAndroid Build Coastguard Worker       CHECK_EQ(terms.size(), static_cast<vector<string>::size_type>(2));
116*5a923131SAndroid Build Coastguard Worker       string& range = terms[1];
117*5a923131SAndroid Build Coastguard Worker       LOG(INFO) << "range attribute: " << range;
118*5a923131SAndroid Build Coastguard Worker       CHECK(android::base::StartsWith(range, "bytes=") &&
119*5a923131SAndroid Build Coastguard Worker             range.find('-') != string::npos);
120*5a923131SAndroid Build Coastguard Worker       request->start_offset = atoll(range.c_str() + strlen("bytes="));
121*5a923131SAndroid Build Coastguard Worker       // Decode end offset and increment it by one (so it is non-inclusive).
122*5a923131SAndroid Build Coastguard Worker       if (range.find('-') < range.length() - 1)
123*5a923131SAndroid Build Coastguard Worker         request->end_offset = atoll(range.c_str() + range.find('-') + 1) + 1;
124*5a923131SAndroid Build Coastguard Worker       request->return_code = kHttpResponsePartialContent;
125*5a923131SAndroid Build Coastguard Worker       string tmp_str = android::base::StringPrintf(
126*5a923131SAndroid Build Coastguard Worker           "decoded range offsets: "
127*5a923131SAndroid Build Coastguard Worker           "start=%jd end=",
128*5a923131SAndroid Build Coastguard Worker           (intmax_t)request->start_offset);
129*5a923131SAndroid Build Coastguard Worker       if (request->end_offset > 0)
130*5a923131SAndroid Build Coastguard Worker         android::base::StringAppendF(
131*5a923131SAndroid Build Coastguard Worker             &tmp_str, "%jd (non-inclusive)", (intmax_t)request->end_offset);
132*5a923131SAndroid Build Coastguard Worker       else
133*5a923131SAndroid Build Coastguard Worker         android::base::StringAppendF(&tmp_str, "unspecified");
134*5a923131SAndroid Build Coastguard Worker       LOG(INFO) << tmp_str;
135*5a923131SAndroid Build Coastguard Worker     } else if (terms[0] == "Host:") {
136*5a923131SAndroid Build Coastguard Worker       CHECK_EQ(terms.size(), static_cast<vector<string>::size_type>(2));
137*5a923131SAndroid Build Coastguard Worker       request->host = terms[1];
138*5a923131SAndroid Build Coastguard Worker       LOG(INFO) << "host attribute: " << request->host;
139*5a923131SAndroid Build Coastguard Worker     } else {
140*5a923131SAndroid Build Coastguard Worker       LOG(WARNING) << "ignoring HTTP attribute: `" << lines[i] << "'";
141*5a923131SAndroid Build Coastguard Worker     }
142*5a923131SAndroid Build Coastguard Worker   }
143*5a923131SAndroid Build Coastguard Worker 
144*5a923131SAndroid Build Coastguard Worker   return true;
145*5a923131SAndroid Build Coastguard Worker }
146*5a923131SAndroid Build Coastguard Worker 
Itoa(off_t num)147*5a923131SAndroid Build Coastguard Worker string Itoa(off_t num) {
148*5a923131SAndroid Build Coastguard Worker   char buf[100] = {0};
149*5a923131SAndroid Build Coastguard Worker   snprintf(buf, sizeof(buf), "%" PRIi64, num);
150*5a923131SAndroid Build Coastguard Worker   return buf;
151*5a923131SAndroid Build Coastguard Worker }
152*5a923131SAndroid Build Coastguard Worker 
153*5a923131SAndroid Build Coastguard Worker // Writes a string into a file. Returns total number of bytes written or -1 if a
154*5a923131SAndroid Build Coastguard Worker // write error occurred.
WriteString(int fd,const string & str)155*5a923131SAndroid Build Coastguard Worker ssize_t WriteString(int fd, const string& str) {
156*5a923131SAndroid Build Coastguard Worker   const size_t total_size = str.size();
157*5a923131SAndroid Build Coastguard Worker   size_t remaining_size = total_size;
158*5a923131SAndroid Build Coastguard Worker   char const* data = str.data();
159*5a923131SAndroid Build Coastguard Worker 
160*5a923131SAndroid Build Coastguard Worker   while (remaining_size) {
161*5a923131SAndroid Build Coastguard Worker     ssize_t written = write(fd, data, remaining_size);
162*5a923131SAndroid Build Coastguard Worker     if (written < 0) {
163*5a923131SAndroid Build Coastguard Worker       perror("write");
164*5a923131SAndroid Build Coastguard Worker       LOG(INFO) << "write failed";
165*5a923131SAndroid Build Coastguard Worker       return -1;
166*5a923131SAndroid Build Coastguard Worker     }
167*5a923131SAndroid Build Coastguard Worker     data += written;
168*5a923131SAndroid Build Coastguard Worker     remaining_size -= written;
169*5a923131SAndroid Build Coastguard Worker   }
170*5a923131SAndroid Build Coastguard Worker 
171*5a923131SAndroid Build Coastguard Worker   return total_size;
172*5a923131SAndroid Build Coastguard Worker }
173*5a923131SAndroid Build Coastguard Worker 
174*5a923131SAndroid Build Coastguard Worker // Writes the headers of an HTTP response into a file.
WriteHeaders(int fd,const off_t start_offset,const off_t end_offset,HttpResponseCode return_code)175*5a923131SAndroid Build Coastguard Worker ssize_t WriteHeaders(int fd,
176*5a923131SAndroid Build Coastguard Worker                      const off_t start_offset,
177*5a923131SAndroid Build Coastguard Worker                      const off_t end_offset,
178*5a923131SAndroid Build Coastguard Worker                      HttpResponseCode return_code) {
179*5a923131SAndroid Build Coastguard Worker   ssize_t written = 0, ret{};
180*5a923131SAndroid Build Coastguard Worker 
181*5a923131SAndroid Build Coastguard Worker   ret = WriteString(fd,
182*5a923131SAndroid Build Coastguard Worker                     string("HTTP/1.1 ") + Itoa(return_code) + " " +
183*5a923131SAndroid Build Coastguard Worker                         GetHttpResponseDescription(return_code) +
184*5a923131SAndroid Build Coastguard Worker                         EOL "Content-Type: application/octet-stream" EOL
185*5a923131SAndroid Build Coastguard Worker                             "Connection: close" EOL);
186*5a923131SAndroid Build Coastguard Worker   if (ret < 0)
187*5a923131SAndroid Build Coastguard Worker     return -1;
188*5a923131SAndroid Build Coastguard Worker   written += ret;
189*5a923131SAndroid Build Coastguard Worker 
190*5a923131SAndroid Build Coastguard Worker   // Compute content legnth.
191*5a923131SAndroid Build Coastguard Worker   const off_t content_length = end_offset - start_offset;
192*5a923131SAndroid Build Coastguard Worker 
193*5a923131SAndroid Build Coastguard Worker   // A start offset that equals the end offset indicates that the response
194*5a923131SAndroid Build Coastguard Worker   // should contain the full range of bytes in the requested resource.
195*5a923131SAndroid Build Coastguard Worker   if (start_offset || start_offset == end_offset) {
196*5a923131SAndroid Build Coastguard Worker     ret = WriteString(
197*5a923131SAndroid Build Coastguard Worker         fd,
198*5a923131SAndroid Build Coastguard Worker         string("Accept-Ranges: bytes" EOL "Content-Range: bytes ") +
199*5a923131SAndroid Build Coastguard Worker             Itoa(start_offset == end_offset ? 0 : start_offset) + "-" +
200*5a923131SAndroid Build Coastguard Worker             Itoa(end_offset - 1) + "/" + Itoa(end_offset) + EOL);
201*5a923131SAndroid Build Coastguard Worker     if (ret < 0)
202*5a923131SAndroid Build Coastguard Worker       return -1;
203*5a923131SAndroid Build Coastguard Worker     written += ret;
204*5a923131SAndroid Build Coastguard Worker   }
205*5a923131SAndroid Build Coastguard Worker 
206*5a923131SAndroid Build Coastguard Worker   ret = WriteString(
207*5a923131SAndroid Build Coastguard Worker       fd, string("Content-Length: ") + Itoa(content_length) + EOL EOL);
208*5a923131SAndroid Build Coastguard Worker   if (ret < 0)
209*5a923131SAndroid Build Coastguard Worker     return -1;
210*5a923131SAndroid Build Coastguard Worker   written += ret;
211*5a923131SAndroid Build Coastguard Worker 
212*5a923131SAndroid Build Coastguard Worker   return written;
213*5a923131SAndroid Build Coastguard Worker }
214*5a923131SAndroid Build Coastguard Worker 
215*5a923131SAndroid Build Coastguard Worker // Writes a predetermined payload of lines of ascending bytes to a file. The
216*5a923131SAndroid Build Coastguard Worker // first byte of output is appropriately offset with respect to the request line
217*5a923131SAndroid Build Coastguard Worker // length.  Returns the number of successfully written bytes.
WritePayload(int fd,const off_t start_offset,const off_t end_offset,const char first_byte,const size_t line_len)218*5a923131SAndroid Build Coastguard Worker size_t WritePayload(int fd,
219*5a923131SAndroid Build Coastguard Worker                     const off_t start_offset,
220*5a923131SAndroid Build Coastguard Worker                     const off_t end_offset,
221*5a923131SAndroid Build Coastguard Worker                     const char first_byte,
222*5a923131SAndroid Build Coastguard Worker                     const size_t line_len) {
223*5a923131SAndroid Build Coastguard Worker   CHECK_LE(start_offset, end_offset);
224*5a923131SAndroid Build Coastguard Worker   CHECK_GT(line_len, static_cast<size_t>(0));
225*5a923131SAndroid Build Coastguard Worker 
226*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "writing payload: " << line_len << "-byte lines starting with `"
227*5a923131SAndroid Build Coastguard Worker             << first_byte << "', offset range " << start_offset << " -> "
228*5a923131SAndroid Build Coastguard Worker             << end_offset;
229*5a923131SAndroid Build Coastguard Worker 
230*5a923131SAndroid Build Coastguard Worker   // Populate line of ascending characters.
231*5a923131SAndroid Build Coastguard Worker   string line;
232*5a923131SAndroid Build Coastguard Worker   line.reserve(line_len);
233*5a923131SAndroid Build Coastguard Worker   char byte = first_byte;
234*5a923131SAndroid Build Coastguard Worker   size_t i{};
235*5a923131SAndroid Build Coastguard Worker   for (i = 0; i < line_len; i++)
236*5a923131SAndroid Build Coastguard Worker     line += byte++;
237*5a923131SAndroid Build Coastguard Worker 
238*5a923131SAndroid Build Coastguard Worker   const size_t total_len = end_offset - start_offset;
239*5a923131SAndroid Build Coastguard Worker   size_t remaining_len = total_len;
240*5a923131SAndroid Build Coastguard Worker   bool success = true;
241*5a923131SAndroid Build Coastguard Worker 
242*5a923131SAndroid Build Coastguard Worker   // If start offset is not aligned with line boundary, output partial line up
243*5a923131SAndroid Build Coastguard Worker   // to the first line boundary.
244*5a923131SAndroid Build Coastguard Worker   size_t start_modulo = start_offset % line_len;
245*5a923131SAndroid Build Coastguard Worker   if (start_modulo) {
246*5a923131SAndroid Build Coastguard Worker     string partial = line.substr(start_modulo, remaining_len);
247*5a923131SAndroid Build Coastguard Worker     ssize_t ret = WriteString(fd, partial);
248*5a923131SAndroid Build Coastguard Worker     if ((success = (ret >= 0 && static_cast<size_t>(ret) == partial.length())))
249*5a923131SAndroid Build Coastguard Worker       remaining_len -= partial.length();
250*5a923131SAndroid Build Coastguard Worker   }
251*5a923131SAndroid Build Coastguard Worker 
252*5a923131SAndroid Build Coastguard Worker   // Output full lines up to the maximal line boundary below the end offset.
253*5a923131SAndroid Build Coastguard Worker   while (success && remaining_len >= line_len) {
254*5a923131SAndroid Build Coastguard Worker     ssize_t ret = WriteString(fd, line);
255*5a923131SAndroid Build Coastguard Worker     if ((success = (ret >= 0 && static_cast<size_t>(ret) == line_len)))
256*5a923131SAndroid Build Coastguard Worker       remaining_len -= line_len;
257*5a923131SAndroid Build Coastguard Worker   }
258*5a923131SAndroid Build Coastguard Worker 
259*5a923131SAndroid Build Coastguard Worker   // Output a partial line up to the end offset.
260*5a923131SAndroid Build Coastguard Worker   if (success && remaining_len) {
261*5a923131SAndroid Build Coastguard Worker     string partial = line.substr(0, remaining_len);
262*5a923131SAndroid Build Coastguard Worker     ssize_t ret = WriteString(fd, partial);
263*5a923131SAndroid Build Coastguard Worker     if ((success = (ret >= 0 && static_cast<size_t>(ret) == partial.length())))
264*5a923131SAndroid Build Coastguard Worker       remaining_len -= partial.length();
265*5a923131SAndroid Build Coastguard Worker   }
266*5a923131SAndroid Build Coastguard Worker 
267*5a923131SAndroid Build Coastguard Worker   return (total_len - remaining_len);
268*5a923131SAndroid Build Coastguard Worker }
269*5a923131SAndroid Build Coastguard Worker 
270*5a923131SAndroid Build Coastguard Worker // Write default payload lines of the form 'abcdefghij'.
WritePayload(int fd,const off_t start_offset,const off_t end_offset)271*5a923131SAndroid Build Coastguard Worker inline size_t WritePayload(int fd,
272*5a923131SAndroid Build Coastguard Worker                            const off_t start_offset,
273*5a923131SAndroid Build Coastguard Worker                            const off_t end_offset) {
274*5a923131SAndroid Build Coastguard Worker   return WritePayload(fd, start_offset, end_offset, 'a', 10);
275*5a923131SAndroid Build Coastguard Worker }
276*5a923131SAndroid Build Coastguard Worker 
277*5a923131SAndroid Build Coastguard Worker // Send an empty response, then kill the server.
HandleQuit(int fd)278*5a923131SAndroid Build Coastguard Worker void HandleQuit(int fd) {
279*5a923131SAndroid Build Coastguard Worker   WriteHeaders(fd, 0, 0, kHttpResponseOk);
280*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "pid(" << getpid() << "): HTTP server exiting ...";
281*5a923131SAndroid Build Coastguard Worker   exit(RC_OK);
282*5a923131SAndroid Build Coastguard Worker }
283*5a923131SAndroid Build Coastguard Worker 
284*5a923131SAndroid Build Coastguard Worker // Generates an HTTP response with payload corresponding to requested offsets
285*5a923131SAndroid Build Coastguard Worker // and length.  Optionally, truncate the payload at a given length and add a
286*5a923131SAndroid Build Coastguard Worker // pause midway through the transfer.  Returns the total number of bytes
287*5a923131SAndroid Build Coastguard Worker // delivered or -1 for error.
HandleGet(int fd,const HttpRequest & request,const size_t total_length,const size_t truncate_length,const int sleep_every,const int sleep_secs)288*5a923131SAndroid Build Coastguard Worker ssize_t HandleGet(int fd,
289*5a923131SAndroid Build Coastguard Worker                   const HttpRequest& request,
290*5a923131SAndroid Build Coastguard Worker                   const size_t total_length,
291*5a923131SAndroid Build Coastguard Worker                   const size_t truncate_length,
292*5a923131SAndroid Build Coastguard Worker                   const int sleep_every,
293*5a923131SAndroid Build Coastguard Worker                   const int sleep_secs) {
294*5a923131SAndroid Build Coastguard Worker   ssize_t ret{};
295*5a923131SAndroid Build Coastguard Worker   size_t written = 0;
296*5a923131SAndroid Build Coastguard Worker 
297*5a923131SAndroid Build Coastguard Worker   // Obtain start offset, make sure it is within total payload length.
298*5a923131SAndroid Build Coastguard Worker   const size_t start_offset = request.start_offset;
299*5a923131SAndroid Build Coastguard Worker   if (start_offset >= total_length) {
300*5a923131SAndroid Build Coastguard Worker     LOG(WARNING) << "start offset (" << start_offset
301*5a923131SAndroid Build Coastguard Worker                  << ") exceeds total length (" << total_length
302*5a923131SAndroid Build Coastguard Worker                  << "), generating error response ("
303*5a923131SAndroid Build Coastguard Worker                  << kHttpResponseReqRangeNotSat << ")";
304*5a923131SAndroid Build Coastguard Worker     return WriteHeaders(
305*5a923131SAndroid Build Coastguard Worker         fd, total_length, total_length, kHttpResponseReqRangeNotSat);
306*5a923131SAndroid Build Coastguard Worker   }
307*5a923131SAndroid Build Coastguard Worker 
308*5a923131SAndroid Build Coastguard Worker   // Obtain end offset, adjust to fit in total payload length and ensure it does
309*5a923131SAndroid Build Coastguard Worker   // not preceded the start offset.
310*5a923131SAndroid Build Coastguard Worker   size_t end_offset =
311*5a923131SAndroid Build Coastguard Worker       (request.end_offset > 0 ? request.end_offset : total_length);
312*5a923131SAndroid Build Coastguard Worker   if (end_offset < start_offset) {
313*5a923131SAndroid Build Coastguard Worker     LOG(WARNING) << "end offset (" << end_offset << ") precedes start offset ("
314*5a923131SAndroid Build Coastguard Worker                  << start_offset << "), generating error response";
315*5a923131SAndroid Build Coastguard Worker     return WriteHeaders(fd, 0, 0, kHttpResponseBadRequest);
316*5a923131SAndroid Build Coastguard Worker   }
317*5a923131SAndroid Build Coastguard Worker   if (end_offset > total_length) {
318*5a923131SAndroid Build Coastguard Worker     LOG(INFO) << "requested end offset (" << end_offset
319*5a923131SAndroid Build Coastguard Worker               << ") exceeds total length (" << total_length << "), adjusting";
320*5a923131SAndroid Build Coastguard Worker     end_offset = total_length;
321*5a923131SAndroid Build Coastguard Worker   }
322*5a923131SAndroid Build Coastguard Worker 
323*5a923131SAndroid Build Coastguard Worker   // Generate headers
324*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "generating response header: range=" << start_offset << "-"
325*5a923131SAndroid Build Coastguard Worker             << (end_offset - 1) << "/" << (end_offset - start_offset)
326*5a923131SAndroid Build Coastguard Worker             << ", return code=" << request.return_code;
327*5a923131SAndroid Build Coastguard Worker   if ((ret = WriteHeaders(fd, start_offset, end_offset, request.return_code)) <
328*5a923131SAndroid Build Coastguard Worker       0)
329*5a923131SAndroid Build Coastguard Worker     return -1;
330*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << ret << " header bytes written";
331*5a923131SAndroid Build Coastguard Worker   written += ret;
332*5a923131SAndroid Build Coastguard Worker 
333*5a923131SAndroid Build Coastguard Worker   // Compute payload length, truncate as necessary.
334*5a923131SAndroid Build Coastguard Worker   size_t payload_length = end_offset - start_offset;
335*5a923131SAndroid Build Coastguard Worker   if (truncate_length > 0 && truncate_length < payload_length) {
336*5a923131SAndroid Build Coastguard Worker     LOG(INFO) << "truncating request payload length (" << payload_length
337*5a923131SAndroid Build Coastguard Worker               << ") at " << truncate_length;
338*5a923131SAndroid Build Coastguard Worker     payload_length = truncate_length;
339*5a923131SAndroid Build Coastguard Worker     end_offset = start_offset + payload_length;
340*5a923131SAndroid Build Coastguard Worker   }
341*5a923131SAndroid Build Coastguard Worker 
342*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "generating response payload: range=" << start_offset << "-"
343*5a923131SAndroid Build Coastguard Worker             << (end_offset - 1) << "/" << (end_offset - start_offset);
344*5a923131SAndroid Build Coastguard Worker 
345*5a923131SAndroid Build Coastguard Worker   // Decide about optional midway delay.
346*5a923131SAndroid Build Coastguard Worker   if (truncate_length > 0 && sleep_every > 0 && sleep_secs >= 0 &&
347*5a923131SAndroid Build Coastguard Worker       start_offset % (truncate_length * sleep_every) == 0) {
348*5a923131SAndroid Build Coastguard Worker     const off_t midway_offset = start_offset + payload_length / 2;
349*5a923131SAndroid Build Coastguard Worker 
350*5a923131SAndroid Build Coastguard Worker     if ((ret = WritePayload(fd, start_offset, midway_offset)) < 0)
351*5a923131SAndroid Build Coastguard Worker       return -1;
352*5a923131SAndroid Build Coastguard Worker     LOG(INFO) << ret << " payload bytes written (first chunk)";
353*5a923131SAndroid Build Coastguard Worker     written += ret;
354*5a923131SAndroid Build Coastguard Worker 
355*5a923131SAndroid Build Coastguard Worker     LOG(INFO) << "sleeping for " << sleep_secs << " seconds...";
356*5a923131SAndroid Build Coastguard Worker     sleep(sleep_secs);
357*5a923131SAndroid Build Coastguard Worker 
358*5a923131SAndroid Build Coastguard Worker     if ((ret = WritePayload(fd, midway_offset, end_offset)) < 0)
359*5a923131SAndroid Build Coastguard Worker       return -1;
360*5a923131SAndroid Build Coastguard Worker     LOG(INFO) << ret << " payload bytes written (second chunk)";
361*5a923131SAndroid Build Coastguard Worker     written += ret;
362*5a923131SAndroid Build Coastguard Worker   } else {
363*5a923131SAndroid Build Coastguard Worker     if ((ret = WritePayload(fd, start_offset, end_offset)) < 0)
364*5a923131SAndroid Build Coastguard Worker       return -1;
365*5a923131SAndroid Build Coastguard Worker     LOG(INFO) << ret << " payload bytes written";
366*5a923131SAndroid Build Coastguard Worker     written += ret;
367*5a923131SAndroid Build Coastguard Worker   }
368*5a923131SAndroid Build Coastguard Worker 
369*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "response generation complete, " << written
370*5a923131SAndroid Build Coastguard Worker             << " total bytes written";
371*5a923131SAndroid Build Coastguard Worker   return written;
372*5a923131SAndroid Build Coastguard Worker }
373*5a923131SAndroid Build Coastguard Worker 
HandleGet(int fd,const HttpRequest & request,const size_t total_length)374*5a923131SAndroid Build Coastguard Worker ssize_t HandleGet(int fd,
375*5a923131SAndroid Build Coastguard Worker                   const HttpRequest& request,
376*5a923131SAndroid Build Coastguard Worker                   const size_t total_length) {
377*5a923131SAndroid Build Coastguard Worker   return HandleGet(fd, request, total_length, 0, 0, 0);
378*5a923131SAndroid Build Coastguard Worker }
379*5a923131SAndroid Build Coastguard Worker 
380*5a923131SAndroid Build Coastguard Worker // Handles /redirect/<code>/<url> requests by returning the specified
381*5a923131SAndroid Build Coastguard Worker // redirect <code> with a location pointing to /<url>.
HandleRedirect(int fd,const HttpRequest & request)382*5a923131SAndroid Build Coastguard Worker void HandleRedirect(int fd, const HttpRequest& request) {
383*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "Redirecting...";
384*5a923131SAndroid Build Coastguard Worker   string url = request.url;
385*5a923131SAndroid Build Coastguard Worker   CHECK_EQ(static_cast<size_t>(0), url.find("/redirect/"));
386*5a923131SAndroid Build Coastguard Worker   url.erase(0, strlen("/redirect/"));
387*5a923131SAndroid Build Coastguard Worker   string::size_type url_start = url.find('/');
388*5a923131SAndroid Build Coastguard Worker   CHECK_NE(url_start, string::npos);
389*5a923131SAndroid Build Coastguard Worker   HttpResponseCode code = StringToHttpResponseCode(url.c_str());
390*5a923131SAndroid Build Coastguard Worker   url.erase(0, url_start);
391*5a923131SAndroid Build Coastguard Worker   url = "http://" + request.host + url;
392*5a923131SAndroid Build Coastguard Worker   const char* status = GetHttpResponseDescription(code);
393*5a923131SAndroid Build Coastguard Worker   if (!status)
394*5a923131SAndroid Build Coastguard Worker     CHECK(false) << "Unrecognized redirection code: " << code;
395*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "Code: " << code << " " << status;
396*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "New URL: " << url;
397*5a923131SAndroid Build Coastguard Worker 
398*5a923131SAndroid Build Coastguard Worker   ssize_t ret{};
399*5a923131SAndroid Build Coastguard Worker   if ((ret = WriteString(fd, "HTTP/1.1 " + Itoa(code) + " " + status + EOL)) <
400*5a923131SAndroid Build Coastguard Worker       0)
401*5a923131SAndroid Build Coastguard Worker     return;
402*5a923131SAndroid Build Coastguard Worker   WriteString(fd, "Connection: close" EOL);
403*5a923131SAndroid Build Coastguard Worker   WriteString(fd, "Location: " + url + EOL);
404*5a923131SAndroid Build Coastguard Worker }
405*5a923131SAndroid Build Coastguard Worker 
406*5a923131SAndroid Build Coastguard Worker // Generate a page not found error response with actual text payload. Return
407*5a923131SAndroid Build Coastguard Worker // number of bytes written or -1 for error.
HandleError(int fd,const HttpRequest & request)408*5a923131SAndroid Build Coastguard Worker ssize_t HandleError(int fd, const HttpRequest& request) {
409*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "Generating error HTTP response";
410*5a923131SAndroid Build Coastguard Worker 
411*5a923131SAndroid Build Coastguard Worker   ssize_t ret{};
412*5a923131SAndroid Build Coastguard Worker   size_t written = 0;
413*5a923131SAndroid Build Coastguard Worker 
414*5a923131SAndroid Build Coastguard Worker   const string data("This is an error page.");
415*5a923131SAndroid Build Coastguard Worker 
416*5a923131SAndroid Build Coastguard Worker   if ((ret = WriteHeaders(fd, 0, data.size(), kHttpResponseNotFound)) < 0)
417*5a923131SAndroid Build Coastguard Worker     return -1;
418*5a923131SAndroid Build Coastguard Worker   written += ret;
419*5a923131SAndroid Build Coastguard Worker 
420*5a923131SAndroid Build Coastguard Worker   if ((ret = WriteString(fd, data)) < 0)
421*5a923131SAndroid Build Coastguard Worker     return -1;
422*5a923131SAndroid Build Coastguard Worker   written += ret;
423*5a923131SAndroid Build Coastguard Worker 
424*5a923131SAndroid Build Coastguard Worker   return written;
425*5a923131SAndroid Build Coastguard Worker }
426*5a923131SAndroid Build Coastguard Worker 
427*5a923131SAndroid Build Coastguard Worker // Generate an error response if the requested offset is nonzero, up to a given
428*5a923131SAndroid Build Coastguard Worker // maximal number of successive failures.  The error generated is an "Internal
429*5a923131SAndroid Build Coastguard Worker // Server Error" (500).
HandleErrorIfOffset(int fd,const HttpRequest & request,size_t end_offset,int max_fails)430*5a923131SAndroid Build Coastguard Worker ssize_t HandleErrorIfOffset(int fd,
431*5a923131SAndroid Build Coastguard Worker                             const HttpRequest& request,
432*5a923131SAndroid Build Coastguard Worker                             size_t end_offset,
433*5a923131SAndroid Build Coastguard Worker                             int max_fails) {
434*5a923131SAndroid Build Coastguard Worker   static int num_fails = 0;
435*5a923131SAndroid Build Coastguard Worker 
436*5a923131SAndroid Build Coastguard Worker   if (request.start_offset > 0 && num_fails < max_fails) {
437*5a923131SAndroid Build Coastguard Worker     LOG(INFO) << "Generating error HTTP response";
438*5a923131SAndroid Build Coastguard Worker 
439*5a923131SAndroid Build Coastguard Worker     ssize_t ret{};
440*5a923131SAndroid Build Coastguard Worker     size_t written = 0;
441*5a923131SAndroid Build Coastguard Worker 
442*5a923131SAndroid Build Coastguard Worker     const string data("This is an error page.");
443*5a923131SAndroid Build Coastguard Worker 
444*5a923131SAndroid Build Coastguard Worker     if ((ret = WriteHeaders(
445*5a923131SAndroid Build Coastguard Worker              fd, 0, data.size(), kHttpResponseInternalServerError)) < 0)
446*5a923131SAndroid Build Coastguard Worker       return -1;
447*5a923131SAndroid Build Coastguard Worker     written += ret;
448*5a923131SAndroid Build Coastguard Worker 
449*5a923131SAndroid Build Coastguard Worker     if ((ret = WriteString(fd, data)) < 0)
450*5a923131SAndroid Build Coastguard Worker       return -1;
451*5a923131SAndroid Build Coastguard Worker     written += ret;
452*5a923131SAndroid Build Coastguard Worker 
453*5a923131SAndroid Build Coastguard Worker     num_fails++;
454*5a923131SAndroid Build Coastguard Worker     return written;
455*5a923131SAndroid Build Coastguard Worker   } else {
456*5a923131SAndroid Build Coastguard Worker     num_fails = 0;
457*5a923131SAndroid Build Coastguard Worker     return HandleGet(fd, request, end_offset);
458*5a923131SAndroid Build Coastguard Worker   }
459*5a923131SAndroid Build Coastguard Worker }
460*5a923131SAndroid Build Coastguard Worker 
461*5a923131SAndroid Build Coastguard Worker // Returns a valid response echoing in the body of the response all the headers
462*5a923131SAndroid Build Coastguard Worker // sent by the client.
HandleEchoHeaders(int fd,const HttpRequest & request)463*5a923131SAndroid Build Coastguard Worker void HandleEchoHeaders(int fd, const HttpRequest& request) {
464*5a923131SAndroid Build Coastguard Worker   WriteHeaders(fd, 0, request.raw_headers.size(), kHttpResponseOk);
465*5a923131SAndroid Build Coastguard Worker   WriteString(fd, request.raw_headers);
466*5a923131SAndroid Build Coastguard Worker }
467*5a923131SAndroid Build Coastguard Worker 
HandleHang(int fd)468*5a923131SAndroid Build Coastguard Worker void HandleHang(int fd) {
469*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "Hanging until the other side of the connection is closed.";
470*5a923131SAndroid Build Coastguard Worker   char c{};
471*5a923131SAndroid Build Coastguard Worker   while (HANDLE_EINTR(read(fd, &c, 1)) > 0) {
472*5a923131SAndroid Build Coastguard Worker   }
473*5a923131SAndroid Build Coastguard Worker }
474*5a923131SAndroid Build Coastguard Worker 
HandleDefault(int fd,const HttpRequest & request)475*5a923131SAndroid Build Coastguard Worker void HandleDefault(int fd, const HttpRequest& request) {
476*5a923131SAndroid Build Coastguard Worker   const off_t start_offset = request.start_offset;
477*5a923131SAndroid Build Coastguard Worker   const string data("unhandled path");
478*5a923131SAndroid Build Coastguard Worker   const size_t size = data.size();
479*5a923131SAndroid Build Coastguard Worker   ssize_t ret{};
480*5a923131SAndroid Build Coastguard Worker 
481*5a923131SAndroid Build Coastguard Worker   if ((ret = WriteHeaders(fd, start_offset, size, request.return_code)) < 0)
482*5a923131SAndroid Build Coastguard Worker     return;
483*5a923131SAndroid Build Coastguard Worker   WriteString(
484*5a923131SAndroid Build Coastguard Worker       fd,
485*5a923131SAndroid Build Coastguard Worker       (start_offset < static_cast<off_t>(size) ? data.substr(start_offset)
486*5a923131SAndroid Build Coastguard Worker                                                : ""));
487*5a923131SAndroid Build Coastguard Worker }
488*5a923131SAndroid Build Coastguard Worker 
489*5a923131SAndroid Build Coastguard Worker // Break a URL into terms delimited by slashes.
490*5a923131SAndroid Build Coastguard Worker class UrlTerms {
491*5a923131SAndroid Build Coastguard Worker  public:
UrlTerms(const string & url,size_t num_terms)492*5a923131SAndroid Build Coastguard Worker   UrlTerms(const string& url, size_t num_terms) {
493*5a923131SAndroid Build Coastguard Worker     // URL must be non-empty and start with a slash.
494*5a923131SAndroid Build Coastguard Worker     CHECK_GT(url.size(), static_cast<size_t>(0));
495*5a923131SAndroid Build Coastguard Worker     CHECK_EQ(url[0], '/');
496*5a923131SAndroid Build Coastguard Worker 
497*5a923131SAndroid Build Coastguard Worker     // Split it into terms delimited by slashes, omitting the preceding slash.
498*5a923131SAndroid Build Coastguard Worker     terms = base::SplitString(
499*5a923131SAndroid Build Coastguard Worker         url.substr(1), "/", base::KEEP_WHITESPACE, base::SPLIT_WANT_ALL);
500*5a923131SAndroid Build Coastguard Worker 
501*5a923131SAndroid Build Coastguard Worker     // Ensure expected length.
502*5a923131SAndroid Build Coastguard Worker     CHECK_EQ(terms.size(), num_terms);
503*5a923131SAndroid Build Coastguard Worker   }
504*5a923131SAndroid Build Coastguard Worker 
Get(const off_t index) const505*5a923131SAndroid Build Coastguard Worker   inline const string& Get(const off_t index) const { return terms[index]; }
GetCStr(const off_t index) const506*5a923131SAndroid Build Coastguard Worker   inline const char* GetCStr(const off_t index) const {
507*5a923131SAndroid Build Coastguard Worker     return Get(index).c_str();
508*5a923131SAndroid Build Coastguard Worker   }
GetInt(const off_t index) const509*5a923131SAndroid Build Coastguard Worker   inline int GetInt(const off_t index) const { return atoi(GetCStr(index)); }
GetSizeT(const off_t index) const510*5a923131SAndroid Build Coastguard Worker   inline size_t GetSizeT(const off_t index) const {
511*5a923131SAndroid Build Coastguard Worker     return static_cast<size_t>(atol(GetCStr(index)));
512*5a923131SAndroid Build Coastguard Worker   }
513*5a923131SAndroid Build Coastguard Worker 
514*5a923131SAndroid Build Coastguard Worker  private:
515*5a923131SAndroid Build Coastguard Worker   vector<string> terms;
516*5a923131SAndroid Build Coastguard Worker };
517*5a923131SAndroid Build Coastguard Worker 
HandleConnection(int fd)518*5a923131SAndroid Build Coastguard Worker void HandleConnection(int fd) {
519*5a923131SAndroid Build Coastguard Worker   HttpRequest request;
520*5a923131SAndroid Build Coastguard Worker   ParseRequest(fd, &request);
521*5a923131SAndroid Build Coastguard Worker 
522*5a923131SAndroid Build Coastguard Worker   string& url = request.url;
523*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << "pid(" << getpid() << "): handling url " << url;
524*5a923131SAndroid Build Coastguard Worker   if (url == "/quitquitquit") {
525*5a923131SAndroid Build Coastguard Worker     HandleQuit(fd);
526*5a923131SAndroid Build Coastguard Worker   } else if (android::base::StartsWith(url, "/download/")) {
527*5a923131SAndroid Build Coastguard Worker     const UrlTerms terms(url, 2);
528*5a923131SAndroid Build Coastguard Worker     HandleGet(fd, request, terms.GetSizeT(1));
529*5a923131SAndroid Build Coastguard Worker   } else if (android::base::StartsWith(url, "/flaky/")) {
530*5a923131SAndroid Build Coastguard Worker     const UrlTerms terms(url, 5);
531*5a923131SAndroid Build Coastguard Worker     HandleGet(fd,
532*5a923131SAndroid Build Coastguard Worker               request,
533*5a923131SAndroid Build Coastguard Worker               terms.GetSizeT(1),
534*5a923131SAndroid Build Coastguard Worker               terms.GetSizeT(2),
535*5a923131SAndroid Build Coastguard Worker               terms.GetInt(3),
536*5a923131SAndroid Build Coastguard Worker               terms.GetInt(4));
537*5a923131SAndroid Build Coastguard Worker   } else if (url.find("/redirect/") == 0) {
538*5a923131SAndroid Build Coastguard Worker     HandleRedirect(fd, request);
539*5a923131SAndroid Build Coastguard Worker   } else if (url == "/error") {
540*5a923131SAndroid Build Coastguard Worker     HandleError(fd, request);
541*5a923131SAndroid Build Coastguard Worker   } else if (android::base::StartsWith(url, "/error-if-offset/")) {
542*5a923131SAndroid Build Coastguard Worker     const UrlTerms terms(url, 3);
543*5a923131SAndroid Build Coastguard Worker     HandleErrorIfOffset(fd, request, terms.GetSizeT(1), terms.GetInt(2));
544*5a923131SAndroid Build Coastguard Worker   } else if (url == "/echo-headers") {
545*5a923131SAndroid Build Coastguard Worker     HandleEchoHeaders(fd, request);
546*5a923131SAndroid Build Coastguard Worker   } else if (url == "/hang") {
547*5a923131SAndroid Build Coastguard Worker     HandleHang(fd);
548*5a923131SAndroid Build Coastguard Worker   } else {
549*5a923131SAndroid Build Coastguard Worker     HandleDefault(fd, request);
550*5a923131SAndroid Build Coastguard Worker   }
551*5a923131SAndroid Build Coastguard Worker 
552*5a923131SAndroid Build Coastguard Worker   close(fd);
553*5a923131SAndroid Build Coastguard Worker }
554*5a923131SAndroid Build Coastguard Worker 
555*5a923131SAndroid Build Coastguard Worker }  // namespace chromeos_update_engine
556*5a923131SAndroid Build Coastguard Worker 
557*5a923131SAndroid Build Coastguard Worker using namespace chromeos_update_engine;  // NOLINT(build/namespaces)
558*5a923131SAndroid Build Coastguard Worker 
usage(const char * prog_arg)559*5a923131SAndroid Build Coastguard Worker void usage(const char* prog_arg) {
560*5a923131SAndroid Build Coastguard Worker   fprintf(stderr,
561*5a923131SAndroid Build Coastguard Worker           "Usage: %s [ FILE ]\n"
562*5a923131SAndroid Build Coastguard Worker           "Once accepting connections, the following is written to FILE (or "
563*5a923131SAndroid Build Coastguard Worker           "stdout):\n"
564*5a923131SAndroid Build Coastguard Worker           "\"%sN\" (where N is an integer port number)\n",
565*5a923131SAndroid Build Coastguard Worker           basename(prog_arg),
566*5a923131SAndroid Build Coastguard Worker           kListeningMsgPrefix);
567*5a923131SAndroid Build Coastguard Worker }
568*5a923131SAndroid Build Coastguard Worker 
main(int argc,char ** argv)569*5a923131SAndroid Build Coastguard Worker int main(int argc, char** argv) {
570*5a923131SAndroid Build Coastguard Worker   // Check invocation.
571*5a923131SAndroid Build Coastguard Worker   if (argc > 2)
572*5a923131SAndroid Build Coastguard Worker     errx(RC_BAD_ARGS, "unexpected number of arguments (use -h for usage)");
573*5a923131SAndroid Build Coastguard Worker 
574*5a923131SAndroid Build Coastguard Worker   // Parse (optional) argument.
575*5a923131SAndroid Build Coastguard Worker   int report_fd = STDOUT_FILENO;
576*5a923131SAndroid Build Coastguard Worker   if (argc == 2) {
577*5a923131SAndroid Build Coastguard Worker     if (!strcmp(argv[1], "-h")) {
578*5a923131SAndroid Build Coastguard Worker       usage(argv[0]);
579*5a923131SAndroid Build Coastguard Worker       exit(RC_OK);
580*5a923131SAndroid Build Coastguard Worker     }
581*5a923131SAndroid Build Coastguard Worker 
582*5a923131SAndroid Build Coastguard Worker     report_fd = open(argv[1], O_WRONLY | O_CREAT, 00644);
583*5a923131SAndroid Build Coastguard Worker   }
584*5a923131SAndroid Build Coastguard Worker 
585*5a923131SAndroid Build Coastguard Worker   // Ignore SIGPIPE on write() to sockets.
586*5a923131SAndroid Build Coastguard Worker   signal(SIGPIPE, SIG_IGN);
587*5a923131SAndroid Build Coastguard Worker 
588*5a923131SAndroid Build Coastguard Worker   int listen_fd = socket(AF_INET, SOCK_STREAM, 0);
589*5a923131SAndroid Build Coastguard Worker   if (listen_fd < 0)
590*5a923131SAndroid Build Coastguard Worker     LOG(FATAL) << "socket() failed";
591*5a923131SAndroid Build Coastguard Worker 
592*5a923131SAndroid Build Coastguard Worker   struct sockaddr_in server_addr = sockaddr_in();
593*5a923131SAndroid Build Coastguard Worker   server_addr.sin_family = AF_INET;
594*5a923131SAndroid Build Coastguard Worker   server_addr.sin_addr.s_addr = INADDR_ANY;
595*5a923131SAndroid Build Coastguard Worker   server_addr.sin_port = 0;
596*5a923131SAndroid Build Coastguard Worker 
597*5a923131SAndroid Build Coastguard Worker   {
598*5a923131SAndroid Build Coastguard Worker     // Get rid of "Address in use" error
599*5a923131SAndroid Build Coastguard Worker     int tr = 1;
600*5a923131SAndroid Build Coastguard Worker     if (setsockopt(listen_fd, SOL_SOCKET, SO_REUSEADDR, &tr, sizeof(int)) ==
601*5a923131SAndroid Build Coastguard Worker         -1) {
602*5a923131SAndroid Build Coastguard Worker       perror("setsockopt");
603*5a923131SAndroid Build Coastguard Worker       exit(RC_ERR_SETSOCKOPT);
604*5a923131SAndroid Build Coastguard Worker     }
605*5a923131SAndroid Build Coastguard Worker   }
606*5a923131SAndroid Build Coastguard Worker 
607*5a923131SAndroid Build Coastguard Worker   // Bind the socket and set for listening.
608*5a923131SAndroid Build Coastguard Worker   if (bind(listen_fd,
609*5a923131SAndroid Build Coastguard Worker            reinterpret_cast<struct sockaddr*>(&server_addr),
610*5a923131SAndroid Build Coastguard Worker            sizeof(server_addr)) < 0) {
611*5a923131SAndroid Build Coastguard Worker     perror("bind");
612*5a923131SAndroid Build Coastguard Worker     exit(RC_ERR_BIND);
613*5a923131SAndroid Build Coastguard Worker   }
614*5a923131SAndroid Build Coastguard Worker   if (listen(listen_fd, 5) < 0) {
615*5a923131SAndroid Build Coastguard Worker     perror("listen");
616*5a923131SAndroid Build Coastguard Worker     exit(RC_ERR_LISTEN);
617*5a923131SAndroid Build Coastguard Worker   }
618*5a923131SAndroid Build Coastguard Worker 
619*5a923131SAndroid Build Coastguard Worker   // Check the actual port.
620*5a923131SAndroid Build Coastguard Worker   struct sockaddr_in bound_addr = sockaddr_in();
621*5a923131SAndroid Build Coastguard Worker   socklen_t bound_addr_len = sizeof(bound_addr);
622*5a923131SAndroid Build Coastguard Worker   if (getsockname(listen_fd,
623*5a923131SAndroid Build Coastguard Worker                   reinterpret_cast<struct sockaddr*>(&bound_addr),
624*5a923131SAndroid Build Coastguard Worker                   &bound_addr_len) < 0) {
625*5a923131SAndroid Build Coastguard Worker     perror("getsockname");
626*5a923131SAndroid Build Coastguard Worker     exit(RC_ERR_GETSOCKNAME);
627*5a923131SAndroid Build Coastguard Worker   }
628*5a923131SAndroid Build Coastguard Worker   in_port_t port = ntohs(bound_addr.sin_port);
629*5a923131SAndroid Build Coastguard Worker 
630*5a923131SAndroid Build Coastguard Worker   // Output the listening port, indicating that the server is processing
631*5a923131SAndroid Build Coastguard Worker   // requests. IMPORTANT! (a) the format of this message is as expected by some
632*5a923131SAndroid Build Coastguard Worker   // unit tests, avoid unilateral changes; (b) it is necessary to flush/sync the
633*5a923131SAndroid Build Coastguard Worker   // file to prevent the spawning process from waiting indefinitely for this
634*5a923131SAndroid Build Coastguard Worker   // message.
635*5a923131SAndroid Build Coastguard Worker   string listening_msg =
636*5a923131SAndroid Build Coastguard Worker       android::base::StringPrintf("%s%hu", kListeningMsgPrefix, port);
637*5a923131SAndroid Build Coastguard Worker   LOG(INFO) << listening_msg;
638*5a923131SAndroid Build Coastguard Worker   CHECK_EQ(write(report_fd, listening_msg.c_str(), listening_msg.length()),
639*5a923131SAndroid Build Coastguard Worker            static_cast<int>(listening_msg.length()));
640*5a923131SAndroid Build Coastguard Worker   CHECK_EQ(write(report_fd, "\n", 1), 1);
641*5a923131SAndroid Build Coastguard Worker   if (report_fd == STDOUT_FILENO)
642*5a923131SAndroid Build Coastguard Worker     fsync(report_fd);
643*5a923131SAndroid Build Coastguard Worker   else
644*5a923131SAndroid Build Coastguard Worker     close(report_fd);
645*5a923131SAndroid Build Coastguard Worker 
646*5a923131SAndroid Build Coastguard Worker   while (1) {
647*5a923131SAndroid Build Coastguard Worker     LOG(INFO) << "pid(" << getpid() << "): waiting to accept new connection";
648*5a923131SAndroid Build Coastguard Worker     int client_fd = accept(listen_fd, nullptr, nullptr);
649*5a923131SAndroid Build Coastguard Worker     LOG(INFO) << "got past accept";
650*5a923131SAndroid Build Coastguard Worker     if (client_fd < 0)
651*5a923131SAndroid Build Coastguard Worker       LOG(FATAL) << "ERROR on accept";
652*5a923131SAndroid Build Coastguard Worker     HandleConnection(client_fd);
653*5a923131SAndroid Build Coastguard Worker   }
654*5a923131SAndroid Build Coastguard Worker }
655