aboutsummaryrefslogtreecommitdiff
path: root/examples
diff options
context:
space:
mode:
authorDan Williams <dan@ioncontrol.co>2025-01-29 10:20:18 -0600
committerAleksander Morgado <aleksander@aleksander.es>2025-02-17 21:57:33 +0000
commitc320e50753e544e3277280ee53b10eadcdd46fdb (patch)
tree570c976270bf174df86badc42bd13a3e1a7eb333 /examples
parentbc7198df8a3217d29c936fccde66820774228e30 (diff)
examples: add sms-watch-python
Add an example that watches for and prints all SMS messages of all modems. Signed-off-by: Dan Williams <dan@ioncontrol.co>
Diffstat (limited to 'examples')
-rw-r--r--examples/modem-watcher-python/ModemWatcher.py8
-rwxr-xr-xexamples/modem-watcher-python/modem-watcher-python2
-rw-r--r--examples/sms-send-python/README (renamed from examples/sms-python/README)6
-rwxr-xr-xexamples/sms-send-python/sms-send-python (renamed from examples/sms-python/sms-python)0
-rw-r--r--examples/sms-watch-python/README27
-rw-r--r--examples/sms-watch-python/sms-watch-python166
6 files changed, 204 insertions, 5 deletions
diff --git a/examples/modem-watcher-python/ModemWatcher.py b/examples/modem-watcher-python/ModemWatcher.py
index b7cf4e7c..c42d26a7 100644
--- a/examples/modem-watcher-python/ModemWatcher.py
+++ b/examples/modem-watcher-python/ModemWatcher.py
@@ -28,7 +28,7 @@ class ModemWatcher:
The ModemWatcher class is responsible for monitoring ModemManager.
"""
- def __init__(self):
+ def __init__(self, modem_cb):
# Flag for initial logs
self.initializing = True
# Setup DBus monitoring
@@ -40,6 +40,7 @@ class ModemWatcher:
# IDs for added/removed signals
self.object_added_id = 0
self.object_removed_id = 0
+ self.modem_callback = modem_cb
# Follow availability of the ModemManager process
self.available = False
self.manager.connect('notify::name-owner', self.on_name_owner)
@@ -109,6 +110,8 @@ class ModemWatcher:
obj.get_object_path())
else:
modem.connect('state-changed', self.on_modem_state_updated)
+ if self.modem_callback != None:
+ self.modem_callback(obj, True)
def on_object_removed(self, manager, obj):
"""
@@ -116,3 +119,6 @@ class ModemWatcher:
"""
print('[ModemWatcher] %s: modem unmanaged by ModemManager' %
obj.get_object_path())
+ if self.modem_callback != None:
+ self.modem_callback(obj, False)
+
diff --git a/examples/modem-watcher-python/modem-watcher-python b/examples/modem-watcher-python/modem-watcher-python
index f7dd1fc6..ab7399d1 100755
--- a/examples/modem-watcher-python/modem-watcher-python
+++ b/examples/modem-watcher-python/modem-watcher-python
@@ -33,7 +33,7 @@ def signal_handler(loop):
def main():
"""Main routine."""
# Create modem watcher
- ModemWatcher.ModemWatcher()
+ ModemWatcher.ModemWatcher(None)
# Main loop
main_loop = GLib.MainLoop()
diff --git a/examples/sms-python/README b/examples/sms-send-python/README
index 50a1cc55..3927c743 100644
--- a/examples/sms-python/README
+++ b/examples/sms-send-python/README
@@ -1,5 +1,5 @@
-The sms-python program makes use of the 'libmm-glib' library through
+The sms-send-python program makes use of the 'libmm-glib' library through
GObject Introspection to talk to ModemManager.
The program will:
@@ -11,10 +11,10 @@ The program will:
The output will look like this:
-$ ./sms-python "+1234567890" "hello there, how are you?"
+$ ./sms-send-python "+1234567890" "hello there, how are you?"
/org/freedesktop/ModemManager1/Modem/0: sms sent
Note that the program requires ModemManager and libmm-glib to be installed in
the system and the introspection typelibs available in the standard paths.
-Have fun! \ No newline at end of file
+Have fun!
diff --git a/examples/sms-python/sms-python b/examples/sms-send-python/sms-send-python
index 46d5e692..46d5e692 100755
--- a/examples/sms-python/sms-python
+++ b/examples/sms-send-python/sms-send-python
diff --git a/examples/sms-watch-python/README b/examples/sms-watch-python/README
new file mode 100644
index 00000000..e490c890
--- /dev/null
+++ b/examples/sms-watch-python/README
@@ -0,0 +1,27 @@
+
+The sms-watch-python program makes use of the 'libmm-glib' library through
+GObject Introspection to talk to ModemManager.
+
+The program will:
+ * Detect whether ModemManager is found in the bus
+ * List all existing SMS messages stored in the modem
+ * Print details of all new SMS messages (either from the network or the host)
+
+The output will look like this:
+
+$ ./sms-watch-python
+[ModemWatcher] /org/freedesktop/ModemManager1/Modem/21: modem managed by ModemManager [862342010150533]: QUALCOMM INCORPORATED (0)
+[ModemWatcher] /org/freedesktop/ModemManager1/Modem/22: modem state updated: disabled -> enabling (user-requested)
+[SmsWatcher] PATH: /org/freedesktop/ModemManager1/SMS/57
+[SmsWatcher] NUMBER: 78273
+[SmsWatcher] TEXT: 'Lieber Tchibo mobil Kunde, herzlich Willkommen in den USA. Sie k nnen hier wie gewohnt Ihre Mailbox einfach und bequem unter 333 erreichen.'
+[SmsWatcher] PDU-TYPE: deliver
+[SmsWatcher] STATE: received
+[SmsWatcher] STORAGE: me
+[SmsWatcher] SMSC: +491760000443
+[SmsWatcher] TIMESTAMP: 2014-05-07T04:35:19+02
+
+Note that the program requires ModemManager and libmm-glib to be installed in
+the system and the introspection typelibs available in the standard paths.
+
+Have fun!
diff --git a/examples/sms-watch-python/sms-watch-python b/examples/sms-watch-python/sms-watch-python
new file mode 100644
index 00000000..3a6a945f
--- /dev/null
+++ b/examples/sms-watch-python/sms-watch-python
@@ -0,0 +1,166 @@
+#!/usr/bin/env python3
+# -*- Mode: python; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
+#
+# This program is free software; you can redistribute it and/or modify it under
+# the terms of the GNU Lesser General Public License as published by the Free
+# Software Foundation; either version 2 of the License, or (at your option) any
+# later version.
+#
+# This program is distributed in the hope that it will be useful, but WITHOUT
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
+# FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
+# details.
+#
+# You should have received a copy of the GNU Lesser General Public License along
+# with this program; if not, write to the Free Software Foundation, Inc., 51
+# Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+#
+# Copyright (C) 2016 Aleksander Morgado <aleksander@aleksander.es>
+# Copyright (C) 2025 Dan Williams <dan@ioncontrol.co>
+#
+
+import os
+import signal
+import sys
+
+watchers = {}
+
+import gi
+gi.require_version('ModemManager', '1.0')
+from gi.repository import Gio, GLib, GObject, ModemManager
+
+sys.path.append(os.path.join(os.path.dirname(__file__), '..', 'modem-watcher-python'))
+import ModemWatcher
+
+class SmsWatcher:
+ """
+ The SmsWatcher class watches for and prints complete SMS messages.
+ """
+
+ def __init__(self, sms):
+ self.sms = sms
+ # Connect to the SMS message's 'state' property to know when all parts
+ # have been received.
+ self.state_id = self.sms.connect('notify::state', self.on_state_changed)
+ self.on_state_changed(sms, sms.get_state())
+
+ def cleanup(self):
+ self.sms.disconnect(self.state_id)
+ self.state_id = 0
+
+ def on_state_changed(self, sms, prop):
+ # When all parts have been received, print the SMS
+ if sms.get_state() == ModemManager.SmsState.RECEIVED:
+ self.show()
+
+ def show(self):
+ print('[SmsWatcher] PATH: %s' % self.sms.get_path())
+ print('[SmsWatcher] NUMBER: %s' % self.sms.get_number())
+ print('[SmsWatcher] TEXT: \'%s\'' % self.sms.get_text())
+ print('[SmsWatcher] PDU-TYPE: %s' % ModemManager.SmsPduType.get_string(self.sms.get_pdu_type()))
+ print('[SmsWatcher] STATE: %s' % ModemManager.SmsState.get_string(self.sms.get_state()))
+ if self.sms.get_validity_type() == ModemManager.SmsValidityType.RELATIVE:
+ print('[SmsWatcher] VALIDITY: %u' % self.sms.get_validity_relative())
+ print('[SmsWatcher] STORAGE: %s' % ModemManager.SmsStorage.get_string(self.sms.get_storage()))
+ print('[SmsWatcher] SMSC: %s' % self.sms.get_smsc())
+ if self.sms.get_class() >= 0:
+ print('[SmsWatcher] STORAGE: %d' % self.sms.get_class())
+ if self.sms.get_pdu_type() == ModemManager.SmsPduType.SUBMIT:
+ delivery_report = "not requested"
+ if self.sms.get_delivery_report_request():
+ delivery_report = "requested"
+ print('[SmsWatcher] DELIVERY-REPORT: %s' % delivery_report)
+ if self.sms.get_message_reference() != 0:
+ print('[SmsWatcher] MSG-REFERENCE: %u' % self.sms.get_message_reference())
+ print('[SmsWatcher] TIMESTAMP: %s' % self.sms.get_timestamp())
+ if self.sms.get_delivery_state () != ModemManager.SmsDeliveryState.UNKNOWN:
+ print('[SmsWatcher] DELIVERY-STATE: %s' % ModemManager.SmsDeliveryState.get_string_extended (self.sms.get_delivery_state()))
+ if self.sms.get_discharge_timestamp() is not None:
+ print('[SmsWatcher] DISCHARGE-TIMESTAMP: %s' % self.sms.get_discharge_timestamp())
+ print('')
+
+
+class MessagingWatcher:
+ """
+ The MessagingWatcher class monitors a modem's Messaging interface for SMS messages.
+ """
+
+ def __init__(self, messaging):
+ self.iface = messaging
+ # Connect to the messaging interface's added/removed signals to be
+ # notified when SMS messages are received or deleted
+ self.sms_added_id = self.iface.connect('added', self.on_sms_added)
+ self.sms_removed_id = self.iface.connect('deleted', self.on_sms_removed)
+
+ # List all existing messages
+ self.messages = {}
+ for sms in self.iface.list_sync():
+ self.messages[sms.get_path()] = SmsWatcher(sms)
+
+ def cleanup(self):
+ # We no longer care about signals; disconnect them
+ self.iface.disconnect(self.sms_added_id)
+ self.sms_added_id = 0
+ self.iface.disconnect(self.sms_removed_id)
+ self.sms_removed_id = 0
+ # Clean up each SMS we're tracking
+ for sms in self.messages.values():
+ sms.cleanup()
+ self.messages = {}
+
+ def on_sms_added(self, messaging, path, received):
+ # MM/libmm-glib do not yet provide a way to retrieve
+ # a single SMS object given its D-Bus path. List them
+ # and find the one we want.
+ for sms in self.iface.list_sync():
+ if sms.get_path() == path:
+ # Watch this SMS
+ self.messages[sms.get_path()] = SmsWatcher(sms)
+
+ def on_sms_removed(self, messaging, path):
+ try:
+ # Clean up the SMS and stop tracking it
+ sms_watcher = self.messages[path]
+ sms_watcher.cleanup()
+ del self.messages[path]
+ except KeyError:
+ pass
+
+
+def modem_callback(obj, added):
+ global watchers
+ if added:
+ # Get the messaging interface of this modem and start watching for SMSes
+ messaging = obj.get_modem_messaging()
+ if messaging is not None:
+ watchers[obj] = MessagingWatcher(messaging)
+ else:
+ try:
+ # Modem went away; clean up
+ watcher = watchers[obj]
+ watcher.cleanup()
+ del(watchers, obj)
+ except KeyError:
+ pass
+
+def signal_handler(loop):
+ """SIGHUP and SIGINT handler."""
+ loop.quit()
+
+def main():
+ """Main routine."""
+ modem_watcher = ModemWatcher.ModemWatcher(modem_callback)
+
+ # Main loop to watch for incoming SMSes
+ main_loop = GLib.MainLoop()
+ GLib.unix_signal_add(
+ GLib.PRIORITY_HIGH, signal.SIGHUP, signal_handler, main_loop)
+ GLib.unix_signal_add(
+ GLib.PRIORITY_HIGH, signal.SIGTERM, signal_handler, main_loop)
+ try:
+ main_loop.run()
+ except KeyboardInterrupt:
+ pass
+
+if __name__ == "__main__":
+ main()