mirror of
				https://github.com/actions/setup-node.git
				synced 2025-11-04 04:45:28 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			436 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			436 lines
		
	
	
		
			17 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());
 | 
						|
    });
 | 
						|
};
 | 
						|
Object.defineProperty(exports, "__esModule", { value: true });
 | 
						|
const core = require("@actions/core");
 | 
						|
const io = require("@actions/io");
 | 
						|
const fs = require("fs");
 | 
						|
const os = require("os");
 | 
						|
const path = require("path");
 | 
						|
const httpm = require("typed-rest-client/HttpClient");
 | 
						|
const semver = require("semver");
 | 
						|
const uuidV4 = require("uuid/v4");
 | 
						|
const exec_1 = require("@actions/exec/lib/exec");
 | 
						|
const assert_1 = require("assert");
 | 
						|
class HTTPError extends Error {
 | 
						|
    constructor(httpStatusCode) {
 | 
						|
        super(`Unexpected HTTP response: ${httpStatusCode}`);
 | 
						|
        this.httpStatusCode = httpStatusCode;
 | 
						|
        Object.setPrototypeOf(this, new.target.prototype);
 | 
						|
    }
 | 
						|
}
 | 
						|
exports.HTTPError = HTTPError;
 | 
						|
const IS_WINDOWS = process.platform === 'win32';
 | 
						|
const userAgent = 'actions/tool-cache';
 | 
						|
// On load grab temp directory and cache directory and remove them from env (currently don't want to expose this)
 | 
						|
let tempDirectory = process.env['RUNNER_TEMP'] || '';
 | 
						|
let cacheRoot = process.env['RUNNER_TOOL_CACHE'] || '';
 | 
						|
// If directories not found, place them in common temp locations
 | 
						|
if (!tempDirectory || !cacheRoot) {
 | 
						|
    let baseLocation;
 | 
						|
    if (IS_WINDOWS) {
 | 
						|
        // On windows use the USERPROFILE env variable
 | 
						|
        baseLocation = process.env['USERPROFILE'] || 'C:\\';
 | 
						|
    }
 | 
						|
    else {
 | 
						|
        if (process.platform === 'darwin') {
 | 
						|
            baseLocation = '/Users';
 | 
						|
        }
 | 
						|
        else {
 | 
						|
            baseLocation = '/home';
 | 
						|
        }
 | 
						|
    }
 | 
						|
    if (!tempDirectory) {
 | 
						|
        tempDirectory = path.join(baseLocation, 'actions', 'temp');
 | 
						|
    }
 | 
						|
    if (!cacheRoot) {
 | 
						|
        cacheRoot = path.join(baseLocation, 'actions', 'cache');
 | 
						|
    }
 | 
						|
}
 | 
						|
/**
 | 
						|
 * Download a tool from an url and stream it into a file
 | 
						|
 *
 | 
						|
 * @param url       url of tool to download
 | 
						|
 * @returns         path to downloaded tool
 | 
						|
 */
 | 
						|
function downloadTool(url) {
 | 
						|
    return __awaiter(this, void 0, void 0, function* () {
 | 
						|
        // Wrap in a promise so that we can resolve from within stream callbacks
 | 
						|
        return new Promise((resolve, reject) => __awaiter(this, void 0, void 0, function* () {
 | 
						|
            try {
 | 
						|
                const http = new httpm.HttpClient(userAgent, [], {
 | 
						|
                    allowRetries: true,
 | 
						|
                    maxRetries: 3
 | 
						|
                });
 | 
						|
                const destPath = path.join(tempDirectory, uuidV4());
 | 
						|
                yield io.mkdirP(tempDirectory);
 | 
						|
                core.debug(`Downloading ${url}`);
 | 
						|
                core.debug(`Downloading ${destPath}`);
 | 
						|
                if (fs.existsSync(destPath)) {
 | 
						|
                    throw new Error(`Destination file path ${destPath} already exists`);
 | 
						|
                }
 | 
						|
                const response = yield http.get(url);
 | 
						|
                if (response.message.statusCode !== 200) {
 | 
						|
                    const err = new HTTPError(response.message.statusCode);
 | 
						|
                    core.debug(`Failed to download from "${url}". Code(${response.message.statusCode}) Message(${response.message.statusMessage})`);
 | 
						|
                    throw err;
 | 
						|
                }
 | 
						|
                const file = fs.createWriteStream(destPath);
 | 
						|
                file.on('open', () => __awaiter(this, void 0, void 0, function* () {
 | 
						|
                    try {
 | 
						|
                        const stream = response.message.pipe(file);
 | 
						|
                        stream.on('close', () => {
 | 
						|
                            core.debug('download complete');
 | 
						|
                            resolve(destPath);
 | 
						|
                        });
 | 
						|
                    }
 | 
						|
                    catch (err) {
 | 
						|
                        core.debug(`Failed to download from "${url}". Code(${response.message.statusCode}) Message(${response.message.statusMessage})`);
 | 
						|
                        reject(err);
 | 
						|
                    }
 | 
						|
                }));
 | 
						|
                file.on('error', err => {
 | 
						|
                    file.end();
 | 
						|
                    reject(err);
 | 
						|
                });
 | 
						|
            }
 | 
						|
            catch (err) {
 | 
						|
                reject(err);
 | 
						|
            }
 | 
						|
        }));
 | 
						|
    });
 | 
						|
}
 | 
						|
exports.downloadTool = downloadTool;
 | 
						|
/**
 | 
						|
 * Extract a .7z file
 | 
						|
 *
 | 
						|
 * @param file     path to the .7z file
 | 
						|
 * @param dest     destination directory. Optional.
 | 
						|
 * @param _7zPath  path to 7zr.exe. Optional, for long path support. Most .7z archives do not have this
 | 
						|
 * problem. If your .7z archive contains very long paths, you can pass the path to 7zr.exe which will
 | 
						|
 * gracefully handle long paths. By default 7zdec.exe is used because it is a very small program and is
 | 
						|
 * bundled with the tool lib. However it does not support long paths. 7zr.exe is the reduced command line
 | 
						|
 * interface, it is smaller than the full command line interface, and it does support long paths. At the
 | 
						|
 * time of this writing, it is freely available from the LZMA SDK that is available on the 7zip website.
 | 
						|
 * Be sure to check the current license agreement. If 7zr.exe is bundled with your action, then the path
 | 
						|
 * to 7zr.exe can be pass to this function.
 | 
						|
 * @returns        path to the destination directory
 | 
						|
 */
 | 
						|
function extract7z(file, dest, _7zPath) {
 | 
						|
    return __awaiter(this, void 0, void 0, function* () {
 | 
						|
        assert_1.ok(IS_WINDOWS, 'extract7z() not supported on current OS');
 | 
						|
        assert_1.ok(file, 'parameter "file" is required');
 | 
						|
        dest = dest || (yield _createExtractFolder(dest));
 | 
						|
        const originalCwd = process.cwd();
 | 
						|
        process.chdir(dest);
 | 
						|
        if (_7zPath) {
 | 
						|
            try {
 | 
						|
                const args = [
 | 
						|
                    'x',
 | 
						|
                    '-bb1',
 | 
						|
                    '-bd',
 | 
						|
                    '-sccUTF-8',
 | 
						|
                    file
 | 
						|
                ];
 | 
						|
                const options = {
 | 
						|
                    silent: true
 | 
						|
                };
 | 
						|
                yield exec_1.exec(`"${_7zPath}"`, args, options);
 | 
						|
            }
 | 
						|
            finally {
 | 
						|
                process.chdir(originalCwd);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        else {
 | 
						|
            const escapedScript = path
 | 
						|
                .join(__dirname, '..', 'scripts', 'Invoke-7zdec.ps1')
 | 
						|
                .replace(/'/g, "''")
 | 
						|
                .replace(/"|\n|\r/g, ''); // double-up single quotes, remove double quotes and newlines
 | 
						|
            const escapedFile = file.replace(/'/g, "''").replace(/"|\n|\r/g, '');
 | 
						|
            const escapedTarget = dest.replace(/'/g, "''").replace(/"|\n|\r/g, '');
 | 
						|
            const command = `& '${escapedScript}' -Source '${escapedFile}' -Target '${escapedTarget}'`;
 | 
						|
            const args = [
 | 
						|
                '-NoLogo',
 | 
						|
                '-Sta',
 | 
						|
                '-NoProfile',
 | 
						|
                '-NonInteractive',
 | 
						|
                '-ExecutionPolicy',
 | 
						|
                'Unrestricted',
 | 
						|
                '-Command',
 | 
						|
                command
 | 
						|
            ];
 | 
						|
            const options = {
 | 
						|
                silent: true
 | 
						|
            };
 | 
						|
            try {
 | 
						|
                const powershellPath = yield io.which('powershell', true);
 | 
						|
                yield exec_1.exec(`"${powershellPath}"`, args, options);
 | 
						|
            }
 | 
						|
            finally {
 | 
						|
                process.chdir(originalCwd);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        return dest;
 | 
						|
    });
 | 
						|
}
 | 
						|
exports.extract7z = extract7z;
 | 
						|
/**
 | 
						|
 * Extract a tar
 | 
						|
 *
 | 
						|
 * @param file     path to the tar
 | 
						|
 * @param dest     destination directory. Optional.
 | 
						|
 * @returns        path to the destination directory
 | 
						|
 */
 | 
						|
function extractTar(file, dest) {
 | 
						|
    return __awaiter(this, void 0, void 0, function* () {
 | 
						|
        if (!file) {
 | 
						|
            throw new Error("parameter 'file' is required");
 | 
						|
        }
 | 
						|
        dest = dest || (yield _createExtractFolder(dest));
 | 
						|
        const tarPath = yield io.which('tar', true);
 | 
						|
        yield exec_1.exec(`"${tarPath}"`, ['xzC', dest, '-f', file]);
 | 
						|
        return dest;
 | 
						|
    });
 | 
						|
}
 | 
						|
exports.extractTar = extractTar;
 | 
						|
/**
 | 
						|
 * Extract a zip
 | 
						|
 *
 | 
						|
 * @param file     path to the zip
 | 
						|
 * @param dest     destination directory. Optional.
 | 
						|
 * @returns        path to the destination directory
 | 
						|
 */
 | 
						|
function extractZip(file, dest) {
 | 
						|
    return __awaiter(this, void 0, void 0, function* () {
 | 
						|
        if (!file) {
 | 
						|
            throw new Error("parameter 'file' is required");
 | 
						|
        }
 | 
						|
        dest = dest || (yield _createExtractFolder(dest));
 | 
						|
        if (IS_WINDOWS) {
 | 
						|
            yield extractZipWin(file, dest);
 | 
						|
        }
 | 
						|
        else {
 | 
						|
            yield extractZipNix(file, dest);
 | 
						|
        }
 | 
						|
        return dest;
 | 
						|
    });
 | 
						|
}
 | 
						|
exports.extractZip = extractZip;
 | 
						|
function extractZipWin(file, dest) {
 | 
						|
    return __awaiter(this, void 0, void 0, function* () {
 | 
						|
        // build the powershell command
 | 
						|
        const escapedFile = file.replace(/'/g, "''").replace(/"|\n|\r/g, ''); // double-up single quotes, remove double quotes and newlines
 | 
						|
        const escapedDest = dest.replace(/'/g, "''").replace(/"|\n|\r/g, '');
 | 
						|
        const command = `$ErrorActionPreference = 'Stop' ; try { Add-Type -AssemblyName System.IO.Compression.FileSystem } catch { } ; [System.IO.Compression.ZipFile]::ExtractToDirectory('${escapedFile}', '${escapedDest}')`;
 | 
						|
        // run powershell
 | 
						|
        const powershellPath = yield io.which('powershell');
 | 
						|
        const args = [
 | 
						|
            '-NoLogo',
 | 
						|
            '-Sta',
 | 
						|
            '-NoProfile',
 | 
						|
            '-NonInteractive',
 | 
						|
            '-ExecutionPolicy',
 | 
						|
            'Unrestricted',
 | 
						|
            '-Command',
 | 
						|
            command
 | 
						|
        ];
 | 
						|
        yield exec_1.exec(`"${powershellPath}"`, args);
 | 
						|
    });
 | 
						|
}
 | 
						|
function extractZipNix(file, dest) {
 | 
						|
    return __awaiter(this, void 0, void 0, function* () {
 | 
						|
        const unzipPath = path.join(__dirname, '..', 'scripts', 'externals', 'unzip');
 | 
						|
        yield exec_1.exec(`"${unzipPath}"`, [file], { cwd: dest });
 | 
						|
    });
 | 
						|
}
 | 
						|
/**
 | 
						|
 * Caches a directory and installs it into the tool cacheDir
 | 
						|
 *
 | 
						|
 * @param sourceDir    the directory to cache into tools
 | 
						|
 * @param tool          tool name
 | 
						|
 * @param version       version of the tool.  semver format
 | 
						|
 * @param arch          architecture of the tool.  Optional.  Defaults to machine architecture
 | 
						|
 */
 | 
						|
function cacheDir(sourceDir, tool, version, arch) {
 | 
						|
    return __awaiter(this, void 0, void 0, function* () {
 | 
						|
        version = semver.clean(version) || version;
 | 
						|
        arch = arch || os.arch();
 | 
						|
        core.debug(`Caching tool ${tool} ${version} ${arch}`);
 | 
						|
        core.debug(`source dir: ${sourceDir}`);
 | 
						|
        if (!fs.statSync(sourceDir).isDirectory()) {
 | 
						|
            throw new Error('sourceDir is not a directory');
 | 
						|
        }
 | 
						|
        // Create the tool dir
 | 
						|
        const destPath = yield _createToolPath(tool, version, arch);
 | 
						|
        // copy each child item. do not move. move can fail on Windows
 | 
						|
        // due to anti-virus software having an open handle on a file.
 | 
						|
        for (const itemName of fs.readdirSync(sourceDir)) {
 | 
						|
            const s = path.join(sourceDir, itemName);
 | 
						|
            yield io.cp(s, destPath, { recursive: true });
 | 
						|
        }
 | 
						|
        // write .complete
 | 
						|
        _completeToolPath(tool, version, arch);
 | 
						|
        return destPath;
 | 
						|
    });
 | 
						|
}
 | 
						|
exports.cacheDir = cacheDir;
 | 
						|
/**
 | 
						|
 * Caches a downloaded file (GUID) and installs it
 | 
						|
 * into the tool cache with a given targetName
 | 
						|
 *
 | 
						|
 * @param sourceFile    the file to cache into tools.  Typically a result of downloadTool which is a guid.
 | 
						|
 * @param targetFile    the name of the file name in the tools directory
 | 
						|
 * @param tool          tool name
 | 
						|
 * @param version       version of the tool.  semver format
 | 
						|
 * @param arch          architecture of the tool.  Optional.  Defaults to machine architecture
 | 
						|
 */
 | 
						|
function cacheFile(sourceFile, targetFile, tool, version, arch) {
 | 
						|
    return __awaiter(this, void 0, void 0, function* () {
 | 
						|
        version = semver.clean(version) || version;
 | 
						|
        arch = arch || os.arch();
 | 
						|
        core.debug(`Caching tool ${tool} ${version} ${arch}`);
 | 
						|
        core.debug(`source file: ${sourceFile}`);
 | 
						|
        if (!fs.statSync(sourceFile).isFile()) {
 | 
						|
            throw new Error('sourceFile is not a file');
 | 
						|
        }
 | 
						|
        // create the tool dir
 | 
						|
        const destFolder = yield _createToolPath(tool, version, arch);
 | 
						|
        // copy instead of move. move can fail on Windows due to
 | 
						|
        // anti-virus software having an open handle on a file.
 | 
						|
        const destPath = path.join(destFolder, targetFile);
 | 
						|
        core.debug(`destination file ${destPath}`);
 | 
						|
        yield io.cp(sourceFile, destPath);
 | 
						|
        // write .complete
 | 
						|
        _completeToolPath(tool, version, arch);
 | 
						|
        return destFolder;
 | 
						|
    });
 | 
						|
}
 | 
						|
exports.cacheFile = cacheFile;
 | 
						|
/**
 | 
						|
 * Finds the path to a tool version in the local installed tool cache
 | 
						|
 *
 | 
						|
 * @param toolName      name of the tool
 | 
						|
 * @param versionSpec   version of the tool
 | 
						|
 * @param arch          optional arch.  defaults to arch of computer
 | 
						|
 */
 | 
						|
function find(toolName, versionSpec, arch) {
 | 
						|
    if (!toolName) {
 | 
						|
        throw new Error('toolName parameter is required');
 | 
						|
    }
 | 
						|
    if (!versionSpec) {
 | 
						|
        throw new Error('versionSpec parameter is required');
 | 
						|
    }
 | 
						|
    arch = arch || os.arch();
 | 
						|
    // attempt to resolve an explicit version
 | 
						|
    if (!_isExplicitVersion(versionSpec)) {
 | 
						|
        const localVersions = findAllVersions(toolName, arch);
 | 
						|
        const match = _evaluateVersions(localVersions, versionSpec);
 | 
						|
        versionSpec = match;
 | 
						|
    }
 | 
						|
    // check for the explicit version in the cache
 | 
						|
    let toolPath = '';
 | 
						|
    if (versionSpec) {
 | 
						|
        versionSpec = semver.clean(versionSpec) || '';
 | 
						|
        const cachePath = path.join(cacheRoot, toolName, versionSpec, arch);
 | 
						|
        core.debug(`checking cache: ${cachePath}`);
 | 
						|
        if (fs.existsSync(cachePath) && fs.existsSync(`${cachePath}.complete`)) {
 | 
						|
            core.debug(`Found tool in cache ${toolName} ${versionSpec} ${arch}`);
 | 
						|
            toolPath = cachePath;
 | 
						|
        }
 | 
						|
        else {
 | 
						|
            core.debug('not found');
 | 
						|
        }
 | 
						|
    }
 | 
						|
    return toolPath;
 | 
						|
}
 | 
						|
exports.find = find;
 | 
						|
/**
 | 
						|
 * Finds the paths to all versions of a tool that are installed in the local tool cache
 | 
						|
 *
 | 
						|
 * @param toolName  name of the tool
 | 
						|
 * @param arch      optional arch.  defaults to arch of computer
 | 
						|
 */
 | 
						|
function findAllVersions(toolName, arch) {
 | 
						|
    const versions = [];
 | 
						|
    arch = arch || os.arch();
 | 
						|
    const toolPath = path.join(cacheRoot, toolName);
 | 
						|
    if (fs.existsSync(toolPath)) {
 | 
						|
        const children = fs.readdirSync(toolPath);
 | 
						|
        for (const child of children) {
 | 
						|
            if (_isExplicitVersion(child)) {
 | 
						|
                const fullPath = path.join(toolPath, child, arch || '');
 | 
						|
                if (fs.existsSync(fullPath) && fs.existsSync(`${fullPath}.complete`)) {
 | 
						|
                    versions.push(child);
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
    return versions;
 | 
						|
}
 | 
						|
exports.findAllVersions = findAllVersions;
 | 
						|
function _createExtractFolder(dest) {
 | 
						|
    return __awaiter(this, void 0, void 0, function* () {
 | 
						|
        if (!dest) {
 | 
						|
            // create a temp dir
 | 
						|
            dest = path.join(tempDirectory, uuidV4());
 | 
						|
        }
 | 
						|
        yield io.mkdirP(dest);
 | 
						|
        return dest;
 | 
						|
    });
 | 
						|
}
 | 
						|
function _createToolPath(tool, version, arch) {
 | 
						|
    return __awaiter(this, void 0, void 0, function* () {
 | 
						|
        const folderPath = path.join(cacheRoot, tool, semver.clean(version) || version, arch || '');
 | 
						|
        core.debug(`destination ${folderPath}`);
 | 
						|
        const markerPath = `${folderPath}.complete`;
 | 
						|
        yield io.rmRF(folderPath);
 | 
						|
        yield io.rmRF(markerPath);
 | 
						|
        yield io.mkdirP(folderPath);
 | 
						|
        return folderPath;
 | 
						|
    });
 | 
						|
}
 | 
						|
function _completeToolPath(tool, version, arch) {
 | 
						|
    const folderPath = path.join(cacheRoot, tool, semver.clean(version) || version, arch || '');
 | 
						|
    const markerPath = `${folderPath}.complete`;
 | 
						|
    fs.writeFileSync(markerPath, '');
 | 
						|
    core.debug('finished caching tool');
 | 
						|
}
 | 
						|
function _isExplicitVersion(versionSpec) {
 | 
						|
    const c = semver.clean(versionSpec) || '';
 | 
						|
    core.debug(`isExplicit: ${c}`);
 | 
						|
    const valid = semver.valid(c) != null;
 | 
						|
    core.debug(`explicit? ${valid}`);
 | 
						|
    return valid;
 | 
						|
}
 | 
						|
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;
 | 
						|
}
 | 
						|
//# sourceMappingURL=tool-cache.js.map
 |