@rudderhq/cli 0.2.2 → 0.2.4-canary.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -10,7 +10,7 @@ var __export = (target, all) => {
10
10
  };
11
11
 
12
12
  // ../packages/shared/src/constants.ts
13
- var ORGANIZATION_STATUSES, DEPLOYMENT_MODES, DEPLOYMENT_EXPOSURES, AUTH_BASE_URL_MODES, AGENT_STATUSES, AGENT_RUNTIME_TYPES, AGENT_ROLES, AGENT_ICON_NAMES, ISSUE_STATUSES, ISSUE_PRIORITIES, CALENDAR_SOURCE_TYPES, CALENDAR_OWNER_TYPES, CALENDAR_VISIBILITIES, CALENDAR_SOURCE_STATUSES, CALENDAR_EVENT_KINDS, CALENDAR_EVENT_STATUSES, CALENDAR_SOURCE_MODES, CHAT_CONVERSATION_STATUSES, CHAT_ISSUE_CREATION_MODES, CHAT_MESSAGE_ROLES, CHAT_MESSAGE_KINDS, CHAT_MESSAGE_STATUSES, CHAT_CONTEXT_ENTITY_TYPES, GOAL_LEVELS, GOAL_STATUSES, PROJECT_STATUSES, ORGANIZATION_RESOURCE_KINDS, PROJECT_RESOURCE_ATTACHMENT_ROLES, AUTOMATION_STATUSES, AUTOMATION_CONCURRENCY_POLICIES, AUTOMATION_CATCH_UP_POLICIES, AUTOMATION_TRIGGER_SIGNING_MODES, PROJECT_COLORS, APPROVAL_TYPES, SECRET_PROVIDERS, STORAGE_PROVIDERS, BILLING_TYPES, FINANCE_EVENT_KINDS, FINANCE_DIRECTIONS, FINANCE_UNITS, BUDGET_SCOPE_TYPES, BUDGET_METRICS, BUDGET_WINDOW_KINDS, BUDGET_INCIDENT_RESOLUTION_ACTIONS, INVITE_JOIN_TYPES, JOIN_REQUEST_TYPES, JOIN_REQUEST_STATUSES, PERMISSION_KEYS, PLUGIN_STATUSES, PLUGIN_CATEGORIES, PLUGIN_CAPABILITIES, PLUGIN_UI_SLOT_TYPES, PLUGIN_RESERVED_COMPANY_ROUTE_SEGMENTS, PLUGIN_LAUNCHER_PLACEMENT_ZONES, PLUGIN_LAUNCHER_ACTIONS, PLUGIN_LAUNCHER_BOUNDS, PLUGIN_LAUNCHER_RENDER_ENVIRONMENTS, PLUGIN_UI_SLOT_ENTITY_TYPES, PLUGIN_STATE_SCOPE_KINDS;
13
+ var ORGANIZATION_STATUSES, DEPLOYMENT_MODES, DEPLOYMENT_EXPOSURES, AUTH_BASE_URL_MODES, AGENT_STATUSES, AGENT_RUNTIME_TYPES, AGENT_ROLES, AGENT_ICON_NAMES, AGENT_DICEBEAR_NOTIONISTS_ICON_PREFIX, AGENT_AVATAR_BACKGROUND_PRESET_IDS, ISSUE_STATUSES, ISSUE_PRIORITIES, CALENDAR_SOURCE_TYPES, CALENDAR_OWNER_TYPES, CALENDAR_VISIBILITIES, CALENDAR_SOURCE_STATUSES, CALENDAR_EVENT_KINDS, CALENDAR_EVENT_STATUSES, CALENDAR_SOURCE_MODES, CHAT_CONVERSATION_STATUSES, CHAT_ISSUE_CREATION_MODES, CHAT_MESSAGE_ROLES, CHAT_MESSAGE_KINDS, CHAT_MESSAGE_STATUSES, CHAT_CONTEXT_ENTITY_TYPES, GOAL_LEVELS, GOAL_STATUSES, PROJECT_STATUSES, ORGANIZATION_RESOURCE_KINDS, PROJECT_RESOURCE_ATTACHMENT_ROLES, AUTOMATION_STATUSES, AUTOMATION_CONCURRENCY_POLICIES, AUTOMATION_CATCH_UP_POLICIES, AUTOMATION_TRIGGER_SIGNING_MODES, PROJECT_COLORS, APPROVAL_TYPES, SECRET_PROVIDERS, STORAGE_PROVIDERS, BILLING_TYPES, FINANCE_EVENT_KINDS, FINANCE_DIRECTIONS, FINANCE_UNITS, BUDGET_SCOPE_TYPES, BUDGET_METRICS, BUDGET_WINDOW_KINDS, BUDGET_INCIDENT_RESOLUTION_ACTIONS, INVITE_JOIN_TYPES, JOIN_REQUEST_TYPES, JOIN_REQUEST_STATUSES, PERMISSION_KEYS, PLUGIN_STATUSES, PLUGIN_CATEGORIES, PLUGIN_CAPABILITIES, PLUGIN_UI_SLOT_TYPES, PLUGIN_RESERVED_COMPANY_ROUTE_SEGMENTS, PLUGIN_LAUNCHER_PLACEMENT_ZONES, PLUGIN_LAUNCHER_ACTIONS, PLUGIN_LAUNCHER_BOUNDS, PLUGIN_LAUNCHER_RENDER_ENVIRONMENTS, PLUGIN_UI_SLOT_ENTITY_TYPES, PLUGIN_STATE_SCOPE_KINDS;
14
14
  var init_constants = __esm({
15
15
  "../packages/shared/src/constants.ts"() {
16
16
  "use strict";
@@ -95,6 +95,15 @@ var init_constants = __esm({
95
95
  "pentagon",
96
96
  "fingerprint"
97
97
  ];
98
+ AGENT_DICEBEAR_NOTIONISTS_ICON_PREFIX = "dicebear:notionists:";
99
+ AGENT_AVATAR_BACKGROUND_PRESET_IDS = [
100
+ "mist",
101
+ "slate",
102
+ "sky",
103
+ "mint",
104
+ "peach",
105
+ "violet"
106
+ ];
98
107
  ISSUE_STATUSES = [
99
108
  "backlog",
100
109
  "todo",
@@ -632,9 +641,35 @@ var init_chat = __esm({
632
641
  question: z5.string().trim().min(1).max(240),
633
642
  options: z5.array(chatAskUserOptionSchema).min(2).max(3),
634
643
  allowFreeform: z5.boolean().optional()
644
+ }).superRefine((question, ctx) => {
645
+ const optionIds = /* @__PURE__ */ new Set();
646
+ question.options.forEach((option, index) => {
647
+ if (optionIds.has(option.id)) {
648
+ ctx.addIssue({
649
+ code: z5.ZodIssueCode.custom,
650
+ message: "Option ids must be unique within each question",
651
+ path: ["options", index, "id"]
652
+ });
653
+ return;
654
+ }
655
+ optionIds.add(option.id);
656
+ });
635
657
  });
636
658
  chatAskUserRequestSchema = z5.object({
637
659
  questions: z5.array(chatAskUserQuestionSchema).min(1).max(3)
660
+ }).superRefine((request, ctx) => {
661
+ const questionIds = /* @__PURE__ */ new Set();
662
+ request.questions.forEach((question, index) => {
663
+ if (questionIds.has(question.id)) {
664
+ ctx.addIssue({
665
+ code: z5.ZodIssueCode.custom,
666
+ message: "Question ids must be unique within requestUserInput",
667
+ path: ["questions", index, "id"]
668
+ });
669
+ return;
670
+ }
671
+ questionIds.add(question.id);
672
+ });
638
673
  });
639
674
  chatIssueRichReferenceSchema = z5.object({
640
675
  type: z5.literal("issue"),
@@ -1267,7 +1302,7 @@ var init_model_fallbacks = __esm({
1267
1302
 
1268
1303
  // ../packages/shared/src/validators/agent.ts
1269
1304
  import { z as z11 } from "zod";
1270
- var agentPermissionsSchema, agentInstructionsBundleModeSchema, updateAgentInstructionsBundleSchema, upsertAgentInstructionsFileSchema, agentRuntimeConfigSchema, optionalAgentNameSchema, uploadedAgentIconSchema, customAgentIconSchema, agentIconSchema, createAgentSchema, createAgentHireSchema, updateAgentSchema, updateAgentInstructionsPathSchema, createAgentKeySchema, wakeAgentSchema, resetAgentSessionSchema, testAgentRuntimeEnvironmentSchema, updateAgentPermissionsSchema;
1305
+ var agentPermissionsSchema, agentInstructionsBundleModeSchema, updateAgentInstructionsBundleSchema, upsertAgentInstructionsFileSchema, agentRuntimeConfigSchema, optionalAgentNameSchema, uploadedAgentIconSchema, diceBearNotionistsAgentIconSchema, agentIconSchema, createAgentSchema, createAgentHireSchema, updateAgentSchema, updateAgentInstructionsPathSchema, createAgentKeySchema, wakeAgentSchema, resetAgentSessionSchema, testAgentRuntimeEnvironmentSchema, updateAgentPermissionsSchema;
1271
1306
  var init_agent = __esm({
1272
1307
  "../packages/shared/src/validators/agent.ts"() {
1273
1308
  "use strict";
@@ -1312,10 +1347,19 @@ var init_agent = __esm({
1312
1347
  z11.string().trim().min(1).optional()
1313
1348
  );
1314
1349
  uploadedAgentIconSchema = z11.string().regex(
1315
- /^asset:[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i,
1350
+ new RegExp(
1351
+ `^asset:[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}(?:\\?bg=(?:${AGENT_AVATAR_BACKGROUND_PRESET_IDS.join("|")}))?$`,
1352
+ "i"
1353
+ ),
1316
1354
  "Invalid uploaded avatar reference"
1317
1355
  );
1318
- customAgentIconSchema = z11.string().trim().min(1).max(24).refine((value) => !value.toLowerCase().startsWith("asset:"), "Invalid uploaded avatar reference").refine((value) => !/[<>\u0000-\u001f\u007f]/u.test(value), "Icon cannot contain markup or control characters");
1356
+ diceBearNotionistsAgentIconSchema = z11.string().regex(
1357
+ new RegExp(
1358
+ `^${AGENT_DICEBEAR_NOTIONISTS_ICON_PREFIX}[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}(?:\\?bg=(?:${AGENT_AVATAR_BACKGROUND_PRESET_IDS.join("|")}))?$`,
1359
+ "i"
1360
+ ),
1361
+ "Invalid DiceBear avatar reference"
1362
+ );
1319
1363
  agentIconSchema = z11.preprocess(
1320
1364
  (value) => {
1321
1365
  if (typeof value !== "string") return value;
@@ -1325,7 +1369,7 @@ var init_agent = __esm({
1325
1369
  z11.union([
1326
1370
  z11.enum(AGENT_ICON_NAMES),
1327
1371
  uploadedAgentIconSchema,
1328
- customAgentIconSchema
1372
+ diceBearNotionistsAgentIconSchema
1329
1373
  ]).nullable()
1330
1374
  );
1331
1375
  createAgentSchema = z11.object({
@@ -1756,7 +1800,7 @@ var init_automation = __esm({
1756
1800
  "use strict";
1757
1801
  init_constants();
1758
1802
  createAutomationSchema = z19.object({
1759
- projectId: z19.string().uuid(),
1803
+ projectId: z19.string().uuid().optional().nullable().default(null),
1760
1804
  goalId: z19.string().uuid().optional().nullable(),
1761
1805
  parentIssueId: z19.string().uuid().optional().nullable(),
1762
1806
  title: z19.string().trim().min(1).max(200),
@@ -3716,7 +3760,7 @@ var init_server = __esm({
3716
3760
 
3717
3761
  // src/runtime/install.ts
3718
3762
  import { spawnSync } from "node:child_process";
3719
- import { mkdir, readFile, writeFile } from "node:fs/promises";
3763
+ import { mkdir, readFile, readdir, rm, stat, writeFile } from "node:fs/promises";
3720
3764
  import path6 from "node:path";
3721
3765
  import { createRequire } from "node:module";
3722
3766
  import { pathToFileURL } from "node:url";
@@ -3745,6 +3789,21 @@ async function readRuntimeInstallMetadata(cacheDir) {
3745
3789
  return null;
3746
3790
  }
3747
3791
  }
3792
+ async function writeRuntimeInstallMetadata(cacheDir, metadata) {
3793
+ await writeFile(path6.join(cacheDir, RUNTIME_METADATA_FILE), `${JSON.stringify(metadata, null, 2)}
3794
+ `, "utf8");
3795
+ }
3796
+ async function touchRuntimeInstallMetadata(cacheDir) {
3797
+ try {
3798
+ const metadata = await readRuntimeInstallMetadata(cacheDir);
3799
+ if (!metadata) return;
3800
+ await writeRuntimeInstallMetadata(cacheDir, {
3801
+ ...metadata,
3802
+ lastUsedAt: (/* @__PURE__ */ new Date()).toISOString()
3803
+ });
3804
+ } catch {
3805
+ }
3806
+ }
3748
3807
  async function isRuntimeCacheHit(options) {
3749
3808
  const packageName = options.packageName ?? RUNTIME_NPM_PACKAGE_NAME;
3750
3809
  const packageVersion = resolveRuntimePackageVersion(options.version);
@@ -3767,7 +3826,14 @@ async function ensureRuntimeInstalled(options) {
3767
3826
  const packageSpec = resolveRuntimePackageSpec(packageVersion, packageName);
3768
3827
  const command = `npm install --prefix ${cacheDir} --omit=dev --no-audit --no-fund ${packageSpec}`;
3769
3828
  if (await isRuntimeCacheHit({ cacheDir, version: packageVersion, packageName })) {
3770
- return { status: "hit", cacheDir, packageSpec, command, output: "" };
3829
+ await touchRuntimeInstallMetadata(cacheDir);
3830
+ const prune2 = await maybePruneRuntimeCache({
3831
+ homeDir: options.homeDir,
3832
+ requestedVersion: packageVersion,
3833
+ enabled: options.pruneRuntimeCache !== false,
3834
+ retention: options.retention
3835
+ });
3836
+ return { status: "hit", cacheDir, packageSpec, command, output: "", ...prune2 ? { prune: prune2 } : {} };
3771
3837
  }
3772
3838
  await mkdir(cacheDir, { recursive: true });
3773
3839
  await writeFile(path6.join(cacheDir, "package.json"), `${JSON.stringify({ private: true, type: "module" }, null, 2)}
@@ -3785,11 +3851,17 @@ async function ensureRuntimeInstalled(options) {
3785
3851
  version: 1,
3786
3852
  packageName,
3787
3853
  packageVersion,
3788
- installedAt: (/* @__PURE__ */ new Date()).toISOString()
3854
+ installedAt: (/* @__PURE__ */ new Date()).toISOString(),
3855
+ lastUsedAt: (/* @__PURE__ */ new Date()).toISOString()
3789
3856
  };
3790
- await writeFile(path6.join(cacheDir, RUNTIME_METADATA_FILE), `${JSON.stringify(metadata, null, 2)}
3791
- `, "utf8");
3792
- return { status: "installed", cacheDir, packageSpec, command, output };
3857
+ await writeRuntimeInstallMetadata(cacheDir, metadata);
3858
+ const prune = await maybePruneRuntimeCache({
3859
+ homeDir: options.homeDir,
3860
+ requestedVersion: packageVersion,
3861
+ enabled: options.pruneRuntimeCache !== false,
3862
+ retention: options.retention
3863
+ });
3864
+ return { status: "installed", cacheDir, packageSpec, command, output, ...prune ? { prune } : {} };
3793
3865
  }
3794
3866
  function resolveRuntimeServerEntrypoint(cacheDir, packageName = RUNTIME_NPM_PACKAGE_NAME) {
3795
3867
  return createRequire(path6.join(cacheDir, "package.json")).resolve(packageName);
@@ -3812,13 +3884,238 @@ function runNpmRuntimeInstall(spawnSyncImpl, cacheDir, packageSpec) {
3812
3884
  function collectSpawnOutput(result) {
3813
3885
  return [result.stdout, result.stderr, result.error instanceof Error ? result.error.message : null].filter((value) => typeof value === "string" && value.trim().length > 0).join("\n").trim();
3814
3886
  }
3815
- var RUNTIME_NPM_PACKAGE_NAME, RUNTIME_METADATA_FILE, RuntimeInstallError;
3887
+ async function maybePruneRuntimeCache(options) {
3888
+ if (!options.enabled) return null;
3889
+ return pruneRuntimeCache({
3890
+ ...options.retention,
3891
+ homeDir: options.homeDir,
3892
+ requestedVersion: options.retention?.requestedVersion ?? options.requestedVersion
3893
+ });
3894
+ }
3895
+ async function pruneRuntimeCache(options = {}) {
3896
+ const homeDir = options.homeDir ?? resolveRudderHomeDir();
3897
+ const now = options.now ?? /* @__PURE__ */ new Date();
3898
+ const entries = await scanRuntimeCacheEntries(homeDir);
3899
+ const activeVersions = await readActiveRuntimeVersions(homeDir);
3900
+ const protectedVersions = resolveProtectedRuntimeVersions(entries, {
3901
+ requestedVersion: options.requestedVersion,
3902
+ protectedVersions: [...options.protectedVersions ?? [], ...activeVersions],
3903
+ keepPreviousEntries: options.keepPreviousEntries ?? DEFAULT_RUNTIME_CACHE_KEEP_PREVIOUS
3904
+ });
3905
+ const protectedSet = new Set(protectedVersions);
3906
+ const maxEntries = options.maxEntries ?? DEFAULT_RUNTIME_CACHE_MAX_ENTRIES;
3907
+ const maxAgeMs = options.maxAgeMs ?? DEFAULT_RUNTIME_CACHE_MAX_AGE_MS;
3908
+ const maxTotalBytes = options.maxTotalBytes ?? DEFAULT_RUNTIME_CACHE_MAX_BYTES;
3909
+ const deletions = planRuntimeCacheDeletions(entries, {
3910
+ nowMs: now.getTime(),
3911
+ protectedVersions: protectedSet,
3912
+ maxEntries,
3913
+ maxAgeMs,
3914
+ maxTotalBytes
3915
+ });
3916
+ const deleted = [];
3917
+ const warnings = [];
3918
+ for (const entry of deletions) {
3919
+ try {
3920
+ await rm(entry.cacheDir, { recursive: true, force: true });
3921
+ deleted.push({
3922
+ cacheDir: entry.cacheDir,
3923
+ packageVersion: entry.packageVersion,
3924
+ sizeBytes: entry.sizeBytes
3925
+ });
3926
+ } catch (error) {
3927
+ warnings.push(
3928
+ `Failed to remove runtime cache ${entry.cacheDir}: ${error instanceof Error ? error.message : String(error)}`
3929
+ );
3930
+ }
3931
+ }
3932
+ return {
3933
+ scanned: entries.length,
3934
+ deleted,
3935
+ protectedVersions,
3936
+ freedBytes: deleted.reduce((total, entry) => total + entry.sizeBytes, 0),
3937
+ warnings
3938
+ };
3939
+ }
3940
+ async function scanRuntimeCacheEntries(homeDir) {
3941
+ const runtimesDir = path6.join(homeDir, "runtimes");
3942
+ const dirents = await readdir(runtimesDir, { withFileTypes: true }).catch(() => null);
3943
+ if (!dirents) return [];
3944
+ const entries = [];
3945
+ for (const dirent of dirents) {
3946
+ if (!dirent.isDirectory()) continue;
3947
+ const cacheDir = path6.join(runtimesDir, dirent.name);
3948
+ const metadata = await readRuntimeInstallMetadata(cacheDir);
3949
+ if (!metadata) continue;
3950
+ const fallbackStat = await safeStat(cacheDir);
3951
+ const installedAtMs = parseTimestampMs(metadata.installedAt) ?? Number(fallbackStat?.mtimeMs ?? 0);
3952
+ const lastUsedAtMs = parseTimestampMs(metadata.lastUsedAt) ?? installedAtMs;
3953
+ entries.push({
3954
+ cacheDir,
3955
+ packageVersion: metadata.packageVersion,
3956
+ installedAtMs,
3957
+ lastUsedAtMs,
3958
+ sizeBytes: await directorySizeBytes(cacheDir)
3959
+ });
3960
+ }
3961
+ return entries;
3962
+ }
3963
+ function parseTimestampMs(value) {
3964
+ if (!value) return null;
3965
+ const ms = Date.parse(value);
3966
+ return Number.isFinite(ms) ? ms : null;
3967
+ }
3968
+ async function safeStat(targetPath) {
3969
+ try {
3970
+ return await stat(targetPath);
3971
+ } catch {
3972
+ return null;
3973
+ }
3974
+ }
3975
+ async function directorySizeBytes(targetPath) {
3976
+ const dirents = await readdir(targetPath, { withFileTypes: true }).catch(() => null);
3977
+ if (!dirents) return 0;
3978
+ let total = 0;
3979
+ for (const dirent of dirents) {
3980
+ const entryPath = path6.join(targetPath, dirent.name);
3981
+ if (dirent.isSymbolicLink()) continue;
3982
+ if (dirent.isDirectory()) {
3983
+ total += await directorySizeBytes(entryPath);
3984
+ continue;
3985
+ }
3986
+ const entryStat = await safeStat(entryPath);
3987
+ total += Number(entryStat?.size ?? 0);
3988
+ }
3989
+ return total;
3990
+ }
3991
+ async function readActiveRuntimeVersions(homeDir) {
3992
+ const instancesDir = path6.join(homeDir, "instances");
3993
+ const dirents = await readdir(instancesDir, { withFileTypes: true }).catch(() => null);
3994
+ if (!dirents) return [];
3995
+ const versions = /* @__PURE__ */ new Set();
3996
+ for (const dirent of dirents) {
3997
+ if (!dirent.isDirectory()) continue;
3998
+ try {
3999
+ const descriptorPath = path6.join(instancesDir, dirent.name, "runtime", "server.json");
4000
+ const parsed = JSON.parse(await readFile(descriptorPath, "utf8"));
4001
+ if (typeof parsed.version !== "string") continue;
4002
+ if (typeof parsed.pid === "number" && Number.isInteger(parsed.pid) && parsed.pid > 0 && isPidRunning(parsed.pid)) {
4003
+ versions.add(parsed.version);
4004
+ }
4005
+ } catch {
4006
+ continue;
4007
+ }
4008
+ }
4009
+ return [...versions];
4010
+ }
4011
+ function isPidRunning(pid) {
4012
+ try {
4013
+ process.kill(pid, 0);
4014
+ return true;
4015
+ } catch {
4016
+ return false;
4017
+ }
4018
+ }
4019
+ function resolveProtectedRuntimeVersions(entries, options) {
4020
+ const protectedVersions = /* @__PURE__ */ new Set();
4021
+ const requestedVersion = options.requestedVersion ? resolveRuntimePackageVersion(options.requestedVersion) : null;
4022
+ if (requestedVersion) protectedVersions.add(requestedVersion);
4023
+ for (const version of options.protectedVersions) {
4024
+ const normalized = version.trim();
4025
+ if (normalized) protectedVersions.add(normalized);
4026
+ }
4027
+ const latestStable = latestRuntimeVersion(entries.filter((entry) => isStableVersion(entry.packageVersion)));
4028
+ if (latestStable) protectedVersions.add(latestStable);
4029
+ const latestCanary = latestRuntimeVersion(entries.filter((entry) => isCanaryVersion(entry.packageVersion)));
4030
+ if (latestCanary) protectedVersions.add(latestCanary);
4031
+ const previousEntries = [...entries].filter((entry) => entry.packageVersion !== requestedVersion).sort((a, b) => b.lastUsedAtMs - a.lastUsedAtMs);
4032
+ for (const entry of previousEntries.slice(0, Math.max(0, options.keepPreviousEntries))) {
4033
+ protectedVersions.add(entry.packageVersion);
4034
+ }
4035
+ return [...protectedVersions].sort();
4036
+ }
4037
+ function planRuntimeCacheDeletions(entries, options) {
4038
+ const deletions = /* @__PURE__ */ new Set();
4039
+ const oldestFirst = [...entries].sort((a, b) => a.lastUsedAtMs - b.lastUsedAtMs);
4040
+ const canDelete = (entry) => !options.protectedVersions.has(entry.packageVersion) && !deletions.has(entry.cacheDir);
4041
+ const mark = (entry) => {
4042
+ if (canDelete(entry)) deletions.add(entry.cacheDir);
4043
+ };
4044
+ if (options.maxAgeMs >= 0) {
4045
+ for (const entry of oldestFirst) {
4046
+ if (options.nowMs - entry.lastUsedAtMs > options.maxAgeMs) mark(entry);
4047
+ }
4048
+ }
4049
+ if (options.maxEntries > 0) {
4050
+ for (const entry of oldestFirst) {
4051
+ if (entries.length - deletions.size <= options.maxEntries) break;
4052
+ mark(entry);
4053
+ }
4054
+ }
4055
+ if (options.maxTotalBytes > 0) {
4056
+ let remainingBytes = entries.reduce((total, entry) => total + entry.sizeBytes, 0) - [...deletions].reduce((total, cacheDir) => total + (entries.find((entry) => entry.cacheDir === cacheDir)?.sizeBytes ?? 0), 0);
4057
+ for (const entry of oldestFirst) {
4058
+ if (remainingBytes <= options.maxTotalBytes) break;
4059
+ if (!canDelete(entry)) continue;
4060
+ deletions.add(entry.cacheDir);
4061
+ remainingBytes -= entry.sizeBytes;
4062
+ }
4063
+ }
4064
+ return entries.filter((entry) => deletions.has(entry.cacheDir));
4065
+ }
4066
+ function latestRuntimeVersion(entries) {
4067
+ let latest = null;
4068
+ for (const entry of entries) {
4069
+ if (!latest || compareRuntimeVersions(entry.packageVersion, latest) > 0) {
4070
+ latest = entry.packageVersion;
4071
+ }
4072
+ }
4073
+ return latest;
4074
+ }
4075
+ function isStableVersion(version) {
4076
+ return /^\d+\.\d+\.\d+$/.test(version);
4077
+ }
4078
+ function isCanaryVersion(version) {
4079
+ return /^\d+\.\d+\.\d+-canary\.\d+$/.test(version);
4080
+ }
4081
+ function compareRuntimeVersions(a, b) {
4082
+ const parsedA = parseRuntimeVersion(a);
4083
+ const parsedB = parseRuntimeVersion(b);
4084
+ if (!parsedA || !parsedB) return a.localeCompare(b);
4085
+ for (const key of ["major", "minor", "patch"]) {
4086
+ if (parsedA[key] !== parsedB[key]) return parsedA[key] - parsedB[key];
4087
+ }
4088
+ if (parsedA.prerelease === null && parsedB.prerelease !== null) return 1;
4089
+ if (parsedA.prerelease !== null && parsedB.prerelease === null) return -1;
4090
+ if (parsedA.canaryNumber !== null && parsedB.canaryNumber !== null) {
4091
+ return parsedA.canaryNumber - parsedB.canaryNumber;
4092
+ }
4093
+ return (parsedA.prerelease ?? "").localeCompare(parsedB.prerelease ?? "");
4094
+ }
4095
+ function parseRuntimeVersion(version) {
4096
+ const match = /^(\d+)\.(\d+)\.(\d+)(?:-([0-9A-Za-z.-]+))?$/.exec(version);
4097
+ if (!match) return null;
4098
+ const prerelease = match[4] ?? null;
4099
+ const canaryMatch = prerelease ? /^canary\.(\d+)$/.exec(prerelease) : null;
4100
+ return {
4101
+ major: Number(match[1]),
4102
+ minor: Number(match[2]),
4103
+ patch: Number(match[3]),
4104
+ prerelease,
4105
+ canaryNumber: canaryMatch ? Number(canaryMatch[1]) : null
4106
+ };
4107
+ }
4108
+ var RUNTIME_NPM_PACKAGE_NAME, RUNTIME_METADATA_FILE, DEFAULT_RUNTIME_CACHE_MAX_ENTRIES, DEFAULT_RUNTIME_CACHE_MAX_AGE_MS, DEFAULT_RUNTIME_CACHE_MAX_BYTES, DEFAULT_RUNTIME_CACHE_KEEP_PREVIOUS, RuntimeInstallError;
3816
4109
  var init_install = __esm({
3817
4110
  "src/runtime/install.ts"() {
3818
4111
  "use strict";
3819
4112
  init_home();
3820
4113
  RUNTIME_NPM_PACKAGE_NAME = "@rudderhq/server";
3821
4114
  RUNTIME_METADATA_FILE = "runtime.json";
4115
+ DEFAULT_RUNTIME_CACHE_MAX_ENTRIES = 5;
4116
+ DEFAULT_RUNTIME_CACHE_MAX_AGE_MS = 14 * 24 * 60 * 60 * 1e3;
4117
+ DEFAULT_RUNTIME_CACHE_MAX_BYTES = 2 * 1024 * 1024 * 1024;
4118
+ DEFAULT_RUNTIME_CACHE_KEEP_PREVIOUS = 1;
3822
4119
  RuntimeInstallError = class extends Error {
3823
4120
  cacheDir;
3824
4121
  command;
@@ -6047,11 +6344,12 @@ ${err instanceof Error ? err.message : String(err)}`
6047
6344
 
6048
6345
  // src/commands/start.ts
6049
6346
  init_install2();
6347
+ init_home();
6050
6348
  init_install();
6051
6349
  import { spawn, spawnSync as spawnSync3 } from "node:child_process";
6052
6350
  import { createHash } from "node:crypto";
6053
6351
  import { constants as fsConstants, createWriteStream, mkdirSync, readFileSync as readFileSync2 } from "node:fs";
6054
- import { access, chmod, copyFile, cp, mkdtemp, mkdir as mkdir2, readFile as readFile2, readdir, rm, writeFile as writeFile2 } from "node:fs/promises";
6352
+ import { access, chmod, copyFile, cp, mkdtemp, mkdir as mkdir2, readFile as readFile2, readdir as readdir2, rm as rm2, writeFile as writeFile2 } from "node:fs/promises";
6055
6353
  import { homedir, tmpdir } from "node:os";
6056
6354
  import path11 from "node:path";
6057
6355
  import { Readable, Transform } from "node:stream";
@@ -6177,6 +6475,7 @@ var CLI_REGISTRY_LATEST_URL = "https://registry.npmjs.org/@rudderhq%2fcli/latest
6177
6475
  var DESKTOP_APP_NAME = "Rudder";
6178
6476
  var DESKTOP_METADATA_FILE = ".rudder-desktop-install.json";
6179
6477
  var DESKTOP_CHECKSUM_ASSET_NAME = "SHASUMS256.txt";
6478
+ var DESKTOP_ASSET_CACHE_DIR = "desktop-assets";
6180
6479
  var GITHUB_ASSET_DOWNLOAD_ACCEPT = "application/octet-stream";
6181
6480
  function normalizeProgressTotal(totalBytes) {
6182
6481
  return typeof totalBytes === "number" && Number.isFinite(totalBytes) && totalBytes > 0 ? totalBytes : null;
@@ -6242,6 +6541,38 @@ function createDesktopProgressFactory() {
6242
6541
  };
6243
6542
  };
6244
6543
  }
6544
+ async function waitForDesktopApplySignal() {
6545
+ process.stdin.setEncoding("utf8");
6546
+ process.stdin.resume();
6547
+ await new Promise((resolve, reject) => {
6548
+ let buffer = "";
6549
+ const cleanup = () => {
6550
+ process.stdin.off("data", onData);
6551
+ process.stdin.off("end", onEnd);
6552
+ process.stdin.off("error", onError);
6553
+ };
6554
+ const onData = (chunk) => {
6555
+ buffer += chunk;
6556
+ const lines = buffer.split(/\r?\n/);
6557
+ buffer = lines.pop() ?? "";
6558
+ if (lines.some((line) => line.trim() === "apply")) {
6559
+ cleanup();
6560
+ resolve();
6561
+ }
6562
+ };
6563
+ const onEnd = () => {
6564
+ cleanup();
6565
+ reject(new Error("Desktop update apply signal ended before confirmation."));
6566
+ };
6567
+ const onError = (error) => {
6568
+ cleanup();
6569
+ reject(error);
6570
+ };
6571
+ process.stdin.on("data", onData);
6572
+ process.stdin.on("end", onEnd);
6573
+ process.stdin.on("error", onError);
6574
+ });
6575
+ }
6245
6576
  function resolveCurrentCliVersion(env = process.env) {
6246
6577
  const version = resolveCliVersion(import.meta.url, env);
6247
6578
  return version === "0.0.0" ? "latest" : version;
@@ -6526,6 +6857,44 @@ async function downloadChecksums(checksumAsset, outputDir, progressFactory = cre
6526
6857
  const checksumPath = await downloadAsset(checksumAsset, outputDir, progressFactory);
6527
6858
  return parseChecksumFile(readFileSync2(checksumPath, "utf8"));
6528
6859
  }
6860
+ function normalizeDesktopAssetChecksum(checksum) {
6861
+ const normalized = checksum.trim().toLowerCase();
6862
+ if (!/^[a-f0-9]{64}$/.test(normalized)) {
6863
+ throw new Error("Desktop asset cache requires a SHA-256 checksum.");
6864
+ }
6865
+ return normalized;
6866
+ }
6867
+ function resolveDesktopAssetCacheDir(assetChecksum, homeDir = resolveRudderHomeDir()) {
6868
+ return path11.join(homeDir, DESKTOP_ASSET_CACHE_DIR, normalizeDesktopAssetChecksum(assetChecksum));
6869
+ }
6870
+ function resolveDesktopCachedAssetPath(assetName, assetChecksum, homeDir = resolveRudderHomeDir()) {
6871
+ return path11.join(resolveDesktopAssetCacheDir(assetChecksum, homeDir), path11.basename(assetName));
6872
+ }
6873
+ async function downloadDesktopAssetWithCache(asset, expectedChecksum, options = {}) {
6874
+ const normalizedChecksum = normalizeDesktopAssetChecksum(expectedChecksum);
6875
+ const cachePath = resolveDesktopCachedAssetPath(asset.name, normalizedChecksum, options.homeDir);
6876
+ if (await pathExists(cachePath)) {
6877
+ try {
6878
+ const checksum = assertChecksumMatch(cachePath, normalizedChecksum);
6879
+ return { path: cachePath, checksum, cacheStatus: "hit" };
6880
+ } catch {
6881
+ await rm2(cachePath, { force: true });
6882
+ }
6883
+ }
6884
+ const outputDir = options.outputDir ?? await mkdtemp(path11.join(tmpdir(), "rudder-desktop-installer."));
6885
+ const removeOutputDir = options.outputDir ? false : true;
6886
+ try {
6887
+ const downloadedPath = await downloadAsset(asset, outputDir, options.progressFactory);
6888
+ const checksum = assertChecksumMatch(downloadedPath, normalizedChecksum);
6889
+ await mkdir2(path11.dirname(cachePath), { recursive: true });
6890
+ if (path11.resolve(downloadedPath) !== path11.resolve(cachePath)) {
6891
+ await copyFile(downloadedPath, cachePath);
6892
+ }
6893
+ return { path: cachePath, checksum, cacheStatus: "miss" };
6894
+ } finally {
6895
+ if (removeOutputDir) await rm2(outputDir, { recursive: true, force: true });
6896
+ }
6897
+ }
6529
6898
  async function pathExists(targetPath) {
6530
6899
  try {
6531
6900
  await access(targetPath, fsConstants.F_OK);
@@ -6564,7 +6933,7 @@ function isSuccessfulRobocopyExitCode(status) {
6564
6933
  return typeof status === "number" && status >= 0 && status <= 7;
6565
6934
  }
6566
6935
  async function extractZip(zipPath, outputDir, target) {
6567
- await rm(outputDir, { recursive: true, force: true });
6936
+ await rm2(outputDir, { recursive: true, force: true });
6568
6937
  await mkdir2(outputDir, { recursive: true });
6569
6938
  if (target.platform === "macos") {
6570
6939
  runChecked("ditto", ["-x", "-k", zipPath, outputDir]);
@@ -6579,7 +6948,7 @@ async function extractZip(zipPath, outputDir, target) {
6579
6948
  }
6580
6949
  async function findPath(root, predicate, maxDepth = 5) {
6581
6950
  async function visit(dir, depth) {
6582
- const entries = await readdir(dir, { withFileTypes: true });
6951
+ const entries = await readdir2(dir, { withFileTypes: true });
6583
6952
  for (const entry of entries) {
6584
6953
  const fullPath = path11.join(dir, entry.name);
6585
6954
  if (predicate(fullPath, entry.isDirectory())) return fullPath;
@@ -6654,13 +7023,13 @@ async function requestDesktopQuit(executablePath, target) {
6654
7023
  try {
6655
7024
  return await waitForUpdateQuitResponse(responsePath);
6656
7025
  } finally {
6657
- await rm(responsePath, { force: true });
7026
+ await rm2(responsePath, { force: true });
6658
7027
  }
6659
7028
  }
6660
7029
  async function removePathWithRetry(targetPath, attempts = 5) {
6661
7030
  for (let attempt = 0; attempt < attempts; attempt += 1) {
6662
7031
  try {
6663
- await rm(targetPath, { recursive: true, force: true });
7032
+ await rm2(targetPath, { recursive: true, force: true });
6664
7033
  if (!await pathExists(targetPath)) return true;
6665
7034
  } catch {
6666
7035
  }
@@ -6714,7 +7083,7 @@ async function installPortableDesktop(installerPath, paths, target) {
6714
7083
  await mkdir2(path11.dirname(paths.installRoot), { recursive: true });
6715
7084
  await copyPortableAppBundle(appSource, paths.installRoot);
6716
7085
  } finally {
6717
- await rm(extractDir, { recursive: true, force: true });
7086
+ await rm2(extractDir, { recursive: true, force: true });
6718
7087
  }
6719
7088
  }
6720
7089
  async function copyPortableAppBundle(sourcePath, destinationPath) {
@@ -6965,13 +7334,38 @@ async function startCommand(opts) {
6965
7334
  desktopProgressJson ? "preparing_restart" : null
6966
7335
  );
6967
7336
  } else {
6968
- const installerPath = await downloadAsset(asset, outputDir, progressFactory);
7337
+ const cachedAsset = await downloadDesktopAssetWithCache(asset, expectedChecksum, {
7338
+ outputDir,
7339
+ progressFactory
7340
+ });
7341
+ if (cachedAsset.cacheStatus === "hit") {
7342
+ p13.log.success(`Desktop asset cache hit at ${pc8.cyan(cachedAsset.path)}.`);
7343
+ if (desktopProgressJson) {
7344
+ writeDesktopProgress({
7345
+ phase: "downloading_asset",
7346
+ message: `Desktop asset cache hit for ${asset.name}.`,
7347
+ percent: 100
7348
+ });
7349
+ }
7350
+ }
6969
7351
  const checksum = await runStartPhase(
6970
7352
  "Verifying Desktop checksum...",
6971
- `Verified ${pc8.cyan(path11.basename(installerPath))}.`,
6972
- () => assertChecksumMatch(installerPath, expectedChecksum),
7353
+ `Verified ${pc8.cyan(path11.basename(cachedAsset.path))}.`,
7354
+ () => assertChecksumMatch(cachedAsset.path, expectedChecksum),
6973
7355
  desktopProgressJson ? "verifying_checksum" : null
6974
7356
  );
7357
+ if (desktopProgressJson && opts.desktopWaitForApply === true) {
7358
+ writeDesktopProgress({
7359
+ phase: "ready_to_install",
7360
+ message: "Desktop update is downloaded and verified.",
7361
+ percent: 100
7362
+ });
7363
+ await waitForDesktopApplySignal();
7364
+ writeDesktopProgress({
7365
+ phase: "preparing_restart",
7366
+ message: "Applying Desktop update..."
7367
+ });
7368
+ }
6975
7369
  await runStartPhase(
6976
7370
  "Replacing existing Rudder Desktop if needed...",
6977
7371
  "Existing Desktop install is ready for replacement.",
@@ -6981,7 +7375,7 @@ async function startCommand(opts) {
6981
7375
  await runStartPhase(
6982
7376
  "Installing portable Desktop app...",
6983
7377
  `Installed Rudder Desktop to ${pc8.cyan(installPaths.appPath)}.`,
6984
- () => installPortableDesktop(installerPath, installPaths, target),
7378
+ () => installPortableDesktop(cachedAsset.path, installPaths, target),
6985
7379
  desktopProgressJson ? "preparing_restart" : null
6986
7380
  );
6987
7381
  await runStartPhase(
@@ -8100,7 +8494,7 @@ function registerContextCommands(program) {
8100
8494
  }
8101
8495
 
8102
8496
  // src/commands/client/company.ts
8103
- import { mkdir as mkdir3, readdir as readdir2, readFile as readFile3, stat, writeFile as writeFile3 } from "node:fs/promises";
8497
+ import { mkdir as mkdir3, readdir as readdir3, readFile as readFile3, stat as stat2, writeFile as writeFile3 } from "node:fs/promises";
8104
8498
  import path15 from "node:path";
8105
8499
  import * as p15 from "@clack/prompts";
8106
8500
  import pc14 from "picocolors";
@@ -8835,14 +9229,14 @@ function normalizeGithubImportSource(input, refOverride) {
8835
9229
  }
8836
9230
  async function pathExists2(inputPath) {
8837
9231
  try {
8838
- await stat(path15.resolve(inputPath));
9232
+ await stat2(path15.resolve(inputPath));
8839
9233
  return true;
8840
9234
  } catch {
8841
9235
  return false;
8842
9236
  }
8843
9237
  }
8844
9238
  async function collectPackageFiles(root, current, files) {
8845
- const entries = await readdir2(current, { withFileTypes: true });
9239
+ const entries = await readdir3(current, { withFileTypes: true });
8846
9240
  for (const entry of entries) {
8847
9241
  if (entry.name.startsWith(".git")) continue;
8848
9242
  const absolutePath = path15.join(current, entry.name);
@@ -8858,7 +9252,7 @@ async function collectPackageFiles(root, current, files) {
8858
9252
  }
8859
9253
  async function resolveInlineSourceFromPath(inputPath) {
8860
9254
  const resolved = path15.resolve(inputPath);
8861
- const resolvedStat = await stat(resolved);
9255
+ const resolvedStat = await stat2(resolved);
8862
9256
  if (resolvedStat.isFile() && path15.extname(resolved).toLowerCase() === ".zip") {
8863
9257
  const archive = await readZipArchive(await readFile3(resolved));
8864
9258
  const filteredFiles = Object.fromEntries(
@@ -8894,12 +9288,12 @@ async function writeExportToFolder(outDir, exported) {
8894
9288
  }
8895
9289
  async function confirmOverwriteExportDirectory(outDir) {
8896
9290
  const root = path15.resolve(outDir);
8897
- const stats = await stat(root).catch(() => null);
9291
+ const stats = await stat2(root).catch(() => null);
8898
9292
  if (!stats) return;
8899
9293
  if (!stats.isDirectory()) {
8900
9294
  throw new Error(`Export output path ${root} exists and is not a directory.`);
8901
9295
  }
8902
- const entries = await readdir2(root);
9296
+ const entries = await readdir3(root);
8903
9297
  if (entries.length === 0) return;
8904
9298
  if (!process.stdin.isTTY || !process.stdout.isTTY) {
8905
9299
  throw new Error(`Export output directory ${root} already contains files. Re-run interactively or choose an empty directory.`);
@@ -9321,7 +9715,7 @@ ${organizationUrl}`);
9321
9715
 
9322
9716
  // src/commands/client/issue.ts
9323
9717
  init_src();
9324
- import { readFile as readFile4, stat as stat2 } from "node:fs/promises";
9718
+ import { readFile as readFile4, stat as stat3 } from "node:fs/promises";
9325
9719
  import path16 from "node:path";
9326
9720
 
9327
9721
  // src/agent-v1-registry.ts
@@ -10216,7 +10610,7 @@ ${imageBlock}` : imageBlock;
10216
10610
  }
10217
10611
  async function uploadIssueCommentImage(ctx, issue, imagePath) {
10218
10612
  const resolvedPath = path16.resolve(process.cwd(), imagePath);
10219
- const stats = await stat2(resolvedPath).catch((err) => {
10613
+ const stats = await stat3(resolvedPath).catch((err) => {
10220
10614
  throw new Error(`Unable to read image ${imagePath}: ${err instanceof Error ? err.message : String(err)}`);
10221
10615
  });
10222
10616
  if (!stats.isFile()) {
@@ -11555,7 +11949,7 @@ function createProgram() {
11555
11949
  });
11556
11950
  loadRudderEnvFile(options.config);
11557
11951
  });
11558
- program.command("start").description("Start Rudder Desktop and prepare the matching persistent CLI").option("--no-cli", "Skip persistent CLI installation").option("--no-runtime", "Skip Rudder runtime installation").option("--no-desktop", "Skip desktop app installation").option("--version <version>", "Rudder version to start (default: current CLI version)").option("--target-version <version>", "Rudder version to start; avoids the root CLI version flag").option("--repo <owner/repo>", "GitHub repository that hosts desktop releases").option("--output-dir <path>", "Directory for downloaded desktop release assets").option("--desktop-install-dir <path>", "Directory for the portable Desktop install").option("--no-open", "Install Desktop without launching it").option("--wait-for-active-runs", "Wait for active Rudder runs to finish before replacing Desktop", false).option("--desktop-progress-json", "Emit newline-delimited Desktop update progress events").option("--no-version-check", "Skip checking npm for a newer Rudder CLI version").option("--dry-run", "Print the start actions without changing the machine", false).action(startCommand);
11952
+ program.command("start").description("Start Rudder Desktop and prepare the matching persistent CLI").option("--no-cli", "Skip persistent CLI installation").option("--no-runtime", "Skip Rudder runtime installation").option("--no-desktop", "Skip desktop app installation").option("--version <version>", "Rudder version to start (default: current CLI version)").option("--target-version <version>", "Rudder version to start; avoids the root CLI version flag").option("--repo <owner/repo>", "GitHub repository that hosts desktop releases").option("--output-dir <path>", "Directory for downloaded desktop release assets").option("--desktop-install-dir <path>", "Directory for the portable Desktop install").option("--no-open", "Install Desktop without launching it").option("--wait-for-active-runs", "Wait for active Rudder runs to finish before replacing Desktop", false).option("--desktop-progress-json", "Emit newline-delimited Desktop update progress events").option("--desktop-wait-for-apply", "Wait for an apply signal after downloading and verifying the Desktop update", false).option("--no-version-check", "Skip checking npm for a newer Rudder CLI version").option("--dry-run", "Print the start actions without changing the machine", false).action(startCommand);
11559
11953
  program.command("onboard").description("Interactive first-run setup wizard").option("-c, --config <path>", "Path to config file").option("-d, --data-dir <path>", DATA_DIR_OPTION_HELP).option("-y, --yes", "Accept defaults (quickstart + start immediately)", false).option("--run", "Start Rudder immediately after saving config", false).action(onboard);
11560
11954
  program.command("doctor").description("Run diagnostic checks on your Rudder setup").option("-c, --config <path>", "Path to config file").option("-d, --data-dir <path>", DATA_DIR_OPTION_HELP).option("--repair", "Attempt to repair issues automatically").alias("--fix").option("-y, --yes", "Skip repair confirmation prompts").action(async (opts) => {
11561
11955
  await doctor(opts);