├── .editorconfig ├── .github └── ISSUE_TEMPLATE │ ├── bug_report.md │ ├── config.yml │ ├── feature_request.md │ └── incorrectly_parsed_graphql_schema.yml ├── .gitignore ├── .pre-commit-config.yaml ├── AUTHORS ├── BappDescription.html ├── BappManifest.bmf ├── CONTRIBUTING.md ├── LICENSE ├── README.md ├── Taskfile.yaml ├── assets └── scanner.png ├── build.gradle ├── detekt.sh ├── docs ├── doyensec_logo.svg └── inql.png ├── gradle.properties ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat ├── lib ├── graphiql │ ├── .gitignore │ ├── README.md │ ├── config │ │ ├── env.js │ │ ├── getHttpsConfig.js │ │ ├── jest │ │ │ ├── babelTransform.js │ │ │ ├── cssTransform.js │ │ │ └── fileTransform.js │ │ ├── modules.js │ │ ├── paths.js │ │ ├── webpack.config.js │ │ ├── webpack │ │ │ └── persistentCache │ │ │ │ └── createEnvironmentHash.js │ │ └── webpackDevServer.config.js │ ├── package.json │ ├── public │ │ ├── favicon.ico │ │ ├── index.html │ │ ├── logo192.png │ │ ├── logo512.png │ │ └── manifest.json │ ├── scripts │ │ ├── build.js │ │ ├── start.js │ │ └── test.js │ ├── src │ │ ├── buttons.js │ │ ├── custom.js │ │ ├── index.css │ │ ├── index.js │ │ ├── inql.js │ │ ├── intruder.svg │ │ ├── repeater.svg │ │ ├── swords-dark.png │ │ └── swords-light.png │ └── yarn.lock └── voyager │ ├── config │ ├── env.js │ ├── getHttpsConfig.js │ ├── jest │ │ ├── babelTransform.js │ │ ├── cssTransform.js │ │ └── fileTransform.js │ ├── modules.js │ ├── paths.js │ ├── webpack.config.js │ ├── webpack │ │ └── persistentCache │ │ │ └── createEnvironmentHash.js │ └── webpackDevServer.config.js │ ├── package.json │ ├── public │ ├── favicon.ico │ ├── index.html │ └── manifest.json │ ├── scripts │ ├── build.js │ ├── start.js │ └── test.js │ ├── src │ └── index.js │ └── yarn.lock ├── resources └── keywords.json ├── settings.gradle └── src └── main └── kotlin ├── burp ├── Browser.kt ├── Burp.kt └── BurpExtender.kt └── inql ├── BurpScannerCheck.kt ├── Config.kt ├── InQL.kt ├── Logger.kt ├── Profile.kt ├── ProxyRequestHighlighter.kt ├── attacker ├── Attack.kt ├── Attacker.kt └── HistoryLog.kt ├── externaltools ├── ExternalToolsRequestFixer.kt ├── ExternalToolsService.kt └── WebServer.kt ├── graphql ├── GQLQueryPrinter.kt ├── GQLSchema.kt ├── Introspection.kt ├── IntrospectionQuery.kt ├── Utils.kt ├── formatting │ ├── Formatter.kt │ ├── Style.kt │ ├── StyleMetadata.kt │ ├── SyntaxParser.kt │ ├── Token.kt │ └── Tokenizer.kt └── scanners │ ├── CyclesScanner.kt │ └── POIScanner.kt ├── savestate ├── BurpPersistedObjectList.kt ├── BurpProjectSerializable.kt └── SaveState.kt ├── scanner ├── IntrospectionCache.kt ├── ScanResult.kt ├── Scanner.kt ├── ScannerTab.kt ├── ScannerTabFactory.kt ├── scanconfig │ ├── ScanConfigView.kt │ └── ScannerFileChooser.kt └── scanresults │ ├── GQLQueryElement.kt │ ├── ScanResultElement.kt │ ├── ScanResultsContentView.kt │ ├── ScanResultsTreeNode.kt │ ├── ScanResultsTreeView.kt │ └── ScanResultsView.kt ├── ui ├── ContextMenu.kt ├── EditableTabbedPane.kt ├── GraphQLEditor.kt ├── PayloadEditor.kt ├── SettingsWindow.kt ├── StyledPayloadEditor.kt ├── Utils.kt └── WrapEditorKit.kt └── utils ├── BurpEditorTextArea.kt ├── BurpRequestHeaders.kt ├── JsonFileReader.kt └── JsonPrettifier.kt /.editorconfig: -------------------------------------------------------------------------------- 1 | [*.{kt,kts}] 2 | ktlint_code_style = intellij_idea 3 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/bug_report.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Bug report 3 | about: Create a report to help us improve 4 | title: '' 5 | labels: '' 6 | assignees: '' 7 | 8 | --- 9 | 10 | **Describe the bug** 11 | A clear and concise description of what the bug is. 12 | 13 | **To Reproduce** 14 | Steps to reproduce the behavior: 15 | 1. Go to '...' 16 | 2. Click on '....' 17 | 3. Scroll down to '....' 18 | 4. See error 19 | 20 | **Expected behavior** 21 | A clear and concise description of what you expected to happen. 22 | 23 | **Screenshots** 24 | If applicable, add screenshots to help explain your problem. 25 | 26 | **Desktop (please complete the following information):** 27 | - OS: [e.g. Ubuntu Linux 16.04, Windows 10, macOS Catalina] 28 | - Java Version: [e.g. java -v output, if applicable] 29 | - Burp Version: [eg. 2.1.0 if applicable] 30 | - Version: [e.g. 22] 31 | 32 | **Additional context** 33 | Add any other context about the problem here. 34 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/config.yml: -------------------------------------------------------------------------------- 1 | blank_issues_enabled: true 2 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/feature_request.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Feature request 3 | about: Suggest an idea for this project 4 | title: '' 5 | labels: '' 6 | assignees: '' 7 | 8 | --- 9 | 10 | **Is your feature request related to a problem? Please describe.** 11 | A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] 12 | 13 | **Describe the solution you'd like** 14 | A clear and concise description of what you want to happen. 15 | 16 | **Describe alternatives you've considered** 17 | A clear and concise description of any alternative solutions or features you've considered. 18 | 19 | **Additional context** 20 | Add any other context or screenshots about the feature request here. 21 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/incorrectly_parsed_graphql_schema.yml: -------------------------------------------------------------------------------- 1 | name: Improperly parsed GraphQL schema / Introspection errors 2 | description: Submit a GraphQL API or schema that does not get parsed or is parsed incorrectly 3 | title: "[Improper parsing]: " 4 | labels: ["bug", "parsing"] 5 | assignees: "execveat" 6 | body: 7 | - type: markdown 8 | attributes: 9 | value: | 10 | Thanks for taking the time to fill out this bug report! 11 | - type: dropdown 12 | id: version 13 | attributes: 14 | label: InQL version 15 | description: How did you install InQL (in either case make sure you're using the latest version before reporting)? 16 | options: 17 | - BApp Store 18 | - master branch 19 | - other (explain in the description) 20 | - type: input 21 | id: url 22 | attributes: 23 | label: GraphQL API 24 | description: URL of the GraphQL API (skip this if schema is provided as file) 25 | placeholder: https://.../graphql 26 | validations: 27 | required: false 28 | - type: input 29 | id: specification 30 | attributes: 31 | label: GraphQL specification version 32 | description: Please add GraphQL [specification](https://spec.graphql.org/) of the API / schema, if you know it. 33 | placeholder: unknown 34 | validations: 35 | required: false 36 | - type: textarea 37 | id: description 38 | attributes: 39 | label: What isn't working? 40 | description: | 41 | Explain, what isn't working as expected. Provide a screenshot or log files if relevant. 42 | 43 | Add schema in the attachment, if the GraphQL URL wasn't shared above. 44 | 45 | Tip: You can attach images or files by clicking this area to highlight it and then dragging files in. 46 | validations: 47 | required: true 48 | -------------------------------------------------------------------------------- /.pre-commit-config.yaml: -------------------------------------------------------------------------------- 1 | # See https://pre-commit.com for more information 2 | # See https://pre-commit.com/hooks.html for more hooks 3 | repos: 4 | - repo: https://github.com/pre-commit/pre-commit-hooks 5 | rev: v3.2.0 6 | hooks: 7 | - id: check-added-large-files 8 | - id: check-ast 9 | - id: check-case-conflict 10 | - id: check-docstring-first 11 | - id: check-executables-have-shebangs 12 | - id: check-merge-conflict 13 | - id: check-yaml 14 | - id: fix-encoding-pragma 15 | args: [--pragma, '# coding: utf-8'] 16 | - id: mixed-line-ending 17 | args: [--fix, 'no'] 18 | - id: no-commit-to-branch 19 | args: [--branch, master] 20 | - id: end-of-file-fixer 21 | - id: trailing-whitespace 22 | - repo: local 23 | hooks: 24 | - id: ktlint 25 | name: Run ktlint 26 | entry: ktlint --color --format 27 | language: system 28 | files: '\.kt$' 29 | - id: detekt 30 | name: Run detekt 31 | entry: ./detekt.sh 32 | language: script 33 | -------------------------------------------------------------------------------- /AUTHORS: -------------------------------------------------------------------------------- 1 | # This is the list of InQL's significant contributors. 2 | # 3 | # This does not necessarily list everyone who has contributed code, 4 | # especially since many employees of one corporation may be contributing. 5 | # To see the full list of contributors, see the revision history in 6 | # source control. 7 | 8 | Alex Birsan 9 | Andrea Brancaleoni 10 | Andrew Konstantinov 11 | Bartek Gorkiewicz 12 | Dominique Righetto 13 | fuomag9 14 | Hank Leininger 15 | John Villamil 16 | Kasper Karlsson 17 | Luca Carettoni 18 | Mateusz Swidniak 19 | Mathieu Déziel 20 | Matteo Oldani 21 | Paolo Stagno 22 | Rami McCarthy 23 | Savio Sisco 24 | schoobydrew 25 | -------------------------------------------------------------------------------- /BappDescription.html: -------------------------------------------------------------------------------- 1 |

This Burp extension is designed to assist in your GraphQL security testing efforts.

2 | 3 |

The main tool provided by InQL v6.0 is a customizable scanner to analyze a GraphQL endpoint or a local 4 | introspection schema file. It generates all possible queries, mutations, and subscriptions, presenting them in an organized view 5 | for thorough analysis. Scanner results can be sent to Burp's Repeater or Intruder tools for further testing.

6 | 7 |

Here are some other features that make InQL v6.0 an indispensable tool for your auditing needs:

8 | 9 | 17 | 18 |

Happy testing and stay tuned for more updates!

19 | -------------------------------------------------------------------------------- /BappManifest.bmf: -------------------------------------------------------------------------------- 1 | Uuid: 296e9a0730384be4b2fffef7b4e19b1f 2 | ExtensionType: 1 3 | Name: InQL - GraphQL Scanner 4 | RepoName: inql 5 | ScreenVersion: 6.0.0 6 | SerialVersion: 15 7 | MinPlatformVersion: 2 8 | ProOnly: False 9 | Author: Doyensec 10 | ShortDescription: InQL - A Burp Extension for GraphQL Security Testing 11 | EntryPoint: InQL.jar 12 | BuildCommand: task all 13 | -------------------------------------------------------------------------------- /CONTRIBUTING.md: -------------------------------------------------------------------------------- 1 | # Welcome to InQL contributing guide 2 | 3 | Thank you for investing your time in contributing to InQL! 4 | 5 | ## Where to send patches 6 | 7 | The development is happening in **dev** branches, so this is where you should be sending pull requests (not the default **main** branches, as they only contain released code). 8 | 9 | ## Development workflow 10 | 11 | 1. Fork the repository 12 | 2. Check out the 'dev' branch 13 | 3. Compile the existing code by following the instructions inside of the `README.md` file 14 | 4. Make changes in your fork 15 | 5. Create pull request to the 'dev' branch, explain the rationale in the message or link to an issue that is solved by PR 16 | -------------------------------------------------------------------------------- /Taskfile.yaml: -------------------------------------------------------------------------------- 1 | version: '3' 2 | 3 | env: 4 | GENERATE_SOURCEMAP: false 5 | DISABLE_ESLINT_PLUGIN: true 6 | 7 | interval: '500ms' 8 | output: prefixed 9 | 10 | tasks: 11 | web-deps: 12 | internal: true 13 | label: "Installing dependencies for {{.APP}}" 14 | prefix: "deps-{{.APP}}" 15 | dir: lib/{{.APP}} 16 | cmds: 17 | - yarn --no-progress --non-interactive --silent install --mutex file:../../yarn.mutex 18 | sources: 19 | - package.json 20 | generates: 21 | - build/index.html 22 | 23 | web-build: 24 | label: "Building web app: {{.APP}}" 25 | internal: true 26 | prefix: "build-{{.APP}}" 27 | dir: 'lib/{{.APP}}' 28 | cmds: 29 | - yarn --no-progress --non-interactive --silent build 30 | 31 | web-copy: 32 | label: "Copying web app resources: {{.APP}}" 33 | internal: true 34 | prefix: "copy-{{.APP}}" 35 | cmds: 36 | - 'rm -rf resources/static/{{.APP}}' 37 | - 'mv lib/{{.APP}}/build/ resources/static/{{.APP}}/' 38 | 39 | web-cleanup: 40 | label: "Cleaning up: {{.APP}}" 41 | internal: true 42 | prefix: "cleanup-{{.APP}}" 43 | ignore_error: true 44 | cmds: 45 | - 'rm -rf lib/{{.APP}}/node_modules/' 46 | 47 | web-item: 48 | label: "Build and copy webapp: {{.APP}}" 49 | internal: true 50 | cmds: 51 | - task: web-deps 52 | vars: { APP: '{{.APP}}' } 53 | - task: web-build 54 | vars: { APP: '{{.APP}}' } 55 | - task: web-copy 56 | vars: { APP: '{{.APP}}' } 57 | - task: web-cleanup 58 | vars: { APP: '{{.APP}}' } 59 | sources: 60 | - 'lib/{{.APP}}/src/**/*.json' 61 | - 'lib/{{.APP}}/src/**/*.js' 62 | - 'lib/{{.APP}}/src/**/*.ts' 63 | - 'lib/{{.APP}}/src/**/*.css' 64 | - 'lib/{{.APP}}/src/**/*.html' 65 | - 'lib/{{.APP}}/src/**/*.svg' 66 | - 'lib/{{.APP}}/src/**/*.png' 67 | - 'lib/{{.APP}}/src/**/*.ico' 68 | - 'lib/{{.APP}}/src/**/*.txt' 69 | - 'lib/{{.APP}}/src/**/*.md' 70 | - 'lib/{{.APP}}/src/**/*.woff2' 71 | generates: 72 | - 'resources/static/{{.APP}}/index.html' 73 | 74 | create-resources-folder: 75 | label: "Make sure resources/static/ exists" 76 | internal: true 77 | cmds: 78 | - mkdir -p resources/static/ 79 | generates: 80 | - resources/static/ 81 | status: 82 | - test -d resources/static 83 | 84 | web-apps: 85 | label: "Build web apps" 86 | internal: true 87 | deps: 88 | - task: web-item 89 | vars: { APP: 'graphiql' } 90 | - task: web-item 91 | vars: { APP: 'voyager' } 92 | 93 | 94 | web: 95 | cmds: 96 | - task: create-resources-folder 97 | - task: web-apps 98 | 99 | kotlin: 100 | label: "Build kotlin app" 101 | cmds: 102 | - ./gradlew build 103 | sources: 104 | - build.gradle 105 | - src/**/*.kt 106 | - resources/**/*.js 107 | - resources/**/*.json 108 | - resources/**/*.css 109 | - resources/**/*.html 110 | - resources/**/*.png 111 | - resources/**/*.svg 112 | - resources/**/*.ico 113 | - resources/**/*.txt 114 | - resources/**/*.md 115 | - resources/**/*.woff2 116 | generates: 117 | - InQL.jar 118 | 119 | all: 120 | label: "Run all commands" 121 | cmds: 122 | - task: web 123 | - task: kotlin 124 | 125 | default: 126 | cmds: 127 | - task: all 128 | -------------------------------------------------------------------------------- /assets/scanner.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/doyensec/inql/81bb17d2bc8bef15b57b939d47288bf11d38478a/assets/scanner.png -------------------------------------------------------------------------------- /build.gradle: -------------------------------------------------------------------------------- 1 | import org.jetbrains.kotlin.gradle.dsl.JvmTarget 2 | import org.jetbrains.kotlin.gradle.tasks.KotlinCompile 3 | 4 | buildscript { 5 | repositories { 6 | mavenCentral() 7 | } 8 | dependencies { 9 | classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:2.1.0" 10 | } 11 | } 12 | 13 | plugins { 14 | id("org.jetbrains.kotlin.jvm") version "2.1.0" 15 | } 16 | 17 | dependencies { 18 | implementation 'net.portswigger.burp.extender:burp-extender-api:2.3' 19 | implementation 'net.portswigger.burp.extensions:montoya-api:2023.5' 20 | implementation 'com.google.code.gson:gson:2.10.1' 21 | implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-core:1.7.3' 22 | implementation 'org.apache.commons:commons-text:1.11.0' 23 | implementation 'commons-codec:commons-codec:1.16.1' 24 | implementation 'io.ktor:ktor-server-core:3.1.1' 25 | implementation 'io.ktor:ktor-server-netty:3.1.1' 26 | implementation 'com.formdev:flatlaf:3.4' 27 | implementation 'com.formdev:flatlaf-extras:3.4' 28 | implementation 'com.graphql-java:graphql-java:21.5' 29 | } 30 | 31 | java { 32 | sourceCompatibility = 17 33 | targetCompatibility = 17 34 | } 35 | 36 | tasks.withType(KotlinCompile).configureEach { 37 | compilerOptions.jvmTarget.set(JvmTarget.JVM_17) 38 | } 39 | 40 | compileKotlin { 41 | compilerOptions { 42 | // Configure the Kotlin compiler options 43 | allWarningsAsErrors = true 44 | } 45 | } 46 | 47 | repositories { 48 | mavenCentral() 49 | } 50 | 51 | sourceSets { 52 | main { 53 | java { 54 | srcDir 'kotlin' 55 | } 56 | resources { 57 | srcDirs = ['resources'] 58 | } 59 | } 60 | } 61 | 62 | build { 63 | dependsOn(clean, processResources) 64 | } 65 | 66 | defaultTasks "build" 67 | 68 | // build standalone jar with bundled dependencies 69 | jar { 70 | manifest { 71 | attributes('Class-Path': configurations.compileClasspath.collect { it.getName() }.join(' ')) 72 | } 73 | from { configurations.compileClasspath.collect { it.isDirectory() ? it : zipTree(it) } } 74 | destinationDirectory = rootDir 75 | duplicatesStrategy = DuplicatesStrategy.INCLUDE 76 | archiveFileName = "InQL.jar" 77 | } 78 | -------------------------------------------------------------------------------- /detekt.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | 3 | # Collect the list of files as arguments and convert spaces to commas 4 | FILES=$(echo "$@" | tr ' ' ',' | sed 's/,$//') 5 | 6 | # Run detekt with the comma-separated file list 7 | detekt --input "$FILES" 8 | -------------------------------------------------------------------------------- /docs/inql.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/doyensec/inql/81bb17d2bc8bef15b57b939d47288bf11d38478a/docs/inql.png -------------------------------------------------------------------------------- /gradle.properties: -------------------------------------------------------------------------------- 1 | kotlin.code.style=official 2 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/doyensec/inql/81bb17d2bc8bef15b57b939d47288bf11d38478a/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | distributionBase=GRADLE_USER_HOME 2 | distributionPath=wrapper/dists 3 | distributionUrl=https\://services.gradle.org/distributions/gradle-8.3-bin.zip 4 | networkTimeout=10000 5 | validateDistributionUrl=true 6 | zipStoreBase=GRADLE_USER_HOME 7 | zipStorePath=wrapper/dists 8 | -------------------------------------------------------------------------------- /gradlew.bat: -------------------------------------------------------------------------------- 1 | @rem 2 | @rem Copyright 2015 the original author or authors. 3 | @rem 4 | @rem Licensed under the Apache License, Version 2.0 (the "License"); 5 | @rem you may not use this file except in compliance with the License. 6 | @rem You may obtain a copy of the License at 7 | @rem 8 | @rem https://www.apache.org/licenses/LICENSE-2.0 9 | @rem 10 | @rem Unless required by applicable law or agreed to in writing, software 11 | @rem distributed under the License is distributed on an "AS IS" BASIS, 12 | @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 | @rem See the License for the specific language governing permissions and 14 | @rem limitations under the License. 15 | @rem 16 | 17 | @if "%DEBUG%"=="" @echo off 18 | @rem ########################################################################## 19 | @rem 20 | @rem Gradle startup script for Windows 21 | @rem 22 | @rem ########################################################################## 23 | 24 | @rem Set local scope for the variables with windows NT shell 25 | if "%OS%"=="Windows_NT" setlocal 26 | 27 | set DIRNAME=%~dp0 28 | if "%DIRNAME%"=="" set DIRNAME=. 29 | @rem This is normally unused 30 | set APP_BASE_NAME=%~n0 31 | set APP_HOME=%DIRNAME% 32 | 33 | @rem Resolve any "." and ".." in APP_HOME to make it shorter. 34 | for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi 35 | 36 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 37 | set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" 38 | 39 | @rem Find java.exe 40 | if defined JAVA_HOME goto findJavaFromJavaHome 41 | 42 | set JAVA_EXE=java.exe 43 | %JAVA_EXE% -version >NUL 2>&1 44 | if %ERRORLEVEL% equ 0 goto execute 45 | 46 | echo. 47 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 48 | echo. 49 | echo Please set the JAVA_HOME variable in your environment to match the 50 | echo location of your Java installation. 51 | 52 | goto fail 53 | 54 | :findJavaFromJavaHome 55 | set JAVA_HOME=%JAVA_HOME:"=% 56 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe 57 | 58 | if exist "%JAVA_EXE%" goto execute 59 | 60 | echo. 61 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 62 | echo. 63 | echo Please set the JAVA_HOME variable in your environment to match the 64 | echo location of your Java installation. 65 | 66 | goto fail 67 | 68 | :execute 69 | @rem Setup the command line 70 | 71 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 72 | 73 | 74 | @rem Execute Gradle 75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* 76 | 77 | :end 78 | @rem End local scope for the variables with windows NT shell 79 | if %ERRORLEVEL% equ 0 goto mainEnd 80 | 81 | :fail 82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 83 | rem the _cmd.exe /c_ return code! 84 | set EXIT_CODE=%ERRORLEVEL% 85 | if %EXIT_CODE% equ 0 set EXIT_CODE=1 86 | if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% 87 | exit /b %EXIT_CODE% 88 | 89 | :mainEnd 90 | if "%OS%"=="Windows_NT" endlocal 91 | 92 | :omega 93 | -------------------------------------------------------------------------------- /lib/graphiql/.gitignore: -------------------------------------------------------------------------------- 1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. 2 | 3 | # dependencies 4 | /node_modules 5 | /.pnp 6 | .pnp.js 7 | 8 | # testing 9 | /coverage 10 | 11 | # production 12 | /build 13 | 14 | # misc 15 | .DS_Store 16 | .env.local 17 | .env.development.local 18 | .env.test.local 19 | .env.production.local 20 | 21 | npm-debug.log* 22 | yarn-debug.log* 23 | yarn-error.log* 24 | -------------------------------------------------------------------------------- /lib/graphiql/README.md: -------------------------------------------------------------------------------- 1 | # GraphiQL 2 | 3 | This project includes an embedded version of GraphiQL IDE, which is a third-party project and subject to its own licensing terms and conditions. Please refer to the GraphiQL IDE repository for more information on licensing and usage: https://github.com/graphql/graphiql/ 4 | -------------------------------------------------------------------------------- /lib/graphiql/config/env.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | const fs = require('fs'); 4 | const path = require('path'); 5 | const paths = require('./paths'); 6 | 7 | // Make sure that including paths.js after env.js will read .env variables. 8 | delete require.cache[require.resolve('./paths')]; 9 | 10 | const NODE_ENV = process.env.NODE_ENV; 11 | if (!NODE_ENV) { 12 | throw new Error( 13 | 'The NODE_ENV environment variable is required but was not specified.' 14 | ); 15 | } 16 | 17 | // https://github.com/bkeepers/dotenv#what-other-env-files-can-i-use 18 | const dotenvFiles = [ 19 | `${paths.dotenv}.${NODE_ENV}.local`, 20 | // Don't include `.env.local` for `test` environment 21 | // since normally you expect tests to produce the same 22 | // results for everyone 23 | NODE_ENV !== 'test' && `${paths.dotenv}.local`, 24 | `${paths.dotenv}.${NODE_ENV}`, 25 | paths.dotenv, 26 | ].filter(Boolean); 27 | 28 | // Load environment variables from .env* files. Suppress warnings using silent 29 | // if this file is missing. dotenv will never modify any environment variables 30 | // that have already been set. Variable expansion is supported in .env files. 31 | // https://github.com/motdotla/dotenv 32 | // https://github.com/motdotla/dotenv-expand 33 | dotenvFiles.forEach(dotenvFile => { 34 | if (fs.existsSync(dotenvFile)) { 35 | require('dotenv-expand')( 36 | require('dotenv').config({ 37 | path: dotenvFile, 38 | }) 39 | ); 40 | } 41 | }); 42 | 43 | // We support resolving modules according to `NODE_PATH`. 44 | // This lets you use absolute paths in imports inside large monorepos: 45 | // https://github.com/facebook/create-react-app/issues/253. 46 | // It works similar to `NODE_PATH` in Node itself: 47 | // https://nodejs.org/api/modules.html#modules_loading_from_the_global_folders 48 | // Note that unlike in Node, only *relative* paths from `NODE_PATH` are honored. 49 | // Otherwise, we risk importing Node.js core modules into an app instead of webpack shims. 50 | // https://github.com/facebook/create-react-app/issues/1023#issuecomment-265344421 51 | // We also resolve them to make sure all tools using them work consistently. 52 | const appDirectory = fs.realpathSync(process.cwd()); 53 | process.env.NODE_PATH = (process.env.NODE_PATH || '') 54 | .split(path.delimiter) 55 | .filter(folder => folder && !path.isAbsolute(folder)) 56 | .map(folder => path.resolve(appDirectory, folder)) 57 | .join(path.delimiter); 58 | 59 | // Grab NODE_ENV and REACT_APP_* environment variables and prepare them to be 60 | // injected into the application via DefinePlugin in webpack configuration. 61 | const REACT_APP = /^REACT_APP_/i; 62 | 63 | function getClientEnvironment(publicUrl) { 64 | const raw = Object.keys(process.env) 65 | .filter(key => REACT_APP.test(key)) 66 | .reduce( 67 | (env, key) => { 68 | env[key] = process.env[key]; 69 | return env; 70 | }, 71 | { 72 | // Useful for determining whether we’re running in production mode. 73 | // Most importantly, it switches React into the correct mode. 74 | NODE_ENV: process.env.NODE_ENV || 'development', 75 | // Useful for resolving the correct path to static assets in `public`. 76 | // For example, . 77 | // This should only be used as an escape hatch. Normally you would put 78 | // images into the `src` and `import` them in code to get their paths. 79 | PUBLIC_URL: publicUrl, 80 | // We support configuring the sockjs pathname during development. 81 | // These settings let a developer run multiple simultaneous projects. 82 | // They are used as the connection `hostname`, `pathname` and `port` 83 | // in webpackHotDevClient. They are used as the `sockHost`, `sockPath` 84 | // and `sockPort` options in webpack-dev-server. 85 | WDS_SOCKET_HOST: process.env.WDS_SOCKET_HOST, 86 | WDS_SOCKET_PATH: process.env.WDS_SOCKET_PATH, 87 | WDS_SOCKET_PORT: process.env.WDS_SOCKET_PORT, 88 | // Whether or not react-refresh is enabled. 89 | // It is defined here so it is available in the webpackHotDevClient. 90 | FAST_REFRESH: process.env.FAST_REFRESH !== 'false', 91 | } 92 | ); 93 | // Stringify all values so we can feed into webpack DefinePlugin 94 | const stringified = { 95 | 'process.env': Object.keys(raw).reduce((env, key) => { 96 | env[key] = JSON.stringify(raw[key]); 97 | return env; 98 | }, {}), 99 | }; 100 | 101 | return { raw, stringified }; 102 | } 103 | 104 | module.exports = getClientEnvironment; 105 | -------------------------------------------------------------------------------- /lib/graphiql/config/getHttpsConfig.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | const fs = require('fs'); 4 | const path = require('path'); 5 | const crypto = require('crypto'); 6 | const chalk = require('react-dev-utils/chalk'); 7 | const paths = require('./paths'); 8 | 9 | // Ensure the certificate and key provided are valid and if not 10 | // throw an easy to debug error 11 | function validateKeyAndCerts({ cert, key, keyFile, crtFile }) { 12 | let encrypted; 13 | try { 14 | // publicEncrypt will throw an error with an invalid cert 15 | encrypted = crypto.publicEncrypt(cert, Buffer.from('test')); 16 | } catch (err) { 17 | throw new Error( 18 | `The certificate "${chalk.yellow(crtFile)}" is invalid.\n${err.message}` 19 | ); 20 | } 21 | 22 | try { 23 | // privateDecrypt will throw an error with an invalid key 24 | crypto.privateDecrypt(key, encrypted); 25 | } catch (err) { 26 | throw new Error( 27 | `The certificate key "${chalk.yellow(keyFile)}" is invalid.\n${ 28 | err.message 29 | }` 30 | ); 31 | } 32 | } 33 | 34 | // Read file and throw an error if it doesn't exist 35 | function readEnvFile(file, type) { 36 | if (!fs.existsSync(file)) { 37 | throw new Error( 38 | `You specified ${chalk.cyan( 39 | type 40 | )} in your env, but the file "${chalk.yellow(file)}" can't be found.` 41 | ); 42 | } 43 | return fs.readFileSync(file); 44 | } 45 | 46 | // Get the https config 47 | // Return cert files if provided in env, otherwise just true or false 48 | function getHttpsConfig() { 49 | const { SSL_CRT_FILE, SSL_KEY_FILE, HTTPS } = process.env; 50 | const isHttps = HTTPS === 'true'; 51 | 52 | if (isHttps && SSL_CRT_FILE && SSL_KEY_FILE) { 53 | const crtFile = path.resolve(paths.appPath, SSL_CRT_FILE); 54 | const keyFile = path.resolve(paths.appPath, SSL_KEY_FILE); 55 | const config = { 56 | cert: readEnvFile(crtFile, 'SSL_CRT_FILE'), 57 | key: readEnvFile(keyFile, 'SSL_KEY_FILE'), 58 | }; 59 | 60 | validateKeyAndCerts({ ...config, keyFile, crtFile }); 61 | return config; 62 | } 63 | return isHttps; 64 | } 65 | 66 | module.exports = getHttpsConfig; 67 | -------------------------------------------------------------------------------- /lib/graphiql/config/jest/babelTransform.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | const babelJest = require('babel-jest').default; 4 | 5 | const hasJsxRuntime = (() => { 6 | if (process.env.DISABLE_NEW_JSX_TRANSFORM === 'true') { 7 | return false; 8 | } 9 | 10 | try { 11 | require.resolve('react/jsx-runtime'); 12 | return true; 13 | } catch (e) { 14 | return false; 15 | } 16 | })(); 17 | 18 | module.exports = babelJest.createTransformer({ 19 | presets: [ 20 | [ 21 | require.resolve('babel-preset-react-app'), 22 | { 23 | runtime: hasJsxRuntime ? 'automatic' : 'classic', 24 | }, 25 | ], 26 | ], 27 | babelrc: false, 28 | configFile: false, 29 | }); 30 | -------------------------------------------------------------------------------- /lib/graphiql/config/jest/cssTransform.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | // This is a custom Jest transformer turning style imports into empty objects. 4 | // http://facebook.github.io/jest/docs/en/webpack.html 5 | 6 | module.exports = { 7 | process() { 8 | return 'module.exports = {};'; 9 | }, 10 | getCacheKey() { 11 | // The output is always the same. 12 | return 'cssTransform'; 13 | }, 14 | }; 15 | -------------------------------------------------------------------------------- /lib/graphiql/config/jest/fileTransform.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | const path = require('path'); 4 | const camelcase = require('camelcase'); 5 | 6 | // This is a custom Jest transformer turning file imports into filenames. 7 | // http://facebook.github.io/jest/docs/en/webpack.html 8 | 9 | module.exports = { 10 | process(src, filename) { 11 | const assetFilename = JSON.stringify(path.basename(filename)); 12 | 13 | if (filename.match(/\.svg$/)) { 14 | // Based on how SVGR generates a component name: 15 | // https://github.com/smooth-code/svgr/blob/01b194cf967347d43d4cbe6b434404731b87cf27/packages/core/src/state.js#L6 16 | const pascalCaseFilename = camelcase(path.parse(filename).name, { 17 | pascalCase: true, 18 | }); 19 | const componentName = `Svg${pascalCaseFilename}`; 20 | return `const React = require('react'); 21 | module.exports = { 22 | __esModule: true, 23 | default: ${assetFilename}, 24 | ReactComponent: React.forwardRef(function ${componentName}(props, ref) { 25 | return { 26 | $$typeof: Symbol.for('react.element'), 27 | type: 'svg', 28 | ref: ref, 29 | key: null, 30 | props: Object.assign({}, props, { 31 | children: ${assetFilename} 32 | }) 33 | }; 34 | }), 35 | };`; 36 | } 37 | 38 | return `module.exports = ${assetFilename};`; 39 | }, 40 | }; 41 | -------------------------------------------------------------------------------- /lib/graphiql/config/modules.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | const fs = require('fs'); 4 | const path = require('path'); 5 | const paths = require('./paths'); 6 | const chalk = require('react-dev-utils/chalk'); 7 | const resolve = require('resolve'); 8 | 9 | /** 10 | * Get additional module paths based on the baseUrl of a compilerOptions object. 11 | * 12 | * @param {Object} options 13 | */ 14 | function getAdditionalModulePaths(options = {}) { 15 | const baseUrl = options.baseUrl; 16 | 17 | if (!baseUrl) { 18 | return ''; 19 | } 20 | 21 | const baseUrlResolved = path.resolve(paths.appPath, baseUrl); 22 | 23 | // We don't need to do anything if `baseUrl` is set to `node_modules`. This is 24 | // the default behavior. 25 | if (path.relative(paths.appNodeModules, baseUrlResolved) === '') { 26 | return null; 27 | } 28 | 29 | // Allow the user set the `baseUrl` to `appSrc`. 30 | if (path.relative(paths.appSrc, baseUrlResolved) === '') { 31 | return [paths.appSrc]; 32 | } 33 | 34 | // If the path is equal to the root directory we ignore it here. 35 | // We don't want to allow importing from the root directly as source files are 36 | // not transpiled outside of `src`. We do allow importing them with the 37 | // absolute path (e.g. `src/Components/Button.js`) but we set that up with 38 | // an alias. 39 | if (path.relative(paths.appPath, baseUrlResolved) === '') { 40 | return null; 41 | } 42 | 43 | // Otherwise, throw an error. 44 | throw new Error( 45 | chalk.red.bold( 46 | "Your project's `baseUrl` can only be set to `src` or `node_modules`." + 47 | ' Create React App does not support other values at this time.' 48 | ) 49 | ); 50 | } 51 | 52 | /** 53 | * Get webpack aliases based on the baseUrl of a compilerOptions object. 54 | * 55 | * @param {*} options 56 | */ 57 | function getWebpackAliases(options = {}) { 58 | const baseUrl = options.baseUrl; 59 | 60 | if (!baseUrl) { 61 | return {}; 62 | } 63 | 64 | const baseUrlResolved = path.resolve(paths.appPath, baseUrl); 65 | 66 | if (path.relative(paths.appPath, baseUrlResolved) === '') { 67 | return { 68 | src: paths.appSrc, 69 | }; 70 | } 71 | } 72 | 73 | /** 74 | * Get jest aliases based on the baseUrl of a compilerOptions object. 75 | * 76 | * @param {*} options 77 | */ 78 | function getJestAliases(options = {}) { 79 | const baseUrl = options.baseUrl; 80 | 81 | if (!baseUrl) { 82 | return {}; 83 | } 84 | 85 | const baseUrlResolved = path.resolve(paths.appPath, baseUrl); 86 | 87 | if (path.relative(paths.appPath, baseUrlResolved) === '') { 88 | return { 89 | '^src/(.*)$': '/src/$1', 90 | }; 91 | } 92 | } 93 | 94 | function getModules() { 95 | // Check if TypeScript is setup 96 | const hasTsConfig = fs.existsSync(paths.appTsConfig); 97 | const hasJsConfig = fs.existsSync(paths.appJsConfig); 98 | 99 | if (hasTsConfig && hasJsConfig) { 100 | throw new Error( 101 | 'You have both a tsconfig.json and a jsconfig.json. If you are using TypeScript please remove your jsconfig.json file.' 102 | ); 103 | } 104 | 105 | let config; 106 | 107 | // If there's a tsconfig.json we assume it's a 108 | // TypeScript project and set up the config 109 | // based on tsconfig.json 110 | if (hasTsConfig) { 111 | const ts = require(resolve.sync('typescript', { 112 | basedir: paths.appNodeModules, 113 | })); 114 | config = ts.readConfigFile(paths.appTsConfig, ts.sys.readFile).config; 115 | // Otherwise we'll check if there is jsconfig.json 116 | // for non TS projects. 117 | } else if (hasJsConfig) { 118 | config = require(paths.appJsConfig); 119 | } 120 | 121 | config = config || {}; 122 | const options = config.compilerOptions || {}; 123 | 124 | const additionalModulePaths = getAdditionalModulePaths(options); 125 | 126 | return { 127 | additionalModulePaths: additionalModulePaths, 128 | webpackAliases: getWebpackAliases(options), 129 | jestAliases: getJestAliases(options), 130 | hasTsConfig, 131 | }; 132 | } 133 | 134 | module.exports = getModules(); 135 | -------------------------------------------------------------------------------- /lib/graphiql/config/paths.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | const path = require('path'); 4 | const fs = require('fs'); 5 | const getPublicUrlOrPath = require('react-dev-utils/getPublicUrlOrPath'); 6 | 7 | // Make sure any symlinks in the project folder are resolved: 8 | // https://github.com/facebook/create-react-app/issues/637 9 | const appDirectory = fs.realpathSync(process.cwd()); 10 | const resolveApp = relativePath => path.resolve(appDirectory, relativePath); 11 | 12 | // We use `PUBLIC_URL` environment variable or "homepage" field to infer 13 | // "public path" at which the app is served. 14 | // webpack needs to know it to put the right