// Copyright (c) 2012 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 "chromeos/dbus/introspectable_client.h" #include #include #include "base/bind.h" #include "base/logging.h" #include "dbus/bus.h" #include "dbus/message.h" #include "dbus/object_path.h" #include "dbus/object_proxy.h" #include "third_party/libxml/chromium/libxml_utils.h" namespace { // D-Bus specification constants. const char kIntrospectableInterface[] = "org.freedesktop.DBus.Introspectable"; const char kIntrospect[] = "Introspect"; // String constants used for parsing D-Bus Introspection XML data. const char kInterfaceNode[] = "interface"; const char kInterfaceNameAttribute[] = "name"; } // namespace namespace chromeos { // The IntrospectableClient implementation used in production. class IntrospectableClientImpl : public IntrospectableClient { public: explicit IntrospectableClientImpl(dbus::Bus* bus) : bus_(bus), weak_ptr_factory_(this) { DVLOG(1) << "Creating IntrospectableClientImpl"; } virtual ~IntrospectableClientImpl() { } // IntrospectableClient override. virtual void Introspect(const std::string& service_name, const dbus::ObjectPath& object_path, const IntrospectCallback& callback) OVERRIDE { dbus::MethodCall method_call(kIntrospectableInterface, kIntrospect); dbus::ObjectProxy* object_proxy = bus_->GetObjectProxy(service_name, object_path); object_proxy->CallMethod( &method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, base::Bind(&IntrospectableClientImpl::OnIntrospect, weak_ptr_factory_.GetWeakPtr(), service_name, object_path, callback)); } private: // Called by dbus:: when a response for Introspect() is recieved. void OnIntrospect(const std::string& service_name, const dbus::ObjectPath& object_path, const IntrospectCallback& callback, dbus::Response* response) { // Parse response. bool success = false; std::string xml_data; if (response != NULL) { dbus::MessageReader reader(response); if (!reader.PopString(&xml_data)) { LOG(WARNING) << "Introspect response has incorrect paramters: " << response->ToString(); } else { success = true; } } // Notify client. callback.Run(service_name, object_path, xml_data, success); } dbus::Bus* bus_; // Weak pointer factory for generating 'this' pointers that might live longer // than we do. // Note: This should remain the last member so it'll be destroyed and // invalidate its weak pointers before any other members are destroyed. base::WeakPtrFactory weak_ptr_factory_; DISALLOW_COPY_AND_ASSIGN(IntrospectableClientImpl); }; // The IntrospectableClient implementation used on Linux desktop, which does // nothing. class IntrospectableClientStubImpl : public IntrospectableClient { public: // IntrospectableClient override. virtual void Introspect(const std::string& service_name, const dbus::ObjectPath& object_path, const IntrospectCallback& callback) OVERRIDE { VLOG(1) << "Introspect: " << service_name << " " << object_path.value(); callback.Run(service_name, object_path, "", false); } }; IntrospectableClient::IntrospectableClient() { } IntrospectableClient::~IntrospectableClient() { } // static std::vector IntrospectableClient::GetInterfacesFromIntrospectResult( const std::string& xml_data) { std::vector interfaces; XmlReader reader; if (!reader.Load(xml_data)) return interfaces; do { // Skip to the next open tag, exit when done. while (!reader.SkipToElement()) { if (!reader.Read()) { return interfaces; } } // Only look at interface nodes. if (reader.NodeName() != kInterfaceNode) continue; // Skip if missing the interface name. std::string interface_name; if (!reader.NodeAttribute(kInterfaceNameAttribute, &interface_name)) continue; interfaces.push_back(interface_name); } while (reader.Read()); return interfaces; } // static IntrospectableClient* IntrospectableClient::Create( DBusClientImplementationType type, dbus::Bus* bus) { if (type == REAL_DBUS_CLIENT_IMPLEMENTATION) return new IntrospectableClientImpl(bus); DCHECK_EQ(STUB_DBUS_CLIENT_IMPLEMENTATION, type); return new IntrospectableClientStubImpl(); } } // namespace chromeos