| OLD | NEW |
| (Empty) |
| 1 // Copyright 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 "content/child/indexed_db/indexed_db_dispatcher.h" | |
| 6 | |
| 7 #include <stddef.h> | |
| 8 #include <stdint.h> | |
| 9 | |
| 10 #include <memory> | |
| 11 | |
| 12 #include "base/macros.h" | |
| 13 #include "base/single_thread_task_runner.h" | |
| 14 #include "base/threading/thread_task_runner_handle.h" | |
| 15 #include "base/values.h" | |
| 16 #include "content/child/indexed_db/mock_webidbcallbacks.h" | |
| 17 #include "content/child/indexed_db/webidbcursor_impl.h" | |
| 18 #include "content/child/thread_safe_sender.h" | |
| 19 #include "content/common/indexed_db/indexed_db_key.h" | |
| 20 #include "content/common/indexed_db/indexed_db_key_range.h" | |
| 21 #include "content/common/indexed_db/indexed_db_messages.h" | |
| 22 #include "ipc/ipc_sync_message_filter.h" | |
| 23 #include "testing/gtest/include/gtest/gtest.h" | |
| 24 #include "third_party/WebKit/public/platform/WebBlobInfo.h" | |
| 25 #include "third_party/WebKit/public/platform/WebData.h" | |
| 26 #include "third_party/WebKit/public/web/WebHeap.h" | |
| 27 | |
| 28 using blink::WebBlobInfo; | |
| 29 using blink::WebData; | |
| 30 using blink::WebIDBCursor; | |
| 31 using blink::WebIDBKey; | |
| 32 using blink::WebVector; | |
| 33 using testing::_; | |
| 34 using testing::Invoke; | |
| 35 using testing::StrictMock; | |
| 36 using testing::WithArgs; | |
| 37 | |
| 38 namespace content { | |
| 39 namespace { | |
| 40 | |
| 41 class MockDispatcher : public IndexedDBDispatcher { | |
| 42 public: | |
| 43 explicit MockDispatcher(ThreadSafeSender* sender) | |
| 44 : IndexedDBDispatcher(sender) {} | |
| 45 | |
| 46 bool Send(IPC::Message* msg) override { | |
| 47 delete msg; | |
| 48 return true; | |
| 49 } | |
| 50 | |
| 51 private: | |
| 52 DISALLOW_COPY_AND_ASSIGN(MockDispatcher); | |
| 53 }; | |
| 54 | |
| 55 class MockSyncMessageFilter : public IPC::SyncMessageFilter { | |
| 56 public: | |
| 57 MockSyncMessageFilter() | |
| 58 : SyncMessageFilter(nullptr, false /* is_channel_send_thread_safe */) {} | |
| 59 | |
| 60 private: | |
| 61 ~MockSyncMessageFilter() override {} | |
| 62 }; | |
| 63 | |
| 64 } // namespace | |
| 65 | |
| 66 class IndexedDBDispatcherTest : public testing::Test { | |
| 67 public: | |
| 68 IndexedDBDispatcherTest() | |
| 69 : thread_safe_sender_(new ThreadSafeSender( | |
| 70 base::ThreadTaskRunnerHandle::Get(), new MockSyncMessageFilter)) {} | |
| 71 | |
| 72 void TearDown() override { blink::WebHeap::collectAllGarbageForTesting(); } | |
| 73 | |
| 74 protected: | |
| 75 base::MessageLoop message_loop_; | |
| 76 scoped_refptr<ThreadSafeSender> thread_safe_sender_; | |
| 77 | |
| 78 private: | |
| 79 DISALLOW_COPY_AND_ASSIGN(IndexedDBDispatcherTest); | |
| 80 }; | |
| 81 | |
| 82 TEST_F(IndexedDBDispatcherTest, ValueSizeTest) { | |
| 83 // For testing use a much smaller maximum size to prevent allocating >100 MB | |
| 84 // of memory, which crashes on memory-constrained systems. | |
| 85 const size_t kMaxValueSizeForTesting = 10 * 1024 * 1024; // 10 MB | |
| 86 | |
| 87 const std::vector<char> data(kMaxValueSizeForTesting + 1); | |
| 88 const WebData value(&data.front(), data.size()); | |
| 89 const WebVector<WebBlobInfo> web_blob_info; | |
| 90 const int32_t ipc_dummy_id = -1; | |
| 91 const int64_t transaction_id = 1; | |
| 92 const int64_t object_store_id = 2; | |
| 93 | |
| 94 StrictMock<MockWebIDBCallbacks> callbacks; | |
| 95 EXPECT_CALL(callbacks, onError(_)).Times(1); | |
| 96 | |
| 97 IndexedDBDispatcher dispatcher(thread_safe_sender_.get()); | |
| 98 dispatcher.max_put_value_size_ = kMaxValueSizeForTesting; | |
| 99 IndexedDBKey key(0, blink::WebIDBKeyTypeNumber); | |
| 100 dispatcher.RequestIDBDatabasePut(ipc_dummy_id, | |
| 101 transaction_id, | |
| 102 object_store_id, | |
| 103 value, | |
| 104 web_blob_info, | |
| 105 key, | |
| 106 blink::WebIDBPutModeAddOrUpdate, | |
| 107 &callbacks, | |
| 108 WebVector<long long>(), | |
| 109 WebVector<WebVector<WebIDBKey> >()); | |
| 110 } | |
| 111 | |
| 112 TEST_F(IndexedDBDispatcherTest, KeyAndValueSizeTest) { | |
| 113 // For testing use a much smaller maximum size to prevent allocating >100 MB | |
| 114 // of memory, which crashes on memory-constrained systems. | |
| 115 const size_t kMaxValueSizeForTesting = 10 * 1024 * 1024; // 10 MB | |
| 116 const size_t kKeySize = 1024 * 1024; | |
| 117 | |
| 118 const std::vector<char> data(kMaxValueSizeForTesting - kKeySize); | |
| 119 const WebData value(&data.front(), data.size()); | |
| 120 const WebVector<WebBlobInfo> web_blob_info; | |
| 121 const IndexedDBKey key( | |
| 122 base::string16(kKeySize / sizeof(base::string16::value_type), 'x')); | |
| 123 | |
| 124 const int32_t ipc_dummy_id = -1; | |
| 125 const int64_t transaction_id = 1; | |
| 126 const int64_t object_store_id = 2; | |
| 127 | |
| 128 StrictMock<MockWebIDBCallbacks> callbacks; | |
| 129 EXPECT_CALL(callbacks, onError(_)).Times(1); | |
| 130 | |
| 131 IndexedDBDispatcher dispatcher(thread_safe_sender_.get()); | |
| 132 dispatcher.max_put_value_size_ = kMaxValueSizeForTesting; | |
| 133 dispatcher.RequestIDBDatabasePut(ipc_dummy_id, | |
| 134 transaction_id, | |
| 135 object_store_id, | |
| 136 value, | |
| 137 web_blob_info, | |
| 138 key, | |
| 139 blink::WebIDBPutModeAddOrUpdate, | |
| 140 &callbacks, | |
| 141 WebVector<long long>(), | |
| 142 WebVector<WebVector<WebIDBKey> >()); | |
| 143 } | |
| 144 | |
| 145 TEST_F(IndexedDBDispatcherTest, CursorTransactionId) { | |
| 146 const int32_t ipc_database_id = -1; | |
| 147 const int64_t transaction_id = 1234; | |
| 148 const int64_t object_store_id = 2; | |
| 149 const int32_t index_id = 3; | |
| 150 const blink::WebIDBCursorDirection direction = | |
| 151 blink::WebIDBCursorDirectionNext; | |
| 152 const bool key_only = false; | |
| 153 | |
| 154 MockDispatcher dispatcher(thread_safe_sender_.get()); | |
| 155 | |
| 156 // First case: successful cursor open. | |
| 157 { | |
| 158 std::unique_ptr<WebIDBCursor> cursor; | |
| 159 EXPECT_EQ(0UL, dispatcher.cursor_transaction_ids_.size()); | |
| 160 | |
| 161 auto* callbacks = new StrictMock<MockWebIDBCallbacks>(); | |
| 162 // Reference first param (cursor) to keep it alive. | |
| 163 // TODO(cmumford): Cleanup (and below) once std::addressof() is allowed. | |
| 164 ON_CALL(*callbacks, onSuccess(testing::A<WebIDBCursor*>(), _, _, _)) | |
| 165 .WillByDefault( | |
| 166 WithArgs<0>(Invoke(&cursor.operator=(nullptr), | |
| 167 &std::unique_ptr<WebIDBCursor>::reset))); | |
| 168 EXPECT_CALL(*callbacks, onSuccess(testing::A<WebIDBCursor*>(), _, _, _)) | |
| 169 .Times(1); | |
| 170 | |
| 171 // Make a cursor request. This should record the transaction id. | |
| 172 dispatcher.RequestIDBDatabaseOpenCursor( | |
| 173 ipc_database_id, transaction_id, object_store_id, index_id, | |
| 174 IndexedDBKeyRange(), direction, key_only, blink::WebIDBTaskTypeNormal, | |
| 175 callbacks); | |
| 176 | |
| 177 // Verify that the transaction id was captured. | |
| 178 EXPECT_EQ(1UL, dispatcher.cursor_transaction_ids_.size()); | |
| 179 EXPECT_FALSE(cursor.get()); | |
| 180 | |
| 181 int32_t ipc_callbacks_id = | |
| 182 dispatcher.cursor_transaction_ids_.begin()->first; | |
| 183 | |
| 184 IndexedDBMsg_CallbacksSuccessIDBCursor_Params params; | |
| 185 params.ipc_thread_id = dispatcher.CurrentWorkerId(); | |
| 186 params.ipc_callbacks_id = ipc_callbacks_id; | |
| 187 | |
| 188 // Now simululate the cursor response. | |
| 189 params.ipc_cursor_id = WebIDBCursorImpl::kInvalidCursorId; | |
| 190 dispatcher.OnSuccessOpenCursor(params); | |
| 191 | |
| 192 EXPECT_EQ(0UL, dispatcher.cursor_transaction_ids_.size()); | |
| 193 | |
| 194 EXPECT_TRUE(cursor.get()); | |
| 195 | |
| 196 WebIDBCursorImpl* impl = static_cast<WebIDBCursorImpl*>(cursor.get()); | |
| 197 | |
| 198 // This is the primary expectation of this test: the transaction id was | |
| 199 // applied to the cursor. | |
| 200 EXPECT_EQ(transaction_id, impl->transaction_id()); | |
| 201 } | |
| 202 | |
| 203 // Second case: null cursor (no data in range) | |
| 204 { | |
| 205 EXPECT_EQ(0UL, dispatcher.cursor_transaction_ids_.size()); | |
| 206 | |
| 207 auto* callbacks = new StrictMock<MockWebIDBCallbacks>(); | |
| 208 EXPECT_CALL(*callbacks, onSuccess(testing::A<const blink::WebIDBValue&>())) | |
| 209 .Times(1); | |
| 210 | |
| 211 // Make a cursor request. This should record the transaction id. | |
| 212 dispatcher.RequestIDBDatabaseOpenCursor( | |
| 213 ipc_database_id, transaction_id, object_store_id, index_id, | |
| 214 IndexedDBKeyRange(), direction, key_only, blink::WebIDBTaskTypeNormal, | |
| 215 callbacks); | |
| 216 | |
| 217 // Verify that the transaction id was captured. | |
| 218 EXPECT_EQ(1UL, dispatcher.cursor_transaction_ids_.size()); | |
| 219 | |
| 220 int32_t ipc_callbacks_id = | |
| 221 dispatcher.cursor_transaction_ids_.begin()->first; | |
| 222 | |
| 223 // Now simululate a "null cursor" response. | |
| 224 IndexedDBMsg_CallbacksSuccessValue_Params params; | |
| 225 params.ipc_thread_id = dispatcher.CurrentWorkerId(); | |
| 226 params.ipc_callbacks_id = ipc_callbacks_id; | |
| 227 dispatcher.OnSuccessValue(params); | |
| 228 | |
| 229 // Ensure the map result was deleted. | |
| 230 EXPECT_EQ(0UL, dispatcher.cursor_transaction_ids_.size()); | |
| 231 } | |
| 232 } | |
| 233 | |
| 234 namespace { | |
| 235 | |
| 236 class MockCursor : public WebIDBCursorImpl { | |
| 237 public: | |
| 238 MockCursor(int32_t ipc_cursor_id, | |
| 239 int64_t transaction_id, | |
| 240 ThreadSafeSender* thread_safe_sender) | |
| 241 : WebIDBCursorImpl(ipc_cursor_id, transaction_id, thread_safe_sender), | |
| 242 reset_count_(0) {} | |
| 243 | |
| 244 // This method is virtual so it can be overridden in unit tests. | |
| 245 void ResetPrefetchCache() override { ++reset_count_; } | |
| 246 | |
| 247 int reset_count() const { return reset_count_; } | |
| 248 | |
| 249 private: | |
| 250 int reset_count_; | |
| 251 | |
| 252 DISALLOW_COPY_AND_ASSIGN(MockCursor); | |
| 253 }; | |
| 254 | |
| 255 } // namespace | |
| 256 | |
| 257 TEST_F(IndexedDBDispatcherTest, CursorReset) { | |
| 258 std::unique_ptr<WebIDBCursor> cursor; | |
| 259 MockDispatcher dispatcher(thread_safe_sender_.get()); | |
| 260 | |
| 261 const int32_t ipc_database_id = 0; | |
| 262 const int32_t object_store_id = 0; | |
| 263 const int32_t index_id = 0; | |
| 264 const bool key_only = false; | |
| 265 const int cursor1_ipc_id = 1; | |
| 266 const int cursor2_ipc_id = 2; | |
| 267 const int other_cursor_ipc_id = 2; | |
| 268 const int cursor1_transaction_id = 1; | |
| 269 const int cursor2_transaction_id = 2; | |
| 270 const int other_transaction_id = 3; | |
| 271 | |
| 272 std::unique_ptr<MockCursor> cursor1( | |
| 273 new MockCursor(WebIDBCursorImpl::kInvalidCursorId, cursor1_transaction_id, | |
| 274 thread_safe_sender_.get())); | |
| 275 | |
| 276 std::unique_ptr<MockCursor> cursor2( | |
| 277 new MockCursor(WebIDBCursorImpl::kInvalidCursorId, cursor2_transaction_id, | |
| 278 thread_safe_sender_.get())); | |
| 279 | |
| 280 dispatcher.cursors_[cursor1_ipc_id] = cursor1.get(); | |
| 281 dispatcher.cursors_[cursor2_ipc_id] = cursor2.get(); | |
| 282 | |
| 283 EXPECT_EQ(0, cursor1->reset_count()); | |
| 284 EXPECT_EQ(0, cursor2->reset_count()); | |
| 285 | |
| 286 // Other transaction: | |
| 287 dispatcher.RequestIDBDatabaseGet( | |
| 288 ipc_database_id, other_transaction_id, object_store_id, index_id, | |
| 289 IndexedDBKeyRange(), key_only, new StrictMock<MockWebIDBCallbacks>()); | |
| 290 | |
| 291 EXPECT_EQ(0, cursor1->reset_count()); | |
| 292 EXPECT_EQ(0, cursor2->reset_count()); | |
| 293 | |
| 294 // Same transaction: | |
| 295 dispatcher.RequestIDBDatabaseGet( | |
| 296 ipc_database_id, cursor1_transaction_id, object_store_id, index_id, | |
| 297 IndexedDBKeyRange(), key_only, new StrictMock<MockWebIDBCallbacks>()); | |
| 298 | |
| 299 EXPECT_EQ(1, cursor1->reset_count()); | |
| 300 EXPECT_EQ(0, cursor2->reset_count()); | |
| 301 | |
| 302 // Same transaction and same cursor: | |
| 303 dispatcher.RequestIDBCursorContinue(IndexedDBKey(), IndexedDBKey(), | |
| 304 new StrictMock<MockWebIDBCallbacks>(), | |
| 305 cursor1_ipc_id, cursor1_transaction_id); | |
| 306 | |
| 307 EXPECT_EQ(1, cursor1->reset_count()); | |
| 308 EXPECT_EQ(0, cursor2->reset_count()); | |
| 309 | |
| 310 // Same transaction and different cursor: | |
| 311 dispatcher.RequestIDBCursorContinue( | |
| 312 IndexedDBKey(), IndexedDBKey(), new StrictMock<MockWebIDBCallbacks>(), | |
| 313 other_cursor_ipc_id, cursor1_transaction_id); | |
| 314 | |
| 315 EXPECT_EQ(2, cursor1->reset_count()); | |
| 316 EXPECT_EQ(0, cursor2->reset_count()); | |
| 317 | |
| 318 cursor1.reset(); | |
| 319 cursor2.reset(); | |
| 320 } | |
| 321 | |
| 322 } // namespace content | |
| OLD | NEW |