Spaces:
Paused
Paused
| import fs from "node:fs"; | |
| import os from "node:os"; | |
| import path from "node:path"; | |
| import { afterEach, beforeEach, describe, expect, it, vi } from "vitest"; | |
| const ORIGINAL_ENV = { ...process.env }; | |
| const CI_ENV_VARS = ["CI", "CONTINUOUS_INTEGRATION", "BUILD_NUMBER", "GITHUB_ACTIONS", "GITLAB_CI"]; | |
| function makeConfigPath(root: string, enabled: boolean): string { | |
| const configPath = path.join(root, ".paperclip", "config.json"); | |
| fs.mkdirSync(path.dirname(configPath), { recursive: true }); | |
| fs.writeFileSync(configPath, JSON.stringify({ | |
| $meta: { | |
| version: 1, | |
| updatedAt: "2026-03-31T00:00:00.000Z", | |
| source: "configure", | |
| }, | |
| database: { | |
| mode: "embedded-postgres", | |
| embeddedPostgresDataDir: path.join(root, "runtime", "db"), | |
| embeddedPostgresPort: 54329, | |
| backup: { | |
| enabled: true, | |
| intervalMinutes: 60, | |
| retentionDays: 30, | |
| dir: path.join(root, "runtime", "backups"), | |
| }, | |
| }, | |
| logging: { | |
| mode: "file", | |
| logDir: path.join(root, "runtime", "logs"), | |
| }, | |
| server: { | |
| deploymentMode: "local_trusted", | |
| exposure: "private", | |
| host: "127.0.0.1", | |
| port: 3100, | |
| allowedHostnames: [], | |
| serveUi: true, | |
| }, | |
| auth: { | |
| baseUrlMode: "auto", | |
| disableSignUp: false, | |
| }, | |
| telemetry: { | |
| enabled, | |
| }, | |
| storage: { | |
| provider: "local_disk", | |
| localDisk: { | |
| baseDir: path.join(root, "runtime", "storage"), | |
| }, | |
| s3: { | |
| bucket: "paperclip", | |
| region: "us-east-1", | |
| prefix: "", | |
| forcePathStyle: false, | |
| }, | |
| }, | |
| secrets: { | |
| provider: "local_encrypted", | |
| strictMode: false, | |
| localEncrypted: { | |
| keyFilePath: path.join(root, "runtime", "secrets", "master.key"), | |
| }, | |
| }, | |
| }, null, 2)); | |
| return configPath; | |
| } | |
| describe("cli telemetry", () => { | |
| beforeEach(() => { | |
| process.env = { ...ORIGINAL_ENV }; | |
| for (const key of CI_ENV_VARS) { | |
| delete process.env[key]; | |
| } | |
| vi.stubGlobal("fetch", vi.fn(async () => ({ ok: true }))); | |
| }); | |
| afterEach(() => { | |
| process.env = { ...ORIGINAL_ENV }; | |
| vi.unstubAllGlobals(); | |
| vi.resetModules(); | |
| }); | |
| it("respects telemetry.enabled=false from the config file", async () => { | |
| const root = fs.mkdtempSync(path.join(os.tmpdir(), "paperclip-cli-telemetry-")); | |
| const configPath = makeConfigPath(root, false); | |
| process.env.PAPERCLIP_HOME = path.join(root, "home"); | |
| process.env.PAPERCLIP_INSTANCE_ID = "telemetry-test"; | |
| const { initTelemetryFromConfigFile } = await import("../telemetry.js"); | |
| const client = initTelemetryFromConfigFile(configPath); | |
| expect(client).toBeNull(); | |
| expect(fs.existsSync(path.join(root, "home", "instances", "telemetry-test", "telemetry", "state.json"))).toBe(false); | |
| }); | |
| it("creates telemetry state only after the first event is tracked", async () => { | |
| const root = fs.mkdtempSync(path.join(os.tmpdir(), "paperclip-cli-telemetry-")); | |
| process.env.PAPERCLIP_HOME = path.join(root, "home"); | |
| process.env.PAPERCLIP_INSTANCE_ID = "telemetry-test"; | |
| const { initTelemetry, flushTelemetry } = await import("../telemetry.js"); | |
| const client = initTelemetry({ enabled: true }); | |
| const statePath = path.join(root, "home", "instances", "telemetry-test", "telemetry", "state.json"); | |
| expect(client).not.toBeNull(); | |
| expect(fs.existsSync(statePath)).toBe(false); | |
| client!.track("install.started", { setupMode: "quickstart" }); | |
| expect(fs.existsSync(statePath)).toBe(true); | |
| await flushTelemetry(); | |
| }); | |
| }); | |