// Copyright 2014 The Chromium 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 "components/copresence/rpc/http_post.h"
// TODO(ckehoe): Support third-party protobufs too.
#include <google/protobuf/message_lite.h>
#include "base/bind.h"
#include "google_apis/google_api_keys.h"
#include "net/base/load_flags.h"
#include "net/base/net_errors.h"
#include "net/base/url_util.h"
#include "net/http/http_status_code.h"
#include "net/url_request/url_fetcher.h"
#include "net/url_request/url_request_context_getter.h"
#include "url/gurl.h"
namespace copresence {
const char HttpPost::kApiKeyField[] = "key";
const char HttpPost::kTracingField[] = "trace";
HttpPost::HttpPost(net::URLRequestContextGetter* url_context_getter,
const std::string& server_host,
const std::string& rpc_name,
const std::string& tracing_token,
std::string api_key,
const google::protobuf::MessageLite& request_proto) {
// Create the base URL to call.
GURL url(server_host + "/" + rpc_name);
// Add the tracing token, if specified.
if (!tracing_token.empty()) {
url = net::AppendQueryParameter(
url, kTracingField, "token:" + tracing_token);
}
// If no API key is specified, use the Chrome API key.
if (api_key.empty()) {
#ifdef GOOGLE_CHROME_BUILD
DCHECK(google_apis::HasKeysConfigured());
api_key = google_apis::GetAPIKey();
#else
LOG(ERROR) << "No Copresence API key provided";
#endif
}
url = net::AppendQueryParameter(url, kApiKeyField, api_key);
// Serialize the proto for transmission.
std::string request_data;
bool serialize_success = request_proto.SerializeToString(&request_data);
DCHECK(serialize_success);
// Configure and send the request.
url_fetcher_.reset(net::URLFetcher::Create(
kUrlFetcherId, url, net::URLFetcher::POST, this));
url_fetcher_->SetRequestContext(url_context_getter);
url_fetcher_->SetLoadFlags(net::LOAD_BYPASS_CACHE |
net::LOAD_DISABLE_CACHE |
net::LOAD_DO_NOT_SAVE_COOKIES |
net::LOAD_DO_NOT_SEND_COOKIES |
net::LOAD_DO_NOT_SEND_AUTH_DATA);
url_fetcher_->SetUploadData("application/x-protobuf", request_data);
}
HttpPost::~HttpPost() {}
void HttpPost::Start(const ResponseCallback& response_callback) {
response_callback_ = response_callback;
DVLOG(3) << "Sending Copresence request to "
<< url_fetcher_->GetOriginalURL().spec();
url_fetcher_->Start();
}
void HttpPost::OnURLFetchComplete(const net::URLFetcher* source) {
DCHECK_EQ(url_fetcher_.get(), source);
// Gather response info.
std::string response;
source->GetResponseAsString(&response);
int response_code = source->GetResponseCode();
// Log any errors.
if (response_code < 0) {
net::URLRequestStatus status = source->GetStatus();
LOG(WARNING) << "Couldn't contact the Copresence server at "
<< source->GetURL() << ". Status code " << status.status();
LOG_IF(WARNING, status.error())
<< "Network error: " << net::ErrorToString(status.error());
LOG_IF(WARNING, !response.empty()) << "HTTP response: " << response;
} else if (response_code != net::HTTP_OK) {
LOG(WARNING) << "Copresence request got HTTP response code "
<< response_code << ". Response:\n" << response;
}
// Return the response.
response_callback_.Run(response_code, response);
}
} // namespace copresence