e3af07b1eb
- Integrate cargo fmt --check and clippy -Dwarnings into .githooks/pre-commit - Matches CI's RUSTFLAGS="-Dwarnings" to catch issues before push - Runs before existing secret detection checks
110 lines
3.7 KiB
Bash
Executable File
110 lines
3.7 KiB
Bash
Executable File
#!/bin/bash
|
|
# Pre-commit hook:
|
|
# 1. cargo fmt --check (matches CI)
|
|
# 2. cargo clippy -Dwarnings (matches CI RUSTFLAGS)
|
|
# 3. Secret detection (mnemonic phrases, hex keys)
|
|
#
|
|
# Skip with: git commit --no-verify
|
|
|
|
RED='\033[0;31m'
|
|
GREEN='\033[0;32m'
|
|
YELLOW='\033[0;33m'
|
|
NC='\033[0m'
|
|
|
|
# ── Rust CI checks ──────────────────────────────────────────────────
|
|
echo -e "${YELLOW}pre-commit: checking formatting...${NC}"
|
|
if ! cargo fmt --all -- --check 2>/dev/null; then
|
|
echo -e "${RED}FAILED:${NC} cargo fmt --all -- --check"
|
|
echo -e "Run ${GREEN}cargo fmt --all${NC} to fix, then re-stage and commit."
|
|
exit 1
|
|
fi
|
|
|
|
echo -e "${YELLOW}pre-commit: running clippy (-Dwarnings)...${NC}"
|
|
if ! RUSTFLAGS="-Dwarnings" cargo clippy --workspace --all-targets 2>&1; then
|
|
echo -e "${RED}FAILED:${NC} cargo clippy --workspace --all-targets"
|
|
exit 1
|
|
fi
|
|
|
|
echo -e "${GREEN}pre-commit: fmt + clippy passed${NC}"
|
|
|
|
# ── Secret detection ────────────────────────────────────────────────
|
|
echo "Checking for potential secrets..."
|
|
|
|
# Patterns to detect
|
|
PATTERNS=(
|
|
# 25-word mnemonic phrases (not obviously fake like "bacon bacon...")
|
|
'[a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+'
|
|
)
|
|
|
|
# Allowed fake test patterns (same word repeated)
|
|
ALLOWED_PATTERNS=(
|
|
"bacon bacon bacon"
|
|
"abbey abbey abbey"
|
|
"test test test"
|
|
"word word word"
|
|
)
|
|
|
|
# Get staged files
|
|
STAGED_FILES=$(git diff --cached --name-only --diff-filter=ACM | grep -E '\.(js|ts|json|md|env)$')
|
|
|
|
if [ -z "$STAGED_FILES" ]; then
|
|
exit 0
|
|
fi
|
|
|
|
FOUND_SECRETS=0
|
|
|
|
for file in $STAGED_FILES; do
|
|
# Skip wordlist files
|
|
if [[ "$file" == *"wordlist"* ]]; then
|
|
continue
|
|
fi
|
|
|
|
# Check for potential 25-word phrases
|
|
MATCHES=$(git diff --cached "$file" | grep -E "^\+" | grep -oE "'[a-z]+ [a-z]+ [a-z]+ [a-z]+ [a-z]+[^']*'" | head -5)
|
|
|
|
for match in $MATCHES; do
|
|
# Check if it's an allowed fake pattern
|
|
IS_ALLOWED=0
|
|
for allowed in "${ALLOWED_PATTERNS[@]}"; do
|
|
if echo "$match" | grep -q "$allowed"; then
|
|
IS_ALLOWED=1
|
|
break
|
|
fi
|
|
done
|
|
|
|
# If not allowed and looks like 25 words, flag it
|
|
if [ $IS_ALLOWED -eq 0 ]; then
|
|
WORD_COUNT=$(echo "$match" | tr -cd ' ' | wc -c)
|
|
if [ "$WORD_COUNT" -ge 20 ]; then
|
|
echo "WARNING: Potential mnemonic in $file"
|
|
echo " $match"
|
|
FOUND_SECRETS=1
|
|
fi
|
|
fi
|
|
done
|
|
|
|
# Check for 64-char hex that looks like a key (not all zeros/ones/test patterns)
|
|
HEX_MATCHES=$(git diff --cached "$file" | grep -E "^\+" | grep -oE "['\"][a-f0-9]{64}['\"]" | grep -v "0000000000" | grep -v "ffffffff" | grep -v "01234567" | grep -v "f5f5f5f5" | head -5)
|
|
|
|
for hex in $HEX_MATCHES; do
|
|
# Check if it's in a test file with TEST_ prefix nearby
|
|
if ! grep -B2 "$hex" "$file" 2>/dev/null | grep -qE "TEST_|test_|const test|// test"; then
|
|
echo "WARNING: Potential secret key in $file"
|
|
echo " $hex"
|
|
FOUND_SECRETS=1
|
|
fi
|
|
done
|
|
done
|
|
|
|
if [ $FOUND_SECRETS -eq 1 ]; then
|
|
echo ""
|
|
echo "Potential secrets detected! Review carefully before committing."
|
|
echo "If these are intentional test values, you can bypass with: git commit --no-verify"
|
|
echo ""
|
|
# Warning only - don't block (use exit 1 to block)
|
|
exit 0
|
|
fi
|
|
|
|
echo "No secrets detected."
|
|
exit 0
|