| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  | // Load tempDirectory before it gets wiped by tool-cache
 | 
					
						
							|  |  |  | let tempDirectory = process.env['RUNNER_TEMPDIRECTORY'] || ''; | 
					
						
							| 
									
										
										
										
											2020-01-27 10:37:12 -05:00
										 |  |  | import * as assert from 'assert'; | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  | import * as core from '@actions/core'; | 
					
						
							| 
									
										
										
										
											2020-01-27 10:37:12 -05:00
										 |  |  | import * as hc from '@actions/http-client'; | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  | import * as io from '@actions/io'; | 
					
						
							|  |  |  | import * as tc from '@actions/tool-cache'; | 
					
						
							|  |  |  | import * as os from 'os'; | 
					
						
							|  |  |  | import * as path from 'path'; | 
					
						
							|  |  |  | import * as semver from 'semver'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | let osPlat: string = os.platform(); | 
					
						
							| 
									
										
										
										
											2019-11-20 12:21:56 -05:00
										 |  |  | let osArch: string = translateArchToDistUrl(os.arch()); | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | if (!tempDirectory) { | 
					
						
							|  |  |  |   let baseLocation; | 
					
						
							|  |  |  |   if (process.platform === 'win32') { | 
					
						
							|  |  |  |     // On windows use the USERPROFILE env variable
 | 
					
						
							|  |  |  |     baseLocation = process.env['USERPROFILE'] || 'C:\\'; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     if (process.platform === 'darwin') { | 
					
						
							|  |  |  |       baseLocation = '/Users'; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       baseLocation = '/home'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   tempDirectory = path.join(baseLocation, 'actions', 'temp'); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Node versions interface
 | 
					
						
							|  |  |  | // see https://nodejs.org/dist/index.json
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | interface INodeVersion { | 
					
						
							|  |  |  |   version: string; | 
					
						
							|  |  |  |   files: string[]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export async function getNode(versionSpec: string) { | 
					
						
							|  |  |  |   // check cache
 | 
					
						
							|  |  |  |   let toolPath: string; | 
					
						
							|  |  |  |   toolPath = tc.find('node', versionSpec); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // If not found in cache, download
 | 
					
						
							|  |  |  |   if (!toolPath) { | 
					
						
							|  |  |  |     let version: string; | 
					
						
							|  |  |  |     const c = semver.clean(versionSpec) || ''; | 
					
						
							|  |  |  |     // If explicit version
 | 
					
						
							|  |  |  |     if (semver.valid(c) != null) { | 
					
						
							|  |  |  |       // version to download
 | 
					
						
							|  |  |  |       version = versionSpec; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       // query nodejs.org for a matching version
 | 
					
						
							|  |  |  |       version = await queryLatestMatch(versionSpec); | 
					
						
							|  |  |  |       if (!version) { | 
					
						
							|  |  |  |         throw new Error( | 
					
						
							|  |  |  |           `Unable to find Node version '${versionSpec}' for platform ${osPlat} and architecture ${osArch}.` | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // check cache
 | 
					
						
							|  |  |  |       toolPath = tc.find('node', version); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!toolPath) { | 
					
						
							|  |  |  |       // download, extract, cache
 | 
					
						
							|  |  |  |       toolPath = await acquireNode(version); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   // a tool installer initimately knows details about the layout of that tool
 | 
					
						
							|  |  |  |   // for example, node binary is in the bin folder after the extract on Mac/Linux.
 | 
					
						
							|  |  |  |   // layouts could change by version, by platform etc... but that's the tool installers job
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   if (osPlat != 'win32') { | 
					
						
							|  |  |  |     toolPath = path.join(toolPath, 'bin'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   // prepend the tools path. instructs the agent to prepend for future tasks
 | 
					
						
							|  |  |  |   core.addPath(toolPath); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function queryLatestMatch(versionSpec: string): Promise<string> { | 
					
						
							|  |  |  |   // node offers a json list of versions
 | 
					
						
							|  |  |  |   let dataFileName: string; | 
					
						
							|  |  |  |   switch (osPlat) { | 
					
						
							|  |  |  |     case 'linux': | 
					
						
							| 
									
										
										
										
											2019-11-20 12:21:56 -05:00
										 |  |  |       dataFileName = `linux-${osArch}`; | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case 'darwin': | 
					
						
							| 
									
										
										
										
											2019-11-20 12:21:56 -05:00
										 |  |  |       dataFileName = `osx-${osArch}-tar`; | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case 'win32': | 
					
						
							| 
									
										
										
										
											2019-11-20 12:21:56 -05:00
										 |  |  |       dataFileName = `win-${osArch}-exe`; | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  |       break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       throw new Error(`Unexpected OS '${osPlat}'`); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let versions: string[] = []; | 
					
						
							|  |  |  |   let dataUrl = 'https://nodejs.org/dist/index.json'; | 
					
						
							| 
									
										
										
										
											2020-01-27 10:37:12 -05:00
										 |  |  |   let httpClient = new hc.HttpClient('setup-node', [], { | 
					
						
							|  |  |  |     allowRetries: true, | 
					
						
							|  |  |  |     maxRetries: 3 | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2020-02-05 11:20:33 -05:00
										 |  |  |   let response = await httpClient.getJson<INodeVersion[]>(dataUrl); | 
					
						
							|  |  |  |   let nodeVersions = response.result || []; | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  |   nodeVersions.forEach((nodeVersion: INodeVersion) => { | 
					
						
							|  |  |  |     // ensure this version supports your os and platform
 | 
					
						
							|  |  |  |     if (nodeVersion.files.indexOf(dataFileName) >= 0) { | 
					
						
							|  |  |  |       versions.push(nodeVersion.version); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // get the latest version that matches the version spec
 | 
					
						
							|  |  |  |   let version: string = evaluateVersions(versions, versionSpec); | 
					
						
							|  |  |  |   return version; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // TODO - should we just export this from @actions/tool-cache? Lifted directly from there
 | 
					
						
							|  |  |  | function evaluateVersions(versions: string[], versionSpec: string): string { | 
					
						
							|  |  |  |   let version = ''; | 
					
						
							|  |  |  |   core.debug(`evaluating ${versions.length} versions`); | 
					
						
							|  |  |  |   versions = versions.sort((a, b) => { | 
					
						
							|  |  |  |     if (semver.gt(a, b)) { | 
					
						
							|  |  |  |       return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   for (let i = versions.length - 1; i >= 0; i--) { | 
					
						
							|  |  |  |     const potential: string = versions[i]; | 
					
						
							|  |  |  |     const satisfied: boolean = semver.satisfies(potential, versionSpec); | 
					
						
							|  |  |  |     if (satisfied) { | 
					
						
							|  |  |  |       version = potential; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (version) { | 
					
						
							|  |  |  |     core.debug(`matched: ${version}`); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     core.debug('match not found'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return version; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function acquireNode(version: string): Promise<string> { | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   // Download - a tool installer intimately knows how to get the tool (and construct urls)
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   version = semver.clean(version) || ''; | 
					
						
							|  |  |  |   let fileName: string = | 
					
						
							|  |  |  |     osPlat == 'win32' | 
					
						
							| 
									
										
										
										
											2019-11-20 12:21:56 -05:00
										 |  |  |       ? `node-v${version}-win-${osArch}` | 
					
						
							|  |  |  |       : `node-v${version}-${osPlat}-${osArch}`; | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  |   let urlFileName: string = | 
					
						
							| 
									
										
										
										
											2019-11-20 12:21:56 -05:00
										 |  |  |     osPlat == 'win32' ? `${fileName}.7z` : `${fileName}.tar.gz`; | 
					
						
							|  |  |  |   let downloadUrl = `https://nodejs.org/dist/v${version}/${urlFileName}`; | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   let downloadPath: string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   try { | 
					
						
							|  |  |  |     downloadPath = await tc.downloadTool(downloadUrl); | 
					
						
							|  |  |  |   } catch (err) { | 
					
						
							|  |  |  |     if (err instanceof tc.HTTPError && err.httpStatusCode == 404) { | 
					
						
							|  |  |  |       return await acquireNodeFromFallbackLocation(version); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     throw err; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   // Extract
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   let extPath: string; | 
					
						
							|  |  |  |   if (osPlat == 'win32') { | 
					
						
							|  |  |  |     let _7zPath = path.join(__dirname, '..', 'externals', '7zr.exe'); | 
					
						
							|  |  |  |     extPath = await tc.extract7z(downloadPath, undefined, _7zPath); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     extPath = await tc.extractTar(downloadPath); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   // Install into the local tool cache - node extracts with a root folder that matches the fileName downloaded
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   let toolRoot = path.join(extPath, fileName); | 
					
						
							|  |  |  |   return await tc.cacheDir(toolRoot, 'node', version); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // For non LTS versions of Node, the files we need (for Windows) are sometimes located
 | 
					
						
							|  |  |  | // in a different folder than they normally are for other versions.
 | 
					
						
							|  |  |  | // Normally the format is similar to: https://nodejs.org/dist/v5.10.1/node-v5.10.1-win-x64.7z
 | 
					
						
							|  |  |  | // In this case, there will be two files located at:
 | 
					
						
							|  |  |  | //      /dist/v5.10.1/win-x64/node.exe
 | 
					
						
							|  |  |  | //      /dist/v5.10.1/win-x64/node.lib
 | 
					
						
							|  |  |  | // If this is not the structure, there may also be two files located at:
 | 
					
						
							|  |  |  | //      /dist/v0.12.18/node.exe
 | 
					
						
							|  |  |  | //      /dist/v0.12.18/node.lib
 | 
					
						
							|  |  |  | // This method attempts to download and cache the resources from these alternative locations.
 | 
					
						
							|  |  |  | // Note also that the files are normally zipped but in this case they are just an exe
 | 
					
						
							|  |  |  | // and lib file in a folder, not zipped.
 | 
					
						
							|  |  |  | async function acquireNodeFromFallbackLocation( | 
					
						
							|  |  |  |   version: string | 
					
						
							|  |  |  | ): Promise<string> { | 
					
						
							|  |  |  |   // Create temporary folder to download in to
 | 
					
						
							|  |  |  |   let tempDownloadFolder: string = | 
					
						
							|  |  |  |     'temp_' + Math.floor(Math.random() * 2000000000); | 
					
						
							|  |  |  |   let tempDir: string = path.join(tempDirectory, tempDownloadFolder); | 
					
						
							|  |  |  |   await io.mkdirP(tempDir); | 
					
						
							|  |  |  |   let exeUrl: string; | 
					
						
							|  |  |  |   let libUrl: string; | 
					
						
							|  |  |  |   try { | 
					
						
							| 
									
										
										
										
											2019-11-20 12:21:56 -05:00
										 |  |  |     exeUrl = `https://nodejs.org/dist/v${version}/win-${osArch}/node.exe`; | 
					
						
							|  |  |  |     libUrl = `https://nodejs.org/dist/v${version}/win-${osArch}/node.lib`; | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const exePath = await tc.downloadTool(exeUrl); | 
					
						
							| 
									
										
										
										
											2019-08-05 22:23:46 -04:00
										 |  |  |     await io.cp(exePath, path.join(tempDir, 'node.exe')); | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  |     const libPath = await tc.downloadTool(libUrl); | 
					
						
							| 
									
										
										
										
											2019-08-05 22:23:46 -04:00
										 |  |  |     await io.cp(libPath, path.join(tempDir, 'node.lib')); | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  |   } catch (err) { | 
					
						
							|  |  |  |     if (err instanceof tc.HTTPError && err.httpStatusCode == 404) { | 
					
						
							|  |  |  |       exeUrl = `https://nodejs.org/dist/v${version}/node.exe`; | 
					
						
							|  |  |  |       libUrl = `https://nodejs.org/dist/v${version}/node.lib`; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const exePath = await tc.downloadTool(exeUrl); | 
					
						
							| 
									
										
										
										
											2019-08-05 22:23:46 -04:00
										 |  |  |       await io.cp(exePath, path.join(tempDir, 'node.exe')); | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  |       const libPath = await tc.downloadTool(libUrl); | 
					
						
							| 
									
										
										
										
											2019-08-05 22:23:46 -04:00
										 |  |  |       await io.cp(libPath, path.join(tempDir, 'node.lib')); | 
					
						
							| 
									
										
										
										
											2019-08-03 21:49:54 -04:00
										 |  |  |     } else { | 
					
						
							|  |  |  |       throw err; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return await tc.cacheDir(tempDir, 'node', version); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-11-20 12:21:56 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | // os.arch does not always match the relative download url, e.g.
 | 
					
						
							|  |  |  | // os.arch == 'arm' != node-v12.13.1-linux-armv7l.tar.gz
 | 
					
						
							| 
									
										
										
										
											2019-11-20 15:30:44 -05:00
										 |  |  | // All other currently supported architectures match, e.g.:
 | 
					
						
							|  |  |  | //   os.arch = arm64 => https://nodejs.org/dist/v{VERSION}/node-v{VERSION}-{OS}-arm64.tar.gz
 | 
					
						
							|  |  |  | //   os.arch = x64 => https://nodejs.org/dist/v{VERSION}/node-v{VERSION}-{OS}-x64.tar.gz
 | 
					
						
							| 
									
										
										
										
											2019-11-20 12:21:56 -05:00
										 |  |  | function translateArchToDistUrl(arch: string): string { | 
					
						
							|  |  |  |   switch (arch) { | 
					
						
							|  |  |  |     case 'arm': | 
					
						
							|  |  |  |       return 'armv7l'; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       return arch; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |