• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2013 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 
5 #ifndef DBUS_OBJECT_MANAGER_H_
6 #define DBUS_OBJECT_MANAGER_H_
7 
8 #include <map>
9 
10 #include "base/memory/ref_counted.h"
11 #include "base/memory/weak_ptr.h"
12 #include "dbus/object_path.h"
13 #include "dbus/property.h"
14 
15 // Newer D-Bus services implement the Object Manager interface to inform other
16 // clients about the objects they export, the properties of those objects, and
17 // notification of changes in the set of available objects:
18 //     http://dbus.freedesktop.org/doc/dbus-specification.html
19 //       #standard-interfaces-objectmanager
20 //
21 // This interface is very closely tied to the Properties interface, and uses
22 // even more levels of nested dictionaries and variants. In addition to
23 // simplifying implementation, since there tends to be a single object manager
24 // per service, spanning the complete set of objects an interfaces available,
25 // the classes implemented here make dealing with this interface simpler.
26 //
27 // Except where noted, use of this class replaces the need for the code
28 // documented in dbus/property.h
29 //
30 // Client implementation classes should begin by deriving from the
31 // dbus::ObjectManager::Interface class, and defining a Properties structure as
32 // documented in dbus/property.h.
33 //
34 // Example:
35 //   class ExampleClient : public dbus::ObjectManager::Interface {
36 //    public:
37 //     struct Properties : public dbus::PropertySet {
38 //       dbus::Property<std::string> name;
39 //       dbus::Property<uint16> version;
40 //       dbus::Property<dbus::ObjectPath> parent;
41 //       dbus::Property<std::vector<std::string> > children;
42 //
43 //       Properties(dbus::ObjectProxy* object_proxy,
44 //                  const PropertyChangedCallback callback)
45 //           : dbus::PropertySet(object_proxy, kExampleInterface, callback) {
46 //         RegisterProperty("Name", &name);
47 //         RegisterProperty("Version", &version);
48 //         RegisterProperty("Parent", &parent);
49 //         RegisterProperty("Children", &children);
50 //       }
51 //       virtual ~Properties() {}
52 //     };
53 //
54 // The link between the implementation class and the object manager is set up
55 // in the constructor and removed in the destructor; the class should maintain
56 // a pointer to its object manager for use in other methods and establish
57 // itself as the implementation class for its interface.
58 //
59 // Example:
60 //   explicit ExampleClient::ExampleClient(dbus::Bus* bus)
61 //       : bus_(bus),
62 //         weak_ptr_factory_(this) {
63 //     object_manager_ = bus_->GetObjectManager(kServiceName, kManagerPath);
64 //     object_manager_->RegisterInterface(kInterface, this);
65 //   }
66 //
67 //   virtual ExampleClient::~ExampleClient() {
68 //     object_manager_->UnregisterInterface(kInterface);
69 //   }
70 //
71 // The D-Bus thread manager takes care of issuing the necessary call to
72 // GetManagedObjects() after the implementation classes have been set up.
73 //
74 // The object manager interface class has one abstract method that must be
75 // implemented by the class to create Properties structures on demand. As well
76 // as implementing this, you will want to implement a public GetProperties()
77 // method.
78 //
79 // Example:
80 //   dbus::PropertySet* CreateProperties(dbus::ObjectProxy* object_proxy,
81 //                                       const std::string& interface_name)
82 //       OVERRIDE {
83 //     Properties* properties = new Properties(
84 //           object_proxy, interface_name,
85 //           base::Bind(&PropertyChanged,
86 //                      weak_ptr_factory_.GetWeakPtr(),
87 //                      object_path));
88 //     return static_cast<dbus::PropertySet*>(properties);
89 //   }
90 //
91 //   Properties* GetProperties(const dbus::ObjectPath& object_path) {
92 //     return static_cast<Properties*>(
93 //         object_manager_->GetProperties(object_path, kInterface));
94 //   }
95 //
96 // Note that unlike classes that only use dbus/property.h there is no need
97 // to connect signals or obtain the initial values of properties. The object
98 // manager class handles that for you.
99 //
100 // PropertyChanged is a method of your own to notify your observers of a change
101 // in your properties, either as a result of a signal from the Properties
102 // interface or from the Object Manager interface. You may also wish to
103 // implement the optional ObjectAdded and ObjectRemoved methods of the class
104 // to likewise notify observers.
105 //
106 // When your class needs an object proxy for a given object path, it may
107 // obtain it from the object manager. Unlike the equivalent method on the bus
108 // this will return NULL if the object is not known.
109 //
110 //   object_proxy = object_manager_->GetObjectProxy(object_path);
111 //   if (object_proxy) {
112 //     ...
113 //   }
114 //
115 // There is no need for code using your implementation class to be aware of the
116 // use of object manager behind the scenes, the rules for updating properties
117 // documented in dbus/property.h still apply.
118 
119 namespace dbus {
120 
121 const char kObjectManagerInterface[] = "org.freedesktop.DBus.ObjectManager";
122 const char kObjectManagerGetManagedObjects[] = "GetManagedObjects";
123 const char kObjectManagerInterfacesAdded[] = "InterfacesAdded";
124 const char kObjectManagerInterfacesRemoved[] = "InterfacesRemoved";
125 
126 class Bus;
127 class MessageReader;
128 class ObjectProxy;
129 class Response;
130 class Signal;
131 
132 // ObjectManager implements both the D-Bus client components of the D-Bus
133 // Object Manager interface, as internal methods, and a public API for
134 // client classes to utilize.
135 class CHROME_DBUS_EXPORT ObjectManager
136     : public base::RefCountedThreadSafe<ObjectManager> {
137 public:
138   // ObjectManager::Interface must be implemented by any class wishing to have
139   // its remote objects managed by an ObjectManager.
140   class Interface {
141    public:
~Interface()142     virtual ~Interface() {}
143 
144     // Called by ObjectManager to create a Properties structure for the remote
145     // D-Bus object identified by |object_path| and accessibile through
146     // |object_proxy|. The D-Bus interface name |interface_name| is that passed
147     // to RegisterInterface() by the implementation class.
148     //
149     // The implementation class should create and return an instance of its own
150     // subclass of dbus::PropertySet; ObjectManager will then connect signals
151     // and update the properties from its own internal message reader.
152     virtual PropertySet* CreateProperties(
153         ObjectProxy *object_proxy,
154         const dbus::ObjectPath& object_path,
155         const std::string& interface_name) = 0;
156 
157     // Called by ObjectManager to inform the implementation class that an
158     // object has been added with the path |object_path|. The D-Bus interface
159     // name |interface_name| is that passed to RegisterInterface() by the
160     // implementation class.
161     //
162     // If a new object implements multiple interfaces, this method will be
163     // called on each interface implementation with differing values of
164     // |interface_name| as appropriate. An implementation class will only
165     // receive multiple calls if it has registered for multiple interfaces.
ObjectAdded(const ObjectPath & object_path,const std::string & interface_name)166     virtual void ObjectAdded(const ObjectPath& object_path,
167                              const std::string& interface_name) { }
168 
169     // Called by ObjectManager to inform the implementation class than an
170     // object with the path |object_path| has been removed. Ths D-Bus interface
171     // name |interface_name| is that passed to RegisterInterface() by the
172     // implementation class. Multiple interfaces are handled as with
173     // ObjectAdded().
174     //
175     // This method will be called before the Properties structure and the
176     // ObjectProxy object for the given interface are cleaned up, it is safe
177     // to retrieve them during removal to vary processing.
ObjectRemoved(const ObjectPath & object_path,const std::string & interface_name)178     virtual void ObjectRemoved(const ObjectPath& object_path,
179                                const std::string& interface_name) { }
180   };
181 
182   // Client code should use Bus::GetObjectManager() instead of this constructor.
183   ObjectManager(Bus* bus,
184                 const std::string& service_name,
185                 const ObjectPath& object_path);
186 
187   // Register a client implementation class |interface| for the given D-Bus
188   // interface named in |interface_name|. That object's CreateProperties()
189   // method will be used to create instances of dbus::PropertySet* when
190   // required.
191   void RegisterInterface(const std::string& interface_name,
192                          Interface* interface);
193 
194   // Unregister the implementation class for the D-Bus interface named in
195   // |interface_name|, objects and properties of this interface will be
196   // ignored.
197   void UnregisterInterface(const std::string& interface_name);
198 
199   // Returns a list of object paths, in an undefined order, of objects known
200   // to this manager.
201   std::vector<ObjectPath> GetObjects();
202 
203   // Returns the list of object paths, in an undefined order, of objects
204   // implementing the interface named in |interface_name| known to this manager.
205   std::vector<ObjectPath> GetObjectsWithInterface(
206       const std::string& interface_name);
207 
208   // Returns a ObjectProxy pointer for the given |object_path|. Unlike
209   // the equivalent method on Bus this will return NULL if the object
210   // manager has not been informed of that object's existance.
211   ObjectProxy* GetObjectProxy(const ObjectPath& object_path);
212 
213   // Returns a PropertySet* pointer for the given |object_path| and
214   // |interface_name|, or NULL if the object manager has not been informed of
215   // that object's existance or the interface's properties. The caller should
216   // cast the returned pointer to the appropriate type, e.g.:
217   //   static_cast<Properties*>(GetProperties(object_path, my_interface));
218   PropertySet* GetProperties(const ObjectPath& object_path,
219                              const std::string& interface_name);
220 
221   // Instructs the object manager to refresh its list of managed objects;
222   // automatically called by the D-Bus thread manager, there should never be
223   // a need to call this manually.
224   void GetManagedObjects();
225 
226   // Cleans up any match rules and filter functions added by this ObjectManager.
227   // The Bus object will take care of this so you don't have to do it manually.
228   //
229   // BLOCKING CALL.
230   void CleanUp();
231 
232  protected:
233   virtual ~ObjectManager();
234 
235  private:
236   friend class base::RefCountedThreadSafe<ObjectManager>;
237 
238   // Connects the InterfacesAdded and InterfacesRemoved signals and calls
239   // GetManagedObjects. Called from OnSetupMatchRuleAndFilterComplete.
240   void InitializeObjects();
241 
242   // Called from the constructor to add a match rule for PropertiesChanged
243   // signals on the DBus thread and set up a corresponding filter function.
244   bool SetupMatchRuleAndFilter();
245 
246   // Called on the origin thread once the match rule and filter have been set
247   // up. |success| is false, if an error occurred during set up; it's true
248   // otherwise.
249   void OnSetupMatchRuleAndFilterComplete(bool success);
250 
251   // Called by dbus:: when a message is received. This is used to filter
252   // PropertiesChanged signals from the correct sender and relay the event to
253   // the correct PropertySet.
254   static DBusHandlerResult HandleMessageThunk(DBusConnection* connection,
255                                               DBusMessage* raw_message,
256                                               void* user_data);
257   DBusHandlerResult HandleMessage(DBusConnection* connection,
258                                   DBusMessage* raw_message);
259 
260   // Called when a PropertiesChanged signal is received from the sender.
261   // This method notifies the relevant PropertySet that it should update its
262   // properties based on the received signal. Called from HandleMessage.
263   void NotifyPropertiesChanged(const dbus::ObjectPath object_path,
264                                Signal* signal);
265   void NotifyPropertiesChangedHelper(const dbus::ObjectPath object_path,
266                                      Signal* signal);
267 
268   // Called by dbus:: in response to the GetManagedObjects() method call.
269   void OnGetManagedObjects(Response* response);
270 
271   // Called by dbus:: when an InterfacesAdded signal is received and initially
272   // connected.
273   void InterfacesAddedReceived(Signal* signal);
274   void InterfacesAddedConnected(const std::string& interface_name,
275                                 const std::string& signal_name,
276                                 bool success);
277 
278   // Called by dbus:: when an InterfacesRemoved signal is received and
279   // initially connected.
280   void InterfacesRemovedReceived(Signal* signal);
281   void InterfacesRemovedConnected(const std::string& interface_name,
282                                   const std::string& signal_name,
283                                   bool success);
284 
285   // Updates the map entry for the object with path |object_path| using the
286   // D-Bus message in |reader|, which should consist of an dictionary mapping
287   // interface names to properties dictionaries as recieved by both the
288   // GetManagedObjects() method return and the InterfacesAdded() signal.
289   void UpdateObject(const ObjectPath& object_path, MessageReader* reader);
290 
291   // Updates the properties structure of the object with path |object_path|
292   // for the interface named |interface_name| using the D-Bus message in
293   // |reader| which should consist of the properties dictionary for that
294   // interface.
295   //
296   // Called by UpdateObjects() for each interface in the dictionary; this
297   // method takes care of both creating the entry in the ObjectMap and
298   // ObjectProxy if required, as well as the PropertySet instance for that
299   // interface if necessary.
300   void AddInterface(const ObjectPath& object_path,
301                     const std::string& interface_name,
302                     MessageReader* reader);
303 
304   // Removes the properties structure of the object with path |object_path|
305   // for the interfaces named |interface_name|.
306   //
307   // If no further interfaces remain, the entry in the ObjectMap is discarded.
308   void RemoveInterface(const ObjectPath& object_path,
309                        const std::string& interface_name);
310 
311   // Removes all objects and interfaces from the object manager when
312   // |old_owner| is not the empty string and/or re-requests the set of managed
313   // objects when |new_owner| is not the empty string.
314   void NameOwnerChanged(const std::string& old_owner,
315                         const std::string& new_owner);
316 
317   Bus* bus_;
318   std::string service_name_;
319   std::string service_name_owner_;
320   std::string match_rule_;
321   ObjectPath object_path_;
322   ObjectProxy* object_proxy_;
323   bool setup_success_;
324   bool cleanup_called_;
325 
326   // Maps the name of an interface to the implementation class used for
327   // instantiating PropertySet structures for that interface's properties.
328   typedef std::map<std::string, Interface*> InterfaceMap;
329   InterfaceMap interface_map_;
330 
331   // Each managed object consists of a ObjectProxy used to make calls
332   // against that object and a collection of D-Bus interface names and their
333   // associated PropertySet structures.
334   struct Object {
335     Object();
336     ~Object();
337 
338     ObjectProxy* object_proxy;
339 
340     // Maps the name of an interface to the specific PropertySet structure
341     // of that interface's properties.
342     typedef std::map<const std::string, PropertySet*> PropertiesMap;
343     PropertiesMap properties_map;
344   };
345 
346   // Maps the object path of an object to the Object structure.
347   typedef std::map<const ObjectPath, Object*> ObjectMap;
348   ObjectMap object_map_;
349 
350   // Weak pointer factory for generating 'this' pointers that might live longer
351   // than we do.
352   // Note: This should remain the last member so it'll be destroyed and
353   // invalidate its weak pointers before any other members are destroyed.
354   base::WeakPtrFactory<ObjectManager> weak_ptr_factory_;
355 
356   DISALLOW_COPY_AND_ASSIGN(ObjectManager);
357 };
358 
359 }  // namespace dbus
360 
361 #endif  // DBUS_OBJECT_MANAGER_H_
362