blob: 0fc6cf112c3530f6e2102ea5e2d10f1e7e06ed9b [file] [log] [blame]
// 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/sms_client.h"
#include <map>
#include <memory>
#include <utility>
#include "base/bind.h"
#include "base/location.h"
#include "base/macros.h"
#include "base/memory/weak_ptr.h"
#include "base/stl_util.h"
#include "base/strings/stringprintf.h"
#include "base/threading/thread_task_runner_handle.h"
#include "base/values.h"
#include "dbus/bus.h"
#include "dbus/message.h"
#include "dbus/object_proxy.h"
#include "dbus/property.h"
#include "dbus/values_util.h"
#include "third_party/cros_system_api/dbus/service_constants.h"
namespace chromeos {
namespace {
// See "enum MMSMSState" definition in ModemManager.
constexpr uint32_t kSMSStateReceived = 3; // MM_SMS_STATE_RECEIVED
class SMSReceiveHandler {
public:
SMSReceiveHandler(dbus::ObjectProxy* object_proxy,
SMSClient::GetAllCallback callback)
: callback_(std::move(callback)),
sms_received_(false),
weak_ptr_factory_(this) {
property_set_ = std::make_unique<dbus::PropertySet>(
object_proxy, modemmanager::kModemManager1SmsInterface,
base::Bind(&SMSReceiveHandler::OnPropertyChanged,
weak_ptr_factory_.GetWeakPtr()));
property_set_->RegisterProperty(SMSClient::kSMSPropertyState, &state_);
property_set_->ConnectSignals();
property_set_->Get(&state_, dbus::PropertySet::GetCallback());
}
~SMSReceiveHandler() = default;
private:
void OnPropertyChanged(const std::string& property_name) {
// Initially, we monitor only the "State" property of the SMS object. When
// the "State" property is updated as a result of the initial
// PropertySet::Get() call or subsequent D-Bus signals on property changes,
// we check if the "State" property indicates that the SMS is fully
// received. If the SMS is fully received, we fetch all properties via
// PropertySet::GetAll(). When the properties of interest are updated,
// |callback_| is invoked to notify the observer of the received SMS.
if (callback_.is_null())
return;
if (number_.is_valid() && text_.is_valid() && timestamp_.is_valid()) {
base::DictionaryValue sms;
sms.SetString(SMSClient::kSMSPropertyNumber, number_.value());
sms.SetString(SMSClient::kSMSPropertyText, text_.value());
sms.SetString(SMSClient::kSMSPropertyTimestamp, timestamp_.value());
// Move |callback_| to the task to ensure that |callback_| is only called
// once. Since |callback_| may destruct this object, schedule it to the
// task runner to run after this method returns.
base::ThreadTaskRunnerHandle::Get()->PostTask(
FROM_HERE, base::BindOnce(std::move(callback_), std::move(sms)));
return;
}
if (state_.is_valid() && state_.value() == kSMSStateReceived &&
!sms_received_) {
sms_received_ = true;
property_set_->RegisterProperty(SMSClient::kSMSPropertyNumber, &number_);
property_set_->RegisterProperty(SMSClient::kSMSPropertyText, &text_);
property_set_->RegisterProperty(SMSClient::kSMSPropertyTimestamp,
&timestamp_);
property_set_->GetAll();
}
}
SMSClient::GetAllCallback callback_;
bool sms_received_;
dbus::Property<uint32_t> state_;
dbus::Property<std::string> number_;
dbus::Property<std::string> text_;
dbus::Property<std::string> timestamp_;
std::unique_ptr<dbus::PropertySet> property_set_;
base::WeakPtrFactory<SMSReceiveHandler> weak_ptr_factory_;
DISALLOW_COPY_AND_ASSIGN(SMSReceiveHandler);
};
// SMSClient is used to communicate with the
// org.freedesktop.ModemManager1.SMS service. All methods should be
// called from the origin thread (UI thread) which initializes the
// DBusThreadManager instance.
class SMSClientImpl : public SMSClient {
public:
SMSClientImpl() : bus_(NULL), weak_ptr_factory_(this) {}
~SMSClientImpl() override = default;
// Calls GetAll method. |callback| is called after the method call succeeds.
void GetAll(const std::string& service_name,
const dbus::ObjectPath& object_path,
GetAllCallback callback) override {
dbus::ObjectProxy* proxy = bus_->GetObjectProxy(service_name, object_path);
sms_receive_handlers_[object_path] = std::make_unique<SMSReceiveHandler>(
proxy, base::BindOnce(&SMSClientImpl::OnSMSReceived,
weak_ptr_factory_.GetWeakPtr(), object_path,
std::move(callback)));
}
protected:
void Init(dbus::Bus* bus) override { bus_ = bus; }
private:
void OnSMSReceived(const dbus::ObjectPath& object_path,
GetAllCallback callback,
const base::DictionaryValue& sms) {
sms_receive_handlers_.erase(object_path);
std::move(callback).Run(sms);
}
dbus::Bus* bus_;
std::map<dbus::ObjectPath, std::unique_ptr<SMSReceiveHandler>>
sms_receive_handlers_;
// 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<SMSClientImpl> weak_ptr_factory_;
DISALLOW_COPY_AND_ASSIGN(SMSClientImpl);
};
} // namespace
////////////////////////////////////////////////////////////////////////////////
// SMSClient
// Properties of org.freedesktop.ModemManager1.Sms object:
const char SMSClient::kSMSPropertyState[] = "State";
const char SMSClient::kSMSPropertyNumber[] = "Number";
const char SMSClient::kSMSPropertyText[] = "Text";
const char SMSClient::kSMSPropertyTimestamp[] = "Timestamp";
SMSClient::SMSClient() = default;
SMSClient::~SMSClient() = default;
// static
SMSClient* SMSClient::Create() {
return new SMSClientImpl();
}
} // namespace chromeos