Pin Chrome's shortcut to the Win10 Start menu on install and OS upgrade.
[chromium-blink-merge.git] / content / browser / service_worker / service_worker_version_unittest.cc
blobe52f698de13e452329dfe5a3c21e25ad34c57ec4
1 // Copyright 2014 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.
5 #include "base/basictypes.h"
6 #include "base/run_loop.h"
7 #include "content/browser/message_port_service.h"
8 #include "content/browser/service_worker/embedded_worker_registry.h"
9 #include "content/browser/service_worker/embedded_worker_test_helper.h"
10 #include "content/browser/service_worker/service_worker_context_core.h"
11 #include "content/browser/service_worker/service_worker_registration.h"
12 #include "content/browser/service_worker/service_worker_test_utils.h"
13 #include "content/browser/service_worker/service_worker_utils.h"
14 #include "content/browser/service_worker/service_worker_version.h"
15 #include "content/public/test/test_browser_thread_bundle.h"
16 #include "testing/gtest/include/gtest/gtest.h"
18 // IPC messages for testing ---------------------------------------------------
20 #define IPC_MESSAGE_IMPL
21 #include "ipc/ipc_message_macros.h"
23 #define IPC_MESSAGE_START TestMsgStart
25 IPC_MESSAGE_CONTROL0(TestMsg_Message);
26 IPC_MESSAGE_ROUTED1(TestMsg_MessageFromWorker, int);
28 // ---------------------------------------------------------------------------
30 namespace content {
32 namespace {
34 static const int kRenderProcessId = 1;
36 class MessageReceiver : public EmbeddedWorkerTestHelper {
37 public:
38 MessageReceiver()
39 : EmbeddedWorkerTestHelper(base::FilePath(), kRenderProcessId),
40 current_embedded_worker_id_(0) {}
41 ~MessageReceiver() override {}
43 bool OnMessageToWorker(int thread_id,
44 int embedded_worker_id,
45 const IPC::Message& message) override {
46 if (EmbeddedWorkerTestHelper::OnMessageToWorker(
47 thread_id, embedded_worker_id, message)) {
48 return true;
50 current_embedded_worker_id_ = embedded_worker_id;
51 bool handled = true;
52 IPC_BEGIN_MESSAGE_MAP(MessageReceiver, message)
53 IPC_MESSAGE_HANDLER(TestMsg_Message, OnMessage)
54 IPC_MESSAGE_UNHANDLED(handled = false)
55 IPC_END_MESSAGE_MAP()
56 return handled;
59 void SimulateSendValueToBrowser(int embedded_worker_id, int value) {
60 SimulateSend(new TestMsg_MessageFromWorker(embedded_worker_id, value));
63 private:
64 void OnMessage() {
65 // Do nothing.
68 int current_embedded_worker_id_;
69 DISALLOW_COPY_AND_ASSIGN(MessageReceiver);
72 void VerifyCalled(bool* called) {
73 *called = true;
76 void ObserveStatusChanges(ServiceWorkerVersion* version,
77 std::vector<ServiceWorkerVersion::Status>* statuses) {
78 statuses->push_back(version->status());
79 version->RegisterStatusChangeCallback(
80 base::Bind(&ObserveStatusChanges, base::Unretained(version), statuses));
83 void ReceiveFetchResult(ServiceWorkerStatusCode* status,
84 ServiceWorkerStatusCode actual_status,
85 ServiceWorkerFetchEventResult actual_result,
86 const ServiceWorkerResponse& response) {
87 *status = actual_status;
90 // A specialized listener class to receive test messages from a worker.
91 class MessageReceiverFromWorker : public EmbeddedWorkerInstance::Listener {
92 public:
93 explicit MessageReceiverFromWorker(EmbeddedWorkerInstance* instance)
94 : instance_(instance) {
95 instance_->AddListener(this);
97 ~MessageReceiverFromWorker() override { instance_->RemoveListener(this); }
99 void OnStarted() override { NOTREACHED(); }
100 void OnStopped(EmbeddedWorkerInstance::Status old_status) override {
101 NOTREACHED();
103 bool OnMessageReceived(const IPC::Message& message) override {
104 bool handled = true;
105 IPC_BEGIN_MESSAGE_MAP(MessageReceiverFromWorker, message)
106 IPC_MESSAGE_HANDLER(TestMsg_MessageFromWorker, OnMessageFromWorker)
107 IPC_MESSAGE_UNHANDLED(handled = false)
108 IPC_END_MESSAGE_MAP()
109 return handled;
112 void OnMessageFromWorker(int value) { received_values_.push_back(value); }
113 const std::vector<int>& received_values() const { return received_values_; }
115 private:
116 EmbeddedWorkerInstance* instance_;
117 std::vector<int> received_values_;
118 DISALLOW_COPY_AND_ASSIGN(MessageReceiverFromWorker);
121 void SetUpDummyMessagePort(std::vector<TransferredMessagePort>* ports) {
122 int port_id = -1;
123 MessagePortService::GetInstance()->Create(MSG_ROUTING_NONE, nullptr,
124 &port_id);
125 TransferredMessagePort dummy_port;
126 dummy_port.id = port_id;
127 ports->push_back(dummy_port);
130 base::Time GetYesterday() {
131 return base::Time::Now() - base::TimeDelta::FromDays(1) -
132 base::TimeDelta::FromSeconds(1);
135 } // namespace
137 class ServiceWorkerVersionTest : public testing::Test {
138 protected:
139 struct RunningStateListener : public ServiceWorkerVersion::Listener {
140 RunningStateListener() : last_status(ServiceWorkerVersion::STOPPED) {}
141 ~RunningStateListener() override {}
142 void OnRunningStateChanged(ServiceWorkerVersion* version) override {
143 last_status = version->running_status();
145 ServiceWorkerVersion::RunningStatus last_status;
148 ServiceWorkerVersionTest()
149 : thread_bundle_(TestBrowserThreadBundle::IO_MAINLOOP) {}
151 void SetUp() override {
152 helper_ = GetMessageReceiver();
154 pattern_ = GURL("http://www.example.com/");
155 registration_ = new ServiceWorkerRegistration(
156 pattern_,
158 helper_->context()->AsWeakPtr());
159 version_ = new ServiceWorkerVersion(
160 registration_.get(),
161 GURL("http://www.example.com/service_worker.js"),
163 helper_->context()->AsWeakPtr());
164 std::vector<ServiceWorkerDatabase::ResourceRecord> records;
165 records.push_back(
166 ServiceWorkerDatabase::ResourceRecord(10, version_->script_url(), 100));
167 version_->script_cache_map()->SetResources(records);
169 // Make the registration findable via storage functions.
170 helper_->context()->storage()->LazyInitialize(base::Bind(&base::DoNothing));
171 base::RunLoop().RunUntilIdle();
172 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_FAILED;
173 helper_->context()->storage()->StoreRegistration(
174 registration_.get(),
175 version_.get(),
176 CreateReceiverOnCurrentThread(&status));
177 base::RunLoop().RunUntilIdle();
178 ASSERT_EQ(SERVICE_WORKER_OK, status);
180 // Simulate adding one process to the pattern.
181 helper_->SimulateAddProcessToPattern(pattern_, kRenderProcessId);
182 ASSERT_TRUE(helper_->context()->process_manager()
183 ->PatternHasProcessToRun(pattern_));
186 virtual scoped_ptr<MessageReceiver> GetMessageReceiver() {
187 return make_scoped_ptr(new MessageReceiver());
190 void TearDown() override {
191 version_ = 0;
192 registration_ = 0;
193 helper_.reset();
196 TestBrowserThreadBundle thread_bundle_;
197 scoped_ptr<MessageReceiver> helper_;
198 scoped_refptr<ServiceWorkerRegistration> registration_;
199 scoped_refptr<ServiceWorkerVersion> version_;
200 GURL pattern_;
202 private:
203 DISALLOW_COPY_AND_ASSIGN(ServiceWorkerVersionTest);
206 class MessageReceiverDisallowStart : public MessageReceiver {
207 public:
208 MessageReceiverDisallowStart()
209 : MessageReceiver() {}
210 ~MessageReceiverDisallowStart() override {}
212 void OnStartWorker(int embedded_worker_id,
213 int64 service_worker_version_id,
214 const GURL& scope,
215 const GURL& script_url) override {
216 // Do nothing.
219 private:
220 DISALLOW_COPY_AND_ASSIGN(MessageReceiverDisallowStart);
223 class ServiceWorkerFailToStartTest : public ServiceWorkerVersionTest {
224 protected:
225 ServiceWorkerFailToStartTest()
226 : ServiceWorkerVersionTest() {}
228 scoped_ptr<MessageReceiver> GetMessageReceiver() override {
229 return make_scoped_ptr(new MessageReceiverDisallowStart());
232 private:
233 DISALLOW_COPY_AND_ASSIGN(ServiceWorkerFailToStartTest);
236 class MessageReceiverDisallowFetch : public MessageReceiver {
237 public:
238 MessageReceiverDisallowFetch() : MessageReceiver() {}
239 ~MessageReceiverDisallowFetch() override {}
241 void OnFetchEvent(int embedded_worker_id,
242 int request_id,
243 const ServiceWorkerFetchRequest& request) override {
244 // Do nothing.
247 private:
248 DISALLOW_COPY_AND_ASSIGN(MessageReceiverDisallowFetch);
251 class ServiceWorkerWaitForeverInFetchTest : public ServiceWorkerVersionTest {
252 protected:
253 ServiceWorkerWaitForeverInFetchTest() : ServiceWorkerVersionTest() {}
255 scoped_ptr<MessageReceiver> GetMessageReceiver() override {
256 return make_scoped_ptr(new MessageReceiverDisallowFetch());
259 private:
260 DISALLOW_COPY_AND_ASSIGN(ServiceWorkerWaitForeverInFetchTest);
263 TEST_F(ServiceWorkerVersionTest, ConcurrentStartAndStop) {
264 // Call StartWorker() multiple times.
265 ServiceWorkerStatusCode status1 = SERVICE_WORKER_ERROR_FAILED;
266 ServiceWorkerStatusCode status2 = SERVICE_WORKER_ERROR_FAILED;
267 ServiceWorkerStatusCode status3 = SERVICE_WORKER_ERROR_FAILED;
268 version_->StartWorker(CreateReceiverOnCurrentThread(&status1));
269 version_->StartWorker(CreateReceiverOnCurrentThread(&status2));
271 EXPECT_EQ(ServiceWorkerVersion::STARTING, version_->running_status());
272 base::RunLoop().RunUntilIdle();
273 EXPECT_EQ(ServiceWorkerVersion::RUNNING, version_->running_status());
275 // Call StartWorker() after it's started.
276 version_->StartWorker(CreateReceiverOnCurrentThread(&status3));
277 base::RunLoop().RunUntilIdle();
279 // All should just succeed.
280 EXPECT_EQ(SERVICE_WORKER_OK, status1);
281 EXPECT_EQ(SERVICE_WORKER_OK, status2);
282 EXPECT_EQ(SERVICE_WORKER_OK, status3);
284 // Call StopWorker() multiple times.
285 status1 = SERVICE_WORKER_ERROR_FAILED;
286 status2 = SERVICE_WORKER_ERROR_FAILED;
287 version_->StopWorker(CreateReceiverOnCurrentThread(&status1));
288 version_->StopWorker(CreateReceiverOnCurrentThread(&status2));
290 EXPECT_EQ(ServiceWorkerVersion::STOPPING, version_->running_status());
291 base::RunLoop().RunUntilIdle();
292 EXPECT_EQ(ServiceWorkerVersion::STOPPED, version_->running_status());
294 // All StopWorker should just succeed.
295 EXPECT_EQ(SERVICE_WORKER_OK, status1);
296 EXPECT_EQ(SERVICE_WORKER_OK, status2);
298 // Start worker again.
299 status1 = SERVICE_WORKER_ERROR_FAILED;
300 status2 = SERVICE_WORKER_ERROR_FAILED;
301 status3 = SERVICE_WORKER_ERROR_FAILED;
303 version_->StartWorker(CreateReceiverOnCurrentThread(&status1));
305 EXPECT_EQ(ServiceWorkerVersion::STARTING, version_->running_status());
306 base::RunLoop().RunUntilIdle();
307 EXPECT_EQ(ServiceWorkerVersion::RUNNING, version_->running_status());
309 // Call StopWorker()
310 status2 = SERVICE_WORKER_ERROR_FAILED;
311 version_->StopWorker(CreateReceiverOnCurrentThread(&status2));
313 // And try calling StartWorker while StopWorker is in queue.
314 version_->StartWorker(CreateReceiverOnCurrentThread(&status3));
316 EXPECT_EQ(ServiceWorkerVersion::STOPPING, version_->running_status());
317 base::RunLoop().RunUntilIdle();
318 EXPECT_EQ(ServiceWorkerVersion::RUNNING, version_->running_status());
320 // All should just succeed.
321 EXPECT_EQ(SERVICE_WORKER_OK, status1);
322 EXPECT_EQ(SERVICE_WORKER_OK, status2);
323 EXPECT_EQ(SERVICE_WORKER_OK, status3);
326 TEST_F(ServiceWorkerVersionTest, DispatchEventToStoppedWorker) {
327 EXPECT_EQ(ServiceWorkerVersion::STOPPED, version_->running_status());
329 // Dispatch an event without starting the worker.
330 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_FAILED;
331 version_->SetStatus(ServiceWorkerVersion::INSTALLING);
332 version_->DispatchInstallEvent(CreateReceiverOnCurrentThread(&status));
333 base::RunLoop().RunUntilIdle();
334 EXPECT_EQ(SERVICE_WORKER_OK, status);
336 // The worker should be now started.
337 EXPECT_EQ(ServiceWorkerVersion::RUNNING, version_->running_status());
339 // Stop the worker, and then dispatch an event immediately after that.
340 status = SERVICE_WORKER_ERROR_FAILED;
341 ServiceWorkerStatusCode stop_status = SERVICE_WORKER_ERROR_FAILED;
342 version_->StopWorker(CreateReceiverOnCurrentThread(&stop_status));
343 version_->DispatchInstallEvent(CreateReceiverOnCurrentThread(&status));
344 base::RunLoop().RunUntilIdle();
345 EXPECT_EQ(SERVICE_WORKER_OK, stop_status);
347 // Dispatch an event should return SERVICE_WORKER_OK since the worker
348 // should have been restarted to dispatch the event.
349 EXPECT_EQ(SERVICE_WORKER_OK, status);
351 // The worker should be now started again.
352 EXPECT_EQ(ServiceWorkerVersion::RUNNING, version_->running_status());
355 TEST_F(ServiceWorkerVersionTest, ReceiveMessageFromWorker) {
356 // Start worker.
357 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_FAILED;
358 version_->StartWorker(CreateReceiverOnCurrentThread(&status));
359 EXPECT_EQ(ServiceWorkerVersion::STARTING, version_->running_status());
360 base::RunLoop().RunUntilIdle();
361 EXPECT_EQ(SERVICE_WORKER_OK, status);
362 EXPECT_EQ(ServiceWorkerVersion::RUNNING, version_->running_status());
364 MessageReceiverFromWorker receiver(version_->embedded_worker());
366 // Simulate sending some dummy values from the worker.
367 helper_->SimulateSendValueToBrowser(
368 version_->embedded_worker()->embedded_worker_id(), 555);
369 helper_->SimulateSendValueToBrowser(
370 version_->embedded_worker()->embedded_worker_id(), 777);
372 // Verify the receiver received the values.
373 ASSERT_EQ(2U, receiver.received_values().size());
374 EXPECT_EQ(555, receiver.received_values()[0]);
375 EXPECT_EQ(777, receiver.received_values()[1]);
378 TEST_F(ServiceWorkerVersionTest, InstallAndWaitCompletion) {
379 version_->SetStatus(ServiceWorkerVersion::INSTALLING);
381 // Dispatch an install event.
382 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_FAILED;
383 version_->DispatchInstallEvent(CreateReceiverOnCurrentThread(&status));
385 // Wait for the completion.
386 bool status_change_called = false;
387 version_->RegisterStatusChangeCallback(
388 base::Bind(&VerifyCalled, &status_change_called));
390 base::RunLoop().RunUntilIdle();
392 // Version's status must not have changed during installation.
393 EXPECT_EQ(SERVICE_WORKER_OK, status);
394 EXPECT_FALSE(status_change_called);
395 EXPECT_EQ(ServiceWorkerVersion::INSTALLING, version_->status());
398 TEST_F(ServiceWorkerVersionTest, ActivateAndWaitCompletion) {
399 version_->SetStatus(ServiceWorkerVersion::ACTIVATING);
401 // Dispatch an activate event.
402 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_FAILED;
403 version_->DispatchActivateEvent(CreateReceiverOnCurrentThread(&status));
405 // Wait for the completion.
406 bool status_change_called = false;
407 version_->RegisterStatusChangeCallback(
408 base::Bind(&VerifyCalled, &status_change_called));
410 base::RunLoop().RunUntilIdle();
412 // Version's status must not have changed during activation.
413 EXPECT_EQ(SERVICE_WORKER_OK, status);
414 EXPECT_FALSE(status_change_called);
415 EXPECT_EQ(ServiceWorkerVersion::ACTIVATING, version_->status());
418 TEST_F(ServiceWorkerVersionTest, RepeatedlyObserveStatusChanges) {
419 EXPECT_EQ(ServiceWorkerVersion::NEW, version_->status());
421 // Repeatedly observe status changes (the callback re-registers itself).
422 std::vector<ServiceWorkerVersion::Status> statuses;
423 version_->RegisterStatusChangeCallback(
424 base::Bind(&ObserveStatusChanges, version_, &statuses));
426 version_->SetStatus(ServiceWorkerVersion::INSTALLING);
427 version_->SetStatus(ServiceWorkerVersion::INSTALLED);
428 version_->SetStatus(ServiceWorkerVersion::ACTIVATING);
429 version_->SetStatus(ServiceWorkerVersion::ACTIVATED);
430 version_->SetStatus(ServiceWorkerVersion::REDUNDANT);
432 // Verify that we could successfully observe repeated status changes.
433 ASSERT_EQ(5U, statuses.size());
434 ASSERT_EQ(ServiceWorkerVersion::INSTALLING, statuses[0]);
435 ASSERT_EQ(ServiceWorkerVersion::INSTALLED, statuses[1]);
436 ASSERT_EQ(ServiceWorkerVersion::ACTIVATING, statuses[2]);
437 ASSERT_EQ(ServiceWorkerVersion::ACTIVATED, statuses[3]);
438 ASSERT_EQ(ServiceWorkerVersion::REDUNDANT, statuses[4]);
441 TEST_F(ServiceWorkerVersionTest, IdleTimeout) {
442 // Used to reliably test when the idle time gets reset regardless of clock
443 // granularity.
444 const base::TimeDelta kOneSecond = base::TimeDelta::FromSeconds(1);
446 // Verify the timer is not running when version initializes its status.
447 version_->SetStatus(ServiceWorkerVersion::ACTIVATED);
448 EXPECT_FALSE(version_->timeout_timer_.IsRunning());
450 // Verify the timer is running after the worker is started.
451 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_FAILED;
452 version_->StartWorker(CreateReceiverOnCurrentThread(&status));
453 base::RunLoop().RunUntilIdle();
454 EXPECT_EQ(SERVICE_WORKER_OK, status);
455 EXPECT_TRUE(version_->timeout_timer_.IsRunning());
456 EXPECT_FALSE(version_->idle_time_.is_null());
458 // The idle time should be reset if the worker is restarted without
459 // controllee.
460 status = SERVICE_WORKER_ERROR_FAILED;
461 version_->idle_time_ -= kOneSecond;
462 base::TimeTicks idle_time = version_->idle_time_;
463 version_->StopWorker(CreateReceiverOnCurrentThread(&status));
464 base::RunLoop().RunUntilIdle();
465 EXPECT_EQ(SERVICE_WORKER_OK, status);
466 status = SERVICE_WORKER_ERROR_FAILED;
467 version_->StartWorker(CreateReceiverOnCurrentThread(&status));
468 base::RunLoop().RunUntilIdle();
469 EXPECT_EQ(SERVICE_WORKER_OK, status);
470 EXPECT_TRUE(version_->timeout_timer_.IsRunning());
471 EXPECT_LT(idle_time, version_->idle_time_);
473 // Adding a controllee resets the idle time.
474 version_->idle_time_ -= kOneSecond;
475 idle_time = version_->idle_time_;
476 scoped_ptr<ServiceWorkerProviderHost> host(new ServiceWorkerProviderHost(
477 33 /* dummy render process id */, MSG_ROUTING_NONE /* render_frame_id */,
478 1 /* dummy provider_id */, SERVICE_WORKER_PROVIDER_FOR_WINDOW,
479 helper_->context()->AsWeakPtr(), NULL));
480 version_->AddControllee(host.get());
481 EXPECT_TRUE(version_->timeout_timer_.IsRunning());
482 EXPECT_LT(idle_time, version_->idle_time_);
484 // Completing an event resets the idle time.
485 status = SERVICE_WORKER_ERROR_FAILED;
486 version_->idle_time_ -= kOneSecond;
487 idle_time = version_->idle_time_;
488 version_->DispatchFetchEvent(ServiceWorkerFetchRequest(),
489 base::Bind(&base::DoNothing),
490 base::Bind(&ReceiveFetchResult, &status));
491 base::RunLoop().RunUntilIdle();
493 EXPECT_EQ(SERVICE_WORKER_OK, status);
494 EXPECT_LT(idle_time, version_->idle_time_);
496 // Dispatching a message event resets the idle time.
497 std::vector<TransferredMessagePort> ports;
498 SetUpDummyMessagePort(&ports);
499 status = SERVICE_WORKER_ERROR_FAILED;
500 version_->idle_time_ -= kOneSecond;
501 idle_time = version_->idle_time_;
502 version_->DispatchMessageEvent(base::string16(), ports,
503 CreateReceiverOnCurrentThread(&status));
504 base::RunLoop().RunUntilIdle();
505 MessagePortService::GetInstance()->Destroy(ports[0].id);
507 EXPECT_EQ(SERVICE_WORKER_OK, status);
508 EXPECT_LT(idle_time, version_->idle_time_);
511 TEST_F(ServiceWorkerVersionTest, SetDevToolsAttached) {
512 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_FAILED;
513 version_->StartWorker(CreateReceiverOnCurrentThread(&status));
515 ASSERT_EQ(ServiceWorkerVersion::STARTING, version_->running_status());
517 ASSERT_TRUE(version_->timeout_timer_.IsRunning());
518 ASSERT_FALSE(version_->start_time_.is_null());
519 ASSERT_FALSE(version_->skip_recording_startup_time_);
521 // Simulate DevTools is attached. This should deactivate the timer for start
522 // timeout, but not stop the timer itself.
523 version_->SetDevToolsAttached(true);
524 EXPECT_TRUE(version_->timeout_timer_.IsRunning());
525 EXPECT_TRUE(version_->start_time_.is_null());
526 EXPECT_TRUE(version_->skip_recording_startup_time_);
528 // Simulate DevTools is detached. This should reactivate the timer for start
529 // timeout.
530 version_->SetDevToolsAttached(false);
531 EXPECT_TRUE(version_->timeout_timer_.IsRunning());
532 EXPECT_FALSE(version_->start_time_.is_null());
533 EXPECT_TRUE(version_->skip_recording_startup_time_);
535 base::RunLoop().RunUntilIdle();
536 EXPECT_EQ(SERVICE_WORKER_OK, status);
537 EXPECT_EQ(ServiceWorkerVersion::RUNNING, version_->running_status());
540 TEST_F(ServiceWorkerVersionTest, StoppingBeforeDestruct) {
541 RunningStateListener listener;
542 version_->AddListener(&listener);
544 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_FAILED;
545 version_->StartWorker(CreateReceiverOnCurrentThread(&status));
546 base::RunLoop().RunUntilIdle();
547 EXPECT_EQ(SERVICE_WORKER_OK, status);
548 EXPECT_EQ(ServiceWorkerVersion::RUNNING, version_->running_status());
549 EXPECT_EQ(ServiceWorkerVersion::RUNNING, listener.last_status);
551 version_ = nullptr;
552 EXPECT_EQ(ServiceWorkerVersion::STOPPING, listener.last_status);
555 // Test that update isn't triggered for a non-stale worker.
556 TEST_F(ServiceWorkerVersionTest, StaleUpdate_FreshWorker) {
557 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_FAILED;
559 version_->SetStatus(ServiceWorkerVersion::ACTIVATED);
560 registration_->SetActiveVersion(version_);
561 registration_->set_last_update_check(base::Time::Now());
562 version_->DispatchPushEvent(CreateReceiverOnCurrentThread(&status),
563 std::string());
564 base::RunLoop().RunUntilIdle();
566 EXPECT_EQ(SERVICE_WORKER_OK, status);
567 EXPECT_TRUE(version_->stale_time_.is_null());
568 EXPECT_FALSE(version_->update_timer_.IsRunning());
571 // Test that update isn't triggered for a non-active worker.
572 TEST_F(ServiceWorkerVersionTest, StaleUpdate_NonActiveWorker) {
573 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_FAILED;
575 version_->SetStatus(ServiceWorkerVersion::INSTALLING);
576 registration_->SetInstallingVersion(version_);
577 registration_->set_last_update_check(GetYesterday());
578 version_->DispatchInstallEvent(CreateReceiverOnCurrentThread(&status));
579 base::RunLoop().RunUntilIdle();
581 EXPECT_EQ(SERVICE_WORKER_OK, status);
582 EXPECT_TRUE(version_->stale_time_.is_null());
583 EXPECT_FALSE(version_->update_timer_.IsRunning());
586 // Test that staleness is detected when starting a worker.
587 TEST_F(ServiceWorkerVersionTest, StaleUpdate_StartWorker) {
588 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_FAILED;
590 // Starting the worker marks it as stale.
591 version_->SetStatus(ServiceWorkerVersion::ACTIVATED);
592 registration_->SetActiveVersion(version_);
593 registration_->set_last_update_check(GetYesterday());
594 version_->DispatchPushEvent(CreateReceiverOnCurrentThread(&status),
595 std::string());
596 base::RunLoop().RunUntilIdle();
597 EXPECT_EQ(SERVICE_WORKER_OK, status);
598 EXPECT_FALSE(version_->stale_time_.is_null());
599 EXPECT_FALSE(version_->update_timer_.IsRunning());
601 // Update is actually scheduled after the worker stops.
602 version_->StopWorker(CreateReceiverOnCurrentThread(&status));
603 base::RunLoop().RunUntilIdle();
604 EXPECT_EQ(SERVICE_WORKER_OK, status);
605 EXPECT_TRUE(version_->stale_time_.is_null());
606 EXPECT_TRUE(version_->update_timer_.IsRunning());
609 // Test that staleness is detected on a running worker.
610 TEST_F(ServiceWorkerVersionTest, StaleUpdate_RunningWorker) {
611 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_FAILED;
613 // Start a fresh worker.
614 version_->SetStatus(ServiceWorkerVersion::ACTIVATED);
615 registration_->SetActiveVersion(version_);
616 registration_->set_last_update_check(base::Time::Now());
617 version_->DispatchPushEvent(CreateReceiverOnCurrentThread(&status),
618 std::string());
619 base::RunLoop().RunUntilIdle();
620 EXPECT_EQ(SERVICE_WORKER_OK, status);
621 EXPECT_TRUE(version_->stale_time_.is_null());
623 // Simulate it running for a day. It will be marked stale.
624 registration_->set_last_update_check(GetYesterday());
625 version_->OnTimeoutTimer();
626 EXPECT_FALSE(version_->stale_time_.is_null());
627 EXPECT_FALSE(version_->update_timer_.IsRunning());
629 // Simulate it running for past the wait threshold. The update will be
630 // scheduled.
631 version_->stale_time_ =
632 base::TimeTicks::Now() -
633 base::TimeDelta::FromMinutes(
634 ServiceWorkerVersion::kStartWorkerTimeoutMinutes + 1);
635 version_->OnTimeoutTimer();
636 EXPECT_TRUE(version_->stale_time_.is_null());
637 EXPECT_TRUE(version_->update_timer_.IsRunning());
640 // Test that a stream of events doesn't restart the timer.
641 TEST_F(ServiceWorkerVersionTest, StaleUpdate_DoNotDeferTimer) {
642 // Make a stale worker.
643 version_->SetStatus(ServiceWorkerVersion::ACTIVATED);
644 registration_->SetActiveVersion(version_);
645 registration_->set_last_update_check(GetYesterday());
646 base::TimeTicks stale_time =
647 base::TimeTicks::Now() -
648 base::TimeDelta::FromMinutes(
649 ServiceWorkerVersion::kStartWorkerTimeoutMinutes + 1);
650 version_->stale_time_ = stale_time;
652 // Stale time is not deferred.
653 version_->DispatchPushEvent(
654 base::Bind(&ServiceWorkerUtils::NoOpStatusCallback), std::string());
655 version_->DispatchPushEvent(
656 base::Bind(&ServiceWorkerUtils::NoOpStatusCallback), std::string());
657 EXPECT_EQ(stale_time, version_->stale_time_);
659 // Timeout triggers the update.
660 version_->OnTimeoutTimer();
661 EXPECT_TRUE(version_->stale_time_.is_null());
662 EXPECT_TRUE(version_->update_timer_.IsRunning());
664 // Update timer is not deferred.
665 base::TimeTicks run_time = version_->update_timer_.desired_run_time();
666 version_->DispatchPushEvent(
667 base::Bind(&ServiceWorkerUtils::NoOpStatusCallback), std::string());
668 version_->DispatchPushEvent(
669 base::Bind(&ServiceWorkerUtils::NoOpStatusCallback), std::string());
670 version_->DispatchPushEvent(
671 base::Bind(&ServiceWorkerUtils::NoOpStatusCallback), std::string());
672 base::RunLoop().RunUntilIdle();
673 EXPECT_TRUE(version_->stale_time_.is_null());
674 EXPECT_EQ(run_time, version_->update_timer_.desired_run_time());
677 TEST_F(ServiceWorkerWaitForeverInFetchTest, RequestTimeout) {
678 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_NETWORK; // dummy value
680 version_->SetStatus(ServiceWorkerVersion::ACTIVATED);
681 version_->DispatchFetchEvent(ServiceWorkerFetchRequest(),
682 base::Bind(&base::DoNothing),
683 base::Bind(&ReceiveFetchResult, &status));
684 base::RunLoop().RunUntilIdle();
686 // Callback has not completed yet.
687 EXPECT_EQ(SERVICE_WORKER_ERROR_NETWORK, status);
688 EXPECT_EQ(ServiceWorkerVersion::RUNNING, version_->running_status());
690 // Simulate timeout.
691 EXPECT_TRUE(version_->timeout_timer_.IsRunning());
692 version_->SetAllRequestTimes(
693 base::TimeTicks::Now() -
694 base::TimeDelta::FromMinutes(
695 ServiceWorkerVersion::kRequestTimeoutMinutes + 1));
696 version_->timeout_timer_.user_task().Run();
697 base::RunLoop().RunUntilIdle();
698 EXPECT_EQ(SERVICE_WORKER_ERROR_TIMEOUT, status);
699 EXPECT_EQ(ServiceWorkerVersion::STOPPED, version_->running_status());
702 TEST_F(ServiceWorkerFailToStartTest, RendererCrash) {
703 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_NETWORK; // dummy value
704 version_->StartWorker(
705 CreateReceiverOnCurrentThread(&status));
706 base::RunLoop().RunUntilIdle();
708 // Callback has not completed yet.
709 EXPECT_EQ(SERVICE_WORKER_ERROR_NETWORK, status);
710 EXPECT_EQ(ServiceWorkerVersion::STARTING, version_->running_status());
712 // Simulate renderer crash: do what
713 // ServiceWorkerDispatcherHost::OnFilterRemoved does.
714 int process_id = helper_->mock_render_process_id();
715 helper_->context()->RemoveAllProviderHostsForProcess(process_id);
716 helper_->context()->embedded_worker_registry()->RemoveChildProcessSender(
717 process_id);
718 base::RunLoop().RunUntilIdle();
720 // Callback completed.
721 EXPECT_EQ(SERVICE_WORKER_ERROR_START_WORKER_FAILED, status);
722 EXPECT_EQ(ServiceWorkerVersion::STOPPED, version_->running_status());
725 TEST_F(ServiceWorkerFailToStartTest, Timeout) {
726 ServiceWorkerStatusCode status = SERVICE_WORKER_ERROR_NETWORK; // dummy value
728 // We could just call StartWorker but make it interesting and test
729 // starting the worker as part of dispatching an event.
730 version_->SetStatus(ServiceWorkerVersion::ACTIVATING);
731 version_->DispatchActivateEvent(CreateReceiverOnCurrentThread(&status));
732 base::RunLoop().RunUntilIdle();
734 // Callback has not completed yet.
735 EXPECT_EQ(SERVICE_WORKER_ERROR_NETWORK, status);
736 EXPECT_EQ(ServiceWorkerVersion::STARTING, version_->running_status());
738 // Simulate timeout.
739 EXPECT_TRUE(version_->timeout_timer_.IsRunning());
740 version_->start_time_ =
741 base::TimeTicks::Now() -
742 base::TimeDelta::FromMinutes(
743 ServiceWorkerVersion::kStartWorkerTimeoutMinutes + 1);
744 version_->timeout_timer_.user_task().Run();
745 base::RunLoop().RunUntilIdle();
746 EXPECT_EQ(SERVICE_WORKER_ERROR_TIMEOUT, status);
747 EXPECT_EQ(ServiceWorkerVersion::STOPPED, version_->running_status());
750 } // namespace content