Skip to content

Commit

Permalink
Merge pull request #95 from geigerzaehler/stricter-type-checking
Browse files Browse the repository at this point in the history
More type hints and stricter checking
  • Loading branch information
geigerzaehler authored Jul 11, 2024
2 parents 03ece2e + 4934186 commit 0e6dd63
Show file tree
Hide file tree
Showing 5 changed files with 86 additions and 51 deletions.
81 changes: 50 additions & 31 deletions beetsplug/alternatives.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,31 +11,33 @@
# all copies or substantial portions of the Software.

import argparse
import logging
import os.path
import threading
import traceback
from concurrent import futures
from enum import Enum
from typing import Iterator, List, Optional, Tuple, cast
from typing import Callable, Iterable, Iterator, Optional, Sequence, Set, Tuple, cast

import beets
import confuse
from beets import art, util
from beets.library import Item, Library, parse_query_string
from beets.plugins import BeetsPlugin
from beets.ui import Subcommand, UserError, decargs, get_path_formats, input_yn, print_
from beets.util import FilesystemError, bytestring_path, displayable_path, syspath
from typing_extensions import override
from typing_extensions import Never, override

import beetsplug.convert as convert


def _remove(path, soft=True):
def _remove(path_: bytes, soft: bool = True):
"""Remove the file. If `soft`, then no error will be raised if the
file does not exist.
In contrast to beets' util.remove, this uses lexists such that it can
actually remove symlink links.
"""
path = syspath(path)
path = syspath(path_)
if soft and not os.path.lexists(path):
return
try:
Expand All @@ -48,7 +50,7 @@ class AlternativesPlugin(BeetsPlugin):
def __init__(self):
super().__init__()

def commands(self):
def commands(self): # pyright: ignore[reportIncompatibleMethodOverride]
return [AlternativesCommand(self)]

def update(self, lib: Library, options: argparse.Namespace):
Expand All @@ -67,10 +69,10 @@ def update(self, lib: Library, options: argparse.Namespace):
) from e
alt.update(create=options.create)

def list_tracks(self, lib, options):
def list_tracks(self, lib: Library, options: argparse.Namespace):
if options.format is not None:
(fmt,) = decargs([options.format])
beets.config[Item._format_config_key].set(fmt)
beets.config[Item._format_config_key].set(fmt) # pyright: ignore[reportPrivateUsage]

alt = self.alternative(options.name, lib)

Expand Down Expand Up @@ -100,7 +102,7 @@ class AlternativesCommand(Subcommand):
name = "alt"
help = "manage alternative files"

def __init__(self, plugin):
def __init__(self, plugin: AlternativesPlugin):
parser = ArgumentParser()
subparsers = parser.add_subparsers(prog=parser.prog + " alt")
subparsers.required = True
Expand Down Expand Up @@ -147,10 +149,10 @@ def __init__(self, plugin):

super().__init__(self.name, parser, self.help)

def func(self, lib, opts, _):
def func(self, lib: Library, opts: argparse.Namespace, _): # pyright: ignore[reportIncompatibleMethodOverride]
opts.func(lib, opts)

def parse_args(self, args):
def parse_args(self, args: Sequence[str]): # pyright: ignore
return self.parser.parse_args(args), []


Expand All @@ -160,7 +162,7 @@ class ArgumentParser(argparse.ArgumentParser):
`_get_all_options()` to generate shell completion.
"""

def _get_all_options(self):
def _get_all_options(self) -> Sequence[Never]:
# FIXME return options like ``OptionParser._get_all_options``.
return []

Expand All @@ -174,15 +176,17 @@ class Action(Enum):


class External:
def __init__(self, log, name, lib, config):
def __init__(
self, log: logging.Logger, name: str, lib: Library, config: confuse.ConfigView
):
self._log = log
self.name = name
self.lib = lib
self.path_key = f"alt.{name}"
self.max_workers = int(str(beets.config["convert"]["threads"]))
self.parse_config(config)

def parse_config(self, config):
def parse_config(self, config: confuse.ConfigView):
if "paths" in config:
path_config = config["paths"]
else:
Expand All @@ -191,18 +195,21 @@ def parse_config(self, config):
query = config["query"].as_str()
self.query, _ = parse_query_string(query, Item)

self.removable = config.get(dict).get("removable", True)
self.removable = config.get(dict).get("removable", True) # type: ignore

if "directory" in config:
dir = config["directory"].as_str()
assert isinstance(dir, str)
else:
dir = self.name
dir = bytestring_path(dir)
if not os.path.isabs(syspath(dir)):
dir = os.path.join(self.lib.directory, dir)
self.directory = dir

def item_change_actions(self, item: Item, path: bytes, dest: bytes) -> List[Action]:
def item_change_actions(
self, item: Item, path: bytes, dest: bytes
) -> Sequence[Action]:
"""Returns the necessary actions for items that were previously in the
external collection, but might require metadata updates.
"""
Expand All @@ -216,16 +223,17 @@ def item_change_actions(self, item: Item, path: bytes, dest: bytes) -> List[Acti
actions.append(Action.WRITE)
album = item.get_album()

if album and (
album.artpath
if (
album
and album.artpath
and os.path.isfile(syspath(album.artpath))
and (item_mtime_alt < os.path.getmtime(syspath(album.artpath)))
):
actions.append(Action.SYNC_ART)

return actions

def _matched_item_action(self, item: Item) -> List[Action]:
def _matched_item_action(self, item: Item) -> Sequence[Action]:
path = self._get_stored_path(item)
if path and os.path.lexists(syspath(path)):
dest = self.destination(item)
Expand All @@ -239,7 +247,7 @@ def _matched_item_action(self, item: Item) -> List[Action]:
else:
return [Action.ADD]

def _items_actions(self) -> Iterator[Tuple[Item, List[Action]]]:
def _items_actions(self) -> Iterator[Tuple[Item, Sequence[Action]]]:
matched_ids = set()
for album in self.lib.albums():
if self.query.match(album):
Expand Down Expand Up @@ -299,7 +307,7 @@ def update(self, create: Optional[bool] = None):
self._sync_art(item, path)
elif action == Action.ADD:
print_(f"+{displayable_path(dest)}")
converter.submit(item)
converter.run(item)
elif action == Action.REMOVE:
assert path is not None # action guarantees that `path` is not none
print_(f"-{displayable_path(path)}")
Expand Down Expand Up @@ -344,7 +352,7 @@ def _remove_file(self, item: Item):
del item[self.path_key]

def _converter(self) -> "Worker":
def _convert(item):
def _convert(item: Item):
dest = self.destination(item)
util.mkdirall(dest)
util.copy(item.path, dest, replace=True)
Expand All @@ -363,7 +371,14 @@ def _sync_art(self, item: Item, path: bytes):


class ExternalConvert(External):
def __init__(self, log, name, formats, lib, config):
def __init__(
self,
log: logging.Logger,
name: str,
formats: Iterable[str],
lib: Library,
config: confuse.ConfigView,
):
super().__init__(log, name, lib, config)
convert_plugin = convert.ConvertPlugin()
self._encode = convert_plugin.encode
Expand All @@ -376,7 +391,7 @@ def __init__(self, log, name, formats, lib, config):
def _converter(self) -> "Worker":
fs_lock = threading.Lock()

def _convert(item):
def _convert(item: Item):
dest = self.destination(item)
with fs_lock:
util.mkdirall(dest)
Expand All @@ -402,7 +417,7 @@ def destination(self, item: Item) -> bytes:
else:
return dest

def _should_transcode(self, item):
def _should_transcode(self, item: Item):
return item.format.lower() not in self.formats


Expand All @@ -413,7 +428,7 @@ class SymlinkType(Enum):

class SymlinkView(External):
@override
def parse_config(self, config):
def parse_config(self, config: confuse.ConfigView):
if "query" not in config:
config["query"] = "" # This is a TrueQuery()
if "link_type" not in config:
Expand All @@ -428,7 +443,9 @@ def parse_config(self, config):
super().parse_config(config)

@override
def item_change_actions(self, item: Item, path: bytes, dest: bytes):
def item_change_actions(
self, item: Item, path: bytes, dest: bytes
) -> Sequence[Action]:
"""Returns the necessary actions for items that were previously in the
external collection, but might require metadata updates.
"""
Expand All @@ -447,7 +464,7 @@ def item_change_actions(self, item: Item, path: bytes, dest: bytes):
return [Action.MOVE]

@override
def update(self, create=None):
def update(self, create: Optional[bool] = None):
for item, actions in self._items_actions():
dest = self.destination(item)
path = self._get_stored_path(item)
Expand Down Expand Up @@ -486,13 +503,15 @@ def _sync_art(self, item: Item, path: bytes):


class Worker(futures.ThreadPoolExecutor):
def __init__(self, fn, max_workers: Optional[int]):
def __init__(
self, fn: Callable[[Item], Tuple[Item, bytes]], max_workers: Optional[int]
):
super().__init__(max_workers)
self._tasks = set()
self._tasks: Set[futures.Future[Tuple[Item, bytes]]] = set()
self._fn = fn

def submit(self, *args, **kwargs):
fut = super().submit(self._fn, *args, **kwargs)
def run(self, item: Item):
fut = self.submit(self._fn, item)
self._tasks.add(fut)
return fut

Expand Down
12 changes: 9 additions & 3 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,13 @@ filterwarnings = [
]

[tool.pyright]
typeCheckingMode = "basic"
typeCheckingMode = "strict"
reportMissingTypeStubs = "none"
# We haven’t made an effort to fix the following issues. Most are caused by
# missing type annotations from beets.
reportUnknownMemberType = "none"
reportUnknownArgumentType = "none"
reportUnknownVariableType = "none"

[tool.ruff]
target-version = "py38"
Expand All @@ -61,9 +67,9 @@ preview = true
extend-select = [
"I", # Sort imports
"C", # Pyflakes conventions
"UP", # Enforce modern Python syntx
"PIE", # Misc. lints
"FURB", # Elso enforce more modern Python syntax
"UP", # Enforce modern Python syntax
"FURB", # Also enforce more modern Python syntax
"PT", # Pytest style
"B", # Bugbear, avoid common sources of bugs
"SIM", # Simplify
Expand Down
3 changes: 3 additions & 0 deletions test/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
import pytest

pytest.register_assert_rewrite("test.helper")
1 change: 0 additions & 1 deletion test/cli_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@
from time import sleep

import pytest
from beets import util
from beets.library import Item
from beets.ui import UserError
from confuse import ConfigValueError
Expand Down
Loading

0 comments on commit 0e6dd63

Please sign in to comment.