mirror of
				https://kkgithub.com/actions/setup-node.git
				synced 2025-11-04 04:31:55 +08:00 
			
		
		
		
	Initial Import
This commit is contained in:
		
							
								
								
									
										234
									
								
								src/installer.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										234
									
								
								src/installer.ts
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,234 @@
 | 
			
		||||
// Load tempDirectory before it gets wiped by tool-cache
 | 
			
		||||
let tempDirectory = process.env['RUNNER_TEMPDIRECTORY'] || '';
 | 
			
		||||
import * as core from '@actions/core';
 | 
			
		||||
import * as io from '@actions/io';
 | 
			
		||||
import * as tc from '@actions/tool-cache';
 | 
			
		||||
import * as restm from 'typed-rest-client/RestClient';
 | 
			
		||||
import * as os from 'os';
 | 
			
		||||
import * as path from 'path';
 | 
			
		||||
import * as semver from 'semver';
 | 
			
		||||
 | 
			
		||||
let osPlat: string = os.platform();
 | 
			
		||||
let osArch: string = 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');
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
//
 | 
			
		||||
// 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':
 | 
			
		||||
      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: string[] = [];
 | 
			
		||||
  let dataUrl = 'https://nodejs.org/dist/index.json';
 | 
			
		||||
  let rest: restm.RestClient = new restm.RestClient('vsts-node-tool');
 | 
			
		||||
  let nodeVersions: INodeVersion[] =
 | 
			
		||||
    (await rest.get<INodeVersion[]>(dataUrl)).result || [];
 | 
			
		||||
  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'
 | 
			
		||||
      ? 'node-v' + version + '-win-' + os.arch()
 | 
			
		||||
      : 'node-v' + version + '-' + osPlat + '-' + os.arch();
 | 
			
		||||
  let urlFileName: string =
 | 
			
		||||
    osPlat == 'win32' ? fileName + '.7z' : fileName + '.tar.gz';
 | 
			
		||||
 | 
			
		||||
  let downloadUrl = 'https://nodejs.org/dist/v' + version + '/' + urlFileName;
 | 
			
		||||
 | 
			
		||||
  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 {
 | 
			
		||||
    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 = await tc.downloadTool(exeUrl);
 | 
			
		||||
    await io.mv(exePath, path.join(tempDir, 'node.exe'));
 | 
			
		||||
    const libPath = await tc.downloadTool(libUrl);
 | 
			
		||||
    await 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 = await tc.downloadTool(exeUrl);
 | 
			
		||||
      await io.mv(exePath, path.join(tempDir, 'node.exe'));
 | 
			
		||||
      const libPath = await tc.downloadTool(libUrl);
 | 
			
		||||
      await io.mv(libPath, path.join(tempDir, 'node.lib'));
 | 
			
		||||
    } else {
 | 
			
		||||
      throw err;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  return await tc.cacheDir(tempDir, 'node', version);
 | 
			
		||||
}
 | 
			
		||||
		Reference in New Issue
	
	Block a user