diff options
author | Dan Williams <dan@ioncontrol.co> | 2025-01-29 10:20:18 -0600 |
---|---|---|
committer | Aleksander Morgado <aleksander@aleksander.es> | 2025-02-17 21:57:33 +0000 |
commit | c320e50753e544e3277280ee53b10eadcdd46fdb (patch) | |
tree | 570c976270bf174df86badc42bd13a3e1a7eb333 /examples | |
parent | bc7198df8a3217d29c936fccde66820774228e30 (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.py | 8 | ||||
-rwxr-xr-x | examples/modem-watcher-python/modem-watcher-python | 2 | ||||
-rw-r--r-- | examples/sms-send-python/README (renamed from examples/sms-python/README) | 6 | ||||
-rwxr-xr-x | examples/sms-send-python/sms-send-python (renamed from examples/sms-python/sms-python) | 0 | ||||
-rw-r--r-- | examples/sms-watch-python/README | 27 | ||||
-rw-r--r-- | examples/sms-watch-python/sms-watch-python | 166 |
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() |