SQL Injection
Last updated
Last updated
URL = https://portswigger.net/web-security/all-labs
Si le damos a una de las etiquetas del filtro vemos que en la URL hace una comparacion, por lo que si hacemos lo siguiente filtrariamos todo lo que hay en la base de datos que es el proposito...
Lo que hemos añadido es lo siguiente...
Que quedaria visto desde la terminal de mysql
de la siguiente manera...
En este caso vemos una pagina web con un login y lo que tenemos es que pasarlo utilizando SQL Injection, haremos lo siguiente...
Lo que añado en el username
es...
o tambien haciendo lo siguiente sirve, ya que estamos comanetando la comparativa de la autenticacion de la password
...
Se ve de esta manera desde la consola de mysql
...
Ya que es una logica que siempre se va a cumplir 1=1
y despues lo comentamos utilizando -- -
u otro signo que sea para comentar la comparativa de la contraseña, respecto a la '
lo ponemos para que cierre cualquier logica que este haciendo mysql
...
Con esto comprobamos que somos admin
automaticamente ya que petamos la logica de mysql
, pero donde pone (admin) puedes poner lo que quieras, es por poner una palabra...
URL (Payloads genericos para este tipo de casos) = https://github.com/swisskyrepo/PayloadsAllTheThings/blob/master/SQL%20Injection/MySQL%20Injection.md
En este caso para determinar el numero de columnas que tiene la tabla en la base de datos mediante la URL seria de la siguiente manera...
Con esto lo que estamos haciendo es finalizar con la '
el input que haya antes y lo de despues es que estamos haciendo que nos marque esos 3 campos, pero si hubiera 4 se pone hasta el numero 4 y asi sucesivamente, de primeras se tiene que probar con la siguiente injeccion...
Si sale un error como Error internal server
o algun otro tipo de error del servidor o de mysql
significa que funciona pero que no tiene esos campos, por lo que habria que ir probando poco a poco hasta que nos devuelva un 200 OK
como que dimos en el clavo de los campos que tiene, una vez sabiendo los campos que tiene hacemos lo siguiente (Pongamos que solo tiene 2 columnas el servidor)
Ahora sabiendo los campos que tiene, tendremos que representar las columnas para que se nos muestre la informacion...
o
Esto utlimo se pone cuando la URL o el servidor no admite numeros para este tipo de inyecciones, por lo que se le pone un NULL
para que no haya "nada" pero que aun asi te lo muestre...
De forma representada en una base de datos de mysql
seria de la siguiente manera...
o para determinal una version o base de datos...
Con esto ya sabriamos las columnas que tiene y estaria hecho
En este caso el objetivo es ponernos la cadena 'mFdsaD' en uno de los campos de las columnas (Entendiendo que ya sabemos que hay 3 columnas o campos...) tienes que ir probando sucesivamente hasta que en uno de los campos te deje inyectarle eso y con eso sabremos que ese campo en concreto es vulnerable a inyecciones...
Con esto ya habriamos logrado con exito inyectar esa palabra en uno de los campos siendo 1 solo el unico que deja ser inyectable...
Sabiendo que es el campo 2 el que se puede inyectar probariamos ya diferentes cosas como por ejemplo listar la version...
Como el objetivo de esta otra practica es averiguar el user y password del usuario admin, haremos lo siguiente...
Sabemos que con esto podemos saber las databases
que hay en mysql
...
o
Pero si no fuera por que te lo limitara la pagina, se puede ir 1 por 1...
Y se van poniendo numeros sucesivamente dependiendo de la base de datos que quieras mostrar, para ello tendrias que ir probando varios numeros hasta que encuentres una interesante...
Con esto listamos la base de datos existentes sabiendo que hay 2 columnas...
Una vez sabiendo que y visualizando la bases de datos que nos aparecen veremos uno llamado public
por lo que ahora veremos las tablas que se encuentran dentro de public
...
Hecho esto nos apareceran 2, users
y production
...
REPRESENTATIVO:
Con esto listamos las columnas que se encuentran dentro de la tabla users
...
o
o
o
Este ultimo ejemplo es por si no admitiera concatenar la informacion de las columnas con separaciones de :
se podria poner los :
pero de manera hexadecimal y su valor seria...
Esto se puede mirar en la terminal y filtrando por los :
poniendo...
o
Pero si queremos ver el contenido de la tabla users
pero esa tabla no se encuentra en la base de datos donde estamos inyectando el codigo de mysql
tendriamos que conocer el nombre de esa base de datos y hacer lo siguiente...
o
o
Aqui nos dice casi lo mismo de antes, que saquemos las credenciales listando las columnas para registrarnos en la pagina web...
o
URL (Oracle SQL Injection) = https://github.com/swisskyrepo/PayloadsAllTheThings/blob/master/SQL%20Injection/OracleSQL%20Injection.md
Al ser en Oracle
cambian algunos detalles a la hora de hacer este tipo de inyecciones...
Nos va a dar ERROR
si ponemos lo siguiente ya que nos estamos enfrentando ante un Oracle
...
Pero como estamos en Oracle
para que nos funcione eso de aqui arriba, tendriamos que hacer lo siguiente...
Eso de from dual
es una regla que se tiene que cumplir al ser Oracle
y para hacer las otras inyecciones como lo que nos piden aqui que es ver la version y base de datos de esa pagina web hariamos lo siguiente...
Si nos vamos a la ayuda que nos da de la misma pagina de los creadores de PortSwigger
encontrariamos esto de Oracle
...
Por lo que tendriamos que cambiar uno de los NULL
por banner
y donde pone v$version
lo cambiaremos por dual
...
Info:
En este caso tenemos que saber la version de la base de datos de Microsoft
sabiendo que hay 2 columnas todavia haremos lo siguiente...
Guiandonos de la chuleta de la propia pagina en la seccion de Microsoft
pondria algo tal que asi...
Por lo que hariamos lo siguiente...
Y con esto ya habriamos resuelto esta parte...
Info:
Aqui nos pide que saquemos el usuario y contraseña de la base de datos de esta pagina, por lo que haremos lo siguiente en la URL...
Con esto sabemos que no es nada de Oracle
ni nada parecido...
Veremos que nos aparece una llamada public
por lo que haremos lo siguiente...
Y nos aparecera lo siguiente...
Por lo que elegimos esa misma...
Y nos aparecera lo siguiente...
En este caso vamos hacer los dos a la vez concatenando comandos para elegir el de password_noopnn
y el de username_fwnbqa
...
Y estaremos viendo entre los usuarios y contraseñas la de admin
, por lo que nos registramos y ya lo habriamos hecho...
URL = https://pentestmonkey.net/cheat-sheet/sql-injection/oracle-sql-injection-cheat-sheet
En este caso tenemos que logearnos como administradores
en esta pagina web...
Al ser Oracle
ya cambia la cosa en algunos aspectos a la hora de poder ver una base de datos o una tabla...
Para saber que es a Oracle
a los que nos estamos enfrentando, por lo que ahora vamos a pasar a ver las bases de datos que se encuentran en esta pagina...
Ahi mostraria todas las tablas que hay, pero si solo queremos filtrar por las tablas que sean con privilegios altos, hariamos lo siguiente...
Y aparecerian estas...
La mas curiosa es la de PETER
, si queremos filtrar por esa tabla en concreto, lo que haremos sera...
Con esto lo que hacemos es que nos muestre las tablas o columnas en las que PETER
tiene privilegios altos indicandole el owner
...
Sabiendo que USERS_NADZKF
es la que queremos ver, para ver las columnas que contiene esa tabla, lo que haremos sera lo siguiente...
Y aparecera lo siguiente...
Y para poder ver el contenido del username
y del password
haremos lo siguiente...
Veremos esto...
Por lo que nos logeamos con el de admin
y ya estaria...
En este caso te dice que tienes que rellenar todas las columnas de NULL
, por lo que haremos lo siguiente...
Veremos cuantas columnas tiene...
Con esto sabemos que si ponemos 4
ya nos devuelve un error, por lo que son 3
...
Y si ponemos eso, ya habriamos rellenado las 3 columnas que hay de NULL
por lo que ya lo tendriamos resuelto...
En este caso lo que hay que hacer es poner la cadena qnUgvU
en la columna que sea inyectable, para ver eso tendremos que ver lo siguiente...
Con esto sabemos que el NULL
del medio es inyectable, por lo que haremos lo siguiente...
Y hecho esto ya lo habriamos resuelto...
Aqui lo que necesitamos es obtener la informacion de la tabla de usuarios
para ver las credenciales de todos los usuarios
incluyendo el de admin
para poder registrarnos con el mismo...
Con esto sabemos que no es nada de Oracle
ni nada parecido...
Veremos que nos aparece una llamada public
por lo que haremos lo siguiente...
Y nos aparecera lo siguiente...
Por lo que haremos lo siguiente...
Aparecera lo siguiente...
Con esto ya nos podremos registrar como admin
en la pagina web...
En esta practica tendremos que hacer basicamente lo mismo que en las anteriores, conseguir las credenciales completas de todos los usuarios y entre ellas registrarnos como admin
...
Sabiendo que sigue la misma mecanica y que las bases de datos y tablas son iguales, para comprobarlo haremos lo siguiente...
Con esto veremos su contenido para ahora mirar lo que contiene cada columna...
Para poder ver su contenido haremos lo siguiente...
Y con esto ya podriamos registrarnos como admin
en la pagina web...
En esta practica sera mas a ciegas, ya que no aparecera lo que estas metiendo representado en la pagina como tal, como en las anteriores practicas y tendremos que conseguir las credenciales de admin
para logearnos...
Lo que nos dice el ejercicio es que se puede inyectar desde la coockie, para ello utilizaremos BurpSuit...
Dentro de la peticion de BurpSuit se puede hacer de mcuhas manera, la mas optima es automatizarlo con un script de Python
o con el propio Intruder
de BurpSuit para hacer fuerza bruta a la contraseña, pero antes de hacer todo esto hay que explicar el funcionamiento del mismo...
Como sabemos que el campo inyectable es la coockie
en especifico el campo de TrackingId
aparecera unas letras seguidas unas de otras, al final es donde se empieza a poner la inyeccion de la '
por lo que el mecanismo seria de la siguiente manera...
o
Esas 2 situaciones anteriores hacen el mismo efecto, ya que dentro de la terminal de mysql
se veria de la siguiente manera...
o
Lo que estamos haciendo aqui es concatenar comandos que en este caso se puede, por lo que estamos haciendo un ataque a ciegas por lo que tendremos que ir probando letra por letra si estamos haciendo bien la inyeccion o automatizarlo, pero antes de hacer eso para saber que vamos por buen camino, dentro de la pagina hay una palabra llamada Welcome back!
que cuando hacemos mal la inyeccion desaparece, pero cuando la hacemos bien, se mantiene visible, por lo que aprovecharemos eso para ir viendo como vamos, para hacer esas comprobaciones haremos lo siguiente...
Ahora lo que vamos hacer es ir probando poco a poco, la informacion que estamos acertando y la que no...
Como eso siempre va a ser igual deberiamos de visualizar la palabra que mencione anteriormente, si le pusieramos una b
u otra letra que no sea la a
en este caso, desapareceria ya que la comparacion que estamos haciendo no coincide...
Como sabemos que hay una tabla llamada users
y que estamos haciendo la comparacion que siempre va a ser cierta, seguiremos viendo la palabra, por lo que lo estaremos haciendo bien (Ya que dentro de la tabla mencionada tiene por ahi una a
)...
o
A lo que se refiere con username,1,1
esta haciendo una comparacion de la palabra administrator
con la primera letra de la palabra a
, que en este caso se cumple ya que es la a
...
o
Sabemos que el usuario que estamos intentando adivinar su contraseña es la del administrator
por eso hacemos eso de ahi arriba, pero lo podemos hacer de mas maneras distintas...
TODO ESTO ES POR QUE EN ESTA PRACTICA SE ESTA PRACTICANDO LO QUE SE CONOCE COMO
CONDITIONAL ERROR
POR LO QUE TENDREMOS QUE IR PROBANDO LETRA POR LETRA PARA IR SABEINDO CUANDO DA UN ERROR Y CUANDO NO (CUANDO NO DE EL ERROR SIGNFICA QUE LA LETRA ESTA...)
Ahora si queremos saber la contraseña de este usuario, haremos lo siguiente...
Donde pone password
se le pone el nombre de la tabla que contenga las contraseñas y tendremos que ir probando letra por letra con comparaciones las que te van dando fallos y las que no, pero para no hacerlo de forma manual, utilizaremos el intruder
...
Todo el request
nos lo llevamos al intruder
utilizando ^I
, dentro de el, seleccionaremos la seccion en la que pone a
ubicado en la ultima linea del codigo...
Y le daremos al boton de add §
para que se nos coloquen esos 2 simbolos a los lados de la siguiente manera...
Vamos hacer un ataque de Sniper
asi que lo dejamos por defecto, seguidamente vamos a la pestaña donde pone payload
para cargar un diccionario, en nuestro caso tendremos que sustituir la a
por todas las letras del abecedario y numeros del 0-9
para que nos saque la contraseña a fuerza bruta, cuando tengamos el diccionario algo tal que asi...
Lo cargaremos en la pestaña de payload
donde pone Payload Settings [Simple List]
le damos al boton Load...
para cargar ese diccionario .txt
y una vez hecho eso le damos al boton Start attack
para iniciar el ataque de fuerza bruta, nos tendremos que fijar en los codigos de respuesta para ir sacando la contraseña poco a poco, por ejemplo aparecera algo ral que asi...
Por lo que el que se repetira mucho es el codigo 11409
ese significa que dio error, por lo que tendremos que fijarnos solo en los codigos 11470
y asi ir cogiendo las letaras 1 por 1, hasta montar la contraseña, pero esto seria con la funcion Intruder
, si lo queremos hacer con un script de python
seria de la siguiente manera...
Antes de hacer el script para que funcione instalaremos lo siguiente...
Una vez echo eso tendremos que saber de cuantos caracteres se compone la contraseña del usuario, lo haremos de la siguiente manera...
Con esto lo que estamos haciendo es que decimos que la contraseña es mayor a 5
caracteres y como tiene mas de 5
caracteres sera correcto por lo que el Welcome back!
seguira ahi...
En este caso nos pasamos poniendo que si es mayor de 50
caracteres, por lo que probaremos con un numero menor...
Y si finalmente hacemos lo siguiente veremos que la contraseña tendra 20
caracteres, ya que si ponemos 21
nos da error, por lo que 20
seria el correcto y tambien se puede comprobar poniendo un >=
....
o
Una vez sabiendo todo esto el script quedaria de la siguiente manera...
Yo lo llame SQLI_Conditional_Response.py
Donde pone TrackingId
tendremos que poner el nuestro que aparece en la captura de BurpSuit hasta antes de la '
, donde pone session
igual, lo que nos aparezca a nosotros y por ultimo donde pone main_url
lo que contiene nuestro host
en el request
capturado añadiendole https://
...
Info:
Una vez hecho todo esto tendriamos la contraseña, la probamos y ya estaria hecha...
En esta parctica lo que debemos de hacer es parecido a la anterior pero en este caso hay que jugar con los errores de la pagina, por lo que nos tendremos que guiar por eso...
Utilizaremos el BurpSuit para capturar la peticion de la pagina y llevandolo al Repeater
con ^R
podremos ir probando hay inyecciones de comandos...
Si lo dejamos asi nos dara error por lo que nos da a entender que tiene que ser otro tipo de base de datos la que se esta manejando...
Si ponemos eso si nos va a ir, por lo que estamos trabajando con un Oracle
, ahora para seguir viendo si vamos bien haremos diferentes comprobaciones de si estamos acertando la tabla o no...
Esto nos da un 200 OK
por lo que la tabla users
existe de por si y no nos esta dando ningun fallo, asi que vamos bien...
Lo que estamos haciendo aqui es que primero va a comprobar si la linea from users where username='administrator'
es correcta (Si existe el usuario administrator
) y si es asi hace la accion de lo siguiente que es select case when (1=1) then to_char[1/0] else '' end
basicamente como existe y como 1=1
siempre va hacer la operacion de 1/0
como eso siempre va a ser un error nos dara el error 500
, pero si ponemos lo siguiente...
Vera que 2=1
no es igual y hara else ''
por lo que nos dara un 200 OK
a todo esto sabiendo que el usuario administrator
es el correcto, por que de lo contrario pasaria lo siguiente...
Si lo dejamos en ese usuario que no existe cuando lo compruebe vera el servidor que no existe, por lo que no hara la siguiente comprobacion select case when (1=1) then to_char[1/0] else '' end
y dentro de esta comprobacion da igual lo que cambiemos siempre reciviremos un 200 OK
por lo que poniendo el usuario bien y jugando con la logica de errores, podremos adivinar la contraseña...
Ahora adivinaremos la longitud de la contraseña, guiandonos por la practica anterior que tenia 20 caracteres, esta sera igual y lo comprobamos de la siguiente manera...
Nos deberia de salir un error 500
, esto significa que lo estamos haciendo bien y que la contraseña si tiene 20 caracteres, ya que lo primero que comprueba es que esto se cumpla from users where username='administrat' and length(password)>=20
para seguir haciendo lo demas select case when (1=1) then to_char[1/0] else '' end
...
Ahora lo que estamos haciendo es hacer la comprobacion para saber si estamos llendo bien a la hora de adivinar la letra, en este caso, estamos haciendo que la primera inicial del username administrator
es igual a la letra a
en este caso si es cierto, por lo que nos tendria que devolver un error 500
y tenemos que utilizar substr
para Oracle
por que en otros seria substring
...
Ahora sabiendo que eso funciona pondremos lo siguiente para que en nuestro script de python
nos descubra la contarseña...
Ahora vamos a crear el script de python3
recopilando un poco del anterior, quedaria de la siguiente manera...
Yo llame al script SQLI_Conditional_Error.py
En este caso cambiaremos la comparacion de Welcome back!
por r.status_code == 500
para que haga la comparacion del codigo de respuesta y ya lo demas que se ajuste a nuestro request
Info:
Una vez hecho esto ya lo tendriamos, nos logeamos y listo...
En esta practica hay que hacer algo parecido a lo anterior, en este caso tenemos que sacar la contraseña del admin
con inyeccion de SQL en una base de datos manejada por PostgreSQL
y dependiendo de los errores que te de sera una cosa u otra...
Para sacar la contraseña lo automatizaremos...
Yo llame a este archivo .py
SQLI_Conditional_Error_PostgreSQL.py
Y esto nos daria la contraseña del admin
, nos registramos y ya estaria listo...
Lo que hay que hacer aqui basicamente es que la web tarde en responder 10 segundos, para ello lo que haremos sera una inyeccion de mysql
con tiempos de espera (sleep
), por lo que hariamos lo siguiente...
Antes de nada vemos que se esta trabajando con una base de datos PostgreSQL
por lo que se utiliza de otra manera para mandar sleep
aunque eso se puede ir viendo haciendo pruebas cuales funionan y cuales no...
Esto hay que ubicarlo donde pone TrackingId
ya que es la zona inyectable...
Con esto lo que hacemos es que tarde en respondernos la pagina 10 segundos, por lo que ya estaria hecho...
En esta practica lo que hay que hacer es sacar la contraseña del admin
mediante sleep
jugando con esas opciones, como si fuera un true/false
...
Lo que estamos haciendo aqui es que cuando se cumpla la comparacion 1=1
(Que es siempre) se tardara la pagina en respondernos 10 segundos, pero de lo contrario (else
) tardara 0 segundos, entendiendo que existe la tabla users
con el usuario administrator
, con esto podemos comprobar que todo se cumple correctamente...
Con esto comprobamos que la contarseña tenga 20 caracteres, si va bien tardara 10 segundos en respondernos...
Con esto comprobamos que vaya esta mecanica de comprobar letras por filas, con el nombre de usuario, ya que se cumple la comparacion de que el username
administrator
empieza por la letra a
tardara 10 segundos en respondernos la pagina por lo que funciona, ahora lo mismo pero con la password
implementandolo en el script...
Ahora todo esto lo vamos a implementar en el script de python3
quedando de la siguiente manera...
Unicamente tenemos que cambiar la mecanica de la comparacion que hace r.status_code == 500
por time_end - time_start > 1.5
a parte de que añadimos time_start = time.time() y time_end = time.time()
y ya todo lo demas a lo que se nos ajuste a nuestro request
...
Info:
Con esto ya tendriamos la contraseña, por lo que nos logeamos y estaria listo...
Aqui no va a servir nada de lo anterior ya que se tiene que hacer de una manera especifica, utilizando BurpSuit e interactuando con los servidores DNS
y hacer un DNS Lookup
...
En este caso tienes que tener BurpSuit comprado para hacer la parctica ya que tendriamos que obtener el codigo del BurpSuit Colaboration Client
para poder ponerlo en el siguiente codigo, pero si se tuviera se inyectaria de la siguiente manera...
Y todo esto se tendria que URL Encodear para que funcione presionando ^U
en BurpSuit...
Quedaria algo tal que asi encodeado en URL...
En este caso es lo mismo que antes pero en modo exfiltration
, lo hariamos de la siguiente manera...
Esto habria que URL Encodearlo, por lo que tendria que quedar algo tal que asi...
Lo que tienes que hacer aqui es inyectar codigo SQL en algun campo de los que se muestra en la pagina de seleccion, por lo que con el BurpSuit capturamos la peticion estando a la escucha y dandole al boton check stock
, una vez hecho esto estariamos viendo el request
, por lo que el codigo que tendremos que inyectar sera en el siguiente campo...
Nuestro codigo lo inyectaremos en la seccion de storeId
pero no lo podemos inyectar asi a pelo...
Info:
Por que no va a funcionar, por lo que utilizaremos una cosa especial que hay que instalar en BurpSuit que nos facilitara toda esta tarea de codificarlo...
En BurpSuit nos vamos a la seccion de Extensiones
y en la casilla de BAppStore
dentro de ahi buscamos la herramienta llamada Hackvertor
y la instalamos...
Una vez instalada nos volvemos a donde tenemos el codigo, seleccionamos el codigo 1 union select NULL-- -
, click derecho, nos vamos a la opcion que pone Extensiones
, de ahi vamos a Hackvector
, nos vamos a Encode
y dentro de esta opcion habra varias codificaciones, en mi caso elijo la llamada hex_entities
, se nos quedara de esta manera...
Con esto ahora si va a funcionar ese comando y sabiendo que funciona ese comando hacemos lo siguiente...
Info:
credentials
Nos registramos y ya estaria hecho...