| name | application-security |
| description | OWASP Top 10 with code examples, SAST/DAST tools, dependency scanning, CSP headers, and input validation patterns. Use when hardening applications, reviewing security posture, or implementing defensive coding practices. |
Application Security Skill
Secure coding patterns, vulnerability prevention, and security tooling for web applications.
OWASP Top 10 (2021) with Code Examples
A01: Broken Access Control
app.get('/api/users/:id', async (req, res) => {
const user = await db.user.findUnique({ where: { id: req.params.id } });
res.json(user);
});
app.get('/api/users/:id', authenticate, async (req, res) => {
if (req.user.id !== req.params.id && req.user.role !== 'ADMIN') {
return res.status(403).json({ error: 'Forbidden' });
}
const user = await db.user.findUnique({ where: { id: req.params.id } });
res.json(user);
});
A02: Cryptographic Failures
import crypto from 'crypto';
const hash = crypto.createHash('md5').update(password).digest('hex');
import bcrypt from 'bcrypt';
const hash = await bcrypt.hash(password, 12);
const isValid = await bcrypt.compare(password, hash);
A03: Injection
const query = `SELECT * FROM users WHERE email = '${email}'`;
const user = await prisma.user.findUnique({ where: { email } });
const users = await prisma.$queryRaw`SELECT * FROM users WHERE email = ${email}`;
A07: Cross-Site Scripting (XSS)
element.innerHTML = userInput;
element.textContent = userInput;
return <div>{userInput}</div>;
return <div dangerouslySetInnerHTML={{ __html: userInput }} />;
Content Security Policy (CSP)
Recommended Headers
import { NextResponse } from 'next/server';
export function middleware(request: Request) {
const nonce = crypto.randomUUID();
const csp = [
`default-src 'self'`,
`script-src 'self' 'nonce-${nonce}'`,
`style-src 'self' 'unsafe-inline'`,
`img-src 'self' data: https:`,
`font-src 'self'`,
`connect-src 'self' https://api.example.com`,
`frame-ancestors 'none'`,
`base-uri 'self'`,
`form-action 'self'`,
].join('; ');
const response = NextResponse.next();
response.headers.set('Content-Security-Policy', csp);
response.headers.set('X-Content-Type-Options', 'nosniff');
response.headers.set('X-Frame-Options', 'DENY');
response.headers.set('Referrer-Policy', 'strict-origin-when-cross-origin');
response.headers.set('Permissions-Policy', 'camera=(), microphone=(), geolocation=()');
return response;
}
SAST/DAST Tooling
Static Analysis (SAST)
| Tool | Language | Usage |
|---|
| ESLint security plugins | JS/TS | eslint-plugin-security, @microsoft/eslint-plugin-sdl |
| Semgrep | Multi | semgrep --config=auto . |
| Bandit | Python | bandit -r src/ |
| gosec | Go | gosec ./... |
| cargo-audit | Rust | cargo audit |
Dynamic Analysis (DAST)
| Tool | Purpose | Usage |
|---|
| OWASP ZAP | Web app scanning | Proxy-based scanner, API scan mode |
| Nuclei | Vulnerability scanning | Template-based scanner |
| Burp Suite | Manual + automated | Professional penetration testing |
Dependency Scanning
npm audit
npm audit fix
pip-audit
safety check
govulncheck ./...
cargo audit
Input Validation Patterns
Server-Side Validation (Always Required)
import { z } from 'zod';
const CreateUserSchema = z.object({
email: z.string().email().max(255),
name: z.string().min(1).max(100).regex(/^[a-zA-Z\s'-]+$/),
age: z.number().int().min(0).max(150).optional(),
});
function createUser(input: unknown) {
const validated = CreateUserSchema.parse(input);
}
Rate Limiting
import rateLimit from 'express-rate-limit';
const authLimiter = rateLimit({
windowMs: 15 * 60 * 1000,
max: 5,
message: 'Too many login attempts, please try again later',
standardHeaders: true,
legacyHeaders: false,
});
app.use('/api/auth/login', authLimiter);
Security Checklist
Related Resources
~/.claude/docs/reference/checklists/security-hardening.md - Security hardening checklist
~/.claude/agents/security-auditor.md - Security audit agent
~/.claude/skills/authentication-patterns/SKILL.md - Auth patterns
Secure by default. Validate at boundaries. Defense in depth.