Skip to content

Commit

Permalink
feat(node): support node 20 in tasks
Browse files Browse the repository at this point in the history
Addresses aws#566. Use Node 20 using microsoft guidance: https://learn.microsoft.com/en-us/azure/devops/extend/develop/add-build-task?view=azure-devops#q-how-can-i-upgrade-my-custom-task-to-the-latest-node

Keep Node 10 as a fallback. However, we are still building the node10 with esbuild to ensure backwards compatibility. In the future, we can consider building for a higher node version only.

Adding support requires a hacky patch to fix shell.js, which doesn't support bundlers.
Mitigates the issue described in aws#539

Credit to ivanduplenskikh:  ivanduplenskikh#1

Co-authored-by: Ivan Duplenskikh <[email protected]>
  • Loading branch information
hayemaxi and ivanduplenskikh committed Dec 2, 2024
1 parent f4b9ae1 commit 6b999e6
Show file tree
Hide file tree
Showing 26 changed files with 292 additions and 308 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
{
"type": "Feature",
"description": "Tasks now support Node 20"
}
11 changes: 2 additions & 9 deletions .vscode/settings.json
Original file line number Diff line number Diff line change
@@ -1,14 +1,7 @@
{
"files.exclude": {
"build": true,
"package": true,
"package-lock.json": true,
".vscode": true
},
"files.exclude": {},
"search.exclude": {
"build": true,
"package": true,
".vscode": true
"package": true
},
"json.schemas": [
{
Expand Down
84 changes: 62 additions & 22 deletions build-scripts/packageExtension.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,20 +21,54 @@ interface CommandLineOptions {
publisher?: string
}

/**
* Patch shelljs, because it is not compatible with esbuild
* More info: https://github.com/aws/aws-toolkit-azure-devops/pull/539
*/
class ShelljsPatch {
private static readonly shelljsRootPath = path.resolve(process.cwd(), './node_modules/shelljs/')
private static readonly shelljsEntryPath = path.join(ShelljsPatch.shelljsRootPath, 'shell.js')

private originalContent: string | undefined

patch() {
const sourceRequireString =
"require('./commands').forEach(function (command) {\n require('./src/' + command);\n});"
const originalContent = fs.readFileSync(ShelljsPatch.shelljsEntryPath, 'utf-8')

// eslint-disable-next-line @typescript-eslint/no-var-requires
const fixedRequireString = require(path.join(ShelljsPatch.shelljsRootPath, 'commands.js'))
.map((command: string) => `require('./src/${command}.js');`)
.join('\n')

const patchedContent = originalContent.replace(sourceRequireString, fixedRequireString)
if (originalContent === patchedContent) {
throw new Error('Could not patch shelljs, was this npm package updated?')
}

fs.writeFileSync(ShelljsPatch.shelljsEntryPath, patchedContent)

this.originalContent = originalContent
}

unpatch() {
if (this.originalContent) {
fs.writeFileSync(ShelljsPatch.shelljsEntryPath, this.originalContent)
this.originalContent = undefined
}
}
}
const shelljsPatch = new ShelljsPatch()

function findMatchingFiles(directory: string) {
return fs.readdirSync(directory)
}

function installNodePackages(directory: string) {
fs.mkdirpSync(directory)
const npmCmd = `npm install --prefix ${directory} azure-pipelines-task-lib --only=production`
try {
const output = ncp.execSync(npmCmd)
console.log(output.toString('utf8'))
} catch (err) {
console.error(err.output ? err.output.toString() : err?.message)
process.exit(1)
}
const output = ncp.execSync(npmCmd)
console.log(output.toString('utf8'))
}

function generateGitHashFile() {
Expand Down Expand Up @@ -69,6 +103,8 @@ function packagePlugin(options: CommandLineOptions) {
// get required npm packages that will be copied
installNodePackages(npmFolder)

shelljsPatch.patch()

// clean, dedupe and pack each task as needed
findMatchingFiles(folders.sourceTasks).forEach(function(taskName) {
console.log('Processing task ' + taskName)
Expand Down Expand Up @@ -111,22 +147,19 @@ function packagePlugin(options: CommandLineOptions) {
const inputFilename = path.join(taskBuildFolder, taskName + '.runner.js')

console.log('packing node-based task')
try {
const result = esbuild.buildSync({
entryPoints: [inputFilename],
bundle: true,
platform: 'node',
target: ['node10'],
minify: true,
outfile: `${taskPackageFolder}/${taskName}.js`
})
result.warnings.forEach(warning => console.log(warning))
} catch (err) {
console.error(err.output ? err.output.toString() : err.message)
process.exit(1)
}
const result = esbuild.buildSync({
entryPoints: [inputFilename],
bundle: true,
platform: 'node',
target: ['node10'],
minify: true,
outfile: `${taskPackageFolder}/${taskName}.js`
})
result.warnings.forEach(warning => console.log(warning))
})

shelljsPatch.unpatch()

console.log('Creating deployment vsix')

const binName = os.platform() === 'win32' ? `tfx.cmd` : 'tfx'
Expand Down Expand Up @@ -156,5 +189,12 @@ const parsedOptions: CommandLineOptions = {}
if (commandLineInput.length > 0 && commandLineInput[0].split('=')[0] === 'publisher') {
parsedOptions.publisher = commandLineInput[0].split('=')[1]
}
packagePlugin(parsedOptions)

try {
packagePlugin(parsedOptions)
} catch (e) {
shelljsPatch.unpatch()
throw e
}

console.timeEnd(timeMessage)
Loading

0 comments on commit 6b999e6

Please sign in to comment.