// 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 #include "base/bind.h" #include "base/command_line.h" #include "base/memory/scoped_ptr.h" #include "base/memory/weak_ptr.h" #include "base/message_loop/message_loop.h" #include "base/stl_util.h" #include "base/strings/stringprintf.h" #include "base/values.h" #include "chromeos/chromeos_switches.h" #include "chromeos/dbus/fake_gsm_sms_client.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: explicit GsmSMSClientImpl(dbus::Bus* bus) : bus_(bus), proxies_deleter_(&proxies_) { } // GsmSMSClient override. virtual 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. virtual void ResetSmsReceivedHandler( const std::string& service_name, const dbus::ObjectPath& object_path) OVERRIDE { GetProxy(service_name, object_path)->ResetSmsReceivedHandler(); } // GsmSMSClient override. virtual 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. virtual 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. virtual 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. virtual void RequestUpdate(const std::string& service_name, const dbus::ObjectPath& object_path) OVERRIDE { } private: typedef std::map, SMSProxy*> 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::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. SMSProxy* proxy = new SMSProxy(bus_, service_name, object_path); proxies_.insert(ProxyMap::value_type(key, proxy)); return proxy; } dbus::Bus* bus_; ProxyMap proxies_; STLValueDeleter proxies_deleter_; DISALLOW_COPY_AND_ASSIGN(GsmSMSClientImpl); }; } // namespace //////////////////////////////////////////////////////////////////////////////// // GsmSMSClient GsmSMSClient::GsmSMSClient() {} GsmSMSClient::~GsmSMSClient() {} // static GsmSMSClient* GsmSMSClient::Create(DBusClientImplementationType type, dbus::Bus* bus) { if (type == REAL_DBUS_CLIENT_IMPLEMENTATION) return new GsmSMSClientImpl(bus); DCHECK_EQ(STUB_DBUS_CLIENT_IMPLEMENTATION, type); FakeGsmSMSClient* fake = new FakeGsmSMSClient(); fake->set_sms_test_message_switch_present( CommandLine::ForCurrentProcess()->HasSwitch( chromeos::switches::kSmsTestMessages)); return fake; } } // namespace chromeos