hid_chooser_context.cc 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. // Copyright 2019 The Chromium Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style license that can be
  3. // found in the LICENSE file.
  4. #include "shell/browser/hid/hid_chooser_context.h"
  5. #include <utility>
  6. #include "base/command_line.h"
  7. #include "base/containers/contains.h"
  8. #include "base/strings/string_number_conversions.h"
  9. #include "base/strings/string_util.h"
  10. #include "base/strings/stringprintf.h"
  11. #include "base/strings/utf_string_conversions.h"
  12. #include "base/values.h"
  13. #include "components/content_settings/core/common/content_settings_types.h"
  14. #include "components/prefs/pref_service.h"
  15. #include "content/public/browser/device_service.h"
  16. #include "electron/grit/electron_resources.h"
  17. #include "services/device/public/cpp/hid/hid_blocklist.h"
  18. #include "services/device/public/cpp/hid/hid_switches.h"
  19. #include "shell/browser/web_contents_permission_helper.h"
  20. #include "ui/base/l10n/l10n_util.h"
  21. namespace electron {
  22. const char kHidDeviceNameKey[] = "name";
  23. const char kHidGuidKey[] = "guid";
  24. const char kHidVendorIdKey[] = "vendorId";
  25. const char kHidProductIdKey[] = "productId";
  26. const char kHidSerialNumberKey[] = "serialNumber";
  27. HidChooserContext::HidChooserContext(ElectronBrowserContext* context)
  28. : browser_context_(context) {}
  29. HidChooserContext::~HidChooserContext() {
  30. // Notify observers that the chooser context is about to be destroyed.
  31. // Observers must remove themselves from the observer lists.
  32. for (auto& observer : device_observer_list_) {
  33. observer.OnHidChooserContextShutdown();
  34. DCHECK(!device_observer_list_.HasObserver(&observer));
  35. }
  36. }
  37. // static
  38. std::u16string HidChooserContext::DisplayNameFromDeviceInfo(
  39. const device::mojom::HidDeviceInfo& device) {
  40. if (device.product_name.empty()) {
  41. auto device_id_string = base::ASCIIToUTF16(
  42. base::StringPrintf("%04X:%04X", device.vendor_id, device.product_id));
  43. return l10n_util::GetStringFUTF16(IDS_HID_CHOOSER_ITEM_WITHOUT_NAME,
  44. device_id_string);
  45. }
  46. return base::UTF8ToUTF16(device.product_name);
  47. }
  48. // static
  49. bool HidChooserContext::CanStorePersistentEntry(
  50. const device::mojom::HidDeviceInfo& device) {
  51. return !device.serial_number.empty() && !device.product_name.empty();
  52. }
  53. // static
  54. base::Value HidChooserContext::DeviceInfoToValue(
  55. const device::mojom::HidDeviceInfo& device) {
  56. base::Value value(base::Value::Type::DICTIONARY);
  57. value.SetStringKey(
  58. kHidDeviceNameKey,
  59. base::UTF16ToUTF8(HidChooserContext::DisplayNameFromDeviceInfo(device)));
  60. value.SetIntKey(kHidVendorIdKey, device.vendor_id);
  61. value.SetIntKey(kHidProductIdKey, device.product_id);
  62. if (HidChooserContext::CanStorePersistentEntry(device)) {
  63. // Use the USB serial number as a persistent identifier. If it is
  64. // unavailable, only ephemeral permissions may be granted.
  65. value.SetStringKey(kHidSerialNumberKey, device.serial_number);
  66. } else {
  67. // The GUID is a temporary ID created on connection that remains valid until
  68. // the device is disconnected. Ephemeral permissions are keyed by this ID
  69. // and must be granted again each time the device is connected.
  70. value.SetStringKey(kHidGuidKey, device.guid);
  71. }
  72. return value;
  73. }
  74. void HidChooserContext::GrantDevicePermission(
  75. const url::Origin& origin,
  76. const device::mojom::HidDeviceInfo& device,
  77. content::RenderFrameHost* render_frame_host) {
  78. DCHECK(base::Contains(devices_, device.guid));
  79. if (CanStorePersistentEntry(device)) {
  80. auto* web_contents =
  81. content::WebContents::FromRenderFrameHost(render_frame_host);
  82. auto* permission_helper =
  83. WebContentsPermissionHelper::FromWebContents(web_contents);
  84. permission_helper->GrantHIDDevicePermission(
  85. origin, DeviceInfoToValue(device), render_frame_host);
  86. } else {
  87. ephemeral_devices_[origin].insert(device.guid);
  88. }
  89. }
  90. bool HidChooserContext::HasDevicePermission(
  91. const url::Origin& origin,
  92. const device::mojom::HidDeviceInfo& device,
  93. content::RenderFrameHost* render_frame_host) {
  94. if (!base::CommandLine::ForCurrentProcess()->HasSwitch(
  95. switches::kDisableHidBlocklist) &&
  96. device::HidBlocklist::IsDeviceExcluded(device))
  97. return false;
  98. auto it = ephemeral_devices_.find(origin);
  99. if (it != ephemeral_devices_.end() &&
  100. base::Contains(it->second, device.guid)) {
  101. return true;
  102. }
  103. auto* web_contents =
  104. content::WebContents::FromRenderFrameHost(render_frame_host);
  105. auto* permission_helper =
  106. WebContentsPermissionHelper::FromWebContents(web_contents);
  107. return permission_helper->CheckHIDDevicePermission(
  108. origin, DeviceInfoToValue(device), render_frame_host);
  109. }
  110. void HidChooserContext::AddDeviceObserver(DeviceObserver* observer) {
  111. EnsureHidManagerConnection();
  112. device_observer_list_.AddObserver(observer);
  113. }
  114. void HidChooserContext::RemoveDeviceObserver(DeviceObserver* observer) {
  115. device_observer_list_.RemoveObserver(observer);
  116. }
  117. void HidChooserContext::GetDevices(
  118. device::mojom::HidManager::GetDevicesCallback callback) {
  119. if (!is_initialized_) {
  120. EnsureHidManagerConnection();
  121. pending_get_devices_requests_.push(std::move(callback));
  122. return;
  123. }
  124. std::vector<device::mojom::HidDeviceInfoPtr> device_list;
  125. device_list.reserve(devices_.size());
  126. for (const auto& pair : devices_)
  127. device_list.push_back(pair.second->Clone());
  128. base::SequencedTaskRunnerHandle::Get()->PostTask(
  129. FROM_HERE, base::BindOnce(std::move(callback), std::move(device_list)));
  130. }
  131. const device::mojom::HidDeviceInfo* HidChooserContext::GetDeviceInfo(
  132. const std::string& guid) {
  133. DCHECK(is_initialized_);
  134. auto it = devices_.find(guid);
  135. return it == devices_.end() ? nullptr : it->second.get();
  136. }
  137. device::mojom::HidManager* HidChooserContext::GetHidManager() {
  138. EnsureHidManagerConnection();
  139. return hid_manager_.get();
  140. }
  141. base::WeakPtr<HidChooserContext> HidChooserContext::AsWeakPtr() {
  142. return weak_factory_.GetWeakPtr();
  143. }
  144. void HidChooserContext::DeviceAdded(device::mojom::HidDeviceInfoPtr device) {
  145. DCHECK(device);
  146. // Update the device list.
  147. if (!base::Contains(devices_, device->guid))
  148. devices_.insert({device->guid, device->Clone()});
  149. // Notify all observers.
  150. for (auto& observer : device_observer_list_)
  151. observer.OnDeviceAdded(*device);
  152. }
  153. void HidChooserContext::DeviceRemoved(device::mojom::HidDeviceInfoPtr device) {
  154. DCHECK(device);
  155. DCHECK(base::Contains(devices_, device->guid));
  156. // Update the device list.
  157. devices_.erase(device->guid);
  158. // Notify all device observers.
  159. for (auto& observer : device_observer_list_)
  160. observer.OnDeviceRemoved(*device);
  161. // Next we'll notify observers for revoked permissions. If the device does not
  162. // support persistent permissions then device permissions are revoked on
  163. // disconnect.
  164. if (CanStorePersistentEntry(*device))
  165. return;
  166. std::vector<url::Origin> revoked_origins;
  167. for (auto& map_entry : ephemeral_devices_) {
  168. if (map_entry.second.erase(device->guid) > 0)
  169. revoked_origins.push_back(map_entry.first);
  170. }
  171. if (revoked_origins.empty())
  172. return;
  173. }
  174. void HidChooserContext::DeviceChanged(device::mojom::HidDeviceInfoPtr device) {
  175. DCHECK(device);
  176. DCHECK(base::Contains(devices_, device->guid));
  177. // Update the device list.
  178. devices_[device->guid] = device->Clone();
  179. // Notify all observers.
  180. for (auto& observer : device_observer_list_)
  181. observer.OnDeviceChanged(*device);
  182. }
  183. void HidChooserContext::EnsureHidManagerConnection() {
  184. if (hid_manager_)
  185. return;
  186. mojo::PendingRemote<device::mojom::HidManager> manager;
  187. content::GetDeviceService().BindHidManager(
  188. manager.InitWithNewPipeAndPassReceiver());
  189. SetUpHidManagerConnection(std::move(manager));
  190. }
  191. void HidChooserContext::SetUpHidManagerConnection(
  192. mojo::PendingRemote<device::mojom::HidManager> manager) {
  193. hid_manager_.Bind(std::move(manager));
  194. hid_manager_.set_disconnect_handler(base::BindOnce(
  195. &HidChooserContext::OnHidManagerConnectionError, base::Unretained(this)));
  196. hid_manager_->GetDevicesAndSetClient(
  197. client_receiver_.BindNewEndpointAndPassRemote(),
  198. base::BindOnce(&HidChooserContext::InitDeviceList,
  199. weak_factory_.GetWeakPtr()));
  200. }
  201. void HidChooserContext::InitDeviceList(
  202. std::vector<device::mojom::HidDeviceInfoPtr> devices) {
  203. for (auto& device : devices)
  204. devices_.insert({device->guid, std::move(device)});
  205. is_initialized_ = true;
  206. while (!pending_get_devices_requests_.empty()) {
  207. std::vector<device::mojom::HidDeviceInfoPtr> device_list;
  208. device_list.reserve(devices.size());
  209. for (const auto& entry : devices_)
  210. device_list.push_back(entry.second->Clone());
  211. std::move(pending_get_devices_requests_.front())
  212. .Run(std::move(device_list));
  213. pending_get_devices_requests_.pop();
  214. }
  215. }
  216. void HidChooserContext::OnHidManagerConnectionError() {
  217. hid_manager_.reset();
  218. client_receiver_.reset();
  219. devices_.clear();
  220. std::vector<url::Origin> revoked_origins;
  221. revoked_origins.reserve(ephemeral_devices_.size());
  222. for (const auto& map_entry : ephemeral_devices_)
  223. revoked_origins.push_back(map_entry.first);
  224. ephemeral_devices_.clear();
  225. // Notify all device observers.
  226. for (auto& observer : device_observer_list_)
  227. observer.OnHidManagerConnectionError();
  228. }
  229. } // namespace electron