| OLD | NEW |
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "apps/saved_devices_service.h" | |
| 6 #include "base/run_loop.h" | 5 #include "base/run_loop.h" |
| 7 #include "base/strings/utf_string_conversions.h" | 6 #include "base/strings/utf_string_conversions.h" |
| 8 #include "base/test/values_test_util.h" | 7 #include "base/test/values_test_util.h" |
| 9 #include "chrome/browser/extensions/test_extension_environment.h" | 8 #include "chrome/browser/extensions/test_extension_environment.h" |
| 10 #include "chrome/test/base/testing_profile.h" | 9 #include "chrome/test/base/testing_profile.h" |
| 11 #include "device/usb/usb_device.h" | 10 #include "device/usb/usb_device.h" |
| 12 #include "device/usb/usb_device_handle.h" | 11 #include "device/usb/usb_device_handle.h" |
| 12 #include "extensions/browser/api/device_permissions_manager.h" |
| 13 #include "extensions/browser/extension_prefs.h" | 13 #include "extensions/browser/extension_prefs.h" |
| 14 #include "extensions/common/extension.h" | 14 #include "extensions/common/extension.h" |
| 15 #include "testing/gmock/include/gmock/gmock.h" | 15 #include "testing/gmock/include/gmock/gmock.h" |
| 16 #include "testing/gtest/include/gtest/gtest.h" | 16 #include "testing/gtest/include/gtest/gtest.h" |
| 17 | 17 |
| 18 namespace apps { | 18 namespace extensions { |
| 19 | 19 |
| 20 namespace { | 20 namespace { |
| 21 | 21 |
| 22 using device::UsbDevice; | 22 using device::UsbDevice; |
| 23 using device::UsbDeviceHandle; | 23 using device::UsbDeviceHandle; |
| 24 using device::UsbEndpointDirection; | |
| 25 using device::UsbTransferCallback; | |
| 26 using testing::Return; | 24 using testing::Return; |
| 27 | 25 |
| 28 class MockUsbDevice : public UsbDevice { | 26 class MockUsbDevice : public UsbDevice { |
| 29 public: | 27 public: |
| 30 MockUsbDevice(const std::string& serial_number, uint32 unique_id) | 28 MockUsbDevice(const std::string& serial_number, uint32 unique_id) |
| 31 : UsbDevice(0, 0, unique_id), serial_number_(serial_number) {} | 29 : UsbDevice(0, 0, unique_id), serial_number_(serial_number) {} |
| 32 | 30 |
| 33 MOCK_METHOD0(Open, scoped_refptr<UsbDeviceHandle>()); | 31 MOCK_METHOD0(Open, scoped_refptr<UsbDeviceHandle>()); |
| 34 MOCK_METHOD1(Close, bool(scoped_refptr<UsbDeviceHandle>)); | 32 MOCK_METHOD1(Close, bool(scoped_refptr<UsbDeviceHandle>)); |
| 35 #if defined(OS_CHROMEOS) | 33 #if defined(OS_CHROMEOS) |
| 36 MOCK_METHOD2(RequestUsbAccess, void(int, const base::Callback<void(bool)>&)); | 34 MOCK_METHOD2(RequestUsbAccess, void(int, const base::Callback<void(bool)>&)); |
| 37 #endif | 35 #endif |
| 38 MOCK_METHOD0(GetConfiguration, const device::UsbConfigDescriptor&()); | 36 MOCK_METHOD0(GetConfiguration, const device::UsbConfigDescriptor&()); |
| 39 MOCK_METHOD1(GetManufacturer, bool(base::string16*)); | 37 MOCK_METHOD1(GetManufacturer, bool(base::string16*)); |
| 40 MOCK_METHOD1(GetProduct, bool(base::string16*)); | 38 MOCK_METHOD1(GetProduct, bool(base::string16*)); |
| 41 | 39 |
| 42 bool GetSerialNumber(base::string16* serial) OVERRIDE { | 40 virtual bool GetSerialNumber(base::string16* serial_number) OVERRIDE { |
| 43 if (serial_number_.empty()) { | 41 if (serial_number_.empty()) { |
| 44 return false; | 42 return false; |
| 45 } | 43 } |
| 46 | 44 |
| 47 *serial = base::UTF8ToUTF16(serial_number_); | 45 *serial_number = base::UTF8ToUTF16(serial_number_); |
| 48 return true; | 46 return true; |
| 49 } | 47 } |
| 50 | 48 |
| 51 void NotifyDisconnect() { UsbDevice::NotifyDisconnect(); } | 49 void NotifyDisconnect() { UsbDevice::NotifyDisconnect(); } |
| 52 | 50 |
| 53 private: | 51 private: |
| 54 virtual ~MockUsbDevice() {} | 52 virtual ~MockUsbDevice() {} |
| 55 | 53 |
| 56 const std::string serial_number_; | 54 const std::string serial_number_; |
| 57 }; | 55 }; |
| 58 } | |
| 59 | 56 |
| 60 class SavedDevicesServiceTest : public testing::Test { | 57 } // namespace |
| 58 |
| 59 class DevicePermissionsManagerTest : public testing::Test { |
| 61 protected: | 60 protected: |
| 62 virtual void SetUp() OVERRIDE { | 61 virtual void SetUp() OVERRIDE { |
| 63 testing::Test::SetUp(); | 62 testing::Test::SetUp(); |
| 64 env_.GetExtensionPrefs(); // Force creation before adding extensions. | 63 env_.GetExtensionPrefs(); // Force creation before adding extensions. |
| 65 extension_ = env_.MakeExtension(*base::test::ParseJson( | 64 extension_ = env_.MakeExtension(*base::test::ParseJson( |
| 66 "{" | 65 "{" |
| 67 " \"app\": {" | 66 " \"app\": {" |
| 68 " \"background\": {" | 67 " \"background\": {" |
| 69 " \"scripts\": [\"background.js\"]" | 68 " \"scripts\": [\"background.js\"]" |
| 70 " }" | 69 " }" |
| 71 " }," | 70 " }," |
| 72 " \"permissions\": [" | 71 " \"permissions\": [" |
| 73 " \"usb\"" | 72 " \"usb\"" |
| 74 " ]" | 73 " ]" |
| 75 "}")); | 74 "}")); |
| 76 service_ = SavedDevicesService::Get(env_.profile()); | |
| 77 device0 = new MockUsbDevice("ABCDE", 0); | 75 device0 = new MockUsbDevice("ABCDE", 0); |
| 78 device1 = new MockUsbDevice("", 1); | 76 device1 = new MockUsbDevice("", 1); |
| 79 device2 = new MockUsbDevice("12345", 2); | 77 device2 = new MockUsbDevice("12345", 2); |
| 80 device3 = new MockUsbDevice("", 3); | 78 device3 = new MockUsbDevice("", 3); |
| 81 } | 79 } |
| 82 | 80 |
| 83 extensions::TestExtensionEnvironment env_; | 81 extensions::TestExtensionEnvironment env_; |
| 84 const extensions::Extension* extension_; | 82 const extensions::Extension* extension_; |
| 85 SavedDevicesService* service_; | |
| 86 scoped_refptr<MockUsbDevice> device0; | 83 scoped_refptr<MockUsbDevice> device0; |
| 87 scoped_refptr<MockUsbDevice> device1; | 84 scoped_refptr<MockUsbDevice> device1; |
| 88 scoped_refptr<MockUsbDevice> device2; | 85 scoped_refptr<MockUsbDevice> device2; |
| 89 scoped_refptr<MockUsbDevice> device3; | 86 scoped_refptr<MockUsbDevice> device3; |
| 90 }; | 87 }; |
| 91 | 88 |
| 92 TEST_F(SavedDevicesServiceTest, RegisterDevices) { | 89 TEST_F(DevicePermissionsManagerTest, RegisterDevices) { |
| 93 SavedDevicesService::SavedDevices* saved_devices = | 90 DevicePermissionsManager* manager = |
| 94 service_->GetOrInsert(extension_->id()); | 91 DevicePermissionsManager::Get(env_.profile()); |
| 92 manager->AllowUsbDevice( |
| 93 extension_->id(), device0, base::ASCIIToUTF16("ABCDE")); |
| 94 manager->AllowUsbDevice(extension_->id(), device1, base::string16()); |
| 95 | 95 |
| 96 base::string16 serial_number(base::ASCIIToUTF16("ABCDE")); | 96 scoped_ptr<DevicePermissions> device_permissions = |
| 97 saved_devices->RegisterDevice(device0, &serial_number); | 97 manager->GetForExtension(extension_->id()); |
| 98 saved_devices->RegisterDevice(device1, NULL); | 98 ASSERT_TRUE(device_permissions->CheckUsbDevice(device0)); |
| 99 ASSERT_TRUE(device_permissions->CheckUsbDevice(device1)); |
| 100 ASSERT_FALSE(device_permissions->CheckUsbDevice(device2)); |
| 101 ASSERT_FALSE(device_permissions->CheckUsbDevice(device3)); |
| 99 | 102 |
| 100 // This is necessary as writing out registered devices happens in a task on | 103 std::vector<base::string16> device_messages = |
| 101 // the UI thread. | 104 manager->GetPermissionMessageStrings(extension_->id()); |
| 102 base::RunLoop run_loop; | 105 ASSERT_EQ(1U, device_messages.size()); |
| 103 run_loop.RunUntilIdle(); | 106 ASSERT_NE(device_messages[0].find(base::ASCIIToUTF16("ABCDE")), |
| 104 | 107 base::string16::npos); |
| 105 ASSERT_TRUE(saved_devices->IsRegistered(device0)); | |
| 106 ASSERT_TRUE(saved_devices->IsRegistered(device1)); | |
| 107 ASSERT_FALSE(saved_devices->IsRegistered(device2)); | |
| 108 ASSERT_FALSE(saved_devices->IsRegistered(device3)); | |
| 109 | |
| 110 std::vector<SavedDeviceEntry> device_entries = | |
| 111 service_->GetAllDevices(extension_->id()); | |
| 112 ASSERT_EQ(1U, device_entries.size()); | |
| 113 ASSERT_EQ(base::ASCIIToUTF16("ABCDE"), device_entries[0].serial_number); | |
| 114 | 108 |
| 115 device1->NotifyDisconnect(); | 109 device1->NotifyDisconnect(); |
| 116 | 110 |
| 117 ASSERT_TRUE(saved_devices->IsRegistered(device0)); | 111 device_permissions = manager->GetForExtension(extension_->id()); |
| 118 ASSERT_FALSE(saved_devices->IsRegistered(device1)); | 112 ASSERT_TRUE(device_permissions->CheckUsbDevice(device0)); |
| 119 ASSERT_FALSE(saved_devices->IsRegistered(device2)); | 113 ASSERT_FALSE(device_permissions->CheckUsbDevice(device1)); |
| 120 ASSERT_FALSE(saved_devices->IsRegistered(device3)); | 114 ASSERT_FALSE(device_permissions->CheckUsbDevice(device2)); |
| 115 ASSERT_FALSE(device_permissions->CheckUsbDevice(device3)); |
| 121 | 116 |
| 122 service_->Clear(extension_->id()); | 117 manager->Clear(extension_->id()); |
| 123 | 118 |
| 124 // App is normally restarted, clearing its reference to the SavedDevices. | 119 device_permissions = manager->GetForExtension(extension_->id()); |
| 125 saved_devices = service_->GetOrInsert(extension_->id()); | 120 ASSERT_FALSE(device_permissions->CheckUsbDevice(device0)); |
| 126 ASSERT_FALSE(saved_devices->IsRegistered(device0)); | 121 device_messages = manager->GetPermissionMessageStrings(extension_->id()); |
| 127 device_entries = service_->GetAllDevices(extension_->id()); | 122 ASSERT_EQ(0U, device_messages.size()); |
| 128 ASSERT_EQ(0U, device_entries.size()); | |
| 129 } | 123 } |
| 130 | 124 |
| 131 TEST_F(SavedDevicesServiceTest, LoadPrefs) { | 125 TEST_F(DevicePermissionsManagerTest, LoadPrefs) { |
| 132 scoped_ptr<base::Value> prefs_value = base::test::ParseJson( | 126 scoped_ptr<base::Value> prefs_value = base::test::ParseJson( |
| 133 "[" | 127 "[" |
| 134 " {" | 128 " {" |
| 135 " \"product_id\": 0," | 129 " \"product_id\": 0," |
| 136 " \"serial_number\": \"ABCDE\"," | 130 " \"serial_number\": \"ABCDE\"," |
| 137 " \"type\": \"usb\"," | 131 " \"type\": \"usb\"," |
| 138 " \"vendor_id\": 0" | 132 " \"vendor_id\": 0" |
| 139 " }" | 133 " }" |
| 140 "]"); | 134 "]"); |
| 141 env_.GetExtensionPrefs()->UpdateExtensionPref( | 135 env_.GetExtensionPrefs()->UpdateExtensionPref( |
| 142 extension_->id(), "devices", prefs_value.release()); | 136 extension_->id(), "devices", prefs_value.release()); |
| 143 | 137 |
| 144 SavedDevicesService::SavedDevices* saved_devices = | 138 DevicePermissionsManager* manager = |
| 145 service_->GetOrInsert(extension_->id()); | 139 DevicePermissionsManager::Get(env_.profile()); |
| 146 ASSERT_TRUE(saved_devices->IsRegistered(device0)); | 140 scoped_ptr<DevicePermissions> device_permissions = |
| 147 ASSERT_FALSE(saved_devices->IsRegistered(device1)); | 141 manager->GetForExtension(extension_->id()); |
| 148 ASSERT_FALSE(saved_devices->IsRegistered(device2)); | 142 ASSERT_TRUE(device_permissions->CheckUsbDevice(device0)); |
| 149 ASSERT_FALSE(saved_devices->IsRegistered(device3)); | 143 ASSERT_FALSE(device_permissions->CheckUsbDevice(device1)); |
| 144 ASSERT_FALSE(device_permissions->CheckUsbDevice(device2)); |
| 145 ASSERT_FALSE(device_permissions->CheckUsbDevice(device3)); |
| 150 } | 146 } |
| 151 | 147 |
| 152 } // namespace apps | 148 } // namespace extensions |
| OLD | NEW |