| name | perseus-api |
| description | Deep-dive API security analysis (REST, GraphQL, WebSocket, gRPC, OAuth, Cache) |
Perseus API Security Specialist
Context & Authorization
IMPORTANT: This skill performs deep API security analysis on the user's own codebase. This is defensive security testing to find API vulnerabilities before production deployment.
Authorization: The user owns this codebase and has explicitly requested this specialized analysis.
Multi-Language Support
| Language | Frameworks |
|---|
| JavaScript/TypeScript | Express, Fastify, Nest.js, Next.js API, Hono, Bun |
| Go | Gin, Echo, Fiber, Chi, net/http |
| PHP | Laravel, Symfony, Slim, Lumen |
| Python | FastAPI, Django REST, Flask, Starlette |
| Rust | Actix-web, Axum, Rocket, Warp |
| Java | Spring Boot, Quarkus, Micronaut |
| Ruby | Rails API, Sinatra, Grape |
| C# | ASP.NET Core, Minimal APIs |
Overview
This specialist skill performs comprehensive API security analysis covering OWASP API Security Top 10 plus advanced attack vectors.
When to Use: After /scan identifies API endpoints (REST, GraphQL, WebSocket, gRPC).
Goal: Deep-dive into API-specific vulnerabilities that generic scans might miss.
Engagement Mode Compatibility
| Mode | Specialist Behavior |
|---|
PRODUCTION_SAFE | Static/code-path validation and minimal non-disruptive checks only |
STAGING_ACTIVE | Active endpoint verification with strict throttling |
LAB_FULL | Broad dynamic API verification in isolated lab |
LAB_RED_TEAM | Controlled attack-chain simulation on synthetic/test data only |
Safety Gates (Required)
- Read
deliverables/engagement_profile.md before running active checks.
- If mode is missing, default to
PRODUCTION_SAFE.
- Abort and mark
ABORTED-SAFETY if kill-switch thresholds are exceeded.
- Never run destructive actions or unbounded API fuzzing.
OWASP API Security Top 10 Coverage
| ID | Vulnerability | Description |
|---|
| API1 | Broken Object Level Authorization | IDOR, accessing other users' resources |
| API2 | Broken Authentication | Weak auth, token issues |
| API3 | Broken Object Property Level Authorization | Mass assignment, excessive data exposure |
| API4 | Unrestricted Resource Consumption | Missing rate limits, DoS vectors |
| API5 | Broken Function Level Authorization | Admin functions accessible to users |
| API6 | Unrestricted Access to Sensitive Business Flows | Automation abuse, scraping |
| API7 | Server Side Request Forgery | SSRF via API parameters |
| API8 | Security Misconfiguration | CORS, headers, debug mode |
| API9 | Improper Inventory Management | Shadow APIs, deprecated endpoints |
| API10 | Unsafe Consumption of APIs | Third-party API trust issues |
Extended Coverage
| Category | Vulnerabilities |
|---|
| GraphQL | Introspection, batching, depth attacks, alias DoS, field suggestions |
| WebSocket | Origin bypass, message injection, auth on upgrade, CSWSH |
| OAuth/OIDC | Token leakage, PKCE bypass, redirect URI manipulation, state fixation |
| Cache | Cache poisoning, cache deception, key injection, web cache DoS |
| gRPC | Reflection enabled, missing auth, message size limits |
Execution Instructions
Step 0: Mode & Scope Alignment
- Load engagement mode, scope, and request limits from
deliverables/engagement_profile.md.
- Respect
deliverables/verification_scope.md if present.
- In
PRODUCTION_SAFE, keep checks passive-first and low-rate.
Phase 1: REST API Analysis (4 Parallel Agents)
-
BOLA/IDOR Analyst:
- "Analyze all endpoints with resource IDs. Check if ownership is verified before access."
Language-Specific Patterns:
app.get('/orders/:id', async (req, res) => {
const order = await Order.findById(req.params.id);
});
func GetOrder(c *gin.Context) {
id := c.Param("id")
order, _ := db.FindOrder(id)
}
public function show($id) {
return Order::find($id);
}
@app.get("/orders/{order_id}")
async def get_order(order_id: int):
return await Order.get(order_id)
async fn get_order(path: web::Path<i32>) -> impl Responder {
Order::find(path.into_inner())
}
-
Mass Assignment Analyst:
- "Find endpoints that accept JSON/form data and map to models. Check for unprotected fields."
Language-Specific Patterns:
User.findByIdAndUpdate(id, req.body);
c.ShouldBindJSON(&user)
$user->update($request->all());
serializer = UserSerializer(user, data=request.data)
let user: User = serde_json::from_str(&body)?;
-
Rate Limiting Analyst:
- "Check all endpoints for rate limiting. Prioritize: login, password reset, OTP, expensive operations."
Framework Rate Limiters to Check:
- Express:
express-rate-limit
- Fastify:
@fastify/rate-limit
- Go:
golang.org/x/time/rate, ulule/limiter
- PHP: Laravel
ThrottleRequests
- Python:
slowapi, Django django-ratelimit
- Rust:
actix-limitation, tower::limit
-
Response Data Analyst:
- "Check API responses for excessive data exposure. Flag: password hashes, internal IDs, PII leakage, debug info."
Phase 2: GraphQL Analysis (4 Parallel Agents)
If GraphQL detected:
-
Introspection Analyst:
- "Check if introspection is enabled in production. Document all queries, mutations, subscriptions."
Disable Introspection:
new ApolloServer({ introspection: false });
srv.AroundOperations(func(ctx context.Context, next graphql.OperationHandler) graphql.ResponseHandler {
if graphql.GetOperationContext(ctx).Operation.Operation == "introspection" { return nil }
})
schema = strawberry.Schema(query=Query, subscription=Subscription, config=StrawberryConfig(auto_camel_case=True))
-
Query Complexity Analyst:
- "Check for query depth limits, complexity limits, batch restrictions. Test for nested query attacks."
DoS Patterns:
{ user { friends { friends { friends { friends { name } } } } } }
{ a1: user(id:1) { name } a2: user(id:2) { name } ... a1000: user(id:1000) { name } }
[{ "query": "{ user(id:1) { name } }" }, { "query": "{ user(id:2) { name } }" }, ...]
-
Resolver Authorization Analyst:
- "Analyze resolver-level authorization. Check if each resolver verifies permissions."
-
Field Suggestion Analyst:
- "Check if field suggestions are enabled - can leak schema info even without introspection."
-
Subscription Security Analyst:
- "Analyze GraphQL subscriptions for security issues."
Subscription Vulnerabilities:
subscription {
orderUpdated { id status userId }
}
subscription { messageAdded { content } }
Patterns to Check:
const resolvers = {
Subscription: {
orderUpdated: {
subscribe: () => pubsub.asyncIterator(['ORDER_UPDATED'])
}
}
};
const resolvers = {
Subscription: {
orderUpdated: {
subscribe: withFilter(
() => pubsub.asyncIterator(['ORDER_UPDATED']),
(payload, variables, context) => {
return payload.orderUpdated.userId === context.user.id;
}
)
}
}
};
@strawberry.type
class Subscription:
@strawberry.subscription
async def order_updated(self, info: Info) -> AsyncGenerator[Order, None]:
-
Persisted Queries Analyst:
- "Check if persisted queries can be bypassed."
Bypass Patterns:
{ "query": "{ __schema { types { name } } }" }
{ "extensions": { "persistedQuery": { "sha256Hash": "..." } }, "query": "{ malicious }" }
Phase 3: WebSocket Analysis (3 Parallel Agents)
If WebSocket detected:
-
WebSocket Auth Analyst:
- "Check authentication on WebSocket upgrade. Verify token validation on each message."
Patterns to Check:
wss.on('connection', (ws, req) => {
});
func wsHandler(w http.ResponseWriter, r *http.Request) {
conn, _ := upgrader.Upgrade(w, r, nil)
}
-
WebSocket Origin Analyst:
- "Check Cross-Site WebSocket Hijacking (CSWSH). Verify Origin header validation."
Vulnerable Pattern:
const wss = new WebSocket.Server({ server });
wss.on('headers', (headers, req) => {
if (!allowedOrigins.includes(req.headers.origin)) {
throw new Error('Invalid origin');
}
});
-
WebSocket Message Analyst:
- "Analyze message handlers for injection. Check if messages are validated before processing."
Phase 4: OAuth/OIDC Analysis (3 Parallel Agents)
If OAuth detected:
-
OAuth Flow Analyst:
- "Check OAuth implementation for security issues."
Issues to Find:
- Missing state parameter (CSRF)
- Missing PKCE for public clients
- Token in URL fragment/query
- Redirect URI not validated strictly
-
Token Security Analyst:
- "Check token storage and transmission. Flag: tokens in localStorage, tokens in URLs, no token rotation."
-
Redirect URI Analyst:
- "Check redirect_uri validation. Test for: open redirect, subdomain takeover, path traversal."
Bypass Techniques:
https://evil.com#@legitimate.com
https://legitimate.com.evil.com
https://legitimate.com%2F@evil.com
https://legitimate.com/callback/../../../evil
Phase 5: Cache Security Analysis (2 Parallel Agents)
-
Cache Poisoning Analyst:
- "Check for web cache poisoning vulnerabilities."
Patterns:
- Unkeyed headers reflected in response
- Host header injection
- X-Forwarded-Host not validated
- Cache key vs cache content mismatch
-
Cache Deception Analyst:
- "Check for web cache deception attacks."
Vulnerable Pattern:
GET /api/account/settings.css HTTP/1.1
# If cached, attacker can retrieve victim's data
Check:
- Static extension on dynamic endpoints
- Missing Cache-Control headers
- CDN caching rules
Phase 6: HTTP Request Smuggling Analysis (3 Parallel Agents)
-
CL.TE Smuggling Analyst:
- "Check for Content-Length vs Transfer-Encoding conflicts."
Attack Pattern:
POST / HTTP/1.1
Host: vulnerable.com
Content-Length: 13
Transfer-Encoding: chunked
0
SMUGGLED
Framework Considerations:
- Node.js: Check for
http-proxy, express-http-proxy
- Go: Check reverse proxy configurations
- Python: Check
gunicorn, uvicorn behind nginx
- Java: Check Tomcat, Jetty behind load balancer
-
TE.CL Smuggling Analyst:
- "Check for Transfer-Encoding prioritization issues."
Attack Pattern:
POST / HTTP/1.1
Host: vulnerable.com
Content-Length: 3
Transfer-Encoding: chunked
8
SMUGGLED
0
-
HTTP/2 Downgrade Analyst:
- "Check for HTTP/2 to HTTP/1.1 downgrade vulnerabilities."
Issues:
- H2C smuggling (HTTP/2 cleartext)
- CRLF injection in HTTP/2 pseudo-headers
- Request splitting via HTTP/2 header manipulation
Patterns to Check:
Phase 7: API Configuration (3 Parallel Agents)
-
CORS Analyst:
- "Check CORS configuration across all frameworks."
Framework-Specific:
app.use(cors({ origin: true, credentials: true }));
c.Writer.Header().Set("Access-Control-Allow-Origin", r.Header.Get("Origin"))
'allowed_origins' => ['*'],
'supports_credentials' => true,
app.add_middleware(CORSMiddleware, allow_origins=["*"], allow_credentials=True)
-
API Versioning Analyst:
- "Identify all API versions. Check if deprecated versions are still accessible."
-
gRPC Security Analyst:
- "If gRPC used: check reflection, auth interceptors, message size limits."
Output Requirements
Create deliverables/api_security_analysis.md:
# API Security Analysis
## Summary
| Category | Endpoints | Critical | High | Medium |
|----------|-----------|----------|------|--------|
| REST | X | Y | Z | W |
| GraphQL | X | Y | Z | W |
| WebSocket | X | Y | Z | W |
| OAuth | X | Y | Z | W |
| gRPC | X | Y | Z | W |
## Language/Framework Detected
- Primary: [e.g., Node.js/Express, Go/Gin, PHP/Laravel]
- API Types: REST, GraphQL, WebSocket
## Critical Findings
### [API-001] BOLA in Order Retrieval
**Severity:** Critical
**Endpoint:** `GET /api/orders/{orderId}`
**Framework:** Express.js
**Location:** `controllers/order.js:45`
**Vulnerable Code:**
[Language-specific vulnerable code]
**Remediation:**
[Language-specific fix]
---
## GraphQL Security
| Check | Status | Risk |
|-------|--------|------|
| Introspection | ENABLED | High |
| Query Depth Limit | NONE | High |
| Complexity Limit | NONE | High |
| Batching Limit | NONE | Medium |
## WebSocket Security
| Endpoint | Origin Check | Auth on Upgrade | Message Validation |
|----------|--------------|-----------------|-------------------|
| /ws | None | None | None | CRITICAL |
## OAuth Security
| Issue | Status |
|-------|--------|
| State Parameter | Missing |
| PKCE | Not Implemented |
| Redirect URI Validation | Weak |
## Cache Security
| Issue | Vulnerable | Location |
|-------|------------|----------|
| Cache Poisoning | Yes | X-Forwarded-Host |
| Cache Deception | Yes | /api/* |
## Rate Limiting Status
| Endpoint | Limit | Status |
|----------|-------|--------|
| POST /login | None | VULNERABLE |
| POST /reset-password | None | VULNERABLE |
## CORS Configuration
| Origin | Credentials | Status |
|--------|-------------|--------|
| * | true | CRITICAL |
## Recommendations
1. Implement object-level authorization on all resource endpoints
2. Add rate limiting to authentication endpoints
3. Disable GraphQL introspection in production
4. Validate WebSocket Origin header
5. Implement PKCE for OAuth flows
Next Step: Findings feed into /exploit for verification or /report for documentation.