1 // Copyright (c) 2012 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 #include "chromeos/network/network_sms_handler.h"
6
7 #include <algorithm>
8 #include <deque>
9 #include <string>
10 #include <vector>
11
12 #include "base/bind.h"
13 #include "base/values.h"
14 #include "chromeos/dbus/dbus_thread_manager.h"
15 #include "chromeos/dbus/shill_device_client.h"
16 #include "chromeos/dbus/shill_manager_client.h"
17 #include "chromeos/dbus/gsm_sms_client.h"
18 #include "chromeos/dbus/modem_messaging_client.h"
19 #include "chromeos/dbus/sms_client.h"
20 #include "dbus/object_path.h"
21 #include "third_party/cros_system_api/dbus/service_constants.h"
22
23 namespace {
24
25 // Not exposed/exported:
26 const char kIndexKey[] = "index";
27
28 // Keys from ModemManager1
29 const char kModemManager1NumberKey[] = "Number";
30 const char kModemManager1TextKey[] = "Text";
31 const char kModemManager1TimestampKey[] = "Timestamp";
32
33 // Maximum number of messages stored for RequestUpdate(true).
34 const size_t kMaxReceivedMessages = 100;
35
36 } // namespace
37
38 namespace chromeos {
39
40 // static
41 const char NetworkSmsHandler::kNumberKey[] = "number";
42 const char NetworkSmsHandler::kTextKey[] = "text";
43 const char NetworkSmsHandler::kTimestampKey[] = "timestamp";
44
45 class NetworkSmsHandler::NetworkSmsDeviceHandler {
46 public:
NetworkSmsDeviceHandler()47 NetworkSmsDeviceHandler() {}
~NetworkSmsDeviceHandler()48 virtual ~NetworkSmsDeviceHandler() {}
49
50 virtual void RequestUpdate() = 0;
51 };
52
53 class NetworkSmsHandler::ModemManagerNetworkSmsDeviceHandler
54 : public NetworkSmsHandler::NetworkSmsDeviceHandler {
55 public:
56 ModemManagerNetworkSmsDeviceHandler(NetworkSmsHandler* host,
57 std::string dbus_connection,
58 dbus::ObjectPath object_path);
59
60 virtual void RequestUpdate() OVERRIDE;
61
62 private:
63 void ListCallback(const base::ListValue& message_list);
64 void SmsReceivedCallback(uint32 index, bool complete);
65 void GetCallback(uint32 index, const base::DictionaryValue& dictionary);
66 void DeleteMessages();
67 void MessageReceived(const base::DictionaryValue& dictionary);
68
69 NetworkSmsHandler* host_;
70 std::string dbus_connection_;
71 dbus::ObjectPath object_path_;
72 bool deleting_messages_;
73 base::WeakPtrFactory<ModemManagerNetworkSmsDeviceHandler> weak_ptr_factory_;
74 std::vector<uint32> delete_queue_;
75
76 DISALLOW_COPY_AND_ASSIGN(ModemManagerNetworkSmsDeviceHandler);
77 };
78
79 NetworkSmsHandler::
ModemManagerNetworkSmsDeviceHandler(NetworkSmsHandler * host,std::string dbus_connection,dbus::ObjectPath object_path)80 ModemManagerNetworkSmsDeviceHandler::ModemManagerNetworkSmsDeviceHandler(
81 NetworkSmsHandler* host,
82 std::string dbus_connection,
83 dbus::ObjectPath object_path)
84 : host_(host),
85 dbus_connection_(dbus_connection),
86 object_path_(object_path),
87 deleting_messages_(false),
88 weak_ptr_factory_(this) {
89 // Set the handler for received Sms messaages.
90 DBusThreadManager::Get()->GetGsmSMSClient()->SetSmsReceivedHandler(
91 dbus_connection_, object_path_,
92 base::Bind(&ModemManagerNetworkSmsDeviceHandler::SmsReceivedCallback,
93 weak_ptr_factory_.GetWeakPtr()));
94
95 // List the existing messages.
96 DBusThreadManager::Get()->GetGsmSMSClient()->List(
97 dbus_connection_, object_path_,
98 base::Bind(&NetworkSmsHandler::
99 ModemManagerNetworkSmsDeviceHandler::ListCallback,
100 weak_ptr_factory_.GetWeakPtr()));
101 }
102
RequestUpdate()103 void NetworkSmsHandler::ModemManagerNetworkSmsDeviceHandler::RequestUpdate() {
104 DBusThreadManager::Get()->GetGsmSMSClient()->RequestUpdate(
105 dbus_connection_, object_path_);
106 }
107
ListCallback(const base::ListValue & message_list)108 void NetworkSmsHandler::ModemManagerNetworkSmsDeviceHandler::ListCallback(
109 const base::ListValue& message_list) {
110 // This receives all messages, so clear any pending deletes.
111 delete_queue_.clear();
112 for (base::ListValue::const_iterator iter = message_list.begin();
113 iter != message_list.end(); ++iter) {
114 base::DictionaryValue* message = NULL;
115 if (!(*iter)->GetAsDictionary(&message))
116 continue;
117 MessageReceived(*message);
118 double index = 0;
119 if (message->GetDoubleWithoutPathExpansion(kIndexKey, &index))
120 delete_queue_.push_back(static_cast<uint32>(index));
121 }
122 DeleteMessages();
123 }
124
125 // Messages must be deleted one at a time, since we can not guarantee
126 // the order the deletion will be executed in. Delete messages from
127 // the back of the list so that the indices are valid.
DeleteMessages()128 void NetworkSmsHandler::ModemManagerNetworkSmsDeviceHandler::DeleteMessages() {
129 if (delete_queue_.empty()) {
130 deleting_messages_ = false;
131 return;
132 }
133 deleting_messages_ = true;
134 uint32 index = delete_queue_.back();
135 delete_queue_.pop_back();
136 DBusThreadManager::Get()->GetGsmSMSClient()->Delete(
137 dbus_connection_, object_path_, index,
138 base::Bind(&NetworkSmsHandler::
139 ModemManagerNetworkSmsDeviceHandler::DeleteMessages,
140 weak_ptr_factory_.GetWeakPtr()));
141 }
142
143 void NetworkSmsHandler::
SmsReceivedCallback(uint32 index,bool complete)144 ModemManagerNetworkSmsDeviceHandler::SmsReceivedCallback(
145 uint32 index,
146 bool complete) {
147 // Only handle complete messages.
148 if (!complete)
149 return;
150 DBusThreadManager::Get()->GetGsmSMSClient()->Get(
151 dbus_connection_, object_path_, index,
152 base::Bind(&NetworkSmsHandler::
153 ModemManagerNetworkSmsDeviceHandler::GetCallback,
154 weak_ptr_factory_.GetWeakPtr(), index));
155 }
156
GetCallback(uint32 index,const base::DictionaryValue & dictionary)157 void NetworkSmsHandler::ModemManagerNetworkSmsDeviceHandler::GetCallback(
158 uint32 index,
159 const base::DictionaryValue& dictionary) {
160 MessageReceived(dictionary);
161 delete_queue_.push_back(index);
162 if (!deleting_messages_)
163 DeleteMessages();
164 }
165
166 void NetworkSmsHandler::
MessageReceived(const base::DictionaryValue & dictionary)167 ModemManagerNetworkSmsDeviceHandler::MessageReceived(
168 const base::DictionaryValue& dictionary) {
169 // The keys of the ModemManager.Modem.Gsm.SMS interface match the
170 // exported keys, so the dictionary used as a notification argument
171 // unchanged.
172 host_->MessageReceived(dictionary);
173 }
174
175 class NetworkSmsHandler::ModemManager1NetworkSmsDeviceHandler
176 : public NetworkSmsHandler::NetworkSmsDeviceHandler {
177 public:
178 ModemManager1NetworkSmsDeviceHandler(NetworkSmsHandler* host,
179 std::string dbus_connection,
180 dbus::ObjectPath object_path);
181
182 virtual void RequestUpdate() OVERRIDE;
183
184 private:
185 void ListCallback(const std::vector<dbus::ObjectPath>& paths);
186 void SmsReceivedCallback(const dbus::ObjectPath& path, bool complete);
187 void GetCallback(const base::DictionaryValue& dictionary);
188 void DeleteMessages();
189 void GetMessages();
190 void MessageReceived(const base::DictionaryValue& dictionary);
191
192 NetworkSmsHandler* host_;
193 std::string dbus_connection_;
194 dbus::ObjectPath object_path_;
195 bool deleting_messages_;
196 bool retrieving_messages_;
197 base::WeakPtrFactory<ModemManager1NetworkSmsDeviceHandler> weak_ptr_factory_;
198 std::vector<dbus::ObjectPath> delete_queue_;
199 std::deque<dbus::ObjectPath> retrieval_queue_;
200
201 DISALLOW_COPY_AND_ASSIGN(ModemManager1NetworkSmsDeviceHandler);
202 };
203
204 NetworkSmsHandler::
ModemManager1NetworkSmsDeviceHandler(NetworkSmsHandler * host,std::string dbus_connection,dbus::ObjectPath object_path)205 ModemManager1NetworkSmsDeviceHandler::ModemManager1NetworkSmsDeviceHandler(
206 NetworkSmsHandler* host,
207 std::string dbus_connection,
208 dbus::ObjectPath object_path)
209 : host_(host),
210 dbus_connection_(dbus_connection),
211 object_path_(object_path),
212 deleting_messages_(false),
213 retrieving_messages_(false),
214 weak_ptr_factory_(this) {
215 // Set the handler for received Sms messaages.
216 DBusThreadManager::Get()->GetModemMessagingClient()->SetSmsReceivedHandler(
217 dbus_connection_, object_path_,
218 base::Bind(
219 &NetworkSmsHandler::
220 ModemManager1NetworkSmsDeviceHandler::SmsReceivedCallback,
221 weak_ptr_factory_.GetWeakPtr()));
222
223 // List the existing messages.
224 DBusThreadManager::Get()->GetModemMessagingClient()->List(
225 dbus_connection_, object_path_,
226 base::Bind(&NetworkSmsHandler::
227 ModemManager1NetworkSmsDeviceHandler::ListCallback,
228 weak_ptr_factory_.GetWeakPtr()));
229 }
230
RequestUpdate()231 void NetworkSmsHandler::ModemManager1NetworkSmsDeviceHandler::RequestUpdate() {
232 // Calling List using the service "AddSMS" causes the stub
233 // implementation to deliver new sms messages.
234 DBusThreadManager::Get()->GetModemMessagingClient()->List(
235 std::string("AddSMS"), dbus::ObjectPath("/"),
236 base::Bind(&NetworkSmsHandler::
237 ModemManager1NetworkSmsDeviceHandler::ListCallback,
238 weak_ptr_factory_.GetWeakPtr()));
239 }
240
ListCallback(const std::vector<dbus::ObjectPath> & paths)241 void NetworkSmsHandler::ModemManager1NetworkSmsDeviceHandler::ListCallback(
242 const std::vector<dbus::ObjectPath>& paths) {
243 // This receives all messages, so clear any pending gets and deletes.
244 retrieval_queue_.clear();
245 delete_queue_.clear();
246
247 retrieval_queue_.resize(paths.size());
248 std::copy(paths.begin(), paths.end(), retrieval_queue_.begin());
249 if (!retrieving_messages_)
250 GetMessages();
251 }
252
253 // Messages must be deleted one at a time, since we can not guarantee
254 // the order the deletion will be executed in. Delete messages from
255 // the back of the list so that the indices are valid.
DeleteMessages()256 void NetworkSmsHandler::ModemManager1NetworkSmsDeviceHandler::DeleteMessages() {
257 if (delete_queue_.empty()) {
258 deleting_messages_ = false;
259 return;
260 }
261 deleting_messages_ = true;
262 dbus::ObjectPath sms_path = delete_queue_.back();
263 delete_queue_.pop_back();
264 DBusThreadManager::Get()->GetModemMessagingClient()->Delete(
265 dbus_connection_, object_path_, sms_path,
266 base::Bind(&NetworkSmsHandler::
267 ModemManager1NetworkSmsDeviceHandler::DeleteMessages,
268 weak_ptr_factory_.GetWeakPtr()));
269 }
270
271 // Messages must be fetched one at a time, so that we do not queue too
272 // many requests to a single threaded server.
GetMessages()273 void NetworkSmsHandler::ModemManager1NetworkSmsDeviceHandler::GetMessages() {
274 if (retrieval_queue_.empty()) {
275 retrieving_messages_ = false;
276 if (!deleting_messages_)
277 DeleteMessages();
278 return;
279 }
280 retrieving_messages_ = true;
281 dbus::ObjectPath sms_path = retrieval_queue_.front();
282 retrieval_queue_.pop_front();
283 DBusThreadManager::Get()->GetSMSClient()->GetAll(
284 dbus_connection_, sms_path,
285 base::Bind(&NetworkSmsHandler::
286 ModemManager1NetworkSmsDeviceHandler::GetCallback,
287 weak_ptr_factory_.GetWeakPtr()));
288 delete_queue_.push_back(sms_path);
289 }
290
291 void NetworkSmsHandler::
SmsReceivedCallback(const dbus::ObjectPath & sms_path,bool complete)292 ModemManager1NetworkSmsDeviceHandler::SmsReceivedCallback(
293 const dbus::ObjectPath& sms_path,
294 bool complete) {
295 // Only handle complete messages.
296 if (!complete)
297 return;
298 retrieval_queue_.push_back(sms_path);
299 if (!retrieving_messages_)
300 GetMessages();
301 }
302
GetCallback(const base::DictionaryValue & dictionary)303 void NetworkSmsHandler::ModemManager1NetworkSmsDeviceHandler::GetCallback(
304 const base::DictionaryValue& dictionary) {
305 MessageReceived(dictionary);
306 GetMessages();
307 }
308
309 void NetworkSmsHandler::
MessageReceived(const base::DictionaryValue & dictionary)310 ModemManager1NetworkSmsDeviceHandler::MessageReceived(
311 const base::DictionaryValue& dictionary) {
312 // The keys of the ModemManager1.SMS interface do not match the
313 // exported keys, so a new dictionary is created with the expected
314 // key namaes.
315 base::DictionaryValue new_dictionary;
316 std::string text, number, timestamp;
317 if (dictionary.GetStringWithoutPathExpansion(kModemManager1NumberKey,
318 &number))
319 new_dictionary.SetString(kNumberKey, number);
320 if (dictionary.GetStringWithoutPathExpansion(kModemManager1TextKey, &text))
321 new_dictionary.SetString(kTextKey, text);
322 // TODO(jglasgow): consider normalizing timestamp.
323 if (dictionary.GetStringWithoutPathExpansion(kModemManager1TimestampKey,
324 ×tamp))
325 new_dictionary.SetString(kTimestampKey, timestamp);
326 host_->MessageReceived(new_dictionary);
327 }
328
329 ///////////////////////////////////////////////////////////////////////////////
330 // NetworkSmsHandler
331
NetworkSmsHandler()332 NetworkSmsHandler::NetworkSmsHandler()
333 : weak_ptr_factory_(this) {
334 }
335
~NetworkSmsHandler()336 NetworkSmsHandler::~NetworkSmsHandler() {
337 DBusThreadManager::Get()->GetShillManagerClient()->
338 RemovePropertyChangedObserver(this);
339 }
340
Init()341 void NetworkSmsHandler::Init() {
342 // Add as an observer here so that new devices added after this call are
343 // recognized.
344 DBusThreadManager::Get()->GetShillManagerClient()->AddPropertyChangedObserver(
345 this);
346 // Request network manager properties so that we can get the list of devices.
347 DBusThreadManager::Get()->GetShillManagerClient()->GetProperties(
348 base::Bind(&NetworkSmsHandler::ManagerPropertiesCallback,
349 weak_ptr_factory_.GetWeakPtr()));
350 }
351
RequestUpdate(bool request_existing)352 void NetworkSmsHandler::RequestUpdate(bool request_existing) {
353 // If we already received messages and |request_existing| is true, send
354 // updates for existing messages.
355 for (ScopedVector<base::DictionaryValue>::iterator iter =
356 received_messages_.begin();
357 iter != received_messages_.end(); ++iter) {
358 base::DictionaryValue* message = *iter;
359 NotifyMessageReceived(*message);
360 }
361 // Request updates from each device.
362 for (ScopedVector<NetworkSmsDeviceHandler>::iterator iter =
363 device_handlers_.begin(); iter != device_handlers_.end(); ++iter) {
364 (*iter)->RequestUpdate();
365 }
366 }
367
AddObserver(Observer * observer)368 void NetworkSmsHandler::AddObserver(Observer* observer) {
369 observers_.AddObserver(observer);
370 }
371
RemoveObserver(Observer * observer)372 void NetworkSmsHandler::RemoveObserver(Observer* observer) {
373 observers_.RemoveObserver(observer);
374 }
375
OnPropertyChanged(const std::string & name,const base::Value & value)376 void NetworkSmsHandler::OnPropertyChanged(const std::string& name,
377 const base::Value& value) {
378 if (name != shill::kDevicesProperty)
379 return;
380 const base::ListValue* devices = NULL;
381 if (!value.GetAsList(&devices) || !devices)
382 return;
383 UpdateDevices(devices);
384 }
385
386 // Private methods
387
AddReceivedMessage(const base::DictionaryValue & message)388 void NetworkSmsHandler::AddReceivedMessage(
389 const base::DictionaryValue& message) {
390 base::DictionaryValue* new_message = message.DeepCopy();
391 if (received_messages_.size() >= kMaxReceivedMessages)
392 received_messages_.erase(received_messages_.begin());
393 received_messages_.push_back(new_message);
394 }
395
NotifyMessageReceived(const base::DictionaryValue & message)396 void NetworkSmsHandler::NotifyMessageReceived(
397 const base::DictionaryValue& message) {
398 FOR_EACH_OBSERVER(Observer, observers_, MessageReceived(message));
399 }
400
MessageReceived(const base::DictionaryValue & message)401 void NetworkSmsHandler::MessageReceived(const base::DictionaryValue& message) {
402 AddReceivedMessage(message);
403 NotifyMessageReceived(message);
404 }
405
ManagerPropertiesCallback(DBusMethodCallStatus call_status,const base::DictionaryValue & properties)406 void NetworkSmsHandler::ManagerPropertiesCallback(
407 DBusMethodCallStatus call_status,
408 const base::DictionaryValue& properties) {
409 if (call_status != DBUS_METHOD_CALL_SUCCESS) {
410 LOG(ERROR) << "NetworkSmsHandler: Failed to get manager properties.";
411 return;
412 }
413 const base::Value* value;
414 if (!properties.GetWithoutPathExpansion(shill::kDevicesProperty, &value) ||
415 value->GetType() != base::Value::TYPE_LIST) {
416 LOG(ERROR) << "NetworkSmsHandler: No list value for: "
417 << shill::kDevicesProperty;
418 return;
419 }
420 const base::ListValue* devices = static_cast<const base::ListValue*>(value);
421 UpdateDevices(devices);
422 }
423
UpdateDevices(const base::ListValue * devices)424 void NetworkSmsHandler::UpdateDevices(const base::ListValue* devices) {
425 for (base::ListValue::const_iterator iter = devices->begin();
426 iter != devices->end(); ++iter) {
427 std::string device_path;
428 (*iter)->GetAsString(&device_path);
429 if (!device_path.empty()) {
430 // Request device properties.
431 VLOG(1) << "GetDeviceProperties: " << device_path;
432 DBusThreadManager::Get()->GetShillDeviceClient()->GetProperties(
433 dbus::ObjectPath(device_path),
434 base::Bind(&NetworkSmsHandler::DevicePropertiesCallback,
435 weak_ptr_factory_.GetWeakPtr(),
436 device_path));
437 }
438 }
439 }
440
DevicePropertiesCallback(const std::string & device_path,DBusMethodCallStatus call_status,const base::DictionaryValue & properties)441 void NetworkSmsHandler::DevicePropertiesCallback(
442 const std::string& device_path,
443 DBusMethodCallStatus call_status,
444 const base::DictionaryValue& properties) {
445 if (call_status != DBUS_METHOD_CALL_SUCCESS) {
446 LOG(ERROR) << "NetworkSmsHandler: ERROR: " << call_status
447 << " For: " << device_path;
448 return;
449 }
450
451 std::string device_type;
452 if (!properties.GetStringWithoutPathExpansion(
453 shill::kTypeProperty, &device_type)) {
454 LOG(ERROR) << "NetworkSmsHandler: No type for: " << device_path;
455 return;
456 }
457 if (device_type != shill::kTypeCellular)
458 return;
459
460 std::string dbus_connection;
461 if (!properties.GetStringWithoutPathExpansion(
462 shill::kDBusConnectionProperty, &dbus_connection)) {
463 LOG(ERROR) << "Device has no DBusConnection Property: " << device_path;
464 return;
465 }
466
467 std::string object_path_string;
468 if (!properties.GetStringWithoutPathExpansion(
469 shill::kDBusObjectProperty, &object_path_string)) {
470 LOG(ERROR) << "Device has no DBusObject Property: " << device_path;
471 return;
472 }
473 dbus::ObjectPath object_path(object_path_string);
474 if (object_path_string.compare(
475 0, sizeof(modemmanager::kModemManager1ServicePath) - 1,
476 modemmanager::kModemManager1ServicePath) == 0) {
477 device_handlers_.push_back(
478 new ModemManager1NetworkSmsDeviceHandler(
479 this, dbus_connection, object_path));
480 } else {
481 device_handlers_.push_back(
482 new ModemManagerNetworkSmsDeviceHandler(
483 this, dbus_connection, object_path));
484 }
485 }
486
487
488 } // namespace chromeos
489