badwitch/badwitch.py

198 lines
7.9 KiB
Python
Raw Normal View History

#! /usr/bin/env python3
# -*- coding: utf-8 -*-
# vim:fenc=utf-8:ft=python
#
# Bad Witch
# Copyright © 2020 Vintage Salt <rehashedsalt@cock.li>
#
# Distributed under terms of the MIT license.
#
2020-03-03 18:39:04 -06:00
from appdirs import AppDirs
from pathlib import Path
import argparse
import eyed3
2020-03-03 18:39:04 -06:00
import json
import logging
2020-03-03 23:06:47 -06:00
import pathlib
import sys
2020-03-04 00:04:15 -06:00
import youtube_dl
class Library:
2020-03-03 18:39:04 -06:00
# A thing full of albums
def __init__(self, file, albums={}):
self.albums = albums
self.file = file
# Load from file
2020-03-03 19:14:38 -06:00
def load(self):
2020-03-04 02:04:42 -06:00
try:
with open(self.file, 'r+') as libfd:
libfd.seek(0)
self.albums = json.load(libfd)
except FileNotFoundError:
logging.debug('Could not find library, loading empty')
self.albums = {}
return
2020-03-03 18:39:04 -06:00
# Save to file
def save(self):
with open(self.file, 'w+') as libfd:
libfd.seek(0)
json.dump(self.albums, libfd, indent='\t')
return
def validate(self):
self.load()
2020-03-03 23:06:47 -06:00
try:
for album, albumcontent in self.albums.items():
for song, songcontent in albumcontent.items():
2020-03-04 03:49:37 -06:00
for field in ['track', 'artist', 'source']:
if field not in songcontent:
raise Exception('Song is missing required field', song, field)
# Download library
2020-03-04 00:04:15 -06:00
def download(self, targetalbum=None):
if targetalbum is not None:
print('Downloading album: ' + album)
else:
print('Downloading entire library')
for album, albumcontent in self.albums.items():
2020-03-04 00:04:15 -06:00
# Skip albums that don't match our criterea
if targetalbum is not None and not album == targetalbum:
logging.debug('Skipping album ' + album)
continue
2020-03-03 23:06:47 -06:00
# God have mercy on my soul
2020-03-04 00:04:15 -06:00
artist = next(iter(albumcontent.values()))['artist']
destpath = (Path.home() / 'Music' / artist / album)
Path(destpath).mkdir(parents=True, exist_ok=True)
for song, songcontent in albumcontent.items():
2020-03-04 00:23:52 -06:00
destfile = str(destpath / song) + '.%(ext)s'
logging.debug('Saving to: ' + destfile)
2020-03-04 00:04:15 -06:00
# See if we already have it
if Path(destfile).exists():
2020-03-04 00:23:52 -06:00
logging.info('Already have song: ' + song)
2020-03-04 00:04:15 -06:00
continue
# Download the song
ytdl_opts = {
2020-03-04 00:17:29 -06:00
'format': 'bestaudio',
'postprocessors': [{
'key': 'FFmpegExtractAudio',
'preferredcodec': 'mp3',
2020-03-04 00:17:29 -06:00
'preferredquality': '192'
}],
2020-03-04 00:23:52 -06:00
'quiet': True,
2020-03-04 00:17:29 -06:00
'playlist_items': 1,
2020-03-04 00:23:52 -06:00
'outtmpl': destfile
2020-03-04 00:04:15 -06:00
}
with youtube_dl.YoutubeDL(ytdl_opts) as ydl:
ydl.download([songcontent['source']])
# Add tags
logging.debug('Adding tags')
resultfile = eyed3.load(str(destpath / song) + '.mp3')
print(resultfile)
resultfile.tag.artist = songcontent['artist']
resultfile.tag.album = album
resultfile.tag.title = song
resultfile.tag.track_num = songcontent['track']
resultfile.tag.save()
print('Downloaded song: ' + song)
class BadWitch:
# Our program
def __init__(self):
2020-03-03 19:25:40 -06:00
# Flags and arguments
self.argparser = argparse.ArgumentParser(
description='Manage a declarative music library through YouTube scraping')
2020-03-04 02:05:03 -06:00
self.argparser.add_argument('-l', '--library', metavar='FILE', nargs='?',
help='Override default library file with this one')
self.argparser.add_argument('-v', '--verbose', action='store_true',
help='Show more status messages')
self.argparser.add_argument('-d', '--debug', action='store_true',
help='Show even more status messages')
2020-03-03 19:25:40 -06:00
self.argparser.add_argument('action', metavar='action', nargs='?',
choices=['download', 'edit', 'list'],
2020-03-03 19:25:40 -06:00
help='Action to perform on the library')
2020-03-03 18:39:04 -06:00
# Set up appdirs
self.dirs = AppDirs('badwitch', 'rehashedsalt')
Path(self.dirs.user_data_dir).mkdir(parents=True, exist_ok=True)
def execute(self):
self.args = self.argparser.parse_args()
# Parse flags
if self.args.debug:
logging.basicConfig(level=logging.DEBUG)
elif self.args.verbose:
logging.basicConfig(level=logging.INFO)
2020-03-03 18:39:04 -06:00
# Initialize library
2020-03-03 19:25:40 -06:00
libfile = self.args.library or self.dirs.user_data_dir + '/lib.json'
2020-03-03 18:39:04 -06:00
lib = Library(file=libfile)
2020-03-03 23:06:47 -06:00
lib.load()
lib.validate()
2020-03-03 18:39:04 -06:00
# Perform action
2020-03-03 19:25:40 -06:00
if self.args.action == 'download':
lib.download()
2020-03-03 19:25:40 -06:00
return
elif self.args.action == 'edit':
print('Bad Witch interactive $ibrary editor')
print('^C to abort, ^D to finish changes')
print('Loaded library ' + lib.file)
try:
while True:
in_album = input('\tAlbum: ')
2020-03-04 03:47:22 -06:00
auto_artist = input('\tArtist (leave blank to assign per song): ')
auto_track = int(input('\tStarting track number (leave blank to assign per song): ') or -1)
if in_album not in lib.albums:
lib.albums[in_album] = {}
album = lib.albums[in_album]
try:
while True:
in_song = input('\t\tSong title: ')
2020-03-04 03:47:22 -06:00
if auto_artist is not '':
in_artist = auto_artist
else:
in_artist = input('\t\tArtist: ')
if auto_track is not -1:
in_track = auto_track
auto_track += 1
else:
in_track = input('\t\tTrack number: ')
in_source = input('\t\tSource URL: ')
# Only assign values if we gave them
if in_song not in album:
album[in_song] = {}
song = album[in_song]
if in_track is not '': song['track'] = int(in_track)
2020-03-04 03:47:22 -06:00
if auto_artist is not '': song['artist'] = in_artist
if in_source is not '': song['source'] = in_source
# Bail if song is bad
for field in ['track', 'artist', 'source']:
if field not in song:
print('Critical field is empty: ' + field)
continue
except KeyboardInterrupt:
print('\n\t\tAborting, changes were not saved')
except EOFError:
album[in_song] = song
print('\n\t\tSaving changes')
lib.albums[in_album] = album
lib.save()
except KeyboardInterrupt:
print('\n\tAborting, changes were not saved')
except EOFError:
lib.save()
print('\n\tSaving changes')
print('Closing library')
2020-03-03 19:25:40 -06:00
elif self.args.action == 'list':
for album, albumcontent in lib.albums.items():
2020-03-04 00:42:28 -06:00
print(album)
2020-03-03 19:25:40 -06:00
for song, songcontent in albumcontent.items():
2020-03-04 00:42:28 -06:00
print(str(songcontent['track'])
2020-03-04 00:04:15 -06:00
+ ' - ' + song
+ ' by ' + songcontent['artist'])
2020-03-03 19:25:40 -06:00
return
badwitch = BadWitch()
badwitch.execute()