| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | // Copyright (c) 2021 Microsoft, Inc.
 | 
					
						
							|  |  |  | // Use of this source code is governed by the MIT license that can be
 | 
					
						
							|  |  |  | // found in the LICENSE file.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-22 08:34:31 +01:00
										 |  |  | #ifndef ELECTRON_SHELL_BROWSER_HID_HID_CHOOSER_CONTEXT_H_
 | 
					
						
							|  |  |  | #define ELECTRON_SHELL_BROWSER_HID_HID_CHOOSER_CONTEXT_H_
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <map>
 | 
					
						
							|  |  |  | #include <set>
 | 
					
						
							|  |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2024-11-04 12:27:49 -06:00
										 |  |  | #include <string_view>
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "base/containers/queue.h"
 | 
					
						
							| 
									
										
										
										
											2023-05-11 16:07:39 -04:00
										 |  |  | #include "base/memory/raw_ptr.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include "base/memory/weak_ptr.h"
 | 
					
						
							|  |  |  | #include "base/observer_list.h"
 | 
					
						
							| 
									
										
										
										
											2024-07-29 12:42:57 -05:00
										 |  |  | #include "base/observer_list_types.h"
 | 
					
						
							| 
									
										
										
										
											2023-10-17 22:54:53 +02:00
										 |  |  | #include "base/scoped_observation_traits.h"
 | 
					
						
							| 
									
										
										
										
											2022-10-17 10:22:24 -04:00
										 |  |  | #include "components/keyed_service/core/keyed_service.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include "mojo/public/cpp/bindings/associated_receiver.h"
 | 
					
						
							|  |  |  | #include "mojo/public/cpp/bindings/remote.h"
 | 
					
						
							|  |  |  | #include "services/device/public/mojom/hid.mojom.h"
 | 
					
						
							|  |  |  | #include "url/origin.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace base { | 
					
						
							|  |  |  | class Value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-29 12:42:57 -05:00
										 |  |  | namespace mojo { | 
					
						
							|  |  |  | template <typename T> | 
					
						
							|  |  |  | class PendingRemote; | 
					
						
							|  |  |  | }  // namespace mojo
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | namespace electron { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-25 04:25:45 -05:00
										 |  |  | class ElectronBrowserContext; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-04 12:27:49 -06:00
										 |  |  | inline constexpr std::string_view kHidDeviceNameKey = "name"; | 
					
						
							|  |  |  | inline constexpr std::string_view kHidGuidKey = "guid"; | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Manages the internal state and connection to the device service for the
 | 
					
						
							|  |  |  | // Human Interface Device (HID) chooser UI.
 | 
					
						
							|  |  |  | class HidChooserContext : public KeyedService, | 
					
						
							|  |  |  |                           public device::mojom::HidManagerClient { | 
					
						
							|  |  |  |  public: | 
					
						
							|  |  |  |   // This observer can be used to be notified when HID devices are connected or
 | 
					
						
							|  |  |  |   // disconnected.
 | 
					
						
							|  |  |  |   class DeviceObserver : public base::CheckedObserver { | 
					
						
							|  |  |  |    public: | 
					
						
							|  |  |  |     virtual void OnDeviceAdded(const device::mojom::HidDeviceInfo&) = 0; | 
					
						
							|  |  |  |     virtual void OnDeviceRemoved(const device::mojom::HidDeviceInfo&) = 0; | 
					
						
							|  |  |  |     virtual void OnDeviceChanged(const device::mojom::HidDeviceInfo&) = 0; | 
					
						
							|  |  |  |     virtual void OnHidManagerConnectionError() = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Called when the HidChooserContext is shutting down. Observers must remove
 | 
					
						
							|  |  |  |     // themselves before returning.
 | 
					
						
							|  |  |  |     virtual void OnHidChooserContextShutdown() = 0; | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   explicit HidChooserContext(ElectronBrowserContext* context); | 
					
						
							|  |  |  |   HidChooserContext(const HidChooserContext&) = delete; | 
					
						
							|  |  |  |   HidChooserContext& operator=(const HidChooserContext&) = delete; | 
					
						
							|  |  |  |   ~HidChooserContext() override; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Returns a human-readable string identifier for |device|.
 | 
					
						
							|  |  |  |   static std::u16string DisplayNameFromDeviceInfo( | 
					
						
							|  |  |  |       const device::mojom::HidDeviceInfo& device); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Returns true if a persistent permission can be granted for |device|.
 | 
					
						
							|  |  |  |   static bool CanStorePersistentEntry( | 
					
						
							|  |  |  |       const device::mojom::HidDeviceInfo& device); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   static base::Value DeviceInfoToValue( | 
					
						
							|  |  |  |       const device::mojom::HidDeviceInfo& device); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // HID-specific interface for granting and checking permissions.
 | 
					
						
							|  |  |  |   void GrantDevicePermission(const url::Origin& origin, | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |                              const device::mojom::HidDeviceInfo& device); | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  |   void RevokeDevicePermission(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
										 |  |  |   bool 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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  |   // Returns true if `origin` is allowed to access FIDO reports.
 | 
					
						
							|  |  |  |   bool IsFidoAllowedForOrigin(const url::Origin& origin); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   // For ScopedObserver.
 | 
					
						
							|  |  |  |   void AddDeviceObserver(DeviceObserver* observer); | 
					
						
							|  |  |  |   void RemoveDeviceObserver(DeviceObserver* observer); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Forward HidManager::GetDevices.
 | 
					
						
							|  |  |  |   void GetDevices(device::mojom::HidManager::GetDevicesCallback callback); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Only call this if you're sure |devices_| has been initialized before-hand.
 | 
					
						
							|  |  |  |   // The returned raw pointer is owned by |devices_| and will be destroyed when
 | 
					
						
							|  |  |  |   // the device is removed.
 | 
					
						
							|  |  |  |   const device::mojom::HidDeviceInfo* GetDeviceInfo(const std::string& guid); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   device::mojom::HidManager* GetHidManager(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   base::WeakPtr<HidChooserContext> AsWeakPtr(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  private: | 
					
						
							|  |  |  |   // device::mojom::HidManagerClient implementation:
 | 
					
						
							|  |  |  |   void DeviceAdded(device::mojom::HidDeviceInfoPtr device_info) override; | 
					
						
							|  |  |  |   void DeviceRemoved(device::mojom::HidDeviceInfoPtr device_info) override; | 
					
						
							|  |  |  |   void DeviceChanged(device::mojom::HidDeviceInfoPtr device_info) override; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   void EnsureHidManagerConnection(); | 
					
						
							|  |  |  |   void SetUpHidManagerConnection( | 
					
						
							|  |  |  |       mojo::PendingRemote<device::mojom::HidManager> manager); | 
					
						
							|  |  |  |   void InitDeviceList(std::vector<device::mojom::HidDeviceInfoPtr> devices); | 
					
						
							|  |  |  |   void OnHidManagerInitializedForTesting( | 
					
						
							|  |  |  |       device::mojom::HidManager::GetDevicesCallback callback, | 
					
						
							|  |  |  |       std::vector<device::mojom::HidDeviceInfoPtr> devices); | 
					
						
							|  |  |  |   void OnHidManagerConnectionError(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  |   // HID-specific interface for revoking device permissions.
 | 
					
						
							|  |  |  |   void RevokePersistentDevicePermission( | 
					
						
							|  |  |  |       const url::Origin& origin, | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |       const device::mojom::HidDeviceInfo& device); | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  |   void RevokeEphemeralDevicePermission( | 
					
						
							|  |  |  |       const url::Origin& origin, | 
					
						
							|  |  |  |       const device::mojom::HidDeviceInfo& device); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-11 16:07:39 -04:00
										 |  |  |   raw_ptr<ElectronBrowserContext> browser_context_; | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   bool is_initialized_ = false; | 
					
						
							|  |  |  |   base::queue<device::mojom::HidManager::GetDevicesCallback> | 
					
						
							|  |  |  |       pending_get_devices_requests_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Tracks the set of devices to which an origin has access to.
 | 
					
						
							|  |  |  |   std::map<url::Origin, std::set<std::string>> ephemeral_devices_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Map from device GUID to device info.
 | 
					
						
							|  |  |  |   std::map<std::string, device::mojom::HidDeviceInfoPtr> devices_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   mojo::Remote<device::mojom::HidManager> hid_manager_; | 
					
						
							|  |  |  |   mojo::AssociatedReceiver<device::mojom::HidManagerClient> client_receiver_{ | 
					
						
							|  |  |  |       this}; | 
					
						
							|  |  |  |   base::ObserverList<DeviceObserver> device_observer_list_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   base::WeakPtrFactory<HidChooserContext> weak_factory_{this}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace electron
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-17 22:54:53 +02:00
										 |  |  | namespace base { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | struct ScopedObservationTraits<electron::HidChooserContext, | 
					
						
							|  |  |  |                                electron::HidChooserContext::DeviceObserver> { | 
					
						
							|  |  |  |   static void AddObserver( | 
					
						
							|  |  |  |       electron::HidChooserContext* source, | 
					
						
							|  |  |  |       electron::HidChooserContext::DeviceObserver* observer) { | 
					
						
							|  |  |  |     source->AddDeviceObserver(observer); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   static void RemoveObserver( | 
					
						
							|  |  |  |       electron::HidChooserContext* source, | 
					
						
							|  |  |  |       electron::HidChooserContext::DeviceObserver* observer) { | 
					
						
							|  |  |  |     source->RemoveDeviceObserver(observer); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace base
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-22 08:34:31 +01:00
										 |  |  | #endif  // ELECTRON_SHELL_BROWSER_HID_HID_CHOOSER_CONTEXT_H_
 |