summaryrefslogtreecommitdiff
path: root/Biz/Lint.py
blob: c3e51df318a98023ae7da1f3b3da2d757e02c06c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
#!/usr/bin/env python
"""
all your lint are belong to us
"""
import os
import subprocess
import sys


# pylint: disable=missing-class-docstring,too-few-public-methods
class Color:
    HEAD = "\033[95m"
    BLUE = "\033[94m"
    GREEN = "\033[92m"
    WARN = "\033[93m"
    FAIL = "\033[91m"
    BOLD = "\033[1m"
    UNDER = "\033[4m"
    END = "\033[0m"


def run(cmd, file):
    "Exec a linter for a file."
    global ERRORS  # pylint: disable=global-statement
    args = {
        "ormolu": ["--mode", "check"],
        "hlint": [],
        "black": ["--quiet", "--check"],
        "pylint": ["--disable=invalid-name"],
    }
    # pylint: disable=subprocess-run-check
    ret = subprocess.run([cmd, *args[cmd], file], stdout=subprocess.PIPE)
    if ret.returncode != 0:
        ERRORS += 1  # pylint: disable=undefined-variable
        msg = ret.stdout.decode("utf-8").strip()
        print(Color.WARN + f"lint error: {cmd}: {file}" + Color.END)
        if msg:
            for line in msg.split("\n"):
                print("  " + line)


def changed_files():
    "Return a list of changed files according to git."
    merge_base = (
        subprocess.check_output(["git", "merge-base", "HEAD", "origin/master"])
        .decode("utf-8")
        .strip()
    )
    return (
        subprocess.check_output(["git", "diff", "--name-only", merge_base])
        .decode("utf-8")
        .strip()
        .split()
    )


def group_files(files, extensions):
    """Given a list of files and list of extensions, return a dict of:
       {ext: [files]}

    """
    root = os.getenv("BIZ_ROOT")
    ret = {k: [] for k in extensions}
    for ext in extensions:
        for file in files:
            if file.endswith(ext):
                ret[ext].append(os.path.join(root, file))
    return ret


def guard_todos(files):
    "Fail if TODO found in text"
    global ERRORS  # pylint: disable=global-statement
    for fname in files:
        with open(fname) as text:
            if "TODO" in text.read():
                ERRORS += 1
                print("found todo:", fname)


if __name__ == "__main__":
    ERRORS = 0
    if "-h" in sys.argv:
        print(f"usage: {os.path.basename(__file__)} <files...>")
        print("if no files given, lint changed files in this branch")
        sys.exit(0)
    elif len(sys.argv) == 1:
        FILES = group_files(changed_files(), [".hs", ".py"])
    else:
        FILES = group_files(sys.argv[1:], [".hs", ".py"])
    for hs in FILES[".hs"]:
        if not os.path.exists(hs):
            print("lint: does not exist:", hs)
            continue
        print(f"lint: {hs}")
        run("ormolu", hs)
        run("hlint", hs)
    for py in FILES[".py"]:
        if not os.path.exists(py):
            print("lint: does not exist:", py)
            continue
        print(f"lint: {py}")
        # Broken in our nixpkgs
        # run("black", py)
        run("pylint", py)
    if ERRORS:
        print("lint: errors:", ERRORS)
    sys.exit(ERRORS)