| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | // Copyright 2019 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 "shell/browser/hid/hid_chooser_context.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-10 19:00:37 -06:00
										 |  |  | #include <string_view>
 | 
					
						
							| 
									
										
										
										
											2024-07-25 04:25:45 -05:00
										 |  |  | #include <utility>
 | 
					
						
							| 
									
										
										
										
											2024-01-10 19:00:37 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include "base/command_line.h"
 | 
					
						
							| 
									
										
										
										
											2024-07-15 16:41:38 +02:00
										 |  |  | #include "base/containers/map_util.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include "base/strings/utf_string_conversions.h"
 | 
					
						
							|  |  |  | #include "base/values.h"
 | 
					
						
							| 
									
										
										
										
											2022-04-06 09:34:35 -07:00
										 |  |  | #include "chrome/grit/generated_resources.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include "components/content_settings/core/common/content_settings_types.h"
 | 
					
						
							|  |  |  | #include "components/prefs/pref_service.h"
 | 
					
						
							|  |  |  | #include "content/public/browser/device_service.h"
 | 
					
						
							|  |  |  | #include "services/device/public/cpp/hid/hid_blocklist.h"
 | 
					
						
							|  |  |  | #include "services/device/public/cpp/hid/hid_switches.h"
 | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  | #include "shell/browser/api/electron_api_session.h"
 | 
					
						
							| 
									
										
										
										
											2024-07-25 04:25:45 -05:00
										 |  |  | #include "shell/browser/electron_browser_context.h"
 | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  | #include "shell/browser/electron_permission_manager.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include "shell/browser/web_contents_permission_helper.h"
 | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  | #include "shell/common/electron_constants.h"
 | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  | #include "shell/common/gin_converters/content_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_converters/frame_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_converters/hid_device_info_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_converters/value_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_helper/dictionary.h"
 | 
					
						
							| 
									
										
										
										
											2024-11-04 12:58:16 -06:00
										 |  |  | #include "third_party/abseil-cpp/absl/strings/str_format.h"
 | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  | #include "third_party/blink/public/common/permissions/permission_utils.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include "ui/base/l10n/l10n_util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  | #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
 | 
					
						
							|  |  |  | #include "base/containers/fixed_flat_set.h"
 | 
					
						
							|  |  |  | #include "extensions/common/constants.h"
 | 
					
						
							|  |  |  | #endif  // BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | namespace electron { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | HidChooserContext::HidChooserContext(ElectronBrowserContext* context) | 
					
						
							|  |  |  |     : browser_context_(context) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | HidChooserContext::~HidChooserContext() { | 
					
						
							|  |  |  |   // Notify observers that the chooser context is about to be destroyed.
 | 
					
						
							|  |  |  |   // Observers must remove themselves from the observer lists.
 | 
					
						
							|  |  |  |   for (auto& observer : device_observer_list_) { | 
					
						
							|  |  |  |     observer.OnHidChooserContextShutdown(); | 
					
						
							|  |  |  |     DCHECK(!device_observer_list_.HasObserver(&observer)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // static
 | 
					
						
							|  |  |  | std::u16string HidChooserContext::DisplayNameFromDeviceInfo( | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							|  |  |  |   if (device.product_name.empty()) { | 
					
						
							|  |  |  |     auto device_id_string = base::ASCIIToUTF16( | 
					
						
							| 
									
										
										
										
											2024-11-04 12:58:16 -06:00
										 |  |  |         absl::StrFormat("%04X:%04X", device.vendor_id, device.product_id)); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |     return l10n_util::GetStringFUTF16(IDS_HID_CHOOSER_ITEM_WITHOUT_NAME, | 
					
						
							|  |  |  |                                       device_id_string); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return base::UTF8ToUTF16(device.product_name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // static
 | 
					
						
							|  |  |  | bool HidChooserContext::CanStorePersistentEntry( | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							|  |  |  |   return !device.serial_number.empty() && !device.product_name.empty(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // static
 | 
					
						
							|  |  |  | base::Value HidChooserContext::DeviceInfoToValue( | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							| 
									
										
										
										
											2023-03-15 14:07:51 +01:00
										 |  |  |   base::Value::Dict value; | 
					
						
							|  |  |  |   value.Set( | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |       kHidDeviceNameKey, | 
					
						
							|  |  |  |       base::UTF16ToUTF8(HidChooserContext::DisplayNameFromDeviceInfo(device))); | 
					
						
							| 
									
										
										
										
											2023-03-15 14:07:51 +01:00
										 |  |  |   value.Set(kDeviceVendorIdKey, device.vendor_id); | 
					
						
							|  |  |  |   value.Set(kDeviceProductIdKey, device.product_id); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   if (HidChooserContext::CanStorePersistentEntry(device)) { | 
					
						
							|  |  |  |     // Use the USB serial number as a persistent identifier. If it is
 | 
					
						
							|  |  |  |     // unavailable, only ephemeral permissions may be granted.
 | 
					
						
							| 
									
										
										
										
											2023-03-15 14:07:51 +01:00
										 |  |  |     value.Set(kDeviceSerialNumberKey, device.serial_number); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     // The GUID is a temporary ID created on connection that remains valid until
 | 
					
						
							|  |  |  |     // the device is disconnected. Ephemeral permissions are keyed by this ID
 | 
					
						
							|  |  |  |     // and must be granted again each time the device is connected.
 | 
					
						
							| 
									
										
										
										
											2023-03-15 14:07:51 +01:00
										 |  |  |     value.Set(kHidGuidKey, device.guid); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-03-15 14:07:51 +01:00
										 |  |  |   return base::Value(std::move(value)); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserContext::GrantDevicePermission( | 
					
						
							|  |  |  |     const url::Origin& origin, | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							| 
									
										
										
										
											2025-02-21 17:33:43 -06:00
										 |  |  |   DCHECK(devices_.contains(device.guid)); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   if (CanStorePersistentEntry(device)) { | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |     auto* permission_manager = static_cast<ElectronPermissionManager*>( | 
					
						
							|  |  |  |         browser_context_->GetPermissionControllerDelegate()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-22 15:53:29 -04:00
										 |  |  |     permission_manager->GrantDevicePermission(blink::PermissionType::HID, | 
					
						
							|  |  |  |                                               origin, DeviceInfoToValue(device), | 
					
						
							|  |  |  |                                               browser_context_); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     ephemeral_devices_[origin].insert(device.guid); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  | void HidChooserContext::RevokeDevicePermission( | 
					
						
							|  |  |  |     const url::Origin& origin, | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							| 
									
										
										
										
											2025-02-21 17:33:43 -06:00
										 |  |  |   DCHECK(devices_.contains(device.guid)); | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  |   if (CanStorePersistentEntry(device)) { | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |     RevokePersistentDevicePermission(origin, device); | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     RevokeEphemeralDevicePermission(origin, device); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |   api::Session* session = api::Session::FromBrowserContext(browser_context_); | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  |   if (session) { | 
					
						
							|  |  |  |     v8::Isolate* isolate = JavascriptEnvironment::GetIsolate(); | 
					
						
							|  |  |  |     v8::HandleScope scope(isolate); | 
					
						
							| 
									
										
										
										
											2023-10-10 12:45:44 +02:00
										 |  |  |     auto details = gin_helper::Dictionary::CreateEmpty(isolate); | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  |     details.Set("device", device.Clone()); | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |     details.Set("origin", origin.Serialize()); | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  |     session->Emit("hid-device-revoked", details); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserContext::RevokePersistentDevicePermission( | 
					
						
							|  |  |  |     const url::Origin& origin, | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							|  |  |  |   auto* permission_manager = static_cast<ElectronPermissionManager*>( | 
					
						
							|  |  |  |       browser_context_->GetPermissionControllerDelegate()); | 
					
						
							| 
									
										
										
										
											2025-04-22 15:53:29 -04:00
										 |  |  |   permission_manager->RevokeDevicePermission(blink::PermissionType::HID, origin, | 
					
						
							|  |  |  |                                              DeviceInfoToValue(device), | 
					
						
							|  |  |  |                                              browser_context_); | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  |   RevokeEphemeralDevicePermission(origin, device); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserContext::RevokeEphemeralDevicePermission( | 
					
						
							|  |  |  |     const url::Origin& origin, | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							|  |  |  |   auto it = ephemeral_devices_.find(origin); | 
					
						
							| 
									
										
										
										
											2024-07-15 16:41:38 +02:00
										 |  |  |   if (it == ephemeral_devices_.end()) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   std::set<std::string>& device_guids = it->second; | 
					
						
							|  |  |  |   std::erase_if(device_guids, [&](const auto& guid) { | 
					
						
							|  |  |  |     auto* device_ptr = base::FindPtrOrNull(devices_, guid); | 
					
						
							|  |  |  |     return device_ptr && | 
					
						
							|  |  |  |            device_ptr->physical_device_id == device.physical_device_id; | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (device_guids.empty()) | 
					
						
							|  |  |  |     ephemeral_devices_.erase(it); | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | bool HidChooserContext::HasDevicePermission( | 
					
						
							|  |  |  |     const url::Origin& origin, | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   if (!base::CommandLine::ForCurrentProcess()->HasSwitch( | 
					
						
							|  |  |  |           switches::kDisableHidBlocklist) && | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |       device.is_excluded_by_blocklist) | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |     return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   auto it = ephemeral_devices_.find(origin); | 
					
						
							| 
									
										
										
										
											2025-02-21 17:33:43 -06:00
										 |  |  |   if (it != ephemeral_devices_.end() && it->second.contains(device.guid)) { | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |   auto* permission_manager = static_cast<ElectronPermissionManager*>( | 
					
						
							|  |  |  |       browser_context_->GetPermissionControllerDelegate()); | 
					
						
							|  |  |  |   return permission_manager->CheckDevicePermission( | 
					
						
							| 
									
										
										
										
											2025-04-22 15:53:29 -04:00
										 |  |  |       blink::PermissionType::HID, origin, DeviceInfoToValue(device), | 
					
						
							|  |  |  |       browser_context_); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  | bool HidChooserContext::IsFidoAllowedForOrigin(const url::Origin& origin) { | 
					
						
							|  |  |  | #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
 | 
					
						
							|  |  |  |   static constexpr auto kPrivilegedExtensionIds = | 
					
						
							| 
									
										
										
										
											2024-01-10 19:00:37 -06:00
										 |  |  |       base::MakeFixedFlatSet<std::string_view>({ | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  |           "ckcendljdlmgnhghiaomidhiiclmapok",  // gnubbyd-v3 dev
 | 
					
						
							|  |  |  |           "lfboplenmmjcmpbkeemecobbadnmpfhi",  // gnubbyd-v3 prod
 | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (origin.scheme() == extensions::kExtensionScheme && | 
					
						
							| 
									
										
										
										
											2025-02-21 17:33:43 -06:00
										 |  |  |       kPrivilegedExtensionIds.contains(origin.host())) { | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif  // BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // This differs from upstream - we want to allow users greater
 | 
					
						
							|  |  |  |   // ability to communicate with FIDO devices in Electron.
 | 
					
						
							|  |  |  |   return base::CommandLine::ForCurrentProcess()->HasSwitch( | 
					
						
							|  |  |  |       switches::kDisableHidBlocklist); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | void HidChooserContext::AddDeviceObserver(DeviceObserver* observer) { | 
					
						
							|  |  |  |   EnsureHidManagerConnection(); | 
					
						
							|  |  |  |   device_observer_list_.AddObserver(observer); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserContext::RemoveDeviceObserver(DeviceObserver* observer) { | 
					
						
							|  |  |  |   device_observer_list_.RemoveObserver(observer); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserContext::GetDevices( | 
					
						
							|  |  |  |     device::mojom::HidManager::GetDevicesCallback callback) { | 
					
						
							|  |  |  |   if (!is_initialized_) { | 
					
						
							|  |  |  |     EnsureHidManagerConnection(); | 
					
						
							|  |  |  |     pending_get_devices_requests_.push(std::move(callback)); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   std::vector<device::mojom::HidDeviceInfoPtr> device_list; | 
					
						
							|  |  |  |   device_list.reserve(devices_.size()); | 
					
						
							|  |  |  |   for (const auto& pair : devices_) | 
					
						
							|  |  |  |     device_list.push_back(pair.second->Clone()); | 
					
						
							| 
									
										
										
										
											2023-02-03 12:43:42 +01:00
										 |  |  |   base::SequencedTaskRunner::GetCurrentDefault()->PostTask( | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |       FROM_HERE, base::BindOnce(std::move(callback), std::move(device_list))); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const device::mojom::HidDeviceInfo* HidChooserContext::GetDeviceInfo( | 
					
						
							|  |  |  |     const std::string& guid) { | 
					
						
							|  |  |  |   DCHECK(is_initialized_); | 
					
						
							|  |  |  |   auto it = devices_.find(guid); | 
					
						
							|  |  |  |   return it == devices_.end() ? nullptr : it->second.get(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | device::mojom::HidManager* HidChooserContext::GetHidManager() { | 
					
						
							|  |  |  |   EnsureHidManagerConnection(); | 
					
						
							|  |  |  |   return hid_manager_.get(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | base::WeakPtr<HidChooserContext> HidChooserContext::AsWeakPtr() { | 
					
						
							|  |  |  |   return weak_factory_.GetWeakPtr(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserContext::DeviceAdded(device::mojom::HidDeviceInfoPtr device) { | 
					
						
							|  |  |  |   DCHECK(device); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Update the device list.
 | 
					
						
							| 
									
										
										
										
											2025-02-21 17:33:43 -06:00
										 |  |  |   if (!devices_.contains(device->guid)) | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |     devices_.insert({device->guid, device->Clone()}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Notify all observers.
 | 
					
						
							| 
									
										
										
										
											2025-05-02 04:10:52 -05:00
										 |  |  |   device_observer_list_.Notify(&DeviceObserver::OnDeviceAdded, *device); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserContext::DeviceRemoved(device::mojom::HidDeviceInfoPtr device) { | 
					
						
							|  |  |  |   DCHECK(device); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Update the device list.
 | 
					
						
							| 
									
										
										
										
											2025-03-28 21:34:24 -05:00
										 |  |  |   const size_t n_erased = devices_.erase(device->guid); | 
					
						
							|  |  |  |   DCHECK_EQ(n_erased, 1U); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Notify all device observers.
 | 
					
						
							| 
									
										
										
										
											2025-05-02 04:10:52 -05:00
										 |  |  |   device_observer_list_.Notify(&DeviceObserver::OnDeviceRemoved, *device); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Next we'll notify observers for revoked permissions. If the device does not
 | 
					
						
							|  |  |  |   // support persistent permissions then device permissions are revoked on
 | 
					
						
							|  |  |  |   // disconnect.
 | 
					
						
							|  |  |  |   if (CanStorePersistentEntry(*device)) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-05 08:31:58 -05:00
										 |  |  |   for (auto& [origin, guids] : ephemeral_devices_) | 
					
						
							|  |  |  |     guids.erase(device->guid); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserContext::DeviceChanged(device::mojom::HidDeviceInfoPtr device) { | 
					
						
							|  |  |  |   DCHECK(device); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Update the device list.
 | 
					
						
							| 
									
										
										
										
											2025-04-03 17:51:37 -05:00
										 |  |  |   auto& mapped = devices_[device->guid]; | 
					
						
							|  |  |  |   DCHECK(!mapped.is_null()); | 
					
						
							|  |  |  |   mapped = device->Clone(); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Notify all observers.
 | 
					
						
							| 
									
										
										
										
											2025-05-02 04:10:52 -05:00
										 |  |  |   device_observer_list_.Notify(&DeviceObserver::OnDeviceChanged, *device); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserContext::EnsureHidManagerConnection() { | 
					
						
							|  |  |  |   if (hid_manager_) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   mojo::PendingRemote<device::mojom::HidManager> manager; | 
					
						
							|  |  |  |   content::GetDeviceService().BindHidManager( | 
					
						
							|  |  |  |       manager.InitWithNewPipeAndPassReceiver()); | 
					
						
							|  |  |  |   SetUpHidManagerConnection(std::move(manager)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserContext::SetUpHidManagerConnection( | 
					
						
							|  |  |  |     mojo::PendingRemote<device::mojom::HidManager> manager) { | 
					
						
							|  |  |  |   hid_manager_.Bind(std::move(manager)); | 
					
						
							|  |  |  |   hid_manager_.set_disconnect_handler(base::BindOnce( | 
					
						
							|  |  |  |       &HidChooserContext::OnHidManagerConnectionError, base::Unretained(this))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   hid_manager_->GetDevicesAndSetClient( | 
					
						
							|  |  |  |       client_receiver_.BindNewEndpointAndPassRemote(), | 
					
						
							|  |  |  |       base::BindOnce(&HidChooserContext::InitDeviceList, | 
					
						
							|  |  |  |                      weak_factory_.GetWeakPtr())); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserContext::InitDeviceList( | 
					
						
							|  |  |  |     std::vector<device::mojom::HidDeviceInfoPtr> devices) { | 
					
						
							|  |  |  |   for (auto& device : devices) | 
					
						
							| 
									
										
										
										
											2025-04-25 13:11:53 -05:00
										 |  |  |     devices_.try_emplace(device->guid, std::move(device)); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   is_initialized_ = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (!pending_get_devices_requests_.empty()) { | 
					
						
							|  |  |  |     std::vector<device::mojom::HidDeviceInfoPtr> device_list; | 
					
						
							|  |  |  |     device_list.reserve(devices.size()); | 
					
						
							|  |  |  |     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 HidChooserContext::OnHidManagerConnectionError() { | 
					
						
							|  |  |  |   hid_manager_.reset(); | 
					
						
							|  |  |  |   client_receiver_.reset(); | 
					
						
							|  |  |  |   devices_.clear(); | 
					
						
							|  |  |  |   ephemeral_devices_.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Notify all device observers.
 | 
					
						
							| 
									
										
										
										
											2025-05-02 04:10:52 -05:00
										 |  |  |   device_observer_list_.Notify(&DeviceObserver::OnHidManagerConnectionError); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace electron
 |