flac support and an argument for choosing when to embed cover art
This commit is contained in:
parent
801b7369c4
commit
124f0f7b42
1 changed files with 69 additions and 48 deletions
47
bcao.py
47
bcao.py
|
@ -6,19 +6,20 @@
|
||||||
# output: it organises it, adds cover art, puts it in the right place...
|
# output: it organises it, adds cover art, puts it in the right place...
|
||||||
|
|
||||||
import argparse
|
import argparse
|
||||||
import base64
|
|
||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
import sys
|
import sys
|
||||||
import tempfile
|
import tempfile
|
||||||
import shutil
|
import shutil
|
||||||
from os import path
|
from os import path
|
||||||
|
from base64 import b64encode
|
||||||
from zipfile import ZipFile
|
from zipfile import ZipFile
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import Optional, List, Dict
|
from typing import Optional, List, Dict
|
||||||
|
|
||||||
import mutagen
|
import mutagen
|
||||||
from mutagen.flac import Picture
|
from mutagen.flac import Picture, FLAC
|
||||||
|
from mutagen.oggvorbis import OggVorbis
|
||||||
from mutagen import id3
|
from mutagen import id3
|
||||||
|
|
||||||
from PIL import Image
|
from PIL import Image
|
||||||
|
@ -37,24 +38,39 @@ def get_tag(mut_song: mutagen.FileType, tag: str) -> str:
|
||||||
tag = tag.replace("_", "")
|
tag = tag.replace("_", "")
|
||||||
return sanitise(mut_song[tag][0] if type(mut_song[tag]) is list else mut_song[tag])
|
return sanitise(mut_song[tag][0] if type(mut_song[tag]) is list else mut_song[tag])
|
||||||
|
|
||||||
|
def has_cover(mut_song: mutagen.FileType):
|
||||||
|
if type(mut_song) is OggVorbis:
|
||||||
|
return "metadata_block_picture" in mut_song and len(mut_song["metadata_block_picture"]) != 0
|
||||||
|
if type(mut_song) is FLAC:
|
||||||
|
return len(mut_song.pictures) != 0
|
||||||
|
raise Exception(f"what is {type(mut_song)}")
|
||||||
|
|
||||||
def sanitise(in_str: str) -> str:
|
def sanitise(in_str: str) -> str:
|
||||||
if args.sanitise:
|
if args.sanitise:
|
||||||
return re.sub(r"[?\\/:|*\"<>]", "_", in_str)
|
return re.sub(r"[?\\/:|*\"<>]", "_", in_str)
|
||||||
return in_str
|
return in_str
|
||||||
|
|
||||||
parser = argparse.ArgumentParser(description="Extracts the given zip file downloaded from Bandcamp and organises it.")
|
def parse_add_cover_images(value: str) -> str:
|
||||||
parser.add_argument('zip', help='The zip file to use')
|
if value not in ["n", "a", "w"]:
|
||||||
|
raise argparse.ArgumentTypeError("Must be one of n, a, w")
|
||||||
|
return value
|
||||||
|
|
||||||
|
parser = argparse.ArgumentParser(usage='%(prog)s zip [options]',
|
||||||
|
description="Extracts the given zip file downloaded from Bandcamp and organises it.")
|
||||||
|
parser.add_argument('zip', help='The zip file to use.')
|
||||||
|
parser.add_argument('-c', '--add-cover-images', dest='process_cover', default='w', type=parse_add_cover_images,
|
||||||
|
help="When to embed cover art into songs. Options: [n]ever, [a]lways, [w]hen necessary. Default: %(default)s")
|
||||||
parser.add_argument('-d', '--destination', dest='destination', default='/home/lynne/Music/Music/',
|
parser.add_argument('-d', '--destination', dest='destination', default='/home/lynne/Music/Music/',
|
||||||
help="The directory to organise the music into. Default: /home/lynne/Music/Music/")
|
help="The directory to organise the music into. Default: %(default)s")
|
||||||
parser.add_argument('-q', '--quiet', dest='quiet', action='store_true',
|
parser.add_argument('-q', '--quiet', dest='quiet', action='store_true',
|
||||||
help='Disable non-error output and assume default artist name.')
|
help='Disable non-error output and assume default artist name.')
|
||||||
parser.add_argument('-u', '--unsanitised', dest='sanitise', action='store_false',
|
parser.add_argument('-u', '--unsanitised', dest='sanitise', action='store_false',
|
||||||
help="Don't replace NTFS-unsafe characters with underscores. Not recommended.")
|
help="Don't replace NTFS-unsafe characters with underscores. Not recommended.")
|
||||||
parser.add_argument('-t', '--threshold', dest='threshold', nargs=1, default=300,
|
parser.add_argument('-t', '--threshold', dest='threshold', nargs=1, default=300,
|
||||||
help="Maximum acceptable cover art file size in kilobytes. Default: 300")
|
help="Maximum acceptable file size for cover art, in kilobytes. Default: %(default)s")
|
||||||
|
|
||||||
args = parser.parse_args()
|
args = parser.parse_args()
|
||||||
# convert args.treshold to bytes
|
# convert args.threshold to bytes
|
||||||
args.threshold *= 1024
|
args.threshold *= 1024
|
||||||
|
|
||||||
if not path.exists(args.zip):
|
if not path.exists(args.zip):
|
||||||
|
@ -68,7 +84,7 @@ song_names: List[str] = []
|
||||||
|
|
||||||
with ZipFile(args.zip, 'r') as zip_file:
|
with ZipFile(args.zip, 'r') as zip_file:
|
||||||
for file in zip_file.namelist():
|
for file in zip_file.namelist():
|
||||||
if re.match(r"^(.+ - ){2}\d{2,} .+\.(ogg|flac|alac|aiff|wav|mp3|opus|m4a|aac|oga)$", file):
|
if re.match(r"^(.+ - ){2}\d{2,} .+\.(ogg|flac|alac|aiff|wav|mp3|opus|m4a|aac)$", file):
|
||||||
# bandcamp zips contains songs with names formatted like "Album - Artist - 01 Song.mp3"
|
# bandcamp zips contains songs with names formatted like "Album - Artist - 01 Song.mp3"
|
||||||
# for example, "King Crimson - In the Wake of Poseidon - 02 Pictures of a City.ogg"
|
# for example, "King Crimson - In the Wake of Poseidon - 02 Pictures of a City.ogg"
|
||||||
# this regex should match only on those, and cut out (hopefully) all of the bonus material stuff, which shouldn't
|
# this regex should match only on those, and cut out (hopefully) all of the bonus material stuff, which shouldn't
|
||||||
|
@ -84,6 +100,7 @@ with ZipFile(args.zip, 'r') as zip_file:
|
||||||
# we'll need this to know what metadata format we should write
|
# we'll need this to know what metadata format we should write
|
||||||
song_format: str = path.splitext(song_names[0])[1][1:]
|
song_format: str = path.splitext(song_names[0])[1][1:]
|
||||||
|
|
||||||
|
if args.process_cover != 'n':
|
||||||
log("Resizing album art to embed in songs...")
|
log("Resizing album art to embed in songs...")
|
||||||
with Image.open(str(Path(tmp, cover))) as image:
|
with Image.open(str(Path(tmp, cover))) as image:
|
||||||
temp_cover: Path = Path(tmp, "cover-lq.jpg")
|
temp_cover: Path = Path(tmp, "cover-lq.jpg")
|
||||||
|
@ -109,13 +126,12 @@ with Image.open(str(Path(tmp, cover))) as image:
|
||||||
# something very bad has happened here
|
# something very bad has happened here
|
||||||
die("Failed to resize image")
|
die("Failed to resize image")
|
||||||
|
|
||||||
|
# read the image file to get its raw data
|
||||||
# read the image file to get the file's raw data
|
|
||||||
with open(temp_cover, 'r+b') as cover_file:
|
with open(temp_cover, 'r+b') as cover_file:
|
||||||
data = cover_file.read()
|
data = cover_file.read()
|
||||||
|
|
||||||
with Image.open(temp_cover) as image:
|
with Image.open(temp_cover) as image:
|
||||||
if song_format == "ogg":
|
if song_format == "ogg" or song_format == "flac":
|
||||||
# i hate this
|
# i hate this
|
||||||
embed_cover = Picture()
|
embed_cover = Picture()
|
||||||
embed_cover.data = data
|
embed_cover.data = data
|
||||||
|
@ -124,7 +140,6 @@ with Image.open(temp_cover) as image:
|
||||||
embed_cover.width = image.size[0]
|
embed_cover.width = image.size[0]
|
||||||
embed_cover.height = image.size[1]
|
embed_cover.height = image.size[1]
|
||||||
embed_cover.depth = image.bits
|
embed_cover.depth = image.bits
|
||||||
embed_cover = base64.b64encode(embed_cover.write()).decode("ascii")
|
|
||||||
else:
|
else:
|
||||||
log(f"Format {song_format} is not fully supported - cover images will not be modified", 1)
|
log(f"Format {song_format} is not fully supported - cover images will not be modified", 1)
|
||||||
|
|
||||||
|
@ -146,9 +161,15 @@ for song in song_names:
|
||||||
map(lambda x: artists.append(sanitise(x)), m['artist'])
|
map(lambda x: artists.append(sanitise(x)), m['artist'])
|
||||||
songs[song] = f"{str(get_tag(m, 'track')).zfill(zeroes)} {get_tag(m, 'title')}.{song_format}"
|
songs[song] = f"{str(get_tag(m, 'track')).zfill(zeroes)} {get_tag(m, 'title')}.{song_format}"
|
||||||
|
|
||||||
|
if args.process_cover == 'a' or (args.process_cover == 'w' and has_cover(m) is False):
|
||||||
|
log("Embedding cover art...")
|
||||||
# embed cover art
|
# embed cover art
|
||||||
if song_format == "ogg":
|
if song_format == "ogg":
|
||||||
m["metadata_block_picture"] = [embed_cover]
|
m["metadata_block_picture"] = [b64encode(embed_cover.write()).decode("ascii")]
|
||||||
|
m.save()
|
||||||
|
elif song_format == "flac":
|
||||||
|
m.clear_pictures()
|
||||||
|
m.add_picture(embed_cover)
|
||||||
m.save()
|
m.save()
|
||||||
|
|
||||||
# remove duplicate artists
|
# remove duplicate artists
|
||||||
|
|
Loading…
Reference in a new issue