123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272 |
- // 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"
- #include <utility>
- #include "base/command_line.h"
- #include "base/containers/contains.h"
- #include "base/strings/string_number_conversions.h"
- #include "base/strings/string_util.h"
- #include "base/strings/stringprintf.h"
- #include "base/strings/utf_string_conversions.h"
- #include "base/values.h"
- #include "components/content_settings/core/common/content_settings_types.h"
- #include "components/prefs/pref_service.h"
- #include "content/public/browser/device_service.h"
- #include "electron/grit/electron_resources.h"
- #include "services/device/public/cpp/hid/hid_blocklist.h"
- #include "services/device/public/cpp/hid/hid_switches.h"
- #include "shell/browser/web_contents_permission_helper.h"
- #include "ui/base/l10n/l10n_util.h"
- namespace electron {
- const char kHidDeviceNameKey[] = "name";
- const char kHidGuidKey[] = "guid";
- const char kHidVendorIdKey[] = "vendorId";
- const char kHidProductIdKey[] = "productId";
- const char kHidSerialNumberKey[] = "serialNumber";
- 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(
- base::StringPrintf("%04X:%04X", device.vendor_id, device.product_id));
- 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) {
- base::Value value(base::Value::Type::DICTIONARY);
- value.SetStringKey(
- kHidDeviceNameKey,
- base::UTF16ToUTF8(HidChooserContext::DisplayNameFromDeviceInfo(device)));
- value.SetIntKey(kHidVendorIdKey, device.vendor_id);
- value.SetIntKey(kHidProductIdKey, device.product_id);
- if (HidChooserContext::CanStorePersistentEntry(device)) {
- // Use the USB serial number as a persistent identifier. If it is
- // unavailable, only ephemeral permissions may be granted.
- value.SetStringKey(kHidSerialNumberKey, device.serial_number);
- } 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.
- value.SetStringKey(kHidGuidKey, device.guid);
- }
- return value;
- }
- void HidChooserContext::GrantDevicePermission(
- const url::Origin& origin,
- const device::mojom::HidDeviceInfo& device,
- content::RenderFrameHost* render_frame_host) {
- DCHECK(base::Contains(devices_, device.guid));
- if (CanStorePersistentEntry(device)) {
- auto* web_contents =
- content::WebContents::FromRenderFrameHost(render_frame_host);
- auto* permission_helper =
- WebContentsPermissionHelper::FromWebContents(web_contents);
- permission_helper->GrantHIDDevicePermission(
- origin, DeviceInfoToValue(device), render_frame_host);
- } else {
- ephemeral_devices_[origin].insert(device.guid);
- }
- }
- bool HidChooserContext::HasDevicePermission(
- const url::Origin& origin,
- const device::mojom::HidDeviceInfo& device,
- content::RenderFrameHost* render_frame_host) {
- if (!base::CommandLine::ForCurrentProcess()->HasSwitch(
- switches::kDisableHidBlocklist) &&
- device::HidBlocklist::IsDeviceExcluded(device))
- return false;
- auto it = ephemeral_devices_.find(origin);
- if (it != ephemeral_devices_.end() &&
- base::Contains(it->second, device.guid)) {
- return true;
- }
- auto* web_contents =
- content::WebContents::FromRenderFrameHost(render_frame_host);
- auto* permission_helper =
- WebContentsPermissionHelper::FromWebContents(web_contents);
- return permission_helper->CheckHIDDevicePermission(
- origin, DeviceInfoToValue(device), render_frame_host);
- }
- 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());
- base::SequencedTaskRunnerHandle::Get()->PostTask(
- 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.
- if (!base::Contains(devices_, device->guid))
- devices_.insert({device->guid, device->Clone()});
- // Notify all observers.
- for (auto& observer : device_observer_list_)
- observer.OnDeviceAdded(*device);
- }
- void HidChooserContext::DeviceRemoved(device::mojom::HidDeviceInfoPtr device) {
- DCHECK(device);
- DCHECK(base::Contains(devices_, device->guid));
- // Update the device list.
- devices_.erase(device->guid);
- // Notify all device observers.
- for (auto& observer : device_observer_list_)
- observer.OnDeviceRemoved(*device);
- // 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;
- std::vector<url::Origin> revoked_origins;
- for (auto& map_entry : ephemeral_devices_) {
- if (map_entry.second.erase(device->guid) > 0)
- revoked_origins.push_back(map_entry.first);
- }
- if (revoked_origins.empty())
- return;
- }
- void HidChooserContext::DeviceChanged(device::mojom::HidDeviceInfoPtr device) {
- DCHECK(device);
- DCHECK(base::Contains(devices_, device->guid));
- // Update the device list.
- devices_[device->guid] = device->Clone();
- // Notify all observers.
- for (auto& observer : device_observer_list_)
- observer.OnDeviceChanged(*device);
- }
- 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)
- devices_.insert({device->guid, std::move(device)});
- 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();
- std::vector<url::Origin> revoked_origins;
- revoked_origins.reserve(ephemeral_devices_.size());
- for (const auto& map_entry : ephemeral_devices_)
- revoked_origins.push_back(map_entry.first);
- ephemeral_devices_.clear();
- // Notify all device observers.
- for (auto& observer : device_observer_list_)
- observer.OnHidManagerConnectionError();
- }
- } // namespace electron
|