Last active
January 26, 2024 20:38
-
-
Save vovaprog/84593a9b0bcc2ada0ed1912d4d569e06 to your computer and use it in GitHub Desktop.
git blame colored output. Outputs different authors in different colors. Also colorizes C++ and Python code.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#!/usr/bin/python | |
import sys | |
import subprocess | |
import re | |
import time | |
from termcolor import colored | |
import keyword | |
import math | |
keyword_color = 'cyan' | |
brace_color = 'yellow' | |
type_color = 'red' | |
value_color = 'green' | |
preproc_color = 'magenta' | |
def colorize(line, keys, types, vals, preps): | |
words = re.split('(\W+)', line) | |
out = "" | |
for word in words: | |
if word in keys: | |
out += colored(word, keyword_color) | |
elif word in types: | |
out += colored(word, type_color) | |
elif re.match('^[0-9]+(\.[0-9]+)?$', word) or word in vals: | |
out += colored(word, value_color) | |
elif word in preps: | |
out += colored(word, preproc_color) | |
else: | |
for brace in '[ ] { } ( )'.split(): | |
word = word.replace(brace, colored(brace, brace_color)) | |
out += word | |
return out | |
def colorize_cpp(line): | |
keys = """ auto const struct | |
break continue else for switch | |
case default enum goto register sizeof typedef volatile | |
do extern if return static union while | |
asm dynamic_cast namespace reinterpret_cast try | |
explicit new static_cast typeid | |
catch operator template typename | |
class friend private this using | |
const_cast inline public throw virtual | |
delete mutable protected | |
and bitand compl not_eq or_eq xor_eq | |
and_eq bitor not or xor""".split() | |
types = 'int char short long double float void unsigned signed bool wchar_t'.split() | |
vals = 'true false'.split() | |
preps = 'include # define ifdef ifndef endif'.split() | |
return colorize(line, keys, types, vals, preps) | |
def colorize_python(line): | |
keys = keyword.kwlist | |
types = 'int float bool str'.split() | |
vals = 'True False'.split() | |
preps = [] | |
return colorize(line, keys, types, vals, preps) | |
def number_of_lines_in_file(file_name): | |
with open(file_name) as f: | |
return sum(1 for _ in f) | |
reload(sys) | |
sys.setdefaultencoding('utf8') | |
file_name = sys.argv[1] | |
fname = file_name.lower() | |
blame_output = subprocess.check_output( | |
['git', 'blame', '--line-porcelain', file_name]) | |
author_color = {} | |
colors = ['grey', 'red', 'green', 'yellow', | |
'blue', 'magenta', 'cyan', 'white'] | |
color_index = 0 | |
line_number = 1 | |
max_author = 0 | |
line_number_chars = int(math.log10(number_of_lines_in_file(file_name))) + 1 | |
lines = blame_output.splitlines() | |
for line in lines: | |
m = re.search("^author\s+(.*)$", line) | |
if m: | |
author = m.group(1) | |
author = author.strip().decode() | |
if len(author) > max_author: | |
max_author = len(author) | |
for line in lines: | |
m = re.search("^author\s+(.*)$", line) | |
if m: | |
author = m.group(1) | |
author = author.strip().decode() | |
if not author in author_color: | |
color_index = (color_index + 1) % len(colors) | |
author_color[author] = colors[color_index] | |
continue | |
m = re.search("^author-time\s+(.*)$", line) | |
if m: | |
author_time = m.group(1) | |
author_time = time.ctime(int(author_time)) | |
author_time = author_time.encode('utf-8') | |
continue | |
m = re.search("^\t(.*)$", line) | |
if m: | |
code = m.group(1) | |
if (fname.endswith('.cpp') or fname.endswith('.c') or fname.endswith('.cc') or | |
fname.endswith('.h') or fname.endswith('.hpp') or fname.endswith('.hh')): | |
code = colorize_cpp(code) | |
elif fname.endswith('.py'): | |
code = colorize_python(code) | |
print(str.format("{0} {1} {2}: {3}", author_time, | |
colored(author.ljust(max_author + 1), author_color[author]), | |
str(line_number).rjust(line_number_chars), code)) | |
line_number += 1 | |
continue |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment