mirror of
https://github.com/github/codeql-action.git
synced 2026-04-28 18:08:53 +00:00
227 lines
7.3 KiB
TypeScript
227 lines
7.3 KiB
TypeScript
/**
|
|
* This file is the entry point for the `post:` hook of `init-action.yml`.
|
|
* It will run after the all steps in this job, in reverse order in relation to
|
|
* other `post:` hooks.
|
|
*/
|
|
|
|
import * as core from "@actions/core";
|
|
|
|
import {
|
|
restoreInputs,
|
|
getTemporaryDirectory,
|
|
printDebugLogs,
|
|
} from "./actions-util";
|
|
import { getGitHubVersion } from "./api-client";
|
|
import { CachingKind } from "./caching-utils";
|
|
import { getCodeQL } from "./codeql";
|
|
import { type Config, getConfig } from "./config-utils";
|
|
import * as debugArtifacts from "./debug-artifacts";
|
|
import {
|
|
DependencyCachingUsageReport,
|
|
getDependencyCacheUsage,
|
|
} from "./dependency-caching";
|
|
import { EnvVar } from "./environment";
|
|
import { Features } from "./feature-flags";
|
|
import * as gitUtils from "./git-utils";
|
|
import * as initActionPostHelper from "./init-action-post-helper";
|
|
import { getActionsLogger } from "./logging";
|
|
import { getRepositoryNwo } from "./repository";
|
|
import {
|
|
StatusReportBase,
|
|
sendStatusReport,
|
|
sendUnhandledErrorStatusReport,
|
|
createStatusReportBase,
|
|
getActionsStatus,
|
|
ActionName,
|
|
getJobStatusDisplayName,
|
|
JobStatus,
|
|
} from "./status-report";
|
|
import { checkDiskUsage, checkGitHubVersionInRange, wrapError } from "./util";
|
|
|
|
interface InitPostStatusReport
|
|
extends StatusReportBase,
|
|
initActionPostHelper.UploadFailedSarifResult,
|
|
initActionPostHelper.JobStatusReport,
|
|
initActionPostHelper.DependencyCachingUsageReport {}
|
|
|
|
async function run(startedAt: Date) {
|
|
// To capture errors appropriately, keep as much code within the try-catch as
|
|
// possible, and only use safe functions outside.
|
|
|
|
const logger = getActionsLogger();
|
|
let config: Config | undefined;
|
|
let uploadFailedSarifResult:
|
|
| initActionPostHelper.UploadFailedSarifResult
|
|
| undefined;
|
|
let dependencyCachingUsage: DependencyCachingUsageReport | undefined;
|
|
try {
|
|
// Restore inputs from `init` Action.
|
|
restoreInputs();
|
|
|
|
const gitHubVersion = await getGitHubVersion();
|
|
checkGitHubVersionInRange(gitHubVersion, logger);
|
|
|
|
const repositoryNwo = getRepositoryNwo();
|
|
const features = new Features(
|
|
gitHubVersion,
|
|
repositoryNwo,
|
|
getTemporaryDirectory(),
|
|
logger,
|
|
);
|
|
|
|
config = await getConfig(getTemporaryDirectory(), logger);
|
|
if (config === undefined) {
|
|
logger.warning(
|
|
"Debugging artifacts are unavailable since the 'init' Action failed before it could produce any.",
|
|
);
|
|
} else {
|
|
const codeql = await getCodeQL(config.codeQLCmd);
|
|
|
|
uploadFailedSarifResult = await initActionPostHelper.run(
|
|
debugArtifacts.tryUploadAllAvailableDebugArtifacts,
|
|
printDebugLogs,
|
|
codeql,
|
|
config,
|
|
repositoryNwo,
|
|
features,
|
|
logger,
|
|
);
|
|
|
|
// If we are analyzing the default branch and some kind of caching is enabled,
|
|
// then try to determine our overall cache usage for dependency caches. We only
|
|
// do this under these circumstances to avoid slowing down analyses for PRs
|
|
// and where caching may not be enabled.
|
|
if (
|
|
(await gitUtils.isAnalyzingDefaultBranch()) &&
|
|
config.dependencyCachingEnabled !== CachingKind.None
|
|
) {
|
|
dependencyCachingUsage = await getDependencyCacheUsage(logger);
|
|
}
|
|
}
|
|
} catch (unwrappedError) {
|
|
const error = wrapError(unwrappedError);
|
|
core.setFailed(error.message);
|
|
|
|
const statusReportBase = await createStatusReportBase(
|
|
ActionName.InitPost,
|
|
getActionsStatus(error),
|
|
startedAt,
|
|
config,
|
|
await checkDiskUsage(logger),
|
|
logger,
|
|
error.message,
|
|
error.stack,
|
|
);
|
|
if (statusReportBase !== undefined) {
|
|
await sendStatusReport(statusReportBase);
|
|
}
|
|
return;
|
|
}
|
|
const jobStatus = getFinalJobStatus(config);
|
|
logger.info(`CodeQL job status was ${getJobStatusDisplayName(jobStatus)}.`);
|
|
|
|
const statusReportBase = await createStatusReportBase(
|
|
ActionName.InitPost,
|
|
"success",
|
|
startedAt,
|
|
config,
|
|
await checkDiskUsage(logger),
|
|
logger,
|
|
);
|
|
if (statusReportBase !== undefined) {
|
|
const statusReport: InitPostStatusReport = {
|
|
...statusReportBase,
|
|
...uploadFailedSarifResult,
|
|
job_status: jobStatus,
|
|
dependency_caching_usage: dependencyCachingUsage,
|
|
};
|
|
logger.info("Sending status report for init-post step.");
|
|
await sendStatusReport(statusReport);
|
|
logger.info("Status report sent for init-post step.");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Determine the final job status to be reported in the status report.
|
|
*
|
|
* If the job status has already been set by another step, we use that.
|
|
* Otherwise, we determine the job status based on whether the analyze step
|
|
* completed successfully and whether we have a valid CodeQL config.
|
|
*/
|
|
function getFinalJobStatus(config: Config | undefined): JobStatus {
|
|
const existingJobStatus = getJobStatusFromEnvironment();
|
|
if (existingJobStatus !== undefined) {
|
|
return existingJobStatus;
|
|
}
|
|
|
|
let jobStatus: JobStatus;
|
|
|
|
if (process.env[EnvVar.ANALYZE_DID_COMPLETE_SUCCESSFULLY] === "true") {
|
|
core.exportVariable(EnvVar.JOB_STATUS, JobStatus.SuccessStatus);
|
|
jobStatus = JobStatus.SuccessStatus;
|
|
} else if (config !== undefined) {
|
|
// - We have computed a CodeQL config
|
|
// - Analyze didn't complete successfully
|
|
// - The job status hasn't already been set to Failure/ConfigurationError
|
|
//
|
|
// This means that something along the way failed in a step that is not
|
|
// owned by the Action, for example a manual build step. We consider this a
|
|
// configuration error.
|
|
jobStatus = JobStatus.ConfigErrorStatus;
|
|
} else {
|
|
// If we didn't manage to compute a CodeQL config, it is unclear at this
|
|
// point why the analyze Action didn't complete.
|
|
// - One possibility is that the workflow run was cancelled. We could
|
|
// consider determining workflow cancellation using the GitHub API, but
|
|
// for now we treat all these cases as unknown.
|
|
// - Another possibility is that we're running a workflow that only runs
|
|
// `init`, for instance a workflow that was created before `setup-codeql`
|
|
// was available and uses `init` just to set up the CodeQL tools.
|
|
jobStatus = JobStatus.UnknownStatus;
|
|
}
|
|
|
|
// This shouldn't be necessary, but in the odd case that we run more than one
|
|
// `init` post step, ensure the job status is consistent between them.
|
|
core.exportVariable(EnvVar.JOB_STATUS, jobStatus);
|
|
return jobStatus;
|
|
}
|
|
|
|
/**
|
|
* Get the job status from the environment variable, if it has been set.
|
|
*
|
|
* If the job status is invalid, return `UnknownStatus`.
|
|
*/
|
|
function getJobStatusFromEnvironment(): JobStatus | undefined {
|
|
const jobStatusFromEnvironment = process.env[EnvVar.JOB_STATUS];
|
|
|
|
if (jobStatusFromEnvironment !== undefined) {
|
|
// Validate the job status from the environment. If it is invalid, return unknown.
|
|
if (
|
|
Object.values(JobStatus).includes(jobStatusFromEnvironment as JobStatus)
|
|
) {
|
|
return jobStatusFromEnvironment as JobStatus;
|
|
}
|
|
return JobStatus.UnknownStatus;
|
|
}
|
|
|
|
return undefined;
|
|
}
|
|
|
|
async function runWrapper() {
|
|
const startedAt = new Date();
|
|
const logger = getActionsLogger();
|
|
try {
|
|
await run(startedAt);
|
|
} catch (error) {
|
|
core.setFailed(`init post action failed: ${wrapError(error).message}`);
|
|
await sendUnhandledErrorStatusReport(
|
|
ActionName.InitPost,
|
|
startedAt,
|
|
error,
|
|
logger,
|
|
);
|
|
}
|
|
}
|
|
|
|
void runWrapper();
|