Compare commits
No commits in common. "875ed7a6688a5d41858caecab3a91a1db6876c21" and "0889c40bb5f0c08a8d39611a78f2fb4d7f77470d" have entirely different histories.
875ed7a668
...
0889c40bb5
|
@ -1,4 +0,0 @@
|
||||||
from .directory import Directory
|
|
||||||
from .root import Root
|
|
||||||
from .artist import Artist
|
|
||||||
from .album import Album
|
|
|
@ -1,22 +0,0 @@
|
||||||
from .directory import Directory
|
|
||||||
from pathlib import Path
|
|
||||||
from log import Log
|
|
||||||
|
|
||||||
|
|
||||||
class Album(Directory):
|
|
||||||
def __init__(self, path: Path, log: Log):
|
|
||||||
super().__init__(path, log, 'ALB')
|
|
||||||
|
|
||||||
@property
|
|
||||||
def all_files(self) -> list:
|
|
||||||
# todo: handle unexpected dirs
|
|
||||||
return self.contents
|
|
||||||
|
|
||||||
def populate(self, log: Log) -> list:
|
|
||||||
contents = list()
|
|
||||||
for e in self.path.iterdir():
|
|
||||||
if e.is_file():
|
|
||||||
contents.append(self.create_file(e))
|
|
||||||
elif e.is_dir():
|
|
||||||
self.log.warning('POP', f"Directory {e} ignored.")
|
|
||||||
return contents
|
|
|
@ -1,19 +0,0 @@
|
||||||
from log import Log
|
|
||||||
|
|
||||||
from .directory import Directory
|
|
||||||
from .album import Album
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
|
|
||||||
class Artist(Directory):
|
|
||||||
def __init__(self, path: Path, log: Log):
|
|
||||||
super().__init__(path, log, 'ART')
|
|
||||||
|
|
||||||
def populate(self, log: Log) -> list:
|
|
||||||
contents = list()
|
|
||||||
for e in self.path.iterdir():
|
|
||||||
if e.is_file():
|
|
||||||
self.log.warning("POP", f"Warning, skipping non-dir '{e}' found in artist '{self.path.parts[-1]}'")
|
|
||||||
elif e.is_dir():
|
|
||||||
contents.append(Album(e, log))
|
|
||||||
return contents
|
|
|
@ -1,61 +0,0 @@
|
||||||
from pathlib import Path
|
|
||||||
from .file import File, Track, Art, MiscFile
|
|
||||||
from abc import ABC, abstractmethod
|
|
||||||
from log import Log, LogCat
|
|
||||||
|
|
||||||
|
|
||||||
class Directory(ABC):
|
|
||||||
def __init__(self, path: Path, log: Log, logcat: str):
|
|
||||||
self.path = path
|
|
||||||
self.log = LogCat(log.queue, logcat)
|
|
||||||
|
|
||||||
self.contents = self.populate(log)
|
|
||||||
|
|
||||||
def __iter__(self):
|
|
||||||
return self.contents.__iter__()
|
|
||||||
|
|
||||||
def __getitem__(self, name):
|
|
||||||
for e in self:
|
|
||||||
if e.name == name:
|
|
||||||
return e
|
|
||||||
raise KeyError
|
|
||||||
|
|
||||||
def prune(self, name):
|
|
||||||
for e in self:
|
|
||||||
if e.name == name:
|
|
||||||
self.contents.remove(e)
|
|
||||||
return
|
|
||||||
raise KeyError
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return self.name
|
|
||||||
|
|
||||||
@property
|
|
||||||
def all_files(self) -> list:
|
|
||||||
files = list()
|
|
||||||
for c in self:
|
|
||||||
# todo: handle unexpected files
|
|
||||||
files += c.all_files
|
|
||||||
return files
|
|
||||||
|
|
||||||
@property
|
|
||||||
def name(self):
|
|
||||||
return self.path.name
|
|
||||||
|
|
||||||
def by_name(self):
|
|
||||||
return [e.name for e in self.contents]
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def populate(self, log: Log) -> list:
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def create_file(file: Path) -> File:
|
|
||||||
suffix = file.suffix
|
|
||||||
|
|
||||||
if suffix in ['.flac']:
|
|
||||||
return Track(file)
|
|
||||||
elif suffix in ['.jpg', '.jpeg', '.png']:
|
|
||||||
return Art(file)
|
|
||||||
else:
|
|
||||||
return MiscFile(file)
|
|
|
@ -1,4 +0,0 @@
|
||||||
from .file import File
|
|
||||||
from .track import Track
|
|
||||||
from .miscfile import MiscFile
|
|
||||||
from .art import Art
|
|
|
@ -1,7 +0,0 @@
|
||||||
from . import File
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
|
|
||||||
class Art(File):
|
|
||||||
def __init__(self, location: Path):
|
|
||||||
super().__init__(location)
|
|
|
@ -1,7 +0,0 @@
|
||||||
from . import File
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
|
|
||||||
class MiscFile(File):
|
|
||||||
def __init__(self, location: Path):
|
|
||||||
super().__init__(location)
|
|
|
@ -1,7 +0,0 @@
|
||||||
from . import File
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
|
|
||||||
class Track(File):
|
|
||||||
def __init__(self, location: Path):
|
|
||||||
super().__init__(location)
|
|
|
@ -1,18 +0,0 @@
|
||||||
from .directory import Directory
|
|
||||||
from .artist import Artist
|
|
||||||
from pathlib import Path
|
|
||||||
from log import Log
|
|
||||||
|
|
||||||
|
|
||||||
class Root(Directory):
|
|
||||||
def __init__(self, path: Path, log: Log):
|
|
||||||
super().__init__(path, log, 'ROOT')
|
|
||||||
|
|
||||||
def populate(self, log: Log) -> list:
|
|
||||||
contents = list()
|
|
||||||
for e in self.path.iterdir():
|
|
||||||
if e.is_file():
|
|
||||||
self.log.warning("POP", f"Warning, skipping non-dir '{e}' found in root")
|
|
||||||
elif e.is_dir():
|
|
||||||
contents.append(Artist(e, log))
|
|
||||||
return contents
|
|
|
@ -1,31 +0,0 @@
|
||||||
from log import Log
|
|
||||||
from dir import Root
|
|
||||||
from . import Layer
|
|
||||||
from dir import Artist
|
|
||||||
|
|
||||||
|
|
||||||
class Dedupe(Layer):
|
|
||||||
def __init__(self, other: Root, log: Log):
|
|
||||||
super().__init__(log, "TCD")
|
|
||||||
self.other = other
|
|
||||||
|
|
||||||
def _process(self, left: Root):
|
|
||||||
right = self.other
|
|
||||||
existing_artists = right.by_name()
|
|
||||||
for artist in left:
|
|
||||||
artist_name = artist.name
|
|
||||||
if artist_name in existing_artists:
|
|
||||||
self.prune_artist(artist, right[artist_name])
|
|
||||||
if len(artist.contents) == 0:
|
|
||||||
left.prune(artist_name)
|
|
||||||
self.log.info('PRN', f"Pruned artist: {artist_name}")
|
|
||||||
else:
|
|
||||||
continue # todo: fuzzy matching
|
|
||||||
|
|
||||||
def prune_artist(self, left: Artist, right: Artist):
|
|
||||||
existing_albums = right.by_name()
|
|
||||||
for album in left:
|
|
||||||
album_name = album.name
|
|
||||||
if album_name in existing_albums:
|
|
||||||
left.prune(album_name)
|
|
||||||
self.log.info('PRN', f"Pruned album: {album_name}")
|
|
|
@ -1,14 +0,0 @@
|
||||||
from abc import ABC, abstractmethod
|
|
||||||
from dir import Root
|
|
||||||
from log import Log, LogCat
|
|
||||||
|
|
||||||
class Layer(ABC):
|
|
||||||
def __init__(self, log: Log, log_category: str):
|
|
||||||
self.log = LogCat(log.queue, log_category)
|
|
||||||
|
|
||||||
def process(self, root: Root):
|
|
||||||
self._process(root)
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def _process(self, root: Root):
|
|
||||||
raise NotImplementedError
|
|
|
@ -1,45 +0,0 @@
|
||||||
from pathlib import Path
|
|
||||||
from multiprocessing import Pool, Manager, set_start_method
|
|
||||||
from log import Log, LogCat
|
|
||||||
from .worker import Worker
|
|
||||||
from dir import Root
|
|
||||||
from .layer import Layer
|
|
||||||
|
|
||||||
|
|
||||||
class Transcoder(Layer):
|
|
||||||
def __init__(self, encoder: Path, extension: str, output_root: Path, log: Log, log_path: Path):
|
|
||||||
super().__init__(log, 'TCD')
|
|
||||||
self.encoder = encoder
|
|
||||||
self.extension = extension
|
|
||||||
self.output_root = output_root
|
|
||||||
|
|
||||||
self.log_path = log_path
|
|
||||||
|
|
||||||
def _process(self, root: Root):
|
|
||||||
transcode_list = root.all_files
|
|
||||||
self._transcode(transcode_list, self.encoder)
|
|
||||||
|
|
||||||
def _transcode(self, transcode_list: list, encoder: Path, workers=16):
|
|
||||||
manager = Manager()
|
|
||||||
queue = manager.Queue()
|
|
||||||
log = Log(self.log_path, queue)
|
|
||||||
logcat = LogCat(log.queue, "TCD")
|
|
||||||
args = [(str(self.output_root), self.extension, track, encoder, logcat) for track in transcode_list]
|
|
||||||
with Pool(workers) as pool:
|
|
||||||
pool.starmap(self.worker, args)
|
|
||||||
pool.close()
|
|
||||||
pool.join()
|
|
||||||
log.stop()
|
|
||||||
|
|
||||||
def _transcode_single_thread(self, transcode_list: list, encoder: Path):
|
|
||||||
log = Log(self.log_path)
|
|
||||||
logcat = LogCat(log.queue, "TCD")
|
|
||||||
worker_args = [(track, encoder) for track in transcode_list]
|
|
||||||
for track, encoder in worker_args:
|
|
||||||
self.worker(str(self.output_root), self.extension, track, encoder, logcat)
|
|
||||||
log.stop()
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def worker(output_root, extension, track, encoder, log):
|
|
||||||
w = Worker(output_root, extension)
|
|
||||||
w.transcode_worker(track, encoder, log)
|
|
15
src/main.py
15
src/main.py
|
@ -1,9 +1,8 @@
|
||||||
import argparse
|
import argparse
|
||||||
from os.path import realpath
|
from os.path import realpath
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from dir import Root
|
from transcode import Transcoder
|
||||||
from log import Log
|
from log import Log
|
||||||
from layers import Dedupe, Transcoder
|
|
||||||
|
|
||||||
def get_args():
|
def get_args():
|
||||||
parser = argparse.ArgumentParser()
|
parser = argparse.ArgumentParser()
|
||||||
|
@ -18,17 +17,9 @@ def main(input_dir: Path, output_dir: Path, encoder: Path, out_extension: str =
|
||||||
log_path = wd / "logs"
|
log_path = wd / "logs"
|
||||||
if encoder.parts[-1] == "qaac64.exe":
|
if encoder.parts[-1] == "qaac64.exe":
|
||||||
out_extension = "m4a"
|
out_extension = "m4a"
|
||||||
log = Log(log_path)
|
transcoder = Transcoder(encoder, out_extension, input_dir, output_dir, log_path)
|
||||||
input_root = Root(input_dir, log)
|
transcoder.transcode()
|
||||||
output_root = Root(output_dir, log)
|
|
||||||
|
|
||||||
dedupe = Dedupe(output_root, log)
|
|
||||||
dedupe.process(input_root)
|
|
||||||
|
|
||||||
transcoder = Transcoder(encoder, out_extension, output_dir, log, log_path)
|
|
||||||
transcoder.process(input_root)
|
|
||||||
|
|
||||||
log.stop()
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
args = get_args()
|
args = get_args()
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
from .layer import Layer
|
from .file import File
|
||||||
from .worker import Worker
|
from .worker import Worker
|
||||||
from .transcoder import Transcoder
|
from .transcoder import Transcoder
|
||||||
from .dedupe import Dedupe
|
|
|
@ -1,11 +1,10 @@
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from abc import ABC
|
|
||||||
|
|
||||||
|
|
||||||
audio_extensions = ['.flac']
|
audio_extensions = ['.flac']
|
||||||
art_extensions = ['.jpg', '.jpeg', '.png']
|
art_extensions = ['.jpg', '.jpeg', '.png']
|
||||||
|
|
||||||
class File(ABC):
|
class File:
|
||||||
def __init__(self, location: Path):
|
def __init__(self, location: Path):
|
||||||
self.path = location
|
self.path = location
|
||||||
|
|
64
src/transcode/transcoder.py
Normal file
64
src/transcode/transcoder.py
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
from pathlib import Path
|
||||||
|
from multiprocessing import Pool, Manager, set_start_method
|
||||||
|
from log import Log, LogCat
|
||||||
|
from . import Worker
|
||||||
|
|
||||||
|
|
||||||
|
class Transcoder:
|
||||||
|
def __init__(self, encoder: Path, extension: str, input_root: Path, output_root: Path, log_path: Path):
|
||||||
|
self.encoder = encoder
|
||||||
|
self.extension = extension
|
||||||
|
self.input_root = input_root
|
||||||
|
self.output_root = output_root
|
||||||
|
|
||||||
|
self.log_path = log_path
|
||||||
|
self.__log = Log(log_path)
|
||||||
|
self.log = LogCat(self.__log.queue, "TCD")
|
||||||
|
|
||||||
|
def transcode(self):
|
||||||
|
transcode_list = []
|
||||||
|
try:
|
||||||
|
for artist in self.input_root.iterdir():
|
||||||
|
if artist.is_dir():
|
||||||
|
for album in artist.iterdir():
|
||||||
|
if album.is_dir():
|
||||||
|
for file in album.iterdir():
|
||||||
|
if file.is_file():
|
||||||
|
if file.name == "DONE":
|
||||||
|
break
|
||||||
|
else:
|
||||||
|
transcode_list.append(file)
|
||||||
|
else:
|
||||||
|
self.log.warning("TRK", f"Warning, skipping non-dir '{album}' found in artist '{artist.parts[-1]}'")
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
self.log.warning("TRK", f"Warning, skipping non-dir '{artist}' found in root")
|
||||||
|
continue
|
||||||
|
self._transcode(transcode_list, self.encoder)
|
||||||
|
finally:
|
||||||
|
self.__log.stop()
|
||||||
|
|
||||||
|
def _transcode(self, transcode_list: list, encoder: Path, workers=16):
|
||||||
|
manager = Manager()
|
||||||
|
queue = manager.Queue()
|
||||||
|
log = Log(self.log_path, queue)
|
||||||
|
logcat = LogCat(log.queue, "TCD")
|
||||||
|
args = [(str(self.output_root), self.extension, track, encoder, logcat) for track in transcode_list]
|
||||||
|
with Pool(workers) as pool:
|
||||||
|
pool.starmap(self.worker, args)
|
||||||
|
pool.close()
|
||||||
|
pool.join()
|
||||||
|
log.stop()
|
||||||
|
|
||||||
|
def _transcode_single_thread(self, transcode_list: list, encoder: Path):
|
||||||
|
log = Log(self.log_path)
|
||||||
|
logcat = LogCat(log.queue, "TCD")
|
||||||
|
worker_args = [(track, encoder) for track in transcode_list]
|
||||||
|
for track, encoder in worker_args:
|
||||||
|
self.worker(str(self.output_root), self.extension, track, encoder, logcat)
|
||||||
|
log.stop()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def worker(output_root, extension, track, encoder, log):
|
||||||
|
w = Worker(output_root, extension)
|
||||||
|
w.transcode_worker(track, encoder, log)
|
|
@ -1,14 +1,15 @@
|
||||||
import shutil
|
import shutil
|
||||||
import subprocess
|
import subprocess
|
||||||
|
from . import File
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from dir.file import File
|
|
||||||
|
|
||||||
class Worker:
|
class Worker:
|
||||||
def __init__(self, output_root, extension):
|
def __init__(self, output_root, extension):
|
||||||
self.output_root = Path(output_root)
|
self.output_root = Path(output_root)
|
||||||
self.extension = extension
|
self.extension = extension
|
||||||
|
|
||||||
def transcode_worker(self, track: File, encoder, log):
|
def transcode_worker(self, track, encoder, log):
|
||||||
|
track = File(track)
|
||||||
if track.is_art:
|
if track.is_art:
|
||||||
return self.copy_album_art(track, log)
|
return self.copy_album_art(track, log)
|
||||||
elif track.is_audio:
|
elif track.is_audio:
|
Loading…
Reference in New Issue
Block a user