 05dfd14913
			
		
	
	
	
	
	05dfd14913* refactor: add EmitWarning(Isolate*, ...) warning * chore: remove EmitWarning(node::Environment*, ...) * chore: add code comments * fixup! refactor: add EmitWarning(Isolate*, ...) warning * chore: remove unused node #includes
		
			
				
	
	
		
			338 lines
		
	
	
	
		
			12 KiB
			
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			338 lines
		
	
	
	
		
			12 KiB
			
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright (c) 2022 Microsoft, Inc.
 | |
| // Use of this source code is governed by the MIT license that can be
 | |
| // found in the LICENSE file.
 | |
| 
 | |
| #include "shell/browser/usb/usb_chooser_context.h"
 | |
| 
 | |
| #include <string_view>
 | |
| #include <utility>
 | |
| #include <vector>
 | |
| 
 | |
| #include "base/containers/contains.h"
 | |
| #include "base/functional/bind.h"
 | |
| #include "base/task/sequenced_task_runner.h"
 | |
| #include "base/values.h"
 | |
| #include "components/content_settings/core/common/content_settings.h"
 | |
| #include "content/public/browser/device_service.h"
 | |
| #include "services/device/public/cpp/usb/usb_ids.h"
 | |
| #include "services/device/public/mojom/usb_device.mojom.h"
 | |
| #include "shell/browser/api/electron_api_session.h"
 | |
| #include "shell/browser/electron_browser_context.h"
 | |
| #include "shell/browser/electron_permission_manager.h"
 | |
| #include "shell/browser/web_contents_permission_helper.h"
 | |
| #include "shell/common/electron_constants.h"
 | |
| #include "shell/common/gin_converters/usb_device_info_converter.h"
 | |
| #include "shell/common/gin_helper/dictionary.h"
 | |
| #include "ui/base/l10n/l10n_util.h"
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| constexpr char kDeviceNameKey[] = "productName";
 | |
| constexpr char kDeviceIdKey[] = "deviceId";
 | |
| constexpr int kUsbClassMassStorage = 0x08;
 | |
| 
 | |
| bool CanStorePersistentEntry(const device::mojom::UsbDeviceInfo& device_info) {
 | |
|   return device_info.serial_number && !device_info.serial_number->empty();
 | |
| }
 | |
| 
 | |
| bool IsMassStorageInterface(const device::mojom::UsbInterfaceInfo& interface) {
 | |
|   for (auto& alternate : interface.alternates) {
 | |
|     if (alternate->class_code == kUsbClassMassStorage)
 | |
|       return true;
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| bool ShouldExposeDevice(const device::mojom::UsbDeviceInfo& device_info) {
 | |
|   // blink::USBDevice::claimInterface() disallows claiming mass storage
 | |
|   // interfaces, but explicitly prevent access in the browser process as
 | |
|   // ChromeOS would allow these interfaces to be claimed.
 | |
|   for (auto& configuration : device_info.configurations) {
 | |
|     if (configuration->interfaces.size() == 0) {
 | |
|       return true;
 | |
|     }
 | |
|     for (auto& interface : configuration->interfaces) {
 | |
|       if (!IsMassStorageInterface(*interface))
 | |
|         return true;
 | |
|     }
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| }  // namespace
 | |
| 
 | |
| namespace electron {
 | |
| 
 | |
| UsbChooserContext::UsbChooserContext(ElectronBrowserContext* context)
 | |
|     : browser_context_(context) {}
 | |
| 
 | |
| // static
 | |
| base::Value UsbChooserContext::DeviceInfoToValue(
 | |
|     const device::mojom::UsbDeviceInfo& device_info) {
 | |
|   base::Value::Dict device_value;
 | |
|   device_value.Set(kDeviceNameKey, device_info.product_name
 | |
|                                        ? *device_info.product_name
 | |
|                                        : std::u16string_view());
 | |
|   device_value.Set(kDeviceVendorIdKey, device_info.vendor_id);
 | |
|   device_value.Set(kDeviceProductIdKey, device_info.product_id);
 | |
| 
 | |
|   if (device_info.manufacturer_name) {
 | |
|     device_value.Set("manufacturerName", *device_info.manufacturer_name);
 | |
|   }
 | |
| 
 | |
|   // CanStorePersistentEntry checks if |device_info.serial_number| is not empty.
 | |
|   if (CanStorePersistentEntry(device_info)) {
 | |
|     device_value.Set(kDeviceSerialNumberKey, *device_info.serial_number);
 | |
|   }
 | |
| 
 | |
|   device_value.Set(kDeviceIdKey, device_info.guid);
 | |
| 
 | |
|   device_value.Set("usbVersionMajor", device_info.usb_version_major);
 | |
|   device_value.Set("usbVersionMinor", device_info.usb_version_minor);
 | |
|   device_value.Set("usbVersionSubminor", device_info.usb_version_subminor);
 | |
|   device_value.Set("deviceClass", device_info.class_code);
 | |
|   device_value.Set("deviceSubclass", device_info.subclass_code);
 | |
|   device_value.Set("deviceProtocol", device_info.protocol_code);
 | |
|   device_value.Set("deviceVersionMajor", device_info.device_version_major);
 | |
|   device_value.Set("deviceVersionMinor", device_info.device_version_minor);
 | |
|   device_value.Set("deviceVersionSubminor",
 | |
|                    device_info.device_version_subminor);
 | |
|   return base::Value(std::move(device_value));
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::InitDeviceList(
 | |
|     std::vector<device::mojom::UsbDeviceInfoPtr> devices) {
 | |
|   for (auto& device_info : devices) {
 | |
|     DCHECK(device_info);
 | |
|     if (ShouldExposeDevice(*device_info)) {
 | |
|       devices_.insert(
 | |
|           std::make_pair(device_info->guid, std::move(device_info)));
 | |
|     }
 | |
|   }
 | |
|   is_initialized_ = true;
 | |
| 
 | |
|   while (!pending_get_devices_requests_.empty()) {
 | |
|     std::vector<device::mojom::UsbDeviceInfoPtr> device_list;
 | |
|     for (const auto& entry : devices_) {
 | |
|       device_list.push_back(entry.second->Clone());
 | |
|     }
 | |
|     std::move(pending_get_devices_requests_.front())
 | |
|         .Run(std::move(device_list));
 | |
|     pending_get_devices_requests_.pop();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::EnsureConnectionWithDeviceManager() {
 | |
|   if (device_manager_)
 | |
|     return;
 | |
| 
 | |
|   // Receive mojo::Remote<UsbDeviceManager> from DeviceService.
 | |
|   content::GetDeviceService().BindUsbDeviceManager(
 | |
|       device_manager_.BindNewPipeAndPassReceiver());
 | |
| 
 | |
|   SetUpDeviceManagerConnection();
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::SetUpDeviceManagerConnection() {
 | |
|   DCHECK(device_manager_);
 | |
|   device_manager_.set_disconnect_handler(
 | |
|       base::BindOnce(&UsbChooserContext::OnDeviceManagerConnectionError,
 | |
|                      base::Unretained(this)));
 | |
| 
 | |
|   // Listen for added/removed device events.
 | |
|   DCHECK(!client_receiver_.is_bound());
 | |
|   device_manager_->EnumerateDevicesAndSetClient(
 | |
|       client_receiver_.BindNewEndpointAndPassRemote(),
 | |
|       base::BindOnce(&UsbChooserContext::InitDeviceList,
 | |
|                      weak_factory_.GetWeakPtr()));
 | |
| }
 | |
| 
 | |
| UsbChooserContext::~UsbChooserContext() {
 | |
|   OnDeviceManagerConnectionError();
 | |
|   for (auto& observer : device_observer_list_) {
 | |
|     observer.OnBrowserContextShutdown();
 | |
|     DCHECK(!device_observer_list_.HasObserver(&observer));
 | |
|   }
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::RevokeDevicePermissionWebInitiated(
 | |
|     const url::Origin& origin,
 | |
|     const device::mojom::UsbDeviceInfo& device) {
 | |
|   DCHECK(base::Contains(devices_, device.guid));
 | |
|   RevokeObjectPermissionInternal(origin, DeviceInfoToValue(device),
 | |
|                                  /*revoked_by_website=*/true);
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::RevokeObjectPermissionInternal(
 | |
|     const url::Origin& origin,
 | |
|     const base::Value& object,
 | |
|     bool revoked_by_website = false) {
 | |
|   const base::Value::Dict* object_dict = object.GetIfDict();
 | |
|   DCHECK(object_dict != nullptr);
 | |
| 
 | |
|   if (object_dict->FindString(kDeviceSerialNumberKey) != nullptr) {
 | |
|     auto* permission_manager = static_cast<ElectronPermissionManager*>(
 | |
|         browser_context_->GetPermissionControllerDelegate());
 | |
|     permission_manager->RevokeDevicePermission(
 | |
|         static_cast<blink::PermissionType>(
 | |
|             WebContentsPermissionHelper::PermissionType::USB),
 | |
|         origin, object, browser_context_);
 | |
|   } else {
 | |
|     const std::string* guid = object_dict->FindString(kDeviceIdKey);
 | |
|     auto it = ephemeral_devices_.find(origin);
 | |
|     if (it != ephemeral_devices_.end()) {
 | |
|       it->second.erase(*guid);
 | |
|       if (it->second.empty())
 | |
|         ephemeral_devices_.erase(it);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   api::Session* session = api::Session::FromBrowserContext(browser_context_);
 | |
|   if (session) {
 | |
|     v8::Isolate* isolate = JavascriptEnvironment::GetIsolate();
 | |
|     v8::HandleScope scope(isolate);
 | |
|     auto details = gin_helper::Dictionary::CreateEmpty(isolate);
 | |
|     details.Set("device", object);
 | |
|     details.Set("origin", origin.Serialize());
 | |
|     session->Emit("usb-device-revoked", details);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::GrantDevicePermission(
 | |
|     const url::Origin& origin,
 | |
|     const device::mojom::UsbDeviceInfo& device_info) {
 | |
|   if (CanStorePersistentEntry(device_info)) {
 | |
|     auto* permission_manager = static_cast<ElectronPermissionManager*>(
 | |
|         browser_context_->GetPermissionControllerDelegate());
 | |
|     permission_manager->GrantDevicePermission(
 | |
|         static_cast<blink::PermissionType>(
 | |
|             WebContentsPermissionHelper::PermissionType::USB),
 | |
|         origin, DeviceInfoToValue(device_info), browser_context_);
 | |
|   } else {
 | |
|     ephemeral_devices_[origin].insert(device_info.guid);
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool UsbChooserContext::HasDevicePermission(
 | |
|     const url::Origin& origin,
 | |
|     const device::mojom::UsbDeviceInfo& device_info) {
 | |
|   auto it = ephemeral_devices_.find(origin);
 | |
|   if (it != ephemeral_devices_.end() &&
 | |
|       base::Contains(it->second, device_info.guid)) {
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   auto* permission_manager = static_cast<ElectronPermissionManager*>(
 | |
|       browser_context_->GetPermissionControllerDelegate());
 | |
| 
 | |
|   return permission_manager->CheckDevicePermission(
 | |
|       static_cast<blink::PermissionType>(
 | |
|           WebContentsPermissionHelper::PermissionType::USB),
 | |
|       origin, DeviceInfoToValue(device_info), browser_context_);
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::GetDevices(
 | |
|     device::mojom::UsbDeviceManager::GetDevicesCallback callback) {
 | |
|   if (!is_initialized_) {
 | |
|     EnsureConnectionWithDeviceManager();
 | |
|     pending_get_devices_requests_.push(std::move(callback));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   std::vector<device::mojom::UsbDeviceInfoPtr> device_list;
 | |
|   for (const auto& pair : devices_) {
 | |
|     device_list.push_back(pair.second->Clone());
 | |
|   }
 | |
|   base::SequencedTaskRunner::GetCurrentDefault()->PostTask(
 | |
|       FROM_HERE, base::BindOnce(std::move(callback), std::move(device_list)));
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::GetDevice(
 | |
|     const std::string& guid,
 | |
|     base::span<const uint8_t> blocked_interface_classes,
 | |
|     mojo::PendingReceiver<device::mojom::UsbDevice> device_receiver,
 | |
|     mojo::PendingRemote<device::mojom::UsbDeviceClient> device_client) {
 | |
|   EnsureConnectionWithDeviceManager();
 | |
|   device_manager_->GetDevice(
 | |
|       guid,
 | |
|       std::vector<uint8_t>(blocked_interface_classes.begin(),
 | |
|                            blocked_interface_classes.end()),
 | |
|       std::move(device_receiver), std::move(device_client));
 | |
| }
 | |
| 
 | |
| const device::mojom::UsbDeviceInfo* UsbChooserContext::GetDeviceInfo(
 | |
|     const std::string& guid) {
 | |
|   DCHECK(is_initialized_);
 | |
|   auto it = devices_.find(guid);
 | |
|   return it == devices_.end() ? nullptr : it->second.get();
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::AddObserver(DeviceObserver* observer) {
 | |
|   EnsureConnectionWithDeviceManager();
 | |
|   device_observer_list_.AddObserver(observer);
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::RemoveObserver(DeviceObserver* observer) {
 | |
|   device_observer_list_.RemoveObserver(observer);
 | |
| }
 | |
| 
 | |
| base::WeakPtr<UsbChooserContext> UsbChooserContext::AsWeakPtr() {
 | |
|   return weak_factory_.GetWeakPtr();
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::OnDeviceAdded(
 | |
|     device::mojom::UsbDeviceInfoPtr device_info) {
 | |
|   DCHECK(device_info);
 | |
|   // Update the device list.
 | |
|   DCHECK(!base::Contains(devices_, device_info->guid));
 | |
|   if (!ShouldExposeDevice(*device_info))
 | |
|     return;
 | |
|   devices_.insert(std::make_pair(device_info->guid, device_info->Clone()));
 | |
| 
 | |
|   // Notify all observers.
 | |
|   for (auto& observer : device_observer_list_)
 | |
|     observer.OnDeviceAdded(*device_info);
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::OnDeviceRemoved(
 | |
|     device::mojom::UsbDeviceInfoPtr device_info) {
 | |
|   DCHECK(device_info);
 | |
| 
 | |
|   if (!ShouldExposeDevice(*device_info)) {
 | |
|     DCHECK(!base::Contains(devices_, device_info->guid));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Update the device list.
 | |
|   DCHECK(base::Contains(devices_, device_info->guid));
 | |
|   devices_.erase(device_info->guid);
 | |
| 
 | |
|   // Notify all device observers.
 | |
|   for (auto& observer : device_observer_list_)
 | |
|     observer.OnDeviceRemoved(*device_info);
 | |
| 
 | |
|   // If the device was persistent, return. Otherwise, notify all permission
 | |
|   // observers that its permissions were revoked.
 | |
|   if (device_info->serial_number &&
 | |
|       !device_info->serial_number.value().empty()) {
 | |
|     return;
 | |
|   }
 | |
|   for (auto& map_entry : ephemeral_devices_) {
 | |
|     map_entry.second.erase(device_info->guid);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void UsbChooserContext::OnDeviceManagerConnectionError() {
 | |
|   device_manager_.reset();
 | |
|   client_receiver_.reset();
 | |
|   devices_.clear();
 | |
|   is_initialized_ = false;
 | |
| 
 | |
|   ephemeral_devices_.clear();
 | |
| 
 | |
|   // Notify all device observers.
 | |
|   for (auto& observer : device_observer_list_)
 | |
|     observer.OnDeviceManagerConnectionError();
 | |
| }
 | |
| 
 | |
| }  // namespace electron
 |