diff --git a/.gitignore b/.gitignore index b35fb6b3f..3de2b39c3 100644 --- a/.gitignore +++ b/.gitignore @@ -27,3 +27,6 @@ bin/ # Ignore Gradle build output directory build + +# Jar files +*.jar diff --git a/.vscode/launch.json b/.vscode/launch.json index 8546a7999..825ca64e4 100644 --- a/.vscode/launch.json +++ b/.vscode/launch.json @@ -41,6 +41,37 @@ "hidden": true } }, + { + "name": "Debug Language Server: Launch Extension", + "type": "extensionHost", + "request": "launch", + "runtimeExecutable": "${execPath}", + "args": ["--extensionDevelopmentPath=${workspaceFolder}/extension"], + "outFiles": ["${workspaceFolder}/extension/dist/**/*.js"], + "preLaunchTask": "Gradle: Build", + "env": { + "VSCODE_DEBUG_LANGUAGE_SERVER": "true", + "VSCODE_GRADLE_PORT": "6006" + }, + "presentation": { + "group": "debug", + "order": 4 + } + }, + { + "type": "java", + "name": "Debug Language Server: Launch Language Server", + "request": "launch", + "mainClass": "com.microsoft.gradle.GradleLanguageServer", + "projectName": "gradle-language-server", + "env": { + "VSCODE_GRADLE_PORT": "6006" + }, + "presentation": { + "group": "debug", + "order": 5 + } + }, { "name": "Test: Groovy Default", "type": "extensionHost", diff --git a/CHANGELOG.md b/CHANGELOG.md index c009e2589..132de9bd1 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,18 +4,37 @@ All notable changes to the "vscode-gradle" extension will be documented in this The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.html). +## 3.8.0 +### Added +- Support highlighting of Gradle file. [PR#960](https://github.com/microsoft/vscode-gradle/pull/960), [PR#967](https://github.com/microsoft/vscode-gradle/pull/967) +- Provide document outline of Gradle file. [PR#969](https://github.com/microsoft/vscode-gradle/pull/969) +- Show syntax diagnostics of Gradle file. [PR#962](https://github.com/microsoft/vscode-gradle/pull/962) +- Support auto completion for dependencies. [PR#970](https://github.com/microsoft/vscode-gradle/pull/970) +- Support auto completion for basic Gradle closures. [PR#971](https://github.com/microsoft/vscode-gradle/pull/971) +- Support basic projects view. [PR#1002](https://github.com/microsoft/vscode-gradle/pull/1002) + +### Changed +- Upgrade vscode requirement to `1.60.0`. [PR#997](https://github.com/microsoft/vscode-gradle/pull/997) +- Adopt the new `folder-library` icon. [PR#997](https://github.com/microsoft/vscode-gradle/pull/997) + +### Fixed +- [Bugs fixed](https://github.com/microsoft/vscode-gradle/issues?q=is%3Aissue+label%3Abug+milestone%3A3.8.0+is%3Aclosed) + ## 3.7.1 ### Fixed - Fix the `Details` and `Changelog` tabs in the marketplace page. [PR#1012](https://github.com/microsoft/vscode-gradle/pull/1012) + ## 3.7.0 ### Added - Support dependency view. [PR#887](https://github.com/microsoft/vscode-gradle/pull/887) - Support local Gradle installation. [PR#926](https://github.com/microsoft/vscode-gradle/pull/926) + ### Changed - Rename `Gradle Tasks` view to `Gradle Projects` view. - Hide `STOPPED` daemons in Gradle Daemons view by default. [PR#940](https://github.com/microsoft/vscode-gradle/pull/940) - Refine UX when there is no item in pinned tasks and recent tasks view. [PR#937](https://github.com/microsoft/vscode-gradle/pull/937) + ### Fixed - [Bugs fixed](https://github.com/microsoft/vscode-gradle/issues?q=is%3Aissue+label%3Abug+milestone%3A3.7.0+is%3Aclosed) diff --git a/extension/gradle-language-configuration.json b/extension/gradle-language-configuration.json new file mode 100644 index 000000000..532802aa3 --- /dev/null +++ b/extension/gradle-language-configuration.json @@ -0,0 +1,31 @@ +{ + "comments": { + "lineComment": "//", + "blockComment": ["/*", "*/"] + }, + "brackets": [ + ["{", "}"], + ["[", "]"], + ["(", ")"] + ], + "autoClosingPairs": [ + ["{", "}"], + ["[", "]"], + ["(", ")"], + ["\"", "\""], + ["'", "'"], + { + "open": "/**", + "close": " */", + "notIn": ["string"] + } + ], + "surroundingPairs": [ + ["{", "}"], + ["[", "]"], + ["(", ")"], + ["\"", "\""], + ["'", "'"], + ["<", ">"] + ] +} diff --git a/extension/package-lock.json b/extension/package-lock.json index 57e8c6ed8..b8612e690 100644 --- a/extension/package-lock.json +++ b/extension/package-lock.json @@ -1,35 +1,35 @@ { "name": "vscode-gradle", - "version": "3.7.0", + "version": "3.8.0", "lockfileVersion": 1, "requires": true, "dependencies": { + "@babel/code-frame": { + "version": "7.12.11", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.12.11.tgz", + "integrity": "sha512-Zt1yodBx1UcyiePMSkWnU4hPqhwq7hGi2nFL1LeA3EUl+q2LQx16MISgJ0+z7dnmgvP9QtIleuETGOiOH1RcIw==", + "dev": true, + "requires": { + "@babel/highlight": "^7.10.4" + } + }, "@babel/helper-validator-identifier": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.10.4.tgz", - "integrity": "sha512-3U9y+43hz7ZM+rzG24Qe2mufW5KhvFg/NhnNph+i9mgCtdTCtMJuI1TMkrIUiK7Ix4PYlRF9I5dhqaLYA/ADXw==", + "version": "7.15.7", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.15.7.tgz", + "integrity": "sha512-K4JvCtQqad9OY2+yTU8w+E82ywk/fe+ELNlt1G8z3bVGlZfn/hOcQQsUhGhW/N+tb3fxK800wLtKOE/aM0m72w==", "dev": true }, "@babel/highlight": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.10.4.tgz", - "integrity": "sha512-i6rgnR/YgPEQzZZnbTHHuZdlE8qyoBNalD6F+q4vAFlcMEcqmkoG+mPqJYJCo63qPf74+Y1UZsl3l6f7/RIkmA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.14.5.tgz", + "integrity": "sha512-qf9u2WFWVV0MppaL877j2dBtQIDgmidgjGk5VIMw3OadXvYaXn66U1BFlH2t4+t3i+8PhedppRv+i40ABzd+gg==", "dev": true, "requires": { - "@babel/helper-validator-identifier": "^7.10.4", + "@babel/helper-validator-identifier": "^7.14.5", "chalk": "^2.0.0", "js-tokens": "^4.0.0" }, "dependencies": { - "ansi-styles": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", - "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", - "dev": true, - "requires": { - "color-convert": "^1.9.0" - } - }, "chalk": { "version": "2.4.2", "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", @@ -41,33 +41,24 @@ "supports-color": "^5.3.0" } }, - "color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", - "dev": true, - "requires": { - "color-name": "1.1.3" - } - }, - "color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", + "escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", "dev": true } } }, "@eslint/eslintrc": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-0.4.0.tgz", - "integrity": "sha512-2ZPCc+uNbjV5ERJr+aKSPRwZgKd2z11x0EgLvb1PURmUrn9QNRXFqje0Ldq454PfAVyaJYyrDvvIKSFP4NnBog==", + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-0.4.3.tgz", + "integrity": "sha512-J6KFFz5QCYUJq3pf0mjEcCJVERbzv71PUIDczuh9JkwGEzced6CO5ADLHB1rbf/+oPBtoPfMYNOpGDzCANlbXw==", "dev": true, "requires": { "ajv": "^6.12.4", "debug": "^4.1.1", "espree": "^7.3.0", - "globals": "^12.1.0", + "globals": "^13.9.0", "ignore": "^4.0.6", "import-fresh": "^3.2.1", "js-yaml": "^3.13.1", @@ -84,16 +75,31 @@ } }, "@grpc/grpc-js": { - "version": "1.2.10", - "resolved": "https://registry.npmjs.org/@grpc/grpc-js/-/grpc-js-1.2.10.tgz", - "integrity": "sha512-wj6GkNiorWYaPiIZ767xImmw7avMMVUweTvPFg4mJWOxz2180DKwfuxhJJZ7rpc1+7D3mX/v8vJdxTuIo71Ieg==", + "version": "1.3.7", + "resolved": "https://registry.npmjs.org/@grpc/grpc-js/-/grpc-js-1.3.7.tgz", + "integrity": "sha512-CKQVuwuSPh40tgOkR7c0ZisxYRiN05PcKPW72mQL5y++qd7CwBRoaJZvU5xfXnCJDFBmS3qZGQ71Frx6Ofo2XA==", "dev": true, "requires": { - "@types/node": ">=12.12.47", - "google-auth-library": "^6.1.1", - "semver": "^6.2.0" + "@types/node": ">=12.12.47" } }, + "@humanwhocodes/config-array": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.5.0.tgz", + "integrity": "sha512-FagtKFz74XrTl7y6HCzQpwDfXP0yhxe9lHLD1UZxjvZIcbyRz8zTFF/yYNfSfzU414eDwZ1SrO0Qvtyf+wFMQg==", + "dev": true, + "requires": { + "@humanwhocodes/object-schema": "^1.2.0", + "debug": "^4.1.1", + "minimatch": "^3.0.4" + } + }, + "@humanwhocodes/object-schema": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-1.2.0.tgz", + "integrity": "sha512-wdppn25U8z/2yiaT6YGquE6X8sSv7hNMWSXYSSU1jGv/yd6XqjXgTDJ8KP4NgjTXfJ3GbRjeeb8RTV7a/VpM+w==", + "dev": true + }, "@mapbox/node-pre-gyp": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/@mapbox/node-pre-gyp/-/node-pre-gyp-1.0.5.tgz", @@ -111,15 +117,6 @@ "tar": "^6.1.0" }, "dependencies": { - "rimraf": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", - "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, "semver": { "version": "7.3.5", "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", @@ -132,35 +129,35 @@ } }, "@nodelib/fs.scandir": { - "version": "2.1.4", - "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.4.tgz", - "integrity": "sha512-33g3pMJk3bg5nXbL/+CY6I2eJDzZAni49PfJnL5fghPTggPvBd/pFNSgJsdAgWptuFu7qq/ERvOYFlhvsLTCKA==", + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", "dev": true, "requires": { - "@nodelib/fs.stat": "2.0.4", + "@nodelib/fs.stat": "2.0.5", "run-parallel": "^1.1.9" } }, "@nodelib/fs.stat": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.4.tgz", - "integrity": "sha512-IYlHJA0clt2+Vg7bccq+TzRdJvv19c2INqBSsoOLp1je7xjtr7J26+WXR72MCdvU9q1qTzIWDfhMf+DRvQJK4Q==", + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", "dev": true }, "@nodelib/fs.walk": { - "version": "1.2.6", - "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.6.tgz", - "integrity": "sha512-8Broas6vTtW4GIXTAHDoE32hnN2M5ykgCpWGbuXHQ15vEMqr23pB76e/GZcYsZCHALv50ktd24qhEyKr6wBtow==", + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", "dev": true, "requires": { - "@nodelib/fs.scandir": "2.1.4", + "@nodelib/fs.scandir": "2.1.5", "fastq": "^1.6.0" } }, "@sinonjs/commons": { - "version": "1.8.2", - "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-1.8.2.tgz", - "integrity": "sha512-sruwd86RJHdsVf/AtBoijDmUqJp3B6hF/DGC23C+JaegnDHaZyewCjoVGTdg3J0uz3Zs7NnIT05OBOmML72lQw==", + "version": "1.8.3", + "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-1.8.3.tgz", + "integrity": "sha512-xkNcLAn/wZaX14RPlwizcKicDk9G3F8m2nU3L7Ukm5zBgTwiT0wsoFAHx9Jq56fJA1z/7uKGtCRu16sOUCLIHQ==", "dev": true, "requires": { "type-detect": "4.0.8" @@ -199,18 +196,18 @@ "dev": true }, "@types/fs-extra": { - "version": "9.0.8", - "resolved": "https://registry.npmjs.org/@types/fs-extra/-/fs-extra-9.0.8.tgz", - "integrity": "sha512-bnlTVTwq03Na7DpWxFJ1dvnORob+Otb8xHyUqUWhqvz/Ksg8+JXPlR52oeMSZ37YEOa5PyccbgUNutiQdi13TA==", + "version": "9.0.13", + "resolved": "https://registry.npmjs.org/@types/fs-extra/-/fs-extra-9.0.13.tgz", + "integrity": "sha512-nEnwB++1u5lVDM2UI4c1+5R+FYaKfaAzS4OococimjVm3nQw3TuzH5UNsocrcTBbhnerblyHj4A49qXbIiZdpA==", "dev": true, "requires": { "@types/node": "*" } }, "@types/glob": { - "version": "7.1.3", - "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.1.3.tgz", - "integrity": "sha512-SEYeGAIQIQX8NN6LDKprLjbrd5dARM5EXsd8GI/A5l0apYI1fGMWgPHSe4ZKL4eozlAyI+doUE9XbYS4xCkQ1w==", + "version": "7.1.4", + "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.1.4.tgz", + "integrity": "sha512-w+LsMxKyYQm347Otw+IfBXOv9UWVjpHpCDdbBMt8Kz/xbvCYNjP+0qPh91Km3iKfSRLBB0P7fAMf0KHrPu+MyA==", "dev": true, "requires": { "@types/minimatch": "*", @@ -218,48 +215,48 @@ } }, "@types/google-protobuf": { - "version": "3.7.4", - "resolved": "https://registry.npmjs.org/@types/google-protobuf/-/google-protobuf-3.7.4.tgz", - "integrity": "sha512-6PjMFKl13cgB4kRdYtvyjKl8VVa0PXS2IdVxHhQ8GEKbxBkyJtSbaIeK1eZGjDKN7dvUh4vkOvU9FMwYNv4GQQ==", + "version": "3.15.5", + "resolved": "https://registry.npmjs.org/@types/google-protobuf/-/google-protobuf-3.15.5.tgz", + "integrity": "sha512-6bgv24B+A2bo9AfzReeg5StdiijKzwwnRflA8RLd1V4Yv995LeTmo0z69/MPbBDFSiZWdZHQygLo/ccXhMEDgw==", "dev": true }, "@types/json-schema": { - "version": "7.0.7", - "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.7.tgz", - "integrity": "sha512-cxWFQVseBm6O9Gbw1IWb8r6OS4OhSt3hPZLkFApLjM8TEXROBuQGLAH2i2gZpcXdLBIrpXuTDhH7Vbm1iXmNGA==", + "version": "7.0.9", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.9.tgz", + "integrity": "sha512-qcUXuemtEu+E5wZSJHNxUXeCZhAfXKQ41D+duX+VYPde7xyEVZci+/oXKJL13tnRs9lR2pr4fod59GT6/X1/yQ==", "dev": true }, "@types/minimatch": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.3.tgz", - "integrity": "sha512-tHq6qdbT9U1IRSGf14CL0pUlULksvY9OZ+5eEgl1N7t+OA3tGvNpxJCzuKQlsNgCVwbAs670L1vcVQi8j9HjnA==", + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.5.tgz", + "integrity": "sha512-Klz949h02Gz2uZCMGwDUSDS1YBlTdDDgbWHi+81l29tQALUtvz4rAYi5uoVhE5Lagoq6DeqAUlbrHvW/mXDgdQ==", "dev": true }, "@types/mocha": { - "version": "8.2.1", - "resolved": "https://registry.npmjs.org/@types/mocha/-/mocha-8.2.1.tgz", - "integrity": "sha512-NysN+bNqj6E0Hv4CTGWSlPzMW6vTKjDpOteycDkV4IWBsO+PU48JonrPzV9ODjiI2XrjmA05KInLgF5ivZ/YGQ==", + "version": "8.2.3", + "resolved": "https://registry.npmjs.org/@types/mocha/-/mocha-8.2.3.tgz", + "integrity": "sha512-ekGvFhFgrc2zYQoX4JeZPmVzZxw6Dtllga7iGHzfbYIYkAMUx/sAFP2GdFpLff+vdHXu5fl7WX9AT+TtqYcsyw==", "dev": true }, "@types/node": { - "version": "14.14.31", - "resolved": "https://registry.npmjs.org/@types/node/-/node-14.14.31.tgz", - "integrity": "sha512-vFHy/ezP5qI0rFgJ7aQnjDXwAMrG0KqqIH7tQG5PPv3BWBayOPIQNBjVc/P6hhdZfMx51REc6tfDNXHUio893g==", + "version": "14.17.18", + "resolved": "https://registry.npmjs.org/@types/node/-/node-14.17.18.tgz", + "integrity": "sha512-haYyibw4pbteEhkSg0xdDLAI3679L75EJ799ymVrPxOA922bPx3ML59SoDsQ//rHlvqpu+e36kcbR3XRQtFblA==", "dev": true }, "@types/sinon": { - "version": "9.0.10", - "resolved": "https://registry.npmjs.org/@types/sinon/-/sinon-9.0.10.tgz", - "integrity": "sha512-/faDC0erR06wMdybwI/uR8wEKV/E83T0k4sepIpB7gXuy2gzx2xiOjmztq6a2Y6rIGJ04D+6UU0VBmWy+4HEMA==", + "version": "9.0.11", + "resolved": "https://registry.npmjs.org/@types/sinon/-/sinon-9.0.11.tgz", + "integrity": "sha512-PwP4UY33SeeVKodNE37ZlOsR9cReypbMJOhZ7BVE0lB+Hix3efCOxiJWiE5Ia+yL9Cn2Ch72EjFTRze8RZsNtg==", "dev": true, "requires": { "@types/sinonjs__fake-timers": "*" } }, "@types/sinonjs__fake-timers": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/@types/sinonjs__fake-timers/-/sinonjs__fake-timers-6.0.2.tgz", - "integrity": "sha512-dIPoZ3g5gcx9zZEszaxLSVTvMReD3xxyyDnQUjA6IYDG9Ba2AV0otMPs+77sG9ojB4Qr2N2Vk5RnKeuA0X/0bg==", + "version": "6.0.4", + "resolved": "https://registry.npmjs.org/@types/sinonjs__fake-timers/-/sinonjs__fake-timers-6.0.4.tgz", + "integrity": "sha512-IFQTJARgMUBF+xVd2b+hIgXWrZEjND3vJtRCvIelcFB5SIXfjV4bOHbHJ0eXKh+0COrBRc8MqteKAz/j88rE0A==", "dev": true }, "@types/vscode": { @@ -269,25 +266,24 @@ "dev": true }, "@typescript-eslint/eslint-plugin": { - "version": "4.16.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-4.16.1.tgz", - "integrity": "sha512-SK777klBdlkUZpZLC1mPvyOWk9yAFCWmug13eAjVQ4/Q1LATE/NbcQL1xDHkptQkZOLnPmLUA1Y54m8dqYwnoQ==", + "version": "4.31.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-4.31.2.tgz", + "integrity": "sha512-w63SCQ4bIwWN/+3FxzpnWrDjQRXVEGiTt9tJTRptRXeFvdZc/wLiz3FQUwNQ2CVoRGI6KUWMNUj/pk63noUfcA==", "dev": true, "requires": { - "@typescript-eslint/experimental-utils": "4.16.1", - "@typescript-eslint/scope-manager": "4.16.1", - "debug": "^4.1.1", + "@typescript-eslint/experimental-utils": "4.31.2", + "@typescript-eslint/scope-manager": "4.31.2", + "debug": "^4.3.1", "functional-red-black-tree": "^1.0.1", - "lodash": "^4.17.15", - "regexpp": "^3.0.0", - "semver": "^7.3.2", - "tsutils": "^3.17.1" + "regexpp": "^3.1.0", + "semver": "^7.3.5", + "tsutils": "^3.21.0" }, "dependencies": { "semver": { - "version": "7.3.4", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.4.tgz", - "integrity": "sha512-tCfb2WLjqFAtXn4KEdxIhalnRtoKFN7nAwj0B3ZXCbQloV2tq5eDbcTmT68JJD3nRJq24/XgxtQKFIpQdtvmVw==", + "version": "7.3.5", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", + "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", "dev": true, "requires": { "lru-cache": "^6.0.0" @@ -296,66 +292,66 @@ } }, "@typescript-eslint/experimental-utils": { - "version": "4.16.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-4.16.1.tgz", - "integrity": "sha512-0Hm3LSlMYFK17jO4iY3un1Ve9x1zLNn4EM50Lia+0EV99NdbK+cn0er7HC7IvBA23mBg3P+8dUkMXy4leL33UQ==", + "version": "4.31.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-4.31.2.tgz", + "integrity": "sha512-3tm2T4nyA970yQ6R3JZV9l0yilE2FedYg8dcXrTar34zC9r6JB7WyBQbpIVongKPlhEMjhQ01qkwrzWy38Bk1Q==", "dev": true, "requires": { - "@types/json-schema": "^7.0.3", - "@typescript-eslint/scope-manager": "4.16.1", - "@typescript-eslint/types": "4.16.1", - "@typescript-eslint/typescript-estree": "4.16.1", - "eslint-scope": "^5.0.0", - "eslint-utils": "^2.0.0" + "@types/json-schema": "^7.0.7", + "@typescript-eslint/scope-manager": "4.31.2", + "@typescript-eslint/types": "4.31.2", + "@typescript-eslint/typescript-estree": "4.31.2", + "eslint-scope": "^5.1.1", + "eslint-utils": "^3.0.0" } }, "@typescript-eslint/parser": { - "version": "4.16.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-4.16.1.tgz", - "integrity": "sha512-/c0LEZcDL5y8RyI1zLcmZMvJrsR6SM1uetskFkoh3dvqDKVXPsXI+wFB/CbVw7WkEyyTKobC1mUNp/5y6gRvXg==", + "version": "4.31.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-4.31.2.tgz", + "integrity": "sha512-EcdO0E7M/sv23S/rLvenHkb58l3XhuSZzKf6DBvLgHqOYdL6YFMYVtreGFWirxaU2mS1GYDby3Lyxco7X5+Vjw==", "dev": true, "requires": { - "@typescript-eslint/scope-manager": "4.16.1", - "@typescript-eslint/types": "4.16.1", - "@typescript-eslint/typescript-estree": "4.16.1", - "debug": "^4.1.1" + "@typescript-eslint/scope-manager": "4.31.2", + "@typescript-eslint/types": "4.31.2", + "@typescript-eslint/typescript-estree": "4.31.2", + "debug": "^4.3.1" } }, "@typescript-eslint/scope-manager": { - "version": "4.16.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-4.16.1.tgz", - "integrity": "sha512-6IlZv9JaurqV0jkEg923cV49aAn8V6+1H1DRfhRcvZUrptQ+UtSKHb5kwTayzOYTJJ/RsYZdcvhOEKiBLyc0Cw==", + "version": "4.31.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-4.31.2.tgz", + "integrity": "sha512-2JGwudpFoR/3Czq6mPpE8zBPYdHWFGL6lUNIGolbKQeSNv4EAiHaR5GVDQaLA0FwgcdcMtRk+SBJbFGL7+La5w==", "dev": true, "requires": { - "@typescript-eslint/types": "4.16.1", - "@typescript-eslint/visitor-keys": "4.16.1" + "@typescript-eslint/types": "4.31.2", + "@typescript-eslint/visitor-keys": "4.31.2" } }, "@typescript-eslint/types": { - "version": "4.16.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-4.16.1.tgz", - "integrity": "sha512-nnKqBwMgRlhzmJQF8tnFDZWfunXmJyuXj55xc8Kbfup4PbkzdoDXZvzN8//EiKR27J6vUSU8j4t37yUuYPiLqA==", + "version": "4.31.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-4.31.2.tgz", + "integrity": "sha512-kWiTTBCTKEdBGrZKwFvOlGNcAsKGJSBc8xLvSjSppFO88AqGxGNYtF36EuEYG6XZ9vT0xX8RNiHbQUKglbSi1w==", "dev": true }, "@typescript-eslint/typescript-estree": { - "version": "4.16.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-4.16.1.tgz", - "integrity": "sha512-m8I/DKHa8YbeHt31T+UGd/l8Kwr0XCTCZL3H4HMvvLCT7HU9V7yYdinTOv1gf/zfqNeDcCgaFH2BMsS8x6NvJg==", + "version": "4.31.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-4.31.2.tgz", + "integrity": "sha512-ieBq8U9at6PvaC7/Z6oe8D3czeW5d//Fo1xkF/s9394VR0bg/UaMYPdARiWyKX+lLEjY3w/FNZJxitMsiWv+wA==", "dev": true, "requires": { - "@typescript-eslint/types": "4.16.1", - "@typescript-eslint/visitor-keys": "4.16.1", - "debug": "^4.1.1", - "globby": "^11.0.1", + "@typescript-eslint/types": "4.31.2", + "@typescript-eslint/visitor-keys": "4.31.2", + "debug": "^4.3.1", + "globby": "^11.0.3", "is-glob": "^4.0.1", - "semver": "^7.3.2", - "tsutils": "^3.17.1" + "semver": "^7.3.5", + "tsutils": "^3.21.0" }, "dependencies": { "semver": { - "version": "7.3.4", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.4.tgz", - "integrity": "sha512-tCfb2WLjqFAtXn4KEdxIhalnRtoKFN7nAwj0B3ZXCbQloV2tq5eDbcTmT68JJD3nRJq24/XgxtQKFIpQdtvmVw==", + "version": "7.3.5", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", + "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", "dev": true, "requires": { "lru-cache": "^6.0.0" @@ -364,12 +360,12 @@ } }, "@typescript-eslint/visitor-keys": { - "version": "4.16.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-4.16.1.tgz", - "integrity": "sha512-s/aIP1XcMkEqCNcPQtl60ogUYjSM8FU2mq1O7y5cFf3Xcob1z1iXWNB6cC43Op+NGRTFgGolri6s8z/efA9i1w==", + "version": "4.31.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-4.31.2.tgz", + "integrity": "sha512-PrBId7EQq2Nibns7dd/ch6S6/M4/iwLM9McbgeEbCXfxdwRUNxJ4UNreJ6Gh3fI2GNKNrWnQxKL7oCPmngKBug==", "dev": true, "requires": { - "@typescript-eslint/types": "4.16.1", + "@typescript-eslint/types": "4.31.2", "eslint-visitor-keys": "^2.0.0" } }, @@ -389,28 +385,6 @@ "https-proxy-agent": "^5.0.0", "rimraf": "^3.0.2", "unzipper": "^0.10.11" - }, - "dependencies": { - "http-proxy-agent": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz", - "integrity": "sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==", - "dev": true, - "requires": { - "@tootallnate/once": "1", - "agent-base": "6", - "debug": "4" - } - }, - "rimraf": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", - "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - } } }, "@webassemblyjs/ast": { @@ -606,15 +580,6 @@ "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", "dev": true }, - "abort-controller": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz", - "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==", - "dev": true, - "requires": { - "event-target-shim": "^5.0.0" - } - }, "acorn": { "version": "7.4.1", "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", @@ -622,9 +587,9 @@ "dev": true }, "acorn-jsx": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.1.tgz", - "integrity": "sha512-K0Ptm/47OKfQRpNQ2J/oIN/3QYiK6FwW+eJbILhsdxh2WTLdl+30o8aGdTbm5JbffpFFAg/g+zi1E+jvJha5ng==", + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", "dev": true }, "agent-base": { @@ -667,24 +632,24 @@ "dev": true }, "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", "dev": true }, "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", "dev": true, "requires": { - "color-convert": "^2.0.1" + "color-convert": "^1.9.0" } }, "anymatch": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", - "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", + "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", "dev": true, "requires": { "normalize-path": "^3.0.0", @@ -709,9 +674,9 @@ "dev": true }, "are-we-there-yet": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.5.tgz", - "integrity": "sha512-5hYdAkZlcG8tOLujVDTgCT+uPX0VnpAH28gWsLfzpXYm7wP6mp5Q/gYyR7YQ0cKVJcXJnl3j2kpBan13PtQf6w==", + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.7.tgz", + "integrity": "sha512-nxwy40TuMiUGqMyRHgCSWZ9FM4VAoRP4xUYSTv5ImRog+h9yISPbVH7H8fASCIzYn9wlEv4zvFL7uKDMCFQm3g==", "dev": true, "requires": { "delegates": "^1.0.0", @@ -757,12 +722,6 @@ "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", "dev": true }, - "arrify": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/arrify/-/arrify-2.0.1.tgz", - "integrity": "sha512-3duEwti880xqi4eAMN8AyR4a0ByT90zoYdLlevfrvU43vb0YZwZVfxOgxWrLXXXpyugL0hNZc9G6BiB5B3nUug==", - "dev": true - }, "asn1.js": { "version": "5.4.1", "resolved": "https://registry.npmjs.org/asn1.js/-/asn1.js-5.4.1.tgz", @@ -844,13 +803,6 @@ "requires": { "semver": "^5.3.0", "shimmer": "^1.1.0" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" - } } }, "at-least-node": { @@ -866,9 +818,9 @@ "dev": true }, "azure-devops-node-api": { - "version": "10.2.2", - "resolved": "https://registry.npmjs.org/azure-devops-node-api/-/azure-devops-node-api-10.2.2.tgz", - "integrity": "sha512-4TVv2X7oNStT0vLaEfExmy3J4/CzfuXolEcQl/BRUmvGySqKStTG2O55/hUQ0kM7UJlZBLgniM0SBq4d/WkKow==", + "version": "11.0.1", + "resolved": "https://registry.npmjs.org/azure-devops-node-api/-/azure-devops-node-api-11.0.1.tgz", + "integrity": "sha512-YMdjAw9l5p/6leiyIloxj3k7VIvYThKjvqgiQn88r3nhT93ENwsoDS3A83CyJ4uTWzCZ5f5jCi6c27rTU5Pz+A==", "dev": true, "requires": { "tunnel": "0.0.6", @@ -876,10 +828,9 @@ } }, "balanced-match": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", - "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", - "dev": true + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" }, "base": { "version": "0.11.2", @@ -943,9 +894,9 @@ "dev": true }, "big-integer": { - "version": "1.6.48", - "resolved": "https://registry.npmjs.org/big-integer/-/big-integer-1.6.48.tgz", - "integrity": "sha512-j51egjPa7/i+RdiRuJbPdJ2FIUYYPhvYLjzoYbcMMm62ooO6F94fETG4MTs46zPAF9Brs04OajboA/qTGuz78w==", + "version": "1.6.49", + "resolved": "https://registry.npmjs.org/big-integer/-/big-integer-1.6.49.tgz", + "integrity": "sha512-KJ7VhqH+f/BOt9a3yMwJNmcZjG53ijWMTjSAGMveQWyLwqIiwkjNP5PFgDob3Snnx86SjDj6I89fIbv0dkQeNw==", "dev": true }, "big.js": { @@ -954,12 +905,6 @@ "integrity": "sha512-vyL2OymJxmarO8gxMr0mhChsO9QGwhynfuu4+MHTAW6czfq9humCB7rKpUjDd9YUiDPU4mzpyupFSvOClAwbmQ==", "dev": true }, - "bignumber.js": { - "version": "9.0.1", - "resolved": "https://registry.npmjs.org/bignumber.js/-/bignumber.js-9.0.1.tgz", - "integrity": "sha512-IdZR9mh6ahOBv/hYGiXyVuyCetmGJhtYkqLBpTStdhEGjegpPlUawydyaF3pbIOFynJTpllEs+NP+CS9jKFLjA==", - "dev": true - }, "binary": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/binary/-/binary-0.3.0.tgz", @@ -977,9 +922,9 @@ "dev": true }, "bluebird": { - "version": "3.7.2", - "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", - "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==", + "version": "3.4.7", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.4.7.tgz", + "integrity": "sha1-9y12C+Cbf3bQjtj66Ysomo0F+rM=", "dev": true }, "bn.js": { @@ -998,7 +943,6 @@ "version": "1.1.11", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", - "dev": true, "requires": { "balanced-match": "^1.0.0", "concat-map": "0.0.1" @@ -1099,6 +1043,12 @@ "string_decoder": "^1.1.1", "util-deprecate": "^1.0.1" } + }, + "safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "dev": true } } }, @@ -1128,12 +1078,6 @@ "integrity": "sha1-DTM+PwDqxQqhRUq9MO+MKl2ackI=", "dev": true }, - "buffer-equal-constant-time": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", - "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=", - "dev": true - }, "buffer-from": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", @@ -1187,6 +1131,12 @@ "y18n": "^4.0.0" }, "dependencies": { + "bluebird": { + "version": "3.7.2", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", + "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==", + "dev": true + }, "chownr": { "version": "1.1.4", "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", @@ -1202,13 +1152,13 @@ "yallist": "^3.0.2" } }, - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", "dev": true, "requires": { - "minimist": "^1.2.5" + "glob": "^7.1.3" } }, "y18n": { @@ -1274,15 +1224,39 @@ } }, "chalk": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", - "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", "dev": true, "requires": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" }, "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, "has-flag": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", @@ -1301,30 +1275,39 @@ } }, "cheerio": { - "version": "1.0.0-rc.6", - "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.0.0-rc.6.tgz", - "integrity": "sha512-hjx1XE1M/D5pAtMgvWwE21QClmAEeGHOIDfycgmndisdNgI6PE1cGRQkMGBcsbUbmEQyWu5PJLUcAOjtQS8DWw==", + "version": "1.0.0-rc.10", + "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.0.0-rc.10.tgz", + "integrity": "sha512-g0J0q/O6mW8z5zxQ3A8E8J1hUgp4SMOvEoW/x84OwyHKe/Zccz83PVT4y5Crcr530FV6NgmKI1qvGTKVl9XXVw==", "dev": true, "requires": { - "cheerio-select": "^1.3.0", - "dom-serializer": "^1.3.1", - "domhandler": "^4.1.0", + "cheerio-select": "^1.5.0", + "dom-serializer": "^1.3.2", + "domhandler": "^4.2.0", "htmlparser2": "^6.1.0", "parse5": "^6.0.1", - "parse5-htmlparser2-tree-adapter": "^6.0.1" + "parse5-htmlparser2-tree-adapter": "^6.0.1", + "tslib": "^2.2.0" + }, + "dependencies": { + "tslib": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.3.1.tgz", + "integrity": "sha512-77EbyPPpMz+FRFRuAFlWMtmgUWGe9UOG2Z25NqCwiIjRhOf5iKGuzSe5P2w1laq+FkRy4p+PCuVkJSGkzTEKVw==", + "dev": true + } } }, "cheerio-select": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/cheerio-select/-/cheerio-select-1.4.0.tgz", - "integrity": "sha512-sobR3Yqz27L553Qa7cK6rtJlMDbiKPdNywtR95Sj/YgfpLfy0u6CGJuaBKe5YE/vTc23SCRKxWSdlon/w6I/Ew==", + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/cheerio-select/-/cheerio-select-1.5.0.tgz", + "integrity": "sha512-qocaHPv5ypefh6YNxvnbABM07KMxExbtbfuJoIie3iZXX1ERwYmJcIiRrr9H05ucQP1k28dav8rpdDgjQd8drg==", "dev": true, "requires": { - "css-select": "^4.1.2", - "css-what": "^5.0.0", + "css-select": "^4.1.3", + "css-what": "^5.0.1", "domelementtype": "^2.2.0", "domhandler": "^4.2.0", - "domutils": "^2.6.0" + "domutils": "^2.7.0" } }, "chokidar": { @@ -1407,13 +1390,6 @@ "async-hook-jl": "^1.7.6", "emitter-listener": "^1.0.1", "semver": "^5.4.1" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" - } } }, "code-point-at": { @@ -1433,24 +1409,24 @@ } }, "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", "dev": true, "requires": { - "color-name": "~1.1.4" + "color-name": "1.1.3" } }, "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", "dev": true }, "commander": { - "version": "2.20.3", - "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", - "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/commander/-/commander-6.2.1.tgz", + "integrity": "sha512-U7VdrJFnJgo4xjrHpTzu0yrHPGImdsmD95ZlgYSEajAn2JKzDhDTPG9kBTefmObL2w/ngeZnilk+OV9CG3d7UA==", "dev": true }, "commondir": { @@ -1468,8 +1444,7 @@ "concat-map": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", - "dev": true + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" }, "concat-stream": { "version": "1.6.2", @@ -1524,13 +1499,13 @@ "run-queue": "^1.0.0" }, "dependencies": { - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", "dev": true, "requires": { - "minimist": "^1.2.5" + "glob": "^7.1.3" } } } @@ -1542,9 +1517,9 @@ "dev": true }, "core-util-is": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", - "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz", + "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==", "dev": true }, "create-ecdh": { @@ -1623,9 +1598,9 @@ } }, "css-select": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/css-select/-/css-select-4.1.2.tgz", - "integrity": "sha512-nu5ye2Hg/4ISq4XqdLY2bEatAcLIdt3OYGFc9Tm9n7VSlFBcfRv0gBNksHRgSdUDQGtN3XrZ94ztW+NfzkFSUw==", + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/css-select/-/css-select-4.1.3.tgz", + "integrity": "sha512-gT3wBNd9Nj49rAbmtFHj1cljIAOLYSX1nZ8CB7TBO3INYckygm5B7LISU/szY//YmdiSLbJvDLOx9VnMVpMBxA==", "dev": true, "requires": { "boolbase": "^1.0.0", @@ -1648,9 +1623,9 @@ "dev": true }, "debug": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", - "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.2.tgz", + "integrity": "sha512-mOp8wKcvj7XxC78zLgw/ZA+6TSgkoE2C/ienthhRD298T7UNwAg9diBpLRxC0mOezLl4B0xV7M0cCO6P/O0Xhw==", "dev": true, "requires": { "ms": "2.1.2" @@ -1669,9 +1644,9 @@ "dev": true }, "deep-is": { - "version": "0.1.3", - "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz", - "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=", + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", "dev": true }, "define-property": { @@ -1755,13 +1730,6 @@ "integrity": "sha1-zJmvlhLCP7H/8TYSxy8sv6qNWhc=", "requires": { "semver": "^5.3.0" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" - } } }, "diagnostic-channel-publishers": { @@ -1813,13 +1781,13 @@ } }, "dom-serializer": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-1.3.1.tgz", - "integrity": "sha512-Pv2ZluG5ife96udGgEDovOOOA5UELkltfJpnIExPrAk1LTvecolUGn6lIaoLh86d83GiB86CjzciMd9BuRB71Q==", + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-1.3.2.tgz", + "integrity": "sha512-5c54Bk5Dw4qAxNOI1pFEizPSjVsx5+bpJKmL2kPn8JhBUq2q09tTCa3mjijun2NfK78NMouDYNMBkOrPZiS+ig==", "dev": true, "requires": { "domelementtype": "^2.0.1", - "domhandler": "^4.0.0", + "domhandler": "^4.2.0", "entities": "^2.0.0" } }, @@ -1836,18 +1804,18 @@ "dev": true }, "domhandler": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-4.2.0.tgz", - "integrity": "sha512-zk7sgt970kzPks2Bf+dwT/PLzghLnsivb9CcxkvR8Mzr66Olr0Ofd8neSbglHJHaHa2MadfoSdNlKYAaafmWfA==", + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-4.2.2.tgz", + "integrity": "sha512-PzE9aBMsdZO8TK4BnuJwH0QT41wgMbRzuZrHUcpYncEjmQazq8QEaBWgLG7ZyC/DAZKEgglpIA6j4Qn/HmxS3w==", "dev": true, "requires": { "domelementtype": "^2.2.0" } }, "domutils": { - "version": "2.6.0", - "resolved": "https://registry.npmjs.org/domutils/-/domutils-2.6.0.tgz", - "integrity": "sha512-y0BezHuy4MDYxh6OvolXYsH+1EMGmFbwv5FKW7ovwMG6zTPWqNPq3WF9ayZssFq+UlKdffGLbOEaghNdaOm1WA==", + "version": "2.8.0", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-2.8.0.tgz", + "integrity": "sha512-w96Cjofp72M5IIhpjgobBimYEfoPjx1Vx0BSX9P30WBdZW2WIKU0T1Bd0kz2eNZ9ikjKgHbEyKx8BB6H1L3h3A==", "dev": true, "requires": { "dom-serializer": "^1.0.1", @@ -1876,15 +1844,6 @@ "stream-shift": "^1.0.0" } }, - "ecdsa-sig-formatter": { - "version": "1.0.11", - "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", - "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", - "dev": true, - "requires": { - "safe-buffer": "^5.0.1" - } - }, "elliptic": { "version": "6.5.4", "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.4.tgz", @@ -1979,35 +1938,38 @@ "dev": true }, "escape-string-regexp": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", - "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", "dev": true }, "eslint": { - "version": "7.21.0", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-7.21.0.tgz", - "integrity": "sha512-W2aJbXpMNofUp0ztQaF40fveSsJBjlSCSWpy//gzfTvwC+USs/nceBrKmlJOiM8r1bLwP2EuYkCqArn/6QTIgg==", + "version": "7.32.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-7.32.0.tgz", + "integrity": "sha512-VHZ8gX+EDfz+97jGcgyGCyRia/dPOd6Xh9yPv8Bl1+SoaIwD+a/vlrOmGRUyOYu7MwUhc7CxqeaDZU13S4+EpA==", "dev": true, "requires": { "@babel/code-frame": "7.12.11", - "@eslint/eslintrc": "^0.4.0", + "@eslint/eslintrc": "^0.4.3", + "@humanwhocodes/config-array": "^0.5.0", "ajv": "^6.10.0", "chalk": "^4.0.0", "cross-spawn": "^7.0.2", "debug": "^4.0.1", "doctrine": "^3.0.0", "enquirer": "^2.3.5", + "escape-string-regexp": "^4.0.0", "eslint-scope": "^5.1.1", "eslint-utils": "^2.1.0", "eslint-visitor-keys": "^2.0.0", "espree": "^7.3.1", "esquery": "^1.4.0", "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", "file-entry-cache": "^6.0.1", "functional-red-black-tree": "^1.0.1", - "glob-parent": "^5.0.0", - "globals": "^12.1.0", + "glob-parent": "^5.1.2", + "globals": "^13.6.0", "ignore": "^4.0.6", "import-fresh": "^3.0.0", "imurmurhash": "^0.1.4", @@ -2015,7 +1977,7 @@ "js-yaml": "^3.13.1", "json-stable-stringify-without-jsonify": "^1.0.1", "levn": "^0.4.1", - "lodash": "^4.17.20", + "lodash.merge": "^4.6.2", "minimatch": "^3.0.4", "natural-compare": "^1.4.0", "optionator": "^0.9.1", @@ -2024,18 +1986,26 @@ "semver": "^7.2.1", "strip-ansi": "^6.0.0", "strip-json-comments": "^3.1.0", - "table": "^6.0.4", + "table": "^6.0.9", "text-table": "^0.2.0", "v8-compile-cache": "^2.0.3" }, "dependencies": { - "@babel/code-frame": { - "version": "7.12.11", - "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.12.11.tgz", - "integrity": "sha512-Zt1yodBx1UcyiePMSkWnU4hPqhwq7hGi2nFL1LeA3EUl+q2LQx16MISgJ0+z7dnmgvP9QtIleuETGOiOH1RcIw==", + "eslint-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz", + "integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==", "dev": true, "requires": { - "@babel/highlight": "^7.10.4" + "eslint-visitor-keys": "^1.1.0" + }, + "dependencies": { + "eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "dev": true + } } }, "ignore": { @@ -2045,9 +2015,9 @@ "dev": true }, "semver": { - "version": "7.3.4", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.4.tgz", - "integrity": "sha512-tCfb2WLjqFAtXn4KEdxIhalnRtoKFN7nAwj0B3ZXCbQloV2tq5eDbcTmT68JJD3nRJq24/XgxtQKFIpQdtvmVw==", + "version": "7.3.5", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", + "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", "dev": true, "requires": { "lru-cache": "^6.0.0" @@ -2056,15 +2026,15 @@ } }, "eslint-config-prettier": { - "version": "8.1.0", - "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-8.1.0.tgz", - "integrity": "sha512-oKMhGv3ihGbCIimCAjqkdzx2Q+jthoqnXSP+d86M9tptwugycmTFdVR4IpLgq2c4SHifbwO90z2fQ8/Aio73yw==", + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-8.3.0.tgz", + "integrity": "sha512-BgZuLUSeKzvlL/VUjx/Yb787VQ26RU3gGjA3iiFvdsp/2bMfVIWUVP7tjxtjS0e+HP409cPlPvNkQloz8C91ew==", "dev": true }, "eslint-plugin-prettier": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-3.3.1.tgz", - "integrity": "sha512-Rq3jkcFY8RYeQLgk2cCwuc0P7SEFwDravPhsJZOQ5N4YI4DSg50NyqJ/9gdZHzQlHf8MvafSesbNJCcP/FF6pQ==", + "version": "3.4.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-3.4.1.tgz", + "integrity": "sha512-htg25EUYUeIhKHXjOinK4BgCcDwtLHjqaxCDsMy5nbnUMkKFvIhMVCp+5GFUXQ4Nr8lBsPqtGAqBenbpFqAA2g==", "dev": true, "requires": { "prettier-linter-helpers": "^1.0.0" @@ -2087,26 +2057,18 @@ } }, "eslint-utils": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz", - "integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-3.0.0.tgz", + "integrity": "sha512-uuQC43IGctw68pJA1RgbQS8/NP7rch6Cwd4j3ZBtgo4/8Flj4eGE7ZYSZRN3iq5pVUv6GPdW5Z1RFleo84uLDA==", "dev": true, "requires": { - "eslint-visitor-keys": "^1.1.0" - }, - "dependencies": { - "eslint-visitor-keys": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", - "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", - "dev": true - } + "eslint-visitor-keys": "^2.0.0" } }, "eslint-visitor-keys": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.0.0.tgz", - "integrity": "sha512-QudtT6av5WXels9WjIM7qz1XD1cWGvX4gGXvp/zBn9nXG02D0utdU3Em2m/QjTnrsk6bBjmCygl3rmj118msQQ==", + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.1.0.tgz", + "integrity": "sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==", "dev": true }, "espree": { @@ -2180,12 +2142,6 @@ "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", "dev": true }, - "event-target-shim": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz", - "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==", - "dev": true - }, "events": { "version": "3.3.0", "resolved": "https://registry.npmjs.org/events/-/events-3.3.0.tgz", @@ -2261,12 +2217,6 @@ "homedir-polyfill": "^1.0.1" } }, - "extend": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", - "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", - "dev": true - }, "extend-shallow": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", @@ -2366,17 +2316,16 @@ "dev": true }, "fast-glob": { - "version": "3.2.5", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.5.tgz", - "integrity": "sha512-2DtFcgT68wiTTiwZ2hNdJfcHNke9XOfnwmBRWXhmeKM8rF0TGwmC/Qto3S7RoZKp5cilZbxzO5iTNTQsJ+EeDg==", + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.7.tgz", + "integrity": "sha512-rYGMRwip6lUMvYD3BTScMwT1HtAs2d71SMv66Vrxs0IekGZEjhM0pcMfjQPnknBt2zeCwQMEupiN02ZP4DiT1Q==", "dev": true, "requires": { "@nodelib/fs.stat": "^2.0.2", "@nodelib/fs.walk": "^1.2.3", - "glob-parent": "^5.1.0", + "glob-parent": "^5.1.2", "merge2": "^1.3.0", - "micromatch": "^4.0.2", - "picomatch": "^2.2.1" + "micromatch": "^4.0.4" } }, "fast-json-stable-stringify": { @@ -2391,16 +2340,10 @@ "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", "dev": true }, - "fast-text-encoding": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/fast-text-encoding/-/fast-text-encoding-1.0.3.tgz", - "integrity": "sha512-dtm4QZH9nZtcDt8qJiOH9fcQd1NAgi+K1O2DbE6GG1PPCK/BWfOH3idCTRQ4ImXRUOyopDEgDEnVEE7Y/2Wrig==", - "dev": true - }, "fastq": { - "version": "1.11.0", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.11.0.tgz", - "integrity": "sha512-7Eczs8gIPDrVzT+EksYBcupqMyxSHXXrHOLRRxU2/DicV8789MRBRR8+Hc2uWzUupOs4YS4JzBmBxjjCVBxD/g==", + "version": "1.13.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.13.0.tgz", + "integrity": "sha512-YpkpUnK8od0o1hmeSc7UUs/eB/vIPWJYjKck2QKIzAf71Vm1AAQ3EbuZB3g2JIy+pg+ERD0vqI79KyZiB2e2Nw==", "dev": true, "requires": { "reusify": "^1.0.4" @@ -2459,12 +2402,6 @@ "pify": "^4.0.1", "semver": "^5.6.0" } - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true } } }, @@ -2609,23 +2546,12 @@ "requires": { "flatted": "^3.1.0", "rimraf": "^3.0.2" - }, - "dependencies": { - "rimraf": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", - "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - } } }, "flatted": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.1.1.tgz", - "integrity": "sha512-zAoAQiudy+r5SvnSw3KJy5os/oRJYHzrzja/tBDqrZtNhUw8bt6y8OBzMWcjWr+8liV8Eb6yOhw8WZ7VFZ5ZzA==", + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.2.tgz", + "integrity": "sha512-JaTY/wtrcSyvXJl4IMFHPKyFur1sE9AUqc0QnhOaJ0CxHtAoIV8pYDzeEfAaNEtGkOfq4gr3LBFmdXW5mOQFnA==", "dev": true }, "flush-write-stream": { @@ -2721,13 +2647,13 @@ "rimraf": "2" }, "dependencies": { - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", "dev": true, "requires": { - "minimist": "^1.2.5" + "glob": "^7.1.3" } } } @@ -2797,29 +2723,6 @@ } } }, - "gaxios": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/gaxios/-/gaxios-4.2.0.tgz", - "integrity": "sha512-Ms7fNifGv0XVU+6eIyL9LB7RVESeML9+cMvkwGS70xyD6w2Z80wl6RiqiJ9k1KFlJCUTQqFFc8tXmPQfSKUe8g==", - "dev": true, - "requires": { - "abort-controller": "^3.0.0", - "extend": "^3.0.2", - "https-proxy-agent": "^5.0.0", - "is-stream": "^2.0.0", - "node-fetch": "^2.3.0" - } - }, - "gcp-metadata": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/gcp-metadata/-/gcp-metadata-4.2.1.tgz", - "integrity": "sha512-tSk+REe5iq/N+K+SK1XjZJUrFPuDqGZVzCy2vocIHIGmPlTGsa8owXMJwGkrXr73NO0AzhPW4MF2DEHz7P2AVw==", - "dev": true, - "requires": { - "gaxios": "^4.0.0", - "json-bigint": "^1.0.0" - } - }, "get-caller-file": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", @@ -2850,9 +2753,9 @@ "dev": true }, "glob": { - "version": "7.1.6", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", - "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", "dev": true, "requires": { "fs.realpath": "^1.0.0", @@ -2928,18 +2831,18 @@ } }, "globals": { - "version": "12.4.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz", - "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", + "version": "13.11.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.11.0.tgz", + "integrity": "sha512-08/xrJ7wQjK9kkkRoI3OFUBbLx4f+6x3SGwcPvQ0QH6goFDrOU2oyAWrmh3dJezu65buo+HBMzAMQy6rovVC3g==", "dev": true, "requires": { - "type-fest": "^0.8.1" + "type-fest": "^0.20.2" } }, "globby": { - "version": "11.0.2", - "resolved": "https://registry.npmjs.org/globby/-/globby-11.0.2.tgz", - "integrity": "sha512-2ZThXDvvV8fYFRVIxnrMQBipZQDr7MxKAmQK1vujaj9/7eF0efG7BPUKJ7jP7G5SLF37xKDXvO4S/KKLj/Z0og==", + "version": "11.0.4", + "resolved": "https://registry.npmjs.org/globby/-/globby-11.0.4.tgz", + "integrity": "sha512-9O4MVG9ioZJ08ffbcyVYyLOJLk5JQ688pJ4eMGLpdWLHq/Wr1D9BlriLQyL0E+jbkuePVZXYFj47QM/v093wHg==", "dev": true, "requires": { "array-union": "^2.1.0", @@ -2950,42 +2853,16 @@ "slash": "^3.0.0" } }, - "google-auth-library": { - "version": "6.1.6", - "resolved": "https://registry.npmjs.org/google-auth-library/-/google-auth-library-6.1.6.tgz", - "integrity": "sha512-Q+ZjUEvLQj/lrVHF/IQwRo6p3s8Nc44Zk/DALsN+ac3T4HY/g/3rrufkgtl+nZ1TW7DNAw5cTChdVp4apUXVgQ==", - "dev": true, - "requires": { - "arrify": "^2.0.0", - "base64-js": "^1.3.0", - "ecdsa-sig-formatter": "^1.0.11", - "fast-text-encoding": "^1.0.0", - "gaxios": "^4.0.0", - "gcp-metadata": "^4.2.0", - "gtoken": "^5.0.4", - "jws": "^4.0.0", - "lru-cache": "^6.0.0" - } - }, - "google-p12-pem": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/google-p12-pem/-/google-p12-pem-3.0.3.tgz", - "integrity": "sha512-wS0ek4ZtFx/ACKYF3JhyGe5kzH7pgiQ7J5otlumqR9psmWMYc+U9cErKlCYVYHoUaidXHdZ2xbo34kB+S+24hA==", - "dev": true, - "requires": { - "node-forge": "^0.10.0" - } - }, "google-protobuf": { - "version": "3.15.5", - "resolved": "https://registry.npmjs.org/google-protobuf/-/google-protobuf-3.15.5.tgz", - "integrity": "sha512-6bLpAI4nMIQODlegR7OevgkCoyOj5frLVDArUpeuBWad7XWUNWMGP0v5lz1/aeUI6Yf3cG9XA6acZkPxom4SEw==", + "version": "3.18.0", + "resolved": "https://registry.npmjs.org/google-protobuf/-/google-protobuf-3.18.0.tgz", + "integrity": "sha512-WlaQWRkUOo/lm9uTgNH6nk9IQt814RggWPzKBfnAVewOFzSzRUSmS1yUWRT6ixW1vS7er5p6tmLSmwzpPpmc8A==", "dev": true }, "graceful-fs": { - "version": "4.2.6", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.6.tgz", - "integrity": "sha512-nTnJ528pbqxYanhpDYsi4Rd8MAeaBA67+RZ10CM1m3bTAVFEDcd5AuA4a6W5YkGZ1iNXHzZz8T6TBKLeBuNriQ==", + "version": "4.2.8", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.8.tgz", + "integrity": "sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==", "dev": true }, "growl": { @@ -3003,17 +2880,6 @@ "@mapbox/node-pre-gyp": "^1.0.5" } }, - "gtoken": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/gtoken/-/gtoken-5.2.1.tgz", - "integrity": "sha512-OY0BfPKe3QnMsY9MzTHTSKn+Vl2l1CcLe6BwDEQj00mbbkl5nyQ/7EUREstg4fQNZ8iYE7br4JJ7TdKeDOPWmw==", - "dev": true, - "requires": { - "gaxios": "^4.0.0", - "google-p12-pem": "^3.0.3", - "jws": "^4.0.0" - } - }, "has": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", @@ -3114,6 +2980,12 @@ "string_decoder": "^1.1.1", "util-deprecate": "^1.0.1" } + }, + "safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "dev": true } } }, @@ -3153,6 +3025,15 @@ "parse-passwd": "^1.0.0" } }, + "hosted-git-info": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-4.0.2.tgz", + "integrity": "sha512-c9OGXbZ3guC/xOlCg1Ci/VgWlwsqDv1yMQL1CWqXDL0hDjXuNcq0zuR4xqPSuasI3kqFDhqSyTjREz5gzq0fXg==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, "htmlparser2": { "version": "6.1.0", "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-6.1.0.tgz", @@ -3165,6 +3046,17 @@ "entities": "^2.0.0" } }, + "http-proxy-agent": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz", + "integrity": "sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==", + "dev": true, + "requires": { + "@tootallnate/once": "1", + "agent-base": "6", + "debug": "4" + } + }, "https-browserify": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/https-browserify/-/https-browserify-1.0.0.tgz", @@ -3346,9 +3238,9 @@ "dev": true }, "is-fullwidth-code-point": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", "dev": true }, "is-glob": { @@ -3381,12 +3273,6 @@ "isobject": "^3.0.1" } }, - "is-stream": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.0.tgz", - "integrity": "sha512-XCoy+WlUr7d1+Z8GgSuXmpuUFC9fOhRXglJMx+dwLKTkL44Cjd4W1Z5P+BQZpr+cR93aGP4S/s7Ftw6Nd/kiEw==", - "dev": true - }, "is-windows": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", @@ -3433,15 +3319,6 @@ "esprima": "^4.0.0" } }, - "json-bigint": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/json-bigint/-/json-bigint-1.0.0.tgz", - "integrity": "sha512-SiPv/8VpZuWbvLSMtTDU8hEfrZWg/mH/nV/b4o0CYbSxu1UIQPLdwKOCIyLQX+VIPO5vrLX3i8qtqFyhdPSUSQ==", - "dev": true, - "requires": { - "bignumber.js": "^9.0.0" - } - }, "json-parse-better-errors": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", @@ -3480,32 +3357,11 @@ } }, "just-extend": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/just-extend/-/just-extend-4.1.1.tgz", - "integrity": "sha512-aWgeGFW67BP3e5181Ep1Fv2v8z//iBJfrvyTnq8wG86vEESwmonn1zPBJ0VfmT9CJq2FIT0VsETtrNFm2a+SHA==", + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/just-extend/-/just-extend-4.2.1.tgz", + "integrity": "sha512-g3UB796vUFIY90VIv/WX3L2c8CS2MdWUww3CNrYmqza1Fg0DURc2K/O4YrnklBdQarSJ/y8JnJYDGc+1iumQjg==", "dev": true }, - "jwa": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/jwa/-/jwa-2.0.0.tgz", - "integrity": "sha512-jrZ2Qx916EA+fq9cEAeCROWPTfCwi1IVHqT2tapuqLEVVDKFDENFw1oL+MwrTvH6msKxsd1YTDVw6uKEcsrLEA==", - "dev": true, - "requires": { - "buffer-equal-constant-time": "1.0.1", - "ecdsa-sig-formatter": "1.0.11", - "safe-buffer": "^5.0.1" - } - }, - "jws": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/jws/-/jws-4.0.0.tgz", - "integrity": "sha512-KDncfTmOZoOMTFG4mBlG0qUIOlc03fmzH+ru6RgYVZhPkyiy/92Owlt/8UEN+a4TXR1FQetfIpJE8ApdvdVxTg==", - "dev": true, - "requires": { - "jwa": "^2.0.0", - "safe-buffer": "^5.0.1" - } - }, "kind-of": { "version": "6.0.3", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", @@ -3575,12 +3431,30 @@ "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", "dev": true }, + "lodash.clonedeep": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz", + "integrity": "sha1-4j8/nE+Pvd6HJSnBBxhXoIblzO8=", + "dev": true + }, "lodash.get": { "version": "4.4.2", "resolved": "https://registry.npmjs.org/lodash.get/-/lodash.get-4.4.2.tgz", "integrity": "sha1-LRd/ZS+jHpObRDjVNBSZ36OCXpk=", "dev": true }, + "lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true + }, + "lodash.truncate": { + "version": "4.4.2", + "resolved": "https://registry.npmjs.org/lodash.truncate/-/lodash.truncate-4.4.2.tgz", + "integrity": "sha1-WjUNoLERO4N+z//VgSy+WNbq4ZM=", + "dev": true + }, "log-symbols": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.0.0.tgz", @@ -3594,7 +3468,6 @@ "version": "6.0.0", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", - "dev": true, "requires": { "yallist": "^4.0.0" } @@ -3606,6 +3479,14 @@ "dev": true, "requires": { "semver": "^6.0.0" + }, + "dependencies": { + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } } }, "map-cache": { @@ -3678,13 +3559,13 @@ "dev": true }, "micromatch": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", - "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.4.tgz", + "integrity": "sha512-pRmzw/XUcwXGpD9aI9q/0XOwLNygjETJ8y0ao0wdqprrzDa4YnxLcz7fQRZr8voh8V10kGhABbNcHVk5wHgWwg==", "dev": true, "requires": { "braces": "^3.0.1", - "picomatch": "^2.0.5" + "picomatch": "^2.2.3" } }, "miller-rabin": { @@ -3727,7 +3608,6 @@ "version": "3.0.4", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", - "dev": true, "requires": { "brace-expansion": "^1.1.7" } @@ -3739,9 +3619,9 @@ "dev": true }, "minipass": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.1.3.tgz", - "integrity": "sha512-Mgd2GdMVzY+x3IJ+oHnVM+KG3lA5c8tnabyJKmHSaG2kAGpudxuOf8ToDkhumF7UzME7DecbQE9uOZhNm7PuJg==", + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.1.5.tgz", + "integrity": "sha512-+8NzxD82XQoNKNrl1d/FSi+X8wAEWR+sbYAfIvub4Nz0d22plFG72CEVVaufV8PNf4qSslFTD8VMOxNVhHCjTw==", "dev": true, "requires": { "yallist": "^4.0.0" @@ -3797,15 +3677,18 @@ } }, "mkdirp": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", - "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", - "dev": true + "version": "0.5.5", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", + "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", + "dev": true, + "requires": { + "minimist": "^1.2.5" + } }, "mocha": { - "version": "8.3.1", - "resolved": "https://registry.npmjs.org/mocha/-/mocha-8.3.1.tgz", - "integrity": "sha512-5SBMxANWqOv5bw3Hx+HVgaWlcWcFEQDUdaUAr1AUU+qwtx6cowhn7gEDT/DwQP7uYxnvShdUOVLbTYAHOEGfDQ==", + "version": "8.4.0", + "resolved": "https://registry.npmjs.org/mocha/-/mocha-8.4.0.tgz", + "integrity": "sha512-hJaO0mwDXmZS4ghXsvPVriOhsxQ7ofcpQdm8dE+jISUOKopitvnXFQmpRR7jd2K6VBG6E26gU3IAbXXGIbu4sQ==", "dev": true, "requires": { "@ungap/promise-all-settled": "1.1.2", @@ -3841,11 +3724,36 @@ "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", "dev": true }, - "escape-string-regexp": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", - "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", - "dev": true + "debug": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", + "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", + "dev": true, + "requires": { + "ms": "2.1.2" + }, + "dependencies": { + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + } + } + }, + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } }, "has-flag": { "version": "4.0.0", @@ -3868,15 +3776,6 @@ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", "dev": true }, - "serialize-javascript": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-5.0.1.tgz", - "integrity": "sha512-SaaNal9imEO737H2c05Og0/8LUXG7EnsZyMa8MzkmuHoELfT6txuj0cMqRj6zfPKnmQ1yasR4PCJc8x+M4JSPA==", - "dev": true, - "requires": { - "randombytes": "^2.1.0" - } - }, "supports-color": { "version": "8.1.1", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", @@ -3902,13 +3801,13 @@ "run-queue": "^1.0.3" }, "dependencies": { - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", "dev": true, "requires": { - "minimist": "^1.2.5" + "glob": "^7.1.3" } } } @@ -3982,16 +3881,13 @@ } }, "node-fetch": { - "version": "2.6.1", - "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.1.tgz", - "integrity": "sha512-V4aYg89jEoVRxRb2fJdAg8FHvI7cEyYdVAh94HH0UIK8oJxUfkjlDQN9RbMx+bEjP7+ggMiFRprSti032Oipxw==", - "dev": true - }, - "node-forge": { - "version": "0.10.0", - "resolved": "https://registry.npmjs.org/node-forge/-/node-forge-0.10.0.tgz", - "integrity": "sha512-PPmu8eEeG9saEUvI97fm4OYxXVB6bFvyNTyiUOBichBpFG8A1Ljw3bY62+5oOjDEMHRnd0Y7HQ+x7uzxOzC6JA==", - "dev": true + "version": "2.6.5", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.5.tgz", + "integrity": "sha512-mmlIVHJEu5rnIxgEgez6b9GgWXbkZj5YZ7fx+2r94a2E+Uirsp6HsPTPlomfdHtpt/B0cdKviwkoaM6pyvUOpQ==", + "dev": true, + "requires": { + "whatwg-url": "^5.0.0" + } }, "node-libs-browser": { "version": "2.2.1", @@ -4060,9 +3956,9 @@ } }, "nth-check": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.0.0.tgz", - "integrity": "sha512-i4sc/Kj8htBrAiH1viZ0TgU8Y5XqCaV/FziYK6TBczxmeKm3AEFWqqF3195yKudrarqy7Zu80Ra5dobFjn9X/Q==", + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.0.1.tgz", + "integrity": "sha512-it1vE95zF6dTT9lBsYbxvqh0Soy4SPowchj0UBGj/V6cTPnXXtQOPUbhZ6CmGzAD/rW22LQK6E96pcdJXk4A4w==", "dev": true, "requires": { "boolbase": "^1.0.0" @@ -4112,9 +4008,9 @@ } }, "object-inspect": { - "version": "1.10.2", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.10.2.tgz", - "integrity": "sha512-gz58rdPpadwztRrPjZE9DZLOABUpTGdcANUgOwBFO1C+HZZhePoP83M65WGDmbpwFYJSWqavbl4SgDn4k8RYTA==", + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.11.0.tgz", + "integrity": "sha512-jp7ikS6Sd3GxQfZJPyH3cjcbJF6GZPClgdV+EFygjFLQ5FmW/dRUnTd9PQ9k0JhoNDabWFbpF1yCdSWCC6gexg==", "dev": true }, "object-visit": { @@ -4262,14 +4158,6 @@ "dev": true, "requires": { "semver": "^5.1.0" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } } }, "parse5": { @@ -4367,9 +4255,9 @@ "dev": true }, "picomatch": { - "version": "2.2.2", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz", - "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==", + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", + "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", "dev": true }, "pify": { @@ -4445,9 +4333,9 @@ "dev": true }, "prettier": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.2.1.tgz", - "integrity": "sha512-PqyhM2yCjg/oKkFPtTGUojv7gnZAoG80ttl45O6x2Ug/rMJw4wcc9k6aaf2hibP7BGVCCM33gZoGjyvt9mm16Q==", + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.4.1.tgz", + "integrity": "sha512-9fbDAXSBcc6Bs1mZrDYb3XKzDLm4EXXL9sC1LqKP5rZkT6KRr/rf9amVUcODVXgguK/isJz0d0hP72WeaKWsvA==", "dev": true }, "prettier-linter-helpers": { @@ -4572,9 +4460,9 @@ "dev": true }, "queue-microtask": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.2.tgz", - "integrity": "sha512-dB15eXv3p2jDlbOiNLyMabYg1/sXvppd8DP2J3EOCQ0AkuSXCW2tP7mnVouVLJKgUMY6yP0kcQDVpLCN13h4Xg==", + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", "dev": true }, "randombytes": { @@ -4618,14 +4506,6 @@ "safe-buffer": "~5.1.1", "string_decoder": "~1.1.1", "util-deprecate": "~1.0.1" - }, - "dependencies": { - "safe-buffer": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", - "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", - "dev": true - } } }, "readdirp": { @@ -4648,9 +4528,9 @@ } }, "regexpp": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.1.0.tgz", - "integrity": "sha512-ZOIzd8yVsQQA7j8GCSlPGXwg5PfmA1mrq0JP4nGhh54LaKN3xdai/vHUDu74pKwV8OxseMS65u2NImosQcSD0Q==", + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.2.0.tgz", + "integrity": "sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==", "dev": true }, "remove-trailing-separator": { @@ -4755,9 +4635,9 @@ "dev": true }, "rimraf": { - "version": "2.6.3", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", - "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", "dev": true, "requires": { "glob": "^7.1.3" @@ -4792,9 +4672,9 @@ } }, "safe-buffer": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", - "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", "dev": true }, "safe-regex": { @@ -4812,6 +4692,12 @@ "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", "dev": true }, + "sax": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", + "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==", + "dev": true + }, "schema-utils": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", @@ -4824,15 +4710,14 @@ } }, "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", - "dev": true + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" }, "serialize-javascript": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-4.0.0.tgz", - "integrity": "sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-5.0.1.tgz", + "integrity": "sha512-SaaNal9imEO737H2c05Og0/8LUXG7EnsZyMa8MzkmuHoELfT6txuj0cMqRj6zfPKnmQ1yasR4PCJc8x+M4JSPA==", "dev": true, "requires": { "randombytes": "^2.1.0" @@ -4915,9 +4800,9 @@ } }, "signal-exit": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", - "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.4.tgz", + "integrity": "sha512-rqYhcAnZ6d/vTPGghdrw7iumdcbXpsk1b8IG/rz+VWV51DM0p7XCtMoJ3qhPLIbp3tvyt3pKRbaaEMZYpHto8Q==", "dev": true }, "sinon": { @@ -4974,10 +4859,28 @@ "is-fullwidth-code-point": "^3.0.0" }, "dependencies": { - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true } } @@ -5105,9 +5008,9 @@ } }, "snyk": { - "version": "1.695.0", - "resolved": "https://registry.npmjs.org/snyk/-/snyk-1.695.0.tgz", - "integrity": "sha512-Jzt6W21zuka7ZiGZf+XdOHsnTY702uU58NP1dBqAsyGqO2U7lVOVBvKjVQC9EURZPLALHI99dvFiVlTduelbDw==", + "version": "1.720.0", + "resolved": "https://registry.npmjs.org/snyk/-/snyk-1.720.0.tgz", + "integrity": "sha512-NCy+57lvoggyM4WIzsNQyf2+fX6eZAqu2nznGq+RVfYBcwLgiDpg76oafrmp/rUtSIM1E+68cm1bJWSiNZqf7w==", "dev": true }, "source-list-map": { @@ -5136,9 +5039,9 @@ } }, "source-map-support": { - "version": "0.5.19", - "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.19.tgz", - "integrity": "sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw==", + "version": "0.5.20", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.20.tgz", + "integrity": "sha512-n1lZZ8Ve4ksRqizaBQgxXDgKwttHDhyfQjA6YZZn8+AroHbsIz+JjwxQDxbp+7y5OYCI8t1Yk7etjD9CRd2hIw==", "dev": true, "requires": { "buffer-from": "^1.0.0", @@ -5255,22 +5158,14 @@ "dev": true }, "string-width": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", - "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", "dev": true, "requires": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.0" - }, - "dependencies": { - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", - "dev": true - } + "strip-ansi": "^6.0.1" } }, "string_decoder": { @@ -5280,23 +5175,15 @@ "dev": true, "requires": { "safe-buffer": "~5.1.0" - }, - "dependencies": { - "safe-buffer": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", - "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", - "dev": true - } } }, "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", "dev": true, "requires": { - "ansi-regex": "^5.0.0" + "ansi-regex": "^5.0.1" } }, "strip-json-comments": { @@ -5315,21 +5202,23 @@ } }, "table": { - "version": "6.0.7", - "resolved": "https://registry.npmjs.org/table/-/table-6.0.7.tgz", - "integrity": "sha512-rxZevLGTUzWna/qBLObOe16kB2RTnnbhciwgPbMMlazz1yZGVEgnZK762xyVdVznhqxrfCeBMmMkgOOaPwjH7g==", + "version": "6.7.1", + "resolved": "https://registry.npmjs.org/table/-/table-6.7.1.tgz", + "integrity": "sha512-ZGum47Yi6KOOFDE8m223td53ath2enHcYLgOCjGr5ngu8bdIARQk6mN/wRMv4yMRcHnCSnHbCEha4sobQx5yWg==", "dev": true, "requires": { - "ajv": "^7.0.2", - "lodash": "^4.17.20", + "ajv": "^8.0.1", + "lodash.clonedeep": "^4.5.0", + "lodash.truncate": "^4.4.2", "slice-ansi": "^4.0.0", - "string-width": "^4.2.0" + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0" }, "dependencies": { "ajv": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-7.1.1.tgz", - "integrity": "sha512-ga/aqDYnUy/o7vbsRTFhhTsNeXiYb5JWDIcRIeZfwRNCefwjNTVYCGdGSUrEmiu3yDK3vFvNbgJxvrQW4JXrYQ==", + "version": "8.6.3", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.6.3.tgz", + "integrity": "sha512-SMJOdDP6LqTkD0Uq8qLi+gMwSt0imXLSV080qFVwJCpH9U6Mb+SUGHAXM0KNbcBPguytWyvFxcHgMLe2D2XSpw==", "dev": true, "requires": { "fast-deep-equal": "^3.1.1", @@ -5353,9 +5242,9 @@ "dev": true }, "tar": { - "version": "6.1.6", - "resolved": "https://registry.npmjs.org/tar/-/tar-6.1.6.tgz", - "integrity": "sha512-oaWyu5dQbHaYcyZCTfyPpC+VmI62/OM2RTUYavTk1MDr1cwW5Boi3baeYQKiZbY2uSQJGr+iMOzb/JFxLrft+g==", + "version": "6.1.11", + "resolved": "https://registry.npmjs.org/tar/-/tar-6.1.11.tgz", + "integrity": "sha512-an/KZQzQUkZCkuoAA64hM92X0Urb6VpRhAFllDzz44U2mcD5scmT3zBc4VgVpkugF580+DQn8eAFSyoQt0tznA==", "dev": true, "requires": { "chownr": "^2.0.0", @@ -5364,6 +5253,14 @@ "minizlib": "^2.1.1", "mkdirp": "^1.0.3", "yallist": "^4.0.0" + }, + "dependencies": { + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + } } }, "terser": { @@ -5377,6 +5274,12 @@ "source-map-support": "~0.5.12" }, "dependencies": { + "commander": { + "version": "2.20.3", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", + "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", + "dev": true + }, "source-map": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", @@ -5402,6 +5305,15 @@ "worker-farm": "^1.7.0" }, "dependencies": { + "serialize-javascript": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-4.0.0.tgz", + "integrity": "sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw==", + "dev": true, + "requires": { + "randombytes": "^2.1.0" + } + }, "source-map": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", @@ -5435,6 +5347,15 @@ "setimmediate": "^1.0.4" } }, + "tmp": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.2.1.tgz", + "integrity": "sha512-76SUhtfqR2Ijn+xllcI5P1oyannHNHByD80W1q447gU3mp9G9PSpGdWmjUOHRDPiHYacIk66W7ubDTuPF3BEtQ==", + "dev": true, + "requires": { + "rimraf": "^3.0.0" + } + }, "to-arraybuffer": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/to-arraybuffer/-/to-arraybuffer-1.0.1.tgz", @@ -5482,6 +5403,12 @@ "is-number": "^7.0.0" } }, + "tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha1-gYT9NH2snNwYWZLzpmIuFLnZq2o=", + "dev": true + }, "traverse": { "version": "0.3.9", "resolved": "https://registry.npmjs.org/traverse/-/traverse-0.3.9.tgz", @@ -5507,15 +5434,6 @@ "semver": "^5.0.1" }, "dependencies": { - "ansi-styles": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", - "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", - "dev": true, - "requires": { - "color-convert": "^1.9.0" - } - }, "braces": { "version": "2.3.2", "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", @@ -5556,19 +5474,10 @@ "supports-color": "^5.3.0" } }, - "color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", - "dev": true, - "requires": { - "color-name": "1.1.3" - } - }, - "color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", + "escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", "dev": true }, "fill-range": { @@ -5635,12 +5544,6 @@ "to-regex": "^3.0.2" } }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - }, "to-regex-range": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", @@ -5669,9 +5572,9 @@ "dev": true }, "tsutils": { - "version": "3.20.0", - "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.20.0.tgz", - "integrity": "sha512-RYbuQuvkhuqVeXweWT3tJLKOEJ/UUw9GjNEZGWdrLLlM+611o1gwLHBpxoFJKKl25fLprp2eVthtKs5JOrNeXg==", + "version": "3.21.0", + "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.21.0.tgz", + "integrity": "sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==", "dev": true, "requires": { "tslib": "^1.8.1" @@ -5705,15 +5608,15 @@ "dev": true }, "type-fest": { - "version": "0.8.1", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", - "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", "dev": true }, "typed-rest-client": { - "version": "1.8.4", - "resolved": "https://registry.npmjs.org/typed-rest-client/-/typed-rest-client-1.8.4.tgz", - "integrity": "sha512-MyfKKYzk3I6/QQp6e1T50py4qg+c+9BzOEl2rBmQIpStwNUoqQ73An+Tkfy9YuV7O+o2mpVVJpe+fH//POZkbg==", + "version": "1.8.6", + "resolved": "https://registry.npmjs.org/typed-rest-client/-/typed-rest-client-1.8.6.tgz", + "integrity": "sha512-xcQpTEAJw2DP7GqVNECh4dD+riS+C1qndXLfBCJ3xk0kqprtGN491P5KlmrDbKdtuW8NEcP/5ChxiJI3S9WYTA==", "dev": true, "requires": { "qs": "^6.9.1", @@ -5837,14 +5740,6 @@ "listenercount": "~1.0.1", "readable-stream": "~2.3.6", "setimmediate": "~1.0.4" - }, - "dependencies": { - "bluebird": { - "version": "3.4.7", - "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.4.7.tgz", - "integrity": "sha1-9y12C+Cbf3bQjtj66Ysomo0F+rM=", - "dev": true - } } }, "upath": { @@ -5922,6 +5817,11 @@ "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", "dev": true }, + "uuid": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", + "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==" + }, "v8-compile-cache": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.3.0.tgz", @@ -5935,17 +5835,18 @@ "dev": true }, "vsce": { - "version": "1.88.0", - "resolved": "https://registry.npmjs.org/vsce/-/vsce-1.88.0.tgz", - "integrity": "sha512-FS5ou3G+WRnPPr/tWVs8b/jVzeDacgZHy/y7/QQW7maSPFEAmRt2bFGUJtJVEUDLBqtDm/3VGMJ7D31cF2U1tw==", + "version": "1.100.0", + "resolved": "https://registry.npmjs.org/vsce/-/vsce-1.100.0.tgz", + "integrity": "sha512-sY1NVSZkesioir/1w04igdSPKKHb4QAn7AngOQIKvNTvtFUFuEE/KrcURcld9Gai9W5167zaeifW5rWUsX3rLg==", "dev": true, "requires": { - "azure-devops-node-api": "^10.2.2", + "azure-devops-node-api": "^11.0.1", "chalk": "^2.4.2", - "cheerio": "^1.0.0-rc.1", + "cheerio": "^1.0.0-rc.9", "commander": "^6.1.0", "denodeify": "^1.2.1", "glob": "^7.0.6", + "hosted-git-info": "^4.0.2", "leven": "^3.1.0", "lodash": "^4.17.15", "markdown-it": "^10.0.0", @@ -5955,22 +5856,14 @@ "parse-semver": "^1.1.1", "read": "^1.0.7", "semver": "^5.1.0", - "tmp": "0.0.29", + "tmp": "^0.2.1", "typed-rest-client": "^1.8.4", "url-join": "^1.1.0", + "xml2js": "^0.4.23", "yauzl": "^2.3.1", "yazl": "^2.2.2" }, "dependencies": { - "ansi-styles": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", - "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", - "dev": true, - "requires": { - "color-convert": "^1.9.0" - } - }, "chalk": { "version": "2.4.2", "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", @@ -5982,41 +5875,11 @@ "supports-color": "^5.3.0" } }, - "color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", - "dev": true, - "requires": { - "color-name": "1.1.3" - } - }, - "color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", - "dev": true - }, - "commander": { - "version": "6.2.1", - "resolved": "https://registry.npmjs.org/commander/-/commander-6.2.1.tgz", - "integrity": "sha512-U7VdrJFnJgo4xjrHpTzu0yrHPGImdsmD95ZlgYSEajAn2JKzDhDTPG9kBTefmObL2w/ngeZnilk+OV9CG3d7UA==", - "dev": true - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", "dev": true - }, - "tmp": { - "version": "0.0.29", - "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.29.tgz", - "integrity": "sha1-8lEl/w3Z2jzLDC3Tce4SiLuRKMA=", - "dev": true, - "requires": { - "os-tmpdir": "~1.0.1" - } } } }, @@ -6035,15 +5898,47 @@ "requires": { "uuid": "^3.4.0", "vscode-extension-telemetry": "^0.1.6" + } + }, + "vscode-jsonrpc": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/vscode-jsonrpc/-/vscode-jsonrpc-6.0.0.tgz", + "integrity": "sha512-wnJA4BnEjOSyFMvjZdpiOwhSq9uDoK8e/kpRJDTaMYzwlkrhG1fwDIZI94CLsLzlCK5cIbMMtFlJlfR57Lavmg==" + }, + "vscode-languageclient": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/vscode-languageclient/-/vscode-languageclient-7.0.0.tgz", + "integrity": "sha512-P9AXdAPlsCgslpP9pRxYPqkNYV7Xq8300/aZDpO35j1fJm/ncize8iGswzYlcvFw5DQUx4eVk+KvfXdL0rehNg==", + "requires": { + "minimatch": "^3.0.4", + "semver": "^7.3.4", + "vscode-languageserver-protocol": "3.16.0" }, "dependencies": { - "uuid": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", - "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==" + "semver": { + "version": "7.3.5", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", + "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "requires": { + "lru-cache": "^6.0.0" + } } } }, + "vscode-languageserver-protocol": { + "version": "3.16.0", + "resolved": "https://registry.npmjs.org/vscode-languageserver-protocol/-/vscode-languageserver-protocol-3.16.0.tgz", + "integrity": "sha512-sdeUoAawceQdgIfTI+sdcwkiK2KU+2cbEYA0agzM2uqaUy2UpnnGHtWTHVEtS0ES4zHU0eMFRGN+oQgDxlD66A==", + "requires": { + "vscode-jsonrpc": "6.0.0", + "vscode-languageserver-types": "3.16.0" + } + }, + "vscode-languageserver-types": { + "version": "3.16.0", + "resolved": "https://registry.npmjs.org/vscode-languageserver-types/-/vscode-languageserver-types-3.16.0.tgz", + "integrity": "sha512-k8luDIWJWyenLc5ToFQQMaSrqCHiLwyKPHKPQZ5zz21vM+vIVUSvsRpcbiECH4WR88K2XZqc4ScRcZ7nk/jbeA==" + }, "watchpack": { "version": "1.7.5", "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-1.7.5.tgz", @@ -6282,6 +6177,12 @@ } } }, + "webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha1-JFNCdeKnvGvnvIZhHMFq4KVlSHE=", + "dev": true + }, "webpack": { "version": "4.46.0", "resolved": "https://registry.npmjs.org/webpack/-/webpack-4.46.0.tgz", @@ -6432,15 +6333,6 @@ "to-regex": "^3.0.2" } }, - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", - "dev": true, - "requires": { - "minimist": "^1.2.5" - } - }, "to-regex-range": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", @@ -6478,15 +6370,6 @@ "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", "dev": true }, - "ansi-styles": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", - "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", - "dev": true, - "requires": { - "color-convert": "^1.9.0" - } - }, "camelcase": { "version": "5.3.1", "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", @@ -6526,21 +6409,6 @@ "wrap-ansi": "^5.1.0" } }, - "color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", - "dev": true, - "requires": { - "color-name": "1.1.3" - } - }, - "color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", - "dev": true - }, "cross-spawn": { "version": "6.0.5", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", @@ -6566,6 +6434,12 @@ "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", "dev": true }, + "escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", + "dev": true + }, "find-up": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", @@ -6575,6 +6449,12 @@ "locate-path": "^3.0.0" } }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, "locate-path": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", @@ -6615,12 +6495,6 @@ "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", "dev": true }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - }, "shebang-command": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", @@ -6739,6 +6613,16 @@ } } }, + "whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha1-lmRU6HZUYuN2RNNib2dCzotwll0=", + "dev": true, + "requires": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, "which": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", @@ -6769,6 +6653,12 @@ "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", "dev": true }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, "string-width": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", @@ -6820,6 +6710,32 @@ "ansi-styles": "^4.0.0", "string-width": "^4.1.0", "strip-ansi": "^6.0.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + } } }, "wrappy": { @@ -6828,6 +6744,22 @@ "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", "dev": true }, + "xml2js": { + "version": "0.4.23", + "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.23.tgz", + "integrity": "sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug==", + "dev": true, + "requires": { + "sax": ">=0.6.0", + "xmlbuilder": "~11.0.0" + } + }, + "xmlbuilder": { + "version": "11.0.1", + "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz", + "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==", + "dev": true + }, "xtend": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", @@ -6835,16 +6767,15 @@ "dev": true }, "y18n": { - "version": "5.0.5", - "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.5.tgz", - "integrity": "sha512-hsRUr4FFrvhhRH12wOdfs38Gy7k2FFzB9qgN9v3aLykRq0dRcdcpz5C9FxdS2NuhOrI/628b/KSTJ3rwHysYSg==", + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", "dev": true }, "yallist": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", - "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", - "dev": true + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==" }, "yargs": { "version": "16.2.0", diff --git a/extension/package.json b/extension/package.json index 163ade797..d2ba51cd6 100644 --- a/extension/package.json +++ b/extension/package.json @@ -2,7 +2,7 @@ "name": "vscode-gradle", "displayName": "Gradle Tasks", "description": "Run Gradle tasks in VS Code", - "version": "3.7.1", + "version": "3.8.0", "private": true, "publisher": "vscjava", "aiKey": "b4aae7d0-c65b-4819-92bf-1d2f537ae7ce", @@ -37,6 +37,25 @@ ], "main": "./dist/index.js", "contributes": { + "languages": [ + { + "id": "gradle", + "extensions": [ + ".gradle" + ], + "aliases": [ + "Gradle" + ], + "configuration": "./gradle-language-configuration.json" + } + ], + "grammars": [ + { + "language": "gradle", + "scopeName": "source.groovy", + "path": "./syntaxes/groovy.tmLanguage.json" + } + ], "problemMatchers": [ { "owner": "gradle", @@ -57,7 +76,12 @@ { "id": "gradleTasksView", "name": "Gradle Projects", - "when": "gradle:activated" + "when": "gradle:activated && !gradle:defaultView" + }, + { + "id": "gradleDefaultProjectsView", + "name": "Gradle Projects", + "when": "gradle:activated && gradle:defaultView" }, { "id": "pinnedTasksView", @@ -437,7 +461,7 @@ }, { "command": "gradle.openSettings", - "when": "view == gradleTasksView" + "when": "view == gradleTasksView || view == gradleDefaultProjectsView" }, { "command": "gradle.findTask", @@ -445,7 +469,7 @@ }, { "command": "gradle.runBuild", - "when": "view == gradleTasksView", + "when": "view == gradleTasksView || view == gradleDefaultProjectsView", "group": "navigation@0" }, { @@ -497,12 +521,12 @@ "view/item/context": [ { "command": "gradle.runTask", - "when": "view =~ /^gradleTasksView$|^pinnedTasksView$|^recentTasksView$/ && viewItem =~ /^debugTask.*$|^task.*$/", + "when": "view =~ /^gradleTasksView$|^pinnedTasksView$|^recentTasksView$|^gradleDefaultProjectsView$/ && viewItem =~ /^debugTask.*$|^task.*$/", "group": "contextGroup1@0" }, { "command": "gradle.runTaskWithArgs", - "when": "view =~ /^gradleTasksView$|^pinnedTasksView$|^recentTasksView$/ && viewItem =~ /^debugTask(WithTerminals)?$|^task(WithTerminals)?$/", + "when": "view =~ /^gradleTasksView$|^pinnedTasksView$|^recentTasksView$|^gradleDefaultProjectsView$/ && viewItem =~ /^debugTask(WithTerminals)?$|^task(WithTerminals)?$/", "group": "contextGroup1@1" }, { @@ -532,7 +556,7 @@ }, { "command": "gradle.runTask", - "when": "view =~ /^gradleTasksView$|^pinnedTasksView$|^recentTasksView$/ && viewItem =~ /^debugTask.*$|^task.*$/", + "when": "view =~ /^gradleTasksView$|^pinnedTasksView$|^recentTasksView$|^gradleDefaultProjectsView$/ && viewItem =~ /^debugTask.*$|^task.*$/", "group": "inline@3" }, { @@ -823,18 +847,19 @@ "mocha": "^8.3.1", "prettier": "^2.2.1", "sinon": "^9.2.4", - "snyk": "^1.695.0", + "snyk": "^1.697.0", "string-argv": "^0.3.1", "tree-kill": "^1.2.2", "ts-loader": "^4.4.2", "ts-protoc-gen": "^0.14.0", - "typescript": "^4.2.3", + "typescript": "4.2.3", "vsce": "^1.88.0", "webpack": "^4.46.0", "webpack-cli": "^3.3.12" }, "snyk": true, "dependencies": { - "vscode-extension-telemetry-wrapper": "^0.9.0" + "vscode-extension-telemetry-wrapper": "^0.9.0", + "vscode-languageclient": "7.0.0" } } diff --git a/extension/src/Extension.ts b/extension/src/Extension.ts index 8fdedcceb..45101764e 100644 --- a/extension/src/Extension.ts +++ b/extension/src/Extension.ts @@ -26,6 +26,7 @@ import { PINNED_TASKS_VIEW, GRADLE_DAEMONS_VIEW, RECENT_TASKS_VIEW, + GRADLE_DEFAULT_PROJECTS_VIEW, } from './views/constants'; import { focusTaskInGradleTasksTree } from './views/viewUtil'; import { COMMAND_RENDER_TASK, COMMAND_REFRESH } from './commands'; @@ -39,6 +40,8 @@ import { DependencyTreeItem } from './views/gradleTasks/DependencyTreeItem'; import { GRADLE_DEPENDENCY_REVEAL } from './views/gradleTasks/DependencyUtils'; import { GradleDependencyProvider } from './dependencies/GradleDependencyProvider'; import { getSpecificVersionStatus } from './views/gradleDaemons/util'; +import { startLanguageServer } from './languageServer/languageServer'; +import { DefaultProjectsTreeDataProvider } from './views/defaultProject/DefaultProjectsTreeDataProvider'; export class Extension { private readonly client: GradleClient; @@ -62,11 +65,14 @@ export class Extension { private readonly recentTasksTreeDataProvider: RecentTasksTreeDataProvider; private readonly recentTasksTreeView: vscode.TreeView; private readonly gradleTasksTreeDataProvider: GradleTasksTreeDataProvider; + private readonly defaultProjectsTreeView: vscode.TreeView; + private readonly defaultProjectsTreeDataProvider: DefaultProjectsTreeDataProvider; private readonly api: Api; private readonly commands: Commands; - private readonly _onDidTerminalOpen: vscode.EventEmitter = new vscode.EventEmitter(); - private readonly onDidTerminalOpen: vscode.Event = this - ._onDidTerminalOpen.event; + private readonly _onDidTerminalOpen: vscode.EventEmitter = + new vscode.EventEmitter(); + private readonly onDidTerminalOpen: vscode.Event = + this._onDidTerminalOpen.event; private recentTerminal: vscode.Terminal | undefined; public constructor(private readonly context: vscode.ExtensionContext) { @@ -152,6 +158,19 @@ export class Extension { treeDataProvider: this.recentTasksTreeDataProvider, showCollapseAll: false, }); + this.defaultProjectsTreeDataProvider = new DefaultProjectsTreeDataProvider( + this.gradleTaskProvider, + this.rootProjectsStore, + this.client, + this.icons + ); + this.defaultProjectsTreeView = vscode.window.createTreeView( + GRADLE_DEFAULT_PROJECTS_VIEW, + { + treeDataProvider: this.defaultProjectsTreeDataProvider, + showCollapseAll: false, + } + ); this.gradleTaskManager = new GradleTaskManager(context); this.buildFileWatcher = new FileWatcher('**/*.{gradle,gradle.kts}'); @@ -198,6 +217,7 @@ export class Extension { ); void this.activate(); + void startLanguageServer(this.context); } private storeSubscriptions(): void { @@ -215,7 +235,8 @@ export class Extension { this.gradleDaemonsTreeView, this.gradleTasksTreeView, this.pinnedTasksTreeView, - this.recentTasksTreeView + this.recentTasksTreeView, + this.defaultProjectsTreeView ); } @@ -229,6 +250,11 @@ export class Extension { 'gradle:activated', activated ); + await vscode.commands.executeCommand( + 'setContext', + 'gradle:defaultView', + true + ); } private registerCommands(): void { diff --git a/extension/src/client/GradleClient.ts b/extension/src/client/GradleClient.ts index 52fedd19b..1b475b113 100644 --- a/extension/src/client/GradleClient.ts +++ b/extension/src/client/GradleClient.ts @@ -46,6 +46,10 @@ import { } from './CancellationKeys'; import { EventWaiter } from '../util/EventWaiter'; import { getGradleConfig, getConfigJavaDebug } from '../util/config'; +import { + setDefault, + unsetDefault, +} from '../views/defaultProject/DefaultProjectUtils'; function logBuildEnvironment(environment: Environment): void { const javaEnv = environment.getJavaEnvironment()!; @@ -59,11 +63,13 @@ function logBuildEnvironment(environment: Environment): void { export class GradleClient implements vscode.Disposable { private readonly connectDeadline = 30; // seconds private grpcClient: GrpcClient | null = null; - private readonly _onDidConnect: vscode.EventEmitter = new vscode.EventEmitter(); - private readonly _onDidConnectFail: vscode.EventEmitter = new vscode.EventEmitter(); + private readonly _onDidConnect: vscode.EventEmitter = + new vscode.EventEmitter(); + private readonly _onDidConnectFail: vscode.EventEmitter = + new vscode.EventEmitter(); public readonly onDidConnect: vscode.Event = this._onDidConnect.event; - public readonly onDidConnectFail: vscode.Event = this._onDidConnectFail - .event; + public readonly onDidConnectFail: vscode.Event = + this._onDidConnectFail.event; private readonly waitForConnect = new EventWaiter(this.onDidConnect).wait; @@ -189,6 +195,7 @@ export class GradleClient implements vscode.Disposable { ); break; case Output.OutputType.STDERR: + void setDefault(); stdErrLoggerStream.write( getBuildReply.getOutput()!.getOutputBytes_asU8() ); @@ -199,6 +206,7 @@ export class GradleClient implements vscode.Disposable { this.handleGetBuildCancelled(getBuildReply.getCancelled()!); break; case GetBuildReply.KindCase.GET_BUILD_RESULT: + void unsetDefault(); build = getBuildReply.getGetBuildResult()!.getBuild(); break; case GetBuildReply.KindCase.ENVIRONMENT: @@ -625,9 +633,8 @@ export class GradleClient implements vscode.Disposable { this.close(); this._onDidConnectFail.fire(null); if (this.server.isReady()) { - const connectivityState = this.grpcClient!.getChannel().getConnectivityState( - true - ); + const connectivityState = + this.grpcClient!.getChannel().getConnectivityState(true); const enumKey = ConnectivityState[connectivityState]; logger.error('The client has state:', enumKey); await this.showRestartMessage(); diff --git a/extension/src/commands/StopDaemonsCommand.ts b/extension/src/commands/StopDaemonsCommand.ts index 3d039cf37..05dc59120 100644 --- a/extension/src/commands/StopDaemonsCommand.ts +++ b/extension/src/commands/StopDaemonsCommand.ts @@ -22,7 +22,8 @@ export class StopDaemonsCommand extends Command { ) { return; } - const gradleRootFolders = await this.rootProjectsStore.getProjectRootsWithUniqueVersions(); + const gradleRootFolders = + await this.rootProjectsStore.getProjectRootsWithUniqueVersions(); const promises: Promise[] = gradleRootFolders.map( (rootProject) => this.client.stopDaemons(rootProject.getProjectUri().fsPath) diff --git a/extension/src/languageServer/languageServer.ts b/extension/src/languageServer/languageServer.ts new file mode 100644 index 000000000..f33503a2f --- /dev/null +++ b/extension/src/languageServer/languageServer.ts @@ -0,0 +1,138 @@ +/* -------------------------------------------------------------------------------------------- + * Copyright (c) Microsoft Corporation. All rights reserved. + * Licensed under the MIT License. See License.txt in the project root for license information. + * ------------------------------------------------------------------------------------------ */ + +import * as net from 'net'; +import * as path from 'path'; +import * as vscode from 'vscode'; +import { + DidChangeConfigurationNotification, + LanguageClientOptions, +} from 'vscode-languageclient'; +import { LanguageClient, StreamInfo } from 'vscode-languageclient/node'; +import { + getConfigGradleJavaHome, + getConfigJavaImportGradleHome, + getConfigJavaImportGradleUserHome, + getConfigJavaImportGradleVersion, + getConfigJavaImportGradleWrapperEnabled, +} from '../util/config'; +const CHANNEL_NAME = 'Gradle Language Server'; + +export let isLanguageServerStarted = false; + +export async function startLanguageServer( + context: vscode.ExtensionContext +): Promise { + void vscode.window.withProgress( + { location: vscode.ProgressLocation.Window }, + // eslint-disable-next-line sonarjs/cognitive-complexity + (progress) => { + // eslint-disable-next-line @typescript-eslint/no-unused-vars + return new Promise(async (resolve, _reject) => { + progress.report({ + message: 'Initializing Gradle Language Server', + }); + const clientOptions: LanguageClientOptions = { + documentSelector: [{ scheme: 'file', language: 'gradle' }], + outputChannel: vscode.window.createOutputChannel(CHANNEL_NAME), + outputChannelName: CHANNEL_NAME, + initializationOptions: { + settings: getGradleSettings(), + }, + }; + let serverOptions; + if (process.env.VSCODE_DEBUG_LANGUAGE_SERVER === 'true') { + // debug mode + const port = process.env.VSCODE_GRADLE_PORT; + if (!port) { + void vscode.window.showErrorMessage( + 'VSCODE_GRADLE_PORT is invalid, please check it in launch.json.' + ); + return; + } + serverOptions = awaitServerConnection.bind(null, port); + } else { + // keep consistent with gRPC server + const javaHome = getConfigGradleJavaHome() || process.env.JAVA_HOME; + if (!javaHome) { + void vscode.window + .showErrorMessage( + 'There is no valid JAVA_HOME setting to launch Gradle Language Server. Please check your "java.home" setting.', + 'Open Settings' + ) + .then((answer) => { + if (answer === 'Open Settings') { + void vscode.commands.executeCommand( + 'workbench.action.openSettings', + 'java.home' + ); + } + }); + return; + } + const args = [ + '-jar', + path.resolve( + context.extensionPath, + 'lib', + 'gradle-language-server.jar' + ), + ]; + serverOptions = { + command: path.join(javaHome, 'bin', 'java'), + args: args, + }; + } + const languageClient = new LanguageClient( + 'gradle', + 'Gradle Language Server', + serverOptions, + clientOptions + ); + languageClient.onReady().then(resolve, (e) => { + resolve(); + isLanguageServerStarted = true; + void vscode.window.showErrorMessage(e); + }); + const disposable = languageClient.start(); + context.subscriptions.push(disposable); + context.subscriptions.push( + vscode.workspace.onDidChangeConfiguration((e) => { + if (e.affectsConfiguration('java.import.gradle')) { + languageClient.sendNotification( + DidChangeConfigurationNotification.type, + { settings: getGradleSettings() } + ); + } + }) + ); + }); + } + ); +} + +async function awaitServerConnection(port: string): Promise { + const addr = parseInt(port); + return new Promise((resolve, reject) => { + const server = net.createServer((stream) => { + server.close(); + resolve({ reader: stream, writer: stream }); + }); + server.on('error', reject); + server.listen(addr, () => { + server.removeListener('error', reject); + }); + return server; + }); +} + +function getGradleSettings(): unknown { + return { + gradleHome: getConfigJavaImportGradleHome(), + gradleVersion: getConfigJavaImportGradleVersion(), + gradleWrapperEnabled: getConfigJavaImportGradleWrapperEnabled(), + gradleUserHome: getConfigJavaImportGradleUserHome(), + }; +} diff --git a/extension/src/progress/ProgressHandler.ts b/extension/src/progress/ProgressHandler.ts index 3ca23ac04..547176e24 100644 --- a/extension/src/progress/ProgressHandler.ts +++ b/extension/src/progress/ProgressHandler.ts @@ -1,9 +1,10 @@ import * as vscode from 'vscode'; export class ProgressHandler { - private readonly _onDidProgressStart: vscode.EventEmitter = new vscode.EventEmitter(); - public readonly onDidProgressStart: vscode.Event = this - ._onDidProgressStart.event; + private readonly _onDidProgressStart: vscode.EventEmitter = + new vscode.EventEmitter(); + public readonly onDidProgressStart: vscode.Event = + this._onDidProgressStart.event; public constructor( private readonly progress: vscode.Progress<{ message?: string }>, diff --git a/extension/src/server/GradleServer.ts b/extension/src/server/GradleServer.ts index c53f46c17..4ece89bc5 100644 --- a/extension/src/server/GradleServer.ts +++ b/extension/src/server/GradleServer.ts @@ -15,8 +15,10 @@ export interface ServerOptions { } export class GradleServer implements vscode.Disposable { - private readonly _onDidStart: vscode.EventEmitter = new vscode.EventEmitter(); - private readonly _onDidStop: vscode.EventEmitter = new vscode.EventEmitter(); + private readonly _onDidStart: vscode.EventEmitter = + new vscode.EventEmitter(); + private readonly _onDidStop: vscode.EventEmitter = + new vscode.EventEmitter(); private ready = false; private port: number | undefined; private restarting = false; diff --git a/extension/src/stores/EventedStore.ts b/extension/src/stores/EventedStore.ts index 9147d109a..c79b076cf 100644 --- a/extension/src/stores/EventedStore.ts +++ b/extension/src/stores/EventedStore.ts @@ -2,7 +2,8 @@ import * as vscode from 'vscode'; import { debounce } from '../util/decorators'; export abstract class EventedStore implements vscode.Disposable { - private readonly _onDidChange: vscode.EventEmitter = new vscode.EventEmitter(); + private readonly _onDidChange: vscode.EventEmitter = + new vscode.EventEmitter(); public readonly onDidChange: vscode.Event = this._onDidChange.event; @debounce(0) diff --git a/extension/src/stores/RootProjectsStore.ts b/extension/src/stores/RootProjectsStore.ts index 6811e899d..800d71e2f 100644 --- a/extension/src/stores/RootProjectsStore.ts +++ b/extension/src/stores/RootProjectsStore.ts @@ -50,11 +50,12 @@ export class RootProjectsStore extends StoreMap { for (const workspaceFolder of workspaceFolders) { const configNestedFolders = getNestedProjectsConfig(workspaceFolder); - const gradleProjectFoldersOutsideRoot = getGradleProjectFoldersOutsideRoot( - configNestedFolders, - gradleProjectFolders, - workspaceFolder - ); + const gradleProjectFoldersOutsideRoot = + getGradleProjectFoldersOutsideRoot( + configNestedFolders, + gradleProjectFolders, + workspaceFolder + ); if (gradleProjectFolders.includes(workspaceFolder.uri.fsPath)) { this.setRootProjectFolder(buildRootFolder(workspaceFolder.uri)); } diff --git a/extension/src/tasks/GradleTaskManager.ts b/extension/src/tasks/GradleTaskManager.ts index c89532bad..e6d62e47a 100644 --- a/extension/src/tasks/GradleTaskManager.ts +++ b/extension/src/tasks/GradleTaskManager.ts @@ -2,16 +2,19 @@ import * as vscode from 'vscode'; import { isGradleTask, getRunningGradleTasks } from './taskUtil'; export class GradleTaskManager implements vscode.Disposable { - private readonly _onDidStartTask: vscode.EventEmitter = new vscode.EventEmitter(); - private readonly _onDidEndTask: vscode.EventEmitter = new vscode.EventEmitter(); - private readonly _onDidEndAllTasks: vscode.EventEmitter = new vscode.EventEmitter(); + private readonly _onDidStartTask: vscode.EventEmitter = + new vscode.EventEmitter(); + private readonly _onDidEndTask: vscode.EventEmitter = + new vscode.EventEmitter(); + private readonly _onDidEndAllTasks: vscode.EventEmitter = + new vscode.EventEmitter(); - public readonly onDidStartTask: vscode.Event = this - ._onDidStartTask.event; - public readonly onDidEndTask: vscode.Event = this._onDidEndTask - .event; - public readonly onDidEndAllTasks: vscode.Event = this._onDidEndAllTasks - .event; + public readonly onDidStartTask: vscode.Event = + this._onDidStartTask.event; + public readonly onDidEndTask: vscode.Event = + this._onDidEndTask.event; + public readonly onDidEndAllTasks: vscode.Event = + this._onDidEndAllTasks.event; constructor(private readonly context: vscode.ExtensionContext) { this.context.subscriptions.push( diff --git a/extension/src/tasks/GradleTaskProvider.ts b/extension/src/tasks/GradleTaskProvider.ts index d09017520..be3c0211f 100644 --- a/extension/src/tasks/GradleTaskProvider.ts +++ b/extension/src/tasks/GradleTaskProvider.ts @@ -10,25 +10,27 @@ import { getConfigJavaDebug } from '../util/config'; import { EventWaiter } from '../util/EventWaiter'; export class GradleTaskProvider - implements vscode.TaskProvider, vscode.Disposable { + implements vscode.TaskProvider, vscode.Disposable +{ private cachedTasks: vscode.Task[] = []; - private readonly _onDidLoadTasks: vscode.EventEmitter< - vscode.Task[] - > = new vscode.EventEmitter(); - private readonly _onDidStartRefresh: vscode.EventEmitter = new vscode.EventEmitter(); - private readonly _onDidStopRefresh: vscode.EventEmitter = new vscode.EventEmitter(); + private readonly _onDidLoadTasks: vscode.EventEmitter = + new vscode.EventEmitter(); + private readonly _onDidStartRefresh: vscode.EventEmitter = + new vscode.EventEmitter(); + private readonly _onDidStopRefresh: vscode.EventEmitter = + new vscode.EventEmitter(); constructor( private readonly rootProjectsStore: RootProjectsStore, private readonly client: GradleClient ) {} - public readonly onDidLoadTasks: vscode.Event = this - ._onDidLoadTasks.event; - public readonly onDidStartRefresh: vscode.Event = this - ._onDidStartRefresh.event; - public readonly onDidStopRefresh: vscode.Event = this._onDidStopRefresh - .event; + public readonly onDidLoadTasks: vscode.Event = + this._onDidLoadTasks.event; + public readonly onDidStartRefresh: vscode.Event = + this._onDidStartRefresh.event; + public readonly onDidStopRefresh: vscode.Event = + this._onDidStopRefresh.event; private loadTasksPromise?: Promise; private readonly _waitForTasksLoad = new EventWaiter( diff --git a/extension/src/test/unit/gradleDaemons.test.ts b/extension/src/test/unit/gradleDaemons.test.ts index d060892c2..5b183bfad 100644 --- a/extension/src/test/unit/gradleDaemons.test.ts +++ b/extension/src/test/unit/gradleDaemons.test.ts @@ -100,7 +100,8 @@ describe(getSuiteName('Gradle daemons'), () => { }); it('should filter out projects with duplicate gradle versions', async () => { - const projects = await rootProjectsStore.getProjectRootsWithUniqueVersions(); + const projects = + await rootProjectsStore.getProjectRootsWithUniqueVersions(); assert.strictEqual( projects.length, 2, @@ -230,10 +231,9 @@ describe(getSuiteName('Gradle daemons'), () => { mockReply.setMessage('Stopped'); mockClient.stopDaemon.resolves(mockReply); - const showWarningMessageStub = (sinon.stub( - vscode.window, - 'showWarningMessage' - ) as SinonStub).resolves('Yes'); + const showWarningMessageStub = ( + sinon.stub(vscode.window, 'showWarningMessage') as SinonStub + ).resolves('Yes'); const mockDaemonInfoBusy = new DaemonInfo(); mockDaemonInfoBusy.setStatus(DaemonInfo.DaemonStatus.BUSY); @@ -280,10 +280,9 @@ describe(getSuiteName('Gradle daemons'), () => { .withArgs(mockWorkspaceFolder2.uri.fsPath) .resolves(mockReply2); - const showWarningMessageStub = (sinon.stub( - vscode.window, - 'showWarningMessage' - ) as SinonStub).resolves('Yes'); + const showWarningMessageStub = ( + sinon.stub(vscode.window, 'showWarningMessage') as SinonStub + ).resolves('Yes'); await new StopDaemonsCommand(mockClient, rootProjectsStore).run(); diff --git a/extension/src/test/unit/gradleTasks.test.ts b/extension/src/test/unit/gradleTasks.test.ts index 13cec4bbf..5715fbea0 100644 --- a/extension/src/test/unit/gradleTasks.test.ts +++ b/extension/src/test/unit/gradleTasks.test.ts @@ -336,7 +336,8 @@ describe(getSuiteName('Gradle tasks'), () => { task, }, ]); - const gradleProjects = (await gradleTasksTreeDataProvider.getChildren()) as ProjectTreeItem[]; + const gradleProjects = + (await gradleTasksTreeDataProvider.getChildren()) as ProjectTreeItem[]; removeCancellingTask(task); const group = gradleProjects[0].groups[0]; const taskItem = group.tasks[0]; @@ -383,7 +384,8 @@ describe(getSuiteName('Gradle tasks'), () => { executeCommandStub.calledWith(COMMAND_RENDER_TASK, task), 'Task was not rendered' ); - const gradleProjects = (await gradleTasksTreeDataProvider.getChildren()) as ProjectTreeItem[]; + const gradleProjects = + (await gradleTasksTreeDataProvider.getChildren()) as ProjectTreeItem[]; removeCancellingTask(task); const group = gradleProjects[0].groups[0]; const taskItem = group.tasks[0]; @@ -430,7 +432,8 @@ describe(getSuiteName('Gradle tasks'), () => { }); it('should build root RootProject items at top level', async () => { - const rootProjectItems = await gradleTasksTreeDataProvider.getChildren(); + const rootProjectItems = + await gradleTasksTreeDataProvider.getChildren(); assert.ok(rootProjectItems.length > 0, 'No root gradle projects found'); assert.strictEqual( rootProjectItems.length, diff --git a/extension/src/test/unit/pinnedTasks.test.ts b/extension/src/test/unit/pinnedTasks.test.ts index bec2e843f..4345928d8 100644 --- a/extension/src/test/unit/pinnedTasks.test.ts +++ b/extension/src/test/unit/pinnedTasks.test.ts @@ -348,10 +348,9 @@ describe(getSuiteName('Pinned tasks'), () => { childrenBefore[1] instanceof PinnedTaskTreeItem, 'Pinned task is not PinnedTaskTreeItem' ); - const showWarningMessageStub = (sinon.stub( - vscode.window, - 'showWarningMessage' - ) as SinonStub).resolves('Yes'); + const showWarningMessageStub = ( + sinon.stub(vscode.window, 'showWarningMessage') as SinonStub + ).resolves('Yes'); await new ClearAllPinnedTasksCommand(pinnedTasksStore).run(); assert.ok( showWarningMessageStub.calledWith( @@ -379,7 +378,8 @@ describe(getSuiteName('Pinned tasks'), () => { describe('With pinned tasks', () => { it('should build a pinned task treeitem', async () => { - const workspaceTreeItems = await gradleTasksTreeDataProvider.getChildren(); + const workspaceTreeItems = + await gradleTasksTreeDataProvider.getChildren(); assert.ok(workspaceTreeItems.length > 0, 'No gradle projects found'); const workspaceTreeItem = workspaceTreeItems[0] as RootProjectTreeItem; assert.ok( @@ -404,7 +404,8 @@ describe(getSuiteName('Pinned tasks'), () => { await new PinTaskCommand(pinnedTasksTreeDataProvider).run(taskItem); const children = await pinnedTasksTreeDataProvider.getChildren(); assert.strictEqual(children.length, 1); - const pinnedTasksRootProjectTreeItem = children[0] as PinnedTasksRootProjectTreeItem; + const pinnedTasksRootProjectTreeItem = + children[0] as PinnedTasksRootProjectTreeItem; assert.strictEqual( pinnedTasksRootProjectTreeItem.collapsibleState, vscode.TreeItemCollapsibleState.Expanded diff --git a/extension/src/test/unit/recentTasks.test.ts b/extension/src/test/unit/recentTasks.test.ts index 7592d392b..31d770416 100644 --- a/extension/src/test/unit/recentTasks.test.ts +++ b/extension/src/test/unit/recentTasks.test.ts @@ -171,7 +171,8 @@ describe(getSuiteName('Recent tasks'), () => { path.join('resources', 'light', ICON_GRADLE_TASK) ); - const workspaceTreeItem = recentTaskTreeItem.parentTreeItem as RecentTasksRootProjectTreeItem; + const workspaceTreeItem = + recentTaskTreeItem.parentTreeItem as RecentTasksRootProjectTreeItem; assert.ok( workspaceTreeItem, 'parentTreeItem must reference a WorkSpace tree item' @@ -229,7 +230,8 @@ describe(getSuiteName('Recent tasks'), () => { path.join('resources', 'light', ICON_GRADLE_TASK) ); - const workspaceTreeItem = recentTaskTreeItem.parentTreeItem as RecentTasksRootProjectTreeItem; + const workspaceTreeItem = + recentTaskTreeItem.parentTreeItem as RecentTasksRootProjectTreeItem; assert.ok( workspaceTreeItem instanceof RecentTasksRootProjectTreeItem, 'Tree item is not RecentTasksRootProjectTreeItem' @@ -247,10 +249,9 @@ describe(getSuiteName('Recent tasks'), () => { 'Task is not a RecentTaskTreeItem' ); - const showWarningMessageStub = (sinon.stub( - vscode.window, - 'showWarningMessage' - ) as SinonStub).resolves('Yes'); + const showWarningMessageStub = ( + sinon.stub(vscode.window, 'showWarningMessage') as SinonStub + ).resolves('Yes'); await new ClearAllRecentTasksCommand(recentTasksStore).run(); @@ -367,10 +368,9 @@ describe(getSuiteName('Recent tasks'), () => { ); assert.strictEqual(recentTaskTreeItemBefore.description, '(2)'); - const showWarningMessageStub = (sinon.stub( - vscode.window, - 'showWarningMessage' - ) as SinonStub).resolves('Yes'); + const showWarningMessageStub = ( + sinon.stub(vscode.window, 'showWarningMessage') as SinonStub + ).resolves('Yes'); await new CloseAllTaskTerminalsCommand(taskTerminalsStore).run(); diff --git a/extension/src/util/FileWatcher.ts b/extension/src/util/FileWatcher.ts index f9f2d9f7d..754a7f8af 100644 --- a/extension/src/util/FileWatcher.ts +++ b/extension/src/util/FileWatcher.ts @@ -2,9 +2,10 @@ import * as vscode from 'vscode'; import * as minimatch from 'minimatch'; export class FileWatcher implements vscode.Disposable { - private readonly _onDidChange: vscode.EventEmitter = new vscode.EventEmitter(); - public readonly onDidChange: vscode.Event = this._onDidChange - .event; + private readonly _onDidChange: vscode.EventEmitter = + new vscode.EventEmitter(); + public readonly onDidChange: vscode.Event = + this._onDidChange.event; private onDidSaveDisposable: vscode.Disposable; private onDidDeleteDisposable: vscode.Disposable; private onDidCreateDisposable: vscode.Disposable; diff --git a/extension/src/util/compat.ts b/extension/src/util/compat.ts index 50153d3b4..8a93e774b 100644 --- a/extension/src/util/compat.ts +++ b/extension/src/util/compat.ts @@ -4,16 +4,14 @@ export const JAVA_LANGUAGE_EXTENSION_ID = 'redhat.java'; export const JAVA_DEBUGGER_EXTENSION_ID = 'vscjava.vscode-java-debug'; export function isJavaLanguageSupportExtensionActivated(): boolean { - const javaExt: - | vscode.Extension - | undefined = getJavaLanguageSupportExtension(); + const javaExt: vscode.Extension | undefined = + getJavaLanguageSupportExtension(); return javaExt?.isActive || false; } export function isJavaDebuggerExtensionActivated(): boolean { - const javaExt: - | vscode.Extension - | undefined = getJavaDebuggerExtension(); + const javaExt: vscode.Extension | undefined = + getJavaDebuggerExtension(); return javaExt?.isActive || false; } diff --git a/extension/src/views/constants.ts b/extension/src/views/constants.ts index 092a552c9..1f0abeacb 100644 --- a/extension/src/views/constants.ts +++ b/extension/src/views/constants.ts @@ -8,6 +8,7 @@ export const ICON_DAEMON_STOPPED = 'close.svg'; export const GRADLE_CONTAINER_VIEW = 'gradleContainerView'; export const GRADLE_TASKS_VIEW = 'gradleTasksView'; +export const GRADLE_DEFAULT_PROJECTS_VIEW = 'gradleDefaultProjectsView'; export const GRADLE_DAEMONS_VIEW = 'gradleDaemonsView'; export const PINNED_TASKS_VIEW = 'pinnedTasksView'; export const RECENT_TASKS_VIEW = 'recentTasksView'; diff --git a/extension/src/views/defaultProject/DefaultProjectProvider.ts b/extension/src/views/defaultProject/DefaultProjectProvider.ts new file mode 100644 index 000000000..2e99b963f --- /dev/null +++ b/extension/src/views/defaultProject/DefaultProjectProvider.ts @@ -0,0 +1,142 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT license. + +import * as path from 'path'; +import * as vscode from 'vscode'; +import { GradleClient } from '../../client'; +import { isLanguageServerStarted } from '../../languageServer/languageServer'; +import { RootProject } from '../../rootProject'; +import { GradleTaskDefinition } from '../../tasks'; +import { buildTaskId, createTaskFromDefinition } from '../../tasks/taskUtil'; +import { DependencyConfigurationTreeItem } from '../gradleTasks/DependencyConfigurationTreeItem'; +import { DependencyTreeItem } from '../gradleTasks/DependencyTreeItem'; +import { HintItem } from '../gradleTasks/HintItem'; +import { ProjectDependencyTreeItem } from '../gradleTasks/ProjectDependencyTreeItem'; +import { generateDefaultTaskDefinitions } from './DefaultProjectUtils'; +import { DefaultDependencyItem, DefaultTaskDefinition } from './types'; + +export class DefaultProjectProvider { + private static defaultTaskDefinitions: DefaultTaskDefinition[] = + generateDefaultTaskDefinitions(); + + constructor() { + this.defaultTasks = []; + this.defaultDependencies = new Map(); + } + + private defaultTasks: vscode.Task[]; + private defaultDependencies: Map; + + public refresh(): void { + this.defaultTasks = []; + } + + public async getDefaultTasks( + rootProjects: RootProject[], + client: GradleClient + ): Promise { + if (this.defaultTasks.length) { + return this.defaultTasks; + } + const tasks = []; + for (const rootProject of rootProjects) { + for (const defaultTaskDefinition of DefaultProjectProvider.defaultTaskDefinitions) { + tasks.push( + createTaskFromDefinition( + this.generateDefaultTaskDefinition( + rootProject, + defaultTaskDefinition.name, + rootProject.getWorkspaceFolder().name, + path.join(rootProject.getProjectUri().fsPath, 'build.gradle'), + defaultTaskDefinition.description, + defaultTaskDefinition.group + ), + rootProject, + client + ) + ); + } + } + this.defaultTasks = tasks; + return tasks; + } + + private generateDefaultTaskDefinition( + rootProject: RootProject, + script: string, + projectName: string, + projectFile: string, + description: string, + group?: string + ): Required { + return { + type: 'gradle', + id: buildTaskId(rootProject.getProjectUri().fsPath, script, projectName), + script, + description: description, + group: (group || 'other').toLowerCase(), + project: projectName, + buildFile: projectFile, + rootProject: projectName, + projectFolder: rootProject.getProjectUri().fsPath, + workspaceFolder: rootProject.getWorkspaceFolder().uri.fsPath, + args: '', + javaDebug: false, + }; + } + + public async getDefaultDependencyItems( + element: ProjectDependencyTreeItem + ): Promise { + const configurationMap: Map = + new Map(); + const buildFileUri = vscode.Uri.file( + // Due to no project knowledge, we only get informations from project root's build.gradle + path.join(element.projectPath, 'build.gradle') + ); + if (!isLanguageServerStarted) { + return [new HintItem('No dependencies')]; + } + const dependencyItems = await vscode.commands.executeCommand< + DefaultDependencyItem[] + >('gradle.getDependencies', buildFileUri.toString()); + if (!dependencyItems) { + return [new HintItem('No dependencies')]; + } + for (const dependencyItem of dependencyItems) { + const configuration = dependencyItem.configuration; + if (!configurationMap.has(configuration)) { + const configItem: DependencyConfigurationTreeItem = + new DependencyConfigurationTreeItem( + dependencyItem.configuration, + vscode.TreeItemCollapsibleState.Collapsed, + element + ); + configurationMap.set(configuration, configItem); + } + const configurationItem = configurationMap.get(configuration); + if (!configurationItem) { + continue; + } + const dependencyTreeItem: DependencyTreeItem = new DependencyTreeItem( + dependencyItem.name, + vscode.TreeItemCollapsibleState.None, + configurationItem + ); + dependencyTreeItem.command = { + title: 'Show Dependency', + command: 'vscode.open', + arguments: [ + buildFileUri, + { selection: dependencyItem.range }, + ], + }; + configurationItem.getChildren().push(dependencyTreeItem); + } + this.defaultDependencies.set( + buildFileUri.toString(), + Array.from(configurationMap.values()) + ); + return this.defaultDependencies.get(buildFileUri.toString()) || []; + } +} diff --git a/extension/src/views/defaultProject/DefaultProjectUtils.ts b/extension/src/views/defaultProject/DefaultProjectUtils.ts new file mode 100644 index 000000000..88b2b7169 --- /dev/null +++ b/extension/src/views/defaultProject/DefaultProjectUtils.ts @@ -0,0 +1,111 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT license. + +import * as vscode from 'vscode'; +import { DefaultTaskDefinition } from './types'; + +export async function setDefault(): Promise { + await vscode.commands.executeCommand( + 'setContext', + 'gradle:defaultView', + true + ); +} + +export async function unsetDefault(): Promise { + await vscode.commands.executeCommand( + 'setContext', + 'gradle:defaultView', + false + ); +} + +export function generateDefaultTaskDefinitions(): DefaultTaskDefinition[] { + const result = []; + result.push( + getDefaultTaskDefinition( + 'init', + 'build setup', + 'Initializes a new Gradle build.' + ) + ); + result.push( + getDefaultTaskDefinition( + 'wrapper', + 'build setup', + 'Generates Gradle wrapper files.' + ) + ); + result.push( + getDefaultTaskDefinition( + 'buildEnvironment', + 'help', + 'Displays all buildscript dependencies declared in root project.' + ) + ); + result.push( + getDefaultTaskDefinition( + 'dependencies', + 'help', + 'Displays all dependencies declared in root project.' + ) + ); + result.push( + getDefaultTaskDefinition( + 'dependencyInsight', + 'help', + 'Displays the insight into a specific dependency in root project.' + ) + ); + result.push( + getDefaultTaskDefinition('help', 'help', 'Displays a help message.') + ); + result.push( + getDefaultTaskDefinition( + 'javaToolchains', + 'help', + 'Displays the detected java toolchains.' + ) + ); + result.push( + getDefaultTaskDefinition( + 'outgoingVariants', + 'help', + 'Displays the outgoing variants of root project.' + ) + ); + result.push( + getDefaultTaskDefinition( + 'projects', + 'help', + 'Displays the sub-projects of root project.' + ) + ); + result.push( + getDefaultTaskDefinition( + 'properties', + 'help', + 'Displays the properties of root project.' + ) + ); + result.push( + getDefaultTaskDefinition( + 'tasks', + 'help', + 'Displays the tasks runnable from root project.' + ) + ); + return result; +} + +export function getDefaultTaskDefinition( + name: string, + group: string, + description: string +): DefaultTaskDefinition { + return { + name: name, + group: group, + description: description, + }; +} diff --git a/extension/src/views/defaultProject/DefaultProjectsTreeDataProvider.ts b/extension/src/views/defaultProject/DefaultProjectsTreeDataProvider.ts new file mode 100644 index 000000000..b875f6d1f --- /dev/null +++ b/extension/src/views/defaultProject/DefaultProjectsTreeDataProvider.ts @@ -0,0 +1,62 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT license. + +import * as vscode from 'vscode'; +import { + GradleTasksTreeDataProvider, + GroupTreeItem, + ProjectTreeItem, +} from '..'; +import { GradleClient } from '../../client'; +import { RootProjectsStore } from '../../stores'; +import { GradleTaskProvider } from '../../tasks'; +import { DefaultProjectProvider } from './DefaultProjectProvider'; +import { Icons } from '../../icons'; +import { ProjectDependencyTreeItem } from '../gradleTasks/ProjectDependencyTreeItem'; +import { ProjectTaskTreeItem } from '../gradleTasks/ProjectTaskTreeItem'; + +export class DefaultProjectsTreeDataProvider + implements vscode.TreeDataProvider +{ + private defaultProjectProvider: DefaultProjectProvider; + + constructor( + private readonly gradleTaskProvider: GradleTaskProvider, + private readonly rootProjectStore: RootProjectsStore, + private readonly client: GradleClient, + private readonly icons: Icons + ) { + this.defaultProjectProvider = new DefaultProjectProvider(); + } + + public getTreeItem(element: vscode.TreeItem): vscode.TreeItem { + return element; + } + + public async getChildren( + element?: vscode.TreeItem + ): Promise { + if (!element) { + // async configuring + void this.gradleTaskProvider.loadTasks(); + return GradleTasksTreeDataProvider.buildItemsTreeFromTasks( + await this.defaultProjectProvider.getDefaultTasks( + await this.rootProjectStore.getProjectRoots(), + this.client + ), + this.rootProjectStore, + false, + this.icons + ); + } else if (element instanceof ProjectTreeItem) { + return GradleTasksTreeDataProvider.getChildrenForProjectTreeItem(element); + } else if (element instanceof ProjectDependencyTreeItem) { + return this.defaultProjectProvider.getDefaultDependencyItems(element); + } else if (element instanceof ProjectTaskTreeItem) { + return element.getChildren() || []; + } else if (element instanceof GroupTreeItem) { + return element.tasks; + } + return []; + } +} diff --git a/extension/src/views/defaultProject/types.ts b/extension/src/views/defaultProject/types.ts new file mode 100644 index 000000000..23db8d514 --- /dev/null +++ b/extension/src/views/defaultProject/types.ts @@ -0,0 +1,16 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT license. + +import { Range } from 'vscode-languageclient/node'; + +export interface DefaultDependencyItem { + name: string; + configuration: string; + range: Range; +} + +export interface DefaultTaskDefinition { + name: string; + group: string; + description: string; +} diff --git a/extension/src/views/gradleDaemons/GradleDaemonsTreeDataProvider.ts b/extension/src/views/gradleDaemons/GradleDaemonsTreeDataProvider.ts index e34d840fe..d45e65e21 100644 --- a/extension/src/views/gradleDaemons/GradleDaemonsTreeDataProvider.ts +++ b/extension/src/views/gradleDaemons/GradleDaemonsTreeDataProvider.ts @@ -12,13 +12,15 @@ import { HintItem } from '../gradleTasks/HintItem'; import { getSpecificVersionStatus } from './util'; export class GradleDaemonsTreeDataProvider - implements vscode.TreeDataProvider { + implements vscode.TreeDataProvider +{ private cancelDeferred?: Deferred; private treeItems: vscode.TreeItem[] = []; private specificVersion = false; - private readonly _onDidChangeTreeData: vscode.EventEmitter = new vscode.EventEmitter(); - public readonly onDidChangeTreeData: vscode.Event = this - ._onDidChangeTreeData.event; + private readonly _onDidChangeTreeData: vscode.EventEmitter = + new vscode.EventEmitter(); + public readonly onDidChangeTreeData: vscode.Event = + this._onDidChangeTreeData.event; constructor( private readonly context: vscode.ExtensionContext, diff --git a/extension/src/views/gradleTasks/DependencyConfigurationTreeItem.ts b/extension/src/views/gradleTasks/DependencyConfigurationTreeItem.ts index 0403bb942..39cc7291d 100644 --- a/extension/src/views/gradleTasks/DependencyConfigurationTreeItem.ts +++ b/extension/src/views/gradleTasks/DependencyConfigurationTreeItem.ts @@ -4,7 +4,7 @@ import * as vscode from 'vscode'; export class DependencyConfigurationTreeItem extends vscode.TreeItem { - private children: vscode.TreeItem[] | undefined; + private children: vscode.TreeItem[]; constructor( name: string, collapsibleState: vscode.TreeItemCollapsibleState, @@ -13,13 +13,14 @@ export class DependencyConfigurationTreeItem extends vscode.TreeItem { ) { super(name, collapsibleState); this.iconPath = iconPath; + this.children = []; } public setChildren(children: vscode.TreeItem[]): void { this.children = children; } - public getChildren(): vscode.TreeItem[] | undefined { + public getChildren(): vscode.TreeItem[] { return this.children; } } diff --git a/extension/src/views/gradleTasks/DependencyUtils.ts b/extension/src/views/gradleTasks/DependencyUtils.ts index 97ffbbd44..9ba4b3bc9 100644 --- a/extension/src/views/gradleTasks/DependencyUtils.ts +++ b/extension/src/views/gradleTasks/DependencyUtils.ts @@ -38,11 +38,12 @@ function protocolItem2DependencyConfigurationTreeItem( ): DependencyConfigurationTreeItem | undefined { const name = protocolItem.getName(); const storageMap = new Map(); - const configurationItem: DependencyConfigurationTreeItem = new DependencyConfigurationTreeItem( - name, - vscode.TreeItemCollapsibleState.Collapsed, - parent - ); + const configurationItem: DependencyConfigurationTreeItem = + new DependencyConfigurationTreeItem( + name, + vscode.TreeItemCollapsibleState.Collapsed, + parent + ); const children = protocolItem.getChildrenList(); const treeChildren = []; for (const child of children) { diff --git a/extension/src/views/gradleTasks/GradleTasksTreeDataProvider.ts b/extension/src/views/gradleTasks/GradleTasksTreeDataProvider.ts index ebb9a04c9..95ee95123 100644 --- a/extension/src/views/gradleTasks/GradleTasksTreeDataProvider.ts +++ b/extension/src/views/gradleTasks/GradleTasksTreeDataProvider.ts @@ -41,12 +41,13 @@ function resetCachedTreeItems(): void { } export class GradleTasksTreeDataProvider - implements vscode.TreeDataProvider { + implements vscode.TreeDataProvider +{ private collapsed = true; - - private readonly _onDidChangeTreeData: vscode.EventEmitter = new vscode.EventEmitter(); - public readonly onDidChangeTreeData: vscode.Event = this - ._onDidChangeTreeData.event; + private readonly _onDidChangeTreeData: vscode.EventEmitter = + new vscode.EventEmitter(); + public readonly onDidChangeTreeData: vscode.Event = + this._onDidChangeTreeData.event; constructor( private readonly context: vscode.ExtensionContext, @@ -81,7 +82,12 @@ export class GradleTasksTreeDataProvider const tasks = await this.gradleTaskProvider.loadTasks(); return tasks.length === 0 ? [new NoGradleTasksTreeItem()] - : this.buildItemsTreeFromTasks(tasks); + : GradleTasksTreeDataProvider.buildItemsTreeFromTasks( + tasks, + this.rootProjectStore, + this.collapsed, + this.icons + ); } public refresh(treeItem: vscode.TreeItem | null = null): void { @@ -114,7 +120,7 @@ export class GradleTasksTreeDataProvider return element.projects; } if (element instanceof ProjectTreeItem) { - return this.getChildrenForProjectTreeItem(element); + return GradleTasksTreeDataProvider.getChildrenForProjectTreeItem(element); } if (element instanceof GroupTreeItem) { return element.tasks; @@ -136,12 +142,12 @@ export class GradleTasksTreeDataProvider return element.getChildren() || []; } if (!element) { - return await this.buildTreeItems(); + return this.buildTreeItems(); } return []; } - public async getChildrenForProjectTreeItem( + public static async getChildrenForProjectTreeItem( element: ProjectTreeItem ): Promise { const projectTaskItem = new ProjectTaskTreeItem( @@ -155,26 +161,30 @@ export class GradleTasksTreeDataProvider if (!resourceUri) { return results; } - const projectDependencyTreeItem: ProjectDependencyTreeItem = new ProjectDependencyTreeItem( - 'Dependencies', - vscode.TreeItemCollapsibleState.Collapsed, - element, - path.dirname(resourceUri.fsPath), - typeof element.label === 'string' ? element.label : resourceUri.fsPath - ); + const projectDependencyTreeItem: ProjectDependencyTreeItem = + new ProjectDependencyTreeItem( + 'Dependencies', + vscode.TreeItemCollapsibleState.Collapsed, + element, + path.dirname(resourceUri.fsPath), + typeof element.label === 'string' ? element.label : resourceUri.fsPath + ); return [...results, projectDependencyTreeItem]; } // eslint-disable-next-line sonarjs/cognitive-complexity - public buildItemsTreeFromTasks( - tasks: vscode.Task[] + public static buildItemsTreeFromTasks( + tasks: vscode.Task[], + rootProjectStore: RootProjectsStore, + collapsed: boolean, + icons: Icons ): RootProjectTreeItem[] | NoGradleTasksTreeItem[] { let gradleProjectTreeItem = null; tasks.forEach((task) => { const definition = task.definition as GradleTaskDefinition; if (isWorkspaceFolder(task.scope) && isGradleTask(task)) { - const rootProject = this.rootProjectStore.get(definition.projectFolder); + const rootProject = rootProjectStore.get(definition.projectFolder); if (!rootProject) { return; } @@ -208,7 +218,7 @@ export class GradleTasksTreeDataProvider ); let parentTreeItem: ProjectTreeItem | GroupTreeItem = projectTreeItem; - if (!this.collapsed) { + if (!collapsed) { const groupId = definition.group + definition.project; let groupTreeItem = groupTreeItemMap.get(groupId); if (!groupTreeItem) { @@ -229,7 +239,7 @@ export class GradleTasksTreeDataProvider taskName, definition.description || taskName, '', - this.icons, + icons, rootProject.getJavaDebug() ); taskTreeItem.setContext(); diff --git a/extension/src/views/pinnedTasks/PinnedTasksTreeDataProvider.ts b/extension/src/views/pinnedTasks/PinnedTasksTreeDataProvider.ts index 87317dc68..260959a0e 100644 --- a/extension/src/views/pinnedTasks/PinnedTasksTreeDataProvider.ts +++ b/extension/src/views/pinnedTasks/PinnedTasksTreeDataProvider.ts @@ -83,10 +83,12 @@ function buildGradleProjectTreeItem( } export class PinnedTasksTreeDataProvider - implements vscode.TreeDataProvider { - private readonly _onDidChangeTreeData: vscode.EventEmitter = new vscode.EventEmitter(); - public readonly onDidChangeTreeData: vscode.Event = this - ._onDidChangeTreeData.event; + implements vscode.TreeDataProvider +{ + private readonly _onDidChangeTreeData: vscode.EventEmitter = + new vscode.EventEmitter(); + public readonly onDidChangeTreeData: vscode.Event = + this._onDidChangeTreeData.event; constructor( private readonly pinnedTasksStore: PinnedTasksStore, diff --git a/extension/src/views/recentTasks/RecentTasksTreeDataProvider.ts b/extension/src/views/recentTasks/RecentTasksTreeDataProvider.ts index ee17ca9f2..e27695c32 100644 --- a/extension/src/views/recentTasks/RecentTasksTreeDataProvider.ts +++ b/extension/src/views/recentTasks/RecentTasksTreeDataProvider.ts @@ -90,10 +90,12 @@ function buildGradleProjectTreeItem( } export class RecentTasksTreeDataProvider - implements vscode.TreeDataProvider { - private readonly _onDidChangeTreeData: vscode.EventEmitter = new vscode.EventEmitter(); - public readonly onDidChangeTreeData: vscode.Event = this - ._onDidChangeTreeData.event; + implements vscode.TreeDataProvider +{ + private readonly _onDidChangeTreeData: vscode.EventEmitter = + new vscode.EventEmitter(); + public readonly onDidChangeTreeData: vscode.Event = + this._onDidChangeTreeData.event; constructor( private readonly recentTasksStore: RecentTasksStore, diff --git a/extension/syntaxes/groovy.tmLanguage.json b/extension/syntaxes/groovy.tmLanguage.json new file mode 100644 index 000000000..06b61b1b6 --- /dev/null +++ b/extension/syntaxes/groovy.tmLanguage.json @@ -0,0 +1,1387 @@ +{ + "information_for_contributors": [ + "This file has been converted from https://github.com/textmate/groovy.tmbundle/blob/master/Syntaxes/Groovy.tmLanguage", + "If you want to provide a fix or improvement, please create a pull request against the original repository.", + "Once accepted there, we are happy to receive an update request." + ], + "version": "https://github.com/textmate/groovy.tmbundle/blob/6f903cacfb2d5397a350eeb73bc36b2c40f3da70/Syntaxes/Groovy.tmLanguage", + "scopeName": "source.groovy", + "fileTypes": ["groovy", "gvy", "gradle"], + "foldingStartMarker": "(\\{\\s*$|^\\s*// \\{\\{\\{)", + "foldingStopMarker": "^\\s*(\\}|// \\}\\}\\}$)", + "keyEquivalent": "^~G", + "name": "Gradle", + "patterns": [ + { + "captures": { + "1": { + "name": "punctuation.definition.comment.groovy" + } + }, + "match": "^(#!).+$\\n", + "name": "comment.line.hashbang.groovy" + }, + { + "captures": { + "1": { + "name": "keyword.other.package.groovy" + }, + "2": { + "name": "storage.modifier.package.groovy" + }, + "3": { + "name": "punctuation.terminator.groovy" + } + }, + "match": "^\\s*(package)\\b(?:\\s*([^ ;$]+)\\s*(;)?)?", + "name": "meta.package.groovy" + }, + { + "begin": "(import static)\\b\\s*", + "beginCaptures": { + "1": { + "name": "keyword.other.import.static.groovy" + } + }, + "captures": { + "1": { + "name": "keyword.other.import.groovy" + }, + "2": { + "name": "storage.modifier.import.groovy" + }, + "3": { + "name": "punctuation.terminator.groovy" + } + }, + "contentName": "storage.modifier.import.groovy", + "end": "\\s*(?:$|(?=%>)(;))", + "endCaptures": { + "1": { + "name": "punctuation.terminator.groovy" + } + }, + "name": "meta.import.groovy", + "patterns": [ + { + "match": "\\.", + "name": "punctuation.separator.groovy" + }, + { + "match": "\\s", + "name": "invalid.illegal.character_not_allowed_here.groovy" + } + ] + }, + { + "begin": "(import)\\b\\s*", + "beginCaptures": { + "1": { + "name": "keyword.other.import.groovy" + } + }, + "captures": { + "1": { + "name": "keyword.other.import.groovy" + }, + "2": { + "name": "storage.modifier.import.groovy" + }, + "3": { + "name": "punctuation.terminator.groovy" + } + }, + "contentName": "storage.modifier.import.groovy", + "end": "\\s*(?:$|(?=%>)|(;))", + "endCaptures": { + "1": { + "name": "punctuation.terminator.groovy" + } + }, + "name": "meta.import.groovy", + "patterns": [ + { + "match": "\\.", + "name": "punctuation.separator.groovy" + }, + { + "match": "\\s", + "name": "invalid.illegal.character_not_allowed_here.groovy" + } + ] + }, + { + "captures": { + "1": { + "name": "keyword.other.import.groovy" + }, + "2": { + "name": "keyword.other.import.static.groovy" + }, + "3": { + "name": "storage.modifier.import.groovy" + }, + "4": { + "name": "punctuation.terminator.groovy" + } + }, + "match": "^\\s*(import)(?:\\s+(static)\\s+)\\b(?:\\s*([^ ;$]+)\\s*(;)?)?", + "name": "meta.import.groovy" + }, + { + "include": "#groovy" + } + ], + "repository": { + "annotations": { + "patterns": [ + { + "begin": "(?)", + "end": "\\}", + "patterns": [ + { + "begin": "(?<=\\{)(?=[^\\}]*?->)", + "end": "->", + "endCaptures": { + "0": { + "name": "keyword.operator.groovy" + } + }, + "patterns": [ + { + "begin": "(?!->)", + "end": "(?=->)", + "name": "meta.closure.parameters.groovy", + "patterns": [ + { + "begin": "(?!,|->)", + "end": "(?=,|->)", + "name": "meta.closure.parameter.groovy", + "patterns": [ + { + "begin": "=", + "beginCaptures": { + "0": { + "name": "keyword.operator.assignment.groovy" + } + }, + "end": "(?=,|->)", + "name": "meta.parameter.default.groovy", + "patterns": [ + { + "include": "#groovy-code" + } + ] + }, + { + "include": "#parameters" + } + ] + } + ] + } + ] + }, + { + "begin": "(?=[^}])", + "end": "(?=\\})", + "patterns": [ + { + "include": "#groovy-code" + } + ] + } + ] + }, + "comment-block": { + "begin": "/\\*", + "captures": { + "0": { + "name": "punctuation.definition.comment.groovy" + } + }, + "end": "\\*/", + "name": "comment.block.groovy" + }, + "comments": { + "patterns": [ + { + "captures": { + "0": { + "name": "punctuation.definition.comment.groovy" + } + }, + "match": "/\\*\\*/", + "name": "comment.block.empty.groovy" + }, + { + "include": "text.html.javadoc" + }, + { + "include": "#comment-block" + }, + { + "captures": { + "1": { + "name": "punctuation.definition.comment.groovy" + } + }, + "match": "(//).*$\\n?", + "name": "comment.line.double-slash.groovy" + } + ] + }, + "constants": { + "patterns": [ + { + "match": "\\b([A-Z][A-Z0-9_]+)\\b", + "name": "constant.other.groovy" + }, + { + "match": "\\b(true|false|null)\\b", + "name": "constant.language.groovy" + } + ] + }, + "constructors": { + "applyEndPatternLast": 1, + "begin": "(?<=;|^)(?=\\s*(?:(?:private|protected|public|native|synchronized|abstract|threadsafe|transient|static|final)\\s+)*[A-Z]\\w*\\()", + "end": "}", + "patterns": [ + { + "include": "#method-content" + } + ] + }, + "enum-values": { + "patterns": [ + { + "begin": "(?<=;|^)\\s*\\b([A-Z0-9_]+)(?=\\s*(?:,|;|}|\\(|$))", + "beginCaptures": { + "1": { + "name": "constant.enum.name.groovy" + } + }, + "end": ",|;|(?=})|^(?!\\s*\\w+\\s*(?:,|$))", + "patterns": [ + { + "begin": "\\(", + "end": "\\)", + "name": "meta.enum.value.groovy", + "patterns": [ + { + "match": ",", + "name": "punctuation.definition.seperator.parameter.groovy" + }, + { + "include": "#groovy-code" + } + ] + } + ] + } + ] + }, + "groovy": { + "patterns": [ + { + "include": "#comments" + }, + { + "include": "#class" + }, + { + "include": "#variables" + }, + { + "include": "#methods" + }, + { + "include": "#annotations" + }, + { + "include": "#groovy-code" + } + ] + }, + "groovy-code": { + "patterns": [ + { + "include": "#groovy-code-minus-map-keys" + }, + { + "include": "#map-keys" + } + ] + }, + "groovy-code-minus-map-keys": { + "comment": "In some situations, maps can't be declared without enclosing []'s,\n\t\t\t\ttherefore we create a collection of everything but that", + "patterns": [ + { + "include": "#comments" + }, + { + "include": "#annotations" + }, + { + "include": "#support-functions" + }, + { + "include": "#keyword-language" + }, + { + "include": "#values" + }, + { + "include": "#anonymous-classes-and-new" + }, + { + "include": "#keyword-operator" + }, + { + "include": "#types" + }, + { + "include": "#storage-modifiers" + }, + { + "include": "#parens" + }, + { + "include": "#closures" + }, + { + "include": "#braces" + } + ] + }, + "keyword": { + "patterns": [ + { + "include": "#keyword-operator" + }, + { + "include": "#keyword-language" + } + ] + }, + "keyword-language": { + "patterns": [ + { + "match": "\\b(try|catch|finally|throw)\\b", + "name": "keyword.control.exception.groovy" + }, + { + "match": "\\b((?", + "name": "keyword.operator.arrow.groovy" + }, + { + "match": "<<", + "name": "keyword.operator.leftshift.groovy" + }, + { + "match": "(?<=\\S)\\.(?=\\S)", + "name": "keyword.operator.navigation.groovy" + }, + { + "match": "(?<=\\S)\\?\\.(?=\\S)", + "name": "keyword.operator.safe-navigation.groovy" + }, + { + "begin": "\\?", + "beginCaptures": { + "0": { + "name": "keyword.operator.ternary.groovy" + } + }, + "end": "(?=$|\\)|}|])", + "name": "meta.evaluation.ternary.groovy", + "patterns": [ + { + "match": ":", + "name": "keyword.operator.ternary.expression-seperator.groovy" + }, + { + "include": "#groovy-code-minus-map-keys" + } + ] + }, + { + "match": "==~", + "name": "keyword.operator.match.groovy" + }, + { + "match": "=~", + "name": "keyword.operator.find.groovy" + }, + { + "match": "\\b(instanceof)\\b", + "name": "keyword.operator.instanceof.groovy" + }, + { + "match": "(===|==|!=|<=|>=|<=>|<>|<|>|<<)", + "name": "keyword.operator.comparison.groovy" + }, + { + "match": "=", + "name": "keyword.operator.assignment.groovy" + }, + { + "match": "(\\-\\-|\\+\\+)", + "name": "keyword.operator.increment-decrement.groovy" + }, + { + "match": "(\\-|\\+|\\*|\\/|%)", + "name": "keyword.operator.arithmetic.groovy" + }, + { + "match": "(!|&&|\\|\\|)", + "name": "keyword.operator.logical.groovy" + } + ] + }, + "language-variables": { + "patterns": [ + { + "match": "\\b(this|super)\\b", + "name": "variable.language.groovy" + } + ] + }, + "map-keys": { + "patterns": [ + { + "captures": { + "1": { + "name": "constant.other.key.groovy" + }, + "2": { + "name": "punctuation.definition.seperator.key-value.groovy" + } + }, + "match": "(\\w+)\\s*(:)" + } + ] + }, + "method-call": { + "begin": "([\\w$]+)(\\()", + "beginCaptures": { + "1": { + "name": "meta.method.groovy" + }, + "2": { + "name": "punctuation.definition.method-parameters.begin.groovy" + } + }, + "end": "\\)", + "endCaptures": { + "0": { + "name": "punctuation.definition.method-parameters.end.groovy" + } + }, + "name": "meta.method-call.groovy", + "patterns": [ + { + "match": ",", + "name": "punctuation.definition.seperator.parameter.groovy" + }, + { + "include": "#groovy-code" + } + ] + }, + "method-content": { + "patterns": [ + { + "match": "\\s" + }, + { + "include": "#annotations" + }, + { + "begin": "(?=(?:\\w|<)[^\\(]*\\s+(?:[\\w$]|<)+\\s*\\()", + "end": "(?=[\\w$]+\\s*\\()", + "name": "meta.method.return-type.java", + "patterns": [ + { + "include": "#storage-modifiers" + }, + { + "include": "#types" + } + ] + }, + { + "begin": "([\\w$]+)\\s*\\(", + "beginCaptures": { + "1": { + "name": "entity.name.function.java" + } + }, + "end": "\\)", + "name": "meta.definition.method.signature.java", + "patterns": [ + { + "begin": "(?=[^)])", + "end": "(?=\\))", + "name": "meta.method.parameters.groovy", + "patterns": [ + { + "begin": "(?=[^,)])", + "end": "(?=,|\\))", + "name": "meta.method.parameter.groovy", + "patterns": [ + { + "match": ",", + "name": "punctuation.definition.separator.groovy" + }, + { + "begin": "=", + "beginCaptures": { + "0": { + "name": "keyword.operator.assignment.groovy" + } + }, + "end": "(?=,|\\))", + "name": "meta.parameter.default.groovy", + "patterns": [ + { + "include": "#groovy-code" + } + ] + }, + { + "include": "#parameters" + } + ] + } + ] + } + ] + }, + { + "begin": "(?=<)", + "end": "(?=\\s)", + "name": "meta.method.paramerised-type.groovy", + "patterns": [ + { + "begin": "<", + "end": ">", + "name": "storage.type.parameters.groovy", + "patterns": [ + { + "include": "#types" + }, + { + "match": ",", + "name": "punctuation.definition.seperator.groovy" + } + ] + } + ] + }, + { + "begin": "throws", + "beginCaptures": { + "0": { + "name": "storage.modifier.groovy" + } + }, + "end": "(?={|;)|^(?=\\s*(?:[^{\\s]|$))", + "name": "meta.throwables.groovy", + "patterns": [ + { + "include": "#object-types" + } + ] + }, + { + "begin": "{", + "end": "(?=})", + "name": "meta.method.body.java", + "patterns": [ + { + "include": "#groovy-code" + } + ] + } + ] + }, + "methods": { + "applyEndPatternLast": 1, + "begin": "(?x:(?<=;|^|{)(?=\\s*\n (?:\n (?:private|protected|public|native|synchronized|abstract|threadsafe|transient|static|final) # visibility/modifier\n |\n (?:def)\n |\n (?:\n (?:\n (?:void|boolean|byte|char|short|int|float|long|double)\n |\n (?:@?(?:[a-zA-Z]\\w*\\.)*[A-Z]+\\w*) # object type\n )\n [\\[\\]]*\n (?:<.*>)?\n )\n\n )\n \\s+\n ([^=]+\\s+)?\\w+\\s*\\(\n\t\t\t))", + "end": "}|(?=[^{])", + "name": "meta.definition.method.groovy", + "patterns": [ + { + "include": "#method-content" + } + ] + }, + "nest_curly": { + "begin": "\\{", + "captures": { + "0": { + "name": "punctuation.section.scope.groovy" + } + }, + "end": "\\}", + "patterns": [ + { + "include": "#nest_curly" + } + ] + }, + "numbers": { + "patterns": [ + { + "match": "((0(x|X)[0-9a-fA-F]*)|(\\+|-)?\\b(([0-9]+\\.?[0-9]*)|(\\.[0-9]+))((e|E)(\\+|-)?[0-9]+)?)([LlFfUuDdg]|UL|ul)?\\b", + "name": "constant.numeric.groovy" + } + ] + }, + "object-types": { + "patterns": [ + { + "begin": "\\b((?:[a-z]\\w*\\.)*(?:[A-Z]+\\w*[a-z]+\\w*|UR[LI]))<", + "end": ">|[^\\w\\s,\\?<\\[\\]]", + "name": "storage.type.generic.groovy", + "patterns": [ + { + "include": "#object-types" + }, + { + "begin": "<", + "comment": "This is just to support <>'s with no actual type prefix", + "end": ">|[^\\w\\s,\\[\\]<]", + "name": "storage.type.generic.groovy" + } + ] + }, + { + "begin": "\\b((?:[a-z]\\w*\\.)*[A-Z]+\\w*[a-z]+\\w*)(?=\\[)", + "end": "(?=[^\\]\\s])", + "name": "storage.type.object.array.groovy", + "patterns": [ + { + "begin": "\\[", + "end": "\\]", + "patterns": [ + { + "include": "#groovy" + } + ] + } + ] + }, + { + "match": "\\b(?:[a-zA-Z]\\w*\\.)*(?:[A-Z]+\\w*[a-z]+\\w*|UR[LI])\\b", + "name": "storage.type.groovy" + } + ] + }, + "object-types-inherited": { + "patterns": [ + { + "begin": "\\b((?:[a-zA-Z]\\w*\\.)*[A-Z]+\\w*[a-z]+\\w*)<", + "end": ">|[^\\w\\s,\\?<\\[\\]]", + "name": "entity.other.inherited-class.groovy", + "patterns": [ + { + "include": "#object-types-inherited" + }, + { + "begin": "<", + "comment": "This is just to support <>'s with no actual type prefix", + "end": ">|[^\\w\\s,\\[\\]<]", + "name": "storage.type.generic.groovy" + } + ] + }, + { + "captures": { + "1": { + "name": "keyword.operator.dereference.groovy" + } + }, + "match": "\\b(?:[a-zA-Z]\\w*(\\.))*[A-Z]+\\w*[a-z]+\\w*\\b", + "name": "entity.other.inherited-class.groovy" + } + ] + }, + "parameters": { + "patterns": [ + { + "include": "#annotations" + }, + { + "include": "#storage-modifiers" + }, + { + "include": "#types" + }, + { + "match": "\\w+", + "name": "variable.parameter.method.groovy" + } + ] + }, + "parens": { + "begin": "\\(", + "end": "\\)", + "patterns": [ + { + "include": "#groovy-code" + } + ] + }, + "primitive-arrays": { + "patterns": [ + { + "match": "\\b(?:void|boolean|byte|char|short|int|float|long|double)(\\[\\])*\\b", + "name": "storage.type.primitive.array.groovy" + } + ] + }, + "primitive-types": { + "patterns": [ + { + "match": "\\b(?:void|boolean|byte|char|short|int|float|long|double)\\b", + "name": "storage.type.primitive.groovy" + } + ] + }, + "regexp": { + "patterns": [ + { + "begin": "/(?=[^/]+/([^>]|$))", + "beginCaptures": { + "0": { + "name": "punctuation.definition.string.regexp.begin.groovy" + } + }, + "end": "/", + "endCaptures": { + "0": { + "name": "punctuation.definition.string.regexp.end.groovy" + } + }, + "name": "string.regexp.groovy", + "patterns": [ + { + "match": "\\\\.", + "name": "constant.character.escape.groovy" + } + ] + }, + { + "begin": "~\"", + "beginCaptures": { + "0": { + "name": "punctuation.definition.string.regexp.begin.groovy" + } + }, + "end": "\"", + "endCaptures": { + "0": { + "name": "punctuation.definition.string.regexp.end.groovy" + } + }, + "name": "string.regexp.compiled.groovy", + "patterns": [ + { + "match": "\\\\.", + "name": "constant.character.escape.groovy" + } + ] + } + ] + }, + "storage-modifiers": { + "patterns": [ + { + "match": "\\b(private|protected|public)\\b", + "name": "storage.modifier.access-control.groovy" + }, + { + "match": "\\b(static)\\b", + "name": "storage.modifier.static.groovy" + }, + { + "match": "\\b(final)\\b", + "name": "storage.modifier.final.groovy" + }, + { + "match": "\\b(native|synchronized|abstract|threadsafe|transient)\\b", + "name": "storage.modifier.other.groovy" + } + ] + }, + "string-quoted-double": { + "begin": "\"", + "beginCaptures": { + "0": { + "name": "punctuation.definition.string.begin.groovy" + } + }, + "end": "\"", + "endCaptures": { + "0": { + "name": "punctuation.definition.string.end.groovy" + } + }, + "name": "string.quoted.double.groovy", + "patterns": [ + { + "include": "#string-quoted-double-contents" + } + ] + }, + "string-quoted-double-contents": { + "patterns": [ + { + "match": "\\\\.", + "name": "constant.character.escape.groovy" + }, + { + "applyEndPatternLast": 1, + "begin": "\\$\\w", + "end": "(?=\\W)", + "name": "variable.other.interpolated.groovy", + "patterns": [ + { + "match": "\\w", + "name": "variable.other.interpolated.groovy" + }, + { + "match": "\\.", + "name": "keyword.other.dereference.groovy" + } + ] + }, + { + "begin": "\\$\\{", + "captures": { + "0": { + "name": "punctuation.section.embedded.groovy" + } + }, + "end": "\\}", + "name": "source.groovy.embedded.source", + "patterns": [ + { + "include": "#nest_curly" + } + ] + } + ] + }, + "string-quoted-double-multiline": { + "begin": "\"\"\"", + "beginCaptures": { + "0": { + "name": "punctuation.definition.string.begin.groovy" + } + }, + "end": "\"\"\"", + "endCaptures": { + "0": { + "name": "punctuation.definition.string.end.groovy" + } + }, + "name": "string.quoted.double.multiline.groovy", + "patterns": [ + { + "include": "#string-quoted-double-contents" + } + ] + }, + "string-quoted-single": { + "begin": "'", + "beginCaptures": { + "0": { + "name": "punctuation.definition.string.begin.groovy" + } + }, + "end": "'", + "endCaptures": { + "0": { + "name": "punctuation.definition.string.end.groovy" + } + }, + "name": "string.quoted.single.groovy", + "patterns": [ + { + "include": "#string-quoted-single-contents" + } + ] + }, + "string-quoted-single-contents": { + "patterns": [ + { + "match": "\\\\.", + "name": "constant.character.escape.groovy" + } + ] + }, + "string-quoted-single-multiline": { + "begin": "'''", + "beginCaptures": { + "0": { + "name": "punctuation.definition.string.begin.groovy" + } + }, + "end": "'''", + "endCaptures": { + "0": { + "name": "punctuation.definition.string.end.groovy" + } + }, + "name": "string.quoted.single.multiline.groovy", + "patterns": [ + { + "include": "#string-quoted-single-contents" + } + ] + }, + "strings": { + "patterns": [ + { + "include": "#string-quoted-double-multiline" + }, + { + "include": "#string-quoted-single-multiline" + }, + { + "include": "#string-quoted-double" + }, + { + "include": "#string-quoted-single" + }, + { + "include": "#regexp" + } + ] + }, + "structures": { + "begin": "\\[", + "beginCaptures": { + "0": { + "name": "punctuation.definition.structure.begin.groovy" + } + }, + "end": "\\]", + "endCaptures": { + "0": { + "name": "punctuation.definition.structure.end.groovy" + } + }, + "name": "meta.structure.groovy", + "patterns": [ + { + "include": "#groovy-code" + }, + { + "match": ",", + "name": "punctuation.definition.separator.groovy" + } + ] + }, + "support-functions": { + "patterns": [ + { + "match": "(?x)\\b(?:sprintf|print(?:f|ln)?)\\b", + "name": "support.function.print.groovy" + }, + { + "match": "(?x)\\b(?:shouldFail|fail(?:NotEquals)?|ass(?:ume|ert(?:S(?:cript|ame)|N(?:ot(?:Same|\n\t\t\t\t\tNull)|ull)|Contains|T(?:hat|oString|rue)|Inspect|Equals|False|Length|\n\t\t\t\t\tArrayEquals)))\\b", + "name": "support.function.testing.groovy" + } + ] + }, + "types": { + "patterns": [ + { + "match": "\\b(def)\\b", + "name": "storage.type.def.groovy" + }, + { + "include": "#primitive-types" + }, + { + "include": "#primitive-arrays" + }, + { + "include": "#object-types" + } + ] + }, + "values": { + "patterns": [ + { + "include": "#language-variables" + }, + { + "include": "#strings" + }, + { + "include": "#numbers" + }, + { + "include": "#constants" + }, + { + "include": "#types" + }, + { + "include": "#structures" + }, + { + "include": "#method-call" + } + ] + }, + "variables": { + "applyEndPatternLast": 1, + "patterns": [ + { + "begin": "(?x:(?=\n (?:\n (?:private|protected|public|native|synchronized|abstract|threadsafe|transient|static|final) # visibility/modifier\n |\n (?:def)\n |\n (?:void|boolean|byte|char|short|int|float|long|double)\n |\n (?:(?:[a-z]\\w*\\.)*[A-Z]+\\w*) # object type\n )\n \\s+\n [\\w\\d_<>\\[\\],\\s]+\n (?:=|$)\n\n \t\t\t))", + "end": ";|$", + "name": "meta.definition.variable.groovy", + "patterns": [ + { + "match": "\\s" + }, + { + "captures": { + "1": { + "name": "constant.variable.groovy" + } + }, + "match": "([A-Z_0-9]+)\\s+(?=\\=)" + }, + { + "captures": { + "1": { + "name": "meta.definition.variable.name.groovy" + } + }, + "match": "(\\w[^\\s,]*)\\s+(?=\\=)" + }, + { + "begin": "=", + "beginCaptures": { + "0": { + "name": "keyword.operator.assignment.groovy" + } + }, + "end": "$", + "patterns": [ + { + "include": "#groovy-code" + } + ] + }, + { + "captures": { + "1": { + "name": "meta.definition.variable.name.groovy" + } + }, + "match": "(\\w[^\\s=]*)(?=\\s*($|;))" + }, + { + "include": "#groovy-code" + } + ] + } + ] + } + } +} diff --git a/extension/thirdpartynotice.txt b/extension/thirdpartynotice.txt new file mode 100644 index 000000000..8b74a77e2 --- /dev/null +++ b/extension/thirdpartynotice.txt @@ -0,0 +1,149 @@ +THIRD-PARTY SOFTWARE NOTICES AND INFORMATION +For vscode-gradle package + +This extension uses Open Source components. You can find the source code of their +open source projects along with the license information below. We acknowledge and +are grateful to these developers for their contribution to open source. + +1. fs-extra (https://github.com/jprichardson/node-fs-extra) +2. get-port (https://github.com/sindresorhus/get-port) +3. minimatch (https://github.com/isaacs/minimatch) +4. string-argv (https://github.com/mccormicka/string-argv) +5. tree-kill (https://github.com/pkrumins/node-tree-kill) +6. vscode-extension-telemetry-wrapper (https://github.com/Eskibear/vscode-extension-telemetry-wrapper) + +fs-extra NOTICES BEGIN HERE +============================= + +(The MIT License) + +Copyright (c) 2011-2017 JP Richardson + +Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files +(the 'Software'), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, + merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE +WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS +OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, + ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + +END OF fs-extra NOTICES AND INFORMATION +================================== + +get-port NOTICES BEGIN HERE +============================= + +MIT License + +Copyright (c) Sindre Sorhus (sindresorhus.com) + +Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + +END OF get-port NOTICES AND INFORMATION +================================== + +minimatch NOTICES BEGIN HERE +============================= + +The ISC License + +Copyright (c) Isaac Z. Schlueter and Contributors + +Permission to use, copy, modify, and/or distribute this software for any +purpose with or without fee is hereby granted, provided that the above +copyright notice and this permission notice appear in all copies. + +THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES +WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF +MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR +ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES +WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN +ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR +IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + +END OF minimatch NOTICES AND INFORMATION +========================================= + +string-argv NOTICES BEGIN HERE +============================= + +The MIT License (MIT) + +Copyright 2014 Anthony McCormick + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. + +END OF string-argv NOTICES AND INFORMATION +========================================= + +node-tree-kill NOTICES BEGIN HERE +============================= + +MIT License + +Copyright (c) 2018 Peter Krumins + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. + +END OF node-tree-kill NOTICES AND INFORMATION +========================================= + +vscode-extension-telemetry-wrapper NOTICES BEGIN HERE +========================================================== + +Copyright 2018 Yan Zhang + +Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files +(the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, + merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE +WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS +OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, +ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + +END OF vscode-extension-telemetry-wrapper NOTICES AND INFORMATION +========================================================== diff --git a/gradle-language-server/build.gradle b/gradle-language-server/build.gradle new file mode 100644 index 000000000..1f5829cb2 --- /dev/null +++ b/gradle-language-server/build.gradle @@ -0,0 +1,40 @@ +plugins { + id "java" +} + +java { + sourceCompatibility = JavaVersion.VERSION_1_8 + targetCompatibility = JavaVersion.VERSION_1_8 +} + +repositories { + mavenCentral() +} + +dependencies { + implementation "org.eclipse.lsp4j:org.eclipse.lsp4j:0.12.0" + implementation "org.eclipse.lsp4j:org.eclipse.lsp4j.jsonrpc:0.12.0" + implementation "org.codehaus.groovy:groovy-eclipse-batch:3.0.8-01" + implementation "com.google.code.gson:gson:2.8.7" + implementation "org.apache.bcel:bcel:6.5.0" +} + +ext.mainClass = "com.microsoft.gradle.GradleLanguageServer" + +task copyJar(type: Copy) { + from "build/libs/gradle-language-server.jar" + from configurations.runtimeClasspath + into "../extension/lib" +} + +tasks.build.dependsOn tasks.copyJar +tasks.copyJar.dependsOn jar + +jar { + manifest { + attributes( + "Main-Class": "com.microsoft.gradle.GradleLanguageServer", + "Class-Path": configurations.runtimeClasspath.collect { it.getName() }.join(' ') + ) + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/GradleLanguageServer.java b/gradle-language-server/src/main/java/com/microsoft/gradle/GradleLanguageServer.java new file mode 100644 index 000000000..90b135f25 --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/GradleLanguageServer.java @@ -0,0 +1,137 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ + +package com.microsoft.gradle; + +import java.io.IOException; +import java.net.Socket; +import java.net.URI; +import java.nio.file.Paths; +import java.util.Arrays; +import java.util.List; +import java.util.Map; +import java.util.concurrent.CompletableFuture; +import java.util.stream.Collectors; + +import com.google.gson.Gson; +import com.google.gson.JsonElement; +import com.microsoft.gradle.semantictokens.TokenModifier; +import com.microsoft.gradle.semantictokens.TokenType; + +import org.eclipse.lsp4j.CompletionOptions; +import org.eclipse.lsp4j.DocumentFilter; +import org.eclipse.lsp4j.ExecuteCommandOptions; +import org.eclipse.lsp4j.InitializeParams; +import org.eclipse.lsp4j.InitializeResult; +import org.eclipse.lsp4j.SaveOptions; +import org.eclipse.lsp4j.SemanticTokensLegend; +import org.eclipse.lsp4j.SemanticTokensServerFull; +import org.eclipse.lsp4j.SemanticTokensWithRegistrationOptions; +import org.eclipse.lsp4j.ServerCapabilities; +import org.eclipse.lsp4j.TextDocumentSyncKind; +import org.eclipse.lsp4j.TextDocumentSyncOptions; +import org.eclipse.lsp4j.WorkspaceFolder; +import org.eclipse.lsp4j.jsonrpc.Launcher; +import org.eclipse.lsp4j.services.LanguageClient; +import org.eclipse.lsp4j.services.LanguageClientAware; +import org.eclipse.lsp4j.services.LanguageServer; +import org.eclipse.lsp4j.services.TextDocumentService; +import org.eclipse.lsp4j.services.WorkspaceService; + +public class GradleLanguageServer implements LanguageServer, LanguageClientAware { + + private GradleServices gradleServices; + + public static void main(String[] args) { + GradleLanguageServer server = new GradleLanguageServer(); + try { + Launcher launcher; + String port = System.getenv("VSCODE_GRADLE_PORT"); + if (port == null) { + // Launch Mode + launcher = Launcher.createLauncher(server, LanguageClient.class, System.in, System.out); + } else { + // Debug Mode + Socket socket = new Socket("localhost", Integer.parseInt(port)); + launcher = Launcher.createLauncher(server, LanguageClient.class, socket.getInputStream(), + socket.getOutputStream()); + } + server.connect(launcher.getRemoteProxy()); + launcher.startListening(); + } catch (IOException e) { + server.exit(); + } + + } + + public GradleLanguageServer() { + this.gradleServices = new GradleServices(); + } + + @Override + public CompletableFuture initialize(InitializeParams params) { + Map initOptions = new Gson().fromJson((JsonElement) params.getInitializationOptions(), Map.class); + // TODO: support multiple workspace folders + List workspaceFolders = params.getWorkspaceFolders(); + for (WorkspaceFolder folder : workspaceFolders) { + URI uri = URI.create(folder.getUri()); + this.gradleServices.getLibraryResolver().setWorkspacePath(Paths.get(uri)); + break; + } + Object settings = initOptions.get("settings"); + this.gradleServices.applySetting(settings); + ServerCapabilities serverCapabilities = new ServerCapabilities(); + SemanticTokensWithRegistrationOptions semanticOptions = new SemanticTokensWithRegistrationOptions(); + semanticOptions.setFull(new SemanticTokensServerFull(false)); + semanticOptions.setRange(false); + semanticOptions.setDocumentSelector(List.of(new DocumentFilter("gradle", "file", null))); + semanticOptions.setLegend(new SemanticTokensLegend( + Arrays.stream(TokenType.values()).map(TokenType::toString).collect(Collectors.toList()), + Arrays.stream(TokenModifier.values()).map(TokenModifier::toString).collect(Collectors.toList()))); + serverCapabilities.setSemanticTokensProvider(semanticOptions); + serverCapabilities.setDocumentSymbolProvider(true); + TextDocumentSyncOptions textDocumentSyncOptions = new TextDocumentSyncOptions(); + textDocumentSyncOptions.setOpenClose(Boolean.TRUE); + textDocumentSyncOptions.setSave(new SaveOptions(Boolean.TRUE)); + textDocumentSyncOptions.setChange(TextDocumentSyncKind.Incremental); + serverCapabilities.setTextDocumentSync(textDocumentSyncOptions); + CompletionOptions completionOptions = new CompletionOptions(false, Arrays.asList(".", ":")); + serverCapabilities.setCompletionProvider(completionOptions); + serverCapabilities.setExecuteCommandProvider(new ExecuteCommandOptions(GradleServices.supportedCommands)); + InitializeResult initializeResult = new InitializeResult(serverCapabilities); + return CompletableFuture.completedFuture(initializeResult); + } + + @Override + public CompletableFuture shutdown() { + return CompletableFuture.completedFuture(new Object()); + } + + @Override + public void exit() { + System.exit(0); + } + + @Override + public TextDocumentService getTextDocumentService() { + return this.gradleServices; + } + + @Override + public WorkspaceService getWorkspaceService() { + return this.gradleServices; + } + + @Override + public void connect(LanguageClient client) { + this.gradleServices.connect(client); + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/GradleServices.java b/gradle-language-server/src/main/java/com/microsoft/gradle/GradleServices.java new file mode 100644 index 000000000..53b1e8e89 --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/GradleServices.java @@ -0,0 +1,335 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ + +package com.microsoft.gradle; + +import java.io.IOException; +import java.net.URI; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.HashSet; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.concurrent.CompletableFuture; + +import com.google.common.base.Charsets; +import com.google.common.io.Files; +import com.google.gson.Gson; +import com.google.gson.JsonElement; +import com.microsoft.gradle.compile.CompletionVisitor; +import com.microsoft.gradle.compile.CompletionVisitor.DependencyItem; +import com.microsoft.gradle.compile.DocumentSymbolVisitor; +import com.microsoft.gradle.compile.GradleCompilationUnit; +import com.microsoft.gradle.compile.SemanticTokenVisitor; +import com.microsoft.gradle.handlers.CompletionHandler; +import com.microsoft.gradle.handlers.DefaultDependenciesHandler; +import com.microsoft.gradle.handlers.DefaultDependenciesHandler.DefaultDependencyItem; +import com.microsoft.gradle.handlers.DependencyCompletionHandler; +import com.microsoft.gradle.manager.GradleFilesManager; +import com.microsoft.gradle.resolver.GradleLibraryResolver; +import com.microsoft.gradle.semantictokens.SemanticToken; +import com.microsoft.gradle.utils.LSPUtils; + +import org.codehaus.groovy.ast.expr.Expression; +import org.codehaus.groovy.ast.expr.MethodCallExpression; +import org.codehaus.groovy.ast.stmt.Statement; +import org.codehaus.groovy.control.CompilationFailedException; +import org.codehaus.groovy.control.ErrorCollector; +import org.codehaus.groovy.control.Phases; +import org.codehaus.groovy.control.messages.Message; +import org.codehaus.groovy.control.messages.SyntaxErrorMessage; +import org.codehaus.groovy.syntax.SyntaxException; +import org.eclipse.lsp4j.CompletionItem; +import org.eclipse.lsp4j.CompletionList; +import org.eclipse.lsp4j.CompletionParams; +import org.eclipse.lsp4j.Diagnostic; +import org.eclipse.lsp4j.DiagnosticSeverity; +import org.eclipse.lsp4j.DidChangeConfigurationParams; +import org.eclipse.lsp4j.DidChangeTextDocumentParams; +import org.eclipse.lsp4j.DidChangeWatchedFilesParams; +import org.eclipse.lsp4j.DidCloseTextDocumentParams; +import org.eclipse.lsp4j.DidOpenTextDocumentParams; +import org.eclipse.lsp4j.DidSaveTextDocumentParams; +import org.eclipse.lsp4j.DocumentSymbol; +import org.eclipse.lsp4j.DocumentSymbolParams; +import org.eclipse.lsp4j.ExecuteCommandParams; +import org.eclipse.lsp4j.Position; +import org.eclipse.lsp4j.PublishDiagnosticsParams; +import org.eclipse.lsp4j.Range; +import org.eclipse.lsp4j.SemanticTokens; +import org.eclipse.lsp4j.SemanticTokensParams; +import org.eclipse.lsp4j.SymbolInformation; +import org.eclipse.lsp4j.TextDocumentContentChangeEvent; +import org.eclipse.lsp4j.TextDocumentIdentifier; +import org.eclipse.lsp4j.TextDocumentItem; +import org.eclipse.lsp4j.jsonrpc.messages.Either; +import org.eclipse.lsp4j.services.LanguageClient; +import org.eclipse.lsp4j.services.LanguageClientAware; +import org.eclipse.lsp4j.services.TextDocumentService; +import org.eclipse.lsp4j.services.WorkspaceService; +import org.eclipse.lsp4j.util.Ranges; + +public class GradleServices implements TextDocumentService, WorkspaceService, LanguageClientAware { + + public static final List supportedCommands = List.of("gradle.getDependencies"); + + private LanguageClient client; + private GradleFilesManager gradleFilesManager; + private SemanticTokenVisitor semanticTokenVisitor; + private DocumentSymbolVisitor documentSymbolVisitor; + private CompletionVisitor completionVisitor; + private GradleLibraryResolver libraryResolver; + private DefaultDependenciesHandler defaultDependenciesHandler; + + public GradleServices() { + this.gradleFilesManager = new GradleFilesManager(); + this.semanticTokenVisitor = new SemanticTokenVisitor(); + this.documentSymbolVisitor = new DocumentSymbolVisitor(); + this.completionVisitor = new CompletionVisitor(); + this.libraryResolver = new GradleLibraryResolver(); + this.defaultDependenciesHandler = new DefaultDependenciesHandler(); + } + + public GradleLibraryResolver getLibraryResolver() { + return this.libraryResolver; + } + + @Override + public void connect(LanguageClient client) { + this.client = client; + } + + @Override + public void didOpen(DidOpenTextDocumentParams params) { + URI uri = URI.create(params.getTextDocument().getUri()); + gradleFilesManager.didOpen(uri, params.getTextDocument().getText()); + GradleCompilationUnit unit = this.gradleFilesManager.getCompilationUnit(uri, params.getTextDocument().getVersion()); + compile(uri, unit); + } + + @Override + public void didChange(DidChangeTextDocumentParams params) { + URI uri = URI.create(params.getTextDocument().getUri()); + for (TextDocumentContentChangeEvent change : params.getContentChanges()) { + gradleFilesManager.didChange(uri, change); + } + GradleCompilationUnit unit = this.gradleFilesManager.getCompilationUnit(uri, params.getTextDocument().getVersion()); + compile(uri, unit); + } + + @Override + public void didClose(DidCloseTextDocumentParams params) { + URI uri = URI.create(params.getTextDocument().getUri()); + gradleFilesManager.didClose(uri); + } + + @Override + public void didSave(DidSaveTextDocumentParams params) { + // TODO + } + + @Override + public void didChangeWatchedFiles(DidChangeWatchedFilesParams params) { + // TODO + } + + @Override + public void didChangeConfiguration(DidChangeConfigurationParams params) { + Map settings = new Gson().fromJson((JsonElement) params.getSettings(), Map.class); + this.applySetting(settings); + } + + public void applySetting(Object settings) { + if (settings instanceof Map) { + this.getLibraryResolver().setGradleHome((String) ((Map) settings).get("gradleHome")); + this.getLibraryResolver().setGradleVersion((String) ((Map) settings).get("gradleVersion")); + this.getLibraryResolver().setGradleWrapperEnabled((Boolean) ((Map) settings).get("gradleWrapperEnabled")); + this.getLibraryResolver().setGradleUserHome((String) ((Map) settings).get("gradleUserHome")); + this.getLibraryResolver().resolve(); + } + } + + private void compile(URI uri, GradleCompilationUnit unit) { + if (unit == null) { + return; + } + Set diagnostics = new HashSet<>(); + try { + unit.compile(Phases.CANONICALIZATION); + // Send empty diagnostic if there is no error + diagnostics.add(new PublishDiagnosticsParams(uri.toString(), Collections.emptyList())); + } catch (CompilationFailedException e) { + diagnostics = generateDiagnostics(unit.getErrorCollector()); + } + for (PublishDiagnosticsParams diagnostic : diagnostics) { + client.publishDiagnostics(diagnostic); + } + } + + private Set generateDiagnostics(ErrorCollector collector) { + // URI, List + Map> diagnosticsStorage = new HashMap<>(); + for (Message error : collector.getErrors()) { + if (error instanceof SyntaxErrorMessage) { + SyntaxException exp = ((SyntaxErrorMessage) error).getCause(); + Range range = LSPUtils.toRange(exp); + Diagnostic diagnostic = new Diagnostic(); + diagnostic.setRange(range); + diagnostic.setSeverity(DiagnosticSeverity.Error); + diagnostic.setMessage(exp.getMessage()); + diagnostic.setSource("Gradle"); + if (diagnosticsStorage.containsKey(exp.getSourceLocator())) { + diagnosticsStorage.get(exp.getSourceLocator()).add(diagnostic); + } else { + List diagnostics = new ArrayList<>(); + diagnostics.add(diagnostic); + diagnosticsStorage.put(exp.getSourceLocator(), diagnostics); + } + } + } + Set diagnosticsParams = new HashSet<>(); + for (Map.Entry> entry : diagnosticsStorage.entrySet()) { + diagnosticsParams.add(new PublishDiagnosticsParams(entry.getKey(), entry.getValue())); + } + return diagnosticsParams; + } + + @Override + public CompletableFuture semanticTokensFull(SemanticTokensParams params) { + URI uri = URI.create(params.getTextDocument().getUri()); + GradleCompilationUnit unit = this.gradleFilesManager.getCompilationUnit(uri); + if (unit == null) { + return CompletableFuture.completedFuture(new SemanticTokens(Collections.emptyList())); + } + this.semanticTokenVisitor.visitCompilationUnit(uri, unit); + List semanticTokens = this.semanticTokenVisitor.getSemanticTokens(uri); + if (semanticTokens == null) { + return CompletableFuture.completedFuture(new SemanticTokens(Collections.emptyList())); + } + return CompletableFuture.completedFuture(new SemanticTokens(SemanticToken.encodedTokens(semanticTokens))); + } + + @Override + public CompletableFuture>> documentSymbol( + DocumentSymbolParams params) { + URI uri = URI.create(params.getTextDocument().getUri()); + GradleCompilationUnit unit = this.gradleFilesManager.getCompilationUnit(uri); + if (unit == null) { + return CompletableFuture.completedFuture(Collections.emptyList()); + } + this.documentSymbolVisitor.visitCompilationUnit(uri, unit); + List documentSymbols = this.documentSymbolVisitor.getDocumentSymbols(uri); + if (documentSymbols == null) { + return CompletableFuture.completedFuture(Collections.emptyList()); + } + List> result = new ArrayList<>(); + for (DocumentSymbol symbol : documentSymbols) { + result.add(Either.forRight(symbol)); + } + return CompletableFuture.completedFuture(result); + } + + @Override + public CompletableFuture, CompletionList>> completion(CompletionParams params) { + URI uri = URI.create(params.getTextDocument().getUri()); + GradleCompilationUnit unit = this.gradleFilesManager.getCompilationUnit(uri); + if (unit == null) { + return CompletableFuture.completedFuture(Either.forLeft(Collections.emptyList())); + } + this.completionVisitor.visitCompilationUnit(uri, unit); + List dependencies = this.completionVisitor.getDependencies(uri); + if (dependencies == null) { + return CompletableFuture.completedFuture(Either.forLeft(Collections.emptyList())); + } + for (DependencyItem dependency : dependencies) { + if (Ranges.containsPosition(dependency.getRange(), params.getPosition())) { + DependencyCompletionHandler handler = new DependencyCompletionHandler(); + return CompletableFuture + .completedFuture(Either.forLeft(handler.getDependencyCompletionItems(dependency, params.getPosition()))); + } + } + // should return empty if in constants + List constants = this.completionVisitor.getConstants(uri); + for (Expression constant : constants) { + Range range = LSPUtils.toRange(constant); + if (Ranges.containsPosition(range, params.getPosition())) { + return CompletableFuture.completedFuture(Either.forLeft(Collections.emptyList())); + } + } + Set methodCalls = this.completionVisitor.getMethodCalls(uri); + MethodCallExpression containingCall = null; + for (MethodCallExpression call : methodCalls) { + Expression expression = call.getArguments(); + Range range = LSPUtils.toRange(expression); + if (Ranges.containsPosition(range, params.getPosition()) && (containingCall == null || Ranges + .containsRange(LSPUtils.toRange(containingCall.getArguments()), LSPUtils.toRange(call.getArguments())))) { + // find inner containing call + containingCall = call; + } + } + boolean javaPluginsIncluded = this.libraryResolver.isJavaPluginsIncluded(this.completionVisitor.getPlugins(uri)); + CompletionHandler handler = new CompletionHandler(); + // check again + if (containingCall == null && isGradleRoot(uri, params.getPosition())) { + return CompletableFuture.completedFuture(Either + .forLeft(handler.getCompletionItems(null, Paths.get(uri).getFileName().toString(), this.libraryResolver, javaPluginsIncluded))); + } + return CompletableFuture.completedFuture(Either.forLeft( + handler.getCompletionItems(containingCall, Paths.get(uri).getFileName().toString(), this.libraryResolver, javaPluginsIncluded))); + } + + @Override + public CompletableFuture executeCommand(ExecuteCommandParams params) { + String command = params.getCommand(); + if (command.equals("gradle.getDependencies")) { + List arguments = params.getArguments(); + if (arguments.isEmpty()) { + return CompletableFuture.completedFuture(null); + } + String uriString = new Gson().fromJson((JsonElement) arguments.get(0), String.class); + URI uri = URI.create(uriString); + if (this.gradleFilesManager.getCompilationUnit(uri) == null) { + try { + Path uriPath = Paths.get(uri); + String content = Files.asCharSource(uriPath.toFile(), Charsets.UTF_8).read(); + DidOpenTextDocumentParams openDocumentParams = new DidOpenTextDocumentParams(new TextDocumentItem(uriString, "gradle", 1, content)); + this.didOpen(openDocumentParams); + DocumentSymbolParams documentSymbolParams = new DocumentSymbolParams(new TextDocumentIdentifier(uriString)); + this.documentSymbol(documentSymbolParams); + } catch (IOException e) { + return CompletableFuture.completedFuture(null); + } + } + List dependencies = this.documentSymbolVisitor.getDependencies(uri); + if (dependencies == null) { + return CompletableFuture.completedFuture(null); + } + List result = defaultDependenciesHandler.getDefaultDependencies(dependencies); + return CompletableFuture.completedFuture(result); + } + return CompletableFuture.completedFuture(null); + } + + private boolean isGradleRoot(URI uri, Position position) { + List statements = this.completionVisitor.getStatements(uri); + for (Statement statement : statements) { + Range range = LSPUtils.toRange(statement); + if (Ranges.containsPosition(range, position)) { + return false; + } + } + return true; + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/compile/CompletionVisitor.java b/gradle-language-server/src/main/java/com/microsoft/gradle/compile/CompletionVisitor.java new file mode 100644 index 000000000..2fce6c1ff --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/compile/CompletionVisitor.java @@ -0,0 +1,251 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ +package com.microsoft.gradle.compile; + +import java.net.URI; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.HashSet; +import java.util.List; +import java.util.Map; +import java.util.Set; + +import com.microsoft.gradle.utils.LSPUtils; + +import org.codehaus.groovy.ast.ClassCodeVisitorSupport; +import org.codehaus.groovy.ast.ModuleNode; +import org.codehaus.groovy.ast.expr.ArgumentListExpression; +import org.codehaus.groovy.ast.expr.ClosureExpression; +import org.codehaus.groovy.ast.expr.ConstantExpression; +import org.codehaus.groovy.ast.expr.Expression; +import org.codehaus.groovy.ast.expr.GStringExpression; +import org.codehaus.groovy.ast.expr.MapEntryExpression; +import org.codehaus.groovy.ast.expr.MethodCallExpression; +import org.codehaus.groovy.ast.expr.NamedArgumentListExpression; +import org.codehaus.groovy.ast.expr.TupleExpression; +import org.codehaus.groovy.ast.stmt.BlockStatement; +import org.codehaus.groovy.ast.stmt.ExpressionStatement; +import org.codehaus.groovy.ast.stmt.Statement; +import org.codehaus.groovy.control.SourceUnit; +import org.eclipse.lsp4j.Range; + +public class CompletionVisitor extends ClassCodeVisitorSupport { + + public class DependencyItem { + private String text; + private Range range; + + public DependencyItem(String text, Range range) { + this.text = text; + this.range = range; + } + + public String getText() { + return this.text; + } + + public Range getRange() { + return this.range; + } + } + + private URI currentUri; + private Map> dependencies = new HashMap<>(); + private Map> methodCalls = new HashMap<>(); + private Map> statements = new HashMap<>(); + private Map> constants = new HashMap<>(); + private Map> plugins = new HashMap<>(); + + public List getDependencies(URI uri) { + return this.dependencies.get(uri); + } + + public Set getMethodCalls(URI uri) { + return this.methodCalls.get(uri); + } + + public List getStatements(URI uri) { + return this.statements.get(uri); + } + + public List getConstants(URI uri) { + return this.constants.get(uri); + } + + public Set getPlugins(URI uri) { + return this.plugins.get(uri); + } + + public void visitCompilationUnit(URI uri, GradleCompilationUnit compilationUnit) { + this.currentUri = uri; + compilationUnit.iterator().forEachRemaining(unit -> visitSourceUnit(unit)); + } + + public void visitSourceUnit(SourceUnit unit) { + ModuleNode moduleNode = unit.getAST(); + if (moduleNode != null) { + this.dependencies.put(this.currentUri, new ArrayList<>()); + this.methodCalls.put(this.currentUri, new HashSet<>()); + this.statements.put(this.currentUri, new ArrayList<>()); + this.constants.put(this.currentUri, new ArrayList<>()); + this.plugins.put(this.currentUri, new HashSet<>()); + visitModule(moduleNode); + } + } + + public void visitModule(ModuleNode node) { + BlockStatement blockStatement = node.getStatementBlock(); + this.statements.put(currentUri, blockStatement.getStatements()); + node.getClasses().forEach(classNode -> { + super.visitClass(classNode); + }); + } + + @Override + public void visitMethodCallExpression(MethodCallExpression node) { + this.methodCalls.get(this.currentUri).add(node); + if (node.getMethodAsString().equals("dependencies")) { + this.dependencies.get(this.currentUri).addAll(getDependencies(node)); + } else if (node.getMethodAsString().equals("plugins")) { + // match plugins { id: ${id} } + List plugins = getPluginFromPlugins(node); + this.plugins.get(this.currentUri).addAll(plugins); + } else if (node.getMethodAsString().equals("apply")) { + // match apply plugins: '${id}' + String plugin = getPluginFromApply(node); + if (plugin != null) { + this.plugins.get(this.currentUri).add(plugin); + } + } + super.visitMethodCallExpression(node); + } + + private List getDependencies(MethodCallExpression expression) { + Expression argument = expression.getArguments(); + if (argument instanceof ArgumentListExpression) { + return getDependencies((ArgumentListExpression) argument); + } + return Collections.emptyList(); + } + + private List getDependencies(ArgumentListExpression argumentListExpression) { + List expressions = argumentListExpression.getExpressions(); + List symbols = new ArrayList<>(); + for (Expression expression : expressions) { + if (expression instanceof ClosureExpression) { + symbols.addAll(getDependencies((ClosureExpression) expression)); + } else if (expression instanceof GStringExpression || expression instanceof ConstantExpression) { + // GStringExp: implementation "org.gradle:gradle-tooling-api:${gradleToolingApi}" + // ConstantExp: implementation "org.gradle:gradle-tooling-api:6.8.0" + symbols.add(new DependencyItem(expression.getText(), LSPUtils.toDependencyRange(expression))); + } else if (expression instanceof MethodCallExpression) { + symbols.addAll(getDependencies((MethodCallExpression) expression)); + } + } + return symbols; + } + + private List getDependencies(ClosureExpression expression) { + Statement code = expression.getCode(); + if (code instanceof BlockStatement) { + return getDependencies((BlockStatement) code); + } + return Collections.emptyList(); + } + + private List getDependencies(BlockStatement blockStatement) { + List statements = blockStatement.getStatements(); + List results = new ArrayList<>(); + for (Statement statement : statements) { + if (statement instanceof ExpressionStatement) { + results.addAll(getDependencies((ExpressionStatement) statement)); + } + } + return results; + } + + private List getDependencies(ExpressionStatement expressionStatement) { + Expression expression = expressionStatement.getExpression(); + if (expression instanceof MethodCallExpression) { + return getDependencies((MethodCallExpression) expression); + } + return Collections.emptyList(); + } + + private String getPluginFromApply(MethodCallExpression node) { + Expression argument = node.getArguments(); + if (argument instanceof TupleExpression) { + List expressions = ((TupleExpression)argument).getExpressions(); + for (Expression expression : expressions) { + if (expression instanceof NamedArgumentListExpression) { + List mapEntryExpressions = ((NamedArgumentListExpression)expression).getMapEntryExpressions(); + for (MapEntryExpression mapEntryExp: mapEntryExpressions) { + Expression keyExpression = mapEntryExp.getKeyExpression(); + if (keyExpression instanceof ConstantExpression && keyExpression.getText().equals("plugin")) { + return mapEntryExp.getValueExpression().getText(); + } + } + } + } + } + return null; + } + + private List getPluginFromPlugins(MethodCallExpression node) { + Expression objectExpression = node.getObjectExpression(); + if (objectExpression instanceof MethodCallExpression) { + return getPluginFromPlugins((MethodCallExpression)objectExpression); + } + List results = new ArrayList<>(); + Expression argument = node.getArguments(); + if (argument instanceof ArgumentListExpression) { + List expressions = ((ArgumentListExpression)argument).getExpressions(); + for (Expression expression : expressions) { + if (expression instanceof ConstantExpression && node.getMethodAsString().equals("id")) { + results.add(expression.getText()); + } else if (expression instanceof ClosureExpression) { + Statement code = ((ClosureExpression)expression).getCode(); + if (code instanceof BlockStatement) { + results.addAll(getPluginFromPlugins((BlockStatement) code)); + } + } + } + } + return results; + } + + private List getPluginFromPlugins(BlockStatement code) { + List results = new ArrayList<>(); + List statements = code.getStatements(); + for (Statement statement : statements) { + if (statement instanceof ExpressionStatement) { + Expression expression = ((ExpressionStatement)statement).getExpression(); + if (expression instanceof MethodCallExpression) { + results.addAll(getPluginFromPlugins((MethodCallExpression) expression)); + } + } + } + return results; + } + + @Override + public void visitConstantExpression(ConstantExpression expression) { + this.constants.get(currentUri).add(expression); + super.visitConstantExpression(expression); + } + + @Override + public void visitGStringExpression(GStringExpression expression) { + this.constants.get(currentUri).add(expression); + super.visitGStringExpression(expression); + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/compile/DocumentSymbolVisitor.java b/gradle-language-server/src/main/java/com/microsoft/gradle/compile/DocumentSymbolVisitor.java new file mode 100644 index 000000000..ea207f6d5 --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/compile/DocumentSymbolVisitor.java @@ -0,0 +1,289 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ +package com.microsoft.gradle.compile; + +import java.net.URI; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import com.microsoft.gradle.utils.LSPUtils; + +import org.codehaus.groovy.ast.ModuleNode; +import org.codehaus.groovy.ast.expr.ArgumentListExpression; +import org.codehaus.groovy.ast.expr.BinaryExpression; +import org.codehaus.groovy.ast.expr.ClosureExpression; +import org.codehaus.groovy.ast.expr.ConstantExpression; +import org.codehaus.groovy.ast.expr.Expression; +import org.codehaus.groovy.ast.expr.GStringExpression; +import org.codehaus.groovy.ast.expr.MapEntryExpression; +import org.codehaus.groovy.ast.expr.MethodCallExpression; +import org.codehaus.groovy.ast.expr.NamedArgumentListExpression; +import org.codehaus.groovy.ast.expr.PropertyExpression; +import org.codehaus.groovy.ast.expr.TupleExpression; +import org.codehaus.groovy.ast.expr.VariableExpression; +import org.codehaus.groovy.ast.stmt.BlockStatement; +import org.codehaus.groovy.ast.stmt.ExpressionStatement; +import org.codehaus.groovy.ast.stmt.Statement; +import org.codehaus.groovy.control.SourceUnit; +import org.eclipse.lsp4j.DocumentSymbol; +import org.eclipse.lsp4j.SymbolKind; + +public class DocumentSymbolVisitor { + + private URI currentUri; + private Map> documentSymbols = new HashMap<>(); + private Map> dependencies = new HashMap<>(); + + public List getDocumentSymbols(URI uri) { + return this.documentSymbols.get(uri); + } + + public List getDependencies(URI uri) { + return this.dependencies.get(uri); + } + + public void visitCompilationUnit(URI uri, GradleCompilationUnit compilationUnit) { + this.currentUri = uri; + compilationUnit.iterator().forEachRemaining(unit -> visitSourceUnit(uri, unit)); + } + + public void visitSourceUnit(URI uri, SourceUnit unit) { + ModuleNode moduleNode = unit.getAST(); + if (moduleNode != null) { + this.documentSymbols.put(uri, new ArrayList<>()); + this.dependencies.put(uri, new ArrayList<>()); + visitModule(moduleNode); + } + } + + public void visitModule(ModuleNode node) { + BlockStatement blockStatement = node.getStatementBlock(); + List statements = blockStatement.getStatements(); + for (Statement statement : statements) { + if (statement instanceof ExpressionStatement) { + DocumentSymbol symbol = getDocumentSymbol((ExpressionStatement) statement); + if (symbol != null) { + this.documentSymbols.get(this.currentUri).add(symbol); + } + } + } + } + + public DocumentSymbol getDocumentSymbol(ExpressionStatement statement) { + Expression expression = statement.getExpression(); + DocumentSymbol symbol = null; + if (expression instanceof MethodCallExpression) { + symbol = getDocumentSymbol((MethodCallExpression) expression); + } else if (expression instanceof BinaryExpression) { + symbol = getDocumentSymbol((BinaryExpression) expression); + } + if (symbol == null || symbol.getName() == null) { + return null; + } + return symbol; + } + + private DocumentSymbol getDocumentSymbol(BinaryExpression expression) { + Expression left = expression.getLeftExpression(); + Expression right = expression.getRightExpression(); + DocumentSymbol symbol = new DocumentSymbol(); + symbol.setName(left.getText()); + if (right instanceof ConstantExpression) { + symbol.setDetail(right.getText()); + } + symbol.setKind(SymbolKind.Property); + symbol.setRange(LSPUtils.toRange(expression)); + symbol.setSelectionRange(LSPUtils.toRange(expression)); + return symbol; + } + + private DocumentSymbol getDocumentSymbol(MethodCallExpression expression) { + DocumentSymbol symbol = new DocumentSymbol(); + symbol.setKind(SymbolKind.Function); + String name = getSymbolName(expression); + if (name == null) { + return null; + } + symbol.setName(name); + String detail = getSymbolDetail(expression); + if (detail != null) { + symbol.setDetail(detail); + } + symbol.setSelectionRange(LSPUtils.toRange(expression)); + symbol.setRange(LSPUtils.toRange(expression)); + if (expression.getMethodAsString().equals("dependencies")) { + List dependencySymbols = getDependencies(expression); + symbol.setChildren(dependencySymbols); + this.dependencies.get(currentUri).addAll(dependencySymbols); + } + return symbol; + } + + private String getSymbolName(MethodCallExpression expression) { + Expression objectExpression = expression.getObjectExpression(); + if (objectExpression instanceof VariableExpression) { + StringBuilder builder = new StringBuilder(); + String objectText = objectExpression.getText(); + if (!objectText.equals("this")) { + // variable "this" should be ignored + builder.append(objectText); + builder.append("."); + } + builder.append(expression.getMethodAsString()); + Expression arguments = expression.getArguments(); + if (arguments instanceof ArgumentListExpression) { + List expressions = ((ArgumentListExpression) arguments).getExpressions(); + for (Expression exp : expressions) { + if (exp instanceof MethodCallExpression) { + // for case: task taskName(Closure), we show "task taskName" in outline + builder.append(" "); + builder.append(getSymbolName((MethodCallExpression) exp)); + } + } + } + return builder.toString(); + } else if (objectExpression instanceof PropertyExpression) { + // for case: a.b.c.d("string"), we show "a.b.c.d" in outline + StringBuilder builder = new StringBuilder(); + builder.append(getSymbolName((PropertyExpression) objectExpression)); + builder.append("."); + builder.append(expression.getMethodAsString()); + return builder.toString(); + } + return null; + } + + private String getSymbolName(PropertyExpression expression) { + Expression objectExpression = expression.getObjectExpression(); + Expression property = expression.getProperty(); + StringBuilder builder = new StringBuilder(); + if (objectExpression instanceof PropertyExpression) { + builder.append(getSymbolName((PropertyExpression) objectExpression)); + } else if (objectExpression instanceof VariableExpression) { + builder.append(objectExpression.getText()); + } + if (property instanceof ConstantExpression) { + builder.append("."); + builder.append(property.getText()); + } + return builder.toString(); + } + + private String getSymbolDetail(MethodCallExpression expression) { + Expression argument = expression.getArguments(); + if (argument instanceof ArgumentListExpression) { + List arguments = ((ArgumentListExpression) argument).getExpressions(); + if (!arguments.isEmpty() && arguments.get(0) instanceof ConstantExpression) { + // if first arg is constantExpression, show it as detail + return arguments.get(0).getText(); + } + return null; + } else if (argument instanceof TupleExpression) { + // if argument is tupleExpression, show first argument as detail + List arguments = ((TupleExpression) argument).getExpressions(); + if (!arguments.isEmpty() && arguments.get(0) instanceof NamedArgumentListExpression) { + NamedArgumentListExpression namedArgumentListExpression = (NamedArgumentListExpression) arguments.get(0); + List mapEntryExpressions = namedArgumentListExpression.getMapEntryExpressions(); + if (!mapEntryExpressions.isEmpty()) { + MapEntryExpression firstExpression = mapEntryExpressions.get(0); + if (firstExpression.getValueExpression() instanceof ConstantExpression) { + StringBuilder detail = new StringBuilder(); + detail.append(firstExpression.getKeyExpression().getText()); + detail.append(": "); + detail.append(firstExpression.getValueExpression().getText()); + return detail.toString(); + } + } + return null; + } + } + return null; + } + + private List getDependencies(MethodCallExpression expression) { + Expression argument = expression.getArguments(); + if (expression.getMethodAsString().equals("dependencies")) { + return getDependencies((ArgumentListExpression) argument); + } + List results = new ArrayList<>(); + DocumentSymbol symbol = new DocumentSymbol(); + String name = expression.getMethodAsString(); + symbol.setName(name); + String detail = getDetail(expression); + if (detail != null) { + symbol.setDetail(detail); + } + symbol.setKind(SymbolKind.Constant); + symbol.setRange(LSPUtils.toRange(expression)); + symbol.setSelectionRange(LSPUtils.toRange(expression)); + results.add(symbol); + return results; + } + + private List getDependencies(ArgumentListExpression argumentListExpression) { + List expressions = argumentListExpression.getExpressions(); + List symbols = new ArrayList<>(); + for (Expression expression : expressions) { + if (expression instanceof ClosureExpression) { + symbols.addAll(getDependencies((ClosureExpression) expression)); + } else if (expression instanceof MethodCallExpression) { + symbols.addAll(getDependencies((MethodCallExpression) expression)); + } + } + return symbols; + } + + private List getDependencies(ClosureExpression expression) { + Statement code = expression.getCode(); + if (code instanceof BlockStatement) { + return getDependencies((BlockStatement) code); + } + return Collections.emptyList(); + } + + private List getDependencies(BlockStatement blockStatement) { + List statements = blockStatement.getStatements(); + List symbols = new ArrayList<>(); + for (Statement statement : statements) { + if (statement instanceof ExpressionStatement) { + symbols.addAll(getDependencies((ExpressionStatement) statement)); + } + } + return symbols; + } + + private List getDependencies(ExpressionStatement expressionStatement) { + Expression expression = expressionStatement.getExpression(); + List symbols = new ArrayList<>(); + if (expression instanceof MethodCallExpression) { + symbols.addAll(getDependencies((MethodCallExpression) expression)); + } + return symbols; + } + + private String getDetail(MethodCallExpression node) { + Expression arguments = node.getArguments(); + if (arguments instanceof ArgumentListExpression) { + List expressions = ((ArgumentListExpression) arguments).getExpressions(); + for (Expression expression : expressions) { + if (expression instanceof MethodCallExpression) { + return getDetail((MethodCallExpression) expression); + } else if (expression instanceof GStringExpression || expression instanceof ConstantExpression) { + return expression.getText(); + } + } + } + return null; + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/compile/GradleCompilationUnit.java b/gradle-language-server/src/main/java/com/microsoft/gradle/compile/GradleCompilationUnit.java new file mode 100644 index 000000000..e5ab55e99 --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/compile/GradleCompilationUnit.java @@ -0,0 +1,32 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ + +package com.microsoft.gradle.compile; + +import java.security.CodeSource; + +import org.codehaus.groovy.control.CompilationUnit; +import org.codehaus.groovy.control.CompilerConfiguration; + +import groovy.lang.GroovyClassLoader; + +public class GradleCompilationUnit extends CompilationUnit { + private Integer version; + + public GradleCompilationUnit(CompilerConfiguration configuration, CodeSource codeSource, GroovyClassLoader loader, Integer version) { + super(configuration, codeSource, loader); + this.version = version; + } + + public Integer getVersion() { + return this.version; + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/compile/SemanticTokenVisitor.java b/gradle-language-server/src/main/java/com/microsoft/gradle/compile/SemanticTokenVisitor.java new file mode 100644 index 000000000..c7ee40a49 --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/compile/SemanticTokenVisitor.java @@ -0,0 +1,101 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ +package com.microsoft.gradle.compile; + +import java.net.URI; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import com.microsoft.gradle.semantictokens.SemanticToken; +import com.microsoft.gradle.semantictokens.TokenModifier; +import com.microsoft.gradle.semantictokens.TokenType; + +import org.codehaus.groovy.ast.ASTNode; +import org.codehaus.groovy.ast.ClassCodeVisitorSupport; +import org.codehaus.groovy.ast.ModuleNode; +import org.codehaus.groovy.ast.expr.MapEntryExpression; +import org.codehaus.groovy.ast.expr.MethodCallExpression; +import org.codehaus.groovy.ast.expr.PropertyExpression; +import org.codehaus.groovy.ast.expr.VariableExpression; +import org.codehaus.groovy.control.SourceUnit; + +public class SemanticTokenVisitor extends ClassCodeVisitorSupport { + + private URI currentUri; + private Map> tokens = new HashMap<>(); + + public List getSemanticTokens(URI uri) { + return this.tokens.get(uri); + } + + private void addToken(int line, int column, int length, TokenType tokenType, int modifiers) { + if (length > 0) { + tokens.get(currentUri).add(new SemanticToken(line, column, length, tokenType, modifiers)); + } + } + + private void addToken(ASTNode node, TokenType tokenType, int modifiers) { + addToken(node.getLineNumber(), node.getColumnNumber(), node.getLength(), tokenType, modifiers); + } + + private void addToken(ASTNode node, TokenType tokenType) { + addToken(node.getLineNumber(), node.getColumnNumber(), node.getLength(), tokenType, 0); + } + + public void visitCompilationUnit(URI uri, GradleCompilationUnit compilationUnit) { + this.currentUri = uri; + compilationUnit.iterator().forEachRemaining(unit -> visitSourceUnit(uri, unit)); + } + + public void visitSourceUnit(URI uri, SourceUnit unit) { + ModuleNode moduleNode = unit.getAST(); + if (moduleNode != null) { + this.tokens.put(uri, new ArrayList<>()); + visitModule(moduleNode); + } + } + + public void visitModule(ModuleNode node) { + node.getClasses().forEach(classNode -> { + super.visitClass(classNode); + }); + } + + @Override + public void visitMethodCallExpression(MethodCallExpression node) { + if (TokenModifier.isDefaultLibrary(node.getMethod().getText())) { + addToken(node.getMethod(), TokenType.FUNCTION, TokenModifier.DEFAULT_LIBRARY.bitmask); + } else { + addToken(node.getMethod(), TokenType.FUNCTION); + } + super.visitMethodCallExpression(node); + } + + @Override + public void visitMapEntryExpression(MapEntryExpression node) { + addToken(node.getKeyExpression(), TokenType.PARAMETER); + super.visitMapEntryExpression(node); + } + + @Override + public void visitVariableExpression(VariableExpression node) { + addToken(node, TokenType.VARIABLE); + super.visitVariableExpression(node); + } + + @Override + public void visitPropertyExpression(PropertyExpression node) { + addToken(node.getProperty(), TokenType.PROPERTY); + super.visitPropertyExpression(node); + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/delegate/GradleDelegate.java b/gradle-language-server/src/main/java/com/microsoft/gradle/delegate/GradleDelegate.java new file mode 100644 index 000000000..802cba7a8 --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/delegate/GradleDelegate.java @@ -0,0 +1,61 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ + +package com.microsoft.gradle.delegate; + +import java.util.HashMap; +import java.util.Map; + +public class GradleDelegate { + private static String PROJECT = "org.gradle.api.Project"; + private static String SETTINGS = "org.gradle.api.initialization.Settings"; + private static Map delegateMap; + + static { + delegateMap = new HashMap<>(); + // plugins + delegateMap.put("application", "org.gradle.api.plugins.ApplicationPluginConvention"); + delegateMap.put("base", "org.gradle.api.plugins.BasePluginConvention"); + delegateMap.put("java", "org.gradle.api.plugins.JavaPluginConvention"); + delegateMap.put("war", "org.gradle.api.plugins.WarPluginConvention"); + // basic closures + delegateMap.put("plugins", "org.gradle.plugin.use.PluginDependenciesSpec"); + delegateMap.put("configurations", "org.gradle.api.artifacts.Configuration"); + delegateMap.put("dependencySubstitution", "org.gradle.api.artifacts.DependencySubstitutions"); + delegateMap.put("resolutionStrategy", "org.gradle.api.artifacts.ResolutionStrategy"); + delegateMap.put("artifacts", "org.gradle.api.artifacts.dsl.ArtifactHandler"); + delegateMap.put("components", "org.gradle.api.artifacts.dsl.ComponentMetadataHandler"); + delegateMap.put("modules", "org.gradle.api.artifacts.dsl.ComponentModuleMetadataHandler"); + delegateMap.put("dependencies", "org.gradle.api.artifacts.dsl.DependencyHandler"); + delegateMap.put("repositories", "org.gradle.api.artifacts.dsl.RepositoryHandler"); + delegateMap.put("publishing", "org.gradle.api.publish.PublishingExtension"); + delegateMap.put("publications", "org.gradle.api.publish.PublicationContainer"); + delegateMap.put("sourceSets", "org.gradle.api.tasks.SourceSet"); + delegateMap.put("distributions", "org.gradle.api.distribution.Distribution"); + delegateMap.put("fileTree", "org.gradle.api.file.ConfigurableFileTree"); + delegateMap.put("copySpec", "org.gradle.api.file.CopySpec"); + delegateMap.put("exec", "org.gradle.process.ExecSpec"); + delegateMap.put("files", "org.gradle.api.file.ConfigurableFileCollection"); + delegateMap.put("task", "org.gradle.api.Task"); + } + + public static Map getDelegateMap() { + return delegateMap; + } + + public static String getDefault() { + return PROJECT; + } + + public static String getSettings() { + return SETTINGS; + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/handlers/CompletionHandler.java b/gradle-language-server/src/main/java/com/microsoft/gradle/handlers/CompletionHandler.java new file mode 100644 index 000000000..464d7bc81 --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/handlers/CompletionHandler.java @@ -0,0 +1,150 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ +package com.microsoft.gradle.handlers; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashSet; +import java.util.List; +import java.util.Set; + +import com.microsoft.gradle.delegate.GradleDelegate; +import com.microsoft.gradle.resolver.GradleLibraryResolver; + +import org.apache.bcel.classfile.JavaClass; +import org.apache.bcel.classfile.Method; +import org.apache.bcel.generic.ObjectType; +import org.apache.bcel.generic.Type; +import org.codehaus.groovy.ast.expr.MethodCallExpression; +import org.eclipse.lsp4j.CompletionItem; +import org.eclipse.lsp4j.CompletionItemKind; +import org.eclipse.lsp4j.InsertTextFormat; + +public class CompletionHandler { + + private static String BUILD_GRADLE = "build.gradle"; + private static String SETTING_GRADLE = "settings.gradle"; + private static String DEPENDENCYHANDLER_CLASS = "org.gradle.api.artifacts.dsl.DependencyHandler"; + + public List getCompletionItems(MethodCallExpression containingCall, String fileName, GradleLibraryResolver resolver, boolean javaPluginsIncluded) { + String delegateClassName = null; + if (containingCall == null) { + if (fileName.equals(BUILD_GRADLE)) { + delegateClassName = GradleDelegate.getDefault(); + } else if (fileName.equals(SETTING_GRADLE)) { + delegateClassName = GradleDelegate.getSettings(); + } + } else { + delegateClassName = GradleDelegate.getDelegateMap().get(containingCall.getMethodAsString()); + } + if (delegateClassName == null) { + return Collections.emptyList(); + } + JavaClass delegateClass = resolver.getGradleLibraries().get(delegateClassName); + if (delegateClass == null) { + return Collections.emptyList(); + } + return getCompletionItemsFromClass(delegateClass, resolver, javaPluginsIncluded, new HashSet<>()); + } + + private List getCompletionItemsFromClass(JavaClass javaClass, GradleLibraryResolver resolver, boolean javaPluginsIncluded, Set resultSet) { + if (javaClass == null) { + return Collections.emptyList(); + } + List results = new ArrayList<>(); + for (String superInterface : javaClass.getInterfaceNames()) { + if (resolver.getGradleLibraries().containsKey(superInterface)) { + results.addAll(getCompletionItemsFromClass(resolver.getGradleLibraries().get(superInterface), resolver, javaPluginsIncluded, resultSet)); + } + } + String superClass = javaClass.getSuperclassName(); + if (resolver.getGradleLibraries().containsKey(superClass)) { + results.addAll(getCompletionItemsFromClass(resolver.getGradleLibraries().get(superClass), resolver, javaPluginsIncluded, resultSet)); + } + List methodNames = new ArrayList<>(); + Method[] methods = javaClass.getMethods(); + for (Method method : methods) { + StringBuilder labelBuilder = new StringBuilder(); + String methodName = method.getName(); + // When parsing a abstract class, we'll get a "" method which can't be called directly, + // So we filter it here. + if (methodName.equals("")) { + continue; + } + methodNames.add(methodName); + labelBuilder.append(methodName); + labelBuilder.append("("); + for (Type type : method.getArgumentTypes()) { + if (type instanceof ObjectType) { + String[] classNameSplits = ((ObjectType) type).getClassName().split("\\."); + String className = classNameSplits[classNameSplits.length - 1]; + String variableName = className.substring(0, 1).toLowerCase(); + labelBuilder.append(className); + labelBuilder.append(" "); + labelBuilder.append(variableName); + labelBuilder.append(","); + } + } + if (labelBuilder.charAt(labelBuilder.length() - 1) == ',') { + labelBuilder.deleteCharAt(labelBuilder.length() - 1); + } + labelBuilder.append(")"); + String label = labelBuilder.toString(); + CompletionItem item = new CompletionItem(label); + item.setKind(CompletionItemKind.Function); + item.setInsertTextFormat(InsertTextFormat.Snippet); + StringBuilder builder = new StringBuilder(); + builder.append(methodName); + if (label.endsWith("(Closure c)")) { + // for single closure, we offer curly brackets + builder.append(" {$0}"); + } else { + builder.append("($0)"); + } + item.setInsertText(builder.toString()); + if (resultSet.add(label)) { + results.add(item); + } + } + for (String methodName : methodNames) { + if (methodName.startsWith("set") && methodName.length() > 3) { + // for cases like setVersion() and getVersion(), + // we offer version as a property + String getMethod = "get" + methodName.substring(3); + if (methodNames.contains(getMethod)) { + String property = methodName.substring(3, 4).toLowerCase() + methodName.substring(4); + CompletionItem item = new CompletionItem(property); + item.setKind(CompletionItemKind.Property); + if (resultSet.add(property)) { + results.add(item); + } + } + } + } + if (javaPluginsIncluded && javaClass.getClassName().equals(DEPENDENCYHANDLER_CLASS)) { + // for dependency {}, we offer java configurations if there is any applied java plugin + for (String plugin : resolver.getJavaConfigurations()) { + StringBuilder builder = new StringBuilder(); + builder.append(plugin); + builder.append("(Object... o)"); + StringBuilder insertBuilder = new StringBuilder(); + insertBuilder.append(plugin); + insertBuilder.append("($0)"); + CompletionItem item = new CompletionItem(builder.toString()); + item.setKind(CompletionItemKind.Function); + item.setInsertTextFormat(InsertTextFormat.Snippet); + item.setInsertText(insertBuilder.toString()); + results.add(item); + } + } + return results; + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/handlers/DefaultDependenciesHandler.java b/gradle-language-server/src/main/java/com/microsoft/gradle/handlers/DefaultDependenciesHandler.java new file mode 100644 index 000000000..46422fe4b --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/handlers/DefaultDependenciesHandler.java @@ -0,0 +1,45 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License 2.0 + * which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + * + * Contributors: + * Microsoft Corporation - initial API and implementation +*******************************************************************************/ +package com.microsoft.gradle.handlers; + +import java.util.ArrayList; +import java.util.List; + +import org.eclipse.lsp4j.DocumentSymbol; +import org.eclipse.lsp4j.Range; + +public class DefaultDependenciesHandler { + public class DefaultDependencyItem { + private String name; + private String configuration; + private Range range; + + public DefaultDependencyItem(String name, String configuration, Range range) { + this.name = name; + this.configuration = configuration; + this.range = range; + } + } + + public List getDefaultDependencies(List symbols) { + List dependencies = new ArrayList<>(); + for (DocumentSymbol symbol : symbols) { + String configuration = symbol.getName(); + String id = symbol.getDetail(); + Range range = symbol.getRange(); + DefaultDependencyItem item = new DefaultDependencyItem(id, configuration, range); + dependencies.add(item); + } + return dependencies; + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/handlers/DependencyCompletionHandler.java b/gradle-language-server/src/main/java/com/microsoft/gradle/handlers/DependencyCompletionHandler.java new file mode 100644 index 000000000..511935b36 --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/handlers/DependencyCompletionHandler.java @@ -0,0 +1,136 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ +package com.microsoft.gradle.handlers; + +import java.io.InputStreamReader; +import java.net.URL; +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; + +import com.google.gson.Gson; +import com.google.gson.JsonArray; +import com.google.gson.JsonElement; +import com.google.gson.JsonObject; +import com.microsoft.gradle.compile.CompletionVisitor.DependencyItem; +import com.microsoft.gradle.utils.LSPUtils; + +import org.eclipse.lsp4j.CompletionItem; +import org.eclipse.lsp4j.CompletionItemKind; +import org.eclipse.lsp4j.Position; +import org.eclipse.lsp4j.Range; +import org.eclipse.lsp4j.TextEdit; +import org.eclipse.lsp4j.jsonrpc.messages.Either; + +public class DependencyCompletionHandler { + private DependencyItem dependency; + private Position position; + private static String URL_BASIC_SEARCH = "https://search.maven.org/solrsearch/select?q="; + + private enum DependencyCompletionKind { + ID, VERSION + } + + public List getDependencyCompletionItems(DependencyItem dependency, Position position) { + this.dependency = dependency; + this.position = position; + String validText = LSPUtils.getStringBeforePosition(dependency.getText(), dependency.getRange(), position); + String[] validTexts = validText.split(":", -1); + switch (validTexts.length) { + case 1: + return getDependenciesForInCompleteGroup(validTexts[0]); + case 2: + return getDependenciesForInCompleteArtifact(validTexts[0]); + case 3: + return getDependenciesForVersion(validTexts[0], validTexts[1]); + default: + return Collections.emptyList(); + } + } + + private List getDependenciesForInCompleteGroup(String group) { + if (group.length() < 3) { + return Collections.emptyList(); + } + StringBuilder builder = new StringBuilder(); + builder.append(URL_BASIC_SEARCH); + builder.append(group); + // limit the number of result to 50 + builder.append("&rows=50&wt=json"); + return getDependenciesFromRestAPI(builder.toString(), DependencyCompletionKind.ID); + } + + private List getDependenciesForInCompleteArtifact(String group) { + if (group.length() < 3) { + return Collections.emptyList(); + } + StringBuilder builder = new StringBuilder(); + builder.append(URL_BASIC_SEARCH); + builder.append("g:%22"); + builder.append(group); + // limit the number of result to 50 + builder.append("%22&rows=50&wt=json"); + return getDependenciesFromRestAPI(builder.toString(), DependencyCompletionKind.ID); + } + + private List getDependenciesForVersion(String group, String artifact) { + if (group.length() < 3 || artifact.length() < 3) { + return Collections.emptyList(); + } + StringBuilder builder = new StringBuilder(); + builder.append(URL_BASIC_SEARCH); + builder.append("g:%22"); + builder.append(group); + builder.append("%22+AND+a:%22"); + builder.append(artifact); + // limit the number of result to 50 + builder.append("%22&core=gav&rows=50&wt=json"); + return getDependenciesFromRestAPI(builder.toString(), DependencyCompletionKind.VERSION); + } + + private List getDependenciesFromRestAPI(String url, DependencyCompletionKind kind) { + try (InputStreamReader reader = new InputStreamReader(new URL(url).openStream())) { + JsonObject jsonResult = new Gson().fromJson(reader, JsonObject.class); + JsonObject response = jsonResult.getAsJsonObject("response"); + JsonArray docs = response.getAsJsonArray("docs"); + List completions = new ArrayList<>(); + for (int i = 0; i < docs.size(); i++) { + JsonElement element = docs.get(i); + if (element instanceof JsonObject) { + CompletionItem completionItem = new CompletionItem(); + JsonElement labelContent = ((JsonObject) element).get("id"); + String label = labelContent.getAsJsonPrimitive().getAsString(); + TextEdit textEdit = new TextEdit(new Range(this.dependency.getRange().getStart(), this.position), label); + completionItem.setTextEdit(Either.forLeft(textEdit)); + if (kind == DependencyCompletionKind.ID) { + completionItem.setLabel(label); + completionItem.setTextEdit(Either.forLeft(textEdit)); + completionItem.setKind(CompletionItemKind.Module); + completionItem.setDetail("mavenCentral"); + } else { + String version = ((JsonObject) element).get("v").getAsJsonPrimitive().getAsString(); + completionItem.setLabel(version); + completionItem.setFilterText(label + ":" + version); + completionItem.setKind(CompletionItemKind.Constant); + completionItem.setDetail("version"); + } + // Currently we have no more than 50 results, so the padding values should have at least 3 digits. + completionItem.setSortText(String.format("%03d", i)); + completions.add(completionItem); + } + } + return completions; + } catch (Exception e) { + // TODO + } + return Collections.emptyList(); + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/manager/GradleFilesManager.java b/gradle-language-server/src/main/java/com/microsoft/gradle/manager/GradleFilesManager.java new file mode 100644 index 000000000..e033a73a5 --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/manager/GradleFilesManager.java @@ -0,0 +1,119 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ + +package com.microsoft.gradle.manager; + +import java.io.BufferedReader; +import java.io.IOException; +import java.io.StringReader; +import java.net.URI; +import java.util.HashMap; +import java.util.Map; + +import com.microsoft.gradle.compile.GradleCompilationUnit; + +import org.codehaus.groovy.control.CompilerConfiguration; +import org.codehaus.groovy.control.SourceUnit; +import org.codehaus.groovy.control.io.StringReaderSource; +import org.eclipse.lsp4j.Position; +import org.eclipse.lsp4j.Range; +import org.eclipse.lsp4j.TextDocumentContentChangeEvent; + +import groovy.lang.GroovyClassLoader; + +public class GradleFilesManager { + private Map openFiles = new HashMap<>(); + private Map unitStorage = new HashMap<>(); + + public void didOpen(URI uri, String content) { + openFiles.put(uri, content); + } + + public void didChange(URI uri, TextDocumentContentChangeEvent change) { + String oldText = openFiles.get(uri); + Range range = change.getRange(); + if (range == null) { + openFiles.put(uri, change.getText()); + } else { + int offsetStart = getOffset(oldText, change.getRange().getStart()); + int offsetEnd = getOffset(oldText, change.getRange().getEnd()); + StringBuilder builder = new StringBuilder(); + builder.append(oldText.substring(0, offsetStart)); + builder.append(change.getText()); + builder.append(oldText.substring(offsetEnd)); + openFiles.put(uri, builder.toString()); + } + } + + public void didClose(URI uri) { + openFiles.remove(uri); + } + + public String getContents(URI uri) { + if (openFiles.containsKey(uri)) { + return openFiles.get(uri); + } + return null; + } + + public int getOffset(String string, Position position) { + int line = position.getLine(); + int character = position.getCharacter(); + int currentIndex = 0; + if (line > 0) { + BufferedReader reader = new BufferedReader(new StringReader(string)); + try { + int readLines = 0; + while (true) { + char currentChar = (char) reader.read(); + if (currentChar == -1) { + return -1; + } + currentIndex++; + if (currentChar == '\n') { + readLines++; + if (readLines == line) { + break; + } + } + } + reader.close(); + } catch (IOException e) { + return -1; + } + } + return currentIndex + character; + } + + public GradleCompilationUnit getCompilationUnit(URI uri, Integer version) { + if (this.unitStorage.containsKey(uri) && this.unitStorage.get(uri).getVersion().equals(version)) { + return this.unitStorage.get(uri); + } + CompilerConfiguration config = new CompilerConfiguration(); + GroovyClassLoader classLoader = new GroovyClassLoader(ClassLoader.getSystemClassLoader().getParent(), config, true); + GradleCompilationUnit unit = new GradleCompilationUnit(config, null, classLoader, version); + SourceUnit sourceUnit = new SourceUnit(uri.toString(), + new StringReaderSource(getContents(uri), unit.getConfiguration()), + unit.getConfiguration(), unit.getClassLoader(), unit.getErrorCollector()); + unit.addSource(sourceUnit); + this.unitStorage.put(uri, unit); + return unit; + } + + public GradleCompilationUnit getCompilationUnit(URI uri) { + // if there is no version info provided, we return the newest version + // when the previous cu exists, otherwise return null + if (this.unitStorage.containsKey(uri)) { + return this.unitStorage.get(uri); + } + return null; + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/resolver/GradleLibraryResolver.java b/gradle-language-server/src/main/java/com/microsoft/gradle/resolver/GradleLibraryResolver.java new file mode 100644 index 000000000..f7200c6b8 --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/resolver/GradleLibraryResolver.java @@ -0,0 +1,240 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ +package com.microsoft.gradle.resolver; + +import java.io.BufferedReader; +import java.io.File; +import java.io.FileInputStream; +import java.io.IOException; +import java.io.InputStreamReader; +import java.nio.file.Path; +import java.util.Arrays; +import java.util.Enumeration; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Set; +import java.util.jar.JarEntry; +import java.util.jar.JarFile; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +import org.apache.bcel.classfile.ClassFormatException; +import org.apache.bcel.classfile.ClassParser; +import org.apache.bcel.classfile.Field; +import org.apache.bcel.classfile.JavaClass; + +public class GradleLibraryResolver { + + private static String JAVA_PLUGIN = "org.gradle.api.plugins.JavaPlugin"; + + private Map gradleLibraries = new HashMap<>(); + private Set javaConfigurations = new HashSet<>(); + private Set javaPlugins = new HashSet<>(); + private String gradleHome; + private String gradleVersion; + private boolean gradleWrapperEnabled; + private String gradleUserHome; + private Path workspacePath; + + public GradleLibraryResolver() { + this.javaPlugins.addAll(Arrays.asList("java", "application", "groovy", "java-library", "war")); + } + + public void setGradleHome(String gradleHome) { + this.gradleHome = gradleHome; + } + + public void setGradleVersion(String gradleVersion) { + this.gradleVersion = gradleVersion; + } + + public void setGradleWrapperEnabled(boolean gradleWrapperEnabled) { + this.gradleWrapperEnabled = gradleWrapperEnabled; + } + + public void setGradleUserHome(String gradleUserHome) { + this.gradleUserHome = gradleUserHome; + } + + public void setWorkspacePath(Path workspacePath) { + this.workspacePath = workspacePath; + } + + public Map getGradleLibraries() { + return this.gradleLibraries; + } + + public Set getJavaConfigurations() { + return this.javaConfigurations; + } + + public void resolve() { + Path gradleUserHomePath = (this.gradleUserHome == null) ? Path.of(System.getProperty("user.home"), ".gradle") + : Path.of(this.gradleUserHome); + File libFile = null; + if (this.gradleWrapperEnabled) { + libFile = findLibWithWrapper(gradleUserHomePath); + } else if (this.gradleVersion != null) { + libFile = findLibWithDist(gradleUserHomePath, "gradle-" + this.gradleVersion); + } else if (this.gradleHome != null) { + Path libPath = Path.of(this.gradleHome).resolve("lib"); + libFile = findLibFile(libPath.toFile()); + } else { + return; + } + if (libFile == null || !libFile.exists()) { + return; + } + try { + JarFile libJar = new JarFile(libFile); + getGradleLibraries(libFile.toPath(), libJar); + File pluginLibFile = findPluginLibFile(libFile.toPath().getParent().resolve(Path.of("plugins")).toFile()); + if (pluginLibFile == null) { + return; + } + JarFile pluginLibJar = new JarFile(pluginLibFile); + getGradleLibraries(pluginLibFile.toPath(), pluginLibJar); + resolveJavaConfigurations(); + } catch (Exception e) { + // Do Nothing + } + } + + private File findLibWithWrapper(Path gradleUserHomePath) { + if (this.workspacePath == null) { + return null; + } + Path propertiesRelativePath = Path.of("gradle", "wrapper", "gradle-wrapper.properties"); + Path propertiesPath = this.workspacePath.resolve(propertiesRelativePath); + File propertiesFile = propertiesPath.toFile(); + if (!propertiesFile.exists()) { + return null; + } + try (FileInputStream stream = new FileInputStream(propertiesFile)) { + BufferedReader reader = new BufferedReader(new InputStreamReader(stream)); + String content = null; + while ((content = reader.readLine()) != null) { + if (content.startsWith("distributionUrl")) { + Pattern p = Pattern.compile("(gradle-(?s)(.*))-bin"); + Matcher matcher = p.matcher(content); + if (matcher.find()) { + String gradleDist = matcher.group(1); + return findLibWithDist(gradleUserHomePath, gradleDist); + } + } + } + } catch (IOException e) { + // Do Nothing + } + return null; + } + + private File findLibWithDist(Path gradleUserHomePath, String gradleDist) { + Path distPath = gradleUserHomePath.resolve(Path.of("wrapper", "dists")); + File distFolder = searchInFolder(gradleDist, distPath.toFile()); + if (distFolder != null && distFolder.exists()) { + Path libPath = distFolder.toPath().resolve("lib"); + return findLibFile(libPath.toFile()); + } + return null; + } + + private File searchInFolder(String gradleDist, File folder) { + for (File file : folder.listFiles()) { + if (file.isDirectory()) { + if (file.getName().equals(gradleDist)) { + return file; + } else { + File searchResult = searchInFolder(gradleDist, file); + if (searchResult != null) { + return searchResult; + } + } + } + } + return null; + } + + private File findLibFile(File folder) { + for (File file : folder.listFiles()) { + String name = file.getName(); + if (name.startsWith("gradle-core-api") && name.endsWith(".jar")) { + return file; + } + } + // For Gradle version under 5.6, the name of library file is like gradle-core-${version}.jar + for (File file : folder.listFiles()) { + String name = file.getName(); + if (name.startsWith("gradle-core") && name.endsWith(".jar")) { + return file; + } + } + return null; + } + + private File findPluginLibFile(File folder) { + for (File file : folder.listFiles()) { + String name = file.getName(); + if (name.startsWith("gradle-plugins") && name.endsWith(".jar")) { + return file; + } + } + return null; + } + + private void getGradleLibraries(Path jarPath, JarFile jarFile) { + Enumeration entries = jarFile.entries(); + while (entries.hasMoreElements()) { + JarEntry entry = entries.nextElement(); + if (!entry.getName().endsWith(".class")) { + continue; + } + ClassParser parser = new ClassParser(jarPath.toString(), entry.getName()); + try { + JavaClass javaClass = parser.parse(); + String className = javaClass.getClassName(); + this.gradleLibraries.put(className, javaClass); + } catch (IOException | ClassFormatException e) { + // Do Nothing + } + } + } + + private void resolveJavaConfigurations() { + JavaClass javaPluginClass = this.gradleLibraries.get(GradleLibraryResolver.JAVA_PLUGIN); + if (javaPluginClass == null) { + return; + } + for (Field field : javaPluginClass.getFields()) { + if (field.getName().endsWith("CONFIGURATION_NAME")) { + this.javaConfigurations.add(removeQuotes(field.getConstantValue().toString())); + } + } + } + + private static String removeQuotes(String original) { + // for those fields parsed from class files, we get ""values"", so we remove the starting and ending quotes here + if (original.length() < 3) { + return original; + } + return original.substring(1, original.length() - 1); + } + + public boolean isJavaPluginsIncluded(Set plugins) { + for (String plugin : plugins) { + if (this.javaPlugins.contains(plugin)) { + return true; + } + } + return false; + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/semantictokens/SemanticToken.java b/gradle-language-server/src/main/java/com/microsoft/gradle/semantictokens/SemanticToken.java new file mode 100644 index 000000000..636c566df --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/semantictokens/SemanticToken.java @@ -0,0 +1,95 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ +package com.microsoft.gradle.semantictokens; + +import java.util.ArrayList; +import java.util.Comparator; +import java.util.List; + +public class SemanticToken { + private final TokenType tokenType; + private final int tokenModifiers; + private final int line; + private final int column; + private final int length; + + public SemanticToken(int line, int column, int length, TokenType tokenType, int tokenModifiers) { + this.line = line; + this.column = column; + this.length = length; + this.tokenType = tokenType; + this.tokenModifiers = tokenModifiers; + } + + public TokenType getTokenType() { + return tokenType; + } + + public int getTokenModifiers() { + return tokenModifiers; + } + + public int getLine() { + return line; + } + + public int getColumn() { + return column; + } + + public int getLength() { + return length; + } + + // Note: similar logics as JDT.LS, but in groovy AST ranges start from 1 + public static List encodedTokens(List tokens) { + tokens.sort(new Comparator() { + @Override + public int compare(final SemanticToken a, final SemanticToken b) { + int lineResult = Integer.valueOf(a.getLine()).compareTo(Integer.valueOf(b.getLine())); + if (lineResult == 0) { + return Integer.valueOf(a.getColumn()).compareTo(Integer.valueOf(b.getColumn())); + } + return lineResult; + } + }); + int numTokens = tokens.size(); + List data = new ArrayList<>(numTokens * 5); + int currentLine = 0; + int currentColumn = 0; + for (int i = 0; i < numTokens; i++) { + SemanticToken token = tokens.get(i); + int line = token.getLine() - 1; + int column = token.getColumn() - 1; + if (line < 0 || column < 0) { + continue; + } + int deltaLine = line - currentLine; + if (deltaLine != 0) { + currentLine = line; + currentColumn = 0; + } + int deltaColumn = column - currentColumn; + currentColumn = column; + // Disallow duplicate/conflict token (if exists) + if (deltaLine != 0 || deltaColumn != 0 || i == 0) { + int tokenTypeIndex = token.getTokenType().ordinal(); + int tokenModifiers = token.getTokenModifiers(); + data.add(deltaLine); + data.add(deltaColumn); + data.add(token.getLength()); + data.add(tokenTypeIndex); + data.add(tokenModifiers); + } + } + return data; + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/semantictokens/TokenModifier.java b/gradle-language-server/src/main/java/com/microsoft/gradle/semantictokens/TokenModifier.java new file mode 100644 index 000000000..95a98ed2e --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/semantictokens/TokenModifier.java @@ -0,0 +1,46 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License 2.0 + * which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + * + * Contributors: + * Microsoft Corporation - initial API and implementation +*******************************************************************************/ +package com.microsoft.gradle.semantictokens; + +import java.util.List; + +import org.eclipse.lsp4j.SemanticTokenModifiers; + +public enum TokenModifier { + + DEFAULT_LIBRARY(SemanticTokenModifiers.DefaultLibrary); + + private String genericName; + // See https://docs.gradle.org/current/javadoc/org/gradle/api/Project.html + private static List defaultLibrary = List.of("afterEvaluate", "allprojects", "ant", "apply", "artifacts", + "beforeEvaluate", "buildscript", "configurations", "configure", "copy", "copySpec", "dependencies", "javaexec", + "repositories", "subprojects", "task"); + + public final int bitmask = 1 << ordinal(); + + TokenModifier(String genericName) { + this.genericName = genericName; + } + + @Override + public String toString() { + return genericName; + } + + public static boolean isDefaultLibrary(String method) { + if (TokenModifier.defaultLibrary.contains(method)) { + return true; + } + return false; + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/semantictokens/TokenType.java b/gradle-language-server/src/main/java/com/microsoft/gradle/semantictokens/TokenType.java new file mode 100644 index 000000000..f3c7d1398 --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/semantictokens/TokenType.java @@ -0,0 +1,31 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License 2.0 + * which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + * + * Contributors: + * Microsoft Corporation - initial API and implementation +*******************************************************************************/ +package com.microsoft.gradle.semantictokens; + +import org.eclipse.lsp4j.SemanticTokenTypes; + +public enum TokenType { + FUNCTION(SemanticTokenTypes.Function), PROPERTY(SemanticTokenTypes.Property), VARIABLE(SemanticTokenTypes.Variable), + PARAMETER(SemanticTokenTypes.Parameter); + + private String genericName; + + TokenType(String genericName) { + this.genericName = genericName; + } + + @Override + public String toString() { + return genericName; + } +} diff --git a/gradle-language-server/src/main/java/com/microsoft/gradle/utils/LSPUtils.java b/gradle-language-server/src/main/java/com/microsoft/gradle/utils/LSPUtils.java new file mode 100644 index 000000000..2bb97924e --- /dev/null +++ b/gradle-language-server/src/main/java/com/microsoft/gradle/utils/LSPUtils.java @@ -0,0 +1,52 @@ +/******************************************************************************* + * Copyright (c) 2021 Microsoft Corporation and others. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * Microsoft Corporation - initial API and implementation + *******************************************************************************/ +package com.microsoft.gradle.utils; + +import org.codehaus.groovy.ast.expr.Expression; +import org.codehaus.groovy.ast.stmt.Statement; +import org.codehaus.groovy.syntax.SyntaxException; +import org.eclipse.lsp4j.Position; +import org.eclipse.lsp4j.Range; + +public class LSPUtils { + public static Range toRange(SyntaxException exp) { + // LSP Range start from 0, while groovy classes start from 1 + return new Range(new Position(exp.getStartLine() - 1, exp.getStartColumn() - 1), + new Position(exp.getEndLine() - 1, exp.getEndColumn() - 1)); + } + + public static Range toRange(Expression expression) { + // LSP Range start from 0, while groovy expressions start from 1 + return new Range(new Position(expression.getLineNumber() - 1, expression.getColumnNumber() - 1), + new Position(expression.getLastLineNumber() - 1, expression.getLastColumnNumber() - 1)); + } + + public static Range toRange(Statement statement) { + // LSP Range start from 0, while groovy expressions start from 1 + return new Range(new Position(statement.getLineNumber() - 1, statement.getColumnNumber() - 1), + new Position(statement.getLastLineNumber() - 1, statement.getLastColumnNumber() - 1)); + } + + public static Range toDependencyRange(Expression expression) { + // For dependency, the string includes open/close quotes should be excluded + return new Range(new Position(expression.getLineNumber() - 1, expression.getColumnNumber()), + new Position(expression.getLastLineNumber() - 1, expression.getLastColumnNumber() - 2)); + } + + public static String getStringBeforePosition(String text, Range range, Position position) { + Position start = range.getStart(); + // Since it's used for extract dependency info, we doesn't support multiple line + if (start.getLine() != position.getLine()) { + return text; + } + return text.substring(0, position.getCharacter() - start.getCharacter()); + } +} diff --git a/settings.gradle b/settings.gradle index d6f05a9b2..7e570b28b 100644 --- a/settings.gradle +++ b/settings.gradle @@ -4,3 +4,4 @@ include('extension') include('npm-package') include('gradle-plugin-api') include('gradle-plugin') +include('gradle-language-server')