beets/beetsplug/missing.py
Quentin Young 339a1ef671 beetsplug: add '-a' to show missing albums to 'missing' plugin
Passing -a to 'beet missing' shows albums missing by all artists
in the library.

Signed-off-by: Quentin Young <qlyoung@qlyoung.net>
2017-03-18 17:52:01 -04:00

223 lines
7.9 KiB
Python

# -*- coding: utf-8 -*-
# This file is part of beets.
# Copyright 2016, Pedro Silva.
# Copyright 2017, Quentin Young.
#
# Permission is hereby granted, free of charge, to any person obtaining
# a copy of this software and associated documentation files (the
# "Software"), to deal in the Software without restriction, including
# without limitation the rights to use, copy, modify, merge, publish,
# distribute, sublicense, and/or sell copies of the Software, and to
# permit persons to whom the Software is furnished to do so, subject to
# the following conditions:
#
# The above copyright notice and this permission notice shall be
# included in all copies or substantial portions of the Software.
"""List missing tracks.
"""
from __future__ import division, absolute_import, print_function
import time
import beets
import musicbrainzngs as m
import sys
from musicbrainzngs.musicbrainz import MusicBrainzError
from beets.autotag import hooks
from beets.library import Item
from beets.plugins import BeetsPlugin
from beets.ui import decargs, print_, Subcommand
from beets import config
from beets.dbcore import types
def _missing_count(album):
"""Return number of missing items in `album`.
"""
return (album.albumtotal or 0) - len(album.items())
def _item(track_info, album_info, album_id):
"""Build and return `item` from `track_info` and `album info`
objects. `item` is missing what fields cannot be obtained from
MusicBrainz alone (encoder, rg_track_gain, rg_track_peak,
rg_album_gain, rg_album_peak, original_year, original_month,
original_day, length, bitrate, format, samplerate, bitdepth,
channels, mtime.)
"""
t = track_info
a = album_info
return Item(**{
'album_id': album_id,
'album': a.album,
'albumartist': a.artist,
'albumartist_credit': a.artist_credit,
'albumartist_sort': a.artist_sort,
'albumdisambig': a.albumdisambig,
'albumstatus': a.albumstatus,
'albumtype': a.albumtype,
'artist': t.artist,
'artist_credit': t.artist_credit,
'artist_sort': t.artist_sort,
'asin': a.asin,
'catalognum': a.catalognum,
'comp': a.va,
'country': a.country,
'day': a.day,
'disc': t.medium,
'disctitle': t.disctitle,
'disctotal': a.mediums,
'label': a.label,
'language': a.language,
'length': t.length,
'mb_albumid': a.album_id,
'mb_artistid': t.artist_id,
'mb_releasegroupid': a.releasegroup_id,
'mb_trackid': t.track_id,
'media': t.media,
'month': a.month,
'script': a.script,
'title': t.title,
'track': t.index,
'tracktotal': len(a.tracks),
'year': a.year,
})
class MissingPlugin(BeetsPlugin):
"""List missing tracks
"""
album_types = {
'missing': types.INTEGER,
}
def __init__(self):
super(MissingPlugin, self).__init__()
self.config.add({
'count': False,
'total': False,
'albums': False,
})
self.album_template_fields['missing'] = _missing_count
self._command = Subcommand('missing',
help=__doc__,
aliases=['miss'])
self._command.parser.add_option(
u'-c', u'--count', dest='count', action='store_true',
help=u'count missing tracks per album')
self._command.parser.add_option(
u'-t', u'--total', dest='total', action='store_true',
help=u'count total of missing tracks')
self._command.parser.add_option(
u'-a', u'--albums', dest='albums', action='store_true',
help=u'show missing albums for artist instead of tracks')
self._command.parser.add_format_option()
def commands(self):
def _miss(lib, opts, args):
self.config.set_args(opts)
albms = self.config['albums'].get()
helper = self._missing_albums if albms else self._missing_tracks
helper(lib, args)
self._command.func = _miss
return [self._command]
def _missing_tracks(self, lib, args):
"""Logic for determining missing tracks from each album in the library
(the usual case)
"""
albums = lib.albums(decargs(args))
count = self.config['count'].get()
total = self.config['total'].get()
fmt = config['format_album' if count else 'format_item'].get()
if total:
print(sum([_missing_count(a) for a in albums]))
return
# Default format string for count mode.
if count:
fmt += ': $missing'
for album in albums:
if count:
if _missing_count(album):
print_(format(album, fmt))
else:
for item in self._missing(album):
print_(format(item, fmt))
def _missing_albums(self, lib, args):
"""Logic for determining missing albums from each artist in the library
(-a case)
"""
albums = lib.albums(decargs(args))
# build dict mapping artist to list of their albums in library
albums_by_artist = {}
for alb in albums:
artist = (alb['albumartist'], alb['mb_albumartistid'])
artistalbs = albums_by_artist.setdefault(artist, [])
artistalbs.append(alb)
albums_by_artist[artist] = artistalbs
# build dict mapping artist to list of all albums
m.set_useragent('beets', beets.__version__, 'http://beets.io/')
for artist, albums in albums_by_artist.items():
if artist[1] is None or artist[1] == "":
print_(u"[!] No musicbrainz ID for artist {}, skipping."
.format(artist[0]), stream=sys.stderr)
continue
try:
releases = m.browse_releases(artist=artist[1])['release-list']
except MusicBrainzError as err:
print_(u"[!] Couldn't fetch info for artist '{}' ({}) - '{}'"
.format(artist[0], artist[1], err), stream=sys.stderr)
continue
missing = []
present = []
for release in releases:
missing.append(release)
for alb in albums:
# compare by title, don't care about specific releases
if alb['album'].lower() == release['title'].lower():
missing.remove(release)
present.append(release)
break
missing_titles = {release['title'] for release in missing}
prefix_miss = "{} - ".format(artist[0])
for release in missing_titles:
print_(u"{}{}".format(prefix_miss, release))
# respect musicbrainz rate limits
time.sleep(1)
def _missing(self, album):
"""Query MusicBrainz to determine items missing from `album`.
"""
item_mbids = [x.mb_trackid for x in album.items()]
if len([i for i in album.items()]) < album.albumtotal:
# fetch missing items
# TODO: Implement caching that without breaking other stuff
album_info = hooks.album_for_mbid(album.mb_albumid)
for track_info in getattr(album_info, 'tracks', []):
if track_info.track_id not in item_mbids:
item = _item(track_info, album_info, album.id)
self._log.debug(u'track {0} in album {1}',
track_info.track_id, album_info.album_id)
yield item