| name | vbs-scan-security |
| description | Use when scanning code for security vulnerabilities. Use when user says "scan security", "kiểm tra bảo mật", "security audit", "review security", or invokes `/vbs-scan-security`. Auto-delegates to sub-agents for large scans (>20 main-language files OR >30 total OR >14 days). Outputs bilingual reports (vi/en). |
| userInvocable | true |
vbsec — Security Scanner cho Vibe Coders
Quét lỗ hổng bảo mật cho code do AI sinh ra (vibe code). Bộ skill này check 21 lỗi bảo mật phổ biến nhất của vibe code, kế thừa kiến trúc SMALL/LARGE mode từ bộ rule production của SePay, tổng quát hóa cross-language (mặc định) + chuyên sâu cho Go/PHP (phase 1).
Public repo: https://github.com/tanviet12/vbsec
License: MIT (sẽ chốt khi public)
Invocation
| Command | Scope | Mô tả |
|---|
/vbs-scan-security | Toàn repo (default từ v0.3) | Mặc định — quét toàn bộ repo |
/vbs-scan-security all | Toàn repo | Alias explicit của default |
/vbs-scan-security uncommitted | Uncommitted changes | Quét staged + unstaged (cần explicit từ v0.3) |
/vbs-scan-security diff | Uncommitted changes | Alias intuitive cho uncommitted |
/vbs-scan-security staged | Staged files only | Pre-commit scan |
/vbs-scan-security commit within Xdays | Recent commits | Quét commit X ngày gần đây |
/vbs-scan-security commit id <sha> | Specific commit | Quét 1 commit |
/vbs-scan-security pr id <number> | Pull request | Quét PR diff (cần gh CLI) |
v0.3 change: Default scope đổi từ uncommitted → all. Non-tech user lần đầu chạy không bị confused bởi report rỗng. Để giữ behavior cũ, dùng uncommitted hoặc diff explicit.
Lựa chọn ngôn ngữ output (thêm vào bất kỳ scope nào):
lang=vi hoặc --vi → Tiếng Việt (mặc định)
lang=en hoặc --en → English
Ví dụ:
/vbs-scan-security pr id 42 lang=en
/vbs-scan-security staged --vi
/vbs-scan-security commit within 7days
CRITICAL: Cách dùng skill này (cho LLM agent)
Các pattern bash/grep trong rule files là VÍ DỤ minh họa, KHÔNG phải lệnh chạy literal.
Nguyên tắc
- Lý luận, không pattern-match thuần — Hiểu intent bảo mật đằng sau mỗi check, không chỉ tìm chuỗi
- Dùng tool phù hợp —
Grep, Read, Glob thay vì bash grep/find
- Đọc context đầy đủ — Khi gặp pattern, READ hàm xung quanh để hiểu đây có thực sự là lỗ hổng không
- Phân loại trust level — Một query có format chuỗi chỉ nguy hiểm nếu data ghép vào là L1 (untrusted)
Phân loại nguồn dữ liệu (L1–L4)
| Level | Nguồn | Tin cậy | Ví dụ |
|---|
| L1 | Input người dùng | KHÔNG tin | req.body, $_GET, request.params, HTTP header, file upload |
| L2 | Database | Bán tin | Giá trị từ DB nhưng nguồn gốc là user input |
| L3 | Code nội bộ | Tin | Hardcoded strings, config keys, computed values |
| L4 | Hệ thống | Tin | Env vars, file paths nội bộ, framework constants |
Key insight: f"SELECT ... {x}" SAFE nếu x là L3+. CRITICAL nếu x là L1 không qua parameterization.
Tham khảo chi tiết: references/data-flow-classification.md.
Workflow
┌─────────────────────────────────────────────────────────────────────┐
│ vbsec SCAN WORKFLOW │
├─────────────────────────────────────────────────────────────────────┤
│ │
│ [Step 0] Parse args │
│ ├─ Scope (uncommitted/staged/commit/pr/all) │
│ └─ Output lang (vi default | en) │
│ ↓ │
│ [Step 1] Gather files (git) │
│ ↓ │
│ [Step 2] Detect primary code language │
│ └─ Đọc references/language-detection.md │
│ ↓ │
│ [Step 3] Route by size │
│ ┌──────────────────┬──────────────────┐ │
│ │ SMALL (inline) │ LARGE (delegate)│ │
│ │ ≤20 main+≤30tot │ >20 OR >30 OR │ │
│ │ AND ≤14d │ >14 ngày │ │
│ └─────┬────────────┴─────────┬────────┘ │
│ ↓ ↓ │
│ workflows/small- workflows/large- │
│ review.md review.md │
│ │
│ Both apply: │
│ - rules/generic/*.md (21 rules, luôn chạy) │
│ - rules/languages/<detected>/*.md (override nếu trùng tên) │
│ ↓ │
│ [Step 4] Generate report │
│ ├─ Markdown report (theo lang chọn) │
│ └─ JSON summary (canonical EN, ở cuối) │
│ │
└─────────────────────────────────────────────────────────────────────┘
Step 0: Parse Arguments
Dùng Bash tool ĐÚNG MỘT LẦN cho step này (gather files là việc của git, không phải reasoning).
ARGS="${ARGUMENTS:-}"
IS_GIT_REPO=true
git rev-parse --is-inside-work-tree >/dev/null 2>&1 || IS_GIT_REPO=false
LANG="vi"
if echo "$ARGS" | grep -qE 'lang=en|--en|\ben\b'; then LANG="en"; fi
if echo "$ARGS" | grep -qE 'lang=vi|--vi'; then LANG="vi"; fi
SCOPE=$(echo "$ARGS" | sed -E 's/(lang=(vi|en)|--vi|--en)//g' | xargs)
NO_GIT_NOTE=""
case "$SCOPE" in
"staged"|"uncommitted"|"diff"|"commit within "*|"commit id "*|"pr id "*)
if [ "$IS_GIT_REPO" = false ]; then
echo "{msg_scope_needs_git}"
exit 1
fi
case "$SCOPE" in
"staged") FILES=$(git diff --cached --name-only) ;;
"uncommitted"|"diff") FILES=$(git diff --name-only HEAD); [ -z "$FILES" ] && FILES=$(git diff --cached --name-only) ;;
"commit within "*) DAYS=$(echo "$SCOPE" | grep -oE '[0-9]+'); FILES=$(git log --since="${DAYS} days ago" --name-only --pretty=format: | sort -u | grep -v '^$') ;;
"commit id "*) SHA=$(echo "$SCOPE" | sed 's/commit id //'); FILES=$(git diff-tree --no-commit-id --name-only -r "$SHA") ;;
"pr id "*) PR=$(echo "$SCOPE" | sed 's/pr id //'); FILES=$(gh pr diff "$PR" --name-only) ;;
esac
;;
"all"|"")
if [ "$IS_GIT_REPO" = true ]; then
FILES=$(git ls-files)
else
FILES=$(find . -type f \
-not -path '*/.git/*' \
-not -path '*/.next/*' \
-not -path '*/.nuxt/*' \
-not -path '*/.venv/*' \
-not -path '*/.idea/*' \
-not -path '*/.vscode/*' \
-not -path '*/node_modules/*' \
-not -path '*/vendor/*' \
-not -path '*/dist/*' \
-not -path '*/build/*' \
-not -path '*/target/*' \
-not -path '*/__pycache__/*' \
-not -path '*/vbsec-reports/*' \
2>/dev/null | sed 's|^\./||')
NO_GIT_NOTE="true"
fi
;;
*)
echo "Unknown scope: $SCOPE"
exit 1
;;
esac
FILES=$(echo "$FILES" | grep -vE '(^|/)(node_modules|vendor|dist|build|\.next|\.nuxt|target|\.venv|__pycache__|\.git|vbsec-reports)/' || true)
TIMESTAMP=$(date +"%Y-%m-%d-%H%M%S")
REPORT_DIR="vbsec-reports"
REPORT_FILE="${REPORT_DIR}/scan-${TIMESTAMP}.md"
mkdir -p "${REPORT_DIR}"
GITIGNORE_WARNING=""
if [ "$IS_GIT_REPO" = true ]; then
if [ -f .gitignore ]; then
grep -qE '^vbsec-reports/?$' .gitignore || GITIGNORE_WARNING="missing"
else
GITIGNORE_WARNING="missing"
fi
fi
echo "Scope: ${SCOPE:-all (default)}"
echo "Lang: $LANG"
echo "Git repo: $IS_GIT_REPO"
echo "Files: $(echo "$FILES" | wc -l)"
echo "Report file: $REPORT_FILE"
[ "$NO_GIT_NOTE" = "true" ] && echo "Note: non-git folder — scanning all files via find"
[ "$GITIGNORE_WARNING" = "missing" ] && echo "Note: vbsec-reports/ not in .gitignore — will warn user at end"
Quan trọng:
vbsec-reports/ được excluded khỏi scan list — không scan chính báo cáo của mình
- Path output
vbsec-reports/scan-<timestamp>.md cần được mkdir trước khi scan, để workflows save vào
- v0.5.1+: skill chạy được trên cả non-git folder. Default scope (
all) dùng find thay git ls-files. Các scope dựa vào git (staged, uncommitted, commit within, commit id, pr id) BẮT BUỘC git — báo msg_scope_needs_git rồi exit.
- Nếu
NO_GIT_NOTE=true, report header phải in {msg_no_git_note} để user biết folder không có git → không lọc theo .gitignore.
Step 1: Load i18n Strings
Đọc file i18n tương ứng với $LANG:
File i18n chứa bảng key→text cho toàn bộ user-facing strings (section headers, severity labels, verdict, fix recommendations templates). Mọi text trong report final phải lấy từ i18n, KHÔNG hardcode.
Strings KHÔNG bao giờ dịch: rule ID (SQL-INJECTION, XSS, IDOR...), file path, code snippet, command name (/vbs-scan-security).
Step 2: Detect Primary Code Language
Đọc references/language-detection.md để biết cách detect. Tóm tắt:
- Count extension trong file list (đã strip vendored):
.go, .py, .php, .js, .ts, .jsx, .tsx, .rb, .java, .rs, .cs, .csproj, .sln
- Primary lang = lang chiếm ≥30% tổng files
- Có
rules/languages/<lang>/ → load overlay; không có → chỉ dùng generic
- Multi-lang repo (cả Go backend + Vue frontend) → load cả 2 overlay
Hiện hỗ trợ chuyên sâu: go, php, typescript (gộp JS+TS), python, dotnet. Các lang khác chỉ dùng generic rules.
Step 3: Route by Size
| Điều kiện | Ngưỡng | Mode |
|---|
| Files ngôn ngữ chính | ≤20 | SMALL |
| Files ngôn ngữ chính | >20 | LARGE |
| Tổng files | ≤30 | SMALL |
| Tổng files | >30 | LARGE |
Timespan (chỉ với scope commit within) | ≤14 ngày | SMALL |
| Timespan | >14 ngày | LARGE |
BẤT KỲ điều kiện nào sang LARGE → dùng LARGE mode.
- SMALL mode: Read
workflows/small-review.md và follow workflow đó (inline, không sub-agent)
- LARGE mode: Read
workflows/large-review.md, trở thành orchestrator only:
- TodoWrite cho từng chunk (resume được nếu interrupt)
- Chunk files theo top-level folder (xem
references/chunking-strategy.md)
- Spawn sub-agents (general-purpose) cho mỗi chunk với prompt từ
references/sub-agent-prompts.md
- Sub-agents ghi findings ra
.vbsec-tmp/findings-<chunk>.md (luôn dùng EN canonical + rule ID)
- Main agent aggregate → translate sang
$LANG → final report
- Cleanup
.vbsec-tmp/ sau khi done
Step 4: Apply Rules
Cho mỗi rule trong rules/generic/ (01-21):
- Read rule file → hiểu intent, severity, search patterns gợi ý
- Apply lên files trong scope (dùng Grep/Read tool)
- Với mỗi match: trace data flow (L1-L4), phân loại có phải vulnerability thật không
- Nếu có rule cùng tên (cùng
id) trong rules/languages/<detected-lang>/, rule chuyên sâu thắng generic (đè hoàn toàn pattern + reasoning steps cho lang đó).
21 rules generic:
| # | ID | Severity max |
|---|
| 1 | HARDCODED-SECRET | CRITICAL |
| 2 | SQL-INJECTION | CRITICAL |
| 3 | XSS | HIGH |
| 4 | IDOR | HIGH |
| 5 | SLOPSQUATTING | CRITICAL |
| 6 | BRUTE-FORCE | HIGH |
| 7 | MASS-ASSIGNMENT | CRITICAL |
| 8 | INSECURE-DESERIALIZATION | CRITICAL |
| 9 | SSRF | HIGH |
| 10 | PATH-TRAVERSAL | HIGH |
| 11 | CSRF | HIGH |
| 12 | BROKEN-ACCESS-CONTROL | CRITICAL |
| 13 | WEAK-PASSWORD-HASHING | CRITICAL |
| 14 | JWT-NONE-ALGORITHM | CRITICAL |
| 15 | CORS-MISCONFIG | HIGH |
| 16 | UNRESTRICTED-FILE-UPLOAD | CRITICAL |
| 17 | VERBOSE-ERROR-DEBUG-MODE | HIGH |
| 18 | MISSING-RATE-LIMIT | HIGH |
| 19 | RACE-CONDITION | HIGH |
| 20 | OUTDATED-DEPENDENCY | HIGH |
| 21 | COMMAND-INJECTION | CRITICAL |
Step 5: Generate Report (v0.3+ — verbose + persistent)
Tham khảo template trong references/output-format.md. Quy tắc cốt lõi:
Verbose level theo severity:
- CRITICAL → bảng overview + full verbose block per finding (Mô tả ngắn + Tại sao nguy hiểm + Hacker khai thác + Code before/after + Đọc thêm)
- HIGH → bảng overview + medium block per finding (Mô tả + Tác động + Code fix + Đọc thêm)
- MEDIUM → chỉ bảng compact
- LOW → chỉ bảng compact
Layout:
- Header block (scope, file count, primary lang, mode, date, lang code)
- VERDICT + 1-line description
- CRITICAL section (overview table → verbose blocks)
- HIGH section (overview table → medium blocks)
- MEDIUM section (compact table)
- LOW section (compact table)
- PASSED CHECKS (list)
- Next steps (1-2 dòng)
- Save notification (path file đã ghi)
- Gitignore warning (nếu cần)
- Footer + disclaimer
- JSON summary (canonical EN — không phụ thuộc lang)
Save-to-file (v0.3+):
Sau khi render report:
cat > "$REPORT_FILE" <<'REPORT_EOF'
<full report content here>
REPORT_EOF
echo ""
echo "📄 {msg_report_saved}: $REPORT_FILE"
[ "$GITIGNORE_WARNING" = "missing" ] && echo "⚠️ {msg_gitignore_warning_title}: {msg_gitignore_warning_text}"
LLM agent thực thi bằng Write tool (NOT bash heredoc) để ghi file, sau đó in 1-2 dòng note ra stdout. Nội dung file PHẢI IDENTICAL với output trên stdout.
Mọi section header, severity label, verdict text lấy từ i18n file đã load ở Step 1.
Verdict Logic
| Điều kiện | Verdict |
|---|
| Có ≥1 CRITICAL | FAIL |
| Không CRITICAL, có ≥1 HIGH | WARN |
| Không CRITICAL, không HIGH | PASS |
WARN ≠ approve. Báo cáo cần nêu rõ HIGH issues cần khắc phục trước production.
Cấu trúc skill (cho người contribute)
~/.claude/skills/vbs-scan-security/
├── SKILL.md # File này
├── workflows/
│ ├── small-review.md # Inline scan (default cho repo nhỏ-vừa)
│ └── large-review.md # Sub-agent delegation
├── rules/
│ ├── generic/ # 21 rules cross-language (bắt buộc apply)
│ │ ├── 01-hardcoded-secret.md
│ │ ├── 02-sql-injection.md
│ │ ├── ... (đến 21)
│ │ └── 21-command-injection.md
│ └── languages/ # Override chuyên sâu per language
│ ├── go/ # GORM, slog, Colly...
│ ├── php/ # mysqli/PDO, $_GET, eval/include, Laravel CSRF
│ └── README.md # Hướng dẫn add language mới
└── references/
├── chunking-strategy.md
├── sub-agent-prompts.md
├── language-detection.md
├── data-flow-classification.md
├── output-format.md
└── i18n/
├── vi.md
└── en.md
Thêm rule mới (cross-language): tạo file số tiếp theo trong rules/generic/, frontmatter có id, severity_max, applies_to: all. Update bảng ở Step 4 trong file này.
Thêm language specialization mới (e.g., Ruby): tạo rules/languages/ruby/<rule-id>.md với cùng id như generic — sẽ tự override. Đọc rules/languages/README.md để biết template.
Reasoning-First (cốt lõi)
DO:
- Đọc full function khi gặp pattern, KHÔNG flag luôn
- Trace nguồn dữ liệu: input → transformations → sink
- Phân loại L1-L4 trước khi flag CRITICAL
- Đọc rule file trước khi áp dụng
DON'T:
- Copy bash example chạy thẳng (đó là minh họa)
- Flag mọi
fmt.Sprintf là SQLi (chỉ flag nếu data là L1 và không parameterize)
- Bỏ qua "but" clauses (nhiều pattern legitimate)
- Skip context (1 dòng grep không đủ để verdict)
Mục tiêu là hiểu bảo mật, không phải đếm pattern.