dnd-music-console/croaker/playlist.py

100 lines
2.8 KiB
Python
Raw Normal View History

2024-03-01 01:00:17 -08:00
import logging
import os
from dataclasses import dataclass
from functools import cached_property
from itertools import chain
from pathlib import Path
from random import shuffle
from typing import List
import croaker.path
2024-03-05 22:21:56 -08:00
logger = logging.getLogger('playlist')
2024-03-01 01:00:17 -08:00
playlists = {}
NowPlaying = None
2024-03-01 01:00:17 -08:00
def _stripped(name):
name.replace('"', "")
name.replace("'", "")
return name
@dataclass
class Playlist:
name: str
position: int = 0
2024-03-01 01:00:17 -08:00
theme: Path = Path("_theme.mp3")
@property
def current(self):
return self.tracks[self.position]
2024-03-01 01:00:17 -08:00
@cached_property
def path(self):
return croaker.path.playlist_root() / Path(self.name)
@cached_property
def tracks(self):
if not self.path.exists():
raise RuntimeError(f"Playlist {self.name} not found at {self.path}.")
entries = []
theme = self.path / self.theme
if theme.exists():
entries.append(theme)
2024-03-01 01:00:17 -08:00
files = [e for e in self.get_audio_files() if e.name != "_theme.mp3"]
if files:
shuffle(files)
entries += files
return entries
def skip(self):
logging.debug(f"Skipping from {self.position} on {self.name}")
if self.position == len(self.tracks) - 1:
self.position = 0
else:
self.position += 1
2024-03-01 01:00:17 -08:00
def get_audio_files(self, path: Path = None):
if not path:
path = self.path
logging.debug(f"Getting files matching {os.environ['MEDIA_GLOB']} from {path}")
pats = os.environ["MEDIA_GLOB"].split(",")
return chain(*[list(path.glob(pat)) for pat in pats])
def _add_track(self, target: Path, source: Path, make_theme: bool = False):
if source.is_dir():
for file in self.get_audio_files(source):
self._add_track(self.path / _stripped(file.name), file)
return
if target.exists():
if not target.is_symlink():
logging.warning(f"{target}: target already exists and is not a symlink; skipping.")
return
target.unlink()
target.symlink_to(source)
def add(self, tracks: List[Path], make_theme: bool = False):
self.path.mkdir(parents=True, exist_ok=True)
if make_theme:
target = self.path / "_theme.mp3"
source = tracks.pop(0)
self._add_track(target, source, make_theme=True)
for track in tracks:
self._add_track(target=self.path / _stripped(track.name), source=track)
return sorted(self.get_audio_files())
def __repr__(self):
lines = [f"Playlist {self.name}"]
lines += [f" * {track}" for track in self.tracks]
return "\n".join(lines)
def load_playlist(name: str):
if name not in playlists:
playlists[name] = Playlist(name=name)
return playlists[name]