| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  | // Copyright (c) 2020 Microsoft, Inc.
 | 
					
						
							|  |  |  | // Use of this source code is governed by the MIT license that can be
 | 
					
						
							|  |  |  | // found in the LICENSE file.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "shell/browser/serial/serial_chooser_controller.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <utility>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-16 23:08:40 -05:00
										 |  |  | #include "base/containers/contains.h"
 | 
					
						
							| 
									
										
										
										
											2023-02-03 12:43:42 +01:00
										 |  |  | #include "base/functional/bind.h"
 | 
					
						
							| 
									
										
										
										
											2024-07-29 12:42:57 -05:00
										 |  |  | #include "content/public/browser/web_contents.h"
 | 
					
						
							| 
									
										
										
										
											2024-03-28 18:23:13 +01:00
										 |  |  | #include "device/bluetooth/public/cpp/bluetooth_uuid.h"
 | 
					
						
							|  |  |  | #include "services/device/public/cpp/bluetooth/bluetooth_utils.h"
 | 
					
						
							|  |  |  | #include "services/device/public/mojom/serial.mojom.h"
 | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  | #include "shell/browser/api/electron_api_session.h"
 | 
					
						
							| 
									
										
										
										
											2024-07-29 12:42:57 -05:00
										 |  |  | #include "shell/browser/serial/electron_serial_delegate.h"
 | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  | #include "shell/browser/serial/serial_chooser_context.h"
 | 
					
						
							|  |  |  | #include "shell/browser/serial/serial_chooser_context_factory.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_converters/callback_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_converters/content_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_helper/dictionary.h"
 | 
					
						
							| 
									
										
										
										
											2024-07-29 12:42:57 -05:00
										 |  |  | #include "shell/common/gin_helper/promise.h"
 | 
					
						
							| 
									
										
										
										
											2024-11-04 12:58:16 -06:00
										 |  |  | #include "third_party/abseil-cpp/absl/strings/str_format.h"
 | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  | #include "ui/base/l10n/l10n_util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace gin { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | struct Converter<device::mojom::SerialPortInfoPtr> { | 
					
						
							|  |  |  |   static v8::Local<v8::Value> ToV8( | 
					
						
							|  |  |  |       v8::Isolate* isolate, | 
					
						
							|  |  |  |       const device::mojom::SerialPortInfoPtr& port) { | 
					
						
							| 
									
										
										
										
											2023-08-21 03:43:41 +02:00
										 |  |  |     auto dict = gin_helper::Dictionary::CreateEmpty(isolate); | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |     dict.Set("portId", port->token.ToString()); | 
					
						
							|  |  |  |     dict.Set("portName", port->path.BaseName().LossyDisplayName()); | 
					
						
							|  |  |  |     if (port->display_name && !port->display_name->empty()) { | 
					
						
							|  |  |  |       dict.Set("displayName", *port->display_name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (port->has_vendor_id) { | 
					
						
							| 
									
										
										
										
											2024-11-04 12:58:16 -06:00
										 |  |  |       dict.Set("vendorId", absl::StrFormat("%u", port->vendor_id)); | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (port->has_product_id) { | 
					
						
							| 
									
										
										
										
											2024-11-04 12:58:16 -06:00
										 |  |  |       dict.Set("productId", absl::StrFormat("%u", port->product_id)); | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-10-15 18:30:41 -07:00
										 |  |  |     if (port->serial_number && !port->serial_number->empty()) { | 
					
						
							|  |  |  |       dict.Set("serialNumber", *port->serial_number); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-09 18:58:52 -08:00
										 |  |  | #if BUILDFLAG(IS_MAC)
 | 
					
						
							| 
									
										
										
										
											2020-10-15 18:30:41 -07:00
										 |  |  |     if (port->usb_driver_name && !port->usb_driver_name->empty()) { | 
					
						
							|  |  |  |       dict.Set("usbDriverName", *port->usb_driver_name); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-09 18:58:52 -08:00
										 |  |  | #elif BUILDFLAG(IS_WIN)
 | 
					
						
							| 
									
										
										
										
											2020-10-15 18:30:41 -07:00
										 |  |  |     if (!port->device_instance_id.empty()) { | 
					
						
							|  |  |  |       dict.Set("deviceInstanceId", port->device_instance_id); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |     return gin::ConvertToV8(isolate, dict); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace gin
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace electron { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-28 18:23:13 +01:00
										 |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using ::device::mojom::SerialPortType; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool FilterMatchesPort(const blink::mojom::SerialPortFilter& filter, | 
					
						
							|  |  |  |                        const device::mojom::SerialPortInfo& port) { | 
					
						
							|  |  |  |   if (filter.bluetooth_service_class_id) { | 
					
						
							|  |  |  |     if (!port.bluetooth_service_class_id) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return device::BluetoothUUID(*port.bluetooth_service_class_id) == | 
					
						
							|  |  |  |            device::BluetoothUUID(*filter.bluetooth_service_class_id); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (!filter.has_vendor_id) { | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (!port.has_vendor_id || port.vendor_id != filter.vendor_id) { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (!filter.has_product_id) { | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return port.has_product_id && port.product_id == filter.product_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool BluetoothPortIsAllowed( | 
					
						
							|  |  |  |     const std::vector<::device::BluetoothUUID>& allowed_ids, | 
					
						
							|  |  |  |     const device::mojom::SerialPortInfo& port) { | 
					
						
							|  |  |  |   if (!port.bluetooth_service_class_id) { | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   // Serial Port Profile is allowed by default.
 | 
					
						
							|  |  |  |   if (*port.bluetooth_service_class_id == device::GetSerialPortProfileUUID()) { | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return base::Contains(allowed_ids, port.bluetooth_service_class_id.value()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  | SerialChooserController::SerialChooserController( | 
					
						
							|  |  |  |     content::RenderFrameHost* render_frame_host, | 
					
						
							|  |  |  |     std::vector<blink::mojom::SerialPortFilterPtr> filters, | 
					
						
							| 
									
										
										
										
											2024-03-28 18:23:13 +01:00
										 |  |  |     std::vector<::device::BluetoothUUID> allowed_bluetooth_service_class_ids, | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |     content::SerialChooser::Callback callback, | 
					
						
							|  |  |  |     content::WebContents* web_contents, | 
					
						
							|  |  |  |     base::WeakPtr<ElectronSerialDelegate> serial_delegate) | 
					
						
							| 
									
										
										
										
											2024-09-24 00:38:13 -05:00
										 |  |  |     : web_contents_{web_contents ? web_contents->GetWeakPtr() | 
					
						
							|  |  |  |                                  : base::WeakPtr<content::WebContents>()}, | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |       filters_(std::move(filters)), | 
					
						
							| 
									
										
										
										
											2024-03-28 18:23:13 +01:00
										 |  |  |       allowed_bluetooth_service_class_ids_( | 
					
						
							|  |  |  |           std::move(allowed_bluetooth_service_class_ids)), | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |       callback_(std::move(callback)), | 
					
						
							| 
									
										
										
										
											2021-10-06 16:18:00 -04:00
										 |  |  |       serial_delegate_(serial_delegate), | 
					
						
							|  |  |  |       render_frame_host_id_(render_frame_host->GetGlobalId()) { | 
					
						
							| 
									
										
										
										
											2024-09-24 00:38:13 -05:00
										 |  |  |   origin_ = web_contents_->GetPrimaryMainFrame()->GetLastCommittedOrigin(); | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   chooser_context_ = SerialChooserContextFactory::GetForBrowserContext( | 
					
						
							| 
									
										
										
										
											2024-09-24 00:38:13 -05:00
										 |  |  |                          web_contents_->GetBrowserContext()) | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |                          ->AsWeakPtr(); | 
					
						
							|  |  |  |   DCHECK(chooser_context_); | 
					
						
							|  |  |  |   chooser_context_->GetPortManager()->GetDevices(base::BindOnce( | 
					
						
							|  |  |  |       &SerialChooserController::OnGetDevices, weak_factory_.GetWeakPtr())); | 
					
						
							| 
									
										
										
										
											2022-07-25 10:50:19 -04:00
										 |  |  |   observation_.Observe(chooser_context_.get()); | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SerialChooserController::~SerialChooserController() { | 
					
						
							|  |  |  |   RunCallback(/*port=*/nullptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | api::Session* SerialChooserController::GetSession() { | 
					
						
							| 
									
										
										
										
											2024-09-24 00:38:13 -05:00
										 |  |  |   if (!web_contents_) { | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |     return nullptr; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2024-09-24 00:38:13 -05:00
										 |  |  |   return api::Session::FromBrowserContext(web_contents_->GetBrowserContext()); | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SerialChooserController::OnPortAdded( | 
					
						
							|  |  |  |     const device::mojom::SerialPortInfo& port) { | 
					
						
							| 
									
										
										
										
											2024-03-28 18:23:13 +01:00
										 |  |  |   if (!DisplayDevice(port)) | 
					
						
							| 
									
										
										
										
											2024-03-20 15:18:41 +01:00
										 |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |   ports_.push_back(port.Clone()); | 
					
						
							| 
									
										
										
										
											2024-03-28 18:23:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |   api::Session* session = GetSession(); | 
					
						
							|  |  |  |   if (session) { | 
					
						
							| 
									
										
										
										
											2024-09-24 00:38:13 -05:00
										 |  |  |     session->Emit("serial-port-added", port.Clone(), web_contents_.get()); | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SerialChooserController::OnPortRemoved( | 
					
						
							|  |  |  |     const device::mojom::SerialPortInfo& port) { | 
					
						
							| 
									
										
										
										
											2024-08-02 21:21:59 -05:00
										 |  |  |   const auto it = std::ranges::find(ports_, port.token, | 
					
						
							|  |  |  |                                     &device::mojom::SerialPortInfo::token); | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |   if (it != ports_.end()) { | 
					
						
							| 
									
										
										
										
											2024-09-24 00:38:13 -05:00
										 |  |  |     if (api::Session* session = GetSession()) | 
					
						
							|  |  |  |       session->Emit("serial-port-removed", port.Clone(), web_contents_.get()); | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |     ports_.erase(it); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-17 12:48:40 -04:00
										 |  |  | void SerialChooserController::OnPortManagerConnectionError() { | 
					
						
							| 
									
										
										
										
											2022-07-25 10:50:19 -04:00
										 |  |  |   observation_.Reset(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SerialChooserController::OnSerialChooserContextShutdown() { | 
					
						
							|  |  |  |   observation_.Reset(); | 
					
						
							| 
									
										
										
										
											2022-05-17 12:48:40 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  | void SerialChooserController::OnDeviceChosen(const std::string& port_id) { | 
					
						
							|  |  |  |   if (port_id.empty()) { | 
					
						
							|  |  |  |     RunCallback(/*port=*/nullptr); | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2024-08-26 09:58:32 -05:00
										 |  |  |     const auto it = std::ranges::find_if(ports_, [&port_id](const auto& ptr) { | 
					
						
							|  |  |  |       return ptr->token.ToString() == port_id; | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2021-04-12 13:18:39 +00:00
										 |  |  |     if (it != ports_.end()) { | 
					
						
							| 
									
										
										
										
											2021-10-06 16:18:00 -04:00
										 |  |  |       auto* rfh = content::RenderFrameHost::FromID(render_frame_host_id_); | 
					
						
							|  |  |  |       chooser_context_->GrantPortPermission(origin_, *it->get(), rfh); | 
					
						
							| 
									
										
										
										
											2021-04-12 13:18:39 +00:00
										 |  |  |       RunCallback(it->Clone()); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       RunCallback(/*port=*/nullptr); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SerialChooserController::OnGetDevices( | 
					
						
							|  |  |  |     std::vector<device::mojom::SerialPortInfoPtr> ports) { | 
					
						
							|  |  |  |   // Sort ports by file paths.
 | 
					
						
							| 
									
										
										
										
											2024-08-26 09:58:32 -05:00
										 |  |  |   std::ranges::sort(ports, [](const auto& port1, const auto& port2) { | 
					
						
							|  |  |  |     return port1->path.BaseName() < port2->path.BaseName(); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   for (auto& port : ports) { | 
					
						
							| 
									
										
										
										
											2024-03-28 18:23:13 +01:00
										 |  |  |     if (DisplayDevice(*port)) | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |       ports_.push_back(std::move(port)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool prevent_default = false; | 
					
						
							| 
									
										
										
										
											2024-09-24 00:38:13 -05:00
										 |  |  |   if (api::Session* session = GetSession()) { | 
					
						
							| 
									
										
										
										
											2023-05-18 23:01:44 +02:00
										 |  |  |     prevent_default = session->Emit( | 
					
						
							| 
									
										
										
										
											2024-09-24 00:38:13 -05:00
										 |  |  |         "select-serial-port", ports_, web_contents_.get(), | 
					
						
							| 
									
										
										
										
											2023-05-18 23:01:44 +02:00
										 |  |  |         base::BindRepeating(&SerialChooserController::OnDeviceChosen, | 
					
						
							|  |  |  |                             weak_factory_.GetWeakPtr())); | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (!prevent_default) { | 
					
						
							|  |  |  |     RunCallback(/*port=*/nullptr); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-28 18:23:13 +01:00
										 |  |  | bool SerialChooserController::DisplayDevice( | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |     const device::mojom::SerialPortInfo& port) const { | 
					
						
							| 
									
										
										
										
											2024-03-28 18:23:13 +01:00
										 |  |  |   if (filters_.empty()) { | 
					
						
							|  |  |  |     return BluetoothPortIsAllowed(allowed_bluetooth_service_class_ids_, port); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   for (const auto& filter : filters_) { | 
					
						
							| 
									
										
										
										
											2024-03-28 18:23:13 +01:00
										 |  |  |     if (FilterMatchesPort(*filter, port) && | 
					
						
							|  |  |  |         BluetoothPortIsAllowed(allowed_bluetooth_service_class_ids_, port)) { | 
					
						
							|  |  |  |       return true; | 
					
						
							| 
									
										
										
										
											2020-09-28 12:22:03 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SerialChooserController::RunCallback( | 
					
						
							|  |  |  |     device::mojom::SerialPortInfoPtr port) { | 
					
						
							|  |  |  |   if (callback_) { | 
					
						
							|  |  |  |     std::move(callback_).Run(std::move(port)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace electron
 |