Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
9 changes: 9 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -422,6 +422,11 @@
"title": "Coder: View Logs",
"icon": "$(list-unordered)"
},
{
"command": "coder.exportTelemetry",
"title": "Coder: Export Telemetry",
"icon": "$(save)"
},
{
"command": "coder.openAppStatus",
"title": "Open App Status",
Expand Down Expand Up @@ -536,6 +541,10 @@
"command": "coder.viewLogs",
"when": "true"
},
{
"command": "coder.exportTelemetry",
"when": "true"
},
{
"command": "coder.openAppStatus",
"when": "false"
Expand Down
13 changes: 13 additions & 0 deletions src/commands.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ import {
applySettingOverrides,
} from "./remote/sshOverrides";
import { resolveCliAuth } from "./settings/cli";
import { runExportTelemetryCommand } from "./telemetry/export/command";
import { toRemoteAuthority, toSafeHost } from "./util";
import { vscodeProposed } from "./vscodeProposed";
import { parseSpeedtestResult } from "./webviews/speedtest/types";
Expand All @@ -49,6 +50,7 @@ import type { SecretsManager } from "./core/secretsManager";
import type { DeploymentManager } from "./deployment/deploymentManager";
import type { Logger } from "./logging/logger";
import type { LoginCoordinator } from "./login/loginCoordinator";
import type { TelemetryService } from "./telemetry/service";
import type { SpeedtestPanelFactory } from "./webviews/speedtest/speedtestPanelFactory";
import type {
DuplicateWorkspaceIpc,
Expand Down Expand Up @@ -80,6 +82,7 @@ export class Commands {
private readonly loginCoordinator: LoginCoordinator;
private readonly duplicateWorkspaceIpc: DuplicateWorkspaceIpc;
private readonly speedtestPanelFactory: SpeedtestPanelFactory;
private readonly telemetryService: TelemetryService;

// These will only be populated when actively connected to a workspace and are
// used in commands. Because commands can be executed by the user, it is not
Expand All @@ -97,6 +100,7 @@ export class Commands {
private readonly extensionClient: CoderApi,
private readonly deploymentManager: DeploymentManager,
) {
this.telemetryService = serviceContainer.getTelemetryService();
this.logger = serviceContainer.getLogger();
this.pathResolver = serviceContainer.getPathResolver();
this.mementoManager = serviceContainer.getMementoManager();
Expand Down Expand Up @@ -350,6 +354,15 @@ export class Commands {
});
}

public async exportTelemetry(): Promise<void> {
await runExportTelemetryCommand(
this.pathResolver.getTelemetryPath(),
this.logger,
() => this.telemetryService.flush(),
this.telemetryService.getContext(),
);
}

/**
* View the logs for the currently connected workspace.
*/
Expand Down
1 change: 1 addition & 0 deletions src/core/commandManager.ts
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ export const CODER_COMMAND_IDS = [
"coder.navigateToWorkspaceSettings",
"coder.refreshWorkspaces",
"coder.viewLogs",
"coder.exportTelemetry",
"coder.searchMyWorkspaces",
"coder.searchAllWorkspaces",
"coder.manageCredentials",
Expand Down
4 changes: 4 additions & 0 deletions src/extension.ts
Original file line number Diff line number Diff line change
Expand Up @@ -295,6 +295,10 @@ async function doActivate(
void allWorkspacesProvider.fetchAndRefresh();
});
commandManager.register("coder.viewLogs", commands.viewLogs.bind(commands));
commandManager.register(
"coder.exportTelemetry",
commands.exportTelemetry.bind(commands),
);
commandManager.register("coder.searchMyWorkspaces", async () =>
showTreeViewSearch(MY_WORKSPACES_TREE_ID),
);
Expand Down
283 changes: 283 additions & 0 deletions src/telemetry/export/command.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,283 @@
import * as fs from "node:fs/promises";
import * as os from "node:os";
import * as path from "node:path";
import * as vscode from "vscode";

import { toError } from "../../error/errorUtils";
import { withCancellableProgress } from "../../progress";

import { listTelemetryFilesForRange, streamTelemetryEvents } from "./files";
import {
TELEMETRY_RANGE_PRESETS,
createCustomDateRange,
createPresetDateRange,
validateUtcDateInput,
type TelemetryDateRange,
type TelemetryRangePresetId,
} from "./range";
import { writeJsonArrayExport } from "./writers/json";
import { writeOtlpZipExport } from "./writers/otlp/writer";

import type { Logger } from "../../logging/logger";
import type { TelemetryContext } from "../event";

interface FormatPick extends vscode.QuickPickItem {
readonly id: "json" | "otlp";
}

interface RangePick extends vscode.QuickPickItem {
readonly id: TelemetryRangePresetId | "custom";
}

interface FormatOutput {
readonly ext: string;
readonly filters: NonNullable<vscode.SaveDialogOptions["filters"]>;
}

interface ExportSummary {
readonly filesScanned: number;
readonly eventCount: number;
}

const FORMAT_PICKS: readonly FormatPick[] = [
{
id: "json",
label: "JSON array",
detail: "Single JSON document for human inspection or compliance review.",
},
{
id: "otlp",
label: "OTLP/JSON zip",
detail:
"Zip containing logs.json, traces.json, and metrics.json for OTLP endpoints.",
},
];

const CUSTOM_RANGE_PICK: RangePick = {
id: "custom",
label: "Custom range…",
detail: "Choose inclusive UTC start and end dates.",
};

const FORMAT_OUTPUT: Record<FormatPick["id"], FormatOutput> = {
json: { ext: "json", filters: { "JSON files": ["json"] } },
otlp: { ext: "otlp.zip", filters: { "Zip files": ["zip"] } },
};

export async function runExportTelemetryCommand(
telemetryDir: string,
logger: Logger,
flushTelemetry: () => Promise<void>,
context: TelemetryContext,
): Promise<void> {
const range = await promptDateRange();
if (!range) return;
const format = await promptFormat();
if (!format) return;
const outputUri = await promptSavePath(range, format.id);
if (!outputUri) return;

const onCleanupError = cleanupWarner(
logger,
"Failed to delete telemetry export temp file",
);
const onStagingCleanupError = cleanupWarner(
logger,
"Failed to delete telemetry export staging directory",
);

// Flush + list run inside the progress callback so the on-disk snapshot
// is taken right before streaming and the user can cancel a long flush.
const result = await withCancellableProgress(
async ({ signal, progress }): Promise<ExportSummary> => {
progress.report({ message: "Flushing buffered events..." });
await flushTelemetry();
throwIfAborted(signal);

progress.report({ message: "Locating telemetry files..." });
const filePaths = await listTelemetryFilesForRange(telemetryDir, range);
if (filePaths.length === 0) {
return { filesScanned: 0, eventCount: 0 };
}

progress.report({ message: "Writing export..." });
const events = (async function* () {
for await (const event of streamTelemetryEvents(filePaths, range)) {
throwIfAborted(signal);
yield event;
}
})();

let eventCount: number;
if (format.id === "json") {
eventCount = await writeJsonArrayExport(
outputUri.fsPath,
events,
onCleanupError,
);
} else {
const counts = await writeOtlpZipExport(
outputUri.fsPath,
events,
context,
{
signal,
onTempCleanupError: onCleanupError,
onStagingCleanupError,
},
);
eventCount = counts.logs + counts.traces + counts.metrics;
}
return { filesScanned: filePaths.length, eventCount };
},
{
location: vscode.ProgressLocation.Notification,
title: "Exporting Coder telemetry",
cancellable: true,
},
);

if (!result.ok) {
if (result.cancelled) return;
logger.error("Telemetry export failed", result.error);
vscode.window.showErrorMessage(
`Telemetry export failed: ${toError(result.error).message}`,
);
return;
}

const { filesScanned, eventCount } = result.value;
if (filesScanned === 0) {
vscode.window.showInformationMessage(
`No telemetry files found for ${range.label}.`,
);
return;
}
if (eventCount === 0) {
await notifyNoEventsMatched(range, outputUri, logger);
return;
}
await notifyExportSuccess(outputUri, eventCount, logger);
}

function cleanupWarner(
logger: Logger,
message: string,
): (err: unknown, target: string) => void {
return (err, target) => logger.warn(message, target, err);
}

async function notifyExportSuccess(
outputUri: vscode.Uri,
eventCount: number,
logger: Logger,
): Promise<void> {
const action = await vscode.window.showInformationMessage(
`Exported ${eventCount} telemetry event(s) to ${outputUri.fsPath}.`,
"Reveal in File Explorer",
);
if (action !== "Reveal in File Explorer") return;
try {
await vscode.commands.executeCommand("revealFileInOS", outputUri);
} catch (err) {
// The export already succeeded; a reveal failure is informational.
logger.warn("Failed to reveal exported telemetry file", err);
}
}

async function notifyNoEventsMatched(
range: TelemetryDateRange,
outputUri: vscode.Uri,
logger: Logger,
): Promise<void> {
// Remove the empty file the writer just created so the user isn't left
// with an unwanted artifact.
await fs
.rm(outputUri.fsPath, { force: true })
.catch((err) =>
logger.warn(
"Failed to remove empty telemetry export",
outputUri.fsPath,
err,
),
);
vscode.window.showInformationMessage(
`No telemetry events matched ${range.label}.`,
);
}

function throwIfAborted(signal: AbortSignal): void {
if (!signal.aborted) return;
const reason: unknown = signal.reason;
throw reason instanceof Error
? reason
: Object.assign(new Error("Aborted"), { name: "AbortError" });
}

async function promptDateRange(): Promise<TelemetryDateRange | undefined> {
const pick = await vscode.window.showQuickPick(
[...TELEMETRY_RANGE_PRESETS, CUSTOM_RANGE_PICK],
{
title: "Export Telemetry: Date Range",
placeHolder: "Select telemetry date range",
ignoreFocusOut: true,
},
);
if (!pick) return undefined;
if (pick.id === "custom") return promptCustomDateRange();
return createPresetDateRange(pick.id);
}

async function promptCustomDateRange(): Promise<
TelemetryDateRange | undefined
> {
const todayUtc = new Date().toISOString().slice(0, 10);
const startDate = await vscode.window.showInputBox({
title: "Export Telemetry: Custom Start Date",
prompt: `Start date in UTC (YYYY-MM-DD). Today in UTC is ${todayUtc}; your local date may differ.`,
value: todayUtc,
validateInput: validateUtcDateInput,
ignoreFocusOut: true,
});
if (startDate === undefined) return undefined;

const endDate = await vscode.window.showInputBox({
title: "Export Telemetry: Custom End Date",
prompt: `End date in UTC (YYYY-MM-DD, inclusive). Today in UTC is ${todayUtc}.`,
value: startDate,
validateInput: (value) => {
const invalidDate = validateUtcDateInput(value);
if (invalidDate !== undefined) return invalidDate;
// YYYY-MM-DD strings sort lexicographically as calendar dates.
if (value < startDate) {
return "End date must be on or after start date.";
}
return undefined;
},
ignoreFocusOut: true,
});
if (endDate === undefined) return undefined;

return createCustomDateRange(startDate, endDate);
}

function promptFormat(): Thenable<FormatPick | undefined> {
return vscode.window.showQuickPick(FORMAT_PICKS, {
title: "Export Telemetry: Format",
placeHolder: "Select export format",
ignoreFocusOut: true,
});
}

function promptSavePath(
range: TelemetryDateRange,
format: FormatPick["id"],
): Thenable<vscode.Uri | undefined> {
const { ext, filters } = FORMAT_OUTPUT[format];
const defaultName = `coder-telemetry-${range.filenamePart}.${ext}`;
return vscode.window.showSaveDialog({
defaultUri: vscode.Uri.file(path.join(os.homedir(), defaultName)),
filters,
title: "Save Telemetry Export",
});
}
Loading
Loading