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
|
||||
from os.path import realpath
|
||||
from pathlib import Path
|
||||
from dir import Root
|
||||
from transcode import Transcoder
|
||||
from log import Log
|
||||
from layers import Dedupe, Transcoder
|
||||
|
||||
def get_args():
|
||||
parser = argparse.ArgumentParser()
|
||||
|
@ -18,17 +17,9 @@ def main(input_dir: Path, output_dir: Path, encoder: Path, out_extension: str =
|
|||
log_path = wd / "logs"
|
||||
if encoder.parts[-1] == "qaac64.exe":
|
||||
out_extension = "m4a"
|
||||
log = Log(log_path)
|
||||
input_root = Root(input_dir, log)
|
||||
output_root = Root(output_dir, log)
|
||||
transcoder = Transcoder(encoder, out_extension, input_dir, output_dir, log_path)
|
||||
transcoder.transcode()
|
||||
|
||||
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__':
|
||||
args = get_args()
|
||||
|
|
|
@ -1,4 +1,3 @@
|
|||
from .layer import Layer
|
||||
from .file import File
|
||||
from .worker import Worker
|
||||
from .transcoder import Transcoder
|
||||
from .dedupe import Dedupe
|
|
@ -1,11 +1,10 @@
|
|||
from pathlib import Path
|
||||
from abc import ABC
|
||||
|
||||
|
||||
audio_extensions = ['.flac']
|
||||
art_extensions = ['.jpg', '.jpeg', '.png']
|
||||
|
||||
class File(ABC):
|
||||
class File:
|
||||
def __init__(self, location: Path):
|
||||
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 subprocess
|
||||
from . import File
|
||||
from pathlib import Path
|
||||
from dir.file import File
|
||||
|
||||
class Worker:
|
||||
def __init__(self, output_root, extension):
|
||||
self.output_root = Path(output_root)
|
||||
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:
|
||||
return self.copy_album_art(track, log)
|
||||
elif track.is_audio:
|
Loading…
Reference in New Issue
Block a user