From 3cec35eeb501a22459b31b9d8e7e32b1a8cf8554 Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sun, 3 Nov 2024 23:12:43 +0100
Subject: [PATCH 01/16] Introduce new PlayContentHandler to abstract Second
Swipe
---
documentation/developers/docstring/README.md | 18 +-
src/jukebox/components/playermpd/__init__.py | 201 ++++++++++--------
...ntcallback.py => play_content_callback.py} | 0
.../playermpd/play_content_handler.py | 97 +++++++++
.../synchronisation/rfidcards/__init__.py | 2 +-
5 files changed, 216 insertions(+), 102 deletions(-)
rename src/jukebox/components/playermpd/{playcontentcallback.py => play_content_callback.py} (100%)
create mode 100644 src/jukebox/components/playermpd/play_content_handler.py
diff --git a/documentation/developers/docstring/README.md b/documentation/developers/docstring/README.md
index c48c34a41..b1aea3623 100644
--- a/documentation/developers/docstring/README.md
+++ b/documentation/developers/docstring/README.md
@@ -27,10 +27,10 @@
* [resolve](#misc.simplecolors.resolve)
* [print](#misc.simplecolors.print)
* [components](#components)
-* [components.playermpd.playcontentcallback](#components.playermpd.playcontentcallback)
- * [PlayContentCallbacks](#components.playermpd.playcontentcallback.PlayContentCallbacks)
- * [register](#components.playermpd.playcontentcallback.PlayContentCallbacks.register)
- * [run\_callbacks](#components.playermpd.playcontentcallback.PlayContentCallbacks.run_callbacks)
+* [components.playermpd.play_content_callback](#components.playermpd.play_content_callback)
+ * [PlayContentCallbacks](#components.playermpd.play_content_callback.PlayContentCallbacks)
+ * [register](#components.playermpd.play_content_callback.PlayContentCallbacks.register)
+ * [run\_callbacks](#components.playermpd.play_content_callback.PlayContentCallbacks.run_callbacks)
* [components.playermpd](#components.playermpd)
* [PlayerMPD](#components.playermpd.PlayerMPD)
* [mpd\_retry\_with\_mutex](#components.playermpd.PlayerMPD.mpd_retry_with_mutex)
@@ -761,11 +761,11 @@ Use just as a regular print function, but with first parameter as color
# components
-
+
-# components.playermpd.playcontentcallback
+# components.playermpd.play_content_callback
-
+
## PlayContentCallbacks Objects
@@ -776,7 +776,7 @@ class PlayContentCallbacks(Generic[STATE], CallbackHandler)
Callbacks are executed in various play functions
-
+
#### register
@@ -796,7 +796,7 @@ Callback signature is
- `folder`: relativ path to folder to play
- `state`: indicator of the state inside the calling
-
+
#### run\_callbacks
diff --git a/src/jukebox/components/playermpd/__init__.py b/src/jukebox/components/playermpd/__init__.py
index 86dbc60ab..96969ca95 100644
--- a/src/jukebox/components/playermpd/__init__.py
+++ b/src/jukebox/components/playermpd/__init__.py
@@ -88,6 +88,8 @@
import time
import functools
from pathlib import Path
+from typing import Union, Dict, Any
+
import components.player
import jukebox.cfghandler
import jukebox.utils as utils
@@ -98,8 +100,9 @@
import misc
from jukebox.NvManager import nv_manager
-from .playcontentcallback import PlayContentCallbacks, PlayCardState
+from .play_content_callback import PlayContentCallbacks, PlayCardState
from .coverart_cache_manager import CoverartCacheManager
+from .play_content_handler import PlayContentHandler, PlayContent, PlayContentType
logger = logging.getLogger('jb.PlayerMPD')
cfg = jukebox.cfghandler.get_handler('jukebox')
@@ -222,6 +225,13 @@ def __init__(self):
self.mpd_status = {}
self.mpd_status_poll_interval = 0.25
self.mpd_lock = MpdLock(self.mpd_client, self.mpd_host, 6600)
+
+ global play_card_callbacks
+ play_card_callbacks = PlayContentCallbacks[PlayCardState]('play_card_callbacks', logger, context=self.mpd_lock)
+ self.play_card_callbacks = play_card_callbacks
+ self.play_content_handler = PlayContentHandler(self)
+ self.play_content_handler.set_second_swipe_action(self.second_swipe_action)
+
self.status_is_closing = False
# self.status_thread = threading.Timer(self.mpd_status_poll_interval, self._mpd_status_poll).start()
@@ -522,60 +532,118 @@ def move(self):
# MPDClient.swapid(song1, song2)
raise NotImplementedError
- @plugs.tag
- def play_single(self, song_url):
+ def _play_single_internal(self, song_url: str) -> None:
with self.mpd_lock:
self.mpd_client.clear()
self.mpd_client.addid(song_url)
self.mpd_client.play()
- @plugs.tag
- def resume(self):
+ def _play_album_internal(self, artist: str, album: str) -> None:
with self.mpd_lock:
- songpos = self.current_folder_status["CURRENTSONGPOS"]
- elapsed = self.current_folder_status["ELAPSED"]
- self.mpd_client.seek(songpos, elapsed)
+ logger.info(f"Play album: '{album}' by '{artist}")
+ self.mpd_client.clear()
+ self.mpd_retry_with_mutex(self.mpd_client.findadd, 'albumartist', artist, 'album', album)
self.mpd_client.play()
- @plugs.tag
- def play_card(self, folder: str, recursive: bool = False):
- """
- Main entry point for trigger music playing from RFID reader. Decodes second swipe options before playing folder content
+ def _play_folder_internal(self, folder: str, recursive: bool) -> None:
+ with self.mpd_lock:
+ logger.info(f"Play folder: '{folder}'")
+ self.mpd_client.clear()
+
+ plc = playlistgenerator.PlaylistCollector(components.player.get_music_library_path())
+ plc.parse(folder, recursive)
+ uri = '--unset--'
+ try:
+ for uri in plc:
+ self.mpd_client.addid(uri)
+ except mpd.base.CommandError as e:
+ logger.error(f"{e.__class__.__qualname__}: {e} at uri {uri}")
+ except Exception as e:
+ logger.error(f"{e.__class__.__qualname__}: {e} at uri {uri}")
- Checks for second (or multiple) trigger of the same folder and calls first swipe / second swipe action
- accordingly.
+ self.music_player_status['player_status']['last_played_folder'] = folder
- :param folder: Folder path relative to music library path
- :param recursive: Add folder recursively
+ self.current_folder_status = self.music_player_status['audio_folder_status'].get(folder)
+ if self.current_folder_status is None:
+ self.current_folder_status = self.music_player_status['audio_folder_status'][folder] = {}
+
+ self.mpd_client.play()
+
+ @plugs.tag
+ def play_content(self, content: Union[str, Dict[str, str]], content_type: str = 'folder', recursive: bool = False):
"""
- # Developers notes:
- #
- # * 2nd swipe trigger may also happen, if playlist has already stopped playing
- # --> Generally, treat as first swipe
- # * 2nd swipe of same Card ID may also happen if a different song has been played in between from WebUI
- # --> Treat as first swipe
- # * With place-not-swipe: Card is placed on reader until playlist expieres. Music stop. Card is removed and
- # placed again on the reader: Should be like first swipe
- # * TODO: last_played_folder is restored after box start, so first swipe of last played card may look like
- # second swipe
- #
- logger.debug(f"last_played_folder = {self.music_player_status['player_status']['last_played_folder']}")
- with self.mpd_lock:
- is_second_swipe = self.music_player_status['player_status']['last_played_folder'] == folder
- if self.second_swipe_action is not None and is_second_swipe:
- logger.debug('Calling second swipe action')
+ Main entry point for trigger music playing from any source (RFID reader, web UI, etc.).
+ Supports second swipe for all content types.
+
+ :param content: Content identifier, either:
+ - string path for single/folder types
+ - dict with 'artist' and 'album' keys for album type
+ :param content_type: Type of content ('single', 'album', 'folder')
+ :param recursive: Add folder recursively (only used for folder type)
+ """
+ try:
+ content_type = content_type.lower()
+ if content_type == 'album':
+ if not isinstance(content, dict):
+ raise ValueError("Album content must be a dictionary with 'artist' and 'album' keys")
+ if 'artist' not in content or 'album' not in content:
+ raise ValueError("Album content dictionary must contain both 'artist' and 'album' keys")
+
+ play_content = PlayContent(
+ type=PlayContentType.ALBUM,
+ content=(content['artist'], content['album'])
+ )
+ elif content_type == 'single':
+ if not isinstance(content, str):
+ raise ValueError("Single track content must be a string path")
+ play_content = PlayContent(
+ type=PlayContentType.SINGLE,
+ content=content
+ )
+ else: # folder is default
+ if not isinstance(content, str):
+ raise ValueError("Folder content must be a string path")
+ play_content = PlayContent(
+ type=PlayContentType.FOLDER,
+ content=content,
+ recursive=recursive
+ )
+
+ self.play_content_handler.play_content(play_content)
+
+ except Exception as e:
+ logger.error(f"Error playing content: {e}")
+ raise
+
+ # Legacy/compatibility methods
+
+ @plugs.tag
+ def play_card(self, folder: str, recursive: bool = False):
+ """Legacy method for RFID cards - redirects to play_content"""
+ return self.play_content(folder, content_type='folder', recursive=recursive)
- # run callbacks before second_swipe_action is invoked
- play_card_callbacks.run_callbacks(folder, PlayCardState.secondSwipe)
+ @plugs.tag
+ def play_single(self, song_url):
+ """Deprecated: Use play_content with content_type='single' instead"""
+ self.play_content(song_url, content_type='single')
- self.second_swipe_action()
- else:
- logger.debug('Calling first swipe action')
+ @plugs.tag
+ def play_album(self, albumartist: str, album: str):
+ """Deprecated: Use play_content with content_type='album' instead"""
+ self.play_content({'artist': albumartist, 'album': album}, content_type='album')
- # run callbacks before play_folder is invoked
- play_card_callbacks.run_callbacks(folder, PlayCardState.firstSwipe)
+ @plugs.tag
+ def play_folder(self, folder: str, recursive: bool = False):
+ """Deprecated: Use play_content with content_type='folder' instead"""
+ self.play_content(folder, content_type='folder', recursive=recursive)
- self.play_folder(folder, recursive)
+ @plugs.tag
+ def resume(self):
+ with self.mpd_lock:
+ songpos = self.current_folder_status["CURRENTSONGPOS"]
+ elapsed = self.current_folder_status["ELAPSED"]
+ self.mpd_client.seek(songpos, elapsed)
+ self.mpd_client.play()
@plugs.tag
def get_single_coverart(self, song_url):
@@ -611,58 +679,6 @@ def get_folder_content(self, folder: str):
plc.get_directory_content(folder)
return plc.playlist
- @plugs.tag
- def play_folder(self, folder: str, recursive: bool = False) -> None:
- """
- Playback a music folder.
-
- Folder content is added to the playlist as described by :mod:`jukebox.playlistgenerator`.
- The playlist is cleared first.
-
- :param folder: Folder path relative to music library path
- :param recursive: Add folder recursively
- """
- # TODO: This changes the current state -> Need to save last state
- with self.mpd_lock:
- logger.info(f"Play folder: '{folder}'")
- self.mpd_client.clear()
-
- plc = playlistgenerator.PlaylistCollector(components.player.get_music_library_path())
- plc.parse(folder, recursive)
- uri = '--unset--'
- try:
- for uri in plc:
- self.mpd_client.addid(uri)
- except mpd.base.CommandError as e:
- logger.error(f"{e.__class__.__qualname__}: {e} at uri {uri}")
- except Exception as e:
- logger.error(f"{e.__class__.__qualname__}: {e} at uri {uri}")
-
- self.music_player_status['player_status']['last_played_folder'] = folder
-
- self.current_folder_status = self.music_player_status['audio_folder_status'].get(folder)
- if self.current_folder_status is None:
- self.current_folder_status = self.music_player_status['audio_folder_status'][folder] = {}
-
- self.mpd_client.play()
-
- @plugs.tag
- def play_album(self, albumartist: str, album: str):
- """
- Playback a album found in MPD database.
-
- All album songs are added to the playlist
- The playlist is cleared first.
-
- :param albumartist: Artist of the Album provided by MPD database
- :param album: Album name provided by MPD database
- """
- with self.mpd_lock:
- logger.info(f"Play album: '{album}' by '{albumartist}")
- self.mpd_client.clear()
- self.mpd_retry_with_mutex(self.mpd_client.findadd, 'albumartist', albumartist, 'album', album)
- self.mpd_client.play()
-
@plugs.tag
def queue_load(self, folder):
# There was something playing before -> stop and save state
@@ -762,6 +778,7 @@ def _db_is_updating(self, update_id: int):
#: States:
#: - See :class:`PlayCardState`
#: See :class:`PlayContentCallbacks`
+player_ctrl: PlayerMPD
play_card_callbacks: PlayContentCallbacks[PlayCardState]
diff --git a/src/jukebox/components/playermpd/playcontentcallback.py b/src/jukebox/components/playermpd/play_content_callback.py
similarity index 100%
rename from src/jukebox/components/playermpd/playcontentcallback.py
rename to src/jukebox/components/playermpd/play_content_callback.py
diff --git a/src/jukebox/components/playermpd/play_content_handler.py b/src/jukebox/components/playermpd/play_content_handler.py
new file mode 100644
index 000000000..660b82c28
--- /dev/null
+++ b/src/jukebox/components/playermpd/play_content_handler.py
@@ -0,0 +1,97 @@
+from enum import Enum, auto
+from dataclasses import dataclass
+from typing import Union, Optional, Callable, Protocol
+import logging
+
+from .play_content_callback import PlayCardState # Add this import
+
+logger = logging.getLogger('jb.PlayerMPD')
+
+
+class PlayContentType(Enum):
+ SINGLE = auto()
+ ALBUM = auto()
+ FOLDER = auto()
+
+
+@dataclass
+class PlayContent:
+ """Represents playable content with its type and metadata"""
+ type: PlayContentType
+ content: Union[str, tuple[str, str]] # str for SINGLE/FOLDER, tuple(artist, album) for ALBUM
+ recursive: bool = False
+
+
+class PlayerProtocol(Protocol):
+ """Protocol defining required player methods"""
+ def _play_single_internal(self, song_url: str) -> None:
+ """Play a single track"""
+
+ def _play_album_internal(self, artist: str, album: str) -> None:
+ """Play an album"""
+
+ def _play_folder_internal(self, folder: str, recursive: bool) -> None:
+ """Play a folder"""
+
+ @property
+ def play_card_callbacks(self) -> any:
+ """Access to callbacks"""
+
+
+class PlayContentHandler:
+ """Handles different types of playback content with second swipe support"""
+
+ def __init__(self, player: PlayerProtocol):
+ self.player = player
+ self.last_played_content: Optional[PlayContent] = None
+ self._second_swipe_action = None
+
+ def set_second_swipe_action(self, action: Optional[Callable]) -> None:
+ """Set the action to be performed on second swipe"""
+ self._second_swipe_action = action
+
+ def _play_content(self, content: PlayContent) -> None:
+ """Internal method to play content based on its type"""
+ if content.type == PlayContentType.SINGLE:
+ logger.debug(f"Playing single track: {content.content}")
+ self.player._play_single_internal(content.content)
+ elif content.type == PlayContentType.ALBUM:
+ artist, album = content.content
+ logger.debug(f"Playing album: {album} by {artist}")
+ self.player._play_album_internal(artist, album)
+ elif content.type == PlayContentType.FOLDER:
+ logger.debug(f"Playing folder: {content.content} (recursive={content.recursive})")
+ self.player._play_folder_internal(content.content, content.recursive)
+
+ def play_content(self, content: PlayContent) -> None:
+ """
+ Main entry point for playing content with second swipe support
+
+ Checks for second trigger of the same content and calls first/second swipe
+ action accordingly.
+ """
+ is_second_swipe = False
+
+ if self.last_played_content is not None:
+ if (content.type == self.last_played_content.type
+ and content.content == self.last_played_content.content):
+ is_second_swipe = True
+
+ if self._second_swipe_action is not None and is_second_swipe:
+ logger.debug('Calling second swipe action')
+ # run callbacks before second_swipe_action is invoked
+ self.player.play_card_callbacks.run_callbacks(
+ str(content.content),
+ PlayCardState.secondSwipe # Use imported PlayCardState directly
+ )
+ self._second_swipe_action()
+ else:
+ logger.debug('Calling first swipe action')
+ # run callbacks before play_content is invoked
+ self.player.play_card_callbacks.run_callbacks(
+ str(content.content),
+ PlayCardState.firstSwipe # Use imported PlayCardState directly
+ )
+ self._play_content(content)
+
+ self.last_played_content = content
diff --git a/src/jukebox/components/synchronisation/rfidcards/__init__.py b/src/jukebox/components/synchronisation/rfidcards/__init__.py
index 0fa0969a9..10f58541c 100644
--- a/src/jukebox/components/synchronisation/rfidcards/__init__.py
+++ b/src/jukebox/components/synchronisation/rfidcards/__init__.py
@@ -29,7 +29,7 @@
import shutil
from components.rfid.reader import RfidCardDetectState
-from components.playermpd.playcontentcallback import PlayCardState
+from components.playermpd.play_content_callback import PlayCardState
logger = logging.getLogger('jb.sync_rfidcards')
From 2d7f17e295b2fa50f47a7ea0d13f4585f87ea45b Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sun, 3 Nov 2024 23:44:33 +0100
Subject: [PATCH 02/16] RPC Tool supporting quotes in args
---
src/jukebox/run_rpc_tool.py | 224 ++++++++++++++++++++----------------
1 file changed, 128 insertions(+), 96 deletions(-)
diff --git a/src/jukebox/run_rpc_tool.py b/src/jukebox/run_rpc_tool.py
index 4bd834e12..a4e679fd9 100644
--- a/src/jukebox/run_rpc_tool.py
+++ b/src/jukebox/run_rpc_tool.py
@@ -7,12 +7,9 @@
or for debugging.
The tool features auto-completion and command history.
+Now supports quoted arguments for handling spaces in arguments.
The list of available commands is fetched from the running Jukebox service.
-
-.. todo:
- - kwargs support
-
"""
import argparse
@@ -20,17 +17,7 @@
import curses
import curses.ascii
import jukebox.rpc.client as rpc
-
-# Developers note: Scripting at it's dirty end :-)
-
-
-# Careful: curses and default outputs don't mix!
-# In case you'll get an error, most likely your terminal may become funny
-# Best bet: Just don't configure any logger at all!
-# import logging
-# import misc.loggingext
-# logger = misc.loggingext.configure_default(logging.ERROR)
-
+import shlex # Added for proper command parsing
url: str
client: rpc.RpcClient
@@ -77,7 +64,6 @@ def format_help(scr, topic):
sign: str = value['signature']
sign = sign[sign.find('('):]
func = f"{key}{sign}"
- # print(f"{func:50}: {value['description']}")
if key.startswith(topic):
scr.addstr(f"{func:50}: {value['description']}\n")
[y, x] = scr.getyx()
@@ -107,10 +93,14 @@ def format_usage(scr):
scr.addstr("\n\nUsage:\n")
scr.addstr(" > cmd [arg1] [arg2] [arg3]\n")
scr.addstr("e.g.\n")
- scr.addstr(" > volume.ctrl.set_volume 50\n")
- scr.addstr("Note: NOT yet supported: kwargs, quoting!\n")
+ scr.addstr(' > volume.ctrl.set_volume 50\n')
+ scr.addstr(' > player.ctrl.play_album "Bibi & Tina" "Jetzt in Echt - Soundtrack zum Kinofilm"\n')
scr.addstr("\n")
- scr.addstr("Numbers are supported in decimal and hexadecimal format when prefixed with '0x'")
+ scr.addstr("Quoting:\n")
+ scr.addstr(" - Use double quotes (\") for arguments containing spaces\n")
+ scr.addstr(' - Escape quotes within quoted strings with \\\n')
+ scr.addstr("\n")
+ scr.addstr("Numbers are supported in decimal and hexadecimal format when prefixed with '0x'\n")
scr.addstr("\n")
scr.addstr("Use for auto-completion of commands!\n")
scr.addstr("Use / for command history!\n")
@@ -141,12 +131,8 @@ def get_common_beginning(strings):
def autocomplete(msg):
- # logger.debug(f"Autocomplete {msg}")
- # Get all stings that match the beginning
- # candidates = ["ap1", 'ap2', 'appbbb3', 'appbbb4', 'appbbb5', 'appbbb6', 'exit']
matches = [s for s in candidates if s.startswith(msg)]
if len(matches) == 0:
- # Matches is empty: nothing found
return msg, matches
common = get_common_beginning(matches)
return common, matches
@@ -164,7 +150,7 @@ def reprompt(scr, msg, y, x):
scr.move(y, x)
-def get_input(scr): # noqa: C901
+def get_input(scr):
curses.noecho()
ch = 0
msg = ''
@@ -173,6 +159,11 @@ def get_input(scr): # noqa: C901
[y, x] = scr.getyx()
reprompt(scr, msg, y, len(prompt) + len(msg))
scr.refresh()
+
+ # Track if we're inside quotes
+ in_quotes = False
+ escape_next = False
+
while ch != ord(b'\n'):
try:
ch = scr.getch()
@@ -181,7 +172,8 @@ def get_input(scr): # noqa: C901
break
[y, x] = scr.getyx()
pos = x - len(prompt)
- if ch == ord(b'\t'):
+
+ if ch == ord(b'\t') and not in_quotes:
msg, matches = autocomplete(msg)
if len(matches) > 1:
scr.addstr('\n')
@@ -189,18 +181,32 @@ def get_input(scr): # noqa: C901
scr.addstr('\n')
scr.clrtobot()
reprompt(scr, msg, y, len(prompt) + len(msg))
- if ch == ord(b'\n'):
- break
- if ch == 4:
+ elif ch == ord(b'\n'):
+ # Only accept newline if we're not in the middle of a quote
+ if not in_quotes:
+ break
+ else:
+ # Add the newline to multi-line quoted string
+ msg = msg[0:pos] + "\\n" + msg[pos:]
+ reprompt(scr, msg, y, x + 2)
+ elif ch == 4 and not in_quotes:
msg = 'exit'
break
elif ch == curses.KEY_BACKSPACE or ch == 127:
if pos > 0:
+ # Handle backspace in quotes - need to check if we're deleting a quote char
+ if msg[pos-1] == '"' and not escape_next:
+ in_quotes = not in_quotes
+ elif msg[pos-1] == '\\':
+ escape_next = False
scr.delch(y, x - 1)
msg = msg[0:pos - 1] + msg[pos:]
elif ch == curses.KEY_DC:
- scr.delch(y, x)
- msg = msg[0:pos] + msg[pos + 1:]
+ if pos < len(msg):
+ if msg[pos] == '"' and not escape_next:
+ in_quotes = not in_quotes
+ scr.delch(y, x)
+ msg = msg[0:pos] + msg[pos + 1:]
elif ch == curses.KEY_LEFT:
if pos > 0:
scr.move(y, x - 1)
@@ -211,13 +217,13 @@ def get_input(scr): # noqa: C901
scr.move(y, len(prompt))
elif ch == curses.KEY_END:
scr.move(y, len(prompt) + len(msg))
- elif ch == curses.KEY_UP:
+ elif ch == curses.KEY_UP and not in_quotes:
if hidx == len(history):
ihist = msg
hidx = max(hidx - 1, 0)
msg = history[hidx]
reprompt(scr, msg, y, len(prompt) + len(msg))
- elif ch == curses.KEY_DOWN:
+ elif ch == curses.KEY_DOWN and not in_quotes:
hidx = min(hidx + 1, len(history))
if hidx == len(history):
msg = ihist
@@ -225,17 +231,48 @@ def get_input(scr): # noqa: C901
msg = history[hidx]
reprompt(scr, msg, y, len(prompt) + len(msg))
elif is_printable(ch):
- msg = msg[0:pos] + curses.ascii.unctrl(ch) + msg[pos:]
+ char = curses.ascii.unctrl(ch)
+ if char == '"' and not escape_next:
+ in_quotes = not in_quotes
+ elif char == '\\':
+ escape_next = True
+ else:
+ escape_next = False
+ msg = msg[0:pos] + char + msg[pos:]
reprompt(scr, msg, y, x + 1)
- # else:
- # print(f" {ch} -- {type(ch)}")
+
scr.refresh()
+
scr.refresh()
- history.append(msg)
+ if msg:
+ history.append(msg)
return msg
+def parse_command(cmd_str):
+ """
+ Parse command string using shlex to handle quoted arguments properly.
+ Returns (command_parts, args) tuple.
+ """
+ try:
+ parts = shlex.split(cmd_str)
+ if not parts:
+ return [], []
+
+ # Split the command on dots for package.plugin.method
+ cmd_parts = [v for v in parts[0].split('.') if v]
+
+ # Convert args to appropriate types
+ args = [tonum(arg) for arg in parts[1:]]
+
+ return cmd_parts, args
+ except ValueError as e:
+ # Handle unclosed quotes
+ return None, str(e)
+
+
def tonum(string_value):
+ """Convert string to number if possible, otherwise return original string."""
ret = string_value
try:
ret = int(string_value)
@@ -270,79 +307,74 @@ def main(scr):
while cmd != 'exit':
cmd = get_input(scr)
scr.addstr("\n")
- # Split on whitespaces to separate cmd and arg list
- dec = [v for v in cmd.strip().split(' ') if len(v) > 0]
- if len(dec) == 0:
+
+ if cmd == '':
continue
- elif dec[0] == 'help':
+
+ # Handle built-in commands
+ if cmd.startswith('help'):
topic = ''
- if len(dec) > 1:
- topic = dec[1]
+ try:
+ _, args = parse_command(cmd)
+ if args:
+ topic = args[0]
+ except:
+ pass
format_help(scr, topic)
continue
- elif dec[0] == 'usage':
+ elif cmd == 'usage':
format_usage(scr)
continue
- # scr.addstr(f"\n{cmd}\n")
- # Split cmd on '.' into package.plugin.method
- # Remove duplicate '.' along the way
- sl = [v for v in dec[0].split('.') if len(v) > 0]
- fargs = [tonum(a) for a in dec[1:]]
- scr.addstr(f"\n:: Command = {sl}, args = {fargs}\n")
- response = None
- method = None
- if not (2 <= len(sl) <= 3):
- scr.addstr(":: Error = Ill-formatted command\n")
+ elif cmd == 'exit':
+ break
+
+ # Parse the command
+ cmd_parts, args = parse_command(cmd)
+
+ if isinstance(args, str):
+ scr.addstr(f"Error parsing command: {args}\n")
+ continue
+
+ if not (2 <= len(cmd_parts) <= 3):
+ scr.addstr("Error: Invalid command format. Use: package.plugin.method or package.plugin\n")
continue
- if len(sl) == 3:
- method = sl[2]
+
+ method = cmd_parts[2] if len(cmd_parts) == 3 else None
+
try:
- response = client.enque(sl[0], sl[1], method, args=fargs)
+ response = client.enque(cmd_parts[0], cmd_parts[1], method, args=args)
+ scr.addstr(f"\n:: Response =\n{response}\n\n")
except zmq.error.Again:
scr.addstr("\n\n" + '-' * 70 + "\n")
scr.addstr("Could not reach RPC Server. Jukebox running? Correct Port?\n")
scr.addstr('-' * 70 + "\n\n")
- scr.refresh()
except Exception as e:
- scr.addstr(f":: Exception response =\n{e}\n")
- else:
- scr.addstr(f"\n:: Response =\n{response}\n\n")
+ scr.addstr(f":: Error: {str(e)}\n")
def runcmd(cmd):
- """
- Just run a command.
- Right now duplicates more or less main()
- :todo remove duplication of code
- """
+ """Run a single command and exit."""
+ cmd_parts, args = parse_command(cmd)
- # Split on whitespaces to separate cmd and arg list
- dec = [v for v in cmd.strip().split(' ') if len(v) > 0]
- if len(dec) == 0:
+ if isinstance(args, str):
+ print(f"Error parsing command: {args}")
return
- # Split cmd on '.' into package.plugin.method
- # Remove duplicate '.' along the way
- sl = [v for v in dec[0].split('.') if len(v) > 0]
- fargs = [tonum(a) for a in dec[1:]]
- response = None
- method = None
- if not (2 <= len(sl) <= 3):
- print(":: Error = Ill-formatted command\n")
+
+ if not (2 <= len(cmd_parts) <= 3):
+ print("Error: Invalid command format. Use: package.plugin.method or package.plugin")
return
- if len(sl) == 3:
- method = sl[2]
+
+ method = cmd_parts[2] if len(cmd_parts) == 3 else None
+
try:
- response = client.enque(sl[0], sl[1], method, args=fargs)
+ response = client.enque(cmd_parts[0], cmd_parts[1], method, args=args)
+ print(f"\n:: Response =\n{response}\n")
except zmq.error.Again:
- print("\n\n" + '-' * 70 + "\n")
- print("Could not reach RPC Server. Jukebox running? Correct Port?\n")
- print('-' * 70 + "\n\n")
- return
+ print("\n\n" + '-' * 70)
+ print("Could not reach RPC Server. Jukebox running? Correct Port?")
+ print('-' * 70 + "\n")
except Exception as e:
- print(f":: Exception response =\n{e}\n")
- return
- else:
- print(f"\n:: Response =\n{response}\n\n")
+ print(f":: Error: {str(e)}")
if __name__ == '__main__':
@@ -350,19 +382,19 @@ def runcmd(cmd):
default_ws = 5556
url = f"tcp://localhost:{default_tcp}"
argparser = argparse.ArgumentParser(description='The Jukebox RPC command line tool',
- epilog=f'Default connection: {url}')
+ epilog=f'Default connection: {url}')
port_group = argparser.add_mutually_exclusive_group()
port_group.add_argument("-w", "--websocket",
- help=f"Use websocket protocol on PORT [default: {default_ws}]",
- nargs='?', const=default_ws,
- metavar="PORT", default=None)
+ help=f"Use websocket protocol on PORT [default: {default_ws}]",
+ nargs='?', const=default_ws,
+ metavar="PORT", default=None)
port_group.add_argument("-t", "--tcp",
- help=f"Use tcp protocol on PORT [default: {default_tcp}]",
- nargs='?', const=default_tcp,
- metavar="PORT", default=None)
+ help=f"Use tcp protocol on PORT [default: {default_tcp}]",
+ nargs='?', const=default_tcp,
+ metavar="PORT", default=None)
port_group.add_argument("-c", "--command",
- help="Send command to Jukebox server",
- default=None)
+ help="Send command to Jukebox server",
+ default=None)
args = argparser.parse_args()
if args.websocket is not None:
From d6f5eadbe25cc48fa651d885f4f9c29ed72ca9ca Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sat, 16 Nov 2024 18:18:20 +0100
Subject: [PATCH 03/16] Introduce play_from_reader only to support second_swipe
with RFID cards
---
src/jukebox/components/playermpd/__init__.py | 133 +++++++++++++------
src/jukebox/run_rpc_tool.py | 31 ++++-
2 files changed, 117 insertions(+), 47 deletions(-)
diff --git a/src/jukebox/components/playermpd/__init__.py b/src/jukebox/components/playermpd/__init__.py
index 96969ca95..f5f8669bd 100644
--- a/src/jukebox/components/playermpd/__init__.py
+++ b/src/jukebox/components/playermpd/__init__.py
@@ -88,7 +88,7 @@
import time
import functools
from pathlib import Path
-from typing import Union, Dict, Any
+from typing import Union, Dict, Any, Optional
import components.player
import jukebox.cfghandler
@@ -151,38 +151,38 @@ def __init__(self):
self.music_player_status = self.nvm.load(cfg.getn('playermpd', 'status_file'))
self.second_swipe_action_dict = {'toggle': self.toggle,
- 'play': self.play,
- 'skip': self.next,
- 'rewind': self.rewind,
- 'replay': self.replay,
- 'replay_if_stopped': self.replay_if_stopped}
+ 'play': self.play,
+ 'skip': self.next,
+ 'rewind': self.rewind,
+ 'replay': self.replay,
+ 'replay_if_stopped': self.replay_if_stopped}
self.second_swipe_action = None
self.decode_2nd_swipe_option()
self.end_of_playlist_next_action = utils.get_config_action(cfg,
- 'playermpd',
- 'end_of_playlist_next_action',
- 'none',
- {'rewind': self.rewind,
+ 'playermpd',
+ 'end_of_playlist_next_action',
+ 'none',
+ {'rewind': self.rewind,
'stop': self.stop,
'none': lambda: None},
- logger)
+ logger)
self.stopped_prev_action = utils.get_config_action(cfg,
- 'playermpd',
- 'stopped_prev_action',
- 'prev',
- {'rewind': self.rewind,
+ 'playermpd',
+ 'stopped_prev_action',
+ 'prev',
+ {'rewind': self.rewind,
'prev': self._prev_in_stopped_state,
'none': lambda: None},
- logger)
+ logger)
self.stopped_next_action = utils.get_config_action(cfg,
- 'playermpd',
- 'stopped_next_action',
- 'next',
- {'rewind': self.rewind,
- 'next': self._next_in_stopped_state,
- 'none': lambda: None},
- logger)
+ 'playermpd',
+ 'stopped_next_action',
+ 'next',
+ {'rewind': self.rewind,
+ 'next': self._next_in_stopped_state,
+ 'none': lambda: None},
+ logger)
self.mpd_client = mpd.MPDClient()
self.coverart_cache_manager = CoverartCacheManager()
@@ -236,7 +236,7 @@ def __init__(self):
# self.status_thread = threading.Timer(self.mpd_status_poll_interval, self._mpd_status_poll).start()
self.status_thread = multitimer.GenericEndlessTimerClass('mpd.timer_status',
- self.mpd_status_poll_interval, self._mpd_status_poll)
+ self.mpd_status_poll_interval, self._mpd_status_poll)
self.status_thread.start()
def exit(self):
@@ -570,57 +570,101 @@ def _play_folder_internal(self, folder: str, recursive: bool) -> None:
self.mpd_client.play()
@plugs.tag
- def play_content(self, content: Union[str, Dict[str, str]], content_type: str = 'folder', recursive: bool = False):
+ def play_content(self, content: Union[str, Dict[str, Any]], content_type: str = 'folder', recursive: bool = False):
"""
Main entry point for trigger music playing from any source (RFID reader, web UI, etc.).
- Supports second swipe for all content types.
+ Does NOT support second swipe - use play_from_reader() for that.
- :param content: Content identifier, either:
- - string path for single/folder types
- - dict with 'artist' and 'album' keys for album type
+ :param content: Content identifier:
+ - For singles/folders: file/folder path as string
+ - For albums: dict with 'artist' and 'album' keys
:param content_type: Type of content ('single', 'album', 'folder')
:param recursive: Add folder recursively (only used for folder type)
"""
try:
content_type = content_type.lower()
if content_type == 'album':
- if not isinstance(content, dict):
+ if isinstance(content, dict):
+ artist = content.get('artist')
+ album = content.get('album')
+ if not artist or not album:
+ raise ValueError("Album content must contain both 'artist' and 'album' keys")
+ else:
raise ValueError("Album content must be a dictionary with 'artist' and 'album' keys")
- if 'artist' not in content or 'album' not in content:
- raise ValueError("Album content dictionary must contain both 'artist' and 'album' keys")
play_content = PlayContent(
type=PlayContentType.ALBUM,
- content=(content['artist'], content['album'])
+ content=(artist, album)
)
elif content_type == 'single':
- if not isinstance(content, str):
- raise ValueError("Single track content must be a string path")
+ if isinstance(content, dict):
+ raise ValueError("Single track content should be a direct file path, not a dictionary")
play_content = PlayContent(
type=PlayContentType.SINGLE,
content=content
)
else: # folder is default
- if not isinstance(content, str):
- raise ValueError("Folder content must be a string path")
+ if isinstance(content, dict):
+ raise ValueError("Folder content should be a direct folder path, not a dictionary")
play_content = PlayContent(
type=PlayContentType.FOLDER,
content=content,
recursive=recursive
)
- self.play_content_handler.play_content(play_content)
+ # Ensure no second swipe for regular content playback
+ old_action = self.play_content_handler._second_swipe_action
+ self.play_content_handler._second_swipe_action = None
+
+ try:
+ self.play_content_handler.play_content(play_content)
+ finally:
+ # Restore previous second swipe action
+ self.play_content_handler._second_swipe_action = old_action
except Exception as e:
logger.error(f"Error playing content: {e}")
raise
- # Legacy/compatibility methods
-
@plugs.tag
- def play_card(self, folder: str, recursive: bool = False):
- """Legacy method for RFID cards - redirects to play_content"""
- return self.play_content(folder, content_type='folder', recursive=recursive)
+ def play_from_reader(self, content: Union[str, Dict[str, str]], content_type: str = 'folder',
+ recursive: bool = False, second_swipe: Optional[str] = None):
+ """
+ Special entry point for reader-triggered playback with second swipe support.
+ Used when content is identified via RFID, barcode, or other physical readers.
+
+ :param content: Content identifier, either:
+ - string path for single/folder types
+ - dict with 'artist' and 'album' keys for album type
+ :param content_type: Type of content ('single', 'album', 'folder')
+ :param recursive: Add folder recursively (only used for folder type)
+ :param second_swipe: Override default second swipe action for this reader:
+ - None/not specified: use default from config
+ - 'none': disable second swipe
+ - One of: 'toggle', 'play', 'skip', 'rewind', 'replay', 'replay_if_stopped'
+ """
+ # Determine second swipe action
+ if second_swipe is None:
+ action = self.second_swipe_action
+ elif second_swipe.lower() == 'none':
+ action = None
+ else:
+ action = self.second_swipe_action_dict.get(second_swipe.lower())
+ if action is None:
+ logger.error(f"Unknown second swipe action '{second_swipe}', using default")
+ action = self.second_swipe_action
+
+ # Temporarily set the chosen second swipe action
+ old_action = self.play_content_handler._second_swipe_action
+ self.play_content_handler.set_second_swipe_action(action)
+
+ try:
+ self.play_content(content, content_type, recursive)
+ finally:
+ # Restore previous second swipe action
+ self.play_content_handler.set_second_swipe_action(old_action)
+
+ # The following methods are kept for backward compatibility but now use play_content internally
@plugs.tag
def play_single(self, song_url):
@@ -630,7 +674,8 @@ def play_single(self, song_url):
@plugs.tag
def play_album(self, albumartist: str, album: str):
"""Deprecated: Use play_content with content_type='album' instead"""
- self.play_content({'artist': albumartist, 'album': album}, content_type='album')
+ content = {'artist': albumartist, 'album': album}
+ self.play_content(content, content_type='album')
@plugs.tag
def play_folder(self, folder: str, recursive: bool = False):
diff --git a/src/jukebox/run_rpc_tool.py b/src/jukebox/run_rpc_tool.py
index a4e679fd9..65da74540 100644
--- a/src/jukebox/run_rpc_tool.py
+++ b/src/jukebox/run_rpc_tool.py
@@ -94,13 +94,38 @@ def format_usage(scr):
scr.addstr(" > cmd [arg1] [arg2] [arg3]\n")
scr.addstr("e.g.\n")
scr.addstr(' > volume.ctrl.set_volume 50\n')
- scr.addstr(' > player.ctrl.play_album "Bibi & Tina" "Jetzt in Echt - Soundtrack zum Kinofilm"\n')
+
+ # General content playback examples (for webapp/CLI)
+ scr.addstr("\nPlaying content (using play_content):\n")
+ scr.addstr(' > player.ctrl.play_content \'{"artist":"Pink Floyd","album":"The Wall"}\' album\n')
+ scr.addstr(' > player.ctrl.play_content "/music/classical" folder true\n')
+ scr.addstr(' > player.ctrl.play_content "/music/favorites/track.mp3" single\n')
+
+ # RFID card specific examples
+ scr.addstr("\nPlaying content with card behavior (using play_card):\n")
+ scr.addstr(' > player.ctrl.play_card \'{"artist":"Pink Floyd","album":"The Wall"}\' album false toggle\n')
+ scr.addstr(' > player.ctrl.play_card "/music/classical" folder true replay\n')
+ scr.addstr(' > player.ctrl.play_card "/music/stories" folder false none\n')
+
scr.addstr("\n")
scr.addstr("Quoting:\n")
- scr.addstr(" - Use double quotes (\") for arguments containing spaces\n")
+ scr.addstr(" - Use single quotes (\') for JSON content\n")
+ scr.addstr(' - Use double quotes (") for simple string arguments containing spaces\n')
scr.addstr(' - Escape quotes within quoted strings with \\\n')
scr.addstr("\n")
- scr.addstr("Numbers are supported in decimal and hexadecimal format when prefixed with '0x'\n")
+ scr.addstr("Content Types:\n")
+ scr.addstr(" - album: requires JSON with artist and album\n")
+ scr.addstr(" - single: direct path to audio file\n")
+ scr.addstr(" - folder: path to folder (optional recursive flag)\n")
+ scr.addstr("\n")
+ scr.addstr("Second Swipe Actions (for play_card):\n")
+ scr.addstr(" - none: disable second swipe\n")
+ scr.addstr(" - toggle: toggle play/pause\n")
+ scr.addstr(" - play: start playing\n")
+ scr.addstr(" - skip: next track\n")
+ scr.addstr(" - rewind: restart playlist\n")
+ scr.addstr(" - replay: restart folder\n")
+ scr.addstr(" - replay_if_stopped: restart if stopped\n")
scr.addstr("\n")
scr.addstr("Use for auto-completion of commands!\n")
scr.addstr("Use / for command history!\n")
From c792037665e9bf411f6443cae855a70844f4f43f Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sat, 16 Nov 2024 18:19:29 +0100
Subject: [PATCH 04/16] Update Webapp to use new play_content api
---
src/webapp/src/commands/index.js | 18 +++---------------
.../albums/song-list/song-list-controls.js | 2 +-
.../lists/albums/song-list/song-list-item.js | 2 +-
.../Library/lists/folders/folder-list-item.js | 4 ++--
4 files changed, 7 insertions(+), 19 deletions(-)
diff --git a/src/webapp/src/commands/index.js b/src/webapp/src/commands/index.js
index 1e984997e..ffbc01f03 100644
--- a/src/webapp/src/commands/index.js
+++ b/src/webapp/src/commands/index.js
@@ -59,23 +59,11 @@ const commands = {
plugin: 'ctrl',
method: 'play',
},
- play_single: {
+ play_content: {
_package: 'player',
plugin: 'ctrl',
- method: 'play_single',
- argKeys: ['song_url']
- },
- play_folder: {
- _package: 'player',
- plugin: 'ctrl',
- method: 'play_folder',
- argKeys: ['folder']
- },
- play_album: {
- _package: 'player',
- plugin: 'ctrl',
- method: 'play_album',
- argKeys: ['albumartist', 'album']
+ method: 'play_content',
+ argKeys: ['content', 'content_type', 'recursive']
},
pause: {
_package: 'player',
diff --git a/src/webapp/src/components/Library/lists/albums/song-list/song-list-controls.js b/src/webapp/src/components/Library/lists/albums/song-list/song-list-controls.js
index b2391819e..18b240382 100644
--- a/src/webapp/src/components/Library/lists/albums/song-list/song-list-controls.js
+++ b/src/webapp/src/components/Library/lists/albums/song-list/song-list-controls.js
@@ -21,7 +21,7 @@ const SongListControls = ({
const command = 'play_album';
const playAlbum = () => (
- request(command, { albumartist, album })
+ request('play_content', { content: { "artist": albumartist, album }, content_type: 'album' })
);
const registerAlbumToCard = () => (
diff --git a/src/webapp/src/components/Library/lists/albums/song-list/song-list-item.js b/src/webapp/src/components/Library/lists/albums/song-list/song-list-item.js
index 0f22d2df3..4b76a984c 100644
--- a/src/webapp/src/components/Library/lists/albums/song-list/song-list-item.js
+++ b/src/webapp/src/components/Library/lists/albums/song-list/song-list-item.js
@@ -26,7 +26,7 @@ const SongListItem = ({
} = song;
const playSingle = () => {
- request(command, { song_url: file })
+ request('play_content', { content: file, content_type: 'single' })
}
const registerSongToCard = () => (
diff --git a/src/webapp/src/components/Library/lists/folders/folder-list-item.js b/src/webapp/src/components/Library/lists/folders/folder-list-item.js
index 3be77fbbc..90bb1d6f6 100644
--- a/src/webapp/src/components/Library/lists/folders/folder-list-item.js
+++ b/src/webapp/src/components/Library/lists/folders/folder-list-item.js
@@ -24,8 +24,8 @@ const FolderListItem = ({
const playItem = () => {
switch(type) {
- case 'directory': return request('play_folder', { folder: relpath, recursive: true });
- case 'file': return request('play_single', { song_url: relpath });
+ case 'directory': return request('play_content', { content: relpath, content_type: 'folder', recursive: true });
+ case 'file': request('play_content', { content: relpath, content_type: 'single' });
// TODO: Add missing Podcast
// TODO: Add missing Stream
default: return;
From 76402e195fbfe294b35582457d3644987535d1d2 Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sat, 16 Nov 2024 18:33:33 +0100
Subject: [PATCH 05/16] fix flake8
---
src/jukebox/run_rpc_tool.py | 214 ++++++++++++++++--------------------
1 file changed, 93 insertions(+), 121 deletions(-)
diff --git a/src/jukebox/run_rpc_tool.py b/src/jukebox/run_rpc_tool.py
index 65da74540..d09a713db 100644
--- a/src/jukebox/run_rpc_tool.py
+++ b/src/jukebox/run_rpc_tool.py
@@ -7,9 +7,12 @@
or for debugging.
The tool features auto-completion and command history.
-Now supports quoted arguments for handling spaces in arguments.
The list of available commands is fetched from the running Jukebox service.
+
+.. todo:
+ - kwargs support
+
"""
import argparse
@@ -17,7 +20,17 @@
import curses
import curses.ascii
import jukebox.rpc.client as rpc
-import shlex # Added for proper command parsing
+
+# Developers note: Scripting at it's dirty end :-)
+
+
+# Careful: curses and default outputs don't mix!
+# In case you'll get an error, most likely your terminal may become funny
+# Best bet: Just don't configure any logger at all!
+# import logging
+# import misc.loggingext
+# logger = misc.loggingext.configure_default(logging.ERROR)
+
url: str
client: rpc.RpcClient
@@ -64,6 +77,7 @@ def format_help(scr, topic):
sign: str = value['signature']
sign = sign[sign.find('('):]
func = f"{key}{sign}"
+ # print(f"{func:50}: {value['description']}")
if key.startswith(topic):
scr.addstr(f"{func:50}: {value['description']}\n")
[y, x] = scr.getyx()
@@ -156,8 +170,12 @@ def get_common_beginning(strings):
def autocomplete(msg):
+ # logger.debug(f"Autocomplete {msg}")
+ # Get all stings that match the beginning
+ # candidates = ["ap1", 'ap2', 'appbbb3', 'appbbb4', 'appbbb5', 'appbbb6', 'exit']
matches = [s for s in candidates if s.startswith(msg)]
if len(matches) == 0:
+ # Matches is empty: nothing found
return msg, matches
common = get_common_beginning(matches)
return common, matches
@@ -175,7 +193,7 @@ def reprompt(scr, msg, y, x):
scr.move(y, x)
-def get_input(scr):
+def get_input(scr): # noqa: C901
curses.noecho()
ch = 0
msg = ''
@@ -184,11 +202,6 @@ def get_input(scr):
[y, x] = scr.getyx()
reprompt(scr, msg, y, len(prompt) + len(msg))
scr.refresh()
-
- # Track if we're inside quotes
- in_quotes = False
- escape_next = False
-
while ch != ord(b'\n'):
try:
ch = scr.getch()
@@ -197,8 +210,7 @@ def get_input(scr):
break
[y, x] = scr.getyx()
pos = x - len(prompt)
-
- if ch == ord(b'\t') and not in_quotes:
+ if ch == ord(b'\t'):
msg, matches = autocomplete(msg)
if len(matches) > 1:
scr.addstr('\n')
@@ -206,32 +218,18 @@ def get_input(scr):
scr.addstr('\n')
scr.clrtobot()
reprompt(scr, msg, y, len(prompt) + len(msg))
- elif ch == ord(b'\n'):
- # Only accept newline if we're not in the middle of a quote
- if not in_quotes:
- break
- else:
- # Add the newline to multi-line quoted string
- msg = msg[0:pos] + "\\n" + msg[pos:]
- reprompt(scr, msg, y, x + 2)
- elif ch == 4 and not in_quotes:
+ if ch == ord(b'\n'):
+ break
+ if ch == 4:
msg = 'exit'
break
elif ch == curses.KEY_BACKSPACE or ch == 127:
if pos > 0:
- # Handle backspace in quotes - need to check if we're deleting a quote char
- if msg[pos-1] == '"' and not escape_next:
- in_quotes = not in_quotes
- elif msg[pos-1] == '\\':
- escape_next = False
scr.delch(y, x - 1)
msg = msg[0:pos - 1] + msg[pos:]
elif ch == curses.KEY_DC:
- if pos < len(msg):
- if msg[pos] == '"' and not escape_next:
- in_quotes = not in_quotes
- scr.delch(y, x)
- msg = msg[0:pos] + msg[pos + 1:]
+ scr.delch(y, x)
+ msg = msg[0:pos] + msg[pos + 1:]
elif ch == curses.KEY_LEFT:
if pos > 0:
scr.move(y, x - 1)
@@ -242,13 +240,13 @@ def get_input(scr):
scr.move(y, len(prompt))
elif ch == curses.KEY_END:
scr.move(y, len(prompt) + len(msg))
- elif ch == curses.KEY_UP and not in_quotes:
+ elif ch == curses.KEY_UP:
if hidx == len(history):
ihist = msg
hidx = max(hidx - 1, 0)
msg = history[hidx]
reprompt(scr, msg, y, len(prompt) + len(msg))
- elif ch == curses.KEY_DOWN and not in_quotes:
+ elif ch == curses.KEY_DOWN:
hidx = min(hidx + 1, len(history))
if hidx == len(history):
msg = ihist
@@ -256,48 +254,17 @@ def get_input(scr):
msg = history[hidx]
reprompt(scr, msg, y, len(prompt) + len(msg))
elif is_printable(ch):
- char = curses.ascii.unctrl(ch)
- if char == '"' and not escape_next:
- in_quotes = not in_quotes
- elif char == '\\':
- escape_next = True
- else:
- escape_next = False
- msg = msg[0:pos] + char + msg[pos:]
+ msg = msg[0:pos] + curses.ascii.unctrl(ch) + msg[pos:]
reprompt(scr, msg, y, x + 1)
-
+ # else:
+ # print(f" {ch} -- {type(ch)}")
scr.refresh()
-
scr.refresh()
- if msg:
- history.append(msg)
+ history.append(msg)
return msg
-def parse_command(cmd_str):
- """
- Parse command string using shlex to handle quoted arguments properly.
- Returns (command_parts, args) tuple.
- """
- try:
- parts = shlex.split(cmd_str)
- if not parts:
- return [], []
-
- # Split the command on dots for package.plugin.method
- cmd_parts = [v for v in parts[0].split('.') if v]
-
- # Convert args to appropriate types
- args = [tonum(arg) for arg in parts[1:]]
-
- return cmd_parts, args
- except ValueError as e:
- # Handle unclosed quotes
- return None, str(e)
-
-
def tonum(string_value):
- """Convert string to number if possible, otherwise return original string."""
ret = string_value
try:
ret = int(string_value)
@@ -332,74 +299,79 @@ def main(scr):
while cmd != 'exit':
cmd = get_input(scr)
scr.addstr("\n")
-
- if cmd == '':
+ # Split on whitespaces to separate cmd and arg list
+ dec = [v for v in cmd.strip().split(' ') if len(v) > 0]
+ if len(dec) == 0:
continue
-
- # Handle built-in commands
- if cmd.startswith('help'):
+ elif dec[0] == 'help':
topic = ''
- try:
- _, args = parse_command(cmd)
- if args:
- topic = args[0]
- except:
- pass
+ if len(dec) > 1:
+ topic = dec[1]
format_help(scr, topic)
continue
- elif cmd == 'usage':
+ elif dec[0] == 'usage':
format_usage(scr)
continue
- elif cmd == 'exit':
- break
-
- # Parse the command
- cmd_parts, args = parse_command(cmd)
-
- if isinstance(args, str):
- scr.addstr(f"Error parsing command: {args}\n")
+ # scr.addstr(f"\n{cmd}\n")
+ # Split cmd on '.' into package.plugin.method
+ # Remove duplicate '.' along the way
+ sl = [v for v in dec[0].split('.') if len(v) > 0]
+ fargs = [tonum(a) for a in dec[1:]]
+ scr.addstr(f"\n:: Command = {sl}, args = {fargs}\n")
+ response = None
+ method = None
+ if not (2 <= len(sl) <= 3):
+ scr.addstr(":: Error = Ill-formatted command\n")
continue
-
- if not (2 <= len(cmd_parts) <= 3):
- scr.addstr("Error: Invalid command format. Use: package.plugin.method or package.plugin\n")
- continue
-
- method = cmd_parts[2] if len(cmd_parts) == 3 else None
-
+ if len(sl) == 3:
+ method = sl[2]
try:
- response = client.enque(cmd_parts[0], cmd_parts[1], method, args=args)
- scr.addstr(f"\n:: Response =\n{response}\n\n")
+ response = client.enque(sl[0], sl[1], method, args=fargs)
except zmq.error.Again:
scr.addstr("\n\n" + '-' * 70 + "\n")
scr.addstr("Could not reach RPC Server. Jukebox running? Correct Port?\n")
scr.addstr('-' * 70 + "\n\n")
+ scr.refresh()
except Exception as e:
- scr.addstr(f":: Error: {str(e)}\n")
+ scr.addstr(f":: Exception response =\n{e}\n")
+ else:
+ scr.addstr(f"\n:: Response =\n{response}\n\n")
def runcmd(cmd):
- """Run a single command and exit."""
- cmd_parts, args = parse_command(cmd)
+ """
+ Just run a command.
+ Right now duplicates more or less main()
+ :todo remove duplication of code
+ """
- if isinstance(args, str):
- print(f"Error parsing command: {args}")
+ # Split on whitespaces to separate cmd and arg list
+ dec = [v for v in cmd.strip().split(' ') if len(v) > 0]
+ if len(dec) == 0:
return
-
- if not (2 <= len(cmd_parts) <= 3):
- print("Error: Invalid command format. Use: package.plugin.method or package.plugin")
+ # Split cmd on '.' into package.plugin.method
+ # Remove duplicate '.' along the way
+ sl = [v for v in dec[0].split('.') if len(v) > 0]
+ fargs = [tonum(a) for a in dec[1:]]
+ response = None
+ method = None
+ if not (2 <= len(sl) <= 3):
+ print(":: Error = Ill-formatted command\n")
return
-
- method = cmd_parts[2] if len(cmd_parts) == 3 else None
-
+ if len(sl) == 3:
+ method = sl[2]
try:
- response = client.enque(cmd_parts[0], cmd_parts[1], method, args=args)
- print(f"\n:: Response =\n{response}\n")
+ response = client.enque(sl[0], sl[1], method, args=fargs)
except zmq.error.Again:
- print("\n\n" + '-' * 70)
- print("Could not reach RPC Server. Jukebox running? Correct Port?")
- print('-' * 70 + "\n")
+ print("\n\n" + '-' * 70 + "\n")
+ print("Could not reach RPC Server. Jukebox running? Correct Port?\n")
+ print('-' * 70 + "\n\n")
+ return
except Exception as e:
- print(f":: Error: {str(e)}")
+ print(f":: Exception response =\n{e}\n")
+ return
+ else:
+ print(f"\n:: Response =\n{response}\n\n")
if __name__ == '__main__':
@@ -407,19 +379,19 @@ def runcmd(cmd):
default_ws = 5556
url = f"tcp://localhost:{default_tcp}"
argparser = argparse.ArgumentParser(description='The Jukebox RPC command line tool',
- epilog=f'Default connection: {url}')
+ epilog=f'Default connection: {url}')
port_group = argparser.add_mutually_exclusive_group()
port_group.add_argument("-w", "--websocket",
- help=f"Use websocket protocol on PORT [default: {default_ws}]",
- nargs='?', const=default_ws,
- metavar="PORT", default=None)
+ help=f"Use websocket protocol on PORT [default: {default_ws}]",
+ nargs='?', const=default_ws,
+ metavar="PORT", default=None)
port_group.add_argument("-t", "--tcp",
- help=f"Use tcp protocol on PORT [default: {default_tcp}]",
- nargs='?', const=default_tcp,
- metavar="PORT", default=None)
+ help=f"Use tcp protocol on PORT [default: {default_tcp}]",
+ nargs='?', const=default_tcp,
+ metavar="PORT", default=None)
port_group.add_argument("-c", "--command",
- help="Send command to Jukebox server",
- default=None)
+ help="Send command to Jukebox server",
+ default=None)
args = argparser.parse_args()
if args.websocket is not None:
From 0a54a81995806303bc72698022facad7f240c758 Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sat, 16 Nov 2024 18:37:27 +0100
Subject: [PATCH 06/16] Update docs on rpc for play_from_reader
---
src/jukebox/run_rpc_tool.py | 10 +++++-----
1 file changed, 5 insertions(+), 5 deletions(-)
diff --git a/src/jukebox/run_rpc_tool.py b/src/jukebox/run_rpc_tool.py
index d09a713db..47f861af5 100644
--- a/src/jukebox/run_rpc_tool.py
+++ b/src/jukebox/run_rpc_tool.py
@@ -115,11 +115,11 @@ def format_usage(scr):
scr.addstr(' > player.ctrl.play_content "/music/classical" folder true\n')
scr.addstr(' > player.ctrl.play_content "/music/favorites/track.mp3" single\n')
- # RFID card specific examples
- scr.addstr("\nPlaying content with card behavior (using play_card):\n")
- scr.addstr(' > player.ctrl.play_card \'{"artist":"Pink Floyd","album":"The Wall"}\' album false toggle\n')
- scr.addstr(' > player.ctrl.play_card "/music/classical" folder true replay\n')
- scr.addstr(' > player.ctrl.play_card "/music/stories" folder false none\n')
+ # Update the reader examples section
+ scr.addstr("\nPlaying content from physical readers (using play_from_reader):\n")
+ scr.addstr(' > player.ctrl.play_from_reader \'{"artist":"Pink Floyd","album":"The Wall"}\' album false toggle\n')
+ scr.addstr(' > player.ctrl.play_from_reader "/music/classical" folder true replay\n')
+ scr.addstr(' > player.ctrl.play_from_reader "/music/stories" folder false none\n')
scr.addstr("\n")
scr.addstr("Quoting:\n")
From c3040d1996ce582ddb46945e8730a8748d092a2a Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sun, 17 Nov 2024 13:01:41 +0100
Subject: [PATCH 07/16] Adapt Webapp to new play_content and play_from_reader
API
---
src/jukebox/components/playermpd/__init__.py | 14 +++---
src/jukebox/components/rfid/cards/__init__.py | 46 +++++++++++--------
src/jukebox/components/rpc_command_alias.py | 6 +--
src/webapp/public/locales/de/translation.json | 17 +++----
src/webapp/public/locales/en/translation.json | 17 +++----
src/webapp/src/commands/index.js | 18 ++++++++
.../Cards/controls/actions-controls.js | 8 ++--
.../{play-music => play-content}/index.js | 23 +++++-----
.../no-music-selected.js | 2 +-
.../selected-album.js | 2 +-
.../selected-folder.js | 0
.../selected-single.js | 2 +-
.../Cards/controls/controls-selector.js | 6 +--
src/webapp/src/components/Cards/edit.js | 4 +-
src/webapp/src/components/Cards/list.js | 10 +++-
src/webapp/src/components/Cards/register.js | 1 +
src/webapp/src/components/Cards/utils.js | 5 ++
.../Library/lists/albums/song-list/index.js | 6 +--
.../albums/song-list/song-list-controls.js | 7 ++-
.../lists/albums/song-list/song-list-item.js | 9 ++--
.../Library/lists/folders/folder-list-item.js | 6 +--
.../Library/lists/folders/folder-list.js | 4 +-
.../components/Library/lists/folders/index.js | 4 +-
.../src/components/Library/lists/index.js | 8 ++--
src/webapp/src/config.js | 6 +--
src/webapp/src/utils/utils.js | 11 +++++
26 files changed, 145 insertions(+), 97 deletions(-)
rename src/webapp/src/components/Cards/controls/actions/{play-music => play-content}/index.js (69%)
rename src/webapp/src/components/Cards/controls/actions/{play-music => play-content}/no-music-selected.js (80%)
rename src/webapp/src/components/Cards/controls/actions/{play-music => play-content}/selected-album.js (88%)
rename src/webapp/src/components/Cards/controls/actions/{play-music => play-content}/selected-folder.js (100%)
rename src/webapp/src/components/Cards/controls/actions/{play-music => play-content}/selected-single.js (95%)
diff --git a/src/jukebox/components/playermpd/__init__.py b/src/jukebox/components/playermpd/__init__.py
index f5f8669bd..161b81679 100644
--- a/src/jukebox/components/playermpd/__init__.py
+++ b/src/jukebox/components/playermpd/__init__.py
@@ -577,7 +577,7 @@ def play_content(self, content: Union[str, Dict[str, Any]], content_type: str =
:param content: Content identifier:
- For singles/folders: file/folder path as string
- - For albums: dict with 'artist' and 'album' keys
+ - For albums: dict with 'albumartist' and 'album' keys
:param content_type: Type of content ('single', 'album', 'folder')
:param recursive: Add folder recursively (only used for folder type)
"""
@@ -585,16 +585,16 @@ def play_content(self, content: Union[str, Dict[str, Any]], content_type: str =
content_type = content_type.lower()
if content_type == 'album':
if isinstance(content, dict):
- artist = content.get('artist')
+ albumartist = content.get('albumartist')
album = content.get('album')
- if not artist or not album:
- raise ValueError("Album content must contain both 'artist' and 'album' keys")
+ if not albumartist or not album:
+ raise ValueError("Album content must contain both 'albumartist' and 'album' keys")
else:
- raise ValueError("Album content must be a dictionary with 'artist' and 'album' keys")
+ raise ValueError("Album content must be a dictionary with 'albumartist' and 'album' keys")
play_content = PlayContent(
type=PlayContentType.ALBUM,
- content=(artist, album)
+ content=(albumartist, album)
)
elif content_type == 'single':
if isinstance(content, dict):
@@ -635,7 +635,7 @@ def play_from_reader(self, content: Union[str, Dict[str, str]], content_type: st
:param content: Content identifier, either:
- string path for single/folder types
- - dict with 'artist' and 'album' keys for album type
+ - dict with 'albumartist' and 'album' keys for album type
:param content_type: Type of content ('single', 'album', 'folder')
:param recursive: Add folder recursively (only used for folder type)
:param second_swipe: Override default second swipe action for this reader:
diff --git a/src/jukebox/components/rfid/cards/__init__.py b/src/jukebox/components/rfid/cards/__init__.py
index 65e3ff8b9..1413f791e 100644
--- a/src/jukebox/components/rfid/cards/__init__.py
+++ b/src/jukebox/components/rfid/cards/__init__.py
@@ -17,7 +17,7 @@
import logging
import time
-from typing import (List, Dict, Optional)
+from typing import List, Dict, Optional, Union
import jukebox.utils as utils
import jukebox.cfghandler
import jukebox.plugs as plugs
@@ -89,42 +89,48 @@ def delete_card(card_id: str, auto_save: bool = True):
@plugs.register
def register_card(card_id: str, cmd_alias: str,
- args: Optional[List] = None, kwargs: Optional[Dict] = None,
- ignore_card_removal_action: Optional[bool] = None, ignore_same_id_delay: Optional[bool] = None,
- overwrite: bool = False,
- auto_save: bool = True):
- """Register a new card based on quick-selection
-
- If you are going to call this through the RPC it will get a little verbose
-
- **Example:** Registering a new card with ID *0009* for increment volume with a custom argument to inc_volume
- (*here: 15*) and custom *ignore_same_id_delay value*::
-
- plugin.call_ignore_errors('cards', 'register_card',
- args=['0009', 'inc_volume'],
- kwargs={'args': [15], 'ignore_same_id_delay': True, 'overwrite': True})
-
- """
+ args: Optional[Union[List, Dict]] = None,
+ kwargs: Optional[Dict] = None,
+ ignore_card_removal_action: Optional[bool] = None,
+ ignore_same_id_delay: Optional[bool] = None,
+ overwrite: bool = False,
+ auto_save: bool = True):
+ """Register a new card based on quick-selection"""
if cmd_alias not in cmd_alias_definitions.keys():
msg = f"Unknown RPC command alias: '{cmd_alias}'"
log.error(msg)
raise KeyError(msg)
+
with cfg_cards:
if not overwrite and card_id in cfg_cards.keys():
msg = f"Card already registered: '{card_id}'. Abort. (use overwrite=True to overrule)"
log.error(msg)
raise KeyError(msg)
+
cfg_cards[card_id] = {'alias': cmd_alias}
- if args is not None:
+
+ # For play_from_reader, expect a single dict of args
+ if cmd_alias == 'play_from_reader':
+ # Use either kwargs or args if it's a dict
+ if kwargs is not None:
+ cfg_cards[card_id]['args'] = kwargs
+ elif isinstance(args, dict):
+ cfg_cards[card_id]['args'] = args
+ else:
+ log.error(f"play_from_reader requires dict arguments, got: {type(args)}")
+ raise ValueError("play_from_reader requires dict arguments")
+ # For other commands, maintain list args support
+ elif args is not None:
cfg_cards[card_id]['args'] = args
- if kwargs is not None:
- cfg_cards[card_id]['kwargs'] = args
+
if ignore_same_id_delay is not None:
cfg_cards[card_id]['ignore_same_id_delay'] = ignore_same_id_delay
if ignore_card_removal_action is not None:
cfg_cards[card_id]['ignore_card_removal_action'] = ignore_card_removal_action
+
if auto_save:
cfg_cards.save()
+
publishing.get_publisher().send(f'{plugs.loaded_as(__name__)}.database.has_changed', time.ctime())
diff --git a/src/jukebox/components/rpc_command_alias.py b/src/jukebox/components/rpc_command_alias.py
index 5a7820733..48b76ea7c 100644
--- a/src/jukebox/components/rpc_command_alias.py
+++ b/src/jukebox/components/rpc_command_alias.py
@@ -12,12 +12,12 @@
# --------------------------------------------------------------
cmd_alias_definitions = {
# Player
- 'play_card': {
- 'title': 'Play music folder triggered by card swipe',
+ 'play_from_reader': {
+ 'title': 'Play content triggered by card swipe, supports second swipe',
'note': "This function you'll want to use most often",
'package': 'player',
'plugin': 'ctrl',
- 'method': 'play_card'},
+ 'method': 'play_from_reader'},
'play_album': {
'title': 'Play Album triggered by card swipe',
'note': "This function plays the content of a given album",
diff --git a/src/webapp/public/locales/de/translation.json b/src/webapp/public/locales/de/translation.json
index 1c6729415..42046e7b6 100644
--- a/src/webapp/public/locales/de/translation.json
+++ b/src/webapp/public/locales/de/translation.json
@@ -27,6 +27,7 @@
"next_song": "Nächster Song",
"pause": "Pause",
"play": "Abspielen",
+ "play_content": "Inhalte abspielen",
"prev_song": "Vorheriger Song",
"shuffle": "Zufallswiedergabe",
"repeat": "Wiedergabe wiederholen",
@@ -40,7 +41,7 @@
"label": "Aktionen",
"placeholder": "Wähle eine Aktion aus",
"actions": {
- "play_music": "Musik abspielen",
+ "play_content": "Inhalte abspielen",
"audio": "Audio & Lautstärke",
"host": "System",
"timers": "Timer",
@@ -53,15 +54,15 @@
"label-full": "Gesamte Addresse (z.B. 192.168.1.53)",
"label-short": "Letzter Quadrant (z.B. 53)"
},
- "play-music": {
+ "play-content": {
"commands": {
- "play_album": "Ausgewähltes Album",
- "play_folder": "Ausgewählter Ordner",
- "play_single": "Ausgewählter Song"
+ "album": "Ausgewähltes Album",
+ "folder": "Ausgewählter Ordner",
+ "single": "Ausgewählter Song"
},
- "button-label": "Musik auswählen",
- "no-music-selected": "Es ist keine Musik ausgewählt.",
- "loading-song-error": "Während des Ladens des Songs ist ein Fehler aufgetreten."
+ "button-label": "Inhalt auswählen",
+ "no-music-selected": "Es sind keine Inhalte ausgewählt.",
+ "loading-song-error": "Während des Ladens des Inhalts ist ein Fehler aufgetreten."
},
"audio": {
"repeat": {
diff --git a/src/webapp/public/locales/en/translation.json b/src/webapp/public/locales/en/translation.json
index 20a28bdac..358b6b1ec 100644
--- a/src/webapp/public/locales/en/translation.json
+++ b/src/webapp/public/locales/en/translation.json
@@ -27,6 +27,7 @@
"next_song": "Next song",
"pause": "Pause",
"play": "Play",
+ "play_content": "Play content",
"prev_song": "Previous song",
"shuffle": "Shuffle",
"repeat": "Repeat",
@@ -40,7 +41,7 @@
"label": "Actions",
"placeholder": "Select an action",
"actions": {
- "play_music": "Play music",
+ "play_content": "Play content",
"audio": "Audio & Volume",
"host": "System",
"timers": "Timers",
@@ -53,15 +54,15 @@
"label-full": "Full address (e.g. 192.168.1.53)",
"label-short": "Last quadrant (e.g. 53)"
},
- "play-music": {
+ "play-content": {
"commands": {
- "play_album": "Selected album",
- "play_folder": "Selected folder",
- "play_single": "Selected song"
+ "album": "Selected album",
+ "folder": "Selected folder",
+ "single": "Selected song"
},
- "button-label": "Select music",
- "no-music-selected": "No music selected",
- "loading-song-error": "An error occurred while loading song."
+ "button-label": "Select content",
+ "no-music-selected": "No content selected",
+ "loading-song-error": "An error occurred while loading the content."
},
"audio": {
"repeat": {
diff --git a/src/webapp/src/commands/index.js b/src/webapp/src/commands/index.js
index ffbc01f03..5a419e1f8 100644
--- a/src/webapp/src/commands/index.js
+++ b/src/webapp/src/commands/index.js
@@ -65,6 +65,24 @@ const commands = {
method: 'play_content',
argKeys: ['content', 'content_type', 'recursive']
},
+ // play_single: {
+ // _package: 'player',
+ // plugin: 'ctrl',
+ // method: 'play_single',
+ // argKeys: ['song_url']
+ // },
+ // play_folder: {
+ // _package: 'player',
+ // plugin: 'ctrl',
+ // method: 'play_folder',
+ // argKeys: ['folder']
+ // },
+ // play_album: {
+ // _package: 'player',
+ // plugin: 'ctrl',
+ // method: 'play_album',
+ // argKeys: ['albumartist', 'album']
+ // },
pause: {
_package: 'player',
plugin: 'ctrl',
diff --git a/src/webapp/src/components/Cards/controls/actions-controls.js b/src/webapp/src/components/Cards/controls/actions-controls.js
index 90afb56ad..b1577f25b 100644
--- a/src/webapp/src/components/Cards/controls/actions-controls.js
+++ b/src/webapp/src/components/Cards/controls/actions-controls.js
@@ -10,8 +10,8 @@ import {
import CardsDeleteDialog from '../dialogs/delete';
import request from '../../../utils/request';
import {
+ cleanObject,
getActionAndCommand,
- getArgsValues
} from '../utils';
const ActionsControls = ({
@@ -24,14 +24,14 @@ const ActionsControls = ({
const [deleteDialogOpen, setDeleteDialogOpen] = useState(false);
const handleRegisterCard = async () => {
- const args = getArgsValues(actionData);
+ const { args } = actionData.command || {};
const { command: cmd_alias } = getActionAndCommand(actionData);
const kwargs = {
card_id: cardId.toString(),
- cmd_alias,
+ cmd_alias: cmd_alias === 'play_content' ? 'play_from_reader' : cmd_alias,
overwrite: true,
- ...(args.length && { args }),
+ args: cleanObject(args),
};
const { error } = await request('registerCard', kwargs);
diff --git a/src/webapp/src/components/Cards/controls/actions/play-music/index.js b/src/webapp/src/components/Cards/controls/actions/play-content/index.js
similarity index 69%
rename from src/webapp/src/components/Cards/controls/actions/play-music/index.js
rename to src/webapp/src/components/Cards/controls/actions/play-content/index.js
index 2e8d7332c..0e8fe8eae 100644
--- a/src/webapp/src/components/Cards/controls/actions/play-music/index.js
+++ b/src/webapp/src/components/Cards/controls/actions/play-content/index.js
@@ -20,17 +20,18 @@ import SelectedAlbum from './selected-album';
import SelectedFolder from './selected-folder';
import SelectedSingle from './selected-single';
-const SelectPlayMusic = ({
+const SelectPlayContent = ({
actionData,
cardId,
}) => {
const { t } = useTranslation();
const navigate = useNavigate();
- const { command } = getActionAndCommand(actionData);
+ const { content_type } = actionData.command.args || {};
+
const values = getArgsValues(actionData);
- const selectMusic = () => {
+ const selectContent = () => {
const searchParams = createSearchParams({
isSelecting: true,
cardId
@@ -44,30 +45,30 @@ const SelectPlayMusic = ({
return (
- {command &&
+ {content_type &&
- {t(`cards.controls.actions.play-music.commands.${command}`)}
+ {t(`cards.controls.actions.play-content.commands.${content_type}`)}
}
- {command === 'play_album' && }
- {command === 'play_folder' && }
- {command === 'play_single' && }
+ {content_type === 'album' && }
+ {content_type === 'folder' && }
+ {content_type === 'single' && }
}
>
- {t('cards.controls.actions.play-music.button-label')}
+ {t('cards.controls.actions.play-content.button-label')}
);
};
-export default SelectPlayMusic;
+export default SelectPlayContent;
diff --git a/src/webapp/src/components/Cards/controls/actions/play-music/no-music-selected.js b/src/webapp/src/components/Cards/controls/actions/play-content/no-music-selected.js
similarity index 80%
rename from src/webapp/src/components/Cards/controls/actions/play-music/no-music-selected.js
rename to src/webapp/src/components/Cards/controls/actions/play-content/no-music-selected.js
index 90821d151..ce9478f14 100644
--- a/src/webapp/src/components/Cards/controls/actions/play-music/no-music-selected.js
+++ b/src/webapp/src/components/Cards/controls/actions/play-content/no-music-selected.js
@@ -8,7 +8,7 @@ const NoMusicSelected = () => {
return (
- {t('cards.controls.actions.play-music.no-music-selected')}
+ {t('cards.controls.actions.play-content.no-music-selected')}
);
}
diff --git a/src/webapp/src/components/Cards/controls/actions/play-music/selected-album.js b/src/webapp/src/components/Cards/controls/actions/play-content/selected-album.js
similarity index 88%
rename from src/webapp/src/components/Cards/controls/actions/play-music/selected-album.js
rename to src/webapp/src/components/Cards/controls/actions/play-content/selected-album.js
index b0f5d2bc9..f7f323245 100644
--- a/src/webapp/src/components/Cards/controls/actions/play-music/selected-album.js
+++ b/src/webapp/src/components/Cards/controls/actions/play-content/selected-album.js
@@ -4,7 +4,7 @@ import { List } from '@mui/material';
import AlbumListItem from '../../../../Library/lists/albums/album-list/album-list-item'
import NoMusicSelected from './no-music-selected';
-const SelectedAlbum = ({ values: [albumartist, album] }) => {
+const SelectedAlbum = ({ values: [{ albumartist, album }] }) => {
if (albumartist && album) {
return (
diff --git a/src/webapp/src/components/Cards/controls/actions/play-music/selected-folder.js b/src/webapp/src/components/Cards/controls/actions/play-content/selected-folder.js
similarity index 100%
rename from src/webapp/src/components/Cards/controls/actions/play-music/selected-folder.js
rename to src/webapp/src/components/Cards/controls/actions/play-content/selected-folder.js
diff --git a/src/webapp/src/components/Cards/controls/actions/play-music/selected-single.js b/src/webapp/src/components/Cards/controls/actions/play-content/selected-single.js
similarity index 95%
rename from src/webapp/src/components/Cards/controls/actions/play-music/selected-single.js
rename to src/webapp/src/components/Cards/controls/actions/play-content/selected-single.js
index a7280c11e..a369e93b2 100644
--- a/src/webapp/src/components/Cards/controls/actions/play-music/selected-single.js
+++ b/src/webapp/src/components/Cards/controls/actions/play-content/selected-single.js
@@ -32,7 +32,7 @@ const SelectecSingle = ({ values: [song_url] }) => {
if (error) {
return (
- {t('cards.controls.actions.play-music.loading-song-error')}
+ {t('cards.controls.actions.play-content.loading-song-error')}
);
}
diff --git a/src/webapp/src/components/Cards/controls/controls-selector.js b/src/webapp/src/components/Cards/controls/controls-selector.js
index eea2d5c67..1295bcfe7 100644
--- a/src/webapp/src/components/Cards/controls/controls-selector.js
+++ b/src/webapp/src/components/Cards/controls/controls-selector.js
@@ -7,7 +7,7 @@ import {
} from '@mui/material';
import SelectCommandAliases from './select-command-aliases';
-import SelectPlayMusic from './actions/play-music';
+import SelectPlayContent from './actions/play-content';
import SelectTimers from './actions/timers';
import SelectAudio from './actions/audio';
import { buildActionData } from '../utils';
@@ -61,8 +61,8 @@ const ControlsSelector = ({
/>
}
- {actionData.action === 'play_music' &&
-
diff --git a/src/webapp/src/components/Cards/edit.js b/src/webapp/src/components/Cards/edit.js
index 6bcf11012..7ca5f803c 100644
--- a/src/webapp/src/components/Cards/edit.js
+++ b/src/webapp/src/components/Cards/edit.js
@@ -22,9 +22,11 @@ const CardsEdit = () => {
if (result && result[cardId]) {
const {
action: { args },
- from_alias: command
+ from_alias,
} = result[cardId];
+ const command = from_alias === 'play_from_reader' ? 'play_content' : from_alias;
+
const action = findActionByCommand(command);
const actionData = buildActionData(action, command, args);
diff --git a/src/webapp/src/components/Cards/list.js b/src/webapp/src/components/Cards/list.js
index 4e8211a27..020389b7f 100644
--- a/src/webapp/src/components/Cards/list.js
+++ b/src/webapp/src/components/Cards/list.js
@@ -13,6 +13,7 @@ import {
} from '@mui/material';
import BookmarkIcon from '@mui/icons-material/Bookmark';
+import { printObject } from '../../utils/utils';
const CardsList = ({ cardsList }) => {
const { t } = useTranslation();
@@ -28,10 +29,15 @@ const CardsList = ({ cardsList }) => {
return
});
- const description = cardsList[cardId].from_alias
+ const command = cardsList[cardId].from_alias === 'play_from_reader' ? 'play_content' : cardsList[cardId].from_alias;
+
+ const description = command
? reject(
isNil,
- [cardsList[cardId].from_alias, cardsList[cardId].action.args]
+ [
+ t(`cards.controls.command-selector.commands.${command}`),
+ printObject(cardsList[cardId].action.args)
+ ]
).join(', ')
: cardsList[cardId].func
diff --git a/src/webapp/src/components/Cards/register.js b/src/webapp/src/components/Cards/register.js
index c4d3d32f0..848acb9db 100644
--- a/src/webapp/src/components/Cards/register.js
+++ b/src/webapp/src/components/Cards/register.js
@@ -17,6 +17,7 @@ const CardsRegister = () => {
const [cardId, setCardId] = useState(undefined);
const [actionData, setActionData] = useState(registerCard?.actionData || {});
+ const [args, setArgs] = useState(registerCard?.args || {});
useEffect(() => {
setState(state => (omit(['rfid.card_id'], state)));
diff --git a/src/webapp/src/components/Cards/utils.js b/src/webapp/src/components/Cards/utils.js
index 17eaede6f..0202b783e 100644
--- a/src/webapp/src/components/Cards/utils.js
+++ b/src/webapp/src/components/Cards/utils.js
@@ -1,6 +1,8 @@
import {
isEmpty,
+ isNil,
has,
+ reject,
} from 'ramda';
import commands from '../../commands';
@@ -67,8 +69,11 @@ const getArgsValues = (actionData) => {
);
};
+const cleanObject = reject(isNil);
+
export {
buildActionData,
+ cleanObject,
findActionByCommand,
getActionAndCommand,
getArgsValues,
diff --git a/src/webapp/src/components/Library/lists/albums/song-list/index.js b/src/webapp/src/components/Library/lists/albums/song-list/index.js
index 006ab791b..9fb87a841 100644
--- a/src/webapp/src/components/Library/lists/albums/song-list/index.js
+++ b/src/webapp/src/components/Library/lists/albums/song-list/index.js
@@ -18,7 +18,7 @@ import SongListItem from './song-list-item';
const SongList = ({
isSelecting,
- registerMusicToCard,
+ registerContentToCard,
}) => {
const { t } = useTranslation();
const { artist, album } = useParams();
@@ -59,7 +59,7 @@ const SongList = ({
albumartist={decodeURIComponent(artist)}
disabled={songs.length === 0}
isSelecting={isSelecting}
- registerMusicToCard={registerMusicToCard}
+ registerContentToCard={registerContentToCard}
/>
)}
diff --git a/src/webapp/src/components/Library/lists/albums/song-list/song-list-controls.js b/src/webapp/src/components/Library/lists/albums/song-list/song-list-controls.js
index 18b240382..a3db788e0 100644
--- a/src/webapp/src/components/Library/lists/albums/song-list/song-list-controls.js
+++ b/src/webapp/src/components/Library/lists/albums/song-list/song-list-controls.js
@@ -14,18 +14,17 @@ const SongListControls = ({
albumartist,
album,
disabled,
- registerMusicToCard,
+ registerContentToCard,
isSelecting
}) => {
const { t } = useTranslation();
- const command = 'play_album';
const playAlbum = () => (
- request('play_content', { content: { "artist": albumartist, album }, content_type: 'album' })
+ request('play_content', { content: { albumartist, album }, content_type: 'album' })
);
const registerAlbumToCard = () => (
- registerMusicToCard(command, { albumartist, album })
+ registerContentToCard('play_content', { content: { albumartist, album }, content_type: 'album' })
);
return (
diff --git a/src/webapp/src/components/Library/lists/albums/song-list/song-list-item.js b/src/webapp/src/components/Library/lists/albums/song-list/song-list-item.js
index 4b76a984c..e9edea8eb 100644
--- a/src/webapp/src/components/Library/lists/albums/song-list/song-list-item.js
+++ b/src/webapp/src/components/Library/lists/albums/song-list/song-list-item.js
@@ -12,12 +12,11 @@ import request from '../../../../../utils/request'
const SongListItem = ({
isSelecting,
- registerMusicToCard,
+ registerContentToCard,
song,
}) => {
const { t } = useTranslation();
- const command = 'play_single';
const {
artist,
duration,
@@ -29,15 +28,15 @@ const SongListItem = ({
request('play_content', { content: file, content_type: 'single' })
}
- const registerSongToCard = () => (
- registerMusicToCard(command, { song_url: file })
+ const registerSingleToCard = () => (
+ registerContentToCard('play_content', { content: file, content_type: 'single' })
);
return (
(isSelecting ? registerSongToCard() : playSingle())}
+ onClick={() => (isSelecting ? registerSingleToCard() : playSingle())}
>
{
const { t } = useTranslation();
const { type, name, relpath } = folder;
@@ -34,8 +34,8 @@ const FolderListItem = ({
const registerItemToCard = () => {
switch(type) {
- case 'directory': return registerMusicToCard('play_folder', { folder: relpath, recursive: true });
- case 'file': return registerMusicToCard('play_single', { song_url: relpath });
+ case 'directory': return registerContentToCard('play_content', { content: relpath, content_type: 'folder', recursive: true });
+ case 'file': return registerContentToCard('play_content', { content: relpath, content_type: 'single' });
// TODO: Add missing Podcast
// TODO: Add missing Stream
default: return;
diff --git a/src/webapp/src/components/Library/lists/folders/folder-list.js b/src/webapp/src/components/Library/lists/folders/folder-list.js
index 3222e4234..2c2f90086 100644
--- a/src/webapp/src/components/Library/lists/folders/folder-list.js
+++ b/src/webapp/src/components/Library/lists/folders/folder-list.js
@@ -17,7 +17,7 @@ const FolderList = ({
dir,
folders,
isSelecting,
- registerMusicToCard,
+ registerContentToCard,
}) => {
const { t } = useTranslation();
@@ -47,7 +47,7 @@ const FolderList = ({
key={key}
folder={folder}
isSelecting={isSelecting}
- registerMusicToCard={registerMusicToCard}
+ registerContentToCard={registerContentToCard}
/>
)}
diff --git a/src/webapp/src/components/Library/lists/folders/index.js b/src/webapp/src/components/Library/lists/folders/index.js
index fa0532589..ce279cebd 100644
--- a/src/webapp/src/components/Library/lists/folders/index.js
+++ b/src/webapp/src/components/Library/lists/folders/index.js
@@ -15,7 +15,7 @@ import { ROOT_DIR } from '../../../../config';
const Folders = ({
musicFilter,
isSelecting,
- registerMusicToCard,
+ registerContentToCard,
}) => {
const { t } = useTranslation();
const { dir = ROOT_DIR } = useParams();
@@ -60,7 +60,7 @@ const Folders = ({
dir={dir}
folders={filteredFolders}
isSelecting={isSelecting}
- registerMusicToCard={registerMusicToCard}
+ registerContentToCard={registerContentToCard}
/>
);
};
diff --git a/src/webapp/src/components/Library/lists/index.js b/src/webapp/src/components/Library/lists/index.js
index e2b7a2d46..7390eb451 100644
--- a/src/webapp/src/components/Library/lists/index.js
+++ b/src/webapp/src/components/Library/lists/index.js
@@ -32,8 +32,8 @@ const LibraryLists = () => {
setMusicFilter(event.target.value);
};
- const registerMusicToCard = (command, args) => {
- const actionData = buildActionData('play_music', command, args);
+ const registerContentToCard = (command, args) => {
+ const actionData = buildActionData('play_content', command, args);
const state = {
registerCard: {
actionData,
@@ -71,7 +71,7 @@ const LibraryLists = () => {
element={
}
exact
@@ -86,7 +86,7 @@ const LibraryLists = () => {
}
/>
diff --git a/src/webapp/src/config.js b/src/webapp/src/config.js
index 46a6ec1df..9d5f5d372 100644
--- a/src/webapp/src/config.js
+++ b/src/webapp/src/config.js
@@ -28,11 +28,9 @@ const ROOT_DIR = './';
const JUKEBOX_ACTIONS_MAP = {
// Command Aliases
// Player
- play_music: {
+ play_content: {
commands: {
- play_album: {},
- play_folder: {},
- play_single: {},
+ play_content: {},
}
},
diff --git a/src/webapp/src/utils/utils.js b/src/webapp/src/utils/utils.js
index 170a8994f..5b217e9fb 100644
--- a/src/webapp/src/utils/utils.js
+++ b/src/webapp/src/utils/utils.js
@@ -24,10 +24,21 @@ const flatByAlbum = (albumList, { albumartist, album }) => {
return [...albumList, ...list];
};
+const printObject = (obj) => {
+ return Object.entries(obj)
+ .map(([key, value]) => {
+ if (value && typeof value === 'object') {
+ return `${key}: ${printObject(value)}`;
+ }
+ return `${key}: ${value}`;
+ })
+ .join(', ');
+};
export {
flatByAlbum,
pluginIsLoaded,
+ printObject,
progressToTime,
timeToProgress,
toHHMMSS,
From 101d8161d516fd98e500483715e8bbaabb4bf359 Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sun, 17 Nov 2024 13:03:39 +0100
Subject: [PATCH 08/16] remove comments
---
src/webapp/src/commands/index.js | 18 ------------------
1 file changed, 18 deletions(-)
diff --git a/src/webapp/src/commands/index.js b/src/webapp/src/commands/index.js
index 5a419e1f8..ffbc01f03 100644
--- a/src/webapp/src/commands/index.js
+++ b/src/webapp/src/commands/index.js
@@ -65,24 +65,6 @@ const commands = {
method: 'play_content',
argKeys: ['content', 'content_type', 'recursive']
},
- // play_single: {
- // _package: 'player',
- // plugin: 'ctrl',
- // method: 'play_single',
- // argKeys: ['song_url']
- // },
- // play_folder: {
- // _package: 'player',
- // plugin: 'ctrl',
- // method: 'play_folder',
- // argKeys: ['folder']
- // },
- // play_album: {
- // _package: 'player',
- // plugin: 'ctrl',
- // method: 'play_album',
- // argKeys: ['albumartist', 'album']
- // },
pause: {
_package: 'player',
plugin: 'ctrl',
From 0f5eb48e57cd03113387b0542338020c42622e3f Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Wed, 20 Nov 2024 22:25:31 +0100
Subject: [PATCH 09/16] Refactor RPC tool
---
src/jukebox/run_rpc_tool.py | 825 ++++++++++++++++++++----------------
1 file changed, 460 insertions(+), 365 deletions(-)
diff --git a/src/jukebox/run_rpc_tool.py b/src/jukebox/run_rpc_tool.py
index 47f861af5..40a3391b1 100644
--- a/src/jukebox/run_rpc_tool.py
+++ b/src/jukebox/run_rpc_tool.py
@@ -3,397 +3,487 @@
Command Line Interface to the Jukebox RPC Server
A command line tool for sending RPC commands to the running jukebox app.
-This uses the same interface as the WebUI. Can be used for additional control
-or for debugging.
-
-The tool features auto-completion and command history.
-
-The list of available commands is fetched from the running Jukebox service.
-
-.. todo:
- - kwargs support
-
+Features auto-completion, command history, and RPC command execution.
+Supports JSON arguments for complex data structures.
"""
import argparse
-import zmq
+import json
+from dataclasses import dataclass
+from typing import List, Dict, Any, Tuple
import curses
import curses.ascii
+import zmq
import jukebox.rpc.client as rpc
-# Developers note: Scripting at it's dirty end :-)
+@dataclass
+class CliState:
+ """Encapsulates CLI state and configuration"""
+ url: str
+ client: rpc.RpcClient
+ rpc_help: Dict[str, Dict[str, str]] = None
+ candidates: List[str] = None
+ history: List[str] = None
+ prompt: str = '> '
+
+ def __post_init__(self):
+ self.rpc_help = {}
+ self.candidates = []
+ self.history = ['']
+
+
+class CommandParser:
+ """Handles parsing and execution of RPC commands with JSON and quoted string support"""
+
+ @staticmethod
+ def parse_command(cmd: str) -> Tuple[List[str], List[Any], Dict[str, Any]]:
+ """
+ Parse command string into command parts, positional args, and keyword args
+ Returns: (command_parts, args, kwargs)
+ """
+ # Split while preserving quotes and JSON structures
+ parts = CommandParser._split_preserving_json(cmd.strip())
+ if not parts:
+ return [], [], {}
+
+ # Split cmd on '.' into package.plugin.method
+ command_parts = [v for v in parts[0].split('.') if len(v) > 0]
+
+ # Process remaining parts into args and kwargs
+ args = []
+ kwargs = {}
+ seen_keys = set() # Track seen keyword argument names
+
+ for part in parts[1:]:
+ # Check if part is a kwarg (contains '=')
+ if '=' in part:
+ key, value = part.split('=', 1)
+ key = key.strip()
+ value = value.strip()
+
+ # Check for duplicate keyword arguments
+ if key in seen_keys:
+ raise ValueError(f"Duplicate keyword argument: {key}")
+ seen_keys.add(key)
+
+ # Handle the value based on its format
+ kwargs[key] = CommandParser._parse_value(value)
+ else:
+ # Handle as positional argument
+ args.append(CommandParser._parse_value(part))
+
+ return command_parts, args, kwargs
+
+ @staticmethod
+ def _parse_value(value: str) -> Any:
+ """Parse a value string into appropriate type"""
+ # Strip quotes if present (only if matching quotes at start and end)
+ if (value.startswith('"') and value.endswith('"')) or \
+ (value.startswith("'") and value.endswith("'")):
+ value = value[1:-1]
+
+ # Try to parse as JSON if it looks like JSON
+ if value.startswith('{') or value.startswith('['):
+ try:
+ return json.loads(value)
+ except json.JSONDecodeError:
+ # If JSON parsing fails, continue with other parsing attempts
+ pass
+
+ # Try to convert to number if appropriate
+ return CommandParser.convert_to_number(value)
+
+ @staticmethod
+ def _split_preserving_json(cmd: str) -> List[str]: # noqa: C901
+ """Split command string while preserving quoted strings and JSON structures"""
+ parts = []
+ current = []
+ brace_count = 0
+ bracket_count = 0
+ in_single_quotes = False
+ in_double_quotes = False
+ escape = False
+
+ for char in cmd:
+ if escape:
+ current.append(char)
+ escape = False
+ continue
+
+ if char == '\\':
+ escape = True
+ current.append(char)
+ continue
+
+ if char == '"' and not in_single_quotes:
+ in_double_quotes = not in_double_quotes
+ current.append(char)
+ continue
+
+ if char == "'" and not in_double_quotes:
+ in_single_quotes = not in_single_quotes
+ current.append(char)
+ continue
+
+ if not in_single_quotes and not in_double_quotes:
+ if char == '{':
+ brace_count += 1
+ elif char == '}':
+ brace_count -= 1
+ elif char == '[':
+ bracket_count += 1
+ elif char == ']':
+ bracket_count -= 1
+ elif char.isspace() and brace_count == 0 and bracket_count == 0:
+ if current:
+ parts.append(''.join(current))
+ current = []
+ continue
+
+ current.append(char)
+
+ if current:
+ parts.append(''.join(current))
+
+ # Validate quote matching
+ if in_single_quotes:
+ raise ValueError("Unmatched single quote")
+ if in_double_quotes:
+ raise ValueError("Unmatched double quote")
+
+ return parts
+
+ @staticmethod
+ def convert_to_number(value: str) -> Any:
+ """Convert string to number if possible"""
+ # Try integer
+ try:
+ return int(value)
+ except ValueError:
+ pass
+
+ # Try float
+ try:
+ return float(value)
+ except ValueError:
+ pass
+
+ # Try hex
+ if value.isalnum() and value.startswith('0x'):
+ try:
+ return int(value, base=16)
+ except ValueError:
+ pass
+
+ return value
-# Careful: curses and default outputs don't mix!
-# In case you'll get an error, most likely your terminal may become funny
-# Best bet: Just don't configure any logger at all!
-# import logging
-# import misc.loggingext
-# logger = misc.loggingext.configure_default(logging.ERROR)
+class JukeboxCli:
+ """Main CLI class handling user interaction and command execution"""
-url: str
-client: rpc.RpcClient
-rpc_help = {}
-candidates = []
-history = ['']
-prompt = '> '
+ def __init__(self, url: str):
+ self.state = CliState(url=url, client=rpc.RpcClient(url))
+ self.command_parser = CommandParser()
+ def update_help(self, scr) -> None:
+ """Update available RPC commands from server"""
+ try:
+ rpc_help_tmp = self.state.client.enque('misc', 'rpc_cmd_help')
+ self.state.rpc_help = {k: rpc_help_tmp[k] for k in sorted(rpc_help_tmp.keys())}
+ except Exception:
+ self._show_connection_error(scr)
+ return
+
+ # Add CLI specific commands
+ self._add_cli_commands()
+ self.state.candidates = list(self.state.rpc_help.keys())
+
+ def _add_cli_commands(self) -> None:
+ """Add CLI-specific commands to help"""
+ cli_commands = {
+ "help": {'description': "Print RPC Server command list (all commands that start with ...)",
+ 'signature': "(cmd_starts_with='')"},
+ 'usage': {'description': "Usage help and key bindings", 'signature': "()"},
+ 'exit': {'description': "Exit RPC Client", 'signature': "()"}
+ }
+ self.state.rpc_help.update(cli_commands)
+
+ def execute_command(self, scr, cmd: str) -> None:
+ """Execute a command and display results"""
+ if not cmd.strip():
+ return
+
+ if cmd == 'help':
+ self._show_help(scr)
+ return
+ elif cmd == 'usage':
+ self._show_usage(scr)
+ return
+ elif cmd == 'exit':
+ return
+
+ command_parts, args, kwargs = self.command_parser.parse_command(cmd)
+
+ if not (2 <= len(command_parts) <= 3):
+ scr.addstr(":: Error = Ill-formatted command\n")
+ return
-def add_cli():
- global rpc_help
- rpc_help["help"] = {'description': "Print RPC Server command list (all commands that start with ...)",
- 'signature': "(cmd_starts_with='')"}
- rpc_help['usage'] = {'description': "Usage help and key bindings", 'signature': "()"}
- rpc_help['exit'] = {'description': "Exit RPC Client", 'signature': "()"}
+ method = command_parts[2] if len(command_parts) == 3 else None
+ try:
+ response = self.state.client.enque(
+ command_parts[0],
+ command_parts[1],
+ method,
+ args=args,
+ kwargs=kwargs
+ )
+ scr.addstr(f"\n:: Response =\n{response}\n\n")
+ except zmq.error.Again:
+ self._show_connection_error(scr)
+ except Exception as e:
+ scr.addstr(f":: Exception response =\n{e}\n")
-def get_help(scr):
- global rpc_help
- global candidates
- rpc_help = {}
- try:
- rpc_help_tmp = client.enque('misc', 'rpc_cmd_help')
- except Exception:
+ def run(self, scr) -> None:
+ """Main CLI loop"""
+ self._setup_screen(scr)
+ self._show_welcome(scr)
+ self.update_help(scr)
+ self._show_usage(scr)
+
+ cmd = ''
+ while cmd != 'exit':
+ cmd = self._get_input(scr)
+ scr.addstr("\n")
+ self.execute_command(scr, cmd)
+
+ def _setup_screen(self, scr) -> None:
+ """Configure screen settings"""
+ scr.idlok(True)
+ scr.scrollok(True)
+ curses.noecho()
+
+ def _show_connection_error(self, scr) -> None:
+ """Display connection error message"""
scr.addstr("\n\n" + '-' * 70 + "\n")
scr.addstr("Could not reach RPC Server. Jukebox running? Correct Port?\n")
scr.addstr('-' * 70 + "\n\n")
scr.refresh()
- else:
- # Sort the commands (Python 3.7 has ordered entries in dicts!)
- rpc_help = {k: rpc_help_tmp[k] for k in sorted(rpc_help_tmp.keys())}
- add_cli()
- candidates = rpc_help.keys()
-
-
-def format_help(scr, topic):
- global rpc_help
- # Always update help, in case Jukebox App has been restarted in between
- scr.erase()
- get_help(scr)
- max_y, max_x = scr.getmaxyx()
- scr.addstr("Available commands:\n\n")
- for key, value in rpc_help.items():
- sign: str = value['signature']
- sign = sign[sign.find('('):]
- func = f"{key}{sign}"
- # print(f"{func:50}: {value['description']}")
- if key.startswith(topic):
- scr.addstr(f"{func:50}: {value['description']}\n")
- [y, x] = scr.getyx()
- if y == max_y - 1:
- scr.addstr("--HIT A KEY TO CONTINUE--")
- scr.getch()
- scr.erase()
- scr.addstr("\n")
- scr.refresh()
-
-
-def format_welcome(scr):
- scr.addstr("\n\n" + '-' * 70 + "\n")
- scr.addstr("RPC Tool\n")
- scr.addstr('-' * 70 + "\n")
- scr.addstr(f"Connection url: '{client.address}'\n")
- try:
- jukebox_version = client.enque('misc', 'get_version')
- except Exception:
- jukebox_version = "unknown"
- scr.addstr(f"Jukebox version: {jukebox_version}\n")
- scr.addstr(f"Pyzmq version: {zmq.pyzmq_version()}; ZMQ version: {zmq.zmq_version()}; has draft API: {zmq.DRAFT_API}\n")
- scr.addstr('-' * 70 + "\n")
-
-
-def format_usage(scr):
- scr.addstr("\n\nUsage:\n")
- scr.addstr(" > cmd [arg1] [arg2] [arg3]\n")
- scr.addstr("e.g.\n")
- scr.addstr(' > volume.ctrl.set_volume 50\n')
-
- # General content playback examples (for webapp/CLI)
- scr.addstr("\nPlaying content (using play_content):\n")
- scr.addstr(' > player.ctrl.play_content \'{"artist":"Pink Floyd","album":"The Wall"}\' album\n')
- scr.addstr(' > player.ctrl.play_content "/music/classical" folder true\n')
- scr.addstr(' > player.ctrl.play_content "/music/favorites/track.mp3" single\n')
-
- # Update the reader examples section
- scr.addstr("\nPlaying content from physical readers (using play_from_reader):\n")
- scr.addstr(' > player.ctrl.play_from_reader \'{"artist":"Pink Floyd","album":"The Wall"}\' album false toggle\n')
- scr.addstr(' > player.ctrl.play_from_reader "/music/classical" folder true replay\n')
- scr.addstr(' > player.ctrl.play_from_reader "/music/stories" folder false none\n')
-
- scr.addstr("\n")
- scr.addstr("Quoting:\n")
- scr.addstr(" - Use single quotes (\') for JSON content\n")
- scr.addstr(' - Use double quotes (") for simple string arguments containing spaces\n')
- scr.addstr(' - Escape quotes within quoted strings with \\\n')
- scr.addstr("\n")
- scr.addstr("Content Types:\n")
- scr.addstr(" - album: requires JSON with artist and album\n")
- scr.addstr(" - single: direct path to audio file\n")
- scr.addstr(" - folder: path to folder (optional recursive flag)\n")
- scr.addstr("\n")
- scr.addstr("Second Swipe Actions (for play_card):\n")
- scr.addstr(" - none: disable second swipe\n")
- scr.addstr(" - toggle: toggle play/pause\n")
- scr.addstr(" - play: start playing\n")
- scr.addstr(" - skip: next track\n")
- scr.addstr(" - rewind: restart playlist\n")
- scr.addstr(" - replay: restart folder\n")
- scr.addstr(" - replay_if_stopped: restart if stopped\n")
- scr.addstr("\n")
- scr.addstr("Use for auto-completion of commands!\n")
- scr.addstr("Use / for command history!\n")
- scr.addstr("\n")
- scr.addstr("Type help , to get a list of all commands'\n")
- scr.addstr("Type usage , to get this usage help'\n")
- scr.addstr("\n")
- scr.addstr("After Jukebox app restart, call help once to update command list from jukebox app\n")
- scr.addstr("\n")
- scr.addstr("To exit, press Ctrl-D or type 'exit'\n")
- scr.addstr("\n")
- scr.refresh()
-
-
-def get_common_beginning(strings):
- """
- Return the strings that are common to the beginning of each string in the strings list.
- """
- result = []
- limit = min([len(s) for s in strings])
- for i in range(limit):
- chs = set([s[i] for s in strings])
- if len(chs) == 1:
- result.append(chs.pop())
- else:
- break
- return ''.join(result)
-
-
-def autocomplete(msg):
- # logger.debug(f"Autocomplete {msg}")
- # Get all stings that match the beginning
- # candidates = ["ap1", 'ap2', 'appbbb3', 'appbbb4', 'appbbb5', 'appbbb6', 'exit']
- matches = [s for s in candidates if s.startswith(msg)]
- if len(matches) == 0:
- # Matches is empty: nothing found
- return msg, matches
- common = get_common_beginning(matches)
- return common, matches
-
-
-def is_printable(ch: int):
- return 32 <= ch <= 127
-
-
-def reprompt(scr, msg, y, x):
- scr.move(y, 0)
- scr.clrtoeol()
- scr.addstr(prompt)
- scr.addstr(msg)
- scr.move(y, x)
-
-
-def get_input(scr): # noqa: C901
- curses.noecho()
- ch = 0
- msg = ''
- ihist = ''
- hidx = len(history)
- [y, x] = scr.getyx()
- reprompt(scr, msg, y, len(prompt) + len(msg))
- scr.refresh()
- while ch != ord(b'\n'):
+
+ def _show_welcome(self, scr) -> None:
+ """Display welcome message and connection information"""
+ scr.addstr("\n\n" + '-' * 70 + "\n")
+ scr.addstr("RPC Tool\n")
+ scr.addstr('-' * 70 + "\n")
+ scr.addstr(f"Connection url: '{self.state.client.address}'\n")
+
try:
- ch = scr.getch()
- except KeyboardInterrupt:
- msg = 'exit'
- break
- [y, x] = scr.getyx()
- pos = x - len(prompt)
- if ch == ord(b'\t'):
- msg, matches = autocomplete(msg)
- if len(matches) > 1:
- scr.addstr('\n')
- scr.addstr(', '.join(matches))
- scr.addstr('\n')
- scr.clrtobot()
- reprompt(scr, msg, y, len(prompt) + len(msg))
- if ch == ord(b'\n'):
- break
- if ch == 4:
- msg = 'exit'
- break
- elif ch == curses.KEY_BACKSPACE or ch == 127:
- if pos > 0:
- scr.delch(y, x - 1)
- msg = msg[0:pos - 1] + msg[pos:]
- elif ch == curses.KEY_DC:
- scr.delch(y, x)
- msg = msg[0:pos] + msg[pos + 1:]
- elif ch == curses.KEY_LEFT:
- if pos > 0:
- scr.move(y, x - 1)
- elif ch == curses.KEY_RIGHT:
- if pos < len(msg):
- scr.move(y, x + 1)
- elif ch == curses.KEY_HOME:
- scr.move(y, len(prompt))
- elif ch == curses.KEY_END:
- scr.move(y, len(prompt) + len(msg))
- elif ch == curses.KEY_UP:
- if hidx == len(history):
- ihist = msg
- hidx = max(hidx - 1, 0)
- msg = history[hidx]
- reprompt(scr, msg, y, len(prompt) + len(msg))
- elif ch == curses.KEY_DOWN:
- hidx = min(hidx + 1, len(history))
- if hidx == len(history):
- msg = ihist
- else:
- msg = history[hidx]
- reprompt(scr, msg, y, len(prompt) + len(msg))
- elif is_printable(ch):
- msg = msg[0:pos] + curses.ascii.unctrl(ch) + msg[pos:]
- reprompt(scr, msg, y, x + 1)
- # else:
- # print(f" {ch} -- {type(ch)}")
+ jukebox_version = self.state.client.enque('misc', 'get_version')
+ except Exception:
+ jukebox_version = "unknown"
+
+ scr.addstr(f"Jukebox version: {jukebox_version}\n")
+ scr.addstr(f"Pyzmq version: {zmq.pyzmq_version()}; ZMQ version: {zmq.zmq_version()}; "
+ f"has draft API: {zmq.DRAFT_API}\n")
+ scr.addstr('-' * 70 + "\n")
scr.refresh()
- scr.refresh()
- history.append(msg)
- return msg
+ def _show_help(self, scr, topic: str = '') -> None:
+ """Display help information for commands"""
+ scr.erase()
+ self.update_help(scr)
+ max_y, max_x = scr.getmaxyx()
+ scr.addstr("Available commands:\n\n")
+
+ for key, value in self.state.rpc_help.items():
+ if not key.startswith(topic):
+ continue
+
+ sign: str = value['signature']
+ sign = sign[sign.find('('):]
+ func = f"{key}{sign}"
+ scr.addstr(f"{func:50}: {value['description']}\n")
+
+ # Handle pagination
+ y, x = scr.getyx()
+ if y == max_y - 1:
+ scr.addstr("--HIT A KEY TO CONTINUE--")
+ scr.getch()
+ scr.erase()
-def tonum(string_value):
- ret = string_value
- try:
- ret = int(string_value)
- except ValueError:
- pass
- else:
- return ret
- try:
- ret = float(string_value)
- except ValueError:
- pass
- else:
- return ret
- if string_value.isalnum() and string_value.startswith('0x'):
- try:
- ret = int(string_value, base=16)
- except ValueError:
- pass
- else:
- return ret
- return ret
-
-
-def main(scr):
- global candidates
- scr.idlok(True)
- scr.scrollok(True)
- format_welcome(scr)
- get_help(scr)
- format_usage(scr)
- cmd = ''
- while cmd != 'exit':
- cmd = get_input(scr)
scr.addstr("\n")
- # Split on whitespaces to separate cmd and arg list
- dec = [v for v in cmd.strip().split(' ') if len(v) > 0]
- if len(dec) == 0:
- continue
- elif dec[0] == 'help':
- topic = ''
- if len(dec) > 1:
- topic = dec[1]
- format_help(scr, topic)
- continue
- elif dec[0] == 'usage':
- format_usage(scr)
- continue
- # scr.addstr(f"\n{cmd}\n")
- # Split cmd on '.' into package.plugin.method
- # Remove duplicate '.' along the way
- sl = [v for v in dec[0].split('.') if len(v) > 0]
- fargs = [tonum(a) for a in dec[1:]]
- scr.addstr(f"\n:: Command = {sl}, args = {fargs}\n")
- response = None
- method = None
- if not (2 <= len(sl) <= 3):
- scr.addstr(":: Error = Ill-formatted command\n")
- continue
- if len(sl) == 3:
- method = sl[2]
- try:
- response = client.enque(sl[0], sl[1], method, args=fargs)
- except zmq.error.Again:
- scr.addstr("\n\n" + '-' * 70 + "\n")
- scr.addstr("Could not reach RPC Server. Jukebox running? Correct Port?\n")
- scr.addstr('-' * 70 + "\n\n")
- scr.refresh()
- except Exception as e:
- scr.addstr(f":: Exception response =\n{e}\n")
- else:
- scr.addstr(f"\n:: Response =\n{response}\n\n")
+ scr.refresh()
+
+ def _show_usage(self, scr) -> None:
+ """Display usage information and key bindings"""
+ scr.addstr("\n\nUsage:\n")
+ scr.addstr(" > cmd [arg1] [arg2] [kwarg1=value1]\n")
+ scr.addstr("Examples:\n")
+ scr.addstr(" > volume.ctrl.set_volume 50\n")
+ example = (
+ ' > player.ctrl.play_from_reader '
+ 'content={"albumartist": "Taylor Swift", "album": "Fearless"} '
+ 'content_type=album\n'
+ )
+ scr.addstr(example)
+ scr.addstr("\nSupported argument formats:\n")
+ scr.addstr(" - Simple values (strings, numbers)\n")
+ scr.addstr(" - Hexadecimal numbers (0x...)\n")
+ scr.addstr(" - JSON objects and arrays for keyword arguments\n")
+ scr.addstr("Note: JSON must be valid and properly quoted\n")
+ scr.addstr("\n")
+ scr.addstr("Use for auto-completion of commands!\n")
+ scr.addstr("Use / for command history!\n")
+ scr.addstr("\n")
+ scr.addstr("Type help , to get a list of all commands\n")
+ scr.addstr("Type usage , to get this usage help\n")
+ scr.addstr("\n")
+ scr.addstr("To exit, press Ctrl-D or type 'exit'\n")
+ scr.addstr("\n")
+ scr.refresh()
+ def _get_common_beginning(self, strings: List[str]) -> str:
+ """Find common prefix among a list of strings"""
+ if not strings:
+ return ""
-def runcmd(cmd):
- """
- Just run a command.
- Right now duplicates more or less main()
- :todo remove duplication of code
- """
-
- # Split on whitespaces to separate cmd and arg list
- dec = [v for v in cmd.strip().split(' ') if len(v) > 0]
- if len(dec) == 0:
- return
- # Split cmd on '.' into package.plugin.method
- # Remove duplicate '.' along the way
- sl = [v for v in dec[0].split('.') if len(v) > 0]
- fargs = [tonum(a) for a in dec[1:]]
- response = None
- method = None
- if not (2 <= len(sl) <= 3):
- print(":: Error = Ill-formatted command\n")
- return
- if len(sl) == 3:
- method = sl[2]
- try:
- response = client.enque(sl[0], sl[1], method, args=fargs)
- except zmq.error.Again:
- print("\n\n" + '-' * 70 + "\n")
- print("Could not reach RPC Server. Jukebox running? Correct Port?\n")
- print('-' * 70 + "\n\n")
- return
- except Exception as e:
- print(f":: Exception response =\n{e}\n")
- return
- else:
- print(f"\n:: Response =\n{response}\n\n")
+ result = []
+ limit = min(len(s) for s in strings)
+ for i in range(limit):
+ chars = set(s[i] for s in strings)
+ if len(chars) == 1:
+ result.append(chars.pop())
+ else:
+ break
+
+ return ''.join(result)
+
+ def _autocomplete(self, msg: str) -> Tuple[str, List[str]]:
+ """Handle command autocompletion"""
+ matches = [s for s in self.state.candidates if s.startswith(msg)]
+ if not matches:
+ return msg, matches
+
+ common = self._get_common_beginning(matches)
+ return common, matches
+
+ def _is_printable(self, ch: int) -> bool:
+ """Check if character is printable"""
+ return 32 <= ch <= 127
+
+ def _reprompt(self, scr, msg: str, y: int, x: int) -> None:
+ """Redraw prompt and message"""
+ scr.move(y, 0)
+ scr.clrtoeol()
+ scr.addstr(self.state.prompt)
+ scr.addstr(msg)
+ scr.move(y, x)
+
+ def _get_input(self, scr) -> str: # noqa: C901
+ """Handle user input with history and autocompletion"""
+ ch = 0
+ msg = ''
+ ihist = ''
+ hidx = len(self.state.history)
+
+ y, x = scr.getyx()
+ self._reprompt(scr, msg, y, len(self.state.prompt) + len(msg))
+ scr.refresh()
-if __name__ == '__main__':
+ while ch != ord(b'\n'):
+ try:
+ ch = scr.getch()
+ except KeyboardInterrupt:
+ return 'exit'
+
+ y, x = scr.getyx()
+ pos = x - len(self.state.prompt)
+
+ if ch == ord(b'\t'):
+ msg, matches = self._autocomplete(msg)
+ if len(matches) > 1:
+ scr.addstr('\n')
+ scr.addstr(', '.join(matches))
+ scr.addstr('\n')
+ scr.clrtobot()
+ self._reprompt(scr, msg, y, len(self.state.prompt) + len(msg))
+
+ elif ch == ord(b'\n'):
+ break
+ elif ch == 4: # Ctrl-D
+ return 'exit'
+ elif ch in (curses.KEY_BACKSPACE, 127):
+ if pos > 0:
+ scr.delch(y, x - 1)
+ msg = msg[0:pos - 1] + msg[pos:]
+ elif ch == curses.KEY_DC:
+ scr.delch(y, x)
+ msg = msg[0:pos] + msg[pos + 1:]
+ elif ch == curses.KEY_LEFT:
+ if pos > 0:
+ scr.move(y, x - 1)
+ elif ch == curses.KEY_RIGHT:
+ if pos < len(msg):
+ scr.move(y, x + 1)
+ elif ch == curses.KEY_HOME:
+ scr.move(y, len(self.state.prompt))
+ elif ch == curses.KEY_END:
+ scr.move(y, len(self.state.prompt) + len(msg))
+ elif ch == curses.KEY_UP:
+ if hidx == len(self.state.history):
+ ihist = msg
+ hidx = max(hidx - 1, 0)
+ msg = self.state.history[hidx]
+ self._reprompt(scr, msg, y, len(self.state.prompt) + len(msg))
+ elif ch == curses.KEY_DOWN:
+ hidx = min(hidx + 1, len(self.state.history))
+ msg = ihist if hidx == len(self.state.history) else self.state.history[hidx]
+ self._reprompt(scr, msg, y, len(self.state.prompt) + len(msg))
+ elif self._is_printable(ch):
+ msg = msg[0:pos] + curses.ascii.unctrl(ch) + msg[pos:]
+ self._reprompt(scr, msg, y, x + 1)
+
+ scr.refresh()
+
+ self.state.history.append(msg)
+ return msg
+
+
+def main():
+ """CLI entry point with argument parsing"""
default_tcp = 5555
default_ws = 5556
- url = f"tcp://localhost:{default_tcp}"
- argparser = argparse.ArgumentParser(description='The Jukebox RPC command line tool',
- epilog=f'Default connection: {url}')
- port_group = argparser.add_mutually_exclusive_group()
- port_group.add_argument("-w", "--websocket",
- help=f"Use websocket protocol on PORT [default: {default_ws}]",
- nargs='?', const=default_ws,
- metavar="PORT", default=None)
- port_group.add_argument("-t", "--tcp",
- help=f"Use tcp protocol on PORT [default: {default_tcp}]",
- nargs='?', const=default_tcp,
- metavar="PORT", default=None)
- port_group.add_argument("-c", "--command",
- help="Send command to Jukebox server",
- default=None)
- args = argparser.parse_args()
-
+ default_url = f"tcp://localhost:{default_tcp}"
+
+ parser = argparse.ArgumentParser(
+ description='The Jukebox RPC command line tool',
+ epilog=f'Default connection: {default_url}'
+ )
+
+ port_group = parser.add_mutually_exclusive_group()
+ port_group.add_argument(
+ "-w", "--websocket",
+ help=f"Use websocket protocol on PORT [default: {default_ws}]",
+ nargs='?', const=default_ws,
+ metavar="PORT", default=None
+ )
+ port_group.add_argument(
+ "-t", "--tcp",
+ help=f"Use tcp protocol on PORT [default: {default_tcp}]",
+ nargs='?', const=default_tcp,
+ metavar="PORT", default=None
+ )
+ port_group.add_argument(
+ "-c", "--command",
+ help="Send command to Jukebox server",
+ default=None
+ )
+
+ args = parser.parse_args()
+
+ url = default_url
if args.websocket is not None:
url = f"ws://localhost:{args.websocket}"
elif args.tcp is not None:
@@ -401,12 +491,17 @@ def runcmd(cmd):
print(f">>> RPC Client connect on {url}")
- client = rpc.RpcClient(url)
+ cli = JukeboxCli(url)
if args.command is not None:
- runcmd(args.command)
- exit(0)
+ # Handle single command execution
+ cli.execute_command(None, args.command)
else:
- curses.wrapper(main)
+ # Run interactive CLI
+ curses.wrapper(cli.run)
print(">>> RPC Client exited!")
+
+
+if __name__ == '__main__':
+ main()
From 3fa5b167c2d46db54c4ebf491c62e90178f1e04b Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sat, 23 Nov 2024 16:12:18 +0100
Subject: [PATCH 10/16] Remove C based RPC client
---
src/cli_client/pbc.c | 261 -------------------------------------------
1 file changed, 261 deletions(-)
delete mode 100644 src/cli_client/pbc.c
diff --git a/src/cli_client/pbc.c b/src/cli_client/pbc.c
deleted file mode 100644
index 7f39c493e..000000000
--- a/src/cli_client/pbc.c
+++ /dev/null
@@ -1,261 +0,0 @@
-/**
- \file pbc.c
-
- MIT License
-
- Copyright (C) 2021 Arne Pagel
-
- Permission is hereby granted, free of charge, to any person obtaining a copy
- of this software and associated documentation files (the "Software"), to deal
- in the Software without restriction, including without limitation the rights
- to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- copies of the Software, and to permit persons to whom the Software is
- furnished to do so, subject to the following conditions:
-
- The above copyright notice and this permission notice shall be included in all
- copies or substantial portions of the Software.
-
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
- SOFTWARE.
-*/
-
-/*
-
- pbc -> PhonieBox Command line interface
-
- depends on libczmq:
- apt-get install libczmq-dev
-
- how to compile:
- gcc pbc.c -o pbc -lzmq -Wall
-
-*/
-
-#include
-#include
-
-#define MAX_STRLEN 256
-#define MAX_REQEST_STRLEN (MAX_STRLEN * 16)
-#define MAX_PARAMS 16
-int g_verbose = 0;
-
-typedef struct
-{
- char object [MAX_STRLEN];
- char package [MAX_STRLEN];
- char method [MAX_STRLEN];
- char params [MAX_PARAMS][MAX_STRLEN];
- int num_params;
- char address [MAX_STRLEN];
-} t_request;
-
-
-int send_zmq_request_and_wait_response(char * request, int request_len, char * response, int max_response_len, char * address)
-{
- int zmq_ret,ret = -1;
- void *context = zmq_ctx_new ();
- void *requester = zmq_socket (context, ZMQ_REQ);
- int linger = 200;
-
- if (g_verbose)
- {
- int major, minor, patch;
- zmq_version (&major, &minor, &patch);
- printf ("Current ØMQ version is %d.%d.%d\n", major, minor, patch);
- }
-
- zmq_setsockopt(requester,ZMQ_LINGER,&linger,sizeof(linger));
- zmq_setsockopt(requester,ZMQ_RCVTIMEO,&linger,sizeof(linger));
- zmq_connect (requester, address);
-
- if (g_verbose) printf("connected to: %s",address);
-
-
- zmq_ret = zmq_send (requester, request, request_len, 0);
-
- if (zmq_ret > 0)
- {
- zmq_ret = zmq_recv (requester, response, max_response_len, 0);
-
- if (zmq_ret > 0)
- {
- printf ("Received %s (%d Bytes)\n", response,zmq_ret);
- ret = 0;
- }
- else
- {
- printf ("zmq_recv rturned %d \n", zmq_ret);
- }
- }
- else
- {
- if (g_verbose) printf ("zmq_send returned %d\n", zmq_ret);
- }
-
- zmq_close (requester);
- zmq_ctx_destroy (context);
- return (ret);
-}
-
-
-void * connect_and_send_request(t_request * tr)
-{
- char json_request[MAX_REQEST_STRLEN];
- char json_response[MAX_REQEST_STRLEN];
- char kwargs[MAX_STRLEN * 8];
- size_t json_len;
- int n;
-
- if (tr->num_params > 0)
- {
- sprintf(kwargs, "\"kwargs\":{");
-
- for (n = 0;n < tr->num_params;)
- {
- strcat(kwargs,tr->params[n]);
- n++;
- if (n < tr->num_params) strcat(kwargs,",");
- }
-
- strcat(kwargs,"},");
-
- }
- else sprintf(kwargs, "\"kwargs\":{},");
-
- snprintf(json_request,MAX_REQEST_STRLEN,"{\"package\": \"%s\", \"plugin\": \"%s\", \"method\": \"%s\", %s\"id\":%d}",tr->package,tr->object,tr->method,kwargs,123);
- json_len = strlen(json_request);
-
- if (g_verbose) printf("Sending Request (%ld Bytes):\n%s\n",json_len,json_request);
-
- send_zmq_request_and_wait_response(json_request,json_len,json_response,MAX_REQEST_STRLEN,tr->address);
-
- return 0;
-}
-
-int check_and_map_parameters_to_json(char * arg, t_request * tr)
-{
- char * name;
- char * value;
- char * fmt;
- int ret = 0;
- if (strchr(arg, ':') != NULL)
- {
- name = strtok(arg, ":");
- value = strtok(NULL, ":");
- fmt = (isdigit(*value)||*value=='-') ? "\"%s\":%s" : "\"%s\":\"%s\"";
- snprintf (tr->params[tr->num_params++],MAX_STRLEN, fmt,name,value);
- ret = 1;
- }
- return (ret);
-}
-
-
-void usage(void)
-{
- fprintf(stderr,"\npbc -> PhonieBox Command line interface\nusage: pbc -p package -o plugin -m method param_name:value\n\n");
- fprintf(stderr," -h this screen\n");
- fprintf(stderr," -p, --package package\n");
- fprintf(stderr," -o, --object plugin\n");
- fprintf(stderr," -m, --method method\n");
- fprintf(stderr," -a, --address default=tcp://localhost:5555\n");
- fprintf(stderr," -v verbose\n");
-
- fprintf(stderr,"last change %s\n\n",__DATE__);
- exit (1);
-}
-
-/**
- returns the index of the first argument that is not an option; i.e.
- does not start with a dash or a slash
-*/
-int HandleOptions(int argc,char *argv[], t_request * tr)
-{
- int c;
- sprintf(tr->address,"tcp://localhost:5555");
-
- const struct option long_options[] =
- {
- /* These options set a flag. */
- //{"verbose", no_argument, &verbose_flag, 1},
- //{"brief", no_argument, &verbose_flag, 0},
- /* These options don't set a flag.
- We distinguish them by their indices. */
- {"help", no_argument, 0, 'h'},
- {"package", required_argument, 0, 'p'},
- {"object", required_argument, 0, 'o'},
- {"method", required_argument, 0, 'm'},
- {"address", required_argument, 0, 'a'},
- {0, 0, 0, 0}
- };
-
- const char short_options[] = {"o:m:p:a:?hv"};
-
- while (1)
- {
- int option_index = 0; // getopt_long stores the option index here.
-
- c = getopt_long (argc, argv,short_options,long_options, &option_index);
-
- // Detect the end of the options.
- if (c == -1) break;
-
- switch (c)
- {
- case '?':
- case 'h':
- usage();
- puts ("option -a\n");
- break;
- case 'p':
- strncpy (tr->package,optarg,MAX_STRLEN);
- break;
- case 'o':
- strncpy (tr->object,optarg,MAX_STRLEN);
- break;
-
- case 'm':
- strncpy (tr->method,optarg,MAX_STRLEN);
- break;
-
- case 'v':
- g_verbose = '1';
- break;
-
- case 'a':
- strncpy (tr->address,optarg,MAX_STRLEN);
- break;
-
- default:
- usage();
- abort ();
- }
- }
-
- /* treat remaining command line arguments (not options). */
- if (optind < argc)
- {
- while (optind < argc)
- {
- check_and_map_parameters_to_json(argv[optind++], tr);
- }
- }
-
- return (1);
-}
-
-int main(int argc,char *argv[])
-{
- t_request tr;
-
- bzero(&tr, sizeof(t_request));
-
- HandleOptions(argc,argv,&tr);
- connect_and_send_request(&tr);
-
- return 0;
-}
From e0194298d47ac6c4396b5183675d3253bfc74216 Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sat, 23 Nov 2024 21:34:39 +0100
Subject: [PATCH 11/16] [Installation] Fix function call in helper file
---
installation/includes/02_helpers.sh | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/installation/includes/02_helpers.sh b/installation/includes/02_helpers.sh
index 440e10b08..1b23abe36 100644
--- a/installation/includes/02_helpers.sh
+++ b/installation/includes/02_helpers.sh
@@ -84,7 +84,7 @@ is_debian_based() {
fi
}
-_get_debian_version_number() {
+get_debian_version_number() {
if [ "$(is_debian_based)" = true ]; then
local debian_version_number=$( . /etc/os-release; printf '%s\n' "$VERSION_ID"; )
echo "$debian_version_number"
From 13fe8415bcb8d4ffc38399149418b161691056d6 Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sat, 23 Nov 2024 23:48:46 +0100
Subject: [PATCH 12/16] [Installation] Fix function call in helper file 2
---
installation/includes/02_helpers.sh | 3 +--
1 file changed, 1 insertion(+), 2 deletions(-)
diff --git a/installation/includes/02_helpers.sh b/installation/includes/02_helpers.sh
index 1b23abe36..928274e71 100644
--- a/installation/includes/02_helpers.sh
+++ b/installation/includes/02_helpers.sh
@@ -106,8 +106,7 @@ is_debian_version_at_least() {
_get_boot_file_path() {
local filename="$1"
- local is_debian_version_number_at_least_12=$(is_debian_version_at_least 12)
- if [ "$(is_debian_version_number_at_least_12)" = true ]; then
+ if [ "$(is_debian_version_at_least 12)" = true ]; then
echo "/boot/firmware/${filename}"
else
echo "/boot/${filename}"
From e82e3745d43c1b6b8908c2e2e091d7235b97d379 Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sun, 24 Nov 2024 00:19:58 +0100
Subject: [PATCH 13/16] Update docs for RPC client
---
documentation/builders/cli-client.md | 105 ++++++++++++++++++++++++---
1 file changed, 93 insertions(+), 12 deletions(-)
diff --git a/documentation/builders/cli-client.md b/documentation/builders/cli-client.md
index b75e942ee..ff1c036cd 100644
--- a/documentation/builders/cli-client.md
+++ b/documentation/builders/cli-client.md
@@ -1,21 +1,102 @@
-# CLI Client
+# RPC CLI Client
-The CLI (command line interface) client can be used to send [RPC commands](./rpc-commands.md) from command line to Phoniebox.
+The Python CLI (Command Line Interface) client can be used to send [RPC commands](./rpc-commands.md) to Phoniebox. It provides an interactive shell with autocompletion and command history, as well as direct command execution.
-## Installation
+The RPC tool can be found here:
-* Install prerequisites: `sudo apt-get install libczmq-dev`
-* Change to directory: `cd ~/RPi-Jukebox-RFID/src/cli_client`
-* Compile CLI client: `gcc pbc.c -o pbc -lzmq -Wall`
+```bash
+~/RPi-Jukebox-RFID/src/jukebox/run_rpc_tool.py
+```
## Usage
-* Get help info: `./pbc -h`
-* Example shutdown: `./pbc -p host -o shutdown`
+The CLI tool can be used in two modes:
-See also [RPC Commands](./rpc-commands.md) reference.
+### Interactive Mode
-## Reference
+```bash
+# Start interactive shell
+./run_rpc_tool.py
-*
-*
+# Start with specific connection type
+./run_rpc_tool.py --tcp 5555 # TCP connection on port 5555
+./run_rpc_tool.py --websocket # WebSocket connection on default port
+```
+
+In interactive mode:
+
+- Use TAB for command autocompletion
+- Use UP/DOWN arrows for command history
+- Type `help` to see available commands
+- Type `usage` for detailed usage information
+- Press Ctrl-D or type `exit` to quit
+
+### Direct Command Mode
+
+```bash
+# Execute single command
+./run_rpc_tool.py -c 'command [args...] [key=value...]'
+
+# Examples with positional args:
+./run_rpc_tool.py -c 'volume.ctrl.set_volume 50'
+./run_rpc_tool.py -c 'player.ctrl.play_content "/music/test.mp3" single'
+
+# Examples with kwargs:
+./run_rpc_tool.py -c 'volume.ctrl.set_volume level=50'
+./run_rpc_tool.py -c 'player.ctrl.play_content content="/music/test.mp3" content_type=single'
+```
+
+## Command Format
+
+Commands support both positional arguments and keyword arguments:
+
+```python
+package.plugin.method [arg1] [arg2] [arg3] # Positional args
+package.plugin.method [key1=value1] [key2=value2] # Keyword args
+```
+
+Arguments can be:
+
+- Numbers (50 or level=50)
+- Strings (use quotes for spaces: "my string" or path="my string")
+- JSON objects (use single quotes: '{"key":"value"}')
+- Hexadecimal numbers (prefix with 0x: 0xFF or value=0xFF)
+
+### Examples
+
+```bash
+# Simple commands - both styles work
+volume.ctrl.set_volume 50
+volume.ctrl.set_volume level=50
+
+system.ctrl.shutdown
+
+# Playing content - positional args
+player.ctrl.play_content '{"artist":"Pink Floyd","album":"The Wall"}' album
+player.ctrl.play_content "/music/classical" folder true
+player.ctrl.play_content "/music/track.mp3" single
+
+# Playing content - keyword args
+player.ctrl.play_content content='{"artist":"Pink Floyd","album":"The Wall"}' content_type=album
+player.ctrl.play_content content="/music/classical" content_type=folder recursive=true
+player.ctrl.play_content content="/music/track.mp3" content_type=single
+
+# Reader-based playback - positional args
+player.ctrl.play_from_reader '{"artist":"Pink Floyd","album":"The Wall"}' album false toggle
+player.ctrl.play_from_reader "/music/classical" folder true replay
+
+# Reader-based playback - keyword args
+player.ctrl.play_from_reader content='{"artist":"Pink Floyd","album":"The Wall"}' content_type=album second_swipe=toggle
+player.ctrl.play_from_reader content="/music/classical" content_type=folder recursive=true second_swipe=replay
+```
+
+## Features
+
+- Command autocompletion
+- Command history
+- Support for both positional and keyword arguments
+- JSON argument support
+- Interactive and direct command modes
+- Automatic type conversion (strings, numbers, JSON)
+- Connection error handling
+- Dynamic command help from server
From 91e9a7286090567681be080c8cd71c90d6c5a702 Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sat, 28 Dec 2024 11:48:52 +0100
Subject: [PATCH 14/16] Make second swipe finally work
---
src/jukebox/components/playermpd/__init__.py | 62 +++++++++----------
.../playermpd/play_content_callback.py | 7 +--
.../playermpd/play_content_handler.py | 18 +++---
3 files changed, 39 insertions(+), 48 deletions(-)
diff --git a/src/jukebox/components/playermpd/__init__.py b/src/jukebox/components/playermpd/__init__.py
index 161b81679..1028f6d61 100644
--- a/src/jukebox/components/playermpd/__init__.py
+++ b/src/jukebox/components/playermpd/__init__.py
@@ -157,7 +157,7 @@ def __init__(self):
'replay': self.replay,
'replay_if_stopped': self.replay_if_stopped}
self.second_swipe_action = None
- self.decode_2nd_swipe_option()
+ self.decode_second_swipe_action()
self.end_of_playlist_next_action = utils.get_config_action(cfg,
'playermpd',
@@ -250,21 +250,20 @@ def exit(self):
def connect(self):
self.mpd_client.connect(self.mpd_host, 6600)
- def decode_2nd_swipe_option(self):
- cfg_2nd_swipe_action = cfg.setndefault('playermpd', 'second_swipe_action', 'alias', value='none').lower()
- if cfg_2nd_swipe_action not in [*self.second_swipe_action_dict.keys(), 'none', 'custom']:
- logger.error(f"Config mpd.second_swipe_action must be one of "
- f"{[*self.second_swipe_action_dict.keys(), 'none', 'custom']}. Ignore setting.")
- if cfg_2nd_swipe_action in self.second_swipe_action_dict.keys():
- self.second_swipe_action = self.second_swipe_action_dict[cfg_2nd_swipe_action]
- if cfg_2nd_swipe_action == 'custom':
- custom_action = utils.decode_rpc_call(cfg.getn('playermpd', 'second_swipe_action', default=None))
- self.second_swipe_action = functools.partial(plugs.call_ignore_errors,
- custom_action['package'],
- custom_action['plugin'],
- custom_action['method'],
- custom_action['args'],
- custom_action['kwargs'])
+ def decode_second_swipe_action(self):
+ """
+ Decode the second swipe option from the configuration
+ """
+ logger.debug("Decoding second swipe option")
+ second_swipe_action = cfg.getn('playermpd', 'second_swipe_action', 'none')
+ logger.debug(f"Second swipe option from config: {second_swipe_action}")
+
+ if second_swipe_action in self.second_swipe_action_dict:
+ self.second_swipe_action = self.second_swipe_action_dict[second_swipe_action]
+ logger.debug(f"Second swipe action set to: {self.second_swipe_action}")
+ else:
+ self.second_swipe_action = None
+ logger.debug("No valid second swipe action found, setting to None")
def mpd_retry_with_mutex(self, mpd_cmd, *args):
"""
@@ -570,16 +569,16 @@ def _play_folder_internal(self, folder: str, recursive: bool) -> None:
self.mpd_client.play()
@plugs.tag
- def play_content(self, content: Union[str, Dict[str, Any]], content_type: str = 'folder', recursive: bool = False):
+ def play_content(self, content: Union[str, Dict[str, Any]], content_type: str = 'folder',
+ recursive: bool = False, preserve_second_swipe: bool = False):
"""
- Main entry point for trigger music playing from any source (RFID reader, web UI, etc.).
- Does NOT support second swipe - use play_from_reader() for that.
+ Main entry point for playing content.
- :param content: Content identifier:
- - For singles/folders: file/folder path as string
- - For albums: dict with 'albumartist' and 'album' keys
- :param content_type: Type of content ('single', 'album', 'folder')
- :param recursive: Add folder recursively (only used for folder type)
+ Args:
+ content: Content to play
+ content_type: Type of content ('folder', 'album', etc.)
+ recursive: Whether to play recursively
+ preserve_second_swipe: If True, preserves any existing second_swipe_action
"""
try:
content_type = content_type.lower()
@@ -612,15 +611,16 @@ def play_content(self, content: Union[str, Dict[str, Any]], content_type: str =
recursive=recursive
)
+ old_action = self.play_content_handler.second_swipe_action
# Ensure no second swipe for regular content playback
- old_action = self.play_content_handler._second_swipe_action
- self.play_content_handler._second_swipe_action = None
+ if not preserve_second_swipe:
+ self.play_content_handler.second_swipe_action = None
try:
self.play_content_handler.play_content(play_content)
finally:
# Restore previous second swipe action
- self.play_content_handler._second_swipe_action = old_action
+ self.play_content_handler.second_swipe_action = old_action
except Exception as e:
logger.error(f"Error playing content: {e}")
@@ -643,7 +643,6 @@ def play_from_reader(self, content: Union[str, Dict[str, str]], content_type: st
- 'none': disable second swipe
- One of: 'toggle', 'play', 'skip', 'rewind', 'replay', 'replay_if_stopped'
"""
- # Determine second swipe action
if second_swipe is None:
action = self.second_swipe_action
elif second_swipe.lower() == 'none':
@@ -651,17 +650,14 @@ def play_from_reader(self, content: Union[str, Dict[str, str]], content_type: st
else:
action = self.second_swipe_action_dict.get(second_swipe.lower())
if action is None:
- logger.error(f"Unknown second swipe action '{second_swipe}', using default")
action = self.second_swipe_action
- # Temporarily set the chosen second swipe action
- old_action = self.play_content_handler._second_swipe_action
+ old_action = self.play_content_handler.second_swipe_action
self.play_content_handler.set_second_swipe_action(action)
try:
- self.play_content(content, content_type, recursive)
+ self.play_content(content, content_type, recursive, preserve_second_swipe=True)
finally:
- # Restore previous second swipe action
self.play_content_handler.set_second_swipe_action(old_action)
# The following methods are kept for backward compatibility but now use play_content internally
diff --git a/src/jukebox/components/playermpd/play_content_callback.py b/src/jukebox/components/playermpd/play_content_callback.py
index ce5a1b8fb..d15e9e8bc 100644
--- a/src/jukebox/components/playermpd/play_content_callback.py
+++ b/src/jukebox/components/playermpd/play_content_callback.py
@@ -1,4 +1,3 @@
-
from enum import Enum
from typing import Callable, Generic, TypeVar
@@ -6,8 +5,8 @@
class PlayCardState(Enum):
- firstSwipe = 0,
- secondSwipe = 1
+ FIRST_SWIPE = 0
+ SECOND_SWIPE = 1
STATE = TypeVar('STATE', bound=Enum)
@@ -27,7 +26,7 @@ def register(self, func: Callable[[str, STATE], None]):
.. py:function:: func(folder: str, state: STATE)
:noindex:
- :param folder: relativ path to folder to play
+ :param folder: relative path to folder to play
:param state: indicator of the state inside the calling
"""
super().register(func)
diff --git a/src/jukebox/components/playermpd/play_content_handler.py b/src/jukebox/components/playermpd/play_content_handler.py
index 660b82c28..5273e2857 100644
--- a/src/jukebox/components/playermpd/play_content_handler.py
+++ b/src/jukebox/components/playermpd/play_content_handler.py
@@ -44,23 +44,21 @@ class PlayContentHandler:
def __init__(self, player: PlayerProtocol):
self.player = player
self.last_played_content: Optional[PlayContent] = None
- self._second_swipe_action = None
+ self.second_swipe_action = None
+ self._instance_id = id(self)
def set_second_swipe_action(self, action: Optional[Callable]) -> None:
"""Set the action to be performed on second swipe"""
- self._second_swipe_action = action
+ self.second_swipe_action = action
def _play_content(self, content: PlayContent) -> None:
"""Internal method to play content based on its type"""
if content.type == PlayContentType.SINGLE:
- logger.debug(f"Playing single track: {content.content}")
self.player._play_single_internal(content.content)
elif content.type == PlayContentType.ALBUM:
artist, album = content.content
- logger.debug(f"Playing album: {album} by {artist}")
self.player._play_album_internal(artist, album)
elif content.type == PlayContentType.FOLDER:
- logger.debug(f"Playing folder: {content.content} (recursive={content.recursive})")
self.player._play_folder_internal(content.content, content.recursive)
def play_content(self, content: PlayContent) -> None:
@@ -77,20 +75,18 @@ def play_content(self, content: PlayContent) -> None:
and content.content == self.last_played_content.content):
is_second_swipe = True
- if self._second_swipe_action is not None and is_second_swipe:
- logger.debug('Calling second swipe action')
+ if self.second_swipe_action is not None and is_second_swipe:
# run callbacks before second_swipe_action is invoked
self.player.play_card_callbacks.run_callbacks(
str(content.content),
- PlayCardState.secondSwipe # Use imported PlayCardState directly
+ PlayCardState.SECOND_SWIPE
)
- self._second_swipe_action()
+ self.second_swipe_action()
else:
- logger.debug('Calling first swipe action')
# run callbacks before play_content is invoked
self.player.play_card_callbacks.run_callbacks(
str(content.content),
- PlayCardState.firstSwipe # Use imported PlayCardState directly
+ PlayCardState.FIRST_SWIPE
)
self._play_content(content)
From e2d2e4301a971a38cfbdb66d8aac34731ef99d56 Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sat, 28 Dec 2024 22:31:50 +0100
Subject: [PATCH 15/16] Allow to update second swipe action in web app
---
.../default-settings/jukebox.default.yaml | 5 +-
src/jukebox/components/playermpd/__init__.py | 48 +++++-------
src/webapp/public/locales/de/translation.json | 6 +-
src/webapp/public/locales/en/translation.json | 6 +-
src/webapp/src/commands/index.js | 12 +++
.../src/components/Settings/autohotspot.js | 1 +
.../src/components/Settings/secondswipe.js | 76 ++++++++++++++-----
7 files changed, 93 insertions(+), 61 deletions(-)
diff --git a/resources/default-settings/jukebox.default.yaml b/resources/default-settings/jukebox.default.yaml
index d3326ef55..37349d014 100644
--- a/resources/default-settings/jukebox.default.yaml
+++ b/resources/default-settings/jukebox.default.yaml
@@ -79,10 +79,7 @@ alsawave:
playermpd:
host: localhost
status_file: ../../shared/settings/music_player_status.json
- second_swipe_action:
- # Note: Does not follow the RPC alias convention (yet)
- # Must be one of: 'toggle', 'play', 'skip', 'rewind', 'replay', 'none'
- alias: toggle
+ second_swipe_action: toggle # Must be one of: 'toggle', 'next', 'rewind', 'none'
library:
update_on_startup: true
check_user_rights: true
diff --git a/src/jukebox/components/playermpd/__init__.py b/src/jukebox/components/playermpd/__init__.py
index 1028f6d61..fb12e0cdc 100644
--- a/src/jukebox/components/playermpd/__init__.py
+++ b/src/jukebox/components/playermpd/__init__.py
@@ -151,11 +151,8 @@ def __init__(self):
self.music_player_status = self.nvm.load(cfg.getn('playermpd', 'status_file'))
self.second_swipe_action_dict = {'toggle': self.toggle,
- 'play': self.play,
- 'skip': self.next,
- 'rewind': self.rewind,
- 'replay': self.replay,
- 'replay_if_stopped': self.replay_if_stopped}
+ 'next': self.next,
+ 'rewind': self.rewind}
self.second_swipe_action = None
self.decode_second_swipe_action()
@@ -254,16 +251,12 @@ def decode_second_swipe_action(self):
"""
Decode the second swipe option from the configuration
"""
- logger.debug("Decoding second swipe option")
second_swipe_action = cfg.getn('playermpd', 'second_swipe_action', 'none')
- logger.debug(f"Second swipe option from config: {second_swipe_action}")
if second_swipe_action in self.second_swipe_action_dict:
self.second_swipe_action = self.second_swipe_action_dict[second_swipe_action]
- logger.debug(f"Second swipe action set to: {self.second_swipe_action}")
else:
self.second_swipe_action = None
- logger.debug("No valid second swipe action found, setting to None")
def mpd_retry_with_mutex(self, mpd_cmd, *args):
"""
@@ -419,34 +412,12 @@ def rewind(self):
with self.mpd_lock:
self.mpd_client.play(0)
- @plugs.tag
- def replay(self):
- """
- Re-start playing the last-played folder
-
- Will reset settings to folder config"""
- logger.debug("Replay")
- with self.mpd_lock:
- self.play_folder(self.music_player_status['player_status']['last_played_folder'])
-
@plugs.tag
def toggle(self):
"""Toggle pause state, i.e. do a pause / resume depending on current state"""
with self.mpd_lock:
self.mpd_client.pause()
- @plugs.tag
- def replay_if_stopped(self):
- """
- Re-start playing the last-played folder unless playlist is still playing
-
- > [!NOTE]
- > To me this seems much like the behaviour of play,
- > but we keep it as it is specifically implemented in box 2.X"""
- with self.mpd_lock:
- if self.mpd_status['state'] == 'stop':
- self.play_folder(self.music_player_status['player_status']['last_played_folder'])
-
# Shuffle
def _shuffle(self, random):
# As long as we don't work with waiting lists (aka playlist), this implementation is ok!
@@ -773,6 +744,21 @@ def get_song_by_url(self, song_url):
return song
+ @plugs.tag
+ def get_second_swipe_action(self):
+ action = cfg.getn('playermpd', 'second_swipe_action', default='None')
+
+ return action
+
+ @plugs.tag
+ def set_second_swipe_action(self, action):
+ if action is None:
+ cfg.setn('playermpd', 'second_swipe_action', value='None')
+ else:
+ cfg.setn('playermpd', 'second_swipe_action', value=action)
+
+ self.decode_second_swipe_action()
+
def get_volume(self):
"""
Get the current volume
diff --git a/src/webapp/public/locales/de/translation.json b/src/webapp/public/locales/de/translation.json
index 42046e7b6..a34df941c 100644
--- a/src/webapp/public/locales/de/translation.json
+++ b/src/webapp/public/locales/de/translation.json
@@ -260,10 +260,10 @@
"secondswipe": {
"title": "Erneute Aktivierung (Second Swipe)",
"description": "Aktion, wenn dieselbe Karte ein weiteres Mal aktiviert wird",
- "restart": "Wiedergabeliste neu starten",
+ "rewind": "Wiedergabeliste neu starten",
"toggle": "Pause / Wiedergabe umschalten",
- "skip": "Zum nächsten Track springen",
- "ignore": "Nur Systembefehle mehrfach ausführen"
+ "next": "Zum nächsten Track springen",
+ "none": "Keine Aktion"
},
"status": {
"title": "Systemstatus",
diff --git a/src/webapp/public/locales/en/translation.json b/src/webapp/public/locales/en/translation.json
index 358b6b1ec..15be8562c 100644
--- a/src/webapp/public/locales/en/translation.json
+++ b/src/webapp/public/locales/en/translation.json
@@ -260,10 +260,10 @@
"secondswipe": {
"title": "Second Swipe",
"description": "Second action after the same card swiped again",
- "restart": "Restart playlist",
+ "rewind": "Restart playlist",
"toggle": "Toggle pause / play",
- "skip": "Skip to next track",
- "ignore": "Ignore audio playout triggers, only system commands"
+ "next": "Skip to next track",
+ "none": "No action"
},
"status": {
"title": "System",
diff --git a/src/webapp/src/commands/index.js b/src/webapp/src/commands/index.js
index ffbc01f03..d13a33214 100644
--- a/src/webapp/src/commands/index.js
+++ b/src/webapp/src/commands/index.js
@@ -52,6 +52,18 @@ const commands = {
plugin: 'ctrl',
method: 'playerstatus'
},
+ // Player Options
+ getSecondSwipeAction: {
+ _package: 'player',
+ plugin: 'ctrl',
+ method: 'get_second_swipe_action',
+ },
+ setSecondSwipeAction: {
+ _package: 'player',
+ plugin: 'ctrl',
+ method: 'set_second_swipe_action',
+ argKeys: ['action']
+ },
// Player Actions
play: {
diff --git a/src/webapp/src/components/Settings/autohotspot.js b/src/webapp/src/components/Settings/autohotspot.js
index 0bcbf97db..90b82a63f 100644
--- a/src/webapp/src/components/Settings/autohotspot.js
+++ b/src/webapp/src/components/Settings/autohotspot.js
@@ -16,6 +16,7 @@ import { SwitchWithLoader } from '../general';
import request from '../../utils/request';
+// TODO: Update the help URL
const helpUrl = 'https://rpi-jukebox-rfid.readthedocs.io/en/latest/userguide/autohotspot.html';
const SettingsAutoHotpot = () => {
diff --git a/src/webapp/src/components/Settings/secondswipe.js b/src/webapp/src/components/Settings/secondswipe.js
index 9ecf88df8..844e3abcd 100644
--- a/src/webapp/src/components/Settings/secondswipe.js
+++ b/src/webapp/src/components/Settings/secondswipe.js
@@ -1,6 +1,5 @@
-import React from 'react';
+import React, { useEffect, useState } from 'react';
import { useTranslation } from 'react-i18next';
-
import {
Card,
CardContent,
@@ -11,44 +10,81 @@ import {
Radio,
RadioGroup,
} from '@mui/material';
+import request from '../../utils/request';
const SettingsSecondSwipe = () => {
const { t } = useTranslation();
+ const [secondSwipeAction, setSecondSwipeAction] = useState('None');
+ const [isLoading, setIsLoading] = useState(true);
+
+ const getSecondSwipeStatus = async () => {
+ const { result, error } = await request('getSecondSwipeAction');
+ if (result && result !== 'error') setSecondSwipeAction(result);
+ if ((result && result === 'error') || error) console.error(error);
+ };
+
+ const setSecondSwipeStatus = async (action) => {
+ setIsLoading(true);
+ const { result, error } = await request('setSecondSwipeAction', {
+ action: action
+ });
+
+ if (error || result === 'error') {
+ console.error('An error occurred while setting second swipe status');
+ await getSecondSwipeStatus(); // Revert to previous state
+ } else {
+ setSecondSwipeAction(action);
+ }
+ setIsLoading(false);
+ };
+
+ useEffect(() => {
+ const fetchSecondSwipeStatus = async () => {
+ setIsLoading(true);
+ await getSecondSwipeStatus();
+ setIsLoading(false);
+ };
+ fetchSecondSwipeStatus();
+ }, []);
+
+ const handleChange = (event) => {
+ setSecondSwipeStatus(event.target.value);
+ };
return (
-
+
}
- label={t('settings.secondswipe.restart')}
- disabled={true}
+ value="toggle"
+ control={}
+ label={t('settings.secondswipe.toggle')}
/>
}
- label={t('settings.secondswipe.toggle')}
- disabled={true}
+ value="rewind"
+ control={}
+ label={t('settings.secondswipe.rewind')}
/>
}
- label={t('settings.secondswipe.skip')}
- disabled={true}
+ value="next"
+ control={}
+ label={t('settings.secondswipe.next')}
/>
}
- label={t('settings.secondswipe.ignore')}
- disabled={true}
+ value="None"
+ control={}
+ label={t('settings.secondswipe.none')}
/>
From 9703c99d87e5c50ce3a71eeb19f0b08270d8d64f Mon Sep 17 00:00:00 2001
From: pabera <1260686+pabera@users.noreply.github.com>
Date: Sun, 29 Dec 2024 13:12:50 +0100
Subject: [PATCH 16/16] fix: call the right function name to check on debian
version
---
installation/includes/02_helpers.sh | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/installation/includes/02_helpers.sh b/installation/includes/02_helpers.sh
index 07c0e6851..dfc880754 100644
--- a/installation/includes/02_helpers.sh
+++ b/installation/includes/02_helpers.sh
@@ -84,7 +84,7 @@ is_debian_based() {
fi
}
-get_debian_version_number() {
+_get_debian_version_number() {
if [ "$(is_debian_based)" = true ]; then
local debian_version_number=$( . /etc/os-release; printf '%s\n' "$VERSION_ID"; )
echo "$debian_version_number"