Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(83)

Side by Side Diff: components/ntp_snippets/ntp_snippets_fetcher.cc

Issue 1677073002: Fetch snippets from ChromeReader and show them on the NTP (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Cleaning up Created 4 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(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
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698