# -*- coding: utf-8 -*- # Copyright 2015 Mike Fährmann # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License version 2 as # published by the Free Software Foundation. import os import sys import shutil import platform from . import config def select(): """Automatically select a suitable printer class""" pdict = { "default": Printer, "pipe": Printer, "term": TerminalPrinter, "terminal": TerminalPrinter, "color": ColorPrinter, } omode = config.get(("output", "mode"), "auto").lower() if omode in pdict: return pdict[omode]() elif omode == "auto": if hasattr(sys.stdout, "isatty") and sys.stdout.isatty(): return ColorPrinter() if ANSI else TerminalPrinter() else: return Printer() else: raise Exception("invalid output mode: " + omode) def safeprint(txt, **kwargs): """Handle unicode errors and replace invalid characters""" try: print(txt, **kwargs) except UnicodeEncodeError: enc = sys.stdout.encoding txt = txt.encode(enc, errors="replace").decode(enc) print(txt, **kwargs) class Printer(): def start(self, path): """Print a message indicating the start of a download""" pass def skip(self, path): """Print a message indicating that a download has been skipped""" print(CHAR_SKIP, path, sep="", flush=True) def success(self, path, tries): """Print a message indicating the completion of a download""" print(path, flush=True) def error(self, file, error, tries, max_tries): """Print a message indicating an error during download""" pass class TerminalPrinter(Printer): def __init__(self): self.short = config.get(("output", "shorten"), True) if self.short: self.width = shutil.get_terminal_size().columns - OFFSET def start(self, path): safeprint(self.shorten(" " + path), end="", flush=True) def skip(self, path): safeprint(self.shorten(CHAR_SKIP + path)) def success(self, path, tries): print("\r", end="") safeprint(self.shorten(CHAR_SUCCESS + path)) def error(self, file, error, tries, max_tries): if tries <= 1 and hasattr(file, "name"): print("\r", end="") safeprint(self.shorten(CHAR_ERROR + file.name)) print("[Error] ", end="") safeprint(error, end="") print(" (", tries, "/", max_tries, ")", sep="") def shorten(self, txt): """Reduce the length of 'txt' to the width of the terminal""" if self.short and len(txt) > self.width: hwidth = self.width // 2 - OFFSET return "".join((txt[:hwidth-1], CHAR_ELLIPSIES, txt[-hwidth-(self.width%2):])) return txt class ColorPrinter(TerminalPrinter): def start(self, path): print(self.shorten(path), end="", flush=True) def skip(self, path): print("\033[2m", self.shorten(path), "\033[0m", sep="") def success(self, path, tries): # if tries == 0: # print("\r", end="") print("\r\033[1;32m", self.shorten(path), "\033[0m", sep="") def error(self, file, error, tries, max_tries): if tries <= 1 and hasattr(file, "name"): print("\r\033[1;31m", self.shorten(file.name), sep="") print("\033[0;31m[Error]\033[0m ", error, " (", tries, "/", max_tries, ")", sep="") if platform.system() == "Windows": ANSI = os.environ.get("TERM") == "ANSI" OFFSET = 1 CHAR_SKIP = "# " CHAR_ERROR = "! " CHAR_SUCCESS = "* " CHAR_ELLIPSIES = "..." else: ANSI = True OFFSET = 0 CHAR_SKIP = "# " CHAR_ERROR = "❌ " CHAR_SUCCESS = "✔ " CHAR_ELLIPSIES = "…"