X-Git-Url: https://git.draconx.ca/gitweb/scripts.git/blobdiff_plain/4ff20e03039c9e4abda2adf838b3ed323c766532..HEAD:/caa-fetcher.py diff --git a/caa-fetcher.py b/caa-fetcher.py index 5c23107..de19aeb 100755 --- a/caa-fetcher.py +++ b/caa-fetcher.py @@ -1,6 +1,6 @@ #!/usr/bin/env python # -# Copyright © 2018 Nick Bowler +# Copyright © 2018-2019 Nick Bowler # # Tool to fetch album art for a release from the Cover Art Archive. # @@ -12,23 +12,51 @@ from pathlib import Path import musicbrainzngs import argparse import tempfile +import urllib import sys import os +import re + +umask = os.umask(0) +os.umask(umask) progname = "caa-fetcher" -version = "0" +version = "1" musicbrainzngs.set_useragent(progname, version) parser = argparse.ArgumentParser( description='Download album artwork from the Cover Art Archive' ) +def errmsg(msg, prog=parser.prog): + print("%s: %s" % (prog, msg), file=sys.stderr) + +# NamedTemporaryFile workalike which allows control of the file mode... +def open_tmp(prefix="tmp", suffix="", dir=".", mode=0o600): + names = tempfile._get_candidate_names() + for seq in range(100): + name = os.path.join(dir, "%s%s%s" % (prefix, next(names), suffix)) + try: + f = open(name, "x+", mode) + except FileExistsError: + continue + return tempfile._TemporaryFileWrapper(f, name) + return None + +# Given an arbitrary string, return the first substring that looks like an +# mbid, or None if no such substring is found. +def extract_mbid(arg): + xdigit = r'[0-9abcdefABCDEF]' + m = re.search(r'{0}{{8}}(-{0}{{4}}){{3}}-{0}{{12}}'.format(xdigit), arg) + if m: + return m.group() + class VersionAction(argparse.Action): def __init__(self, **kw): super().__init__(nargs=0, help="show a version message and exit", **kw) def __call__(self, parser, namespace, values, option_string=None): print("%s %s" % (progname, version)) - print('''Copyright © 2018 Nick Bowler + print('''Copyright © 2019 Nick Bowler License WTFPL2: Do What The Fuck You Want To Public License, version 2. This is free software: you are free to do what the fuck you want to. There is NO WARRANTY, to the extent permitted by law.''') @@ -41,10 +69,22 @@ parser.add_argument('-o', '--output-directory', metavar='DIR', parser.add_argument('-r', '--release-mbid', metavar='MBID', required=True) args = parser.parse_args() +release_mbid = extract_mbid(args.release_mbid) +if release_mbid is None: + parser.error("invalid release MBID: %s" % (args.release_mbid)) + # TODO: allow the naming scheme to be configured... name_format = "{num:02d}" -covers = musicbrainzngs.get_image_list(args.release_mbid) +try: + covers = musicbrainzngs.get_image_list(release_mbid) +except musicbrainzngs.ResponseError as e: + errmsg("error: cannot retrieve image list for release") + errmsg(e) + if isinstance(e.cause, urllib.error.HTTPError) and e.cause.code == 404: + errmsg("is %s a release MBID?" % (release_mbid)) + sys.exit(1) + if len(covers["images"]) == 0: print("release has no cover art, nothing to do") sys.exit() @@ -64,10 +104,11 @@ for c in covers["images"]: print("%s already exists, skipping..." % outname) continue - outfile = tempfile.NamedTemporaryFile(suffix=ext, dir=".") + outfile = open_tmp(suffix=ext, dir=".", mode=0o666) rc = os.spawnlp(os.P_WAIT, 'wget', 'wget', '-O', outfile.name, c["image"]) if rc: failed = True + os.link(outfile.name, outname) outfile.close()