OLD | NEW |
| (Empty) |
1 // Copyright (c) 2013 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 | |
5 #include "chrome/browser/policy/cloud/component_cloud_policy_updater.h" | |
6 | |
7 #include "base/callback.h" | |
8 #include "base/compiler_specific.h" | |
9 #include "base/files/scoped_temp_dir.h" | |
10 #include "base/sequenced_task_runner.h" | |
11 #include "base/sha1.h" | |
12 #include "base/test/test_simple_task_runner.h" | |
13 #include "base/values.h" | |
14 #include "chrome/browser/policy/cloud/cloud_policy_constants.h" | |
15 #include "chrome/browser/policy/cloud/component_cloud_policy_store.h" | |
16 #include "chrome/browser/policy/cloud/external_policy_data_fetcher.h" | |
17 #include "chrome/browser/policy/cloud/policy_builder.h" | |
18 #include "chrome/browser/policy/cloud/resource_cache.h" | |
19 #include "chrome/browser/policy/proto/cloud/chrome_extension_policy.pb.h" | |
20 #include "chrome/browser/policy/proto/cloud/device_management_backend.pb.h" | |
21 #include "components/policy/core/common/external_data_fetcher.h" | |
22 #include "components/policy/core/common/policy_bundle.h" | |
23 #include "components/policy/core/common/policy_map.h" | |
24 #include "components/policy/core/common/policy_types.h" | |
25 #include "net/url_request/test_url_fetcher_factory.h" | |
26 #include "net/url_request/url_fetcher_delegate.h" | |
27 #include "net/url_request/url_request_context_getter.h" | |
28 #include "testing/gmock/include/gmock/gmock.h" | |
29 #include "testing/gtest/include/gtest/gtest.h" | |
30 #include "url/gurl.h" | |
31 | |
32 namespace em = enterprise_management; | |
33 | |
34 using testing::Mock; | |
35 | |
36 namespace policy { | |
37 | |
38 namespace { | |
39 | |
40 const char kTestExtension[] = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"; | |
41 const char kTestExtension2[] = "bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb"; | |
42 const char kTestExtension3[] = "cccccccccccccccccccccccccccccccc"; | |
43 const char kTestDownload[] = "http://example.com/getpolicy?id=123"; | |
44 const char kTestDownload2[] = "http://example.com/getpolicy?id=456"; | |
45 const char kTestDownload3[] = "http://example.com/getpolicy?id=789"; | |
46 const char kTestPolicy[] = | |
47 "{" | |
48 " \"Name\": {" | |
49 " \"Value\": \"disabled\"" | |
50 " }," | |
51 " \"Second\": {" | |
52 " \"Value\": \"maybe\"," | |
53 " \"Level\": \"Recommended\"" | |
54 " }" | |
55 "}"; | |
56 | |
57 class MockComponentCloudPolicyStoreDelegate | |
58 : public ComponentCloudPolicyStore::Delegate { | |
59 public: | |
60 virtual ~MockComponentCloudPolicyStoreDelegate() {} | |
61 | |
62 MOCK_METHOD0(OnComponentCloudPolicyStoreUpdated, void()); | |
63 }; | |
64 | |
65 } // namespace | |
66 | |
67 class ComponentCloudPolicyUpdaterTest : public testing::Test { | |
68 protected: | |
69 virtual void SetUp() OVERRIDE; | |
70 virtual void TearDown() OVERRIDE; | |
71 | |
72 scoped_ptr<em::PolicyFetchResponse> CreateResponse(); | |
73 | |
74 scoped_refptr<base::TestSimpleTaskRunner> task_runner_; | |
75 base::ScopedTempDir temp_dir_; | |
76 scoped_ptr<ResourceCache> cache_; | |
77 scoped_ptr<ComponentCloudPolicyStore> store_; | |
78 MockComponentCloudPolicyStoreDelegate store_delegate_; | |
79 net::TestURLFetcherFactory fetcher_factory_; | |
80 scoped_ptr<ExternalPolicyDataFetcherBackend> fetcher_backend_; | |
81 scoped_ptr<ComponentCloudPolicyUpdater> updater_; | |
82 ComponentPolicyBuilder builder_; | |
83 PolicyBundle expected_bundle_; | |
84 }; | |
85 | |
86 void ComponentCloudPolicyUpdaterTest::SetUp() { | |
87 ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); | |
88 task_runner_ = new base::TestSimpleTaskRunner(); | |
89 cache_.reset(new ResourceCache(temp_dir_.path(), task_runner_)); | |
90 store_.reset(new ComponentCloudPolicyStore(&store_delegate_, cache_.get())); | |
91 store_->SetCredentials(ComponentPolicyBuilder::kFakeUsername, | |
92 ComponentPolicyBuilder::kFakeToken); | |
93 fetcher_factory_.set_remove_fetcher_on_delete(true); | |
94 fetcher_backend_.reset(new ExternalPolicyDataFetcherBackend( | |
95 task_runner_, | |
96 scoped_refptr<net::URLRequestContextGetter>())); | |
97 updater_.reset(new ComponentCloudPolicyUpdater( | |
98 task_runner_, | |
99 fetcher_backend_->CreateFrontend(task_runner_), | |
100 store_.get())); | |
101 ASSERT_EQ(store_->policy().end(), store_->policy().begin()); | |
102 | |
103 builder_.policy_data().set_policy_type( | |
104 dm_protocol::kChromeExtensionPolicyType); | |
105 builder_.policy_data().set_settings_entity_id(kTestExtension); | |
106 builder_.payload().set_download_url(kTestDownload); | |
107 builder_.payload().set_secure_hash(base::SHA1HashString(kTestPolicy)); | |
108 | |
109 PolicyNamespace ns(POLICY_DOMAIN_EXTENSIONS, kTestExtension); | |
110 PolicyMap& policy = expected_bundle_.Get(ns); | |
111 policy.Set("Name", POLICY_LEVEL_MANDATORY, POLICY_SCOPE_USER, | |
112 base::Value::CreateStringValue("disabled"), NULL); | |
113 policy.Set("Second", POLICY_LEVEL_RECOMMENDED, POLICY_SCOPE_USER, | |
114 base::Value::CreateStringValue("maybe"), NULL); | |
115 } | |
116 | |
117 void ComponentCloudPolicyUpdaterTest::TearDown() { | |
118 updater_.reset(); | |
119 task_runner_->RunUntilIdle(); | |
120 } | |
121 | |
122 scoped_ptr<em::PolicyFetchResponse> | |
123 ComponentCloudPolicyUpdaterTest::CreateResponse() { | |
124 builder_.Build(); | |
125 return make_scoped_ptr(new em::PolicyFetchResponse(builder_.policy())); | |
126 } | |
127 | |
128 TEST_F(ComponentCloudPolicyUpdaterTest, FetchAndCache) { | |
129 // Submit a policy fetch response. | |
130 updater_->UpdateExternalPolicy(CreateResponse()); | |
131 task_runner_->RunUntilIdle(); | |
132 | |
133 // Verify that a download has been started. | |
134 net::TestURLFetcher* fetcher = fetcher_factory_.GetFetcherByID(0); | |
135 ASSERT_TRUE(fetcher); | |
136 EXPECT_EQ(GURL(kTestDownload), fetcher->GetOriginalURL()); | |
137 | |
138 // Complete the download. | |
139 fetcher->set_response_code(200); | |
140 fetcher->SetResponseString(kTestPolicy); | |
141 fetcher->delegate()->OnURLFetchComplete(fetcher); | |
142 EXPECT_CALL(store_delegate_, OnComponentCloudPolicyStoreUpdated()); | |
143 task_runner_->RunUntilIdle(); | |
144 Mock::VerifyAndClearExpectations(&store_delegate_); | |
145 | |
146 // Verify that the downloaded policy is being served. | |
147 EXPECT_TRUE(store_->policy().Equals(expected_bundle_)); | |
148 } | |
149 | |
150 TEST_F(ComponentCloudPolicyUpdaterTest, PolicyFetchResponseTooLarge) { | |
151 // Submit a policy fetch response that exceeds the allowed maximum size. | |
152 std::string long_download("http://example.com/get?id="); | |
153 long_download.append(20 * 1024, '1'); | |
154 builder_.payload().set_download_url(long_download); | |
155 updater_->UpdateExternalPolicy(CreateResponse()); | |
156 | |
157 // Submit two valid policy fetch responses. | |
158 builder_.policy_data().set_settings_entity_id(kTestExtension2); | |
159 builder_.payload().set_download_url(kTestDownload2); | |
160 updater_->UpdateExternalPolicy(CreateResponse()); | |
161 builder_.policy_data().set_settings_entity_id(kTestExtension3); | |
162 builder_.payload().set_download_url(kTestDownload3); | |
163 updater_->UpdateExternalPolicy(CreateResponse()); | |
164 task_runner_->RunUntilIdle(); | |
165 | |
166 // Verify that the first policy fetch response has been ignored and downloads | |
167 // have been started for the next two fetch responses instead. | |
168 net::TestURLFetcher* fetcher = fetcher_factory_.GetFetcherByID(0); | |
169 ASSERT_TRUE(fetcher); | |
170 EXPECT_EQ(GURL(kTestDownload2), fetcher->GetOriginalURL()); | |
171 fetcher = fetcher_factory_.GetFetcherByID(1); | |
172 ASSERT_TRUE(fetcher); | |
173 EXPECT_EQ(GURL(kTestDownload3), fetcher->GetOriginalURL()); | |
174 } | |
175 | |
176 TEST_F(ComponentCloudPolicyUpdaterTest, PolicyFetchResponseInvalid) { | |
177 // Submit an invalid policy fetch response. | |
178 builder_.policy_data().set_username("wronguser@example.com"); | |
179 updater_->UpdateExternalPolicy(CreateResponse()); | |
180 | |
181 // Submit two valid policy fetch responses. | |
182 builder_.policy_data().set_username(ComponentPolicyBuilder::kFakeUsername); | |
183 builder_.policy_data().set_settings_entity_id(kTestExtension2); | |
184 builder_.payload().set_download_url(kTestDownload2); | |
185 updater_->UpdateExternalPolicy(CreateResponse()); | |
186 builder_.policy_data().set_settings_entity_id(kTestExtension3); | |
187 builder_.payload().set_download_url(kTestDownload3); | |
188 updater_->UpdateExternalPolicy(CreateResponse()); | |
189 task_runner_->RunUntilIdle(); | |
190 | |
191 // Verify that the first policy fetch response has been ignored and downloads | |
192 // have been started for the next two fetch responses instead. | |
193 net::TestURLFetcher* fetcher = fetcher_factory_.GetFetcherByID(0); | |
194 ASSERT_TRUE(fetcher); | |
195 EXPECT_EQ(GURL(kTestDownload2), fetcher->GetOriginalURL()); | |
196 fetcher = fetcher_factory_.GetFetcherByID(1); | |
197 ASSERT_TRUE(fetcher); | |
198 EXPECT_EQ(GURL(kTestDownload3), fetcher->GetOriginalURL()); | |
199 } | |
200 | |
201 TEST_F(ComponentCloudPolicyUpdaterTest, AlreadyCached) { | |
202 // Cache policy for an extension. | |
203 builder_.Build(); | |
204 PolicyNamespace ns(POLICY_DOMAIN_EXTENSIONS, kTestExtension); | |
205 EXPECT_CALL(store_delegate_, OnComponentCloudPolicyStoreUpdated()); | |
206 EXPECT_TRUE(store_->Store(ns, | |
207 builder_.GetBlob(), | |
208 base::SHA1HashString(kTestPolicy), | |
209 kTestPolicy)); | |
210 Mock::VerifyAndClearExpectations(&store_delegate_); | |
211 | |
212 // Submit a policy fetch response whose extension ID and hash match the | |
213 // already cached policy. | |
214 updater_->UpdateExternalPolicy(CreateResponse()); | |
215 task_runner_->RunUntilIdle(); | |
216 | |
217 // Verify that no download has been started. | |
218 EXPECT_FALSE(fetcher_factory_.GetFetcherByID(0)); | |
219 } | |
220 | |
221 TEST_F(ComponentCloudPolicyUpdaterTest, PolicyDataInvalid) { | |
222 // Submit three policy fetch responses. | |
223 updater_->UpdateExternalPolicy(CreateResponse()); | |
224 builder_.payload().set_download_url(kTestDownload2); | |
225 builder_.policy_data().set_settings_entity_id(kTestExtension2); | |
226 updater_->UpdateExternalPolicy(CreateResponse()); | |
227 builder_.policy_data().set_settings_entity_id(kTestExtension3); | |
228 builder_.payload().set_download_url(kTestDownload3); | |
229 updater_->UpdateExternalPolicy(CreateResponse()); | |
230 task_runner_->RunUntilIdle(); | |
231 | |
232 // Verify that the first download has been started. | |
233 net::TestURLFetcher* fetcher = fetcher_factory_.GetFetcherByID(0); | |
234 ASSERT_TRUE(fetcher); | |
235 EXPECT_EQ(GURL(kTestDownload), fetcher->GetOriginalURL()); | |
236 | |
237 // Verify that the second download has been started. | |
238 fetcher = fetcher_factory_.GetFetcherByID(1); | |
239 ASSERT_TRUE(fetcher); | |
240 EXPECT_EQ(GURL(kTestDownload2), fetcher->GetOriginalURL()); | |
241 | |
242 // Indicate that the policy data size will exceed allowed maximum. | |
243 fetcher->delegate()->OnURLFetchDownloadProgress(fetcher, 6 * 1024 * 1024, -1); | |
244 task_runner_->RunUntilIdle(); | |
245 | |
246 // Verify that the third download has been started. | |
247 fetcher = fetcher_factory_.GetFetcherByID(2); | |
248 ASSERT_TRUE(fetcher); | |
249 EXPECT_EQ(GURL(kTestDownload3), fetcher->GetOriginalURL()); | |
250 } | |
251 | |
252 TEST_F(ComponentCloudPolicyUpdaterTest, FetchUpdatedData) { | |
253 // Submit a policy fetch response. | |
254 updater_->UpdateExternalPolicy(CreateResponse()); | |
255 task_runner_->RunUntilIdle(); | |
256 | |
257 // Verify that the first download has been started. | |
258 net::TestURLFetcher* fetcher = fetcher_factory_.GetFetcherByID(0); | |
259 ASSERT_TRUE(fetcher); | |
260 EXPECT_EQ(GURL(kTestDownload), fetcher->GetOriginalURL()); | |
261 | |
262 // Submit a second policy fetch response for the same extension with an | |
263 // updated download URL. | |
264 builder_.payload().set_download_url(kTestDownload2); | |
265 updater_->UpdateExternalPolicy(CreateResponse()); | |
266 task_runner_->RunUntilIdle(); | |
267 | |
268 // Verify that the first download is no longer running. | |
269 EXPECT_FALSE(fetcher_factory_.GetFetcherByID(0)); | |
270 | |
271 // Verify that the second download has been started. | |
272 fetcher = fetcher_factory_.GetFetcherByID(1); | |
273 ASSERT_TRUE(fetcher); | |
274 EXPECT_EQ(GURL(kTestDownload2), fetcher->GetOriginalURL()); | |
275 } | |
276 | |
277 TEST_F(ComponentCloudPolicyUpdaterTest, FetchUpdatedDataWithoutPolicy) { | |
278 // Submit a policy fetch response. | |
279 updater_->UpdateExternalPolicy(CreateResponse()); | |
280 task_runner_->RunUntilIdle(); | |
281 | |
282 // Verify that the download has been started. | |
283 net::TestURLFetcher* fetcher = fetcher_factory_.GetFetcherByID(0); | |
284 ASSERT_TRUE(fetcher); | |
285 EXPECT_EQ(GURL(kTestDownload), fetcher->GetOriginalURL()); | |
286 | |
287 // Complete the download. | |
288 fetcher->set_response_code(200); | |
289 fetcher->SetResponseString(kTestPolicy); | |
290 fetcher->delegate()->OnURLFetchComplete(fetcher); | |
291 EXPECT_CALL(store_delegate_, OnComponentCloudPolicyStoreUpdated()); | |
292 task_runner_->RunUntilIdle(); | |
293 Mock::VerifyAndClearExpectations(&store_delegate_); | |
294 | |
295 // Verify that the downloaded policy is being served. | |
296 EXPECT_TRUE(store_->policy().Equals(expected_bundle_)); | |
297 | |
298 // Submit a second policy fetch response for the same extension with no | |
299 // download URL, meaning that no policy should be provided for this extension. | |
300 builder_.payload().clear_download_url(); | |
301 builder_.payload().clear_secure_hash(); | |
302 EXPECT_CALL(store_delegate_, OnComponentCloudPolicyStoreUpdated()); | |
303 updater_->UpdateExternalPolicy(CreateResponse()); | |
304 Mock::VerifyAndClearExpectations(&store_delegate_); | |
305 task_runner_->RunUntilIdle(); | |
306 | |
307 // Verify that no download has been started. | |
308 EXPECT_FALSE(fetcher_factory_.GetFetcherByID(1)); | |
309 | |
310 // Verify that the policy is no longer being served. | |
311 const PolicyBundle empty_bundle; | |
312 EXPECT_TRUE(store_->policy().Equals(empty_bundle)); | |
313 } | |
314 | |
315 TEST_F(ComponentCloudPolicyUpdaterTest, NoPolicy) { | |
316 // Submit a policy fetch response with a valid download URL. | |
317 updater_->UpdateExternalPolicy(CreateResponse()); | |
318 task_runner_->RunUntilIdle(); | |
319 | |
320 // Verify that the download has been started. | |
321 EXPECT_TRUE(fetcher_factory_.GetFetcherByID(0)); | |
322 | |
323 // Update the policy fetch response before the download has finished. The new | |
324 // policy fetch response has no download URL. | |
325 builder_.payload().Clear(); | |
326 updater_->UpdateExternalPolicy(CreateResponse()); | |
327 task_runner_->RunUntilIdle(); | |
328 | |
329 // Verify that the download is no longer running. | |
330 EXPECT_FALSE(fetcher_factory_.GetFetcherByID(0)); | |
331 } | |
332 | |
333 TEST_F(ComponentCloudPolicyUpdaterTest, CancelUpdate) { | |
334 // Submit a policy fetch response with a valid download URL. | |
335 updater_->UpdateExternalPolicy(CreateResponse()); | |
336 task_runner_->RunUntilIdle(); | |
337 | |
338 // Verify that the download has been started. | |
339 EXPECT_TRUE(fetcher_factory_.GetFetcherByID(0)); | |
340 | |
341 // Now cancel that update before the download completes. | |
342 EXPECT_CALL(store_delegate_, OnComponentCloudPolicyStoreUpdated()).Times(0); | |
343 updater_->CancelUpdate( | |
344 PolicyNamespace(POLICY_DOMAIN_EXTENSIONS, kTestExtension)); | |
345 task_runner_->RunUntilIdle(); | |
346 Mock::VerifyAndClearExpectations(&store_delegate_); | |
347 EXPECT_FALSE(fetcher_factory_.GetFetcherByID(0)); | |
348 } | |
349 | |
350 } // namespace policy | |
OLD | NEW |