#!/usr/bin/env bash # Pre-commit hook: secret scan + gofmt check on staged .go files. # Use `git commit --no-verify` to bypass for confirmed false positives. set -u REPO_ROOT="$(git rev-parse --show-toplevel)" cd "$REPO_ROOT" # --- 1. Secret scan --- # Patterns for common credentials. Scans only added lines in the staged diff. SECRET_PATTERNS='(AKIA[0-9A-Z]{16}|sk-[a-zA-Z0-9_-]{20,}|-----BEGIN (RSA|DSA|EC|OPENSSH|PGP)( PRIVATE)?( KEY| KEY BLOCK)?-----|(api[_-]?key|api[_-]?secret|auth[_-]?token|secret[_-]?key|bearer[_-]?token|access[_-]?token|password)[[:space:]]*[:=][[:space:]]*["'"'"'][^"'"'"']{16,}["'"'"'])' secret_hits="$(git diff --cached -U0 --diff-filter=AM \ | grep '^+' | grep -v '^+++' \ | grep -iEn "$SECRET_PATTERNS" || true)" if [ -n "$secret_hits" ]; then echo "pre-commit: potential secret in staged changes:" >&2 echo "$secret_hits" >&2 echo "" >&2 echo "Review the lines above. If this is a false positive (test fixture, documentation)," >&2 echo "bypass with: git commit --no-verify" >&2 exit 1 fi # --- 2. gofmt check on staged .go files --- # gofmt -l lists files that aren't gofmt-clean. Skip generated and vendored # files the same way the rest of the toolchain does. staged_go="$(git diff --cached --name-only --diff-filter=AM \ | grep '\.go$' \ | grep -vE '(^|/)vendor/' || true)" if [ -n "$staged_go" ] && command -v gofmt >/dev/null 2>&1; then unformatted="" while IFS= read -r f; do [ -z "$f" ] && continue [ -f "$f" ] || continue if [ -n "$(gofmt -l "$f" 2>/dev/null)" ]; then unformatted="${unformatted}${f}"$'\n' fi done <<< "$staged_go" if [ -n "$unformatted" ]; then printf 'pre-commit: gofmt check failed — these files need `gofmt -w`:\n\n%s\n' "$unformatted" >&2 echo "Run: gofmt -w (or your editor's format-on-save), then re-stage." >&2 exit 1 fi fi exit 0