OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 2016 The Chromium 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 #include "components/ntp_snippets/ntp_snippets_fetcher.h" | |
5 | |
6 #include "base/files/file_path.h" | |
7 #include "base/files/file_util.h" | |
8 #include "base/path_service.h" | |
9 #include "base/strings/stringprintf.h" | |
10 #include "base/task_runner_util.h" | |
11 #include "components/signin/core/browser/profile_oauth2_token_service.h" | |
12 #include "components/signin/core/browser/signin_manager.h" | |
13 #include "components/signin/core/browser/signin_tracker.h" | |
14 #include "net/base/load_flags.h" | |
15 #include "net/http/http_request_headers.h" | |
16 #include "net/http/http_response_headers.h" | |
17 #include "net/http/http_status_code.h" | |
18 #include "net/url_request/url_fetcher.h" | |
19 | |
20 using net::URLFetcher; | |
21 using net::URLRequestContextGetter; | |
22 using net::HttpRequestHeaders; | |
23 using net::URLRequestStatus; | |
24 | |
25 namespace ntp_snippets { | |
26 | |
27 const char kSnippetSuggestionsFilename[] = "ntp_snippets.json"; | |
28 const char kApiScope[] = "https://www.googleapis.com/auth/webhistory"; | |
29 const char kContentSnippetsServer[] = | |
30 "https://chromereader-pa.googleapis.com/v1/fetch"; | |
31 const char kAuthorizationRequestHeaderFormat[] = "Bearer %s"; | |
32 | |
33 const char kUnpersonalizedRequestParameters[] = | |
34 "{ \"response_detail_level\": \"FULL_DEBUG\", \"advanced_options\": { " | |
35 "\"local_scoring_params\": {\"content_params\" : { " | |
36 "\"only_return_personalized_results\": false } }, " | |
37 "\"global_scoring_params\": { \"num_to_return\": 10 } } }"; | |
38 | |
39 base::FilePath GetSnippetsSuggestionsPath() { | |
40 base::FilePath dir; | |
41 #if defined(OS_ANDROID) | |
42 CHECK(PathService::Get(base::DIR_ANDROID_APP_DATA, &dir)); | |
Bernhard Bauer
2016/02/08 18:19:26
DCHECK should be fine for this.
May
2016/02/09 17:38:53
Done.
| |
43 #else | |
44 NOTIMPLEMENTED(); | |
Bernhard Bauer
2016/02/08 18:19:26
Maybe also return an empty FilePath in this case?
May
2016/02/09 17:38:53
Done.
| |
45 #endif | |
46 return dir.AppendASCII(kSnippetSuggestionsFilename); | |
47 } | |
48 | |
49 NTPSnippetsFetcher::NTPSnippetsFetcher( | |
50 scoped_refptr<base::SequencedTaskRunner> file_task_runner, | |
51 SigninManager* signin_manager, | |
52 OAuth2TokenService* token_service, | |
53 URLRequestContextGetter* url_request_context_getter) | |
54 : OAuth2TokenService::Consumer("NTP_snippets"), | |
55 file_task_runner_(file_task_runner), | |
56 url_request_context_getter_(url_request_context_getter), | |
57 signin_manager_(signin_manager), | |
58 token_service_(token_service), | |
59 weak_ptr_factory_(this) {} | |
60 | |
61 NTPSnippetsFetcher::~NTPSnippetsFetcher() {} | |
62 | |
63 void NTPSnippetsFetcher::AddObserver(Observer* observer) { | |
64 observers_.AddObserver(observer); | |
65 } | |
66 | |
67 void NTPSnippetsFetcher::RemoveObserver(Observer* observer) { | |
68 observers_.RemoveObserver(observer); | |
69 } | |
70 | |
71 void NTPSnippetsFetcher::FetchSnippets(bool overwrite) { | |
72 if (overwrite) { | |
73 StartFetch(); | |
74 } else { | |
75 base::PostTaskAndReplyWithResult( | |
76 file_task_runner_.get(), FROM_HERE, | |
77 base::Bind(&base::PathExists, GetSnippetsSuggestionsPath()), | |
78 base::Bind(&NTPSnippetsFetcher::OnFileExistsCheckDone, | |
79 weak_ptr_factory_.GetWeakPtr())); | |
80 } | |
81 } | |
82 | |
83 void NTPSnippetsFetcher::OnFileExistsCheckDone(bool exists) { | |
84 if (exists) { | |
85 NotifyObservers(); | |
86 } else { | |
87 StartFetch(); | |
88 } | |
89 } | |
90 | |
91 void NTPSnippetsFetcher::StartFetch() { | |
92 if (signin_manager_->IsAuthenticated()) { | |
93 StartTokenRequest(); | |
94 } else { | |
95 // Wait until we get a refresh token. | |
96 token_service_->AddObserver(this); | |
97 } | |
98 } | |
99 | |
100 void NTPSnippetsFetcher::StartTokenRequest() { | |
101 OAuth2TokenService::ScopeSet scopes; | |
102 scopes.insert(kApiScope); | |
103 oauth_request_ = token_service_->StartRequest( | |
104 signin_manager_->GetAuthenticatedAccountId(), scopes, this); | |
105 } | |
106 | |
107 void NTPSnippetsFetcher::NotifyObservers() { | |
108 FOR_EACH_OBSERVER(Observer, observers_, OnNTPSnippetsDownloaded()); | |
109 } | |
110 | |
111 //////////////////////////////////////////////////////////////////////////////// | |
112 // OAuth2TokenService::Consumer overrides | |
113 void NTPSnippetsFetcher::OnGetTokenSuccess( | |
114 const OAuth2TokenService::Request* request, | |
115 const std::string& access_token, | |
116 const base::Time& expiration_time) { | |
117 oauth_request_.reset(); | |
118 url_fetcher_ = | |
119 URLFetcher::Create(GURL(kContentSnippetsServer), URLFetcher::POST, this); | |
120 url_fetcher_->SetRequestContext(url_request_context_getter_); | |
121 url_fetcher_->SetLoadFlags(net::LOAD_DO_NOT_SEND_COOKIES | | |
122 net::LOAD_DO_NOT_SAVE_COOKIES); | |
123 HttpRequestHeaders headers; | |
124 headers.SetHeader("Authorization", | |
125 base::StringPrintf(kAuthorizationRequestHeaderFormat, | |
126 access_token.c_str())); | |
127 headers.SetHeader("Content-Type", "application/json; charset=UTF-8"); | |
128 headers.SetHeader("X-GFE-SSL", "yes"); | |
Bernhard Bauer
2016/02/08 18:19:26
o_O Why do we need this?
May
2016/02/09 17:38:53
I'm not sure why we need it. I'd copied the reques
| |
129 url_fetcher_->SetExtraRequestHeaders(headers.ToString()); | |
130 url_fetcher_->SetUploadData("application/json", | |
131 kUnpersonalizedRequestParameters); | |
132 url_fetcher_->SaveResponseToTemporaryFile(file_task_runner_.get()); | |
133 url_fetcher_->Start(); | |
134 } | |
135 | |
136 void NTPSnippetsFetcher::OnGetTokenFailure( | |
137 const OAuth2TokenService::Request* request, | |
138 const GoogleServiceAuthError& error) { | |
139 oauth_request_.reset(); | |
140 DLOG(ERROR) << "Unable to get token: " << error.ToString(); | |
141 } | |
142 | |
143 //////////////////////////////////////////////////////////////////////////////// | |
144 // OAuth2TokenService::Observer overrides | |
145 void NTPSnippetsFetcher::OnRefreshTokenAvailable( | |
146 const std::string& account_id) { | |
147 token_service_->RemoveObserver(this); | |
Marc Treib
2016/02/09 09:17:54
We should probably check if account_id matches the
May
2016/02/09 17:38:53
If we're going to keep using signed in accounts, y
Marc Treib
2016/02/10 10:44:43
Acknowledged.
| |
148 StartTokenRequest(); | |
149 } | |
150 | |
151 //////////////////////////////////////////////////////////////////////////////// | |
152 // URLFetcherDelegate overrides | |
153 void NTPSnippetsFetcher::OnURLFetchComplete(const URLFetcher* source) { | |
154 DCHECK_EQ(url_fetcher_.get(), source); | |
155 | |
156 const URLRequestStatus& status = source->GetStatus(); | |
157 if (!status.is_success()) { | |
158 DLOG(WARNING) << "URLRequestStatus error " << status.error() | |
159 << " while trying to download " << source->GetURL().spec(); | |
160 return; | |
161 } | |
162 | |
163 int response_code = source->GetResponseCode(); | |
164 if (response_code != net::HTTP_OK) { | |
Bernhard Bauer
2016/02/08 18:19:26
Technically, you should also handle the case where
Marc Treib
2016/02/09 09:17:54
Yup, I was thinking the same thing while going thr
May
2016/02/09 17:38:53
Yes we should. :) However, since we're moving to a
Marc Treib
2016/02/10 10:44:43
So for the record, I'm fine with landing as-is for
May
2016/02/10 18:16:46
Acknowledged.
| |
165 DLOG(WARNING) << "HTTP error " << response_code | |
166 << " while trying to download " << source->GetURL().spec(); | |
167 return; | |
168 } | |
169 | |
170 base::FilePath response_path; | |
171 source->GetResponseAsFilePath(false, &response_path); | |
172 | |
173 base::PostTaskAndReplyWithResult( | |
174 file_task_runner_.get(), FROM_HERE, | |
175 base::Bind(&base::Move, response_path, GetSnippetsSuggestionsPath()), | |
176 base::Bind(&NTPSnippetsFetcher::OnFileMoveDone, | |
177 weak_ptr_factory_.GetWeakPtr())); | |
178 } | |
179 | |
180 void NTPSnippetsFetcher::OnFileMoveDone(bool success) { | |
181 if (!success) { | |
182 DLOG(WARNING) << "Could not move file to " | |
183 << GetSnippetsSuggestionsPath().LossyDisplayName(); | |
184 return; | |
185 } | |
186 | |
187 NotifyObservers(); | |
188 } | |
189 | |
190 } // namespace ntp_snippets | |
OLD | NEW |