1 // Copyright 2014 The Chromium OS Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include <brillo/http/http_utils.h>
6 
7 #include <algorithm>
8 
9 #include <base/bind.h>
10 #include <base/json/json_reader.h>
11 #include <base/json/json_writer.h>
12 #include <base/values.h>
13 #include <brillo/data_encoding.h>
14 #include <brillo/errors/error_codes.h>
15 #include <brillo/mime_utils.h>
16 #include <brillo/streams/memory_stream.h>
17 
18 using brillo::mime::AppendParameter;
19 using brillo::mime::RemoveParameters;
20 
21 namespace brillo {
22 namespace http {
23 
GetAndBlock(const std::string & url,const HeaderList & headers,std::shared_ptr<Transport> transport,brillo::ErrorPtr * error)24 std::unique_ptr<Response> GetAndBlock(const std::string& url,
25                                       const HeaderList& headers,
26                                       std::shared_ptr<Transport> transport,
27                                       brillo::ErrorPtr* error) {
28   return SendRequestWithNoDataAndBlock(
29       request_type::kGet, url, headers, transport, error);
30 }
31 
Get(const std::string & url,const HeaderList & headers,std::shared_ptr<Transport> transport,const SuccessCallback & success_callback,const ErrorCallback & error_callback)32 RequestID Get(const std::string& url,
33               const HeaderList& headers,
34               std::shared_ptr<Transport> transport,
35               const SuccessCallback& success_callback,
36               const ErrorCallback& error_callback) {
37   return SendRequestWithNoData(request_type::kGet,
38                                url,
39                                headers,
40                                transport,
41                                success_callback,
42                                error_callback);
43 }
44 
HeadAndBlock(const std::string & url,std::shared_ptr<Transport> transport,brillo::ErrorPtr * error)45 std::unique_ptr<Response> HeadAndBlock(const std::string& url,
46                                        std::shared_ptr<Transport> transport,
47                                        brillo::ErrorPtr* error) {
48   return SendRequestWithNoDataAndBlock(
49       request_type::kHead, url, {}, transport, error);
50 }
51 
Head(const std::string & url,std::shared_ptr<Transport> transport,const SuccessCallback & success_callback,const ErrorCallback & error_callback)52 RequestID Head(const std::string& url,
53                std::shared_ptr<Transport> transport,
54                const SuccessCallback& success_callback,
55                const ErrorCallback& error_callback) {
56   return SendRequestWithNoData(request_type::kHead,
57                                url,
58                                {},
59                                transport,
60                                success_callback,
61                                error_callback);
62 }
63 
PostTextAndBlock(const std::string & url,const std::string & data,const std::string & mime_type,const HeaderList & headers,std::shared_ptr<Transport> transport,brillo::ErrorPtr * error)64 std::unique_ptr<Response> PostTextAndBlock(const std::string& url,
65                                            const std::string& data,
66                                            const std::string& mime_type,
67                                            const HeaderList& headers,
68                                            std::shared_ptr<Transport> transport,
69                                            brillo::ErrorPtr* error) {
70   return PostBinaryAndBlock(
71       url, data.data(), data.size(), mime_type, headers, transport, error);
72 }
73 
PostText(const std::string & url,const std::string & data,const std::string & mime_type,const HeaderList & headers,std::shared_ptr<Transport> transport,const SuccessCallback & success_callback,const ErrorCallback & error_callback)74 RequestID PostText(const std::string& url,
75                    const std::string& data,
76                    const std::string& mime_type,
77                    const HeaderList& headers,
78                    std::shared_ptr<Transport> transport,
79                    const SuccessCallback& success_callback,
80                    const ErrorCallback& error_callback) {
81   return PostBinary(url,
82                     data.data(),
83                     data.size(),
84                     mime_type,
85                     headers,
86                     transport,
87                     success_callback,
88                     error_callback);
89 }
90 
SendRequestAndBlock(const std::string & method,const std::string & url,const void * data,size_t data_size,const std::string & mime_type,const HeaderList & headers,std::shared_ptr<Transport> transport,brillo::ErrorPtr * error)91 std::unique_ptr<Response> SendRequestAndBlock(
92     const std::string& method,
93     const std::string& url,
94     const void* data,
95     size_t data_size,
96     const std::string& mime_type,
97     const HeaderList& headers,
98     std::shared_ptr<Transport> transport,
99     brillo::ErrorPtr* error) {
100   Request request(url, method, transport);
101   request.AddHeaders(headers);
102   if (data_size > 0) {
103     CHECK(!mime_type.empty()) << "MIME type must be specified if request body "
104                                  "message is provided";
105     request.SetContentType(mime_type);
106     if (!request.AddRequestBody(data, data_size, error))
107       return std::unique_ptr<Response>();
108   }
109   return request.GetResponseAndBlock(error);
110 }
111 
SendRequestWithNoDataAndBlock(const std::string & method,const std::string & url,const HeaderList & headers,std::shared_ptr<Transport> transport,brillo::ErrorPtr * error)112 std::unique_ptr<Response> SendRequestWithNoDataAndBlock(
113     const std::string& method,
114     const std::string& url,
115     const HeaderList& headers,
116     std::shared_ptr<Transport> transport,
117     brillo::ErrorPtr* error) {
118   return SendRequestAndBlock(
119       method, url, nullptr, 0, {}, headers, transport, error);
120 }
121 
SendRequest(const std::string & method,const std::string & url,StreamPtr stream,const std::string & mime_type,const HeaderList & headers,std::shared_ptr<Transport> transport,const SuccessCallback & success_callback,const ErrorCallback & error_callback)122 RequestID SendRequest(const std::string& method,
123                       const std::string& url,
124                       StreamPtr stream,
125                       const std::string& mime_type,
126                       const HeaderList& headers,
127                       std::shared_ptr<Transport> transport,
128                       const SuccessCallback& success_callback,
129                       const ErrorCallback& error_callback) {
130   Request request(url, method, transport);
131   request.AddHeaders(headers);
132   if (stream && (!stream->CanGetSize() || stream->GetRemainingSize() > 0)) {
133     CHECK(!mime_type.empty()) << "MIME type must be specified if request body "
134                                  "message is provided";
135     request.SetContentType(mime_type);
136     brillo::ErrorPtr error;
137     if (!request.AddRequestBody(std::move(stream), &error)) {
138       transport->RunCallbackAsync(
139           FROM_HERE, base::Bind(error_callback,
140                                 0, base::Owned(error.release())));
141       return 0;
142     }
143   }
144   return request.GetResponse(success_callback, error_callback);
145 }
146 
SendRequest(const std::string & method,const std::string & url,const void * data,size_t data_size,const std::string & mime_type,const HeaderList & headers,std::shared_ptr<Transport> transport,const SuccessCallback & success_callback,const ErrorCallback & error_callback)147 RequestID SendRequest(const std::string& method,
148                       const std::string& url,
149                       const void* data,
150                       size_t data_size,
151                       const std::string& mime_type,
152                       const HeaderList& headers,
153                       std::shared_ptr<Transport> transport,
154                       const SuccessCallback& success_callback,
155                       const ErrorCallback& error_callback) {
156   return SendRequest(method,
157                      url,
158                      MemoryStream::OpenCopyOf(data, data_size, nullptr),
159                      mime_type,
160                      headers,
161                      transport,
162                      success_callback,
163                      error_callback);
164 }
165 
SendRequestWithNoData(const std::string & method,const std::string & url,const HeaderList & headers,std::shared_ptr<Transport> transport,const SuccessCallback & success_callback,const ErrorCallback & error_callback)166 RequestID SendRequestWithNoData(const std::string& method,
167                                 const std::string& url,
168                                 const HeaderList& headers,
169                                 std::shared_ptr<Transport> transport,
170                                 const SuccessCallback& success_callback,
171                                 const ErrorCallback& error_callback) {
172   return SendRequest(method,
173                      url,
174                      {},
175                      {},
176                      headers,
177                      transport,
178                      success_callback,
179                      error_callback);
180 }
181 
PostBinaryAndBlock(const std::string & url,const void * data,size_t data_size,const std::string & mime_type,const HeaderList & headers,std::shared_ptr<Transport> transport,brillo::ErrorPtr * error)182 std::unique_ptr<Response> PostBinaryAndBlock(
183     const std::string& url,
184     const void* data,
185     size_t data_size,
186     const std::string& mime_type,
187     const HeaderList& headers,
188     std::shared_ptr<Transport> transport,
189     brillo::ErrorPtr* error) {
190   return SendRequestAndBlock(request_type::kPost,
191                              url,
192                              data,
193                              data_size,
194                              mime_type,
195                              headers,
196                              transport,
197                              error);
198 }
199 
PostBinary(const std::string & url,StreamPtr stream,const std::string & mime_type,const HeaderList & headers,std::shared_ptr<Transport> transport,const SuccessCallback & success_callback,const ErrorCallback & error_callback)200 RequestID PostBinary(const std::string& url,
201                      StreamPtr stream,
202                      const std::string& mime_type,
203                      const HeaderList& headers,
204                      std::shared_ptr<Transport> transport,
205                      const SuccessCallback& success_callback,
206                      const ErrorCallback& error_callback) {
207   return SendRequest(request_type::kPost,
208                      url,
209                      std::move(stream),
210                      mime_type,
211                      headers,
212                      transport,
213                      success_callback,
214                      error_callback);
215 }
216 
PostBinary(const std::string & url,const void * data,size_t data_size,const std::string & mime_type,const HeaderList & headers,std::shared_ptr<Transport> transport,const SuccessCallback & success_callback,const ErrorCallback & error_callback)217 RequestID PostBinary(const std::string& url,
218                      const void* data,
219                      size_t data_size,
220                      const std::string& mime_type,
221                      const HeaderList& headers,
222                      std::shared_ptr<Transport> transport,
223                      const SuccessCallback& success_callback,
224                      const ErrorCallback& error_callback) {
225   return SendRequest(request_type::kPost,
226                      url,
227                      data,
228                      data_size,
229                      mime_type,
230                      headers,
231                      transport,
232                      success_callback,
233                      error_callback);
234 }
235 
PostFormDataAndBlock(const std::string & url,const FormFieldList & data,const HeaderList & headers,std::shared_ptr<Transport> transport,brillo::ErrorPtr * error)236 std::unique_ptr<Response> PostFormDataAndBlock(
237     const std::string& url,
238     const FormFieldList& data,
239     const HeaderList& headers,
240     std::shared_ptr<Transport> transport,
241     brillo::ErrorPtr* error) {
242   std::string encoded_data = brillo::data_encoding::WebParamsEncode(data);
243   return PostBinaryAndBlock(url,
244                             encoded_data.c_str(),
245                             encoded_data.size(),
246                             brillo::mime::application::kWwwFormUrlEncoded,
247                             headers,
248                             transport,
249                             error);
250 }
251 
PostFormDataAndBlock(const std::string & url,std::unique_ptr<FormData> form_data,const HeaderList & headers,std::shared_ptr<Transport> transport,brillo::ErrorPtr * error)252 std::unique_ptr<Response> PostFormDataAndBlock(
253     const std::string& url,
254     std::unique_ptr<FormData> form_data,
255     const HeaderList& headers,
256     std::shared_ptr<Transport> transport,
257     brillo::ErrorPtr* error) {
258   Request request(url, request_type::kPost, transport);
259   request.AddHeaders(headers);
260   if (!request.AddRequestBodyAsFormData(std::move(form_data), error))
261     return std::unique_ptr<Response>();
262   return request.GetResponseAndBlock(error);
263 }
264 
PostFormData(const std::string & url,const FormFieldList & data,const HeaderList & headers,std::shared_ptr<Transport> transport,const SuccessCallback & success_callback,const ErrorCallback & error_callback)265 RequestID PostFormData(const std::string& url,
266                        const FormFieldList& data,
267                        const HeaderList& headers,
268                        std::shared_ptr<Transport> transport,
269                        const SuccessCallback& success_callback,
270                        const ErrorCallback& error_callback) {
271   std::string encoded_data = brillo::data_encoding::WebParamsEncode(data);
272   return PostBinary(url,
273                     encoded_data.c_str(),
274                     encoded_data.size(),
275                     brillo::mime::application::kWwwFormUrlEncoded,
276                     headers,
277                     transport,
278                     success_callback,
279                     error_callback);
280 }
281 
PostFormData(const std::string & url,std::unique_ptr<FormData> form_data,const HeaderList & headers,std::shared_ptr<Transport> transport,const SuccessCallback & success_callback,const ErrorCallback & error_callback)282 RequestID PostFormData(const std::string& url,
283                        std::unique_ptr<FormData> form_data,
284                        const HeaderList& headers,
285                        std::shared_ptr<Transport> transport,
286                        const SuccessCallback& success_callback,
287                        const ErrorCallback& error_callback) {
288   Request request(url, request_type::kPost, transport);
289   request.AddHeaders(headers);
290   brillo::ErrorPtr error;
291   if (!request.AddRequestBodyAsFormData(std::move(form_data), &error)) {
292     transport->RunCallbackAsync(
293         FROM_HERE, base::Bind(error_callback, 0, base::Owned(error.release())));
294     return 0;
295   }
296   return request.GetResponse(success_callback, error_callback);
297 }
298 
PostJsonAndBlock(const std::string & url,const base::Value * json,const HeaderList & headers,std::shared_ptr<Transport> transport,brillo::ErrorPtr * error)299 std::unique_ptr<Response> PostJsonAndBlock(const std::string& url,
300                                            const base::Value* json,
301                                            const HeaderList& headers,
302                                            std::shared_ptr<Transport> transport,
303                                            brillo::ErrorPtr* error) {
304   std::string data;
305   if (json)
306     base::JSONWriter::Write(*json, &data);
307   std::string mime_type = AppendParameter(brillo::mime::application::kJson,
308                                           brillo::mime::parameters::kCharset,
309                                           "utf-8");
310   return PostBinaryAndBlock(
311       url, data.c_str(), data.size(), mime_type, headers, transport, error);
312 }
313 
PostJson(const std::string & url,std::unique_ptr<base::Value> json,const HeaderList & headers,std::shared_ptr<Transport> transport,const SuccessCallback & success_callback,const ErrorCallback & error_callback)314 RequestID PostJson(const std::string& url,
315                    std::unique_ptr<base::Value> json,
316                    const HeaderList& headers,
317                    std::shared_ptr<Transport> transport,
318                    const SuccessCallback& success_callback,
319                    const ErrorCallback& error_callback) {
320   std::string data;
321   if (json)
322     base::JSONWriter::Write(*json, &data);
323   std::string mime_type = AppendParameter(brillo::mime::application::kJson,
324                                           brillo::mime::parameters::kCharset,
325                                           "utf-8");
326   return PostBinary(url,
327                     data.c_str(),
328                     data.size(),
329                     mime_type,
330                     headers,
331                     transport,
332                     success_callback,
333                     error_callback);
334 }
335 
PatchJsonAndBlock(const std::string & url,const base::Value * json,const HeaderList & headers,std::shared_ptr<Transport> transport,brillo::ErrorPtr * error)336 std::unique_ptr<Response> PatchJsonAndBlock(
337     const std::string& url,
338     const base::Value* json,
339     const HeaderList& headers,
340     std::shared_ptr<Transport> transport,
341     brillo::ErrorPtr* error) {
342   std::string data;
343   if (json)
344     base::JSONWriter::Write(*json, &data);
345   std::string mime_type = AppendParameter(brillo::mime::application::kJson,
346                                           brillo::mime::parameters::kCharset,
347                                           "utf-8");
348   return SendRequestAndBlock(request_type::kPatch,
349                              url,
350                              data.c_str(),
351                              data.size(),
352                              mime_type,
353                              headers,
354                              transport,
355                              error);
356 }
357 
PatchJson(const std::string & url,std::unique_ptr<base::Value> json,const HeaderList & headers,std::shared_ptr<Transport> transport,const SuccessCallback & success_callback,const ErrorCallback & error_callback)358 RequestID PatchJson(const std::string& url,
359                     std::unique_ptr<base::Value> json,
360                     const HeaderList& headers,
361                     std::shared_ptr<Transport> transport,
362                     const SuccessCallback& success_callback,
363                     const ErrorCallback& error_callback) {
364   std::string data;
365   if (json)
366     base::JSONWriter::Write(*json, &data);
367   std::string mime_type =
368       AppendParameter(brillo::mime::application::kJson,
369                       brillo::mime::parameters::kCharset, "utf-8");
370   return SendRequest(request_type::kPatch, url, data.c_str(), data.size(),
371                      mime_type, headers, transport, success_callback,
372                      error_callback);
373 }
374 
ParseJsonResponse(Response * response,int * status_code,brillo::ErrorPtr * error)375 std::unique_ptr<base::DictionaryValue> ParseJsonResponse(
376     Response* response,
377     int* status_code,
378     brillo::ErrorPtr* error) {
379   std::unique_ptr<base::DictionaryValue> result;
380   if (!response)
381     return result;
382 
383   if (status_code)
384     *status_code = response->GetStatusCode();
385 
386   // Make sure we have a correct content type. Do not try to parse
387   // binary files, or HTML output. Limit to application/json and text/plain.
388   auto content_type = RemoveParameters(response->GetContentType());
389   if (content_type != brillo::mime::application::kJson &&
390       content_type != brillo::mime::text::kPlain) {
391     brillo::Error::AddTo(error, FROM_HERE, brillo::errors::json::kDomain,
392                          "non_json_content_type",
393                          "Unexpected response content type: " + content_type);
394     return result;
395   }
396 
397   std::string json = response->ExtractDataAsString();
398   std::string error_message;
399   auto value = base::JSONReader::ReadAndReturnError(json, base::JSON_PARSE_RFC,
400                                                     nullptr, &error_message);
401   if (!value) {
402     brillo::Error::AddToPrintf(error, FROM_HERE, brillo::errors::json::kDomain,
403                                brillo::errors::json::kParseError,
404                                "Error '%s' occurred parsing JSON string '%s'",
405                                error_message.c_str(), json.c_str());
406     return result;
407   }
408   result = base::DictionaryValue::From(std::move(value));
409   if (!result) {
410     brillo::Error::AddToPrintf(error, FROM_HERE, brillo::errors::json::kDomain,
411                                brillo::errors::json::kObjectExpected,
412                                "Response is not a valid JSON object: '%s'",
413                                json.c_str());
414   }
415   return result;
416 }
417 
GetCanonicalHeaderName(const std::string & name)418 std::string GetCanonicalHeaderName(const std::string& name) {
419   std::string canonical_name = name;
420   bool word_begin = true;
421   for (char& c : canonical_name) {
422     if (c == '-') {
423       word_begin = true;
424     } else {
425       if (word_begin) {
426         c = toupper(c);
427       } else {
428         c = tolower(c);
429       }
430       word_begin = false;
431     }
432   }
433   return canonical_name;
434 }
435 
436 }  // namespace http
437 }  // namespace brillo
438