adding cleanup of stale track entries

This commit is contained in:
evilchili 2022-12-10 10:14:06 -08:00
parent 228d44ce98
commit 1a2506742f
9 changed files with 64 additions and 28 deletions

View File

@ -106,6 +106,7 @@ def scan(
initialize()
with database_manager() as manager:
scanner = media_scanner(root=root, db=manager.session)
scanner.cleanup()
count = scanner.scan()
logging.info(f"Imported {count} new tracks.")

View File

@ -5,7 +5,7 @@ import music_tag
from pathlib import Path
from typing import Callable, Union, Iterable
from sqlalchemy import func
from sqlalchemy import func, delete
import groove.db
import groove.path
@ -40,10 +40,31 @@ class MediaScanner:
async def _do_import():
logging.debug("Scanning filesystem (this may take a minute)...")
for path in sources:
if path.exists() and not path.is_dir():
asyncio.create_task(self._import_one_track(path))
asyncio.run(_do_import())
self.db.commit()
def cleanup(self) -> int:
"""
Check for the existence of every track in the databse.
"""
async def _del(track):
path = self.root / Path(track.relpath)
if path.exists():
return
logging.info(f"Deleting missing track {track.relpath}")
self.db.execute(
delete(groove.db.track).where(groove.db.track.c.id == track.id)
)
async def _do_cleanup():
logging.debug("Locating stale track definitions in the database...")
for track in self.db.query(groove.db.track).all():
asyncio.create_task(_del(track))
asyncio.run(_do_cleanup())
self.db.commit()
def _get_tags(self, path): # pragma: no cover
tags = music_tag.load_file(path)
return {

View File

@ -66,7 +66,7 @@ class Playlist:
@property
def info(self):
count = len(self.entries)
return f"{self.name}: {self.url} [{count} tracks]\n{self.description}\n"
return f"{self.name}: {self.url} [{count} tracks]\n{self.description}"
@property
def url(self) -> str:
@ -133,7 +133,7 @@ class Playlist:
@property
def as_string(self) -> str:
text = self.info
text = self.info + self.description
for (tracknum, entry) in enumerate(self.entries):
text += f" - {tracknum+1} {entry.artist} - {entry.title}\n"
return text

View File

@ -14,6 +14,11 @@ class BasePrompt(Completer):
self._values = []
self._parent = parent
self._manager = manager
self._commands = {'help': self.help}
@property
def commands(self):
return self._commands
@property
def usage(self):
@ -40,7 +45,7 @@ class BasePrompt(Completer):
@property
def values(self):
return self._values
return [k for k in self.commands.keys() if not k.startswith('_')]
def get_completions(self, document, complete_event):
word = document.get_word_before_cursor()
@ -59,7 +64,7 @@ class BasePrompt(Completer):
def start(self, cmd=''):
while True:
if not cmd:
cmd = prompt(f'{self.prompt} ', completer=self)
cmd = prompt(f'{self.prompt} ', completer=self, complete_while_typing=True)
if not cmd:
return
cmd, *parts = cmd.split()
@ -67,6 +72,13 @@ class BasePrompt(Completer):
return
cmd = ''
def help(self, parts):
if not parts:
print(self.__doc__)
else:
print(getattr(self, parts[0]).__doc__)
return True
def default_completer(self, document, complete_event):
raise NotImplementedError()

View File

@ -54,8 +54,8 @@ class CommandPrompt(BasePrompt):
session=self.manager.session,
create_ok=True
)
res = self.commands['_playlist'].start()
return True and res
self.commands['_playlist'].start()
return True
def start(): # pragma: no cover

View File

@ -8,21 +8,14 @@ from groove import db
class _playlist(BasePrompt):
def __init__(self, parent, manager=None):
super().__init__(manager=manager, parent=parent)
self._parent = parent
self._prompt = ''
self._commands = None
"""
PLAYLIST
"""
@property
def prompt(self):
return f"{self.parent.playlist}\n{self.parent.playlist.slug}> "
@property
def values(self):
return self.commands.keys()
@property
def commands(self):
if not self._commands:
@ -31,9 +24,20 @@ class _playlist(BasePrompt):
'delete': self.delete,
'add': self.add,
'edit': self.edit,
'help': self.help
}
return self._commands
def _add_track(self, text):
sess = self.parent.manager.session
try:
track = sess.query(db.track).filter(db.track.c.relpath == text).one()
self.parent.playlist.create_entries([track])
except NoResultFound:
print("No match for '{text}'")
return
return text
def process(self, cmd, *parts):
res = True
if cmd in self.commands:
@ -66,16 +70,6 @@ class _playlist(BasePrompt):
return True
self._add_track(text)
def _add_track(self, text):
sess = self.parent.manager.session
try:
track = sess.query(db.track).filter(db.track.c.relpath == text).one()
self.parent.playlist.create_entries([track])
except NoResultFound:
print("No match for '{text}'")
return
return text
def delete(self, parts):
res = prompt(
'Type DELETE to permanently delete the playlist '

0
test/fixtures/themes/alt_theme/static vendored Normal file
View File

View File

@ -38,6 +38,10 @@ def test_scanner(monkeypatch, in_memory_db, media):
# replace the filesystem glob with the test fixture generator
monkeypatch.setattr(scanner.MediaScanner, 'find_sources', MagicMock(return_value=media()))
# pretend things exist
monkeypatch.setattr(scanner.Path, 'exists', MagicMock(return_value=True))
monkeypatch.setattr(scanner.Path, 'is_dir', MagicMock(return_value=False))
def mock_loader(path):
return {
'artist': 'foo',

View File

@ -16,6 +16,10 @@ def response_factory(responses):
return MagicMock(side_effect=responses + ([''] * 10))
def test_commands(cmd_prompt):
assert cmd_prompt.commands.keys() == cmd_prompt.commands.keys()
@pytest.mark.parametrize('inputs, expected', [
(['stats'], 'Database contains 4 playlists'), # match the db fixture
])