1 // Copyright 2014 The Chromium OS 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 
5 #ifndef LIBBRILLO_BRILLO_DBUS_EXPORTED_OBJECT_MANAGER_H_
6 #define LIBBRILLO_BRILLO_DBUS_EXPORTED_OBJECT_MANAGER_H_
7 
8 #include <map>
9 #include <string>
10 #include <vector>
11 
12 #include <base/callback.h>
13 #include <base/memory/weak_ptr.h>
14 #include <brillo/brillo_export.h>
15 #include <brillo/dbus/dbus_object.h>
16 #include <brillo/dbus/exported_property_set.h>
17 #include <brillo/variant_dictionary.h>
18 #include <dbus/bus.h>
19 #include <dbus/exported_object.h>
20 #include <dbus/message.h>
21 #include <dbus/object_path.h>
22 
23 namespace brillo {
24 
25 namespace dbus_utils {
26 
27 // ExportedObjectManager is a delegate that implements the
28 // org.freedesktop.DBus.ObjectManager interface on behalf of another
29 // object. It handles sending signals when new interfaces are added.
30 //
31 // This class is very similar to the ExportedPropertySet class, except that
32 // it allows objects to expose an object manager interface rather than the
33 // properties interface.
34 //
35 //  Example usage:
36 //
37 //   class ExampleObjectManager {
38 //    public:
39 //     ExampleObjectManager(dbus::Bus* bus)
40 //         : object_manager_(bus, "/my/objects/path") { }
41 //
42 //     void RegisterAsync(const CompletionAction& cb) {
43 //          object_manager_.RegisterAsync(cb);
44 //     }
45 //     void ClaimInterface(const dbus::ObjectPath& path,
46 //                         const std::string& interface_name,
47 //                         const ExportedPropertySet::PropertyWriter& writer) {
48 //       object_manager_->ClaimInterface(...);
49 //     }
50 //     void ReleaseInterface(const dbus::ObjectPath& path,
51 //                           const std::string& interface_name) {
52 //       object_manager_->ReleaseInterface(...);
53 //     }
54 //
55 //    private:
56 //     ExportedObjectManager object_manager_;
57 //   };
58 //
59 //   class MyObjectClaimingAnInterface {
60 //    public:
61 //     MyObjectClaimingAnInterface(ExampleObjectManager* object_manager)
62 //       : object_manager_(object_manager) {}
63 //
64 //     void OnInitFinish(bool success) {
65 //       if (!success) { /* handle that */ }
66 //       object_manager_->ClaimInterface(
67 //           my_path_, my_interface_, my_properties_.GetWriter());
68 //     }
69 //
70 //    private:
71 //     struct Properties : public ExportedPropertySet {
72 //      public:
73 //       /* Lots of interesting properties. */
74 //     };
75 //
76 //     Properties my_properties_;
77 //     ExampleObjectManager* object_manager_;
78 //   };
79 class BRILLO_EXPORT ExportedObjectManager
80     : public base::SupportsWeakPtr<ExportedObjectManager> {
81  public:
82   using ObjectMap =
83       std::map<dbus::ObjectPath, std::map<std::string, VariantDictionary>>;
84   using InterfaceProperties =
85       std::map<std::string, ExportedPropertySet::PropertyWriter>;
86 
87   ExportedObjectManager(scoped_refptr<dbus::Bus> bus,
88                         const dbus::ObjectPath& path);
89   virtual ~ExportedObjectManager() = default;
90 
91   // Registers methods implementing the ObjectManager interface on the object
92   // exported on the path given in the constructor. Must be called on the
93   // origin thread.
94   virtual void RegisterAsync(
95       const brillo::dbus_utils::AsyncEventSequencer::CompletionAction&
96           completion_callback);
97 
98   // Trigger a signal that |path| has added an interface |interface_name|
99   // with properties as given by |writer|.
100   virtual void ClaimInterface(
101       const dbus::ObjectPath& path,
102       const std::string& interface_name,
103       const ExportedPropertySet::PropertyWriter& writer);
104 
105   // Trigger a signal that |path| has removed an interface |interface_name|.
106   virtual void ReleaseInterface(const dbus::ObjectPath& path,
107                                 const std::string& interface_name);
108 
GetBus()109   const scoped_refptr<dbus::Bus>& GetBus() const { return bus_; }
110 
111   // Due to D-Bus forwarding, clients may need to access the underlying
112   // DBusObject to handle signals/methods.
113   // TODO(sonnysasaka): Refactor this accessor into a stricter API once we know
114   // what D-Bus forwarding needs when it's completed, without exposing
115   // DBusObject directly.
dbus_object()116   brillo::dbus_utils::DBusObject* dbus_object() { return &dbus_object_; };
117 
118  private:
119   BRILLO_PRIVATE ObjectMap HandleGetManagedObjects();
120 
121   scoped_refptr<dbus::Bus> bus_;
122   brillo::dbus_utils::DBusObject dbus_object_;
123   // Tracks all objects currently known to the ExportedObjectManager.
124   std::map<dbus::ObjectPath, InterfaceProperties> registered_objects_;
125 
126   using SignalInterfacesAdded =
127       DBusSignal<dbus::ObjectPath, std::map<std::string, VariantDictionary>>;
128   using SignalInterfacesRemoved =
129       DBusSignal<dbus::ObjectPath, std::vector<std::string>>;
130 
131   std::weak_ptr<SignalInterfacesAdded> signal_itf_added_;
132   std::weak_ptr<SignalInterfacesRemoved> signal_itf_removed_;
133 
134   friend class ExportedObjectManagerTest;
135   DISALLOW_COPY_AND_ASSIGN(ExportedObjectManager);
136 };
137 
138 }  // namespace dbus_utils
139 
140 }  // namespace brillo
141 
142 #endif  // LIBBRILLO_BRILLO_DBUS_EXPORTED_OBJECT_MANAGER_H_
143