| 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 "extensions/browser/api/device_permissions_prompt.h" | 5 #include "extensions/browser/api/device_permissions_prompt.h" |
| 6 | 6 |
| 7 #include "base/barrier_closure.h" | 7 #include "base/scoped_observer.h" |
| 8 #include "base/strings/stringprintf.h" | 8 #include "base/strings/stringprintf.h" |
| 9 #include "base/strings/utf_string_conversions.h" | 9 #include "base/strings/utf_string_conversions.h" |
| 10 #include "content/public/browser/browser_thread.h" | |
| 11 #include "device/core/device_client.h" | 10 #include "device/core/device_client.h" |
| 12 #include "device/usb/usb_device.h" | 11 #include "device/usb/usb_device.h" |
| 13 #include "device/usb/usb_device_filter.h" | 12 #include "device/usb/usb_device_filter.h" |
| 14 #include "device/usb/usb_ids.h" | 13 #include "device/usb/usb_ids.h" |
| 15 #include "device/usb/usb_service.h" | 14 #include "device/usb/usb_service.h" |
| 16 #include "extensions/browser/api/device_permissions_manager.h" | 15 #include "extensions/browser/api/device_permissions_manager.h" |
| 17 #include "extensions/common/extension.h" | 16 #include "extensions/common/extension.h" |
| 18 #include "extensions/strings/grit/extensions_strings.h" | 17 #include "extensions/strings/grit/extensions_strings.h" |
| 19 #include "ui/base/l10n/l10n_util.h" | 18 #include "ui/base/l10n/l10n_util.h" |
| 20 | 19 |
| 21 using content::BrowserThread; | |
| 22 using device::UsbDevice; | 20 using device::UsbDevice; |
| 23 using device::UsbDeviceFilter; | 21 using device::UsbDeviceFilter; |
| 24 using device::UsbService; | 22 using device::UsbService; |
| 25 | 23 |
| 26 namespace extensions { | 24 namespace extensions { |
| 27 | 25 |
| 28 DevicePermissionsPrompt::Delegate::~Delegate() { | 26 namespace { |
| 29 } | |
| 30 | 27 |
| 31 DevicePermissionsPrompt::Prompt::DeviceInfo::DeviceInfo( | 28 class UsbDeviceInfo : public DevicePermissionsPrompt::Prompt::DeviceInfo { |
| 32 scoped_refptr<UsbDevice> device) | 29 public: |
| 33 : device(device) { | 30 UsbDeviceInfo(scoped_refptr<UsbDevice> device) : device_(device) { |
| 34 base::string16 manufacturer_string = device->manufacturer_string(); | 31 base::string16 manufacturer_string = device->manufacturer_string(); |
| 35 if (manufacturer_string.empty()) { | 32 if (manufacturer_string.empty()) { |
| 36 const char* vendor_name = | 33 const char* vendor_name = |
| 37 device::UsbIds::GetVendorName(device->vendor_id()); | 34 device::UsbIds::GetVendorName(device->vendor_id()); |
| 38 if (vendor_name) { | 35 if (vendor_name) { |
| 39 manufacturer_string = base::UTF8ToUTF16(vendor_name); | 36 manufacturer_string = base::UTF8ToUTF16(vendor_name); |
| 40 } else { | 37 } else { |
| 41 base::string16 vendor_id = | 38 base::string16 vendor_id = base::ASCIIToUTF16( |
| 42 base::ASCIIToUTF16(base::StringPrintf("0x%04x", device->vendor_id())); | 39 base::StringPrintf("0x%04x", device->vendor_id())); |
| 43 manufacturer_string = | 40 manufacturer_string = |
| 44 l10n_util::GetStringFUTF16(IDS_DEVICE_UNKNOWN_VENDOR, vendor_id); | 41 l10n_util::GetStringFUTF16(IDS_DEVICE_UNKNOWN_VENDOR, vendor_id); |
| 42 } |
| 43 } |
| 44 |
| 45 base::string16 product_string = device->product_string(); |
| 46 if (product_string.empty()) { |
| 47 const char* product_name = device::UsbIds::GetProductName( |
| 48 device->vendor_id(), device->product_id()); |
| 49 if (product_name) { |
| 50 product_string = base::UTF8ToUTF16(product_name); |
| 51 } else { |
| 52 base::string16 product_id = base::ASCIIToUTF16( |
| 53 base::StringPrintf("0x%04x", device->product_id())); |
| 54 product_string = |
| 55 l10n_util::GetStringFUTF16(IDS_DEVICE_UNKNOWN_PRODUCT, product_id); |
| 56 } |
| 57 } |
| 58 |
| 59 name_ = l10n_util::GetStringFUTF16(IDS_DEVICE_PERMISSIONS_DEVICE_NAME, |
| 60 product_string, manufacturer_string); |
| 61 serial_number_ = device->serial_number(); |
| 62 } |
| 63 |
| 64 ~UsbDeviceInfo() override {} |
| 65 |
| 66 const scoped_refptr<UsbDevice>& device() const { return device_; } |
| 67 |
| 68 private: |
| 69 // TODO(reillyg): Convert this to a weak reference when UsbDevice has a |
| 70 // connected flag. |
| 71 scoped_refptr<UsbDevice> device_; |
| 72 }; |
| 73 |
| 74 class UsbDevicePermissionsPrompt : public DevicePermissionsPrompt::Prompt, |
| 75 public device::UsbService::Observer { |
| 76 public: |
| 77 UsbDevicePermissionsPrompt( |
| 78 const Extension* extension, |
| 79 content::BrowserContext* context, |
| 80 bool multiple, |
| 81 const std::vector<device::UsbDeviceFilter>& filters, |
| 82 const DevicePermissionsPrompt::UsbDevicesCallback& callback) |
| 83 : Prompt(extension, context, multiple), |
| 84 filters_(filters), |
| 85 callback_(callback), |
| 86 service_observer_(this) {} |
| 87 |
| 88 private: |
| 89 ~UsbDevicePermissionsPrompt() override {} |
| 90 |
| 91 // DevicePermissionsPrompt::Prompt implementation: |
| 92 void SetObserver( |
| 93 DevicePermissionsPrompt::Prompt::Observer* observer) override { |
| 94 DevicePermissionsPrompt::Prompt::SetObserver(observer); |
| 95 |
| 96 if (observer) { |
| 97 UsbService* service = device::DeviceClient::Get()->GetUsbService(); |
| 98 if (service && !service_observer_.IsObserving(service)) { |
| 99 service->GetDevices( |
| 100 base::Bind(&UsbDevicePermissionsPrompt::OnDevicesEnumerated, this)); |
| 101 service_observer_.Add(service); |
| 102 } |
| 45 } | 103 } |
| 46 } | 104 } |
| 47 | 105 |
| 48 base::string16 product_string = device->product_string(); | 106 base::string16 GetHeading() const override { |
| 49 if (product_string.empty()) { | 107 return l10n_util::GetStringUTF16( |
| 50 const char* product_name = device::UsbIds::GetProductName( | 108 multiple() ? IDS_USB_DEVICE_PERMISSIONS_PROMPT_TITLE_MULTIPLE |
| 51 device->vendor_id(), device->product_id()); | 109 : IDS_USB_DEVICE_PERMISSIONS_PROMPT_TITLE_SINGLE); |
| 52 if (product_name) { | 110 } |
| 53 product_string = base::UTF8ToUTF16(product_name); | 111 |
| 54 } else { | 112 void Dismissed() override { |
| 55 base::string16 product_id = base::ASCIIToUTF16( | 113 DevicePermissionsManager* permissions_manager = |
| 56 base::StringPrintf("0x%04x", device->product_id())); | 114 DevicePermissionsManager::Get(browser_context()); |
| 57 product_string = | 115 std::vector<scoped_refptr<UsbDevice>> devices; |
| 58 l10n_util::GetStringFUTF16(IDS_DEVICE_UNKNOWN_PRODUCT, product_id); | 116 for (const DeviceInfo* device : devices_) { |
| 117 if (device->granted()) { |
| 118 const UsbDeviceInfo* usb_device = |
| 119 static_cast<const UsbDeviceInfo*>(device); |
| 120 devices.push_back(usb_device->device()); |
| 121 if (permissions_manager) { |
| 122 permissions_manager->AllowUsbDevice(extension()->id(), |
| 123 usb_device->device()); |
| 124 } |
| 125 } |
| 126 } |
| 127 DCHECK(multiple() || devices.size() <= 1); |
| 128 callback_.Run(devices); |
| 129 callback_.Reset(); |
| 130 } |
| 131 |
| 132 // device::UsbService::Observer implementation: |
| 133 void OnDeviceAdded(scoped_refptr<UsbDevice> device) override { |
| 134 if (!(filters_.empty() || UsbDeviceFilter::MatchesAny(device, filters_))) { |
| 135 return; |
| 136 } |
| 137 |
| 138 device->CheckUsbAccess(base::Bind( |
| 139 &UsbDevicePermissionsPrompt::AddCheckedDevice, this, device)); |
| 140 } |
| 141 |
| 142 void OnDeviceRemoved(scoped_refptr<UsbDevice> device) override { |
| 143 for (auto it = devices_.begin(); it != devices_.end(); ++it) { |
| 144 const UsbDeviceInfo* entry = static_cast<const UsbDeviceInfo*>(*it); |
| 145 if (entry->device() == device) { |
| 146 devices_.erase(it); |
| 147 if (observer()) { |
| 148 observer()->OnDevicesChanged(); |
| 149 } |
| 150 return; |
| 151 } |
| 59 } | 152 } |
| 60 } | 153 } |
| 61 | 154 |
| 62 name = l10n_util::GetStringFUTF16(IDS_DEVICE_PERMISSIONS_DEVICE_NAME, | 155 void OnDevicesEnumerated( |
| 63 product_string, manufacturer_string); | 156 const std::vector<scoped_refptr<UsbDevice>>& devices) { |
| 157 for (const auto& device : devices) { |
| 158 OnDeviceAdded(device); |
| 159 } |
| 160 } |
| 161 |
| 162 void AddCheckedDevice(scoped_refptr<UsbDevice> device, bool allowed) { |
| 163 if (allowed) { |
| 164 // TODO(reillyg): This method could be called after OnDeviceRemoved. We |
| 165 // should check that the device is still connected. |
| 166 devices_.push_back(new UsbDeviceInfo(device)); |
| 167 if (observer()) { |
| 168 observer()->OnDevicesChanged(); |
| 169 } |
| 170 } |
| 171 } |
| 172 |
| 173 std::vector<UsbDeviceFilter> filters_; |
| 174 DevicePermissionsPrompt::UsbDevicesCallback callback_; |
| 175 ScopedObserver<UsbService, UsbService::Observer> service_observer_; |
| 176 }; |
| 177 |
| 178 } // namespace |
| 179 |
| 180 DevicePermissionsPrompt::Prompt::DeviceInfo::DeviceInfo() { |
| 64 } | 181 } |
| 65 | 182 |
| 66 DevicePermissionsPrompt::Prompt::DeviceInfo::~DeviceInfo() { | 183 DevicePermissionsPrompt::Prompt::DeviceInfo::~DeviceInfo() { |
| 67 } | 184 } |
| 68 | 185 |
| 69 DevicePermissionsPrompt::Prompt::Observer::~Observer() { | 186 DevicePermissionsPrompt::Prompt::Observer::~Observer() { |
| 70 } | 187 } |
| 71 | 188 |
| 72 DevicePermissionsPrompt::Prompt::Prompt(Delegate* delegate, | 189 DevicePermissionsPrompt::Prompt::Prompt(const Extension* extension, |
| 73 const Extension* extension, | 190 content::BrowserContext* context, |
| 74 content::BrowserContext* context) | 191 bool multiple) |
| 75 : extension_(extension), | 192 : extension_(extension), browser_context_(context), multiple_(multiple) { |
| 76 browser_context_(context), | |
| 77 delegate_(delegate), | |
| 78 usb_service_observer_(this) { | |
| 79 } | 193 } |
| 80 | 194 |
| 81 void DevicePermissionsPrompt::Prompt::SetObserver(Observer* observer) { | 195 void DevicePermissionsPrompt::Prompt::SetObserver(Observer* observer) { |
| 82 observer_ = observer; | 196 observer_ = observer; |
| 83 | |
| 84 if (observer_) { | |
| 85 UsbService* service = device::DeviceClient::Get()->GetUsbService(); | |
| 86 if (service && !usb_service_observer_.IsObserving(service)) { | |
| 87 service->GetDevices(base::Bind( | |
| 88 &DevicePermissionsPrompt::Prompt::OnDevicesEnumerated, this)); | |
| 89 usb_service_observer_.Add(service); | |
| 90 } | |
| 91 } | |
| 92 } | |
| 93 | |
| 94 base::string16 DevicePermissionsPrompt::Prompt::GetHeading() const { | |
| 95 return l10n_util::GetStringUTF16( | |
| 96 multiple_ ? IDS_DEVICE_PERMISSIONS_PROMPT_TITLE_MULTIPLE | |
| 97 : IDS_DEVICE_PERMISSIONS_PROMPT_TITLE_SINGLE); | |
| 98 } | 197 } |
| 99 | 198 |
| 100 base::string16 DevicePermissionsPrompt::Prompt::GetPromptMessage() const { | 199 base::string16 DevicePermissionsPrompt::Prompt::GetPromptMessage() const { |
| 101 return l10n_util::GetStringFUTF16(multiple_ | 200 return l10n_util::GetStringFUTF16(multiple_ |
| 102 ? IDS_DEVICE_PERMISSIONS_PROMPT_MULTIPLE | 201 ? IDS_DEVICE_PERMISSIONS_PROMPT_MULTIPLE |
| 103 : IDS_DEVICE_PERMISSIONS_PROMPT_SINGLE, | 202 : IDS_DEVICE_PERMISSIONS_PROMPT_SINGLE, |
| 104 base::UTF8ToUTF16(extension_->name())); | 203 base::UTF8ToUTF16(extension_->name())); |
| 105 } | 204 } |
| 106 | 205 |
| 107 base::string16 DevicePermissionsPrompt::Prompt::GetDeviceName( | 206 base::string16 DevicePermissionsPrompt::Prompt::GetDeviceName( |
| 108 size_t index) const { | 207 size_t index) const { |
| 109 DCHECK_LT(index, devices_.size()); | 208 DCHECK_LT(index, devices_.size()); |
| 110 return devices_[index].name; | 209 return devices_[index]->name(); |
| 111 } | 210 } |
| 112 | 211 |
| 113 base::string16 DevicePermissionsPrompt::Prompt::GetDeviceSerialNumber( | 212 base::string16 DevicePermissionsPrompt::Prompt::GetDeviceSerialNumber( |
| 114 size_t index) const { | 213 size_t index) const { |
| 115 DCHECK_LT(index, devices_.size()); | 214 DCHECK_LT(index, devices_.size()); |
| 116 return devices_[index].device->serial_number(); | 215 return devices_[index]->serial_number(); |
| 117 } | 216 } |
| 118 | 217 |
| 119 void DevicePermissionsPrompt::Prompt::GrantDevicePermission(size_t index) { | 218 void DevicePermissionsPrompt::Prompt::GrantDevicePermission(size_t index) { |
| 120 DCHECK_LT(index, devices_.size()); | 219 DCHECK_LT(index, devices_.size()); |
| 121 devices_[index].granted = true; | 220 devices_[index]->set_granted(); |
| 122 } | |
| 123 | |
| 124 void DevicePermissionsPrompt::Prompt::Dismissed() { | |
| 125 DevicePermissionsManager* permissions_manager = | |
| 126 DevicePermissionsManager::Get(browser_context_); | |
| 127 std::vector<scoped_refptr<UsbDevice>> devices; | |
| 128 for (const DeviceInfo& device : devices_) { | |
| 129 if (device.granted) { | |
| 130 devices.push_back(device.device); | |
| 131 if (permissions_manager) { | |
| 132 permissions_manager->AllowUsbDevice(extension_->id(), device.device); | |
| 133 } | |
| 134 } | |
| 135 } | |
| 136 delegate_->OnUsbDevicesChosen(devices); | |
| 137 } | |
| 138 | |
| 139 void DevicePermissionsPrompt::Prompt::set_filters( | |
| 140 const std::vector<UsbDeviceFilter>& filters) { | |
| 141 filters_ = filters; | |
| 142 } | 221 } |
| 143 | 222 |
| 144 DevicePermissionsPrompt::Prompt::~Prompt() { | 223 DevicePermissionsPrompt::Prompt::~Prompt() { |
| 145 } | 224 } |
| 146 | 225 |
| 147 void DevicePermissionsPrompt::Prompt::OnDeviceAdded( | |
| 148 scoped_refptr<UsbDevice> device) { | |
| 149 if (!(filters_.empty() || UsbDeviceFilter::MatchesAny(device, filters_))) { | |
| 150 return; | |
| 151 } | |
| 152 | |
| 153 device->CheckUsbAccess(base::Bind( | |
| 154 &DevicePermissionsPrompt::Prompt::AddCheckedUsbDevice, this, device)); | |
| 155 } | |
| 156 | |
| 157 void DevicePermissionsPrompt::Prompt::OnDeviceRemoved( | |
| 158 scoped_refptr<UsbDevice> device) { | |
| 159 bool removed_entry = false; | |
| 160 for (std::vector<DeviceInfo>::iterator it = devices_.begin(); | |
| 161 it != devices_.end(); ++it) { | |
| 162 if (it->device == device) { | |
| 163 devices_.erase(it); | |
| 164 removed_entry = true; | |
| 165 break; | |
| 166 } | |
| 167 } | |
| 168 if (observer_ && removed_entry) { | |
| 169 observer_->OnDevicesChanged(); | |
| 170 } | |
| 171 } | |
| 172 | |
| 173 void DevicePermissionsPrompt::Prompt::OnDevicesEnumerated( | |
| 174 const std::vector<scoped_refptr<UsbDevice>>& devices) { | |
| 175 for (const auto& device : devices) { | |
| 176 if (filters_.empty() || UsbDeviceFilter::MatchesAny(device, filters_)) { | |
| 177 device->CheckUsbAccess(base::Bind( | |
| 178 &DevicePermissionsPrompt::Prompt::AddCheckedUsbDevice, this, device)); | |
| 179 } | |
| 180 } | |
| 181 } | |
| 182 | |
| 183 void DevicePermissionsPrompt::Prompt::AddCheckedUsbDevice( | |
| 184 scoped_refptr<UsbDevice> device, | |
| 185 bool allowed) { | |
| 186 if (allowed) { | |
| 187 devices_.push_back(DeviceInfo(device)); | |
| 188 if (observer_) { | |
| 189 observer_->OnDevicesChanged(); | |
| 190 } | |
| 191 } | |
| 192 } | |
| 193 | |
| 194 DevicePermissionsPrompt::DevicePermissionsPrompt( | 226 DevicePermissionsPrompt::DevicePermissionsPrompt( |
| 195 content::WebContents* web_contents) | 227 content::WebContents* web_contents) |
| 196 : web_contents_(web_contents) { | 228 : web_contents_(web_contents) { |
| 197 } | 229 } |
| 198 | 230 |
| 199 DevicePermissionsPrompt::~DevicePermissionsPrompt() { | 231 DevicePermissionsPrompt::~DevicePermissionsPrompt() { |
| 200 } | 232 } |
| 201 | 233 |
| 202 void DevicePermissionsPrompt::AskForUsbDevices( | 234 void DevicePermissionsPrompt::AskForUsbDevices( |
| 203 Delegate* delegate, | |
| 204 const Extension* extension, | 235 const Extension* extension, |
| 205 content::BrowserContext* context, | 236 content::BrowserContext* context, |
| 206 bool multiple, | 237 bool multiple, |
| 207 const std::vector<UsbDeviceFilter>& filters) { | 238 const std::vector<UsbDeviceFilter>& filters, |
| 208 prompt_ = new Prompt(delegate, extension, context); | 239 const UsbDevicesCallback& callback) { |
| 209 prompt_->set_multiple(multiple); | 240 prompt_ = new UsbDevicePermissionsPrompt(extension, context, multiple, |
| 210 prompt_->set_filters(filters); | 241 filters, callback); |
| 211 | |
| 212 ShowDialog(); | 242 ShowDialog(); |
| 213 } | 243 } |
| 214 | 244 |
| 215 } // namespace extensions | 245 } // namespace extensions |
| OLD | NEW |