// 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/gsm_sms_client.h" #include #include #include "base/bind.h" #include "base/containers/scoped_ptr_map.h" #include "base/memory/scoped_ptr.h" #include "base/memory/weak_ptr.h" #include "base/message_loop/message_loop.h" #include "base/strings/stringprintf.h" #include "base/values.h" #include "dbus/bus.h" #include "dbus/message.h" #include "dbus/object_proxy.h" #include "dbus/values_util.h" #include "third_party/cros_system_api/dbus/service_constants.h" namespace chromeos { namespace { // A class actually making method calls for SMS services, used by // GsmSMSClientImpl. class SMSProxy { public: typedef GsmSMSClient::SmsReceivedHandler SmsReceivedHandler; typedef GsmSMSClient::DeleteCallback DeleteCallback; typedef GsmSMSClient::GetCallback GetCallback; typedef GsmSMSClient::ListCallback ListCallback; SMSProxy(dbus::Bus* bus, const std::string& service_name, const dbus::ObjectPath& object_path) : proxy_(bus->GetObjectProxy(service_name, object_path)), weak_ptr_factory_(this) { proxy_->ConnectToSignal( modemmanager::kModemManagerSMSInterface, modemmanager::kSMSReceivedSignal, base::Bind(&SMSProxy::OnSmsReceived, weak_ptr_factory_.GetWeakPtr()), base::Bind(&SMSProxy::OnSignalConnected, weak_ptr_factory_.GetWeakPtr())); } // Sets SmsReceived signal handler. void SetSmsReceivedHandler(const SmsReceivedHandler& handler) { DCHECK(sms_received_handler_.is_null()); sms_received_handler_ = handler; } // Resets SmsReceived signal handler. void ResetSmsReceivedHandler() { sms_received_handler_.Reset(); } // Calls Delete method. void Delete(uint32 index, const DeleteCallback& callback) { dbus::MethodCall method_call(modemmanager::kModemManagerSMSInterface, modemmanager::kSMSDeleteFunction); dbus::MessageWriter writer(&method_call); writer.AppendUint32(index); proxy_->CallMethod(&method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, base::Bind(&SMSProxy::OnDelete, weak_ptr_factory_.GetWeakPtr(), callback)); } // Calls Get method. void Get(uint32 index, const GetCallback& callback) { dbus::MethodCall method_call(modemmanager::kModemManagerSMSInterface, modemmanager::kSMSGetFunction); dbus::MessageWriter writer(&method_call); writer.AppendUint32(index); proxy_->CallMethod(&method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, base::Bind(&SMSProxy::OnGet, weak_ptr_factory_.GetWeakPtr(), callback)); } // Calls List method. void List(const ListCallback& callback) { dbus::MethodCall method_call(modemmanager::kModemManagerSMSInterface, modemmanager::kSMSListFunction); proxy_->CallMethod(&method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, base::Bind(&SMSProxy::OnList, weak_ptr_factory_.GetWeakPtr(), callback)); } private: // Handles SmsReceived signal. void OnSmsReceived(dbus::Signal* signal) { uint32 index = 0; bool complete = false; dbus::MessageReader reader(signal); if (!reader.PopUint32(&index) || !reader.PopBool(&complete)) { LOG(ERROR) << "Invalid signal: " << signal->ToString(); return; } if (!sms_received_handler_.is_null()) sms_received_handler_.Run(index, complete); } // Handles the result of signal connection setup. void OnSignalConnected(const std::string& interface, const std::string& signal, bool succeeded) { LOG_IF(ERROR, !succeeded) << "Connect to " << interface << " " << signal << " failed."; } // Handles responses of Delete method calls. void OnDelete(const DeleteCallback& callback, dbus::Response* response) { if (!response) return; callback.Run(); } // Handles responses of Get method calls. void OnGet(const GetCallback& callback, dbus::Response* response) { if (!response) return; dbus::MessageReader reader(response); scoped_ptr value(dbus::PopDataAsValue(&reader)); base::DictionaryValue* dictionary_value = NULL; if (!value.get() || !value->GetAsDictionary(&dictionary_value)) { LOG(WARNING) << "Invalid response: " << response->ToString(); return; } callback.Run(*dictionary_value); } // Handles responses of List method calls. void OnList(const ListCallback& callback, dbus::Response* response) { if (!response) return; dbus::MessageReader reader(response); scoped_ptr value(dbus::PopDataAsValue(&reader)); base::ListValue* list_value = NULL; if (!value.get() || !value->GetAsList(&list_value)) { LOG(WARNING) << "Invalid response: " << response->ToString(); return; } callback.Run(*list_value); } dbus::ObjectProxy* proxy_; SmsReceivedHandler sms_received_handler_; // 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(SMSProxy); }; // The GsmSMSClient implementation. class GsmSMSClientImpl : public GsmSMSClient { public: GsmSMSClientImpl() : bus_(NULL) {} // GsmSMSClient override. void SetSmsReceivedHandler(const std::string& service_name, const dbus::ObjectPath& object_path, const SmsReceivedHandler& handler) override { GetProxy(service_name, object_path)->SetSmsReceivedHandler(handler); } // GsmSMSClient override. void ResetSmsReceivedHandler(const std::string& service_name, const dbus::ObjectPath& object_path) override { GetProxy(service_name, object_path)->ResetSmsReceivedHandler(); } // GsmSMSClient override. void Delete(const std::string& service_name, const dbus::ObjectPath& object_path, uint32 index, const DeleteCallback& callback) override { GetProxy(service_name, object_path)->Delete(index, callback); } // GsmSMSClient override. void Get(const std::string& service_name, const dbus::ObjectPath& object_path, uint32 index, const GetCallback& callback) override { GetProxy(service_name, object_path)->Get(index, callback); } // GsmSMSClient override. void List(const std::string& service_name, const dbus::ObjectPath& object_path, const ListCallback& callback) override { GetProxy(service_name, object_path)->List(callback); } // GsmSMSClient override. void RequestUpdate(const std::string& service_name, const dbus::ObjectPath& object_path) override {} protected: void Init(dbus::Bus* bus) override { bus_ = bus; } private: typedef base::ScopedPtrMap, scoped_ptr> ProxyMap; // Returns a SMSProxy for the given service name and object path. SMSProxy* GetProxy(const std::string& service_name, const dbus::ObjectPath& object_path) { const ProxyMap::key_type key(service_name, object_path.value()); ProxyMap::const_iterator it = proxies_.find(key); if (it != proxies_.end()) return it->second; // There is no proxy for the service_name and object_path, create it. scoped_ptr proxy(new SMSProxy(bus_, service_name, object_path)); SMSProxy* proxy_ptr = proxy.get(); proxies_.insert(key, proxy.Pass()); return proxy_ptr; } dbus::Bus* bus_; ProxyMap proxies_; DISALLOW_COPY_AND_ASSIGN(GsmSMSClientImpl); }; } // namespace //////////////////////////////////////////////////////////////////////////////// // GsmSMSClient GsmSMSClient::GsmSMSClient() {} GsmSMSClient::~GsmSMSClient() {} // static GsmSMSClient* GsmSMSClient::Create() { return new GsmSMSClientImpl(); } } // namespace chromeos