* introduce logcommand.Lager, Whipper(); use argparse for whipper image commands, stub logging * update Lager docstring to mention config.Config() * make incorrect subcommand and --version work on toplevel command * migrate accurip show, expand Lager, do not attempt to return from Lager.__init__. * migrate offset find, add Lager.error * correct offset find drive symlink handling * migrate drive * change Lager.__init__(prog) to arg from kwarg * but actually * remove Whipper.usage * add and use Lager.device_option() context manager * help I married an axe murderer * use unified options namespace for entire command tree * migrate whipper cd without comprehensive config loading * switch to logging module - use logging instead of flog for non-extern modules - use WHIPPER_DEBUG and WHIPPER_LOGFILE env variables * convert self.log calls to logger.debug * convert self.error calls to logger.error * remove log.Loggable, use logger not logging * Logging conversion continues - Convert log.* calls to logger.* - Remove morituri.common.log imports * remove morituri.common.log from tests * remove extern/flog, bare minimum Debug conversion * update README for logging changes * update soxi to use logging * refactor Lager for more declarative subcommands * Refactor Lager.device_option: - inline into __init__ - throw IOError instead of Exception for missing drives - remove CommandError checking in rip/main * rename rip to whipper in rip.main * convert rip.debug commands * Rename logcommand.Lager to command.BaseCommand - remove command.CommandError occurrences - remove python-command external module * remove submodules from README, update rclog formatter * update minor ambiguity in readme for command invocation * update version number to match setup.py * remove gitmodules * update version number in tests as well (boo) * convert logger.error to logger.critical * Change morituri.rip to morituri.command - mv common.command to command.basecommand - move TEMPLATES used only by rip.cd out of rip.common - update entry point for command to command.main * update basecommand documentation * go pyflaking: import fixing * replace self.stdout with sys.stdout * remove BaseCommand.config, alphabetise imports * convert self.stdXXX leftovers * convert last getRootCommand to config.Config * convert last getExceptionMessage's to str * change musicbrainz useragent to whipper
157 lines
5.2 KiB
Python
157 lines
5.2 KiB
Python
# -*- Mode: Python -*-
|
|
# vi:si:et:sw=4:sts=4:ts=4
|
|
|
|
# Morituri - for those about to RIP
|
|
|
|
# Copyright (C) 2009 Thomas Vander Stichele
|
|
|
|
# This file is part of morituri.
|
|
#
|
|
# morituri is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# morituri is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with morituri. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
import os
|
|
import sys
|
|
|
|
from morituri.command.basecommand import BaseCommand
|
|
from morituri.common import accurip, config, program
|
|
from morituri.extern.task import task
|
|
from morituri.image import image
|
|
from morituri.result import result
|
|
|
|
import logging
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
class Retag(BaseCommand):
|
|
summary = "retag image files"
|
|
description = """
|
|
Retags the image from the given .cue files with tags obtained from MusicBrainz.
|
|
"""
|
|
|
|
def add_arguments(self):
|
|
self.parser.add_argument('cuefile', nargs='+', action='store',
|
|
help="cue file to load rip image from")
|
|
self.parser.add_argument(
|
|
'-R', '--release-id',
|
|
action="store", dest="release_id",
|
|
help="MusicBrainz release id to match to (if there are multiple)"
|
|
)
|
|
self.parser.add_argument(
|
|
'-p', '--prompt',
|
|
action="store_true", dest="prompt",
|
|
help="Prompt if there are multiple matching releases"
|
|
)
|
|
self.parser.add_argument(
|
|
'-c', '--country',
|
|
action="store", dest="country",
|
|
help="Filter releases by country"
|
|
)
|
|
|
|
def do(self):
|
|
# here to avoid import gst eating our options
|
|
from morituri.common import encode
|
|
|
|
prog = program.Program(config.Config(), stdout=sys.stdout)
|
|
runner = task.SyncRunner()
|
|
|
|
for arg in self.options.cuefile:
|
|
sys.stdout.write('Retagging image %r\n' % arg)
|
|
arg = arg.decode('utf-8')
|
|
cueImage = image.Image(arg)
|
|
cueImage.setup(runner)
|
|
|
|
mbdiscid = cueImage.table.getMusicBrainzDiscId()
|
|
sys.stdout.write('MusicBrainz disc id is %s\n' % mbdiscid)
|
|
|
|
sys.stdout.write("MusicBrainz lookup URL %s\n" %
|
|
cueImage.table.getMusicBrainzSubmitURL())
|
|
prog.metadata = prog.getMusicBrainz(cueImage.table, mbdiscid,
|
|
release=self.options.release_id,
|
|
country=self.options.country,
|
|
prompt=self.options.prompt)
|
|
|
|
if not prog.metadata:
|
|
print 'Not in MusicBrainz database, skipping'
|
|
continue
|
|
|
|
prog.metadata.discid = mbdiscid
|
|
|
|
# FIXME: this feels like we're poking at internals.
|
|
prog.cuePath = arg
|
|
prog.result = result.RipResult()
|
|
for track in cueImage.table.tracks:
|
|
path = cueImage.getRealPath(track.indexes[1].path)
|
|
|
|
taglist = prog.getTagList(track.number)
|
|
logger.debug(
|
|
'possibly retagging %r from cue path %r with taglist %r',
|
|
path, arg, taglist)
|
|
t = encode.SafeRetagTask(path, taglist)
|
|
runner.run(t)
|
|
path = os.path.basename(path)
|
|
if t.changed:
|
|
print 'Retagged %s' % path
|
|
else:
|
|
print '%s already tagged correctly' % path
|
|
print
|
|
|
|
|
|
class Verify(BaseCommand):
|
|
summary = "verify image"
|
|
description = """
|
|
Verifies the image from the given .cue files against the AccurateRip database.
|
|
"""
|
|
|
|
def add_arguments(self):
|
|
self.parser.add_argument('cuefile', nargs='+', action='store',
|
|
help="cue file to load rip image from")
|
|
|
|
def do(self):
|
|
prog = program.Program(config.Config())
|
|
runner = task.SyncRunner()
|
|
cache = accurip.AccuCache()
|
|
|
|
for arg in self.options.cuefile:
|
|
arg = arg.decode('utf-8')
|
|
cueImage = image.Image(arg)
|
|
cueImage.setup(runner)
|
|
|
|
url = cueImage.table.getAccurateRipURL()
|
|
responses = cache.retrieve(url)
|
|
|
|
# FIXME: this feels like we're poking at internals.
|
|
prog.cuePath = arg
|
|
prog.result = result.RipResult()
|
|
for track in cueImage.table.tracks:
|
|
tr = result.TrackResult()
|
|
tr.number = track.number
|
|
prog.result.tracks.append(tr)
|
|
|
|
prog.verifyImage(runner, responses)
|
|
|
|
print "\n".join(prog.getAccurateRipResults()) + "\n"
|
|
|
|
|
|
class Image(BaseCommand):
|
|
summary = "handle images"
|
|
description = """
|
|
Handle disc images. Disc images are described by a .cue file.
|
|
Disc images can be encoded to another format (for example, to make a
|
|
compressed encoding), retagged and verified.
|
|
"""
|
|
subcommands = {
|
|
'verify': Verify,
|
|
'retag': Retag
|
|
}
|