From 80176cde31bd0bba2bd24ec8c920464f3e3676f4 Mon Sep 17 00:00:00 2001 From: Adam Semenenko <152864218+adam-enko@users.noreply.github.com> Date: Wed, 31 Jan 2024 09:52:17 +0100 Subject: [PATCH 1/2] test for NPM install caching, and comment-out overlapping outputs in NpmInstallTask - add tests for `npmInstall` task caching - add test fixture project for `npmInstall` caching --- .../gradle/node/npm/task/NpmInstallTask.kt | 72 ++- .../npm/task/NpmBuildCache_integTest.groovy | 101 +++ .../fixtures/npm-build-cache/.gitignore | 2 + .../fixtures/npm-build-cache/build.gradle.kts | 53 ++ .../npm-build-cache/gradle.properties | 3 + .../npm-build-cache/package-lock.json | 610 ++++++++++++++++++ .../fixtures/npm-build-cache/package.json | 23 + .../npm-build-cache/settings.gradle.kts | 50 ++ .../fixtures/npm-build-cache/src/main/app.ts | 11 + .../fixtures/npm-build-cache/tsconfig.json | 11 + 10 files changed, 901 insertions(+), 35 deletions(-) create mode 100644 src/test/groovy/com/github/gradle/node/npm/task/NpmBuildCache_integTest.groovy create mode 100644 src/test/resources/fixtures/npm-build-cache/.gitignore create mode 100644 src/test/resources/fixtures/npm-build-cache/build.gradle.kts create mode 100644 src/test/resources/fixtures/npm-build-cache/gradle.properties create mode 100644 src/test/resources/fixtures/npm-build-cache/package-lock.json create mode 100644 src/test/resources/fixtures/npm-build-cache/package.json create mode 100644 src/test/resources/fixtures/npm-build-cache/settings.gradle.kts create mode 100644 src/test/resources/fixtures/npm-build-cache/src/main/app.ts create mode 100644 src/test/resources/fixtures/npm-build-cache/tsconfig.json diff --git a/src/main/kotlin/com/github/gradle/node/npm/task/NpmInstallTask.kt b/src/main/kotlin/com/github/gradle/node/npm/task/NpmInstallTask.kt index 5befc21a..41c0bbc1 100644 --- a/src/main/kotlin/com/github/gradle/node/npm/task/NpmInstallTask.kt +++ b/src/main/kotlin/com/github/gradle/node/npm/task/NpmInstallTask.kt @@ -64,13 +64,14 @@ abstract class NpmInstallTask : NpmTask() { return projectFileIfExists("yarn.lock").orNull } - @Optional - @OutputFile - protected fun getPackageLockFileAsOutput(): File? { - return npmCommand.flatMap { command -> - if (command[0] == "install") projectFileIfExists("package-lock.json") else providers.provider { null } - }.orNull - } + // does this overlap with getNodeModulesFiles()? +// @Optional +// @OutputFile +// protected fun getPackageLockFileAsOutput(): File? { +// return npmCommand.flatMap { command -> +// if (command[0] == "install") projectFileIfExists("package-lock.json") else providers.provider { null } +// }.orNull +// } private fun projectFileIfExists(name: String): Provider { return nodeExtension.nodeProjectDir.map { it.file(name).asFile } @@ -105,34 +106,35 @@ abstract class NpmInstallTask : NpmTask() { } } - @Optional - @OutputFile - protected fun getNodeModulesPackageLock(): File? { - if (isLegacyNpm()) { - return null - } - - return projectFileIfExists("node_modules/.package-lock.json").orNull - } - - /** - * Is our npm likely to be lower than 7? - */ - private fun isLegacyNpm(): Boolean { - if (nodeExtension.oldNpm.get()) { - return true - } - - val npmVersion = nodeExtension.npmVersion.get() - if (npmVersion.isBlank()) { - if (nodeExtension.version.get().split('.').first().toInt() <= 14) - return true - } else if (npmVersion.split('.').first().toInt() < 7 ) { - return true - } - - return false - } + // does this overlap with getNodeModulesFiles()? +// @Optional +// @OutputFile +// protected fun getNodeModulesPackageLock(): File? { +// if (isLegacyNpm()) { +// return null +// } +// +// return projectFileIfExists("node_modules/.package-lock.json").orNull +// } +// +// /** +// * Is our npm likely to be lower than 7? +// */ +// private fun isLegacyNpm(): Boolean { +// if (nodeExtension.oldNpm.get()) { +// return true +// } +// +// val npmVersion = nodeExtension.npmVersion.get() +// if (npmVersion.isBlank()) { +// if (nodeExtension.version.get().split('.').first().toInt() <= 14) +// return true +// } else if (npmVersion.split('.').first().toInt() < 7 ) { +// return true +// } +// +// return false +// } // For DSL @Suppress("unused") diff --git a/src/test/groovy/com/github/gradle/node/npm/task/NpmBuildCache_integTest.groovy b/src/test/groovy/com/github/gradle/node/npm/task/NpmBuildCache_integTest.groovy new file mode 100644 index 00000000..2f9e97fc --- /dev/null +++ b/src/test/groovy/com/github/gradle/node/npm/task/NpmBuildCache_integTest.groovy @@ -0,0 +1,101 @@ +package com.github.gradle.node.npm.task + +import com.github.gradle.AbstractIntegTest + +import static org.gradle.testkit.runner.TaskOutcome.FROM_CACHE +import static org.gradle.testkit.runner.TaskOutcome.SUCCESS + +class NpmBuildCache_integTest extends AbstractIntegTest { + + def 'npmInstall can be loaded from-cache'() { + given: + gradleVersion = gv + + copyResources("fixtures/npm-build-cache/") + + createFile("build-cache/").deleteDir() + createFile("dist/").deleteDir() + + + when: + def assemble1Result = build("assemble", "--stacktrace", "-PenableNpmInstallCaching=true") + + then: + assemble1Result.task(":npmInstall").outcome == SUCCESS + assemble1Result.task(":npmRunBuild").outcome == SUCCESS + + createFile("package-lock.json").exists() + createFile("node_modules").exists() + createFile("dist/app.js").isFile() + + + when: + def cleanResult = build("clean", "--stacktrace", "-PenableNpmInstallCaching=true") + + then: + cleanResult.task(":clean").outcome == SUCCESS + createFile("package-lock.json").exists() + !createFile("node_modules").exists() + !createFile("dist").exists() + + + when: + def assemble2Result = build("assemble", "--stacktrace", "-PenableNpmInstallCaching=true") + + then: + assemble2Result.task(":npmInstall").outcome == FROM_CACHE + assemble2Result.task(":npmRunBuild").outcome == FROM_CACHE + createFile("node_modules").exists() + createFile("dist").exists() + createFile("dist/app.js").isFile() + + where: + gv << GRADLE_VERSIONS_UNDER_TEST + } + + + def 'test npmInstall has cacheable outputs'() { + // check the `org.gradle.caching.debug` logs to verify that the output is cacheable + + given: + gradleVersion = gv + + copyResources("fixtures/npm-build-cache/") + + createFile("build-cache/").deleteDir() + createFile("dist/").deleteDir() + + + when: + def args = ["assemble", "--stacktrace", "-Dorg.gradle.caching.debug=true", "-PenableNpmInstallCaching=true"] + def assemble1Result = build(*args) + def npmInstall1Output = assemble1Result.output + .takeAfter("> Task :npmInstall") + .takeAfter("\n") + .takeBefore("\n\n") + + then: + !npmInstall1Output.contains("Non-cacheable") + !npmInstall1Output.contains("[OVERLAPPING_OUTPUTS]") + !npmInstall1Output.contains("Gradle does not know how file 'node_modules/.package-lock.json' was created") + + + when: + def assemble2Result = build(*args) + def npmInstall2Output = assemble2Result.output + .takeAfter("> Task :npmInstall") + .takeAfter("\n") + .takeBefore("\n\n") + + then: + !npmInstall2Output.contains("Non-cacheable") + !npmInstall2Output.contains("[OVERLAPPING_OUTPUTS]") + !npmInstall2Output.contains("Gradle does not know how file 'node_modules/.package-lock.json' was created") + + // the inputs and outputs shouldn't have changed, so the fingerprinted properties should be the same + npmInstall1Output == npmInstall2Output + + where: + gv << GRADLE_VERSIONS_UNDER_TEST + } +} diff --git a/src/test/resources/fixtures/npm-build-cache/.gitignore b/src/test/resources/fixtures/npm-build-cache/.gitignore new file mode 100644 index 00000000..b592f705 --- /dev/null +++ b/src/test/resources/fixtures/npm-build-cache/.gitignore @@ -0,0 +1,2 @@ +dist/ +build-cache/ diff --git a/src/test/resources/fixtures/npm-build-cache/build.gradle.kts b/src/test/resources/fixtures/npm-build-cache/build.gradle.kts new file mode 100644 index 00000000..01bee04f --- /dev/null +++ b/src/test/resources/fixtures/npm-build-cache/build.gradle.kts @@ -0,0 +1,53 @@ +import com.github.gradle.node.npm.task.NpmTask +import org.gradle.api.tasks.PathSensitivity + +plugins { + id("com.github.node-gradle.node") + base +} + +node { + version.set("16.13.0") + + download.set(true) + distBaseUrl.set(null as String?) + + //fastNpmInstall.set(true) +} + +tasks.npmInstall { + val enableNpmInstallCaching = + providers.gradleProperty("enableNpmInstallCaching").map(String::toBoolean).orElse(false) + inputs.property("enableNpmInstallCaching", enableNpmInstallCaching) + outputs.cacheIf { enableNpmInstallCaching.get() } +} + +val distributionDirectory = layout.projectDirectory.dir("dist") + +val npmRunBuild by tasks.registering(NpmTask::class) { + dependsOn(tasks.npmInstall) + + npmCommand.set(listOf("run", "build")) + + inputs.dir("src/main") + .withPropertyName("sources") + .withPathSensitivity(PathSensitivity.RELATIVE) + + inputs.file("package.json") + .withPropertyName("packageJson") + .withPathSensitivity(PathSensitivity.RELATIVE) + + outputs.dir(distributionDirectory) + .withPropertyName("distributionDirectory") + + outputs.cacheIf("always cache, this task produces files") { true } +} + +tasks.assemble { + dependsOn(npmRunBuild) +} + +tasks.clean { + delete(distributionDirectory) + delete("node_modules") +} diff --git a/src/test/resources/fixtures/npm-build-cache/gradle.properties b/src/test/resources/fixtures/npm-build-cache/gradle.properties new file mode 100644 index 00000000..21b0d8ad --- /dev/null +++ b/src/test/resources/fixtures/npm-build-cache/gradle.properties @@ -0,0 +1,3 @@ +org.gradle.parallel=true +org.gradle.caching=true +org.gradle.caching.debug=true diff --git a/src/test/resources/fixtures/npm-build-cache/package-lock.json b/src/test/resources/fixtures/npm-build-cache/package-lock.json new file mode 100644 index 00000000..1f292cc2 --- /dev/null +++ b/src/test/resources/fixtures/npm-build-cache/package-lock.json @@ -0,0 +1,610 @@ +{ + "name": "kt-npm-1", + "version": "1.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "kt-npm-1", + "version": "1.0.0", + "license": "ISC", + "dependencies": { + "express": "^4.17.1" + }, + "devDependencies": { + "@types/express": "^4.17.1", + "typescript": "^5.3.3" + } + }, + "node_modules/@types/body-parser": { + "version": "1.19.5", + "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.5.tgz", + "integrity": "sha512-fB3Zu92ucau0iQ0JMCFQE7b/dv8Ot07NI3KaZIkIUNXq82k4eBAqUaneXfleGY9JWskeS9y+u0nXMyspcuQrCg==", + "dev": true, + "dependencies": { + "@types/connect": "*", + "@types/node": "*" + } + }, + "node_modules/@types/connect": { + "version": "3.4.38", + "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.38.tgz", + "integrity": "sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==", + "dev": true, + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/express": { + "version": "4.17.1", + "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.1.tgz", + "integrity": "sha512-VfH/XCP0QbQk5B5puLqTLEeFgR8lfCJHZJKkInZ9mkYd+u8byX0kztXEQxEk4wZXJs8HI+7km2ALXjn4YKcX9w==", + "dev": true, + "dependencies": { + "@types/body-parser": "*", + "@types/express-serve-static-core": "*", + "@types/serve-static": "*" + } + }, + "node_modules/@types/express-serve-static-core": { + "version": "4.17.42", + "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.42.tgz", + "integrity": "sha512-ckM3jm2bf/MfB3+spLPWYPUH573plBFwpOhqQ2WottxYV85j1HQFlxmnTq57X1yHY9awZPig06hL/cLMgNWHIQ==", + "dev": true, + "dependencies": { + "@types/node": "*", + "@types/qs": "*", + "@types/range-parser": "*", + "@types/send": "*" + } + }, + "node_modules/@types/http-errors": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.4.tgz", + "integrity": "sha512-D0CFMMtydbJAegzOyHjtiKPLlvnm3iTZyZRSZoLq2mRhDdmLfIWOCYPfQJ4cu2erKghU++QvjcUjp/5h7hESpA==", + "dev": true + }, + "node_modules/@types/mime": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.5.tgz", + "integrity": "sha512-/pyBZWSLD2n0dcHE3hq8s8ZvcETHtEuF+3E7XVt0Ig2nvsVQXdghHVcEkIWjy9A0wKfTn97a/PSDYohKIlnP/w==", + "dev": true + }, + "node_modules/@types/node": { + "version": "20.11.10", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.11.10.tgz", + "integrity": "sha512-rZEfe/hJSGYmdfX9tvcPMYeYPW2sNl50nsw4jZmRcaG0HIAb0WYEpsB05GOb53vjqpyE9GUhlDQ4jLSoB5q9kg==", + "dev": true, + "dependencies": { + "undici-types": "~5.26.4" + } + }, + "node_modules/@types/qs": { + "version": "6.9.11", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.11.tgz", + "integrity": "sha512-oGk0gmhnEJK4Yyk+oI7EfXsLayXatCWPHary1MtcmbAifkobT9cM9yutG/hZKIseOU0MqbIwQ/u2nn/Gb+ltuQ==", + "dev": true + }, + "node_modules/@types/range-parser": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.7.tgz", + "integrity": "sha512-hKormJbkJqzQGhziax5PItDUTMAM9uE2XXQmM37dyd4hVM+5aVl7oVxMVUiVQn2oCQFN/LKCZdvSM0pFRqbSmQ==", + "dev": true + }, + "node_modules/@types/send": { + "version": "0.17.4", + "resolved": "https://registry.npmjs.org/@types/send/-/send-0.17.4.tgz", + "integrity": "sha512-x2EM6TJOybec7c52BX0ZspPodMsQUd5L6PRwOunVyVUhXiBSKf3AezDL8Dgvgt5o0UfKNfuA0eMLr2wLT4AiBA==", + "dev": true, + "dependencies": { + "@types/mime": "^1", + "@types/node": "*" + } + }, + "node_modules/@types/serve-static": { + "version": "1.15.5", + "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.15.5.tgz", + "integrity": "sha512-PDRk21MnK70hja/YF8AHfC7yIsiQHn1rcXx7ijCFBX/k+XQJhQT/gw3xekXKJvx+5SXaMMS8oqQy09Mzvz2TuQ==", + "dev": true, + "dependencies": { + "@types/http-errors": "*", + "@types/mime": "*", + "@types/node": "*" + } + }, + "node_modules/accepts": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", + "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", + "dependencies": { + "mime-types": "~2.1.34", + "negotiator": "0.6.3" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==" + }, + "node_modules/body-parser": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", + "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", + "dependencies": { + "bytes": "3.1.0", + "content-type": "~1.0.4", + "debug": "2.6.9", + "depd": "~1.1.2", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "on-finished": "~2.3.0", + "qs": "6.7.0", + "raw-body": "2.4.0", + "type-is": "~1.6.17" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/bytes": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", + "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/content-disposition": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", + "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", + "dependencies": { + "safe-buffer": "5.1.2" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/content-type": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", + "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", + "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==" + }, + "node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/depd": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", + "integrity": "sha512-7emPTl6Dpo6JRXOXjLRxck+FlLRX5847cLKEn00PLAgc3g2hTZZgr+e4c2v6QpSmLeFP3n5yUo7ft6avBK/5jQ==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/destroy": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", + "integrity": "sha512-3NdhDuEXnfun/z7x9GOElY49LoqVHoGScmOKwmxhsS8N5Y+Z8KyPPDnaSzqWgYt/ji4mqwfTS34Htrk0zPIXVg==" + }, + "node_modules/ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==" + }, + "node_modules/encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==" + }, + "node_modules/etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/express": { + "version": "4.17.1", + "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", + "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", + "dependencies": { + "accepts": "~1.3.7", + "array-flatten": "1.1.1", + "body-parser": "1.19.0", + "content-disposition": "0.5.3", + "content-type": "~1.0.4", + "cookie": "0.4.0", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "~1.1.2", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "~1.1.2", + "fresh": "0.5.2", + "merge-descriptors": "1.0.1", + "methods": "~1.1.2", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "path-to-regexp": "0.1.7", + "proxy-addr": "~2.0.5", + "qs": "6.7.0", + "range-parser": "~1.2.1", + "safe-buffer": "5.1.2", + "send": "0.17.1", + "serve-static": "1.14.1", + "setprototypeof": "1.1.1", + "statuses": "~1.5.0", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + }, + "engines": { + "node": ">= 0.10.0" + } + }, + "node_modules/finalhandler": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", + "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", + "dependencies": { + "debug": "2.6.9", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "statuses": "~1.5.0", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/forwarded": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/http-errors": { + "version": "1.7.2", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", + "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", + "dependencies": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.1", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha512-x00IRNXNy63jwGkJmzPigoySHbaqpNuzKbBOmzK+g2OdZpQ9w+sxCN+VSB3ja7IAge2OP2qpfxTjeNcyjmW1uw==" + }, + "node_modules/ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/merge-descriptors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", + "integrity": "sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w==" + }, + "node_modules/methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, + "node_modules/negotiator": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", + "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/on-finished": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", + "integrity": "sha512-ikqdkGAAyf/X/gPhXGvfgAytDZtDbr+bkNUJ0N9h5MI/dmdgCs3l6hoHrcUv41sRKew3jIwrp4qQDXiK99Utww==", + "dependencies": { + "ee-first": "1.1.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/path-to-regexp": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", + "integrity": "sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ==" + }, + "node_modules/proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "dependencies": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/qs": { + "version": "6.7.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", + "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/raw-body": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", + "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", + "dependencies": { + "bytes": "3.1.0", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/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==" + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "node_modules/send": { + "version": "0.17.1", + "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", + "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", + "dependencies": { + "debug": "2.6.9", + "depd": "~1.1.2", + "destroy": "~1.0.4", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "~1.7.2", + "mime": "1.6.0", + "ms": "2.1.1", + "on-finished": "~2.3.0", + "range-parser": "~1.2.1", + "statuses": "~1.5.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/send/node_modules/ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" + }, + "node_modules/serve-static": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", + "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", + "dependencies": { + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "0.17.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/setprototypeof": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", + "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" + }, + "node_modules/statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha512-OpZ3zP+jT1PI7I8nemJX4AKmAX070ZkYPVWV/AaKTJl+tXCTGyVdC1a4SL8RUQYEwk/f34ZX8UTykN68FwrqAA==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/toidentifier": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", + "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "dependencies": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/typescript": { + "version": "5.3.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.3.3.tgz", + "integrity": "sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw==", + "dev": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/undici-types": { + "version": "5.26.5", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz", + "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==", + "dev": true + }, + "node_modules/unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", + "engines": { + "node": ">= 0.8" + } + } + } +} diff --git a/src/test/resources/fixtures/npm-build-cache/package.json b/src/test/resources/fixtures/npm-build-cache/package.json new file mode 100644 index 00000000..d31e1192 --- /dev/null +++ b/src/test/resources/fixtures/npm-build-cache/package.json @@ -0,0 +1,23 @@ +{ + "name": "npm-build-cache", + "version": "1.0.0", + "private": true, + "config": { + "dist": "./dist" + }, + "description": "", + "main": "app.js", + "scripts": { + "build": "tsc -p ." + }, + "keywords": [], + "author": "", + "license": "ISC", + "devDependencies": { + "@types/express": "^4.17.1", + "typescript": "^5.3.3" + }, + "dependencies": { + "express": "^4.17.1" + } +} diff --git a/src/test/resources/fixtures/npm-build-cache/settings.gradle.kts b/src/test/resources/fixtures/npm-build-cache/settings.gradle.kts new file mode 100644 index 00000000..90e25101 --- /dev/null +++ b/src/test/resources/fixtures/npm-build-cache/settings.gradle.kts @@ -0,0 +1,50 @@ +/* + * Copyright 2014-2024 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license. + */ + +@file:Suppress("UnstableApiUsage") + +rootProject.name = "npm-build-cache" + +pluginManagement { + repositories { + mavenCentral() + gradlePluginPortal() + } +} + +dependencyResolutionManagement { + repositories { + mavenCentral() + + exclusiveContent { + forRepository { + ivy("https://nodejs.org/dist/") { + name = "Node Distributions at $url" + patternLayout { artifact("v[revision]/[artifact](-v[revision]-[classifier]).[ext]") } + metadataSources { artifact() } + content { includeModule("org.nodejs", "node") } + } + } + filter { includeGroup("org.nodejs") } + } + + exclusiveContent { + forRepository { + ivy("https://github.com/yarnpkg/yarn/releases/download") { + name = "Yarn Distributions at $url" + patternLayout { artifact("v[revision]/[artifact](-v[revision]).[ext]") } + metadataSources { artifact() } + content { includeModule("com.yarnpkg", "yarn") } + } + } + filter { includeGroup("com.yarnpkg") } + } + } +} + +buildCache { + local { + directory = file("build-cache") + } +} diff --git a/src/test/resources/fixtures/npm-build-cache/src/main/app.ts b/src/test/resources/fixtures/npm-build-cache/src/main/app.ts new file mode 100644 index 00000000..bd7c4456 --- /dev/null +++ b/src/test/resources/fixtures/npm-build-cache/src/main/app.ts @@ -0,0 +1,11 @@ +import express from 'express'; +const app = express(); +const port = 3000; + +app.get('/', (req, res) => { + res.send('Hello World!'); +}); + +app.listen(port, () => { + return console.log(`Express is listening at http://localhost:${port}`); +}); diff --git a/src/test/resources/fixtures/npm-build-cache/tsconfig.json b/src/test/resources/fixtures/npm-build-cache/tsconfig.json new file mode 100644 index 00000000..d58a717d --- /dev/null +++ b/src/test/resources/fixtures/npm-build-cache/tsconfig.json @@ -0,0 +1,11 @@ +{ + "compilerOptions": { + "module": "commonjs", + "esModuleInterop": true, + "target": "es6", + "moduleResolution": "node", + "sourceMap": true, + "outDir": "dist" + }, + "lib": ["es2015"] +} From 5e9481feef2327c73581a283ce259aebb3491ace Mon Sep 17 00:00:00 2001 From: Adam Semenenko <152864218+adam-enko@users.noreply.github.com> Date: Wed, 31 Jan 2024 10:16:39 +0100 Subject: [PATCH 2/2] tidy --- .../gradle/node/npm/task/NpmInstallTask.kt | 76 ++++++++++--------- 1 file changed, 39 insertions(+), 37 deletions(-) diff --git a/src/main/kotlin/com/github/gradle/node/npm/task/NpmInstallTask.kt b/src/main/kotlin/com/github/gradle/node/npm/task/NpmInstallTask.kt index 41c0bbc1..aab63e4d 100644 --- a/src/main/kotlin/com/github/gradle/node/npm/task/NpmInstallTask.kt +++ b/src/main/kotlin/com/github/gradle/node/npm/task/NpmInstallTask.kt @@ -64,14 +64,15 @@ abstract class NpmInstallTask : NpmTask() { return projectFileIfExists("yarn.lock").orNull } - // does this overlap with getNodeModulesFiles()? -// @Optional -// @OutputFile -// protected fun getPackageLockFileAsOutput(): File? { -// return npmCommand.flatMap { command -> -// if (command[0] == "install") projectFileIfExists("package-lock.json") else providers.provider { null } -// }.orNull -// } + // does this output overlap with getNodeModulesFiles()? + //@Optional + //@OutputFile + @Internal + protected fun getPackageLockFileAsOutput(): File? { + return npmCommand.flatMap { command -> + if (command[0] == "install") projectFileIfExists("package-lock.json") else providers.provider { null } + }.orNull + } private fun projectFileIfExists(name: String): Provider { return nodeExtension.nodeProjectDir.map { it.file(name).asFile } @@ -106,35 +107,36 @@ abstract class NpmInstallTask : NpmTask() { } } - // does this overlap with getNodeModulesFiles()? -// @Optional -// @OutputFile -// protected fun getNodeModulesPackageLock(): File? { -// if (isLegacyNpm()) { -// return null -// } -// -// return projectFileIfExists("node_modules/.package-lock.json").orNull -// } -// -// /** -// * Is our npm likely to be lower than 7? -// */ -// private fun isLegacyNpm(): Boolean { -// if (nodeExtension.oldNpm.get()) { -// return true -// } -// -// val npmVersion = nodeExtension.npmVersion.get() -// if (npmVersion.isBlank()) { -// if (nodeExtension.version.get().split('.').first().toInt() <= 14) -// return true -// } else if (npmVersion.split('.').first().toInt() < 7 ) { -// return true -// } -// -// return false -// } + // does this output overlap with getNodeModulesFiles()? + @Optional + @OutputFile +// @Internal + protected fun getNodeModulesPackageLock(): File? { + if (isLegacyNpm()) { + return null + } + + return projectFileIfExists("node_modules/.package-lock.json").orNull + } + + /** + * Is our npm likely to be lower than 7? + */ + private fun isLegacyNpm(): Boolean { + if (nodeExtension.oldNpm.get()) { + return true + } + + val npmVersion = nodeExtension.npmVersion.get() + if (npmVersion.isBlank()) { + if (nodeExtension.version.get().split('.').first().toInt() <= 14) + return true + } else if (npmVersion.split('.').first().toInt() < 7 ) { + return true + } + + return false + } // For DSL @Suppress("unused")