717 lines
22 KiB
TypeScript
717 lines
22 KiB
TypeScript
import { join } from "path";
|
|
|
|
import {
|
|
submitVariantAnalysis,
|
|
getVariantAnalysisRepo,
|
|
} from "./gh-api/gh-api-client";
|
|
import {
|
|
CancellationToken,
|
|
commands,
|
|
env,
|
|
EventEmitter,
|
|
ExtensionContext,
|
|
Uri,
|
|
ViewColumn,
|
|
window as Window,
|
|
workspace,
|
|
} from "vscode";
|
|
import { DisposableObject } from "../pure/disposable-object";
|
|
import { VariantAnalysisMonitor } from "./variant-analysis-monitor";
|
|
import {
|
|
getActionsWorkflowRunUrl,
|
|
isVariantAnalysisComplete,
|
|
parseVariantAnalysisQueryLanguage,
|
|
VariantAnalysis,
|
|
VariantAnalysisRepositoryTask,
|
|
VariantAnalysisScannedRepository,
|
|
VariantAnalysisScannedRepositoryDownloadStatus,
|
|
VariantAnalysisScannedRepositoryResult,
|
|
VariantAnalysisScannedRepositoryState,
|
|
VariantAnalysisSubmission,
|
|
} from "./shared/variant-analysis";
|
|
import { getErrorMessage } from "../pure/helpers-pure";
|
|
import { VariantAnalysisView } from "./variant-analysis-view";
|
|
import { VariantAnalysisViewManager } from "./variant-analysis-view-manager";
|
|
import {
|
|
LoadResultsOptions,
|
|
VariantAnalysisResultsManager,
|
|
} from "./variant-analysis-results-manager";
|
|
import { getQueryName, prepareRemoteQueryRun } from "./run-remote-query";
|
|
import {
|
|
processVariantAnalysis,
|
|
processVariantAnalysisRepositoryTask,
|
|
} from "./variant-analysis-processor";
|
|
import PQueue from "p-queue";
|
|
import {
|
|
createTimestampFile,
|
|
showAndLogExceptionWithTelemetry,
|
|
showAndLogInformationMessage,
|
|
showAndLogWarningMessage,
|
|
} from "../helpers";
|
|
import { readFile, readJson, remove, pathExists, outputJson } from "fs-extra";
|
|
import { EOL } from "os";
|
|
import { cancelVariantAnalysis } from "./gh-api/gh-actions-api-client";
|
|
import {
|
|
ProgressCallback,
|
|
UserCancellationException,
|
|
withProgress,
|
|
} from "../commandRunner";
|
|
import { CodeQLCliServer } from "../cli";
|
|
import {
|
|
defaultFilterSortState,
|
|
filterAndSortRepositoriesWithResults,
|
|
RepositoriesFilterSortStateWithIds,
|
|
} from "../pure/variant-analysis-filter-sort";
|
|
import { URLSearchParams } from "url";
|
|
import { DbManager } from "../databases/db-manager";
|
|
import { App } from "../common/app";
|
|
import { redactableError } from "../pure/errors";
|
|
import { AppCommandManager, VariantAnalysisCommands } from "../common/commands";
|
|
import { exportVariantAnalysisResults } from "./export-results";
|
|
|
|
export class VariantAnalysisManager
|
|
extends DisposableObject
|
|
implements VariantAnalysisViewManager<VariantAnalysisView>
|
|
{
|
|
private static readonly REPO_STATES_FILENAME = "repo_states.json";
|
|
private static readonly DOWNLOAD_PERCENTAGE_UPDATE_DELAY_MS = 500;
|
|
|
|
private readonly _onVariantAnalysisAdded = this.push(
|
|
new EventEmitter<VariantAnalysis>(),
|
|
);
|
|
public readonly onVariantAnalysisAdded = this._onVariantAnalysisAdded.event;
|
|
private readonly _onVariantAnalysisStatusUpdated = this.push(
|
|
new EventEmitter<VariantAnalysis>(),
|
|
);
|
|
public readonly onVariantAnalysisStatusUpdated =
|
|
this._onVariantAnalysisStatusUpdated.event;
|
|
|
|
private readonly _onVariantAnalysisRemoved = this.push(
|
|
new EventEmitter<VariantAnalysis>(),
|
|
);
|
|
public readonly onVariantAnalysisRemoved =
|
|
this._onVariantAnalysisRemoved.event;
|
|
|
|
private readonly variantAnalysisMonitor: VariantAnalysisMonitor;
|
|
private readonly variantAnalyses = new Map<number, VariantAnalysis>();
|
|
private readonly views = new Map<number, VariantAnalysisView>();
|
|
private static readonly maxConcurrentDownloads = 3;
|
|
private readonly queue = new PQueue({
|
|
concurrency: VariantAnalysisManager.maxConcurrentDownloads,
|
|
});
|
|
|
|
private readonly repoStates = new Map<
|
|
number,
|
|
Record<number, VariantAnalysisScannedRepositoryState>
|
|
>();
|
|
|
|
constructor(
|
|
private readonly ctx: ExtensionContext,
|
|
private readonly app: App,
|
|
private readonly cliServer: CodeQLCliServer,
|
|
private readonly storagePath: string,
|
|
private readonly variantAnalysisResultsManager: VariantAnalysisResultsManager,
|
|
private readonly dbManager: DbManager,
|
|
) {
|
|
super();
|
|
this.variantAnalysisMonitor = this.push(
|
|
new VariantAnalysisMonitor(
|
|
this.shouldCancelMonitorVariantAnalysis.bind(this),
|
|
),
|
|
);
|
|
this.variantAnalysisMonitor.onVariantAnalysisChange(
|
|
this.onVariantAnalysisUpdated.bind(this),
|
|
);
|
|
|
|
this.variantAnalysisResultsManager = variantAnalysisResultsManager;
|
|
this.variantAnalysisResultsManager.onResultLoaded(
|
|
this.onRepoResultLoaded.bind(this),
|
|
);
|
|
}
|
|
|
|
getCommands(): VariantAnalysisCommands {
|
|
return {
|
|
"codeQL.autoDownloadVariantAnalysisResult":
|
|
this.enqueueDownload.bind(this),
|
|
"codeQL.copyVariantAnalysisRepoList":
|
|
this.copyRepoListToClipboard.bind(this),
|
|
"codeQL.loadVariantAnalysisRepoResults": this.loadResults.bind(this),
|
|
"codeQL.monitorVariantAnalysis": this.monitorVariantAnalysis.bind(this),
|
|
"codeQL.openVariantAnalysisLogs": this.openVariantAnalysisLogs.bind(this),
|
|
"codeQL.openVariantAnalysisView": this.showView.bind(this),
|
|
"codeQL.runVariantAnalysis":
|
|
this.runVariantAnalysisFromCommand.bind(this),
|
|
// Since we are tracking extension usage through commands, this command mirrors the "codeQL.runVariantAnalysis" command
|
|
"codeQL.runVariantAnalysisContextEditor":
|
|
this.runVariantAnalysisFromCommand.bind(this),
|
|
};
|
|
}
|
|
|
|
get commandManager(): AppCommandManager {
|
|
return this.app.commands;
|
|
}
|
|
|
|
private async runVariantAnalysisFromCommand(uri?: Uri) {
|
|
return withProgress(
|
|
async (progress, token) =>
|
|
this.runVariantAnalysis(
|
|
uri || Window.activeTextEditor?.document.uri,
|
|
progress,
|
|
token,
|
|
),
|
|
{
|
|
title: "Run Variant Analysis",
|
|
cancellable: true,
|
|
},
|
|
);
|
|
}
|
|
|
|
public async runVariantAnalysis(
|
|
uri: Uri | undefined,
|
|
progress: ProgressCallback,
|
|
token: CancellationToken,
|
|
): Promise<void> {
|
|
progress({
|
|
maxStep: 5,
|
|
step: 0,
|
|
message: "Getting credentials",
|
|
});
|
|
|
|
const {
|
|
actionBranch,
|
|
base64Pack,
|
|
repoSelection,
|
|
queryFile,
|
|
queryMetadata,
|
|
controllerRepo,
|
|
queryStartTime,
|
|
language,
|
|
} = await prepareRemoteQueryRun(
|
|
this.cliServer,
|
|
this.app.credentials,
|
|
uri,
|
|
progress,
|
|
token,
|
|
this.dbManager,
|
|
);
|
|
|
|
const queryName = getQueryName(queryMetadata, queryFile);
|
|
const variantAnalysisLanguage = parseVariantAnalysisQueryLanguage(language);
|
|
if (variantAnalysisLanguage === undefined) {
|
|
throw new UserCancellationException(
|
|
`Found unsupported language: ${language}`,
|
|
);
|
|
}
|
|
|
|
const queryText = await readFile(queryFile, "utf8");
|
|
|
|
const variantAnalysisSubmission: VariantAnalysisSubmission = {
|
|
startTime: queryStartTime,
|
|
actionRepoRef: actionBranch,
|
|
controllerRepoId: controllerRepo.id,
|
|
query: {
|
|
name: queryName,
|
|
filePath: queryFile,
|
|
pack: base64Pack,
|
|
language: variantAnalysisLanguage,
|
|
text: queryText,
|
|
},
|
|
databases: {
|
|
repositories: repoSelection.repositories,
|
|
repositoryLists: repoSelection.repositoryLists,
|
|
repositoryOwners: repoSelection.owners,
|
|
},
|
|
};
|
|
|
|
const variantAnalysisResponse = await submitVariantAnalysis(
|
|
this.app.credentials,
|
|
variantAnalysisSubmission,
|
|
);
|
|
|
|
const processedVariantAnalysis = processVariantAnalysis(
|
|
variantAnalysisSubmission,
|
|
variantAnalysisResponse,
|
|
);
|
|
|
|
await this.onVariantAnalysisSubmitted(processedVariantAnalysis);
|
|
|
|
void showAndLogInformationMessage(
|
|
`Variant analysis ${processedVariantAnalysis.query.name} submitted for processing`,
|
|
);
|
|
|
|
void commands.executeCommand(
|
|
"codeQL.openVariantAnalysisView",
|
|
processedVariantAnalysis.id,
|
|
);
|
|
void commands.executeCommand(
|
|
"codeQL.monitorVariantAnalysis",
|
|
processedVariantAnalysis,
|
|
);
|
|
}
|
|
|
|
public async rehydrateVariantAnalysis(variantAnalysis: VariantAnalysis) {
|
|
if (!(await this.variantAnalysisRecordExists(variantAnalysis.id))) {
|
|
// In this case, the variant analysis was deleted from disk, most likely because
|
|
// it was purged by another workspace.
|
|
this._onVariantAnalysisRemoved.fire(variantAnalysis);
|
|
} else {
|
|
await this.setVariantAnalysis(variantAnalysis);
|
|
|
|
try {
|
|
const repoStates = await readJson(
|
|
this.getRepoStatesStoragePath(variantAnalysis.id),
|
|
);
|
|
this.repoStates.set(variantAnalysis.id, repoStates);
|
|
} catch (e) {
|
|
// Ignore this error, we simply might not have downloaded anything yet
|
|
this.repoStates.set(variantAnalysis.id, {});
|
|
}
|
|
|
|
if (
|
|
!(await isVariantAnalysisComplete(
|
|
variantAnalysis,
|
|
this.makeResultDownloadChecker(variantAnalysis),
|
|
))
|
|
) {
|
|
void commands.executeCommand(
|
|
"codeQL.monitorVariantAnalysis",
|
|
variantAnalysis,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
private makeResultDownloadChecker(
|
|
variantAnalysis: VariantAnalysis,
|
|
): (repo: VariantAnalysisScannedRepository) => Promise<boolean> {
|
|
const storageLocation = this.getVariantAnalysisStorageLocation(
|
|
variantAnalysis.id,
|
|
);
|
|
return (repo) =>
|
|
this.variantAnalysisResultsManager.isVariantAnalysisRepoDownloaded(
|
|
storageLocation,
|
|
repo.repository.fullName,
|
|
);
|
|
}
|
|
|
|
public async removeVariantAnalysis(variantAnalysis: VariantAnalysis) {
|
|
this.variantAnalysisResultsManager.removeAnalysisResults(variantAnalysis);
|
|
await this.removeStorageDirectory(variantAnalysis.id);
|
|
this.variantAnalyses.delete(variantAnalysis.id);
|
|
|
|
// This will automatically unregister the view
|
|
this.views.get(variantAnalysis.id)?.dispose();
|
|
}
|
|
|
|
private async removeStorageDirectory(variantAnalysisId: number) {
|
|
const storageLocation =
|
|
this.getVariantAnalysisStorageLocation(variantAnalysisId);
|
|
await remove(storageLocation);
|
|
}
|
|
|
|
public async showView(variantAnalysisId: number): Promise<void> {
|
|
if (!this.variantAnalyses.get(variantAnalysisId)) {
|
|
void showAndLogExceptionWithTelemetry(
|
|
redactableError`No variant analysis found with id: ${variantAnalysisId}.`,
|
|
);
|
|
}
|
|
if (!this.views.has(variantAnalysisId)) {
|
|
// The view will register itself with the manager, so we don't need to do anything here.
|
|
this.track(new VariantAnalysisView(this.ctx, variantAnalysisId, this));
|
|
}
|
|
|
|
const variantAnalysisView = this.views.get(variantAnalysisId)!;
|
|
await variantAnalysisView.openView();
|
|
return;
|
|
}
|
|
|
|
public async openQueryText(variantAnalysisId: number): Promise<void> {
|
|
const variantAnalysis = await this.getVariantAnalysis(variantAnalysisId);
|
|
if (!variantAnalysis) {
|
|
void showAndLogWarningMessage(
|
|
"Could not open variant analysis query text. Variant analysis not found.",
|
|
);
|
|
return;
|
|
}
|
|
|
|
const filename = variantAnalysis.query.filePath;
|
|
|
|
try {
|
|
const params = new URLSearchParams({
|
|
variantAnalysisId: variantAnalysis.id.toString(),
|
|
});
|
|
const uri = Uri.from({
|
|
scheme: "codeql-variant-analysis",
|
|
path: filename,
|
|
query: params.toString(),
|
|
});
|
|
const doc = await workspace.openTextDocument(uri);
|
|
await Window.showTextDocument(doc, { preview: false });
|
|
} catch (error) {
|
|
void showAndLogWarningMessage(
|
|
"Could not open variant analysis query text. Failed to open text document.",
|
|
);
|
|
}
|
|
}
|
|
|
|
public async openQueryFile(variantAnalysisId: number): Promise<void> {
|
|
const variantAnalysis = await this.getVariantAnalysis(variantAnalysisId);
|
|
|
|
if (!variantAnalysis) {
|
|
void showAndLogWarningMessage(
|
|
"Could not open variant analysis query file",
|
|
);
|
|
return;
|
|
}
|
|
|
|
try {
|
|
const textDocument = await workspace.openTextDocument(
|
|
variantAnalysis.query.filePath,
|
|
);
|
|
await Window.showTextDocument(textDocument, ViewColumn.One);
|
|
} catch (error) {
|
|
void showAndLogWarningMessage(
|
|
`Could not open file: ${variantAnalysis.query.filePath}`,
|
|
);
|
|
}
|
|
}
|
|
|
|
public registerView(view: VariantAnalysisView): void {
|
|
if (this.views.has(view.variantAnalysisId)) {
|
|
throw new Error(
|
|
`View for variant analysis with id: ${view.variantAnalysisId} already exists`,
|
|
);
|
|
}
|
|
|
|
this.views.set(view.variantAnalysisId, view);
|
|
}
|
|
|
|
public unregisterView(view: VariantAnalysisView): void {
|
|
this.views.delete(view.variantAnalysisId);
|
|
this.disposeAndStopTracking(view);
|
|
}
|
|
|
|
public getView(variantAnalysisId: number): VariantAnalysisView | undefined {
|
|
return this.views.get(variantAnalysisId);
|
|
}
|
|
|
|
public async getVariantAnalysis(
|
|
variantAnalysisId: number,
|
|
): Promise<VariantAnalysis | undefined> {
|
|
return this.variantAnalyses.get(variantAnalysisId);
|
|
}
|
|
|
|
public async getRepoStates(
|
|
variantAnalysisId: number,
|
|
): Promise<VariantAnalysisScannedRepositoryState[]> {
|
|
return Object.values(this.repoStates.get(variantAnalysisId) ?? {});
|
|
}
|
|
|
|
public get variantAnalysesSize(): number {
|
|
return this.variantAnalyses.size;
|
|
}
|
|
|
|
public async loadResults(
|
|
variantAnalysisId: number,
|
|
repositoryFullName: string,
|
|
options?: LoadResultsOptions,
|
|
): Promise<VariantAnalysisScannedRepositoryResult> {
|
|
const variantAnalysis = this.variantAnalyses.get(variantAnalysisId);
|
|
if (!variantAnalysis) {
|
|
throw new Error(`No variant analysis with id: ${variantAnalysisId}`);
|
|
}
|
|
|
|
return this.variantAnalysisResultsManager.loadResults(
|
|
variantAnalysisId,
|
|
this.getVariantAnalysisStorageLocation(variantAnalysisId),
|
|
repositoryFullName,
|
|
options,
|
|
);
|
|
}
|
|
|
|
private async variantAnalysisRecordExists(
|
|
variantAnalysisId: number,
|
|
): Promise<boolean> {
|
|
const filePath = this.getVariantAnalysisStorageLocation(variantAnalysisId);
|
|
return await pathExists(filePath);
|
|
}
|
|
|
|
private async shouldCancelMonitorVariantAnalysis(
|
|
variantAnalysisId: number,
|
|
): Promise<boolean> {
|
|
return !this.variantAnalyses.has(variantAnalysisId);
|
|
}
|
|
|
|
public async onVariantAnalysisUpdated(
|
|
variantAnalysis: VariantAnalysis | undefined,
|
|
): Promise<void> {
|
|
if (!variantAnalysis) {
|
|
return;
|
|
}
|
|
|
|
if (!this.variantAnalyses.has(variantAnalysis.id)) {
|
|
return;
|
|
}
|
|
|
|
await this.setVariantAnalysis(variantAnalysis);
|
|
this._onVariantAnalysisStatusUpdated.fire(variantAnalysis);
|
|
}
|
|
|
|
private async onVariantAnalysisSubmitted(
|
|
variantAnalysis: VariantAnalysis,
|
|
): Promise<void> {
|
|
await this.setVariantAnalysis(variantAnalysis);
|
|
|
|
await this.prepareStorageDirectory(variantAnalysis.id);
|
|
|
|
this.repoStates.set(variantAnalysis.id, {});
|
|
|
|
this._onVariantAnalysisAdded.fire(variantAnalysis);
|
|
}
|
|
|
|
private async setVariantAnalysis(
|
|
variantAnalysis: VariantAnalysis,
|
|
): Promise<void> {
|
|
this.variantAnalyses.set(variantAnalysis.id, variantAnalysis);
|
|
await this.getView(variantAnalysis.id)?.updateView(variantAnalysis);
|
|
}
|
|
|
|
private async onRepoResultLoaded(
|
|
repositoryResult: VariantAnalysisScannedRepositoryResult,
|
|
): Promise<void> {
|
|
await this.getView(
|
|
repositoryResult.variantAnalysisId,
|
|
)?.sendRepositoryResults([repositoryResult]);
|
|
}
|
|
|
|
private async onRepoStateUpdated(
|
|
variantAnalysisId: number,
|
|
repoState: VariantAnalysisScannedRepositoryState,
|
|
): Promise<void> {
|
|
await this.getView(variantAnalysisId)?.updateRepoState(repoState);
|
|
|
|
let repoStates = this.repoStates.get(variantAnalysisId);
|
|
if (!repoStates) {
|
|
repoStates = {};
|
|
this.repoStates.set(variantAnalysisId, repoStates);
|
|
}
|
|
|
|
repoStates[repoState.repositoryId] = repoState;
|
|
}
|
|
|
|
public async monitorVariantAnalysis(
|
|
variantAnalysis: VariantAnalysis,
|
|
): Promise<void> {
|
|
await this.variantAnalysisMonitor.monitorVariantAnalysis(
|
|
variantAnalysis,
|
|
this.app.credentials,
|
|
);
|
|
}
|
|
|
|
public async autoDownloadVariantAnalysisResult(
|
|
scannedRepo: VariantAnalysisScannedRepository,
|
|
variantAnalysis: VariantAnalysis,
|
|
): Promise<void> {
|
|
if (
|
|
this.repoStates.get(variantAnalysis.id)?.[scannedRepo.repository.id]
|
|
?.downloadStatus ===
|
|
VariantAnalysisScannedRepositoryDownloadStatus.Succeeded
|
|
) {
|
|
return;
|
|
}
|
|
|
|
const repoState = {
|
|
repositoryId: scannedRepo.repository.id,
|
|
downloadStatus: VariantAnalysisScannedRepositoryDownloadStatus.Pending,
|
|
};
|
|
|
|
await this.onRepoStateUpdated(variantAnalysis.id, repoState);
|
|
|
|
let repoTask: VariantAnalysisRepositoryTask;
|
|
try {
|
|
const repoTaskResponse = await getVariantAnalysisRepo(
|
|
this.app.credentials,
|
|
variantAnalysis.controllerRepo.id,
|
|
variantAnalysis.id,
|
|
scannedRepo.repository.id,
|
|
);
|
|
|
|
repoTask = processVariantAnalysisRepositoryTask(repoTaskResponse);
|
|
} catch (e) {
|
|
repoState.downloadStatus =
|
|
VariantAnalysisScannedRepositoryDownloadStatus.Failed;
|
|
await this.onRepoStateUpdated(variantAnalysis.id, repoState);
|
|
throw new Error(
|
|
`Could not download the results for variant analysis with id: ${
|
|
variantAnalysis.id
|
|
}. Error: ${getErrorMessage(e)}`,
|
|
);
|
|
}
|
|
|
|
if (repoTask.artifactUrl) {
|
|
repoState.downloadStatus =
|
|
VariantAnalysisScannedRepositoryDownloadStatus.InProgress;
|
|
await this.onRepoStateUpdated(variantAnalysis.id, repoState);
|
|
|
|
try {
|
|
let lastRepoStateUpdate = 0;
|
|
const updateRepoStateCallback = async (downloadPercentage: number) => {
|
|
const now = new Date().getTime();
|
|
if (
|
|
lastRepoStateUpdate <
|
|
now - VariantAnalysisManager.DOWNLOAD_PERCENTAGE_UPDATE_DELAY_MS
|
|
) {
|
|
lastRepoStateUpdate = now;
|
|
await this.onRepoStateUpdated(variantAnalysis.id, {
|
|
repositoryId: scannedRepo.repository.id,
|
|
downloadStatus:
|
|
VariantAnalysisScannedRepositoryDownloadStatus.InProgress,
|
|
downloadPercentage,
|
|
});
|
|
}
|
|
};
|
|
await this.variantAnalysisResultsManager.download(
|
|
variantAnalysis.id,
|
|
repoTask,
|
|
this.getVariantAnalysisStorageLocation(variantAnalysis.id),
|
|
updateRepoStateCallback,
|
|
);
|
|
} catch (e) {
|
|
repoState.downloadStatus =
|
|
VariantAnalysisScannedRepositoryDownloadStatus.Failed;
|
|
await this.onRepoStateUpdated(variantAnalysis.id, repoState);
|
|
throw new Error(
|
|
`Could not download the results for variant analysis with id: ${
|
|
variantAnalysis.id
|
|
}. Error: ${getErrorMessage(e)}`,
|
|
);
|
|
}
|
|
}
|
|
|
|
repoState.downloadStatus =
|
|
VariantAnalysisScannedRepositoryDownloadStatus.Succeeded;
|
|
await this.onRepoStateUpdated(variantAnalysis.id, repoState);
|
|
|
|
await outputJson(
|
|
this.getRepoStatesStoragePath(variantAnalysis.id),
|
|
this.repoStates.get(variantAnalysis.id),
|
|
);
|
|
}
|
|
|
|
public async enqueueDownload(
|
|
scannedRepo: VariantAnalysisScannedRepository,
|
|
variantAnalysis: VariantAnalysis,
|
|
): Promise<void> {
|
|
await this.queue.add(() =>
|
|
this.autoDownloadVariantAnalysisResult(scannedRepo, variantAnalysis),
|
|
);
|
|
}
|
|
|
|
public downloadsQueueSize(): number {
|
|
return this.queue.pending;
|
|
}
|
|
|
|
public getVariantAnalysisStorageLocation(variantAnalysisId: number): string {
|
|
return join(this.storagePath, `${variantAnalysisId}`);
|
|
}
|
|
|
|
public async cancelVariantAnalysis(variantAnalysisId: number) {
|
|
const variantAnalysis = this.variantAnalyses.get(variantAnalysisId);
|
|
if (!variantAnalysis) {
|
|
throw new Error(`No variant analysis with id: ${variantAnalysisId}`);
|
|
}
|
|
|
|
if (!variantAnalysis.actionsWorkflowRunId) {
|
|
throw new Error(
|
|
`No workflow run id for variant analysis with id: ${variantAnalysis.id}`,
|
|
);
|
|
}
|
|
|
|
void showAndLogInformationMessage(
|
|
"Cancelling variant analysis. This may take a while.",
|
|
);
|
|
await cancelVariantAnalysis(this.app.credentials, variantAnalysis);
|
|
}
|
|
|
|
public async openVariantAnalysisLogs(variantAnalysisId: number) {
|
|
const variantAnalysis = this.variantAnalyses.get(variantAnalysisId);
|
|
if (!variantAnalysis) {
|
|
throw new Error(`No variant analysis with id: ${variantAnalysisId}`);
|
|
}
|
|
|
|
const actionsWorkflowRunUrl = getActionsWorkflowRunUrl(variantAnalysis);
|
|
|
|
await commands.executeCommand(
|
|
"vscode.open",
|
|
Uri.parse(actionsWorkflowRunUrl),
|
|
);
|
|
}
|
|
|
|
public async copyRepoListToClipboard(
|
|
variantAnalysisId: number,
|
|
filterSort: RepositoriesFilterSortStateWithIds = defaultFilterSortState,
|
|
) {
|
|
const variantAnalysis = this.variantAnalyses.get(variantAnalysisId);
|
|
if (!variantAnalysis) {
|
|
throw new Error(`No variant analysis with id: ${variantAnalysisId}`);
|
|
}
|
|
|
|
const filteredRepositories = filterAndSortRepositoriesWithResults(
|
|
variantAnalysis.scannedRepos,
|
|
filterSort,
|
|
);
|
|
|
|
const fullNames = filteredRepositories
|
|
?.filter((a) => a.resultCount && a.resultCount > 0)
|
|
.map((a) => a.repository.fullName);
|
|
if (!fullNames || fullNames.length === 0) {
|
|
return;
|
|
}
|
|
|
|
const text = [
|
|
"{",
|
|
` "name": "new-repo-list",`,
|
|
` "repositories": [`,
|
|
...fullNames.slice(0, -1).map((repo) => ` "${repo}",`),
|
|
` "${fullNames[fullNames.length - 1]}"`,
|
|
` ]`,
|
|
"}",
|
|
];
|
|
|
|
await env.clipboard.writeText(text.join(EOL));
|
|
}
|
|
|
|
public async exportResults(
|
|
variantAnalysisId: number,
|
|
filterSort?: RepositoriesFilterSortStateWithIds,
|
|
) {
|
|
await exportVariantAnalysisResults(
|
|
this,
|
|
variantAnalysisId,
|
|
filterSort,
|
|
this.app.credentials,
|
|
);
|
|
}
|
|
|
|
private getRepoStatesStoragePath(variantAnalysisId: number): string {
|
|
return join(
|
|
this.getVariantAnalysisStorageLocation(variantAnalysisId),
|
|
VariantAnalysisManager.REPO_STATES_FILENAME,
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Prepares a directory for storing results for a variant analysis.
|
|
* This directory contains a timestamp file, which will be
|
|
* used by the query history manager to determine when the directory
|
|
* should be deleted.
|
|
*/
|
|
private async prepareStorageDirectory(
|
|
variantAnalysisId: number,
|
|
): Promise<void> {
|
|
await createTimestampFile(
|
|
this.getVariantAnalysisStorageLocation(variantAnalysisId),
|
|
);
|
|
}
|
|
}
|