Skip to content

Commit

Permalink
split pkgs & un-optimize cli file
Browse files Browse the repository at this point in the history
  • Loading branch information
azuline committed Apr 20, 2024
1 parent e0b1492 commit 2d5b82a
Show file tree
Hide file tree
Showing 12 changed files with 333 additions and 175 deletions.
2 changes: 1 addition & 1 deletion flake.nix
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,7 @@
];
dev-cli = pkgs.writeShellScriptBin "rose" ''
cd $ROSE_ROOT
python -m rose "$@"
python -m rose_cli "$@"
'';
in
{
Expand Down
253 changes: 213 additions & 40 deletions rose/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,43 +6,216 @@

import appdirs

logger = logging.getLogger()
logger.setLevel(logging.INFO)

# appdirs by default has Unix log to $XDG_CACHE_HOME, but I'd rather write logs to $XDG_STATE_HOME.
LOG_HOME = Path(appdirs.user_state_dir("rose"))
if appdirs.system == "darwin":
LOG_HOME = Path(appdirs.user_log_dir("rose"))

LOG_HOME.mkdir(parents=True, exist_ok=True)
LOGFILE = LOG_HOME / "rose.log"

# Useful for debugging problems with the virtual FS, since pytest doesn't capture that debug logging
# output.
LOG_EVEN_THOUGH_WERE_IN_TEST = os.environ.get("LOG_TEST", False)

# Add a logging handler for stdout unless we are testing. Pytest
# captures logging output on its own, so by default, we do not attach our own.
if "pytest" not in sys.modules or LOG_EVEN_THOUGH_WERE_IN_TEST: # pragma: no cover
simple_formatter = logging.Formatter(
"[%(asctime)s] %(levelname)s: %(message)s",
datefmt="%H:%M:%S",
)
verbose_formatter = logging.Formatter(
"[ts=%(asctime)s.%(msecs)03d] [pid=%(process)d] [src=%(name)s:%(lineno)s] %(levelname)s: %(message)s",
datefmt="%Y-%m-%d %H:%M:%S",
)

stream_handler = logging.StreamHandler(sys.stderr)
stream_handler.setFormatter(
simple_formatter if not LOG_EVEN_THOUGH_WERE_IN_TEST else verbose_formatter
)
logger.addHandler(stream_handler)

file_handler = logging.handlers.RotatingFileHandler(
LOGFILE,
maxBytes=20 * 1024 * 1024,
backupCount=10,
)
file_handler.setFormatter(verbose_formatter)
logger.addHandler(file_handler)
from rose.audiotags import (
SUPPORTED_AUDIO_EXTENSIONS,
AudioTags,
UnsupportedFiletypeError,
)
from rose.cache import (
STORED_DATA_FILE_REGEX,
CachedRelease,
CachedTrack,
artist_exists,
calculate_release_logtext,
calculate_track_logtext,
collage_exists,
genre_exists,
get_collage,
get_path_of_track_in_playlist,
get_playlist,
get_playlist_cover_path,
get_release,
get_track,
get_tracks_associated_with_release,
label_exists,
list_artists,
list_collages,
list_genres,
list_labels,
list_playlists,
maybe_invalidate_cache_database,
playlist_exists,
update_cache,
update_cache_for_releases,
)
from rose.collages import (
add_release_to_collage,
create_collage,
delete_collage,
dump_all_collages,
dump_collage,
edit_collage_in_editor,
remove_release_from_collage,
rename_collage,
)
from rose.common import (
VERSION,
RoseError,
RoseExpectedError,
sanitize_dirname,
sanitize_filename,
)
from rose.config import Config
from rose.playlists import (
add_track_to_playlist,
create_playlist,
delete_playlist,
delete_playlist_cover_art,
dump_all_playlists,
dump_playlist,
edit_playlist_in_editor,
remove_track_from_playlist,
rename_playlist,
set_playlist_cover_art,
)
from rose.releases import (
create_single_release,
delete_release,
delete_release_cover_art,
dump_all_releases,
dump_release,
edit_release,
run_actions_on_release,
set_release_cover_art,
toggle_release_new,
)
from rose.rule_parser import MetadataAction, MetadataMatcher, MetadataRule
from rose.rules import execute_metadata_rule, execute_stored_metadata_rules
from rose.templates import (
PathTemplate,
eval_release_template,
eval_track_template,
preview_path_templates,
)
from rose.tracks import dump_all_tracks, dump_track, run_actions_on_track
from rose.watcher import start_watchdog

__all__ = [
"AudioTags",
"CachedRelease",
"CachedTrack",
"Config",
"MetadataAction",
"MetadataMatcher",
"MetadataRule",
"PathTemplate",
"RoseError",
"RoseExpectedError",
"STORED_DATA_FILE_REGEX", # TODO: Revise: is_release_directory / is_track_file
"SUPPORTED_AUDIO_EXTENSIONS",
"UnsupportedFiletypeError",
"VERSION",
"add_release_to_collage",
"add_track_to_playlist",
"artist_exists",
"calculate_release_logtext", # TODO: Rename.
"calculate_track_logtext", # TODO: Rename.
"collage_exists",
"create_collage",
"create_playlist",
"create_single_release",
"delete_collage",
"delete_playlist",
"delete_playlist_cover_art",
"delete_release",
"delete_release_cover_art",
"dump_all_collages",
"dump_all_playlists",
"dump_all_releases",
"dump_all_tracks",
"dump_collage",
"dump_playlist",
"dump_release",
"dump_track",
"edit_collage_in_editor", # TODO: Move editor part to CLI, make this file-submissions.
"edit_playlist_in_editor", # TODO: Move editor part to CLI, make this file-submissions.
"edit_release",
"eval_release_template", # TODO: Rename.
"eval_track_template", # TODO: Rename.
"execute_metadata_rule",
"execute_stored_metadata_rules",
"genre_exists",
"get_collage",
"get_path_of_track_in_playlist", # TODO: Redesign.
"get_playlist",
"get_playlist_cover_path", # TODO: Remove.
"get_release",
"get_track",
"get_tracks_associated_with_release", # TODO: Rename: `get_tracks_of_release` / `dump_release(with_tracks=tracks)`
"label_exists",
"list_artists",
"list_collages",
"list_genres",
"list_labels",
"list_playlists",
"maybe_invalidate_cache_database",
"playlist_exists",
"preview_path_templates",
"remove_release_from_collage",
"remove_track_from_playlist",
"rename_collage",
"rename_playlist",
"run_actions_on_release",
"run_actions_on_track",
"sanitize_dirname",
"sanitize_filename",
"set_playlist_cover_art",
"set_release_cover_art",
"start_watchdog",
"toggle_release_new",
"update_cache",
"update_cache_for_releases",
]

__logging_initialized = False


def initialize_logging() -> None:
global __logging_initialized
if __logging_initialized:
return
__logging_initialized = True

logger = logging.getLogger()
logger.setLevel(logging.INFO)

# appdirs by default has Unix log to $XDG_CACHE_HOME, but I'd rather write logs to $XDG_STATE_HOME.
log_home = Path(appdirs.user_state_dir("rose"))
if appdirs.system == "darwin":
log_home = Path(appdirs.user_log_dir("rose"))

log_home.mkdir(parents=True, exist_ok=True)
log_file = log_home / "rose.log"

# Useful for debugging problems with the virtual FS, since pytest doesn't capture that debug logging
# output.
log_despite_testing = os.environ.get("LOG_TEST", False)

# Add a logging handler for stdout unless we are testing. Pytest
# captures logging output on its own, so by default, we do not attach our own.
if "pytest" not in sys.modules or log_despite_testing: # pragma: no cover
simple_formatter = logging.Formatter(
"[%(asctime)s] %(levelname)s: %(message)s",
datefmt="%H:%M:%S",
)
verbose_formatter = logging.Formatter(
"[ts=%(asctime)s.%(msecs)03d] [pid=%(process)d] [src=%(name)s:%(lineno)s] %(levelname)s: %(message)s",
datefmt="%Y-%m-%d %H:%M:%S",
)

stream_handler = logging.StreamHandler(sys.stderr)
stream_handler.setFormatter(
simple_formatter if not log_despite_testing else verbose_formatter
)
logger.addHandler(stream_handler)

file_handler = logging.handlers.RotatingFileHandler(
log_file,
maxBytes=20 * 1024 * 1024,
backupCount=10,
)
file_handler.setFormatter(verbose_formatter)
logger.addHandler(file_handler)


initialize_logging()
9 changes: 0 additions & 9 deletions rose/common.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,6 @@
import hashlib
import os.path
import re
import uuid
from collections.abc import Iterator
from pathlib import Path
from typing import Any, TypeVar
Expand Down Expand Up @@ -72,14 +71,6 @@ def items(self) -> Iterator[tuple[str, list[Artist]]]:
yield "djmixer", self.djmixer


def valid_uuid(x: str) -> bool:
try:
uuid.UUID(x)
return True
except ValueError:
return False


def uniq(xs: list[T]) -> list[T]:
rv: list[T] = []
seen: set[T] = set()
Expand Down
7 changes: 6 additions & 1 deletion rose/config_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,12 @@
import click
import pytest

from rose.config import Config, ConfigNotFoundError, InvalidConfigValueError, MissingConfigKeyError
from rose.config import (
Config,
ConfigNotFoundError,
InvalidConfigValueError,
MissingConfigKeyError,
)
from rose.rule_parser import (
MatcherPattern,
MetadataAction,
Expand Down
3 changes: 3 additions & 0 deletions rose_cli/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
from rose import initialize_logging

initialize_logging()
7 changes: 4 additions & 3 deletions rose/__main__.py → rose_cli/__main__.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,14 +2,15 @@

import click

from rose.cli import cli
from rose.common import RoseExpectedError
from rose_cli.cli import CliExpectedError, cli


def main() -> None:
from rose import RoseExpectedError

try:
cli()
except RoseExpectedError as e:
except (RoseExpectedError, CliExpectedError) as e:
click.secho(f"{e.__class__.__module__}.{e.__class__.__name__}: ", fg="red", nl=False)
click.secho(str(e))
sys.exit(1)
Expand Down
Loading

0 comments on commit 2d5b82a

Please sign in to comment.