Index: chrome/browser/ui/desktop_ios_promotion/sms_service.cc |
diff --git a/chrome/browser/ui/desktop_ios_promotion/sms_service.cc b/chrome/browser/ui/desktop_ios_promotion/sms_service.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..8f109d8184705aac860699dd99eccde595361efe |
--- /dev/null |
+++ b/chrome/browser/ui/desktop_ios_promotion/sms_service.cc |
@@ -0,0 +1,321 @@ |
+// Copyright (c) 2017 The Chromium Authors. All rights reserved. |
msramek
2017/01/23 21:13:56
nit: We don't use (c) anymore. s/\(c\)//g in all f
justincohen
2017/01/25 22:52:38
Done.
|
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#include "chrome/browser/ui/desktop_ios_promotion/sms_service.h" |
+ |
+#include "base/bind.h" |
+#include "base/json/json_reader.h" |
+#include "base/memory/ptr_util.h" |
+#include "base/metrics/histogram_macros.h" |
+#include "base/optional.h" |
+#include "base/strings/stringprintf.h" |
+#include "components/signin/core/browser/signin_manager.h" |
+#include "google_apis/gaia/gaia_urls.h" |
+#include "google_apis/gaia/oauth2_token_service.h" |
+#include "net/base/load_flags.h" |
+#include "net/base/url_util.h" |
+#include "net/http/http_status_code.h" |
+#include "net/http/http_util.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" |
+ |
+namespace { |
+ |
+const char kDesktopIOSPromotionOAuthScope[] = |
+ "https://www.googleapis.com/auth/mobile_user_preferences"; |
+ |
+const char kDesktopIOSPromotionQueryPhoneNumber[] = |
+ "https://growth-pa.googleapis.com/v1/get_verified_phone_numbers"; |
+ |
+const char kDesktopIOSPromotionSendSMS[] = |
+ "https://growth-pa.googleapis.com/v1/send_sms"; |
+ |
+const char kPostDataMimeType[] = "application/json"; |
+ |
+const char kSendSMSPromoFormat[] = "{promo_id:%s}"; |
+ |
+// The maximum number of retries for the URLFetcher requests. |
+const size_t kMaxRetries = 1; |
+ |
+class RequestImpl : public DesktopIOSPromotion::SMSService::Request, |
msramek
2017/01/23 21:13:57
WebHistoryService:Request is public. Couldn't we j
justincohen
2017/01/25 22:52:38
Wouldn't we need to reimplement that virtuals?
Ad
msramek
2017/01/27 16:38:57
Ah, right. I think it would make sense to have a s
|
+ private OAuth2TokenService::Consumer, |
+ private net::URLFetcherDelegate { |
+ public: |
+ ~RequestImpl() override {} |
+ |
+ // Returns the response code received from the server, which will only be |
+ // valid if the request succeeded. |
+ int GetResponseCode() override { return response_code_; } |
+ |
+ // Returns the contents of the response body received from the server. |
+ const std::string& GetResponseBody() override { return response_body_; } |
+ |
+ bool IsPending() override { return is_pending_; } |
+ |
+ private: |
+ friend class DesktopIOSPromotion::SMSService; |
+ |
+ RequestImpl( |
+ OAuth2TokenService* token_service, |
+ SigninManagerBase* signin_manager, |
+ const scoped_refptr<net::URLRequestContextGetter>& request_context, |
+ const GURL& url, |
+ const DesktopIOSPromotion::SMSService::CompletionCallback& callback) |
+ : OAuth2TokenService::Consumer("desktop_ios_promotion"), |
+ token_service_(token_service), |
+ signin_manager_(signin_manager), |
+ request_context_(request_context), |
+ url_(url), |
+ post_data_mime_type_(kPostDataMimeType), |
+ response_code_(0), |
+ auth_retry_count_(0), |
+ callback_(callback), |
+ is_pending_(false) { |
+ DCHECK(token_service_); |
+ DCHECK(signin_manager_); |
+ DCHECK(request_context_); |
+ } |
+ |
+ void Start() override { |
+ OAuth2TokenService::ScopeSet oauth_scopes; |
+ oauth_scopes.insert(kDesktopIOSPromotionOAuthScope); |
+ token_request_ = token_service_->StartRequest( |
+ signin_manager_->GetAuthenticatedAccountId(), oauth_scopes, this); |
+ is_pending_ = true; |
+ } |
+ |
+ // content::URLFetcherDelegate interface. |
+ void OnURLFetchComplete(const net::URLFetcher* source) override { |
+ DCHECK_EQ(source, url_fetcher_.get()); |
+ response_code_ = url_fetcher_->GetResponseCode(); |
+ |
+ UMA_HISTOGRAM_CUSTOM_ENUMERATION( |
+ "DesktopIOSPromotion.OAuthTokenResponseCode", |
+ net::HttpUtil::MapStatusCodeForHistogram(response_code_), |
+ net::HttpUtil::GetStatusCodesForHistogram()); |
+ |
+ // If the response code indicates that the token might not be valid, |
+ // invalidate the token and try again. |
+ if (response_code_ == net::HTTP_UNAUTHORIZED && ++auth_retry_count_ <= 1) { |
+ OAuth2TokenService::ScopeSet oauth_scopes; |
+ oauth_scopes.insert(kDesktopIOSPromotionOAuthScope); |
+ token_service_->InvalidateAccessToken( |
+ signin_manager_->GetAuthenticatedAccountId(), oauth_scopes, |
+ access_token_); |
+ |
+ access_token_.clear(); |
+ Start(); |
+ return; |
+ } |
+ url_fetcher_->GetResponseAsString(&response_body_); |
+ url_fetcher_.reset(); |
+ is_pending_ = false; |
+ callback_.Run(this, response_code_ == net::HTTP_OK); |
+ // It is valid for the callback to delete |this|, so do not access any |
+ // members below here. |
+ } |
+ |
+ // OAuth2TokenService::Consumer interface. |
+ void OnGetTokenSuccess(const OAuth2TokenService::Request* request, |
+ const std::string& access_token, |
+ const base::Time& expiration_time) override { |
+ token_request_.reset(); |
+ DCHECK(!access_token.empty()); |
+ access_token_ = access_token; |
+ |
+ UMA_HISTOGRAM_BOOLEAN("DesktopIOSPromotion.OAuthTokenCompletion", true); |
+ |
+ // Got an access token -- start the actual API request. |
+ url_fetcher_ = CreateUrlFetcher(access_token); |
+ url_fetcher_->Start(); |
+ } |
+ |
+ void OnGetTokenFailure(const OAuth2TokenService::Request* request, |
+ const GoogleServiceAuthError& error) override { |
+ token_request_.reset(); |
+ is_pending_ = false; |
+ |
+ UMA_HISTOGRAM_BOOLEAN("DesktopIOSPromotion.OAuthTokenCompletion", false); |
+ |
+ callback_.Run(this, false); |
+ // It is valid for the callback to delete |this|, so do not access any |
+ // members below here. |
+ } |
+ |
+ // Helper for creating a new URLFetcher for the API request. |
+ std::unique_ptr<net::URLFetcher> CreateUrlFetcher( |
+ const std::string& access_token) { |
+ net::URLFetcher::RequestType request_type = |
+ post_data_ ? net::URLFetcher::POST : net::URLFetcher::GET; |
+ std::unique_ptr<net::URLFetcher> fetcher = |
+ net::URLFetcher::Create(url_, request_type, this); |
+ fetcher->SetRequestContext(request_context_.get()); |
+ fetcher->SetMaxRetriesOn5xx(kMaxRetries); |
+ fetcher->SetLoadFlags(net::LOAD_DO_NOT_SEND_COOKIES | |
+ net::LOAD_DO_NOT_SAVE_COOKIES); |
+ fetcher->AddExtraRequestHeader("Authorization: Bearer " + access_token); |
+ fetcher->AddExtraRequestHeader( |
+ "X-Developer-Key: " + |
+ GaiaUrls::GetInstance()->oauth2_chrome_client_id()); |
+ |
+ if (post_data_) |
+ fetcher->SetUploadData(post_data_mime_type_, post_data_.value()); |
+ return fetcher; |
+ } |
+ |
+ void SetPostData(const std::string& post_data) override { |
+ SetPostDataAndType(post_data, kPostDataMimeType); |
+ } |
+ |
+ void SetPostDataAndType(const std::string& post_data, |
+ const std::string& mime_type) override { |
+ post_data_ = post_data; |
+ post_data_mime_type_ = mime_type; |
+ } |
+ |
+ OAuth2TokenService* token_service_; |
+ SigninManagerBase* signin_manager_; |
+ scoped_refptr<net::URLRequestContextGetter> request_context_; |
+ |
+ // The URL of the API endpoint. |
+ GURL url_; |
+ |
+ // POST data to be sent with the request (may be empty). |
+ base::Optional<std::string> post_data_; |
+ |
+ // MIME type of the post requests. Defaults to text/plain. |
+ std::string post_data_mime_type_; |
+ |
+ // The OAuth2 access token request. |
+ std::unique_ptr<OAuth2TokenService::Request> token_request_; |
+ |
+ // The current OAuth2 access token. |
+ std::string access_token_; |
+ |
+ // Handles the actual API requests after the OAuth token is acquired. |
+ std::unique_ptr<net::URLFetcher> url_fetcher_; |
+ |
+ // Holds the response code received from the server. |
+ int response_code_; |
+ |
+ // Holds the response body received from the server. |
+ std::string response_body_; |
+ |
+ // The number of times this request has already been retried due to |
+ // authorization problems. |
+ int auth_retry_count_; |
+ |
+ // The callback to execute when the query is complete. |
+ DesktopIOSPromotion::SMSService::CompletionCallback callback_; |
+ |
+ // True if the request was started and has not yet completed, otherwise false. |
+ bool is_pending_; |
+}; |
+ |
+} // namespace |
+ |
+namespace DesktopIOSPromotion { |
+ |
+SMSService::Request::Request() {} |
+ |
+SMSService::Request::~Request() {} |
+ |
+SMSService::SMSService( |
+ OAuth2TokenService* token_service, |
+ SigninManagerBase* signin_manager, |
+ const scoped_refptr<net::URLRequestContextGetter>& request_context) |
+ : token_service_(token_service), |
+ signin_manager_(signin_manager), |
+ request_context_(request_context), |
+ weak_ptr_factory_(this) {} |
+ |
+SMSService::~SMSService() {} |
+ |
+void SMSService::AddObserver(SMSServiceObserver* observer) { |
+ observer_list_.AddObserver(observer); |
+} |
+ |
+void SMSService::RemoveObserver(SMSServiceObserver* observer) { |
+ observer_list_.RemoveObserver(observer); |
+} |
+ |
+SMSService::Request* SMSService::CreateRequest( |
+ const GURL& url, |
+ const CompletionCallback& callback) { |
+ return new RequestImpl(token_service_, signin_manager_, request_context_, url, |
+ callback); |
+} |
+ |
+void SMSService::QueryPhoneNumber(const PhoneNumberCallback& callback) { |
+ CompletionCallback completion_callback = |
+ base::Bind(&SMSService::QueryPhoneNumberCompletionCallback, |
+ weak_ptr_factory_.GetWeakPtr(), callback); |
+ |
+ GURL url(kDesktopIOSPromotionQueryPhoneNumber); |
+ Request* request = CreateRequest(url, completion_callback); |
+ request->SetPostData("{}"); |
msramek
2017/01/23 21:13:56
Is this a placeholder? In that case please add a T
justincohen
2017/01/25 22:52:38
added a comment.
|
+ pending_requests_[request] = base::WrapUnique(request); |
+ request->Start(); |
+} |
+ |
+void SMSService::SendSMS(std::string promo_id, |
+ const SMSService::PhoneNumberCallback& callback) { |
+ CompletionCallback completion_callback = base::Bind( |
+ &SMSService::SendSMSCallback, weak_ptr_factory_.GetWeakPtr(), callback); |
+ GURL url(kDesktopIOSPromotionSendSMS); |
+ Request* request = CreateRequest(url, completion_callback); |
+ request->SetPostData( |
+ base::StringPrintf(kSendSMSPromoFormat, promo_id.c_str())); |
+ pending_requests_[request] = base::WrapUnique(request); |
+ request->Start(); |
+} |
+ |
+void SMSService::QueryPhoneNumberCompletionCallback( |
+ const SMSService::PhoneNumberCallback& callback, |
+ SMSService::Request* request, |
+ bool success) { |
+ std::unique_ptr<Request> request_ptr = std::move(pending_requests_[request]); |
+ pending_requests_.erase(request); |
+ |
+ std::string phone_number; |
+ std::unique_ptr<base::Value> value = |
+ base::JSONReader::Read(request->GetResponseBody()); |
+ if (value.get() && value.get()->IsType(base::Value::Type::DICTIONARY)) { |
msramek
2017/01/23 21:13:56
This looks complex enough to deserve a test.
justincohen
2017/01/27 05:12:00
Done.
|
+ const base::DictionaryValue* dictionary; |
+ if (value->GetAsDictionary(&dictionary)) { |
+ const base::ListValue* number_list; |
+ if (dictionary->GetList("phoneNumber", &number_list)) { |
+ const base::DictionaryValue* sub_dictionary; |
+ // For now only handle the first number. |
+ if (number_list->GetSize() > 0 && |
+ number_list->GetDictionary(0, &sub_dictionary)) { |
+ sub_dictionary->GetString("phoneNumber", &phone_number); |
+ } |
+ } |
+ } |
+ } |
+ callback.Run(request, success, phone_number); |
+} |
+ |
+void SMSService::SendSMSCallback( |
+ const SMSService::PhoneNumberCallback& callback, |
+ SMSService::Request* request, |
+ bool success) { |
+ std::unique_ptr<Request> request_ptr = std::move(pending_requests_[request]); |
+ pending_requests_.erase(request); |
+ |
+ std::string phone_number; |
+ std::unique_ptr<base::Value> value = |
+ base::JSONReader::Read(request->GetResponseBody()); |
+ if (value.get() && value.get()->IsType(base::Value::Type::DICTIONARY)) { |
+ const base::DictionaryValue* dictionary; |
+ if (value->GetAsDictionary(&dictionary)) { |
+ dictionary->GetString("phoneNumber", &phone_number); |
+ } |
+ } |
+ callback.Run(request, success, phone_number); |
+} |
+ |
+} // namespace DesktopIOSPromotion |