Files
codeql-action/lib/codeql.js
T

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

932 lines
42 KiB
JavaScript
Raw Normal View History

2020-06-26 17:22:19 +01:00
"use strict";
2021-07-27 17:59:59 +01:00
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
2023-01-18 20:00:33 +00:00
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
2021-07-27 17:59:59 +01:00
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
2020-06-26 17:22:19 +01:00
var __importStar = (this && this.__importStar) || function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
2021-07-27 17:59:59 +01:00
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
__setModuleDefault(result, mod);
2020-06-26 17:22:19 +01:00
return result;
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.getGeneratedCodeScanningConfigPath = exports.getTrapCachingExtractorConfigArgsForLang = exports.getTrapCachingExtractorConfigArgs = exports.getExtraOptions = exports.getCodeQLForCmd = exports.getCodeQLForTesting = exports.getCachedCodeQL = exports.setCodeQL = exports.getCodeQL = exports.setupCodeQL = exports.CODEQL_VERSION_SUBLANGUAGE_FILE_COVERAGE = exports.CODEQL_VERSION_ANALYSIS_SUMMARY_V2 = exports.CODEQL_VERSION_LANGUAGE_ALIASING = exports.CODEQL_VERSION_LANGUAGE_BASELINE_CONFIG = exports.CODEQL_VERSION_RESOLVE_ENVIRONMENT = exports.CODEQL_VERSION_DIAGNOSTICS_EXPORT_FIXED = void 0;
2020-06-26 17:22:19 +01:00
const fs = __importStar(require("fs"));
const path = __importStar(require("path"));
const core = __importStar(require("@actions/core"));
2020-11-20 11:35:59 +01:00
const toolrunner = __importStar(require("@actions/exec/lib/toolrunner"));
const yaml = __importStar(require("js-yaml"));
2023-10-25 17:24:22 +01:00
const semver = __importStar(require("semver"));
2020-09-29 14:43:37 +01:00
const actions_util_1 = require("./actions-util");
2024-02-08 09:20:03 -08:00
const cli_errors_1 = require("./cli-errors");
const environment_1 = require("./environment");
2023-03-09 16:44:45 +00:00
const feature_flags_1 = require("./feature-flags");
2023-01-10 13:16:22 +00:00
const setupCodeql = __importStar(require("./setup-codeql"));
2024-01-30 21:53:24 +00:00
const tools_features_1 = require("./tools-features");
const tracer_config_1 = require("./tracer-config");
const util = __importStar(require("./util"));
2023-04-06 17:04:21 +01:00
const util_1 = require("./util");
/**
2020-07-15 17:36:49 +01:00
* Stores the CodeQL object, and is populated by `setupCodeQL` or `getCodeQL`.
* Can be overridden in tests using `setCodeQL`.
*/
let cachedCodeQL = undefined;
2021-08-18 10:15:48 +01:00
/**
* The oldest version of CodeQL that the Action will run with. This should be
2022-11-11 18:00:58 +00:00
* at least three minor versions behind the current version and must include the
* CLI versions shipped with each supported version of GHES.
2021-08-18 10:15:48 +01:00
*
2022-11-11 18:00:58 +00:00
* The version flags below can be used to conditionally enable certain features
* on versions newer than this.
2021-08-18 10:15:48 +01:00
*/
const CODEQL_MINIMUM_VERSION = "2.12.6";
/**
* This version will shortly become the oldest version of CodeQL that the Action will run with.
*/
2024-04-03 15:44:00 +01:00
const CODEQL_NEXT_MINIMUM_VERSION = "2.12.6";
2023-09-14 16:34:07 +01:00
/**
* This is the version of GHES that was most recently deprecated.
*/
2024-04-03 15:44:00 +01:00
const GHES_VERSION_MOST_RECENTLY_DEPRECATED = "3.8";
2023-09-14 16:34:07 +01:00
/**
* This is the deprecation date for the version of GHES that was most recently deprecated.
*/
2024-04-03 15:44:00 +01:00
const GHES_MOST_RECENT_DEPRECATION_DATE = "2024-03-26";
/** The CLI verbosity level to use for extraction in debug mode. */
const EXTRACTION_DEBUG_MODE_VERBOSITY = "progress++";
2023-09-27 16:16:13 +01:00
/*
* Deprecated in favor of ToolsFeature.
*
2021-08-18 09:44:49 +01:00
* Versions of CodeQL that version-flag certain functionality in the Action.
2021-08-18 10:15:48 +01:00
* For convenience, please keep these in descending order. Once a version
* flag is older than the oldest supported version above, it may be removed.
2021-08-18 09:44:49 +01:00
*/
/**
* Versions 2.13.1+ of the CodeQL CLI fix a bug where diagnostics export could produce invalid SARIF.
*/
exports.CODEQL_VERSION_DIAGNOSTICS_EXPORT_FIXED = "2.13.1";
2023-06-15 10:15:03 +01:00
/**
* Versions 2.13.4+ of the CodeQL CLI support the `resolve build-environment` command.
*/
exports.CODEQL_VERSION_RESOLVE_ENVIRONMENT = "2.13.4";
/**
* Versions 2.14.2+ of the CodeQL CLI support language-specific baseline configuration.
*/
exports.CODEQL_VERSION_LANGUAGE_BASELINE_CONFIG = "2.14.2";
2023-09-14 15:26:38 +01:00
/**
* Versions 2.14.4+ of the CodeQL CLI support language aliasing.
*/
exports.CODEQL_VERSION_LANGUAGE_ALIASING = "2.14.4";
2023-10-25 17:24:22 +01:00
/**
* Versions 2.15.0+ of the CodeQL CLI support new analysis summaries.
*/
exports.CODEQL_VERSION_ANALYSIS_SUMMARY_V2 = "2.15.0";
/**
* Versions 2.15.0+ of the CodeQL CLI support sub-language file coverage information.
*/
exports.CODEQL_VERSION_SUBLANGUAGE_FILE_COVERAGE = "2.15.0";
/**
* Versions 2.15.2+ of the CodeQL CLI support the `--sarif-include-query-help` option.
*/
const CODEQL_VERSION_INCLUDE_QUERY_HELP = "2.15.2";
/**
* Set up CodeQL CLI access.
*
2023-01-05 16:54:18 +00:00
* @param toolsInput
* @param apiDetails
* @param tempDir
* @param variant
2023-01-10 13:16:22 +00:00
* @param defaultCliVersion
* @param logger
* @param checkVersion Whether to check that CodeQL CLI meets the minimum
* version requirement. Must be set to true outside tests.
* @returns a { CodeQL, toolsVersion } object.
*/
async function setupCodeQL(toolsInput, apiDetails, tempDir, variant, defaultCliVersion, logger, checkVersion) {
2020-06-26 17:22:19 +01:00
try {
const { codeqlFolder, toolsDownloadDurationMs, toolsSource, toolsVersion } = await setupCodeql.setupCodeQLBundle(toolsInput, apiDetails, tempDir, variant, defaultCliVersion, logger);
2020-09-14 10:44:43 +01:00
let codeqlCmd = path.join(codeqlFolder, "codeql", "codeql");
if (process.platform === "win32") {
2020-06-26 17:22:19 +01:00
codeqlCmd += ".exe";
}
2020-09-14 10:44:43 +01:00
else if (process.platform !== "linux" && process.platform !== "darwin") {
2024-02-08 09:20:03 -08:00
throw new util.ConfigurationError(`Unsupported platform: ${process.platform}`);
2020-06-26 17:22:19 +01:00
}
2021-08-18 10:15:48 +01:00
cachedCodeQL = await getCodeQLForCmd(codeqlCmd, checkVersion);
return {
codeql: cachedCodeQL,
toolsDownloadDurationMs,
toolsSource,
toolsVersion,
};
2020-06-26 17:22:19 +01:00
}
catch (e) {
throw new Error(`Unable to download and extract CodeQL CLI: ${(0, util_1.wrapError)(e).message}`);
2020-06-26 17:22:19 +01:00
}
}
exports.setupCodeQL = setupCodeQL;
2020-08-19 15:54:23 +01:00
/**
* Use the CodeQL executable located at the given path.
*/
2021-08-18 10:15:48 +01:00
async function getCodeQL(cmd) {
if (cachedCodeQL === undefined) {
2021-08-18 10:15:48 +01:00
cachedCodeQL = await getCodeQLForCmd(cmd, true);
}
return cachedCodeQL;
2020-06-26 17:22:19 +01:00
}
exports.getCodeQL = getCodeQL;
2020-08-19 15:54:23 +01:00
function resolveFunction(partialCodeql, methodName, defaultImplementation) {
2020-09-14 10:44:43 +01:00
if (typeof partialCodeql[methodName] !== "function") {
2020-08-19 15:54:23 +01:00
if (defaultImplementation !== undefined) {
return defaultImplementation;
}
const dummyMethod = () => {
2020-09-14 10:44:43 +01:00
throw new Error(`CodeQL ${methodName} method not correctly defined`);
};
return dummyMethod;
}
return partialCodeql[methodName];
}
2020-07-15 17:36:49 +01:00
/**
* Set the functionality for CodeQL methods. Only for use in tests.
*
* Accepts a partial object and any undefined methods will be implemented
* to immediately throw an exception indicating which method is missing.
*/
function setCodeQL(partialCodeql) {
cachedCodeQL = {
2020-09-14 10:44:43 +01:00
getPath: resolveFunction(partialCodeql, "getPath", () => "/tmp/dummy-path"),
2024-01-30 21:53:24 +00:00
getVersion: resolveFunction(partialCodeql, "getVersion", async () => ({
2023-10-04 15:45:42 +01:00
version: "1.0.0",
2024-01-30 21:53:24 +00:00
})),
2020-09-14 10:44:43 +01:00
printVersion: resolveFunction(partialCodeql, "printVersion"),
2024-01-30 21:53:24 +00:00
supportsFeature: resolveFunction(partialCodeql, "supportsFeature", async (feature) => !!partialCodeql.getVersion &&
(0, tools_features_1.isSupportedToolsFeature)(await partialCodeql.getVersion(), feature)),
databaseInitCluster: resolveFunction(partialCodeql, "databaseInitCluster"),
2020-09-14 10:44:43 +01:00
runAutobuild: resolveFunction(partialCodeql, "runAutobuild"),
extractScannedLanguage: resolveFunction(partialCodeql, "extractScannedLanguage"),
extractUsingBuildMode: resolveFunction(partialCodeql, "extractUsingBuildMode"),
2020-09-14 10:44:43 +01:00
finalizeDatabase: resolveFunction(partialCodeql, "finalizeDatabase"),
2021-05-23 16:27:46 +02:00
resolveLanguages: resolveFunction(partialCodeql, "resolveLanguages"),
2022-08-02 17:52:22 +01:00
betterResolveLanguages: resolveFunction(partialCodeql, "betterResolveLanguages"),
2020-09-14 10:44:43 +01:00
resolveQueries: resolveFunction(partialCodeql, "resolveQueries"),
2023-05-18 20:12:12 +01:00
resolveBuildEnvironment: resolveFunction(partialCodeql, "resolveBuildEnvironment"),
2021-06-03 09:32:44 -07:00
packDownload: resolveFunction(partialCodeql, "packDownload"),
databaseCleanup: resolveFunction(partialCodeql, "databaseCleanup"),
2021-06-22 13:05:12 +01:00
databaseBundle: resolveFunction(partialCodeql, "databaseBundle"),
databaseRunQueries: resolveFunction(partialCodeql, "databaseRunQueries"),
databaseInterpretResults: resolveFunction(partialCodeql, "databaseInterpretResults"),
2021-09-28 09:25:35 +01:00
databasePrintBaseline: resolveFunction(partialCodeql, "databasePrintBaseline"),
databaseExportDiagnostics: resolveFunction(partialCodeql, "databaseExportDiagnostics"),
2022-11-09 17:32:21 +00:00
diagnosticsExport: resolveFunction(partialCodeql, "diagnosticsExport"),
resolveExtractor: resolveFunction(partialCodeql, "resolveExtractor"),
mergeResults: resolveFunction(partialCodeql, "mergeResults"),
};
2020-08-19 15:54:23 +01:00
return cachedCodeQL;
}
exports.setCodeQL = setCodeQL;
2020-08-19 15:54:23 +01:00
/**
* Get the cached CodeQL object. Should only be used from tests.
*
* TODO: Work out a good way for tests to get this from the test context
* instead of having to have this method.
*/
function getCachedCodeQL() {
if (cachedCodeQL === undefined) {
// Should never happen as setCodeQL is called by testing-utils.setupTests
2020-09-14 10:44:43 +01:00
throw new Error("cachedCodeQL undefined");
2020-08-19 15:54:23 +01:00
}
return cachedCodeQL;
}
exports.getCachedCodeQL = getCachedCodeQL;
/**
* Get a real, newly created CodeQL instance for testing. The instance refers to
* a non-existent placeholder codeql command, so tests that use this function
* should also stub the toolrunner.ToolRunner constructor.
*/
async function getCodeQLForTesting(cmd = "codeql-for-testing") {
return getCodeQLForCmd(cmd, false);
}
exports.getCodeQLForTesting = getCodeQLForTesting;
/**
* Return a CodeQL object for CodeQL CLI access.
*
* @param cmd Path to CodeQL CLI
* @param checkVersion Whether to check that CodeQL CLI meets the minimum
* version requirement. Must be set to true outside tests.
* @returns A new CodeQL object
*/
2021-08-18 10:15:48 +01:00
async function getCodeQLForCmd(cmd, checkVersion) {
const codeql = {
2020-09-14 10:44:43 +01:00
getPath() {
2020-08-19 15:54:23 +01:00
return cmd;
2020-06-26 17:22:19 +01:00
},
2021-08-12 16:02:19 +01:00
async getVersion() {
let result = util.getCachedCodeQlVersion();
if (result === undefined) {
2023-10-04 15:49:25 +01:00
const output = await runTool(cmd, ["version", "--format=json"]);
try {
result = JSON.parse(output);
}
catch (err) {
throw Error(`Invalid JSON output from \`version --format=json\`: ${output}`);
}
util.cacheCodeQlVersion(result);
}
return result;
2021-08-12 16:02:19 +01:00
},
2020-09-14 10:44:43 +01:00
async printVersion() {
2021-06-09 13:17:25 -07:00
await runTool(cmd, ["version", "--format=json"]);
2020-06-26 17:22:19 +01:00
},
2024-01-30 21:53:24 +00:00
async supportsFeature(feature) {
return (0, tools_features_1.isSupportedToolsFeature)(await this.getVersion(), feature);
},
async databaseInitCluster(config, sourceRoot, processName, qlconfigFile, features, logger) {
const extraArgs = config.languages.map((language) => `--language=${language}`);
if (await (0, tracer_config_1.shouldEnableIndirectTracing)(codeql, config, features)) {
extraArgs.push("--begin-tracing");
extraArgs.push(...(await getTrapCachingExtractorConfigArgs(config)));
extraArgs.push(`--trace-process-name=${processName}`);
}
2024-01-04 14:39:36 +00:00
const codeScanningConfigFile = await generateCodeScanningConfig(config, logger);
const externalRepositoryToken = (0, actions_util_1.getOptionalInput)("external-repository-token");
extraArgs.push(`--codescanning-config=${codeScanningConfigFile}`);
if (externalRepositoryToken) {
extraArgs.push("--external-repository-token-stdin");
}
2024-01-30 21:59:51 +00:00
if (config.buildMode !== undefined &&
(await this.supportsFeature(tools_features_1.ToolsFeature.BuildModeOption))) {
extraArgs.push(`--build-mode=${config.buildMode}`);
}
if (qlconfigFile !== undefined) {
2023-03-06 10:46:36 +00:00
extraArgs.push(`--qlconfig-file=${qlconfigFile}`);
}
if (await util.codeQlVersionAtLeast(this, exports.CODEQL_VERSION_LANGUAGE_BASELINE_CONFIG)) {
extraArgs.push("--calculate-language-specific-baseline");
}
if (await isSublanguageFileCoverageEnabled(config, this)) {
extraArgs.push("--sublanguage-file-coverage");
}
else if (await util.codeQlVersionAtLeast(this, exports.CODEQL_VERSION_SUBLANGUAGE_FILE_COVERAGE)) {
extraArgs.push("--no-sublanguage-file-coverage");
}
2024-06-11 12:31:07 -07:00
const overwriteFlag = (0, tools_features_1.isSupportedToolsFeature)(await this.getVersion(), tools_features_1.ToolsFeature.ForceOverwrite)
2024-06-11 12:00:21 -07:00
? "--force-overwrite"
: "--overwrite";
await runTool(cmd, [
"database",
"init",
overwriteFlag,
"--db-cluster",
config.dbLocation,
`--source-root=${sourceRoot}`,
...(await getLanguageAliasingArguments(this)),
...extraArgs,
2024-04-16 18:08:46 +01:00
...getExtraOptionsFromEnv(["database", "init"], {
ignoringOptions: ["--overwrite"],
}),
], { stdin: externalRepositoryToken });
},
async runAutobuild(config, language) {
applyAutobuildAzurePipelinesTimeoutFix();
const autobuildCmd = path.join(await this.resolveExtractor(language), "tools", process.platform === "win32" ? "autobuild.cmd" : "autobuild.sh");
// Bump the verbosity of the autobuild command if we're in debug mode
if (config.debugMode) {
process.env[environment_1.EnvVar.CLI_VERBOSITY] =
process.env[environment_1.EnvVar.CLI_VERBOSITY] || EXTRACTION_DEBUG_MODE_VERBOSITY;
}
2022-07-25 14:31:52 -07:00
// On macOS, System Integrity Protection (SIP) typically interferes with
// CodeQL build tracing of protected binaries.
// The usual workaround is to prefix `$CODEQL_RUNNER` to build commands:
// `$CODEQL_RUNNER` (not to be confused with the deprecated CodeQL Runner tool)
// points to a simple wrapper binary included with the CLI, and the extra layer of
// process indirection helps the tracer bypass SIP.
// The above SIP workaround is *not* needed here.
// At the `autobuild` step in the Actions workflow, we assume the `init` step
// has successfully run, and will have exported `DYLD_INSERT_LIBRARIES`
// into the environment of subsequent steps, to activate the tracer.
// When `DYLD_INSERT_LIBRARIES` is set in the environment for a step,
// the Actions runtime introduces its own workaround for SIP
// (https://github.com/actions/runner/pull/416).
await runTool(autobuildCmd);
2020-07-02 14:45:14 +01:00
},
async extractScannedLanguage(config, language) {
2023-07-20 18:31:37 +01:00
await runTool(cmd, [
2020-09-14 10:44:43 +01:00
"database",
"trace-command",
"--index-traceless-dbs",
...(await getTrapCachingExtractorConfigArgsForLang(config, language)),
...getExtractionVerbosityArguments(config.debugMode),
2020-09-14 10:44:43 +01:00
...getExtraOptionsFromEnv(["database", "trace-command"]),
util.getCodeQLDatabasePath(config, language),
2023-07-20 18:31:37 +01:00
]);
2020-06-26 17:22:19 +01:00
},
async extractUsingBuildMode(config, language) {
if (config.buildMode === util_1.BuildMode.Autobuild) {
applyAutobuildAzurePipelinesTimeoutFix();
}
try {
await runTool(cmd, [
"database",
"trace-command",
"--use-build-mode",
"--working-dir",
process.cwd(),
...(await getTrapCachingExtractorConfigArgsForLang(config, language)),
...getExtractionVerbosityArguments(config.debugMode),
...getExtraOptionsFromEnv(["database", "trace-command"]),
util.getCodeQLDatabasePath(config, language),
]);
}
catch (e) {
if (config.buildMode === util_1.BuildMode.Autobuild) {
const prefix = "We were unable to automatically build your code. " +
"Please change the build mode for this language to manual and specify build steps " +
"for your project. For more information, see " +
"https://docs.github.com/en/code-security/code-scanning/troubleshooting-code-scanning/automatic-build-failed.";
const ErrorConstructor = e instanceof util.ConfigurationError
? util.ConfigurationError
: Error;
throw new ErrorConstructor(`${prefix} ${util.wrapError(e).message}`);
}
else {
throw e;
}
}
},
async finalizeDatabase(databasePath, threadsFlag, memoryFlag, enableDebugLogging) {
2021-08-12 16:02:19 +01:00
const args = [
2020-09-14 10:44:43 +01:00
"database",
"finalize",
"--finalize-dataset",
threadsFlag,
2022-11-11 18:05:24 +00:00
memoryFlag,
...getExtractionVerbosityArguments(enableDebugLogging),
2020-09-14 10:44:43 +01:00
...getExtraOptionsFromEnv(["database", "finalize"]),
databasePath,
2021-08-12 16:02:19 +01:00
];
await runTool(cmd, args);
2020-06-26 17:22:19 +01:00
},
2021-05-23 16:27:46 +02:00
async resolveLanguages() {
2022-01-21 13:44:52 +00:00
const codeqlArgs = [
"resolve",
"languages",
"--format=json",
...getExtraOptionsFromEnv(["resolve", "languages"]),
];
2021-06-09 13:17:25 -07:00
const output = await runTool(cmd, codeqlArgs);
2021-05-24 11:00:02 +02:00
try {
return JSON.parse(output);
}
catch (e) {
throw new Error(`Unexpected output from codeql resolve languages: ${e}`);
}
2021-05-23 16:27:46 +02:00
},
2022-08-02 17:52:22 +01:00
async betterResolveLanguages() {
const codeqlArgs = [
"resolve",
"languages",
"--format=betterjson",
"--extractor-options-verbosity=4",
...(await getLanguageAliasingArguments(this)),
2022-08-02 17:52:22 +01:00
...getExtraOptionsFromEnv(["resolve", "languages"]),
];
const output = await runTool(cmd, codeqlArgs);
try {
return JSON.parse(output);
}
catch (e) {
throw new Error(`Unexpected output from codeql resolve languages with --format=betterjson: ${e}`);
}
},
2020-09-14 10:44:43 +01:00
async resolveQueries(queries, extraSearchPath) {
const codeqlArgs = [
2020-09-14 10:44:43 +01:00
"resolve",
"queries",
2020-06-26 17:22:19 +01:00
...queries,
2020-09-14 10:44:43 +01:00
"--format=bylanguage",
...getExtraOptionsFromEnv(["resolve", "queries"]),
];
if (extraSearchPath !== undefined) {
codeqlArgs.push("--additional-packs", extraSearchPath);
}
2021-06-09 13:17:25 -07:00
const output = await runTool(cmd, codeqlArgs);
2021-05-24 11:00:02 +02:00
try {
return JSON.parse(output);
}
catch (e) {
throw new Error(`Unexpected output from codeql resolve queries: ${e}`);
}
2020-06-26 17:22:19 +01:00
},
2023-06-02 19:36:07 +01:00
async resolveBuildEnvironment(workingDir, language) {
2023-05-18 20:12:12 +01:00
const codeqlArgs = [
"resolve",
"build-environment",
`--language=${language}`,
...(await getLanguageAliasingArguments(this)),
2023-05-18 20:12:12 +01:00
...getExtraOptionsFromEnv(["resolve", "build-environment"]),
];
2023-06-02 19:36:07 +01:00
if (workingDir !== undefined) {
codeqlArgs.push("--working-dir", workingDir);
}
2023-05-18 20:12:12 +01:00
const output = await runTool(cmd, codeqlArgs);
try {
return JSON.parse(output);
}
catch (e) {
2023-06-06 12:17:21 +01:00
throw new Error(`Unexpected output from codeql resolve build-environment: ${e} in\n${output}`);
2023-05-18 20:12:12 +01:00
}
},
async databaseRunQueries(databasePath, flags) {
2021-06-09 13:17:25 -07:00
const codeqlArgs = [
2020-09-14 10:44:43 +01:00
"database",
"run-queries",
...flags,
2020-07-02 14:45:14 +01:00
databasePath,
"--expect-discarded-cache",
2023-11-20 14:35:28 -08:00
"--min-disk-free=1024", // Try to leave at least 1GB free
"-v",
2024-04-16 18:08:46 +01:00
...getExtraOptionsFromEnv(["database", "run-queries"], {
ignoringOptions: ["--expect-discarded-cache"],
}),
];
if (await util.codeQlVersionAtLeast(this, feature_flags_1.CODEQL_VERSION_FINE_GRAINED_PARALLELISM)) {
codeqlArgs.push("--intra-layer-parallelism");
}
2023-07-20 18:31:37 +01:00
await runTool(cmd, codeqlArgs);
},
async databaseInterpretResults(databasePath, querySuitePaths, sarifFile, addSnippetsFlag, threadsFlag, verbosityFlag, automationDetailsId, config, features, logger) {
const shouldExportDiagnostics = await features.getValue(feature_flags_1.Feature.ExportDiagnosticsEnabled, this);
const shouldWorkaroundInvalidNotifications = shouldExportDiagnostics &&
!(await isDiagnosticsExportInvalidSarifFixed(this));
const codeqlOutputFile = shouldWorkaroundInvalidNotifications
? path.join(config.tempDir, "codeql-intermediate-results.sarif")
: sarifFile;
2021-06-09 13:17:25 -07:00
const codeqlArgs = [
"database",
"interpret-results",
threadsFlag,
"--format=sarif-latest",
verbosityFlag,
`--output=${codeqlOutputFile}`,
addSnippetsFlag,
2022-11-11 18:05:24 +00:00
"--print-diagnostics-summary",
"--print-metrics-summary",
"--sarif-add-baseline-file-info",
`--sarif-codescanning-config=${getGeneratedCodeScanningConfigPath(config)}`,
"--sarif-group-rules-by-pack",
...(await getCodeScanningQueryHelpArguments(this)),
...getExtraOptionsFromEnv(["database", "interpret-results"]),
];
2022-11-11 18:05:24 +00:00
if (automationDetailsId !== undefined) {
2021-06-09 13:17:25 -07:00
codeqlArgs.push("--sarif-category", automationDetailsId);
2021-05-03 19:41:53 +02:00
}
if (await isSublanguageFileCoverageEnabled(config, this)) {
codeqlArgs.push("--sublanguage-file-coverage");
}
else if (await util.codeQlVersionAtLeast(this, exports.CODEQL_VERSION_SUBLANGUAGE_FILE_COVERAGE)) {
codeqlArgs.push("--no-sublanguage-file-coverage");
}
if (shouldExportDiagnostics) {
codeqlArgs.push("--sarif-include-diagnostics");
}
else {
codeqlArgs.push("--no-sarif-include-diagnostics");
}
if ((await util.codeQlVersionAtLeast(this, exports.CODEQL_VERSION_ANALYSIS_SUMMARY_V2)) &&
!(0, tools_features_1.isSupportedToolsFeature)(await this.getVersion(), tools_features_1.ToolsFeature.AnalysisSummaryV2IsDefault)) {
2023-07-05 18:56:55 +01:00
codeqlArgs.push("--new-analysis-summary");
}
codeqlArgs.push(databasePath);
if (querySuitePaths) {
codeqlArgs.push(...querySuitePaths);
}
2023-07-28 12:04:12 +01:00
// Capture the stdout, which contains the analysis summary. Don't stream it to the Actions
// logs to avoid printing it twice.
const analysisSummary = await runTool(cmd, codeqlArgs, {
noStreamStdout: true,
});
if (shouldWorkaroundInvalidNotifications) {
2023-03-27 15:44:47 +01:00
util.fixInvalidNotificationsInFile(codeqlOutputFile, sarifFile, logger);
}
2023-07-28 12:04:12 +01:00
return analysisSummary;
2020-09-14 10:44:43 +01:00
},
2021-09-28 09:25:35 +01:00
async databasePrintBaseline(databasePath) {
const codeqlArgs = [
"database",
"print-baseline",
...getExtraOptionsFromEnv(["database", "print-baseline"]),
databasePath,
];
return await runTool(cmd, codeqlArgs);
},
/**
* Download specified packs into the package cache. If the specified
* package and version already exists (e.g., from a previous analysis run),
* then it is not downloaded again (unless the extra option `--force` is
* specified).
*
* If no version is specified, then the latest version is
* downloaded. The check to determine what the latest version is is done
* each time this package is requested.
2022-08-29 12:57:46 -07:00
*
* Optionally, a `qlconfigFile` is included. If used, then this file
* is used to determine which registry each pack is downloaded from.
*/
2022-08-29 12:57:46 -07:00
async packDownload(packs, qlconfigFile) {
const qlconfigArg = qlconfigFile
? [`--qlconfig-file=${qlconfigFile}`]
: [];
2021-06-09 13:17:25 -07:00
const codeqlArgs = [
2021-06-03 09:32:44 -07:00
"pack",
"download",
2022-08-29 12:57:46 -07:00
...qlconfigArg,
2021-06-04 13:44:24 -07:00
"--format=json",
"--resolve-query-specs",
2021-06-03 09:32:44 -07:00
...getExtraOptionsFromEnv(["pack", "download"]),
2022-04-26 19:47:59 -07:00
...packs,
2021-06-03 09:32:44 -07:00
];
2021-06-09 13:17:25 -07:00
const output = await runTool(cmd, codeqlArgs);
2021-06-03 09:32:44 -07:00
try {
2021-06-07 16:05:32 -07:00
const parsedOutput = JSON.parse(output);
if (Array.isArray(parsedOutput.packs) &&
2021-06-08 10:24:08 -07:00
// TODO PackDownloadOutput will not include the version if it is not specified
// in the input. The version is always the latest version available.
// It should be added to the output, but this requires a CLI change
parsedOutput.packs.every((p) => p.name /* && p.version */)) {
2021-06-07 16:05:32 -07:00
return parsedOutput;
}
else {
throw new Error("Unexpected output from pack download");
}
2021-06-03 09:32:44 -07:00
}
catch (e) {
2021-06-07 16:05:32 -07:00
throw new Error(`Attempted to download specified packs but got an error:\n${output}\n${e}`);
2021-06-03 09:32:44 -07:00
}
2020-09-14 10:44:43 +01:00
},
async databaseCleanup(databasePath, cleanupLevel) {
2021-06-09 13:17:25 -07:00
const codeqlArgs = [
"database",
"cleanup",
databasePath,
`--mode=${cleanupLevel}`,
2022-01-21 13:44:52 +00:00
...getExtraOptionsFromEnv(["database", "cleanup"]),
];
2021-06-09 13:17:25 -07:00
await runTool(cmd, codeqlArgs);
},
async databaseBundle(databasePath, outputFilePath, databaseName) {
2021-06-22 13:05:12 +01:00
const args = [
"database",
"bundle",
databasePath,
`--output=${outputFilePath}`,
`--name=${databaseName}`,
2022-01-21 13:44:52 +00:00
...getExtraOptionsFromEnv(["database", "bundle"]),
2021-06-22 13:05:12 +01:00
];
await new toolrunner.ToolRunner(cmd, args).exec();
},
async databaseExportDiagnostics(databasePath, sarifFile, automationDetailsId, tempDir, logger) {
const shouldWorkaroundInvalidNotifications = !(await isDiagnosticsExportInvalidSarifFixed(this));
const codeqlOutputFile = shouldWorkaroundInvalidNotifications
? path.join(tempDir, "codeql-intermediate-results.sarif")
: sarifFile;
const args = [
"database",
"export-diagnostics",
`${databasePath}`,
2023-11-20 14:35:28 -08:00
"--db-cluster", // Database is always a cluster for CodeQL versions that support diagnostics.
"--format=sarif-latest",
`--output=${codeqlOutputFile}`,
2023-11-20 14:35:28 -08:00
"--sarif-include-diagnostics", // ExportDiagnosticsEnabled is always true if this command is run.
"-vvv",
...getExtraOptionsFromEnv(["diagnostics", "export"]),
];
if (automationDetailsId !== undefined) {
args.push("--sarif-category", automationDetailsId);
}
await new toolrunner.ToolRunner(cmd, args).exec();
if (shouldWorkaroundInvalidNotifications) {
// Fix invalid notifications in the SARIF file output by CodeQL.
util.fixInvalidNotificationsInFile(codeqlOutputFile, sarifFile, logger);
}
},
async diagnosticsExport(sarifFile, automationDetailsId, config) {
2022-11-09 17:32:21 +00:00
const args = [
"diagnostics",
"export",
"--format=sarif-latest",
`--output=${sarifFile}`,
`--sarif-codescanning-config=${getGeneratedCodeScanningConfigPath(config)}`,
2022-11-09 17:32:21 +00:00
...getExtraOptionsFromEnv(["diagnostics", "export"]),
];
if (automationDetailsId !== undefined) {
args.push("--sarif-category", automationDetailsId);
}
await new toolrunner.ToolRunner(cmd, args).exec();
},
async resolveExtractor(language) {
// Request it using `format=json` so we don't need to strip the trailing new line generated by
// the CLI.
let extractorPath = "";
await new toolrunner.ToolRunner(cmd, [
"resolve",
"extractor",
"--format=json",
`--language=${language}`,
...(await getLanguageAliasingArguments(this)),
...getExtraOptionsFromEnv(["resolve", "extractor"]),
], {
silent: true,
listeners: {
stdout: (data) => {
extractorPath += data.toString();
},
stderr: (data) => {
process.stderr.write(data);
},
},
}).exec();
return JSON.parse(extractorPath);
},
async mergeResults(sarifFiles, outputFile, { mergeRunsFromEqualCategory = false, }) {
const args = [
"github",
"merge-results",
"--output",
outputFile,
...getExtraOptionsFromEnv(["github", "merge-results"]),
];
for (const sarifFile of sarifFiles) {
args.push("--sarif", sarifFile);
}
if (mergeRunsFromEqualCategory) {
args.push("--sarif-merge-runs-from-equal-category");
}
2024-03-25 10:03:31 +01:00
await runTool(cmd, args);
},
2020-06-26 17:22:19 +01:00
};
2022-11-11 18:09:53 +00:00
// To ensure that status reports include the CodeQL CLI version wherever
// possible, we want to call getVersion(), which populates the version value
// used by status reporting, at the earliest opportunity. But invoking
// getVersion() directly here breaks tests that only pretend to create a
// CodeQL object. So instead we rely on the assumption that all non-test
// callers would set checkVersion to true, and util.codeQlVersionAbove()
// would call getVersion(), so the CLI version would be cached as soon as the
// CodeQL object is created.
2021-08-18 10:15:48 +01:00
if (checkVersion &&
!(await util.codeQlVersionAtLeast(codeql, CODEQL_MINIMUM_VERSION))) {
2024-02-08 09:20:03 -08:00
throw new util.ConfigurationError(`Expected a CodeQL CLI with version at least ${CODEQL_MINIMUM_VERSION} but got version ${(await codeql.getVersion()).version}`);
2021-08-18 10:15:48 +01:00
}
else if (checkVersion &&
process.env[environment_1.EnvVar.SUPPRESS_DEPRECATED_SOON_WARNING] !== "true" &&
!(await util.codeQlVersionAtLeast(codeql, CODEQL_NEXT_MINIMUM_VERSION))) {
2023-10-04 11:28:28 +01:00
const result = await codeql.getVersion();
core.warning(`CodeQL CLI version ${result.version} was discontinued on ` +
2023-09-14 16:34:07 +01:00
`${GHES_MOST_RECENT_DEPRECATION_DATE} alongside GitHub Enterprise Server ` +
`${GHES_VERSION_MOST_RECENTLY_DEPRECATED} and will not be supported by the next minor ` +
`release of the CodeQL Action. Please update to CodeQL CLI version ` +
`${CODEQL_NEXT_MINIMUM_VERSION} or later. For instance, if you have specified a custom ` +
"version of the CLI using the 'tools' input to the 'init' Action, you can remove this " +
"input to use the default version.\n\n" +
"Alternatively, if you want to continue using CodeQL CLI version " +
`${result.version}, you can replace 'github/codeql-action/*@v${(0, actions_util_1.getActionVersion)().split(".")[0]}' by 'github/codeql-action/*@v${(0, actions_util_1.getActionVersion)()}' in your code scanning workflow to ` +
2023-09-14 16:34:07 +01:00
"continue using this version of the CodeQL Action.");
core.exportVariable(environment_1.EnvVar.SUPPRESS_DEPRECATED_SOON_WARNING, "true");
}
2021-08-18 10:15:48 +01:00
return codeql;
2020-06-26 17:22:19 +01:00
}
2023-01-09 14:53:52 +00:00
exports.getCodeQLForCmd = getCodeQLForCmd;
2020-08-10 09:25:14 +02:00
/**
* Gets the options for `path` of `options` as an array of extra option strings.
2024-04-16 18:08:46 +01:00
*
* @param ignoringOptions Options that should be ignored, for example because they have already
* been passed and it is an error to pass them more than once.
2020-08-10 09:25:14 +02:00
*/
2024-04-16 18:08:46 +01:00
function getExtraOptionsFromEnv(paths, { ignoringOptions } = {}) {
2020-09-14 10:44:43 +01:00
const options = util.getExtraOptionsEnvParam();
2024-04-16 18:08:46 +01:00
return getExtraOptions(options, paths, []).filter((option) => !ignoringOptions?.includes(option));
2020-11-19 23:03:45 +01:00
}
/**
* Gets `options` as an array of extra option strings.
*
* - throws an exception mentioning `pathInfo` if this conversion is impossible.
*/
function asExtraOptions(options, pathInfo) {
if (options === undefined) {
return [];
}
if (!Array.isArray(options)) {
const msg = `The extra options for '${pathInfo.join(".")}' ('${JSON.stringify(options)}') are not in an array.`;
throw new Error(msg);
}
return options.map((o) => {
const t = typeof o;
if (t !== "string" && t !== "number" && t !== "boolean") {
const msg = `The extra option for '${pathInfo.join(".")}' ('${JSON.stringify(o)}') is not a primitive value.`;
throw new Error(msg);
}
return `${o}`;
});
2020-08-10 09:25:14 +02:00
}
/**
* Gets the options for `path` of `options` as an array of extra option strings.
*
* - the special terminal step name '*' in `options` matches all path steps
* - throws an exception if this conversion is impossible.
*
* Exported for testing.
2020-08-10 09:25:14 +02:00
*/
2020-11-19 23:03:45 +01:00
function getExtraOptions(options, paths, pathInfo) {
2023-01-18 20:00:33 +00:00
const all = asExtraOptions(options?.["*"], pathInfo.concat("*"));
2020-11-19 23:03:45 +01:00
const specific = paths.length === 0
2020-09-14 10:44:43 +01:00
? asExtraOptions(options, pathInfo)
2023-01-18 20:00:33 +00:00
: getExtraOptions(options?.[paths[0]], paths?.slice(1), pathInfo.concat(paths[0]));
2020-08-10 09:25:14 +02:00
return all.concat(specific);
}
exports.getExtraOptions = getExtraOptions;
2021-06-24 17:30:42 +01:00
/*
* A constant defining the maximum number of characters we will keep from
* the programs stderr for logging. This serves two purposes:
* (1) It avoids an OOM if a program fails in a way that results it
* printing many log lines.
* (2) It avoids us hitting the limit of how much data we can send in our
* status reports on GitHub.com.
*/
2024-03-12 07:27:21 -07:00
const maxErrorSize = 20_000;
2023-01-10 12:17:26 -08:00
async function runTool(cmd, args = [], opts = {}) {
let stdout = "";
let stderr = "";
2023-08-10 15:29:57 +01:00
process.stdout.write(`[command]${cmd} ${args.join(" ")}\n`);
2021-06-24 17:30:42 +01:00
const exitCode = await new toolrunner.ToolRunner(cmd, args, {
2023-07-28 12:04:12 +01:00
ignoreReturnCode: true,
2021-06-09 13:17:25 -07:00
listeners: {
stdout: (data) => {
stdout += data.toString("utf8");
2023-07-28 12:04:12 +01:00
if (!opts.noStreamStdout) {
process.stdout.write(data);
}
2021-06-09 13:17:25 -07:00
},
2021-06-24 17:30:42 +01:00
stderr: (data) => {
let readStartIndex = 0;
// If the error is too large, then we only take the last 20,000 characters
if (data.length - maxErrorSize > 0) {
// Eg: if we have 20,000 the start index should be 2.
readStartIndex = data.length - maxErrorSize + 1;
}
stderr += data.toString("utf8", readStartIndex);
2023-07-28 12:04:12 +01:00
// Mimic the standard behavior of the toolrunner by writing stderr to stdout
process.stdout.write(data);
2021-06-24 17:30:42 +01:00
},
2021-06-09 13:17:25 -07:00
},
2023-07-28 12:04:12 +01:00
silent: true,
2023-01-10 12:17:26 -08:00
...(opts.stdin ? { input: Buffer.from(opts.stdin || "") } : {}),
2021-06-09 13:17:25 -07:00
}).exec();
2023-07-20 18:31:37 +01:00
if (exitCode !== 0) {
const e = new cli_errors_1.CommandInvocationError(cmd, args, exitCode, stderr, stdout);
throw (0, cli_errors_1.wrapCliConfigurationError)(e);
2023-07-20 18:31:37 +01:00
}
return stdout;
2021-06-09 13:17:25 -07:00
}
2022-06-19 16:44:24 -07:00
/**
2024-01-04 14:39:36 +00:00
* Generates a code scanning configuration that is to be used for a scan.
2022-06-19 16:44:24 -07:00
*
* @param codeql The CodeQL object to use.
* @param config The configuration to use.
* @returns the path to the generated user configuration file.
*/
2024-01-04 14:39:36 +00:00
async function generateCodeScanningConfig(config, logger) {
2023-07-19 17:37:43 +01:00
const codeScanningConfigFile = getGeneratedCodeScanningConfigPath(config);
2022-06-19 16:44:24 -07:00
// make a copy so we can modify it
2022-08-11 09:56:08 -07:00
const augmentedConfig = cloneObject(config.originalUserInput);
2022-06-19 16:44:24 -07:00
// Inject the queries from the input
if (config.augmentationProperties.queriesInput) {
if (config.augmentationProperties.queriesInputCombines) {
augmentedConfig.queries = (augmentedConfig.queries || []).concat(config.augmentationProperties.queriesInput);
}
else {
augmentedConfig.queries = config.augmentationProperties.queriesInput;
}
}
2023-01-18 20:00:33 +00:00
if (augmentedConfig.queries?.length === 0) {
2022-06-19 16:44:24 -07:00
delete augmentedConfig.queries;
}
// Inject the packs from the input
if (config.augmentationProperties.packsInput) {
if (config.augmentationProperties.packsInputCombines) {
// At this point, we already know that this is a single-language analysis
if (Array.isArray(augmentedConfig.packs)) {
augmentedConfig.packs = (augmentedConfig.packs || []).concat(config.augmentationProperties.packsInput);
}
else if (!augmentedConfig.packs) {
augmentedConfig.packs = config.augmentationProperties.packsInput;
}
else {
// At this point, we know there is only one language.
// If there were more than one language, an error would already have been thrown.
const language = Object.keys(augmentedConfig.packs)[0];
augmentedConfig.packs[language] = augmentedConfig.packs[language].concat(config.augmentationProperties.packsInput);
}
}
else {
augmentedConfig.packs = config.augmentationProperties.packsInput;
}
}
if (Array.isArray(augmentedConfig.packs) && !augmentedConfig.packs.length) {
delete augmentedConfig.packs;
}
2023-02-09 11:19:27 -08:00
logger.info(`Writing augmented user configuration file to ${codeScanningConfigFile}`);
2023-01-17 20:59:36 -08:00
logger.startGroup("Augmented user configuration file contents");
logger.info(yaml.dump(augmentedConfig));
logger.endGroup();
2023-02-09 11:19:27 -08:00
fs.writeFileSync(codeScanningConfigFile, yaml.dump(augmentedConfig));
return codeScanningConfigFile;
2022-06-19 16:44:24 -07:00
}
2022-08-11 09:56:08 -07:00
function cloneObject(obj) {
return JSON.parse(JSON.stringify(obj));
}
// This constant sets the size of each TRAP cache in megabytes.
const TRAP_CACHE_SIZE_MB = 1024;
async function getTrapCachingExtractorConfigArgs(config) {
const result = [];
for (const language of config.languages)
result.push(await getTrapCachingExtractorConfigArgsForLang(config, language));
return result.flat();
}
exports.getTrapCachingExtractorConfigArgs = getTrapCachingExtractorConfigArgs;
async function getTrapCachingExtractorConfigArgsForLang(config, language) {
const cacheDir = config.trapCaches[language];
if (cacheDir === undefined)
return [];
const write = await (0, actions_util_1.isAnalyzingDefaultBranch)();
return [
`-O=${language}.trap.cache.dir=${cacheDir}`,
`-O=${language}.trap.cache.bound=${TRAP_CACHE_SIZE_MB}`,
`-O=${language}.trap.cache.write=${write}`,
];
}
exports.getTrapCachingExtractorConfigArgsForLang = getTrapCachingExtractorConfigArgsForLang;
2023-07-19 17:37:43 +01:00
/**
* Get the path to the code scanning configuration generated by the CLI.
*
* This will not exist if the configuration is being parsed in the Action.
*/
function getGeneratedCodeScanningConfigPath(config) {
return path.resolve(config.tempDir, "user-config.yaml");
}
exports.getGeneratedCodeScanningConfigPath = getGeneratedCodeScanningConfigPath;
async function isDiagnosticsExportInvalidSarifFixed(codeql) {
return await util.codeQlVersionAtLeast(codeql, exports.CODEQL_VERSION_DIAGNOSTICS_EXPORT_FIXED);
}
async function getLanguageAliasingArguments(codeql) {
if (await util.codeQlVersionAtLeast(codeql, exports.CODEQL_VERSION_LANGUAGE_ALIASING)) {
return ["--extractor-include-aliases"];
}
return [];
}
async function isSublanguageFileCoverageEnabled(config, codeql) {
return (
// Sub-language file coverage is first supported in GHES 3.12.
(config.gitHubVersion.type !== util.GitHubVariant.GHES ||
semver.gte(config.gitHubVersion.version, "3.12.0")) &&
(await util.codeQlVersionAtLeast(codeql, exports.CODEQL_VERSION_SUBLANGUAGE_FILE_COVERAGE)));
}
async function getCodeScanningQueryHelpArguments(codeql) {
if (await util.codeQlVersionAtLeast(codeql, CODEQL_VERSION_INCLUDE_QUERY_HELP)) {
return ["--sarif-include-query-help=always"];
}
return ["--sarif-add-query-help"];
}
function getExtractionVerbosityArguments(enableDebugLogging) {
return enableDebugLogging
? [`--verbosity=${EXTRACTION_DEBUG_MODE_VERBOSITY}`]
: [];
}
/**
* Updates the `JAVA_TOOL_OPTIONS` environment variable to resolve an issue with Azure Pipelines
* timing out connections after 4 minutes and Maven not properly handling closed connections.
*
* Without the fix, long build processes will timeout when pulling down Java packages
* https://developercommunity.visualstudio.com/content/problem/292284/maven-hosted-agent-connection-timeout.html
*/
function applyAutobuildAzurePipelinesTimeoutFix() {
const javaToolOptions = process.env["JAVA_TOOL_OPTIONS"] || "";
process.env["JAVA_TOOL_OPTIONS"] = [
...javaToolOptions.split(/\s+/),
"-Dhttp.keepAlive=false",
"-Dmaven.wagon.http.pool=false",
].join(" ");
}
2020-06-26 17:22:19 +01:00
//# sourceMappingURL=codeql.js.map