|  | // Copyright 2019 The Chromium Authors | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #ifndef CHROME_BROWSER_SERIAL_SERIAL_CHOOSER_CONTEXT_H_ | 
|  | #define CHROME_BROWSER_SERIAL_SERIAL_CHOOSER_CONTEXT_H_ | 
|  |  | 
|  | #include <map> | 
|  | #include <memory> | 
|  | #include <set> | 
|  | #include <string> | 
|  | #include <utility> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/memory/raw_ptr.h" | 
|  | #include "base/memory/weak_ptr.h" | 
|  | #include "base/observer_list.h" | 
|  | #include "base/scoped_observation.h" | 
|  | #include "base/scoped_observation_traits.h" | 
|  | #include "base/unguessable_token.h" | 
|  | #include "components/permissions/object_permission_context_base.h" | 
|  | #include "content/public/browser/serial_delegate.h" | 
|  | #include "mojo/public/cpp/bindings/pending_remote.h" | 
|  | #include "mojo/public/cpp/bindings/receiver.h" | 
|  | #include "mojo/public/cpp/bindings/remote.h" | 
|  | #include "services/device/public/mojom/serial.mojom-forward.h" | 
|  | #include "third_party/blink/public/mojom/serial/serial.mojom.h" | 
|  | #include "url/origin.h" | 
|  |  | 
|  | class Profile; | 
|  |  | 
|  | namespace base { | 
|  | class Value; | 
|  | } | 
|  |  | 
|  | class SerialChooserContext | 
|  | : public permissions::ObjectPermissionContextBase, | 
|  | public permissions::ObjectPermissionContextBase::PermissionObserver, | 
|  | public device::mojom::SerialPortManagerClient { | 
|  | public: | 
|  | using PortObserver = content::SerialDelegate::Observer; | 
|  |  | 
|  | explicit SerialChooserContext(Profile* profile); | 
|  |  | 
|  | SerialChooserContext(const SerialChooserContext&) = delete; | 
|  | SerialChooserContext& operator=(const SerialChooserContext&) = delete; | 
|  |  | 
|  | ~SerialChooserContext() override; | 
|  |  | 
|  | static base::Value::Dict PortInfoToValue( | 
|  | const device::mojom::SerialPortInfo& port); | 
|  |  | 
|  | // ObjectPermissionContextBase: | 
|  | std::string GetKeyForObject(const base::Value::Dict& object) override; | 
|  | bool IsValidObject(const base::Value::Dict& object) override; | 
|  | std::u16string GetObjectDisplayName(const base::Value::Dict& object) override; | 
|  | // ObjectPermissionContextBase::PermissionObserver: | 
|  | void OnPermissionRevoked(const url::Origin& origin) override; | 
|  |  | 
|  | // In addition these methods from ObjectPermissionContextBase are overridden | 
|  | // in order to expose ephemeral devices through the public interface. | 
|  | std::vector<std::unique_ptr<Object>> GetGrantedObjects( | 
|  | const url::Origin& origin) override; | 
|  | std::vector<std::unique_ptr<Object>> GetAllGrantedObjects() override; | 
|  | void RevokeObjectPermission(const url::Origin& origin, | 
|  | const base::Value::Dict& object) override; | 
|  |  | 
|  | // Serial-specific interface for granting, checking, and revoking permissions. | 
|  | void GrantPortPermission(const url::Origin& origin, | 
|  | const device::mojom::SerialPortInfo& port); | 
|  | bool HasPortPermission(const url::Origin& origin, | 
|  | const device::mojom::SerialPortInfo& port); | 
|  | void RevokePortPermissionWebInitiated(const url::Origin& origin, | 
|  | const base::UnguessableToken& token); | 
|  | static bool CanStorePersistentEntry( | 
|  | const device::mojom::SerialPortInfo& port); | 
|  |  | 
|  | // Only call this if you're sure |port_info_| has been initialized | 
|  | // before-hand. The returned raw pointer is owned by |port_info_| and will be | 
|  | // destroyed when the port is removed. | 
|  | const device::mojom::SerialPortInfo* GetPortInfo( | 
|  | const base::UnguessableToken& token); | 
|  |  | 
|  | device::mojom::SerialPortManager* GetPortManager(); | 
|  |  | 
|  | void AddPortObserver(PortObserver* observer); | 
|  | void RemovePortObserver(PortObserver* observer); | 
|  |  | 
|  | void SetPortManagerForTesting( | 
|  | mojo::PendingRemote<device::mojom::SerialPortManager> manager); | 
|  | void FlushPortManagerConnectionForTesting(); | 
|  | base::WeakPtr<SerialChooserContext> AsWeakPtr(); | 
|  |  | 
|  | // SerialPortManagerClient implementation. | 
|  | void OnPortAdded(device::mojom::SerialPortInfoPtr port) override; | 
|  | void OnPortRemoved(device::mojom::SerialPortInfoPtr port) override; | 
|  | void OnPortConnectedStateChanged( | 
|  | device::mojom::SerialPortInfoPtr port) override; | 
|  |  | 
|  | // KeyedService: | 
|  | void Shutdown() override; | 
|  |  | 
|  | Profile* profile() { return profile_.get(); } | 
|  |  | 
|  | private: | 
|  | void EnsurePortManagerConnection(); | 
|  | void SetUpPortManagerConnection( | 
|  | mojo::PendingRemote<device::mojom::SerialPortManager> manager); | 
|  | void OnGetDevices(std::vector<device::mojom::SerialPortInfoPtr> ports); | 
|  | void OnPortManagerConnectionError(); | 
|  | bool CanApplyPortSpecificPolicy(); | 
|  |  | 
|  | void RevokeObjectPermissionInternal(const url::Origin& origin, | 
|  | const base::Value::Dict& object, | 
|  | bool revoked_by_website); | 
|  |  | 
|  | // This raw pointer is safe because instances of this class are created by | 
|  | // SerialChooserContextFactory as KeyedServices that will be destroyed when | 
|  | // the Profile object is destroyed. | 
|  | const raw_ptr<Profile> profile_; | 
|  |  | 
|  | bool is_initialized_ = false; | 
|  |  | 
|  | // Tracks the set of ports to which an origin has access to. | 
|  | std::map<url::Origin, std::set<base::UnguessableToken>> ephemeral_ports_; | 
|  |  | 
|  | // Map from port token to port info. | 
|  | std::map<base::UnguessableToken, device::mojom::SerialPortInfoPtr> port_info_; | 
|  |  | 
|  | mojo::Remote<device::mojom::SerialPortManager> port_manager_; | 
|  | mojo::Receiver<device::mojom::SerialPortManagerClient> client_receiver_{this}; | 
|  | base::ObserverList<PortObserver> port_observer_list_; | 
|  |  | 
|  | base::ScopedObservation< | 
|  | permissions::ObjectPermissionContextBase, | 
|  | permissions::ObjectPermissionContextBase::PermissionObserver> | 
|  | permission_observation_{this}; | 
|  |  | 
|  | base::WeakPtrFactory<SerialChooserContext> weak_factory_{this}; | 
|  | }; | 
|  |  | 
|  | namespace base { | 
|  |  | 
|  | template <> | 
|  | struct ScopedObservationTraits<SerialChooserContext, | 
|  | SerialChooserContext::PortObserver> { | 
|  | static void AddObserver(SerialChooserContext* source, | 
|  | SerialChooserContext::PortObserver* observer) { | 
|  | source->AddPortObserver(observer); | 
|  | } | 
|  | static void RemoveObserver(SerialChooserContext* source, | 
|  | SerialChooserContext::PortObserver* observer) { | 
|  | source->RemovePortObserver(observer); | 
|  | } | 
|  | }; | 
|  |  | 
|  | }  // namespace base | 
|  |  | 
|  | #endif  // CHROME_BROWSER_SERIAL_SERIAL_CHOOSER_CONTEXT_H_ |