Skip to content

Commit

Permalink
Create BackendType metaclass for easy access to backends
Browse files Browse the repository at this point in the history
  • Loading branch information
snejus committed Sep 28, 2024
1 parent 450a55d commit 2cc8059
Showing 1 changed file with 58 additions and 53 deletions.
111 changes: 58 additions & 53 deletions beetsplug/lyrics.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,8 +25,9 @@
import unicodedata
import urllib
import warnings
from functools import cached_property
from html import unescape
from typing import Any
from typing import Any, Iterator

import requests
from typing_extensions import TypedDict
Expand Down Expand Up @@ -209,7 +210,36 @@ def try_parse_html(html, **kwargs):
return None


class Backend:
class BackendType(type):
"""Metaclass for the :class:`Backend` class.
It keeps track of defined subclasses and provides access to them through
the base class:
>>> Backend["genius"] # beetsplug.lyrics.Genius
>>> list(Backend) # ["lrclib", "musixmatch", "genius", "tekstowo", "google"]
"""

_registry: dict[str, BackendType] = {}
REQUIRES_BS: bool

def __new__(cls, name: str, bases: tuple[type, ...], attrs) -> BackendType:
"""Create a new instance of the class and add it to the registry."""
new_class = super().__new__(cls, name, bases, attrs)
if bases:
cls._registry[name.lower()] = new_class
return new_class

@classmethod
def __getitem__(cls, key: str) -> BackendType:
return cls._registry[key]

@classmethod
def __iter__(cls) -> Iterator[str]:
return iter(cls._registry)


class Backend(metaclass=BackendType):
REQUIRES_BS = False

def __init__(self, config, log):
Expand Down Expand Up @@ -817,14 +847,29 @@ def fetch(self, artist, title, album=None, length=None):


class LyricsPlugin(plugins.BeetsPlugin):
SOURCES = ["lrclib", "google", "musixmatch", "genius", "tekstowo"]
SOURCE_BACKENDS = {
"google": Google,
"musixmatch": MusiXmatch,
"genius": Genius,
"tekstowo": Tekstowo,
"lrclib": LRCLib,
}
@cached_property
def backends(self) -> dict[str, Backend]:
user_sources = self.config["sources"].get()
chosen = plugins.sanitize_choices(user_sources, Backend)

disabled = set()
if not HAS_BEAUTIFUL_SOUP:
disabled |= {n for n in chosen if Backend[n].REQUIRES_BS}
if disabled:
self._log.debug(
"Disabling {} sources: missing beautifulsoup4 module",
disabled,
)

elif "google" in chosen and not self.config["google_API_key"].get():
self._log.debug("Disabling Google source: no API key configured.")
disabled.add("google")

return {
s: Backend[s](self.config, self._log)
for s in chosen
if s not in disabled
}

def __init__(self):
super().__init__()
Expand All @@ -847,7 +892,7 @@ def __init__(self):
"synced": False,
# Musixmatch is disabled by default as they are currently blocking
# requests with the beets user agent.
"sources": [s for s in self.SOURCES if s != "musixmatch"],
"sources": [s for s in Backend if s != "musixmatch"],
"dist_thresh": 0.1,
}
)
Expand All @@ -865,25 +910,6 @@ def __init__(self):
# open yet.
self.rest = None

available_sources = list(self.SOURCES)
sources = plugins.sanitize_choices(
self.config["sources"].as_str_seq(), available_sources
)

if not HAS_BEAUTIFUL_SOUP:
sources = self.sanitize_bs_sources(sources)

if "google" in sources:
if not self.config["google_API_key"].get():
# We log a *debug* message here because the default
# configuration includes `google`. This way, the source
# is silent by default but can be enabled just by
# setting an API key.
self._log.debug(
"Disabling google source: " "no API key configured."
)
sources.remove("google")

self.config["bing_lang_from"] = [
x.lower() for x in self.config["bing_lang_from"].as_str_seq()
]
Expand All @@ -896,25 +922,6 @@ def __init__(self):
"documentation for further details."
)

self.backends = [
self.SOURCE_BACKENDS[source](self.config, self._log)
for source in sources
]

def sanitize_bs_sources(self, sources):
enabled_sources = []
for source in sources:
if self.SOURCE_BACKENDS[source].REQUIRES_BS:
self._log.debug(
"To use the %s lyrics source, you must "
"install the beautifulsoup4 module. See "
"the documentation for further details." % source
)
else:
enabled_sources.append(source)

return enabled_sources

def get_bing_access_token(self):
params = {
"client_id": "beets",
Expand Down Expand Up @@ -1131,12 +1138,10 @@ def get_lyrics(self, artist, title, album=None, length=None):
"""Fetch lyrics, trying each source in turn. Return a string or
None if no lyrics were found.
"""
for backend in self.backends:
for name, backend in self.backends.items():
lyrics = backend.fetch(artist, title, album=album, length=length)
if lyrics:
self._log.debug(
"got lyrics from backend: {0}", backend.__class__.__name__
)
self._log.debug("got lyrics from backend: {0}", name)
return _scrape_strip_cruft(lyrics, True)

def append_translation(self, text, to_lang):
Expand Down

0 comments on commit 2cc8059

Please sign in to comment.