| // Copyright 2014 The Chromium OS Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <chromeos/http/http_utils.h> |
| |
| #include <algorithm> |
| |
| #include <base/bind.h> |
| #include <base/json/json_reader.h> |
| #include <base/json/json_writer.h> |
| #include <base/values.h> |
| #include <chromeos/data_encoding.h> |
| #include <chromeos/errors/error_codes.h> |
| #include <chromeos/mime_utils.h> |
| #include <chromeos/streams/memory_stream.h> |
| |
| using chromeos::mime::AppendParameter; |
| using chromeos::mime::RemoveParameters; |
| |
| namespace chromeos { |
| namespace http { |
| |
| std::unique_ptr<Response> GetAndBlock(const std::string& url, |
| const HeaderList& headers, |
| std::shared_ptr<Transport> transport, |
| chromeos::ErrorPtr* error) { |
| return SendRequestWithNoDataAndBlock( |
| request_type::kGet, url, headers, transport, error); |
| } |
| |
| RequestID Get(const std::string& url, |
| const HeaderList& headers, |
| std::shared_ptr<Transport> transport, |
| const SuccessCallback& success_callback, |
| const ErrorCallback& error_callback) { |
| return SendRequestWithNoData(request_type::kGet, |
| url, |
| headers, |
| transport, |
| success_callback, |
| error_callback); |
| } |
| |
| std::unique_ptr<Response> HeadAndBlock(const std::string& url, |
| std::shared_ptr<Transport> transport, |
| chromeos::ErrorPtr* error) { |
| return SendRequestWithNoDataAndBlock( |
| request_type::kHead, url, {}, transport, error); |
| } |
| |
| RequestID Head(const std::string& url, |
| std::shared_ptr<Transport> transport, |
| const SuccessCallback& success_callback, |
| const ErrorCallback& error_callback) { |
| return SendRequestWithNoData(request_type::kHead, |
| url, |
| {}, |
| transport, |
| success_callback, |
| error_callback); |
| } |
| |
| std::unique_ptr<Response> PostTextAndBlock(const std::string& url, |
| const std::string& data, |
| const std::string& mime_type, |
| const HeaderList& headers, |
| std::shared_ptr<Transport> transport, |
| chromeos::ErrorPtr* error) { |
| return PostBinaryAndBlock( |
| url, data.data(), data.size(), mime_type, headers, transport, error); |
| } |
| |
| RequestID 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) { |
| return PostBinary(url, |
| data.data(), |
| data.size(), |
| mime_type, |
| headers, |
| transport, |
| success_callback, |
| error_callback); |
| } |
| |
| std::unique_ptr<Response> 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, |
| chromeos::ErrorPtr* error) { |
| Request request(url, method, transport); |
| request.AddHeaders(headers); |
| if (data_size > 0) { |
| CHECK(!mime_type.empty()) << "MIME type must be specified if request body " |
| "message is provided"; |
| request.SetContentType(mime_type); |
| if (!request.AddRequestBody(data, data_size, error)) |
| return std::unique_ptr<Response>(); |
| } |
| return request.GetResponseAndBlock(error); |
| } |
| |
| std::unique_ptr<Response> SendRequestWithNoDataAndBlock( |
| const std::string& method, |
| const std::string& url, |
| const HeaderList& headers, |
| std::shared_ptr<Transport> transport, |
| chromeos::ErrorPtr* error) { |
| return SendRequestAndBlock( |
| method, url, nullptr, 0, {}, headers, transport, error); |
| } |
| |
| RequestID 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) { |
| Request request(url, method, transport); |
| request.AddHeaders(headers); |
| if (stream && (!stream->CanGetSize() || stream->GetRemainingSize() > 0)) { |
| CHECK(!mime_type.empty()) << "MIME type must be specified if request body " |
| "message is provided"; |
| request.SetContentType(mime_type); |
| chromeos::ErrorPtr error; |
| if (!request.AddRequestBody(std::move(stream), &error)) { |
| transport->RunCallbackAsync( |
| FROM_HERE, base::Bind(error_callback, |
| 0, base::Owned(error.release()))); |
| return 0; |
| } |
| } |
| return request.GetResponse(success_callback, error_callback); |
| } |
| |
| RequestID 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) { |
| return SendRequest(method, |
| url, |
| MemoryStream::OpenCopyOf(data, data_size, nullptr), |
| mime_type, |
| headers, |
| transport, |
| success_callback, |
| error_callback); |
| } |
| |
| RequestID 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) { |
| return SendRequest(method, |
| url, |
| {}, |
| {}, |
| headers, |
| transport, |
| success_callback, |
| error_callback); |
| } |
| |
| std::unique_ptr<Response> 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, |
| chromeos::ErrorPtr* error) { |
| return SendRequestAndBlock(request_type::kPost, |
| url, |
| data, |
| data_size, |
| mime_type, |
| headers, |
| transport, |
| error); |
| } |
| |
| RequestID 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) { |
| return SendRequest(request_type::kPost, |
| url, |
| std::move(stream), |
| mime_type, |
| headers, |
| transport, |
| success_callback, |
| error_callback); |
| } |
| |
| RequestID 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) { |
| return SendRequest(request_type::kPost, |
| url, |
| data, |
| data_size, |
| mime_type, |
| headers, |
| transport, |
| success_callback, |
| error_callback); |
| } |
| |
| std::unique_ptr<Response> PostFormDataAndBlock( |
| const std::string& url, |
| const FormFieldList& data, |
| const HeaderList& headers, |
| std::shared_ptr<Transport> transport, |
| chromeos::ErrorPtr* error) { |
| std::string encoded_data = chromeos::data_encoding::WebParamsEncode(data); |
| return PostBinaryAndBlock(url, |
| encoded_data.c_str(), |
| encoded_data.size(), |
| chromeos::mime::application::kWwwFormUrlEncoded, |
| headers, |
| transport, |
| error); |
| } |
| |
| std::unique_ptr<Response> PostFormDataAndBlock( |
| const std::string& url, |
| std::unique_ptr<FormData> form_data, |
| const HeaderList& headers, |
| std::shared_ptr<Transport> transport, |
| chromeos::ErrorPtr* error) { |
| Request request(url, request_type::kPost, transport); |
| request.AddHeaders(headers); |
| if (!request.AddRequestBodyAsFormData(std::move(form_data), error)) |
| return std::unique_ptr<Response>(); |
| return request.GetResponseAndBlock(error); |
| } |
| |
| RequestID PostFormData(const std::string& url, |
| const FormFieldList& data, |
| const HeaderList& headers, |
| std::shared_ptr<Transport> transport, |
| const SuccessCallback& success_callback, |
| const ErrorCallback& error_callback) { |
| std::string encoded_data = chromeos::data_encoding::WebParamsEncode(data); |
| return PostBinary(url, |
| encoded_data.c_str(), |
| encoded_data.size(), |
| chromeos::mime::application::kWwwFormUrlEncoded, |
| headers, |
| transport, |
| success_callback, |
| error_callback); |
| } |
| |
| RequestID 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) { |
| Request request(url, request_type::kPost, transport); |
| request.AddHeaders(headers); |
| chromeos::ErrorPtr error; |
| if (!request.AddRequestBodyAsFormData(std::move(form_data), &error)) { |
| transport->RunCallbackAsync( |
| FROM_HERE, base::Bind(error_callback, 0, base::Owned(error.release()))); |
| return 0; |
| } |
| return request.GetResponse(success_callback, error_callback); |
| } |
| |
| std::unique_ptr<Response> PostJsonAndBlock(const std::string& url, |
| const base::Value* json, |
| const HeaderList& headers, |
| std::shared_ptr<Transport> transport, |
| chromeos::ErrorPtr* error) { |
| std::string data; |
| if (json) |
| base::JSONWriter::Write(*json, &data); |
| std::string mime_type = AppendParameter(chromeos::mime::application::kJson, |
| chromeos::mime::parameters::kCharset, |
| "utf-8"); |
| return PostBinaryAndBlock( |
| url, data.c_str(), data.size(), mime_type, headers, transport, error); |
| } |
| |
| RequestID 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) { |
| std::string data; |
| if (json) |
| base::JSONWriter::Write(*json, &data); |
| std::string mime_type = AppendParameter(chromeos::mime::application::kJson, |
| chromeos::mime::parameters::kCharset, |
| "utf-8"); |
| return PostBinary(url, |
| data.c_str(), |
| data.size(), |
| mime_type, |
| headers, |
| transport, |
| success_callback, |
| error_callback); |
| } |
| |
| std::unique_ptr<Response> PatchJsonAndBlock( |
| const std::string& url, |
| const base::Value* json, |
| const HeaderList& headers, |
| std::shared_ptr<Transport> transport, |
| chromeos::ErrorPtr* error) { |
| std::string data; |
| if (json) |
| base::JSONWriter::Write(*json, &data); |
| std::string mime_type = AppendParameter(chromeos::mime::application::kJson, |
| chromeos::mime::parameters::kCharset, |
| "utf-8"); |
| return SendRequestAndBlock(request_type::kPatch, |
| url, |
| data.c_str(), |
| data.size(), |
| mime_type, |
| headers, |
| transport, |
| error); |
| } |
| |
| RequestID 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) { |
| std::string data; |
| if (json) |
| base::JSONWriter::Write(*json, &data); |
| std::string mime_type = AppendParameter(chromeos::mime::application::kJson, |
| chromeos::mime::parameters::kCharset, |
| "utf-8"); |
| return SendRequest(request_type::kPatch, |
| url, |
| data.c_str(), |
| data.size(), |
| mime_type, |
| headers, |
| transport, |
| success_callback, |
| error_callback); |
| } |
| |
| std::unique_ptr<base::DictionaryValue> ParseJsonResponse( |
| Response* response, |
| int* status_code, |
| chromeos::ErrorPtr* error) { |
| if (!response) |
| return std::unique_ptr<base::DictionaryValue>(); |
| |
| if (status_code) |
| *status_code = response->GetStatusCode(); |
| |
| // Make sure we have a correct content type. Do not try to parse |
| // binary files, or HTML output. Limit to application/json and text/plain. |
| auto content_type = RemoveParameters(response->GetContentType()); |
| if (content_type != chromeos::mime::application::kJson && |
| content_type != chromeos::mime::text::kPlain) { |
| chromeos::Error::AddTo(error, |
| FROM_HERE, |
| chromeos::errors::json::kDomain, |
| "non_json_content_type", |
| "Unexpected response content type: " + content_type); |
| return std::unique_ptr<base::DictionaryValue>(); |
| } |
| |
| std::string json = response->ExtractDataAsString(); |
| std::string error_message; |
| auto value = base::JSONReader::ReadAndReturnError(json, base::JSON_PARSE_RFC, |
| nullptr, &error_message); |
| if (!value) { |
| chromeos::Error::AddToPrintf(error, |
| FROM_HERE, |
| chromeos::errors::json::kDomain, |
| chromeos::errors::json::kParseError, |
| "Error '%s' occurred parsing JSON string '%s'", |
| error_message.c_str(), json.c_str()); |
| return std::unique_ptr<base::DictionaryValue>(); |
| } |
| base::DictionaryValue* dict_value = nullptr; |
| if (!value->GetAsDictionary(&dict_value)) { |
| chromeos::Error::AddToPrintf(error, |
| FROM_HERE, |
| chromeos::errors::json::kDomain, |
| chromeos::errors::json::kObjectExpected, |
| "Response is not a valid JSON object: '%s'", |
| json.c_str()); |
| return std::unique_ptr<base::DictionaryValue>(); |
| } else { |
| // |value| is now owned by |dict_value|, so release the scoped_ptr now. |
| base::IgnoreResult(value.release()); |
| } |
| return std::unique_ptr<base::DictionaryValue>(dict_value); |
| } |
| |
| std::string GetCanonicalHeaderName(const std::string& name) { |
| std::string canonical_name = name; |
| bool word_begin = true; |
| for (char& c : canonical_name) { |
| if (c == '-') { |
| word_begin = true; |
| } else { |
| if (word_begin) { |
| c = toupper(c); |
| } else { |
| c = tolower(c); |
| } |
| word_begin = false; |
| } |
| } |
| return canonical_name; |
| } |
| |
| } // namespace http |
| } // namespace chromeos |