diff options
| -rw-r--r-- | .gitignore | 1 | ||||
| -rw-r--r-- | README.md | 1 | ||||
| -rwxr-xr-x | mpris_to_text.py | 102 |
3 files changed, 50 insertions, 54 deletions
diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..600d2d3 --- /dev/null +++ b/.gitignore | |||
| @@ -0,0 +1 @@ | |||
| .vscode \ No newline at end of file | |||
| @@ -7,7 +7,6 @@ A Python script that fetches media data from MPRIS-compliant media players and s | |||
| 7 | ## Required packages | 7 | ## Required packages |
| 8 | 8 | ||
| 9 | - [blessed](https://github.com/jquast/blessed): Terminal UI | 9 | - [blessed](https://github.com/jquast/blessed): Terminal UI |
| 10 | - [dbussy](https://github.com/ldo/dbussy): DBus client | ||
| 11 | 10 | ||
| 12 | ## Parameters | 11 | ## Parameters |
| 13 | 12 | ||
diff --git a/mpris_to_text.py b/mpris_to_text.py index 1284300..16ee1e0 100755 --- a/mpris_to_text.py +++ b/mpris_to_text.py | |||
| @@ -1,31 +1,34 @@ | |||
| 1 | #!/usr/bin/python | 1 | #!/usr/bin/python |
| 2 | 2 | ||
| 3 | import sys | 3 | import sys |
| 4 | import asyncio | ||
| 5 | import re | 4 | import re |
| 6 | import threading | 5 | import threading |
| 7 | import argparse | 6 | import argparse |
| 8 | import signal | 7 | import signal |
| 9 | 8 | ||
| 9 | import dbus | ||
| 10 | import dbus.mainloop.glib | ||
| 11 | from gi.repository import GLib | ||
| 12 | |||
| 10 | from blessed import Terminal | 13 | from blessed import Terminal |
| 11 | 14 | ||
| 12 | import ravel | 15 | |
| 16 | dbus.mainloop.glib.DBusGMainLoop(set_as_default=True) | ||
| 13 | 17 | ||
| 14 | 18 | ||
| 15 | loop = asyncio.get_event_loop() | ||
| 16 | name_regex = re.compile("^org\.mpris\.MediaPlayer2\.") | 19 | name_regex = re.compile("^org\.mpris\.MediaPlayer2\.") |
| 17 | bus = None | 20 | bus = dbus.SessionBus() |
| 18 | term = Terminal() | 21 | term = Terminal() |
| 19 | 22 | ||
| 20 | refresh_cond = threading.Condition() | 23 | refresh_cond = threading.Condition() |
| 21 | refresh_flag = True | 24 | refresh_flag = True |
| 22 | exit_flag = False | 25 | exit_flag = False |
| 23 | exit_task = asyncio.Future() | ||
| 24 | 26 | ||
| 25 | players = [] | 27 | players = [] |
| 26 | player_id_to_index = {} | 28 | player_id_to_index = {} |
| 27 | active_player = "" | 29 | active_player = "" |
| 28 | current_output = "" | 30 | current_output = "" |
| 31 | playing_song_changed_signal_receiver = None | ||
| 29 | 32 | ||
| 30 | filename = "" | 33 | filename = "" |
| 31 | meta_format = "" | 34 | meta_format = "" |
| @@ -35,14 +38,14 @@ meta_format_album = "" | |||
| 35 | 38 | ||
| 36 | 39 | ||
| 37 | def track_string(metadata): | 40 | def track_string(metadata): |
| 38 | artist = metadata["xesam:artist"][1][0] if "xesam:artist" in metadata else "" | 41 | artist = metadata["xesam:artist"][0] if "xesam:artist" in metadata else "" |
| 39 | title = metadata["xesam:title"][1] if "xesam:title" in metadata else "" | 42 | title = metadata["xesam:title"] if "xesam:title" in metadata else "" |
| 40 | album = metadata["xesam:album"][1] if "xesam:album" in metadata else "" | 43 | album = metadata["xesam:album"] if "xesam:album" in metadata else "" |
| 41 | 44 | ||
| 42 | return meta_format.format( | 45 | return meta_format.format( |
| 43 | artist = meta_format_artist.format(artist) if artist != "" else "", | 46 | artist = meta_format_artist.format(artist) if artist != "" else "", |
| 44 | title = meta_format_title.format(title) if title != "" else "", | 47 | title = meta_format_title.format(title) if title != "" else "", |
| 45 | album = meta_format_album.format(album) if title != "" else "" | 48 | album = meta_format_album.format(album) if album != "" else "" |
| 46 | ) | 49 | ) |
| 47 | 50 | ||
| 48 | 51 | ||
| @@ -56,25 +59,21 @@ def write_track(track): | |||
| 56 | f.close() | 59 | f.close() |
| 57 | 60 | ||
| 58 | 61 | ||
| 59 | @ravel.signal(name = "PropertiesChanged", in_signature = "sa{sv}as", args_keyword = "args") | 62 | def playing_song_changed(player, changed_properties, invalidated_properties): |
| 60 | def playing_song_changed(args): | ||
| 61 | global refresh_cond | 63 | global refresh_cond |
| 62 | global refresh_flag | 64 | global refresh_flag |
| 63 | 65 | ||
| 64 | [ player, changed_properties, invalidated_properties ] = args | ||
| 65 | if "Metadata" in changed_properties: | 66 | if "Metadata" in changed_properties: |
| 66 | write_track(track_string(changed_properties["Metadata"][1])) | 67 | write_track(track_string(changed_properties["Metadata"])) |
| 67 | with refresh_cond: | 68 | with refresh_cond: |
| 68 | refresh_flag = True | 69 | refresh_flag = True |
| 69 | refresh_cond.notify() | 70 | refresh_cond.notify() |
| 70 | 71 | ||
| 71 | 72 | ||
| 72 | @ravel.signal(name = "NameOwnerChanged", in_signature = "sss", args_keyword = "args") | 73 | def dbus_name_owner_changed(name, old_owner, new_owner): |
| 73 | def dbus_name_owner_changed(args): | ||
| 74 | global refresh_cond | 74 | global refresh_cond |
| 75 | global refresh_flag | 75 | global refresh_flag |
| 76 | 76 | ||
| 77 | [ name, old_owner, new_owner ] = args | ||
| 78 | if name_regex.match(name): | 77 | if name_regex.match(name): |
| 79 | get_players() | 78 | get_players() |
| 80 | with refresh_cond: | 79 | with refresh_cond: |
| @@ -84,7 +83,10 @@ def dbus_name_owner_changed(args): | |||
| 84 | 83 | ||
| 85 | def set_active_player(player_id): | 84 | def set_active_player(player_id): |
| 86 | global bus | 85 | global bus |
| 86 | global players | ||
| 87 | global player_id_to_index | ||
| 87 | global active_player | 88 | global active_player |
| 89 | global playing_song_changed_signal_receiver | ||
| 88 | 90 | ||
| 89 | if player_id in player_id_to_index: | 91 | if player_id in player_id_to_index: |
| 90 | active_player = player_id | 92 | active_player = player_id |
| @@ -93,26 +95,22 @@ def set_active_player(player_id): | |||
| 93 | else: | 95 | else: |
| 94 | active_player = "" | 96 | active_player = "" |
| 95 | 97 | ||
| 96 | for (name, player_id) in players: | 98 | if playing_song_changed_signal_receiver is not None: |
| 97 | bus.unlisten_propchanged( | 99 | playing_song_changed_signal_receiver.remove() |
| 98 | path = "/org/mpris/MediaPlayer2", | ||
| 99 | interface = name, | ||
| 100 | func = playing_song_changed, | ||
| 101 | fallback = True | ||
| 102 | ) | ||
| 103 | 100 | ||
| 104 | if active_player != "": | 101 | if active_player != "": |
| 105 | bus.listen_propchanged( | 102 | playing_song_changed_signal_receiver = bus.add_signal_receiver( |
| 106 | path = "/org/mpris/MediaPlayer2", | 103 | handler_function = playing_song_changed, |
| 107 | interface = active_player, | 104 | bus_name = active_player, |
| 108 | func = playing_song_changed, | 105 | dbus_interface = "org.freedesktop.DBus.Properties", |
| 109 | fallback = True | 106 | signal_name = "PropertiesChanged", |
| 107 | path = "/org/mpris/MediaPlayer2" | ||
| 110 | ) | 108 | ) |
| 111 | 109 | ||
| 112 | player_path = bus[active_player]["/org/mpris/MediaPlayer2"] | 110 | player_path = bus.get_object(active_player, "/org/mpris/MediaPlayer2") |
| 113 | player_props = player_path.get_interface("org.freedesktop.DBus.Properties") | 111 | player_props = dbus.Interface(player_path, "org.freedesktop.DBus.Properties") |
| 114 | 112 | ||
| 115 | write_track(track_string(player_props.Get("org.mpris.MediaPlayer2.Player", "Metadata")[0][1])) | 113 | write_track(track_string(player_props.Get("org.mpris.MediaPlayer2.Player", "Metadata"))) |
| 116 | else: | 114 | else: |
| 117 | write_track("") | 115 | write_track("") |
| 118 | 116 | ||
| @@ -120,23 +118,25 @@ def set_active_player(player_id): | |||
| 120 | def get_players(): | 118 | def get_players(): |
| 121 | global players | 119 | global players |
| 122 | global player_id_to_index | 120 | global player_id_to_index |
| 121 | global active_player | ||
| 123 | 122 | ||
| 124 | players = [] | 123 | players = [] |
| 125 | player_id_to_index = {} | 124 | player_id_to_index = {} |
| 126 | bus_proxy = bus["org.freedesktop.DBus"]["/org/freedesktop/DBus"].get_interface("org.freedesktop.DBus") | 125 | bus_path = bus.get_object("org.freedesktop.DBus", "/org/freedesktop/DBus") |
| 126 | bus_proxy = dbus.Interface(bus_path, "org.freedesktop.DBus") | ||
| 127 | names = bus_proxy.ListNames() | 127 | names = bus_proxy.ListNames() |
| 128 | 128 | ||
| 129 | for name in names[0]: | 129 | for name in names: |
| 130 | if name_regex.match(name): | 130 | if name_regex.match(name): |
| 131 | split_name = name.split(".") | 131 | split_name = name.split(".") |
| 132 | id_start_index = len(split_name[0]) + len(split_name[1]) + len(split_name[2]) + 3 | 132 | id_start_index = len(split_name[0]) + len(split_name[1]) + len(split_name[2]) + 3 |
| 133 | 133 | ||
| 134 | player_path = bus[name]["/org/mpris/MediaPlayer2"] | 134 | player_path = bus.get_object(name, "/org/mpris/MediaPlayer2") |
| 135 | player_proxy = player_path.get_interface("org.mpris.MediaPlayer2") | 135 | player_props = dbus.Interface(player_path, "org.freedesktop.DBus.Properties") |
| 136 | player_id = name[id_start_index:] | 136 | player_id = name[id_start_index:] |
| 137 | try: | 137 | try: |
| 138 | player_id = player_proxy.Identity | 138 | player_id = player_props.Get("org.mpris.MediaPlayer2.Player", "Identity") |
| 139 | except AttributeError: | 139 | except dbus.exceptions.DBusException: |
| 140 | pass | 140 | pass |
| 141 | 141 | ||
| 142 | players.append((name, player_id)) | 142 | players.append((name, player_id)) |
| @@ -194,17 +194,11 @@ def on_resize(*args): | |||
| 194 | 194 | ||
| 195 | def init_dbus(): | 195 | def init_dbus(): |
| 196 | global bus | 196 | global bus |
| 197 | global loop | ||
| 198 | 197 | ||
| 199 | bus = ravel.session_bus() | 198 | bus_path = bus.get_object("org.freedesktop.DBus", "/org/freedesktop/DBus") |
| 200 | bus.attach_asyncio(loop) | 199 | bus_proxy = dbus.Interface(bus_path, "org.freedesktop.DBus") |
| 201 | bus.listen_signal( | 200 | |
| 202 | path = "/org/freedesktop/DBus", | 201 | bus_proxy.connect_to_signal("NameOwnerChanged", dbus_name_owner_changed) |
| 203 | interface = "org.freedesktop.DBus", | ||
| 204 | name = "NameOwnerChanged", | ||
| 205 | func = dbus_name_owner_changed, | ||
| 206 | fallback = True | ||
| 207 | ) | ||
| 208 | 202 | ||
| 209 | get_players() | 203 | get_players() |
| 210 | 204 | ||
| @@ -215,6 +209,7 @@ def init_blessed(): | |||
| 215 | global refresh_cond | 209 | global refresh_cond |
| 216 | global refresh_flag | 210 | global refresh_flag |
| 217 | global exit_flag | 211 | global exit_flag |
| 212 | global loop | ||
| 218 | 213 | ||
| 219 | with term.cbreak(): | 214 | with term.cbreak(): |
| 220 | val = None | 215 | val = None |
| @@ -235,7 +230,7 @@ def init_blessed(): | |||
| 235 | exit_flag = True | 230 | exit_flag = True |
| 236 | refresh_cond.notify() | 231 | refresh_cond.notify() |
| 237 | 232 | ||
| 238 | exit_task.set_result(True) | 233 | loop.quit() |
| 239 | 234 | ||
| 240 | 235 | ||
| 241 | def read_args(): | 236 | def read_args(): |
| @@ -298,4 +293,5 @@ blessed_thread.start() | |||
| 298 | menu_thread = threading.Thread(target=draw_menu) | 293 | menu_thread = threading.Thread(target=draw_menu) |
| 299 | menu_thread.start() | 294 | menu_thread.start() |
| 300 | 295 | ||
| 301 | loop.run_until_complete(exit_task) | 296 | loop = GLib.MainLoop() |
| 297 | loop.run() | ||
