You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
			
				
					164 lines
				
				5.2 KiB
			
		
		
			
		
	
	
					164 lines
				
				5.2 KiB
			| 
								 
											4 years ago
										 
									 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * @fileoverview Utility to get information about the execution environment.
							 | 
						||
| 
								 | 
							
								 * @author Kai Cataldo
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								"use strict";
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								//------------------------------------------------------------------------------
							 | 
						||
| 
								 | 
							
								// Requirements
							 | 
						||
| 
								 | 
							
								//------------------------------------------------------------------------------
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								const path = require("path");
							 | 
						||
| 
								 | 
							
								const spawn = require("cross-spawn");
							 | 
						||
| 
								 | 
							
								const { isEmpty } = require("lodash");
							 | 
						||
| 
								 | 
							
								const log = require("../shared/logging");
							 | 
						||
| 
								 | 
							
								const packageJson = require("../../package.json");
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								//------------------------------------------------------------------------------
							 | 
						||
| 
								 | 
							
								// Helpers
							 | 
						||
| 
								 | 
							
								//------------------------------------------------------------------------------
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Generates and returns execution environment information.
							 | 
						||
| 
								 | 
							
								 * @returns {string} A string that contains execution environment information.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								function environment() {
							 | 
						||
| 
								 | 
							
								    const cache = new Map();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Checks if a path is a child of a directory.
							 | 
						||
| 
								 | 
							
								     * @param {string} parentPath The parent path to check.
							 | 
						||
| 
								 | 
							
								     * @param {string} childPath The path to check.
							 | 
						||
| 
								 | 
							
								     * @returns {boolean} Whether or not the given path is a child of a directory.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    function isChildOfDirectory(parentPath, childPath) {
							 | 
						||
| 
								 | 
							
								        return !path.relative(parentPath, childPath).startsWith("..");
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Synchronously executes a shell command and formats the result.
							 | 
						||
| 
								 | 
							
								     * @param {string} cmd The command to execute.
							 | 
						||
| 
								 | 
							
								     * @param {Array} args The arguments to be executed with the command.
							 | 
						||
| 
								 | 
							
								     * @returns {string} The version returned by the command.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    function execCommand(cmd, args) {
							 | 
						||
| 
								 | 
							
								        const key = [cmd, ...args].join(" ");
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (cache.has(key)) {
							 | 
						||
| 
								 | 
							
								            return cache.get(key);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        const process = spawn.sync(cmd, args, { encoding: "utf8" });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (process.error) {
							 | 
						||
| 
								 | 
							
								            throw process.error;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        const result = process.stdout.trim();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        cache.set(key, result);
							 | 
						||
| 
								 | 
							
								        return result;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Normalizes a version number.
							 | 
						||
| 
								 | 
							
								     * @param {string} versionStr The string to normalize.
							 | 
						||
| 
								 | 
							
								     * @returns {string} The normalized version number.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    function normalizeVersionStr(versionStr) {
							 | 
						||
| 
								 | 
							
								        return versionStr.startsWith("v") ? versionStr : `v${versionStr}`;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Gets bin version.
							 | 
						||
| 
								 | 
							
								     * @param {string} bin The bin to check.
							 | 
						||
| 
								 | 
							
								     * @returns {string} The normalized version returned by the command.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    function getBinVersion(bin) {
							 | 
						||
| 
								 | 
							
								        const binArgs = ["--version"];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        try {
							 | 
						||
| 
								 | 
							
								            return normalizeVersionStr(execCommand(bin, binArgs));
							 | 
						||
| 
								 | 
							
								        } catch (e) {
							 | 
						||
| 
								 | 
							
								            log.error(`Error finding ${bin} version running the command \`${bin} ${binArgs.join(" ")}\``);
							 | 
						||
| 
								 | 
							
								            throw e;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Gets installed npm package version.
							 | 
						||
| 
								 | 
							
								     * @param {string} pkg The package to check.
							 | 
						||
| 
								 | 
							
								     * @param {boolean} global Whether to check globally or not.
							 | 
						||
| 
								 | 
							
								     * @returns {string} The normalized version returned by the command.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    function getNpmPackageVersion(pkg, { global = false } = {}) {
							 | 
						||
| 
								 | 
							
								        const npmBinArgs = ["bin", "-g"];
							 | 
						||
| 
								 | 
							
								        const npmLsArgs = ["ls", "--depth=0", "--json", "eslint"];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (global) {
							 | 
						||
| 
								 | 
							
								            npmLsArgs.push("-g");
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        try {
							 | 
						||
| 
								 | 
							
								            const parsedStdout = JSON.parse(execCommand("npm", npmLsArgs));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            /*
							 | 
						||
| 
								 | 
							
								             * Checking globally returns an empty JSON object, while local checks
							 | 
						||
| 
								 | 
							
								             * include the name and version of the local project.
							 | 
						||
| 
								 | 
							
								             */
							 | 
						||
| 
								 | 
							
								            if (isEmpty(parsedStdout) || !(parsedStdout.dependencies && parsedStdout.dependencies.eslint)) {
							 | 
						||
| 
								 | 
							
								                return "Not found";
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            const [, processBinPath] = process.argv;
							 | 
						||
| 
								 | 
							
								            let npmBinPath;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            try {
							 | 
						||
| 
								 | 
							
								                npmBinPath = execCommand("npm", npmBinArgs);
							 | 
						||
| 
								 | 
							
								            } catch (e) {
							 | 
						||
| 
								 | 
							
								                log.error(`Error finding npm binary path when running command \`npm ${npmBinArgs.join(" ")}\``);
							 | 
						||
| 
								 | 
							
								                throw e;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            const isGlobal = isChildOfDirectory(npmBinPath, processBinPath);
							 | 
						||
| 
								 | 
							
								            let pkgVersion = parsedStdout.dependencies.eslint.version;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if ((global && isGlobal) || (!global && !isGlobal)) {
							 | 
						||
| 
								 | 
							
								                pkgVersion += " (Currently used)";
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return normalizeVersionStr(pkgVersion);
							 | 
						||
| 
								 | 
							
								        } catch (e) {
							 | 
						||
| 
								 | 
							
								            log.error(`Error finding ${pkg} version running the command \`npm ${npmLsArgs.join(" ")}\``);
							 | 
						||
| 
								 | 
							
								            throw e;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    return [
							 | 
						||
| 
								 | 
							
								        "Environment Info:",
							 | 
						||
| 
								 | 
							
								        "",
							 | 
						||
| 
								 | 
							
								        `Node version: ${getBinVersion("node")}`,
							 | 
						||
| 
								 | 
							
								        `npm version: ${getBinVersion("npm")}`,
							 | 
						||
| 
								 | 
							
								        `Local ESLint version: ${getNpmPackageVersion("eslint", { global: false })}`,
							 | 
						||
| 
								 | 
							
								        `Global ESLint version: ${getNpmPackageVersion("eslint", { global: true })}`
							 | 
						||
| 
								 | 
							
								    ].join("\n");
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Returns version of currently executing ESLint.
							 | 
						||
| 
								 | 
							
								 * @returns {string} The version from the currently executing ESLint's package.json.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								function version() {
							 | 
						||
| 
								 | 
							
								    return `v${packageJson.version}`;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								//------------------------------------------------------------------------------
							 | 
						||
| 
								 | 
							
								// Public Interface
							 | 
						||
| 
								 | 
							
								//------------------------------------------------------------------------------
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								module.exports = {
							 | 
						||
| 
								 | 
							
								    environment,
							 | 
						||
| 
								 | 
							
								    version
							 | 
						||
| 
								 | 
							
								};
							 |