OLD | NEW |
| (Empty) |
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. | |
4 | |
5 #include "base/bind.h" | |
6 #include "base/message_loop/message_loop.h" | |
7 #include "base/run_loop.h" | |
8 #include "extensions/browser/stash_backend.h" | |
9 #include "testing/gtest/include/gtest/gtest.h" | |
10 | |
11 namespace extensions { | |
12 | |
13 class StashServiceTest : public testing::Test, public mojo::ErrorHandler { | |
14 public: | |
15 enum Event { | |
16 EVENT_NONE, | |
17 EVENT_STASH_RETRIEVED, | |
18 }; | |
19 | |
20 StashServiceTest() {} | |
21 | |
22 virtual void SetUp() override { | |
23 expecting_error_ = false; | |
24 expected_event_ = EVENT_NONE; | |
25 stash_backend_.reset(new StashBackend); | |
26 stash_backend_->BindToRequest(mojo::GetProxy(&stash_service_)); | |
27 stash_service_.set_error_handler(this); | |
28 } | |
29 | |
30 void OnConnectionError() override { FAIL() << "Unexpected connection error"; } | |
31 | |
32 mojo::Array<StashedObjectPtr> RetrieveStash() { | |
33 mojo::Array<StashedObjectPtr> stash; | |
34 stash_service_->RetrieveStash(base::Bind( | |
35 &StashServiceTest::StashRetrieved, base::Unretained(this), &stash)); | |
36 WaitForEvent(EVENT_STASH_RETRIEVED); | |
37 return stash.Pass(); | |
38 } | |
39 | |
40 void StashRetrieved(mojo::Array<StashedObjectPtr>* output, | |
41 mojo::Array<StashedObjectPtr> stash) { | |
42 *output = stash.Pass(); | |
43 EventReceived(EVENT_STASH_RETRIEVED); | |
44 } | |
45 | |
46 void WaitForEvent(Event event) { | |
47 expected_event_ = event; | |
48 base::RunLoop run_loop; | |
49 stop_run_loop_ = run_loop.QuitClosure(); | |
50 run_loop.Run(); | |
51 } | |
52 | |
53 void EventReceived(Event event) { | |
54 if (event == expected_event_ && !stop_run_loop_.is_null()) | |
55 stop_run_loop_.Run(); | |
56 } | |
57 | |
58 protected: | |
59 base::MessageLoop message_loop_; | |
60 base::Closure stop_run_loop_; | |
61 scoped_ptr<StashBackend> stash_backend_; | |
62 Event expected_event_; | |
63 bool expecting_error_; | |
64 mojo::InterfacePtr<StashService> stash_service_; | |
65 | |
66 private: | |
67 DISALLOW_COPY_AND_ASSIGN(StashServiceTest); | |
68 }; | |
69 | |
70 // Test that adding stashed objects in multiple calls can all be retrieved by a | |
71 // Retrieve call. | |
72 TEST_F(StashServiceTest, AddTwiceAndRetrieve) { | |
73 mojo::Array<StashedObjectPtr> stashed_objects; | |
74 StashedObjectPtr stashed_object(StashedObject::New()); | |
75 stashed_object->id = "test type"; | |
76 stashed_object->data.push_back(1); | |
77 stashed_object->stashed_handles = mojo::Array<mojo::ScopedHandle>(0); | |
78 stashed_objects.push_back(stashed_object.Pass()); | |
79 stash_service_->AddToStash(stashed_objects.Pass()); | |
80 stashed_object = StashedObject::New(); | |
81 stashed_object->id = "test type2"; | |
82 stashed_object->data.push_back(2); | |
83 stashed_object->data.push_back(3); | |
84 stashed_object->stashed_handles = mojo::Array<mojo::ScopedHandle>(0); | |
85 stashed_objects.push_back(stashed_object.Pass()); | |
86 stash_service_->AddToStash(stashed_objects.Pass()); | |
87 stashed_objects = RetrieveStash(); | |
88 ASSERT_EQ(2u, stashed_objects.size()); | |
89 EXPECT_EQ("test type", stashed_objects[0]->id); | |
90 EXPECT_EQ(0u, stashed_objects[0]->stashed_handles.size()); | |
91 EXPECT_EQ(1u, stashed_objects[0]->data.size()); | |
92 EXPECT_EQ(1, stashed_objects[0]->data[0]); | |
93 EXPECT_EQ("test type2", stashed_objects[1]->id); | |
94 EXPECT_EQ(0u, stashed_objects[1]->stashed_handles.size()); | |
95 EXPECT_EQ(2u, stashed_objects[1]->data.size()); | |
96 EXPECT_EQ(2, stashed_objects[1]->data[0]); | |
97 EXPECT_EQ(3, stashed_objects[1]->data[1]); | |
98 } | |
99 | |
100 // Test that handles survive a round-trip through the stash. | |
101 TEST_F(StashServiceTest, StashAndRetrieveHandles) { | |
102 mojo::Array<StashedObjectPtr> stashed_objects; | |
103 StashedObjectPtr stashed_object(StashedObject::New()); | |
104 stashed_object->id = "test type"; | |
105 stashed_object->data.push_back(1); | |
106 | |
107 mojo::ScopedDataPipeConsumerHandle consumer; | |
108 mojo::ScopedDataPipeProducerHandle producer; | |
109 MojoCreateDataPipeOptions options = { | |
110 sizeof(options), MOJO_CREATE_DATA_PIPE_OPTIONS_FLAG_NONE, 1, 1, | |
111 }; | |
112 mojo::CreateDataPipe(&options, &producer, &consumer); | |
113 uint32_t num_bytes = 1; | |
114 MojoResult result = mojo::WriteDataRaw( | |
115 producer.get(), "1", &num_bytes, MOJO_WRITE_DATA_FLAG_ALL_OR_NONE); | |
116 ASSERT_EQ(MOJO_RESULT_OK, result); | |
117 ASSERT_EQ(1u, num_bytes); | |
118 | |
119 stashed_object->stashed_handles.push_back( | |
120 mojo::ScopedHandle::From(producer.Pass())); | |
121 stashed_object->stashed_handles.push_back( | |
122 mojo::ScopedHandle::From(consumer.Pass())); | |
123 stashed_objects.push_back(stashed_object.Pass()); | |
124 stash_service_->AddToStash(stashed_objects.Pass()); | |
125 stashed_objects = RetrieveStash(); | |
126 ASSERT_EQ(1u, stashed_objects.size()); | |
127 EXPECT_EQ("test type", stashed_objects[0]->id); | |
128 ASSERT_EQ(2u, stashed_objects[0]->stashed_handles.size()); | |
129 | |
130 consumer = mojo::ScopedDataPipeConsumerHandle::From( | |
131 stashed_objects[0]->stashed_handles[1].Pass()); | |
132 result = mojo::Wait( | |
133 consumer.get(), MOJO_HANDLE_SIGNAL_READABLE, MOJO_DEADLINE_INDEFINITE); | |
134 ASSERT_EQ(MOJO_RESULT_OK, result); | |
135 char data = '\0'; | |
136 result = mojo::ReadDataRaw( | |
137 consumer.get(), &data, &num_bytes, MOJO_READ_DATA_FLAG_ALL_OR_NONE); | |
138 ASSERT_EQ(MOJO_RESULT_OK, result); | |
139 ASSERT_EQ(1u, num_bytes); | |
140 EXPECT_EQ('1', data); | |
141 } | |
142 | |
143 TEST_F(StashServiceTest, RetrieveWithoutStashing) { | |
144 mojo::Array<StashedObjectPtr> stashed_objects = RetrieveStash(); | |
145 ASSERT_TRUE(!stashed_objects.is_null()); | |
146 EXPECT_EQ(0u, stashed_objects.size()); | |
147 } | |
148 | |
149 // Test that a stash service discards stashed objects when the backend no longer | |
150 // exists. | |
151 TEST_F(StashServiceTest, ServiceWithDeletedBackend) { | |
152 stash_backend_.reset(); | |
153 stash_service_.set_error_handler(this); | |
154 | |
155 mojo::Array<StashedObjectPtr> stashed_objects; | |
156 StashedObjectPtr stashed_object(StashedObject::New()); | |
157 stashed_object->id = "test type"; | |
158 stashed_object->data.push_back(1); | |
159 mojo::MessagePipe message_pipe; | |
160 stashed_object->stashed_handles.push_back( | |
161 mojo::ScopedHandle::From(message_pipe.handle0.Pass())); | |
162 stashed_objects.push_back(stashed_object.Pass()); | |
163 stash_service_->AddToStash(stashed_objects.Pass()); | |
164 stashed_objects = RetrieveStash(); | |
165 ASSERT_EQ(0u, stashed_objects.size()); | |
166 // Check that the stashed handle has been closed. | |
167 MojoResult result = | |
168 mojo::Wait(message_pipe.handle1.get(), | |
169 MOJO_HANDLE_SIGNAL_READABLE | MOJO_HANDLE_SIGNAL_READABLE, | |
170 MOJO_DEADLINE_INDEFINITE); | |
171 EXPECT_EQ(MOJO_RESULT_FAILED_PRECONDITION, result); | |
172 } | |
173 | |
174 } // namespace extensions | |
OLD | NEW |