-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathlefthook.yml
More file actions
141 lines (126 loc) · 5.06 KB
/
Copy pathlefthook.yml
File metadata and controls
141 lines (126 loc) · 5.06 KB
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
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
# Canonical lefthook config for GrayCodeAI Go repos.
# Source of truth: .shared-templates/lefthook.yml.tmpl
#
# Install lefthook:
# brew install lefthook (macOS)
# go install github.com/evilmartians/lefthook@latest
# npm install -g lefthook (cross-platform)
#
# Activate hooks in this repo (one time, done automatically by `make setup`):
# lefthook install
#
# Skip hooks for a single commit (use sparingly):
# LEFTHOOK=0 git commit -m "..."
# ---------------------------------------------------------------------------
# pre-commit — runs before commit creation, on staged files only.
# ---------------------------------------------------------------------------
pre-commit:
parallel: true
commands:
fmt:
glob: "*.go"
run: |
if ! command -v gofumpt >/dev/null 2>&1; then
echo "lefthook: gofumpt not installed (go install mvdan.cc/gofumpt@latest) — skipping"
exit 0
fi
gofumpt -w {staged_files}
stage_fixed: true
imports:
glob: "*.go"
run: |
if ! command -v goimports >/dev/null 2>&1; then
echo "lefthook: goimports not installed (go install golang.org/x/tools/cmd/goimports@latest) — skipping"
exit 0
fi
goimports -w {staged_files}
stage_fixed: true
lint:
glob: "*.go"
run: |
if ! command -v golangci-lint >/dev/null 2>&1; then
echo "lefthook: golangci-lint not installed — skipping (install: https://golangci-lint.run/usage/install/)"
exit 0
fi
golangci-lint run --new-from-rev=HEAD~1 --fix ./...
stage_fixed: true
ast-grep:
glob: "*.go"
run: |
# Run ast-grep structural code search on staged Go files.
# Falls back gracefully if `sg` (ast-grep) is not installed.
STAGED=$(git diff --cached --name-only --diff-filter=ACMR | grep '\.go$' || true)
if [ -n "$STAGED" ]; then
if command -v sg >/dev/null 2>&1; then
echo "$STAGED" | xargs sg scan 2>&1 | head -20
fi
fi
yaml-lint:
glob: "*.{yml,yaml}"
run: |
# Quick syntax check via Python's yaml module (already on most systems).
for f in {staged_files}; do
python3 -c "import sys, yaml; yaml.safe_load(open(sys.argv[1]))" "$f" || exit 1
done
forbidden-strings:
run: |
# Catch obvious credential-shaped strings in staged additions.
bad=$(git diff --cached --diff-filter=AM -U0 -- {staged_files} \
| grep -E '^\+' \
| grep -Ei '(aws_secret|password\s*=|api[_-]?key\s*=|BEGIN [A-Z]+ PRIVATE KEY)' \
| grep -v 'example\|placeholder\|TODO\|x-release-please' || true)
if [ -n "$bad" ]; then
echo "lefthook: possible secret in staged changes:"
echo "$bad"
echo "If this is a false positive, bypass with: LEFTHOOK=0 git commit"
exit 1
fi
# ---------------------------------------------------------------------------
# pre-push — heavier checks, runs only on push (not every commit).
# ---------------------------------------------------------------------------
pre-push:
commands:
test:
run: go test ./... -count=1 -timeout=120s
vet:
run: go vet ./...
govulncheck:
run: |
if ! command -v govulncheck >/dev/null 2>&1; then
echo "lefthook: govulncheck not installed — skipping"
exit 0
fi
govulncheck ./...
# ---------------------------------------------------------------------------
# commit-msg — validate Conventional Commits and strip AI co-author trailers.
# ---------------------------------------------------------------------------
commit-msg:
commands:
conventional-commit:
run: |
msg=$(head -n1 "{1}")
# Allow merge commits, revert commits, and release-please bot commits to bypass.
case "$msg" in
"Merge "*|"Revert "*|"chore(main): release"*) exit 0 ;;
esac
# Conventional commits regex.
if ! echo "$msg" | grep -qE '^(feat|fix|perf|refactor|test|docs|build|ci|chore|revert|style)(\([a-z0-9 _-]+\))?!?: .{1,72}$'; then
echo "lefthook: commit message does not follow Conventional Commits."
echo " format: <type>(<scope>): <summary>"
echo " example: feat(client): add streaming retry"
echo " full guide: https://www.conventionalcommits.org/"
exit 1
fi
strip-co-authored-by:
run: |
# Strip Co-authored-by: trailers that AI tools (Claude, Cursor, etc.) add.
# This enforces the rule that commits list only the human author.
sed '/^[Cc]o-[Aa]uthored-[Bb]y:/d' "{1}" > "{1}.tmp" && mv "{1}.tmp" "{1}"
# ---------------------------------------------------------------------------
# prepare-commit-msg — strip AI co-author trailers after tools inject them.
# ---------------------------------------------------------------------------
prepare-commit-msg:
commands:
strip-co-authored-by:
run: |
sed '/^[Cc]o-[Aa]uthored-[Bb]y:/d' "{1}" > "{1}.tmp" && mv "{1}.tmp" "{1}"