Index: apps/saved_devices_service_unittest.cc |
diff --git a/apps/saved_devices_service_unittest.cc b/apps/saved_devices_service_unittest.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..b2ed54451b006c6c7b0157cfea15ed172af3a983 |
--- /dev/null |
+++ b/apps/saved_devices_service_unittest.cc |
@@ -0,0 +1,213 @@ |
+// Copyright 2014 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#include "apps/saved_devices_service.h" |
+#include "base/run_loop.h" |
+#include "base/strings/utf_string_conversions.h" |
+#include "base/test/values_test_util.h" |
+#include "chrome/browser/extensions/test_extension_environment.h" |
+#include "chrome/test/base/testing_profile.h" |
+#include "device/usb/usb_device.h" |
+#include "device/usb/usb_device_handle.h" |
+#include "extensions/browser/extension_prefs.h" |
+#include "extensions/common/extension.h" |
+#include "testing/gmock/include/gmock/gmock.h" |
+#include "testing/gtest/include/gtest/gtest.h" |
+ |
+namespace apps { |
+ |
+namespace { |
+ |
+using device::UsbDevice; |
+using device::UsbDeviceHandle; |
+using device::UsbEndpointDirection; |
+using device::UsbTransferCallback; |
+using testing::Return; |
+ |
+class MockUsbDeviceHandle : public UsbDeviceHandle { |
+ public: |
+ MockUsbDeviceHandle(const std::string& serial_number) |
+ : UsbDeviceHandle(), serial_number_(serial_number) {} |
+ |
+ MOCK_CONST_METHOD0(GetDevice, scoped_refptr<UsbDevice>()); |
+ MOCK_METHOD0(Close, void()); |
+ |
+ MOCK_METHOD10(ControlTransfer, |
+ void(UsbEndpointDirection direction, |
+ TransferRequestType request_type, |
+ TransferRecipient recipient, |
+ uint8 request, |
+ uint16 value, |
+ uint16 index, |
+ net::IOBuffer* buffer, |
+ size_t length, |
+ unsigned int timeout, |
+ const UsbTransferCallback& callback)); |
+ |
+ MOCK_METHOD6(BulkTransfer, |
+ void(UsbEndpointDirection direction, |
+ uint8 endpoint, |
+ net::IOBuffer* buffer, |
+ size_t length, |
+ unsigned int timeout, |
+ const UsbTransferCallback& callback)); |
+ |
+ MOCK_METHOD6(InterruptTransfer, |
+ void(UsbEndpointDirection direction, |
+ uint8 endpoint, |
+ net::IOBuffer* buffer, |
+ size_t length, |
+ unsigned int timeout, |
+ const UsbTransferCallback& callback)); |
+ |
+ MOCK_METHOD8(IsochronousTransfer, |
+ void(UsbEndpointDirection direction, |
+ uint8 endpoint, |
+ net::IOBuffer* buffer, |
+ size_t length, |
+ unsigned int packets, |
+ unsigned int packet_length, |
+ unsigned int timeout, |
+ const UsbTransferCallback& callback)); |
+ |
+ MOCK_METHOD0(ResetDevice, bool()); |
+ MOCK_METHOD1(ClaimInterface, bool(int interface_number)); |
+ MOCK_METHOD1(ReleaseInterface, bool(int interface_number)); |
+ MOCK_METHOD2(SetInterfaceAlternateSetting, |
+ bool(int interface_number, int alternate_setting)); |
+ MOCK_METHOD1(GetManufacturer, bool(base::string16* manufacturer)); |
+ MOCK_METHOD1(GetProduct, bool(base::string16* product)); |
+ |
+ bool GetSerial(base::string16* serial) OVERRIDE { |
+ if (serial_number_.empty()) { |
+ return false; |
+ } |
+ |
+ *serial = base::UTF8ToUTF16(serial_number_); |
+ return true; |
+ } |
+ |
+ private: |
+ virtual ~MockUsbDeviceHandle() {} |
+ |
+ const std::string serial_number_; |
+}; |
+ |
+class MockUsbDevice : public UsbDevice { |
+ public: |
+ MockUsbDevice(const std::string& serial_number, uint32 unique_id) |
+ : UsbDevice(0, 0, unique_id), serial_number_(serial_number) {} |
+ |
+ MOCK_METHOD1(Close, bool(scoped_refptr<UsbDeviceHandle>)); |
+#if defined(OS_CHROMEOS) |
+ MOCK_METHOD2(RequestUsbAccess, void(int, const base::Callback<void(bool)>&)); |
+#endif |
+ MOCK_METHOD0(GetConfiguration, const device::UsbConfigDescriptor&()); |
+ |
+ scoped_refptr<UsbDeviceHandle> Open() OVERRIDE { |
+ return new MockUsbDeviceHandle(serial_number_); |
+ } |
+ |
+ void NotifyDisconnect() { UsbDevice::NotifyDisconnect(); } |
+ |
+ private: |
+ virtual ~MockUsbDevice() {} |
+ |
+ const std::string serial_number_; |
+}; |
+} |
+ |
+class SavedDevicesServiceTest : public testing::Test { |
+ protected: |
+ virtual void SetUp() OVERRIDE { |
+ testing::Test::SetUp(); |
+ env_.GetExtensionPrefs(); // Force creation before adding extensions. |
+ extension_ = env_.MakeExtension(*base::test::ParseJson( |
+ "{" |
+ " \"app\": {" |
+ " \"background\": {" |
+ " \"scripts\": [\"background.js\"]" |
+ " }" |
+ " }," |
+ " \"permissions\": [" |
+ " \"usb\"" |
+ " ]" |
+ "}")); |
+ service_ = SavedDevicesService::Get(env_.profile()); |
+ device0 = new MockUsbDevice("ABCDE", 0); |
+ device1 = new MockUsbDevice("", 1); |
+ device2 = new MockUsbDevice("12345", 2); |
+ device3 = new MockUsbDevice("", 3); |
+ } |
+ |
+ extensions::TestExtensionEnvironment env_; |
+ const extensions::Extension* extension_; |
+ SavedDevicesService* service_; |
+ scoped_refptr<MockUsbDevice> device0; |
+ scoped_refptr<MockUsbDevice> device1; |
+ scoped_refptr<MockUsbDevice> device2; |
+ scoped_refptr<MockUsbDevice> device3; |
+}; |
+ |
+TEST_F(SavedDevicesServiceTest, RegisterDevices) { |
+ SavedDevicesService::SavedDevices* saved_devices = |
+ service_->GetOrInsert(extension_->id()); |
+ |
+ base::string16 serial_number(base::ASCIIToUTF16("ABCDE")); |
+ saved_devices->RegisterDevice(device0, &serial_number); |
+ saved_devices->RegisterDevice(device1, NULL); |
+ |
+ // This is necessary as writing out registered devices happens in a task on |
+ // the UI thread. |
+ base::RunLoop run_loop; |
+ run_loop.RunUntilIdle(); |
+ |
+ ASSERT_TRUE(saved_devices->IsRegistered(device0)); |
+ ASSERT_TRUE(saved_devices->IsRegistered(device1)); |
+ ASSERT_FALSE(saved_devices->IsRegistered(device2)); |
+ ASSERT_FALSE(saved_devices->IsRegistered(device3)); |
+ |
+ std::vector<SavedDeviceEntry> device_entries = |
+ service_->GetAllDevices(extension_->id()); |
+ ASSERT_EQ(1U, device_entries.size()); |
+ ASSERT_EQ(base::ASCIIToUTF16("ABCDE"), device_entries[0].serial_number); |
+ |
+ device1->NotifyDisconnect(); |
+ |
+ ASSERT_TRUE(saved_devices->IsRegistered(device0)); |
+ ASSERT_FALSE(saved_devices->IsRegistered(device1)); |
+ ASSERT_FALSE(saved_devices->IsRegistered(device2)); |
+ ASSERT_FALSE(saved_devices->IsRegistered(device3)); |
+ |
+ service_->Clear(extension_->id()); |
+ |
+ // App is normally restarted, clearing its reference to the SavedDevices. |
+ saved_devices = service_->GetOrInsert(extension_->id()); |
+ ASSERT_FALSE(saved_devices->IsRegistered(device0)); |
+ device_entries = service_->GetAllDevices(extension_->id()); |
+ ASSERT_EQ(0U, device_entries.size()); |
+} |
+ |
+TEST_F(SavedDevicesServiceTest, LoadPrefs) { |
+ scoped_ptr<base::Value> prefs_value = base::test::ParseJson( |
+ "[" |
+ " {" |
+ " \"product_id\": 0," |
+ " \"serial_number\": \"ABCDE\"," |
+ " \"type\": \"usb\"," |
+ " \"vendor_id\": 0" |
+ " }" |
+ "]"); |
+ env_.GetExtensionPrefs()->UpdateExtensionPref( |
+ extension_->id(), "devices", prefs_value.release()); |
+ |
+ SavedDevicesService::SavedDevices* saved_devices = |
+ service_->GetOrInsert(extension_->id()); |
+ ASSERT_TRUE(saved_devices->IsRegistered(device0)); |
+ ASSERT_FALSE(saved_devices->IsRegistered(device1)); |
+ ASSERT_FALSE(saved_devices->IsRegistered(device2)); |
+ ASSERT_FALSE(saved_devices->IsRegistered(device3)); |
+} |
+ |
+} // namespace apps |