// Copyright 2013 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 "remoting/host/setup/service_client.h" #include "base/json/json_reader.h" #include "base/json/json_writer.h" #include "base/memory/scoped_ptr.h" #include "base/values.h" #include "net/http/http_status_code.h" #include "net/url_request/url_fetcher.h" #include "net/url_request/url_fetcher_delegate.h" #include "net/url_request/url_request_context_getter.h" #include "url/gurl.h" namespace remoting { class ServiceClient::Core : public base::RefCountedThreadSafe, public net::URLFetcherDelegate { public: Core(const std::string& chromoting_hosts_url, net::URLRequestContextGetter* request_context_getter) : request_context_getter_(request_context_getter), delegate_(NULL), pending_request_type_(PENDING_REQUEST_NONE), chromoting_hosts_url_(chromoting_hosts_url) { } void RegisterHost(const std::string& host_id, const std::string& host_name, const std::string& public_key, const std::string& host_client_id, const std::string& oauth_access_token, ServiceClient::Delegate* delegate); void UnregisterHost(const std::string& host_id, const std::string& oauth_access_token, ServiceClient::Delegate* delegate); // net::URLFetcherDelegate implementation. void OnURLFetchComplete(const net::URLFetcher* source) override; private: friend class base::RefCountedThreadSafe; ~Core() override {} enum PendingRequestType { PENDING_REQUEST_NONE, PENDING_REQUEST_REGISTER_HOST, PENDING_REQUEST_UNREGISTER_HOST }; void MakeChromotingRequest(net::URLFetcher::RequestType request_type, const std::string& post_body, const std::string& url_suffix, const std::string& oauth_access_token, ServiceClient::Delegate* delegate); void HandleResponse(const net::URLFetcher* source); scoped_refptr request_context_getter_; ServiceClient::Delegate* delegate_; scoped_ptr request_; PendingRequestType pending_request_type_; std::string chromoting_hosts_url_; }; void ServiceClient::Core::RegisterHost( const std::string& host_id, const std::string& host_name, const std::string& public_key, const std::string& host_client_id, const std::string& oauth_access_token, Delegate* delegate) { DCHECK(pending_request_type_ == PENDING_REQUEST_NONE); pending_request_type_ = PENDING_REQUEST_REGISTER_HOST; base::DictionaryValue post_body; post_body.SetString("data.hostId", host_id); post_body.SetString("data.hostName", host_name); post_body.SetString("data.publicKey", public_key); std::string url_suffix; if (!host_client_id.empty()) url_suffix = "?hostClientId=" + host_client_id; std::string post_body_str; base::JSONWriter::Write(&post_body, &post_body_str); MakeChromotingRequest(net::URLFetcher::POST, url_suffix, post_body_str, oauth_access_token, delegate); } void ServiceClient::Core::UnregisterHost( const std::string& host_id, const std::string& oauth_access_token, Delegate* delegate) { DCHECK(pending_request_type_ == PENDING_REQUEST_NONE); pending_request_type_ = PENDING_REQUEST_UNREGISTER_HOST; MakeChromotingRequest(net::URLFetcher::DELETE_REQUEST, host_id, std::string(), oauth_access_token, delegate); } void ServiceClient::Core::MakeChromotingRequest( net::URLFetcher::RequestType request_type, const std::string& url_suffix, const std::string& request_body, const std::string& oauth_access_token, ServiceClient::Delegate* delegate) { delegate_ = delegate; request_.reset(net::URLFetcher::Create( 0, GURL(chromoting_hosts_url_ + url_suffix), request_type, this)); request_->SetRequestContext(request_context_getter_.get()); request_->SetUploadData("application/json; charset=UTF-8", request_body); request_->AddExtraRequestHeader("Authorization: OAuth " + oauth_access_token); request_->Start(); } // URLFetcher::Delegate implementation. void ServiceClient::Core::OnURLFetchComplete( const net::URLFetcher* source) { HandleResponse(source); request_.reset(); } void ServiceClient::Core::HandleResponse(const net::URLFetcher* source) { DCHECK(pending_request_type_ != PENDING_REQUEST_NONE); PendingRequestType old_type = pending_request_type_; pending_request_type_ = PENDING_REQUEST_NONE; if (source->GetResponseCode() == net::HTTP_BAD_REQUEST) { delegate_->OnOAuthError(); return; } // Treat codes 2xx as successful; for example, HTTP_NO_CONTENT (204) can be // returned from a DELETE_REQUEST. if (source->GetResponseCode() / 100 == 2) { switch (old_type) { case PENDING_REQUEST_NONE: break; case PENDING_REQUEST_REGISTER_HOST: { std::string data; source->GetResponseAsString(&data); scoped_ptr message_value(base::JSONReader::Read(data)); base::DictionaryValue *dict; std::string code; if (message_value.get() && message_value->IsType(base::Value::TYPE_DICTIONARY) && message_value->GetAsDictionary(&dict) && dict->GetString("data.authorizationCode", &code)) { delegate_->OnHostRegistered(code); } else { delegate_->OnHostRegistered(std::string()); } } break; case PENDING_REQUEST_UNREGISTER_HOST: delegate_->OnHostUnregistered(); break; } return; } delegate_->OnNetworkError(source->GetResponseCode()); } ServiceClient::ServiceClient(const std::string& chromoting_hosts_url, net::URLRequestContextGetter* context_getter) { core_ = new Core(chromoting_hosts_url, context_getter); } ServiceClient::~ServiceClient() { } void ServiceClient::RegisterHost( const std::string& host_id, const std::string& host_name, const std::string& public_key, const std::string& host_client_id, const std::string& oauth_access_token, Delegate* delegate) { return core_->RegisterHost(host_id, host_name, public_key, host_client_id, oauth_access_token, delegate); } void ServiceClient::UnregisterHost( const std::string& host_id, const std::string& oauth_access_token, Delegate* delegate) { return core_->UnregisterHost(host_id, oauth_access_token, delegate); } } // namespace gaia