3 # Copyright (C) 2010-2012 Vinay Sajip. All rights reserved. Licensed under the new BSD license.
11 class ColorizingStreamHandler(logging.StreamHandler):
12 # color names to indices
24 # levels to (background, foreground, bold/intense)
27 logging.DEBUG: (None, 'white', False),
28 logging.INFO: (None, 'green', False),
29 logging.WARNING: (None, 'yellow', False),
30 logging.ERROR: (None, 'red', False),
31 logging.CRITICAL: ('red', 'white', False)
35 logging.DEBUG: (None, 'white', False),
36 logging.INFO: (None, 'green', False),
37 logging.WARNING: (None, 'yellow', False),
38 logging.ERROR: (None, 'red', False),
39 logging.CRITICAL: ('red', 'white', False)
43 disable_coloring = False
47 isatty = getattr(self.stream, 'isatty', None)
48 return isatty and isatty() and not self.disable_coloring
51 def output_colorized(self, message):
52 self.stream.write(message)
54 ansi_esc = re.compile(r'\x1b\[((?:\d+)(?:;(?:\d+))*)m')
67 def output_colorized(self, message):
70 parts = self.ansi_esc.split(message)
71 write = self.stream.write
73 fd = getattr(self.stream, 'fileno', None)
78 if fd in (1, 2): # stdout or stderr
79 h = ctypes.windll.kernel32.GetStdHandle(-10 - fd)
91 params = [int(p) for p in params.split(';')]
96 color |= self.nt_color_map[p - 40] << 4
98 color |= self.nt_color_map[p - 30]
100 color |= 0x08 # foreground intensity on
101 elif p == 0: # reset to default color
104 pass # error condition ignored
106 ctypes.windll.kernel32.SetConsoleTextAttribute(h, color)
108 def colorize(self, message, record):
109 if record.levelno in self.level_map and self.is_tty:
110 bg, fg, bold = self.level_map[record.levelno]
113 if bg in self.color_map:
114 params.append(str(self.color_map[bg] + 40))
116 if fg in self.color_map:
117 params.append(str(self.color_map[fg] + 30))
122 if params and message:
123 if message.lstrip() != message:
124 prefix = re.search(r"\s+", message).group(0)
125 message = message[len(prefix):]
129 message = "%s%s" % (prefix, ''.join((self.csi, ';'.join(params),
130 'm', message, self.reset)))
134 def format(self, record):
135 message = logging.StreamHandler.format(self, record)
136 return self.colorize(message, record)
138 logging.addLevelName(15, "INFO")
139 logger = logging.getLogger('nhentai')
140 LOGGER_HANDLER = ColorizingStreamHandler(sys.stdout)
141 FORMATTER = logging.Formatter("\r[%(asctime)s] [%(levelname)s] %(message)s", "%H:%M:%S")
142 LOGGER_HANDLER.setFormatter(FORMATTER)
143 LOGGER_HANDLER.level_map[logging.getLevelName("INFO")] = (None, "cyan", False)
144 logger.addHandler(LOGGER_HANDLER)
145 logger.setLevel(logging.DEBUG)
148 if __name__ == '__main__':
149 logger.log(15, 'nhentai')
152 logger.debug('debug')
153 logger.error('error')
154 logger.critical('critical')