sast-skills MCP Server

Collection of agent skills that turn your AI coder into a SAST scanner

Documentation

๐Ÿ›ก๏ธ sast-skills

Turn your LLM coding assistant into a fully featured SAST scanner.

Drop-in agent skills for 14 AI assistants โ€” Claude Code, OpenAI Codex CLI, Gemini CLI, GitHub Copilot, Cursor, Windsurf, OpenCode, Cline, Antigravity, Aider, Kilo Code, Augment Code, Hermes Agent, and Mistral Vibe.

npm version CI tests License: MIT Node.js

Claude Code with Opus is recommended for quality; any capable model works.

โšก Quick start

npx sast-skills install   # pick your assistant(s) โ€” ones found on your PATH are pre-enabled

Then open the project in your assistant and prompt:

Run vulnerability scan

It runs all four phases and writes findings to sast/. Aggregate them with npx sast-skills export --format sarif for GitHub Code Scanning or CI.


๐Ÿ“‘ Table of contents


โœจ Highlights

  • 68 skills across 64 vulnerability classes โ€” injection (incl. XPath, expression-language, CSV-formula, XML-bomb variants), broken access control, API & session depth (rate-limiting, OAuth/OIDC, session fixation, shadow routes), weak crypto, file handling, supply chain, CI/CD & cloud-metadata risks, business logic & payment abuse, agentic/MCP security (skill-config poisoning, MCP tool poisoning, config-as-execution, over-privileged agent identity), and LLM/agent runtime (excessive agency, RAG isolation, unsafe tool calling, memory poisoning, denial-of-wallet), plus a tech-stack router.
  • Four-phase orchestration โ€” reconnaissance โ†’ parallel detection โ†’ consolidated report โ†’ evidence-based triage, driven entirely from CLAUDE.md / AGENTS.md.
  • Idempotent & resumable โ€” each phase skips work whose output already exists; re-run after fixing issues to refresh only what's stale.
  • Machine-readable output โ€” every skill emits canonical JSON; sast-skills export aggregates to JSON, SARIF 2.1.0, or HTML for GitHub Code Scanning and CI.
  • Cross-assistant โ€” identical skills ship for Claude Code (.claude/skills) and every AGENTS.md assistant (.agents/skills).
  • Zero-config CLI โ€” install / update / uninstall / doctor / export, published from GitHub Actions with npm provenance (SLSA attestation).

๐Ÿ”„ Flow

The orchestrator executes four phases โ€” reconnaissance, parallel detection, synthesis, and triage:

flowchart TD
    U(["User: Run vulnerability scan"]) --> R{"CLAUDE.md / AGENTS.md orchestrator"}
    R --> S1["Step 1 โ€” sast-analysis<br/>codebase and architecture map"]
    S1 -->|sast/architecture.md| S2["Step 2 โ€” parallel vulnerability scan<br/>64 skills: recon, batched verify, merge"]
    S2 -->|sast/*-results.md and *-results.json| S3["Step 3 โ€” sast-report<br/>consolidate and rank"]
    S3 -->|sast/final-report.md| S4["Step 4 โ€” sast-triage<br/>false-positive elimination,<br/>severity adjustment with evidence"]
    S4 -->|sast/final-report-triaged.md and triaged.json| EXP["npx sast-skills export<br/>JSON, SARIF, HTML"]
    EXP --> CS(["GitHub Code Scanning, CI, dashboards"])

Every step is idempotent: if its output file already exists, the orchestrator skips it. Re-run the scan after fixing issues to refresh only what's stale.


๐Ÿ” What it detects

All skills follow the same three-phase pattern: recon โ†’ batched verify (parallel subagents, 3 per batch) โ†’ merge. Each writes a human-readable markdown report and a canonical JSON findings file that sast-skills export aggregates.

Reconnaissance & Synthesis

SkillRole
sast-analysisCodebase recon, architecture mapping, threat model
sast-stackTech-stack router โ€” picks which detection skills to run per project
sast-reportConsolidate per-class findings into a ranked report
sast-triageRemove false positives and adjust severities with codebase evidence

Injection

SkillVulnerability Class
sast-sqliSQL injection
sast-nosqlNoSQL injection (Mongo, Firestore, DynamoDB)
sast-ldapLDAP filter / DN injection
sast-graphqlUnsafe GraphQL document construction
sast-xssCross-Site Scripting
sast-sstiServer-Side Template Injection
sast-rceRemote Code Execution (command injection, eval, unsafe deserialization)
sast-xxeXML External Entity
sast-ssrfServer-Side Request Forgery
sast-openredirectOpen redirect (phishing / OAuth token theft)
sast-crlfCRLF / HTTP response splitting (header injection)
sast-ssrfimdsCloud metadata SSRF (IMDSv1 credential theft)
sast-unsafeconsumptionUnvalidated third-party API response into a sink (second-order injection)
sast-xpathXPath injection (user input into an XPath expression)
sast-csvinjFormula / CSV injection in spreadsheet exports
sast-elinjExpression-language injection (OGNL / SpEL / MVEL / JEXL)

Access control & Auth

SkillVulnerability Class
sast-idorInsecure Direct Object Reference
sast-missingauthMissing auth / broken function-level authorization
sast-jwtInsecure JWT implementations
sast-csrfCross-Site Request Forgery
sast-corsCORS misconfiguration
sast-cookieflagsMissing HttpOnly / Secure / SameSite on session cookies
sast-massassignMass assignment / overposting (privilege escalation)
sast-secheadersMissing security headers (CSP, HSTS, X-Frame-Options, SRI)
sast-ratelimitMissing rate limit on auth / expensive endpoints
sast-sessionSession fixation + low-entropy session IDs
sast-oauthOAuth / OIDC misconfig (redirect_uri, state, PKCE)
sast-routeinventoryShadow / debug / admin routes left registered
sast-postmessagepostMessage / CSWSH / reverse-tabnabbing origin trust

Files, crypto & runtime

SkillVulnerability Class
sast-pathtraversalPath / directory traversal
sast-fileuploadInsecure file upload
sast-cryptoWeak primitives, bad modes, IV reuse, weak PRNG
sast-prototypeJavaScript prototype pollution
sast-redosCatastrophic-backtracking regex DoS
sast-raceRace conditions and TOCTOU
sast-deserInsecure deserialization (gadget chains, pickle, unserialize)
sast-tlsDisabled TLS certificate / hostname verification
sast-zipslipZip Slip โ€” archive-extraction path traversal
sast-dangerousapiDangerous API sinks (eval / exec / reflection / native bridges)
sast-xmlbombXML entity-expansion DoS (billion laughs / quadratic blowup)

Data exposure & supply chain

SkillVulnerability Class
sast-hardcodedsecretsAPI keys / tokens / credentials in client-facing code
sast-piiPII and credential leakage to logs / telemetry / error pages
sast-depsKnown-vulnerable dependencies (CVE in lockfiles)
sast-iacInsecure IaC (Dockerfile / Terraform / Kubernetes / GitHub Actions)
sast-errorhandlingFail-open logic, stack-trace / secret leaks, debug mode
sast-excessivedataExcessive data exposure in API responses
sast-pipelineinjCI/CD pipeline injection (untrusted event payloads)
sast-depconfusionDependency confusion + install-time script execution
sast-lockfileMissing lockfile / unpinned dependency hashes
sast-cloudsdkCloud SDK misuse (public bucket, hardcoded key, broad IAM)

Business logic & LLM-specific

SkillVulnerability Class
sast-businesslogicPrice manipulation, workflow bypass, reward abuse
sast-paymentlogicE-commerce payment abuse (price / coupon / refund / balance race)
sast-promptinjectionUntrusted text reaching an LLM prompt (OWASP LLM #1)
sast-llmoutputUnvalidated LLM output reaching code / HTML / SQL / shell sinks (OWASP LLM #2)

Agentic & MCP security

The 2026-era attack surface most signature scanners miss โ€” natural-language manipulation, MCP tool poisoning, config-as-execution, and over-privileged agent identities. These skills use an LLM-driven verify step, not regex matching.

SkillVulnerability Class
sast-skillauditHidden-instruction / shell-sink in untrusted skill or agent config
sast-mcpsecMCP server tool poisoning + missing tool-auth
sast-configrceRepo config that auto-executes shell at project open / checkout
sast-agentidentityOver-privileged non-human (agent / CI) identity

LLM & agent runtime

Runtime behaviour of LLM and agent applications โ€” tool authority, retrieval isolation, prompt-secret hygiene, memory trust, and cost bounds. The tech-stack router gates these so they never run on codebases with no LLM/agent dependencies.

SkillVulnerability Class
sast-excessiveagencyExcessive agent authority (state-changing tools, no human-in-the-loop)
sast-toolcallingLLM tool dispatch without an allow-list / argument validation
sast-ragleakRAG cross-tenant leak + indirect injection (missing retrieval ACL filter)
sast-systempromptleakSecrets in system prompts; prompt logged or echoed to callers
sast-memorypoisonUntrusted data persisted to agent memory and later trusted
sast-llmdosUnbounded LLM tokens / agent loops (denial-of-wallet)

๐Ÿ“ฆ Installation

npx sast-skills install

The installer shows a multi-select of all 14 supported assistants (plus All of the above) and asks whether to install into the current project or your user home directory (project / global). To skip prompts, pass a comma-separated list of assistant ids (or all):

npx sast-skills install --yes --assistant claude,cursor,copilot --scope project

If your project already contains a CLAUDE.md or AGENTS.md, the installer refuses to clobber it by default โ€” back it up or pass --force.

Manual install (without npx / npm) โ€” click to expand

Use this path if you can't run npx (corporate proxy, npm registry unreachable, offline environment) or if you want to pin to this fork's source rather than the published npm package. The CLI installer only does two things โ€” drop the orchestrator entry file at the project root and mirror each skill's SKILL.md into the right hidden directory โ€” so a plain cp -R reproduces it exactly.

1. Get the bundled files

git clone https://github.com/mstfknn/sast-skills.git

Everything you need lives under sast-skills/sast-files/:

sast-files/
โ”œโ”€โ”€ CLAUDE.md                       # Orchestrator entry for Claude Code
โ”œโ”€โ”€ AGENTS.md                       # Orchestrator entry for Gemini CLI / Codex / OpenCode / Cursor
โ”œโ”€โ”€ .claude/skills/sast-*/SKILL.md  # 68 skills in Claude Code format
โ””โ”€โ”€ .agents/skills/sast-*/SKILL.md  # Same 68 skills mirrored for AGENTS.md assistants

The two skill trees are kept in sync by npm run sync โ€” content is identical, only the directory name differs.

2a. Install for Claude Code

Set SAST_SRC to the clone path so the commands below stay copy-pasteable:

export SAST_SRC=/absolute/path/to/sast-skills
cd /path/to/your-project

Project scope (recommended โ€” versioned alongside your repo):

cp "$SAST_SRC/sast-files/CLAUDE.md" ./CLAUDE.md
cp -R "$SAST_SRC/sast-files/.claude" ./

If you already use a project-level CLAUDE.md, do not overwrite it โ€” Claude Code reads only one CLAUDE.md per project. Merge the orchestrator content (the four-phase flow) into your existing file instead.

Global scope (skills available in every project; orchestrator still copied per-project):

mkdir -p ~/.claude/skills
cp -R "$SAST_SRC/sast-files/.claude/skills/." ~/.claude/skills/
# Then in any project where you want the scan flow:
cp "$SAST_SRC/sast-files/CLAUDE.md" /path/to/your-project/CLAUDE.md

2b. Install for Gemini CLI (and other AGENTS.md assistants)

export SAST_SRC=/absolute/path/to/sast-skills
cd /path/to/your-project
cp "$SAST_SRC/sast-files/AGENTS.md" ./AGENTS.md
cp -R "$SAST_SRC/sast-files/.agents" ./

Gemini CLI reads GEMINI.md; the installer writes it for you when you pick Gemini.

3. Verify the install

# Project scope (Claude Code)
ls CLAUDE.md && ls .claude/skills/ | head

# Project scope (Gemini / AGENTS.md)
ls AGENTS.md && ls .agents/skills/ | head

# Global scope (Claude Code)
ls ~/.claude/skills/ | head

You should see all 31 sast-* skill directories. Open the project in your assistant and prompt "Run vulnerability scan" โ€” the orchestrator inside CLAUDE.md / AGENTS.md drives the four phases from there.

4. Keeping a manual install up to date

cd "$SAST_SRC" && git pull
# Re-run the cp commands from Step 2a / 2b to refresh โ€” they're idempotent.

If you keep the clone around, cd "$SAST_SRC" && git pull && <rerun cp> is the manual equivalent of npx sast-skills update.

CLI commands

CommandWhat it does
npx sast-skills installCopy CLAUDE.md / AGENTS.md and the skill tree into your project or $HOME
npx sast-skills updateRefresh an existing install with the currently bundled skill files
npx sast-skills uninstallRemove installed skills; refuses to drop a modified CLAUDE.md without --force
npx sast-skills doctorVerify an install and report OK / MISSING / MODIFIED per file; exits non-zero on issues
npx sast-skills export --input sast/ --format sarif --output report.sarifAggregate sast/*-results.json into JSON, SARIF 2.1.0, or HTML
npx sast-skills export --input sast/ --triaged --format sarifPrefer the triaged sast/triaged.json over raw per-skill results
npx sast-skills --versionPrint the installed CLI version

Install-time flags

FlagPurpose
--yesNon-interactive; required when stdin is not a TTY
--assistant <ids>Comma-separated assistant ids (e.g. claude,cursor,copilot) or all
--scope <project|global>Install into ./.claude/skills/ or $HOME/.claude/skills/
--target <path>Explicit install target (overrides --scope)
--forceOverwrite a pre-existing CLAUDE.md / AGENTS.md
--dry-runPrint the file plan without writing

๐Ÿš€ Running a scan

After installing, open the project in your AI assistant and ask:

Run vulnerability scan

or

Find vulnerabilities in this codebase

The orchestrator takes over. It runs all four phases automatically, respects idempotency (re-runs only pick up what's missing), and writes everything into sast/ in your project root.

Output files

FileDescription
sast/architecture.mdTechnology stack, architecture, entry points, data flows
sast/*-results.mdPer-vulnerability-class findings (human-readable)
sast/*-results.jsonCanonical machine-readable findings (fed to sast-skills export)
sast/final-report.mdConsolidated raw report ranked by severity
sast/final-report-triaged.mdTriaged report โ€” false positives removed, severities adjusted with evidence
sast/triaged.jsonCanonical triaged findings (preferred by sast-skills export --triaged)

Finding schema

Each skill writes sast/<skill>-results.json as a bare findings list:

{
  "findings": [
    {
      "id": "sast-sqli-0001",
      "skill": "sast-sqli",
      "severity": "critical|high|medium|low|info",
      "title": "SQL injection in /api/user",
      "description": "โ€ฆ",
      "location": { "file": "src/api/user.js", "line": 42, "column": 10 },
      "remediation": "โ€ฆ"
    }
  ]
}

sast-skills export aggregates those files into one document wrapped in a run envelope โ€” { "run": { "tool": "sast-skills", "version": "<cli-version>" }, "findings": [...] } โ€” stamping the version of the CLI that produced the report. The triage step writes sast/triaged.json in that same enveloped shape.

Triaged findings add triage_status (confirmed|upgraded|downgraded|false_positive), triage_original_severity (when severity changed), and triage_evidence with concrete codebase citations.


๐Ÿ”Œ CI integrations

GitHub Code Scanning (SARIF)

Composite action at .github/actions/scan/action.yml:

- uses: mstfknn/sast-skills/.github/actions/scan@main
  with:
    input: sast/
    output: sast-skills.sarif

This runs sast-skills export --format sarif and uploads the result to Code Scanning via github/codeql-action/upload-sarif@v3.

Pre-commit hook

Copy hooks/pre-commit into .git/hooks/pre-commit to make sast-skills doctor gate every commit.

Docker

docker build -t sast-skills .
docker run --rm -v "$PWD:/work" sast-skills export --input sast/ --format sarif --output report.sarif

The bundled Dockerfile is node:20-alpine-based with sast-skills set as the entrypoint.


๐Ÿฉบ Verify & troubleshoot

# Is the install in the expected shape?
npx sast-skills doctor --target . --assistant claude

# Version check
npx sast-skills --version
npm view sast-skills version    # latest on the registry

# Upgrade
npx sast-skills update

doctor exits 0 if every bundled file in the target matches the installed version's copy, and 1 if any file is MISSING or MODIFIED. MODIFIED means the file diverged from the bundled copy โ€” expected if you edit the entry file, otherwise a signal to run update.


๐Ÿค Contributing

See CONTRIBUTING.md. Developer loop:

npm install
npm test                                  # vitest suite (TDD-guard enabled; count shown in the tests badge)
npm run sync                              # mirror .claude/skills โ†’ .agents/skills
node scripts/scaffold-skill.js sast-foo   # stub a new skill in both trees
node scripts/register-skill.js sast-foo foo "Foo" "Foo injection description"
npm run lint:md                           # markdownlint

prepublishOnly runs npm run sync && npm test โ€” a dirty mirror or a red test aborts npm publish.


๐Ÿ“„ License

MIT โ€” see LICENSE.