228 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			228 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
|  | "use strict"; | ||
|  | var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { | ||
|  |     return new (P || (P = Promise))(function (resolve, reject) { | ||
|  |         function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } | ||
|  |         function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } | ||
|  |         function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); } | ||
|  |         step((generator = generator.apply(thisArg, _arguments || [])).next()); | ||
|  |     }); | ||
|  | }; | ||
|  | var __importStar = (this && this.__importStar) || function (mod) { | ||
|  |     if (mod && mod.__esModule) return mod; | ||
|  |     var result = {}; | ||
|  |     if (mod != null) for (var k in mod) if (Object.hasOwnProperty.call(mod, k)) result[k] = mod[k]; | ||
|  |     result["default"] = mod; | ||
|  |     return result; | ||
|  | }; | ||
|  | Object.defineProperty(exports, "__esModule", { value: true }); | ||
|  | // Load tempDirectory before it gets wiped by tool-cache
 | ||
|  | let tempDirectory = process.env['RUNNER_TEMPDIRECTORY'] || ''; | ||
|  | const core = __importStar(require("@actions/core")); | ||
|  | const io = __importStar(require("@actions/io")); | ||
|  | const tc = __importStar(require("@actions/tool-cache")); | ||
|  | const restm = __importStar(require("typed-rest-client/RestClient")); | ||
|  | const os = __importStar(require("os")); | ||
|  | const path = __importStar(require("path")); | ||
|  | const semver = __importStar(require("semver")); | ||
|  | let osPlat = os.platform(); | ||
|  | let osArch = os.arch(); | ||
|  | 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'); | ||
|  | } | ||
|  | function getNode(versionSpec) { | ||
|  |     return __awaiter(this, void 0, void 0, function* () { | ||
|  |         // check cache
 | ||
|  |         let toolPath; | ||
|  |         toolPath = tc.find('node', versionSpec); | ||
|  |         // If not found in cache, download
 | ||
|  |         if (!toolPath) { | ||
|  |             let version; | ||
|  |             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 = yield 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 = yield 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); | ||
|  |     }); | ||
|  | } | ||
|  | exports.getNode = getNode; | ||
|  | function queryLatestMatch(versionSpec) { | ||
|  |     return __awaiter(this, void 0, void 0, function* () { | ||
|  |         // node offers a json list of versions
 | ||
|  |         let dataFileName; | ||
|  |         switch (osPlat) { | ||
|  |             case 'linux': | ||
|  |                 dataFileName = 'linux-' + osArch; | ||
|  |                 break; | ||
|  |             case 'darwin': | ||
|  |                 dataFileName = 'osx-' + osArch + '-tar'; | ||
|  |                 break; | ||
|  |             case 'win32': | ||
|  |                 dataFileName = 'win-' + osArch + '-exe'; | ||
|  |                 break; | ||
|  |             default: | ||
|  |                 throw new Error(`Unexpected OS '${osPlat}'`); | ||
|  |         } | ||
|  |         let versions = []; | ||
|  |         let dataUrl = 'https://nodejs.org/dist/index.json'; | ||
|  |         let rest = new restm.RestClient('vsts-node-tool'); | ||
|  |         let nodeVersions = (yield rest.get(dataUrl)).result || []; | ||
|  |         nodeVersions.forEach((nodeVersion) => { | ||
|  |             // 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 = evaluateVersions(versions, versionSpec); | ||
|  |         return version; | ||
|  |     }); | ||
|  | } | ||
|  | // TODO - should we just export this from @actions/tool-cache? Lifted directly from there
 | ||
|  | function evaluateVersions(versions, versionSpec) { | ||
|  |     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 = versions[i]; | ||
|  |         const satisfied = semver.satisfies(potential, versionSpec); | ||
|  |         if (satisfied) { | ||
|  |             version = potential; | ||
|  |             break; | ||
|  |         } | ||
|  |     } | ||
|  |     if (version) { | ||
|  |         core.debug(`matched: ${version}`); | ||
|  |     } | ||
|  |     else { | ||
|  |         core.debug('match not found'); | ||
|  |     } | ||
|  |     return version; | ||
|  | } | ||
|  | function acquireNode(version) { | ||
|  |     return __awaiter(this, void 0, void 0, function* () { | ||
|  |         //
 | ||
|  |         // Download - a tool installer intimately knows how to get the tool (and construct urls)
 | ||
|  |         //
 | ||
|  |         version = semver.clean(version) || ''; | ||
|  |         let fileName = osPlat == 'win32' | ||
|  |             ? 'node-v' + version + '-win-' + os.arch() | ||
|  |             : 'node-v' + version + '-' + osPlat + '-' + os.arch(); | ||
|  |         let urlFileName = osPlat == 'win32' ? fileName + '.7z' : fileName + '.tar.gz'; | ||
|  |         let downloadUrl = 'https://nodejs.org/dist/v' + version + '/' + urlFileName; | ||
|  |         let downloadPath; | ||
|  |         try { | ||
|  |             downloadPath = yield tc.downloadTool(downloadUrl); | ||
|  |         } | ||
|  |         catch (err) { | ||
|  |             if (err instanceof tc.HTTPError && err.httpStatusCode == 404) { | ||
|  |                 return yield acquireNodeFromFallbackLocation(version); | ||
|  |             } | ||
|  |             throw err; | ||
|  |         } | ||
|  |         //
 | ||
|  |         // Extract
 | ||
|  |         //
 | ||
|  |         let extPath; | ||
|  |         if (osPlat == 'win32') { | ||
|  |             let _7zPath = path.join(__dirname, '..', 'externals', '7zr.exe'); | ||
|  |             extPath = yield tc.extract7z(downloadPath, undefined, _7zPath); | ||
|  |         } | ||
|  |         else { | ||
|  |             extPath = yield 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 yield 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.
 | ||
|  | function acquireNodeFromFallbackLocation(version) { | ||
|  |     return __awaiter(this, void 0, void 0, function* () { | ||
|  |         // Create temporary folder to download in to
 | ||
|  |         let tempDownloadFolder = 'temp_' + Math.floor(Math.random() * 2000000000); | ||
|  |         let tempDir = path.join(tempDirectory, tempDownloadFolder); | ||
|  |         yield io.mkdirP(tempDir); | ||
|  |         let exeUrl; | ||
|  |         let libUrl; | ||
|  |         try { | ||
|  |             exeUrl = `https://nodejs.org/dist/v${version}/win-${os.arch()}/node.exe`; | ||
|  |             libUrl = `https://nodejs.org/dist/v${version}/win-${os.arch()}/node.lib`; | ||
|  |             const exePath = yield tc.downloadTool(exeUrl); | ||
|  |             yield io.mv(exePath, path.join(tempDir, 'node.exe')); | ||
|  |             const libPath = yield tc.downloadTool(libUrl); | ||
|  |             yield io.mv(libPath, path.join(tempDir, 'node.lib')); | ||
|  |         } | ||
|  |         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 = yield tc.downloadTool(exeUrl); | ||
|  |                 yield io.mv(exePath, path.join(tempDir, 'node.exe')); | ||
|  |                 const libPath = yield tc.downloadTool(libUrl); | ||
|  |                 yield io.mv(libPath, path.join(tempDir, 'node.lib')); | ||
|  |             } | ||
|  |             else { | ||
|  |                 throw err; | ||
|  |             } | ||
|  |         } | ||
|  |         return yield tc.cacheDir(tempDir, 'node', version); | ||
|  |     }); | ||
|  | } |