primitive-admin 1.0.49 → 1.0.51

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.
Files changed (153) hide show
  1. package/README.md +129 -10
  2. package/assets/skill/skills/primitive-platform/SKILL.md +85 -30
  3. package/dist/bin/primitive.d.ts +2 -0
  4. package/dist/bin/primitive.js +66 -1
  5. package/dist/bin/primitive.js.map +1 -1
  6. package/dist/src/commands/admins.d.ts +2 -0
  7. package/dist/src/commands/admins.js +25 -27
  8. package/dist/src/commands/admins.js.map +1 -1
  9. package/dist/src/commands/analytics.d.ts +2 -0
  10. package/dist/src/commands/apps.d.ts +2 -0
  11. package/dist/src/commands/apps.js +28 -0
  12. package/dist/src/commands/apps.js.map +1 -1
  13. package/dist/src/commands/auth.d.ts +2 -0
  14. package/dist/src/commands/blob-buckets.d.ts +2 -0
  15. package/dist/src/commands/blob-buckets.js +30 -26
  16. package/dist/src/commands/blob-buckets.js.map +1 -1
  17. package/dist/src/commands/catalog.d.ts +2 -0
  18. package/dist/src/commands/catalog.js +17 -18
  19. package/dist/src/commands/catalog.js.map +1 -1
  20. package/dist/src/commands/collection-type-configs.d.ts +2 -0
  21. package/dist/src/commands/collection-type-configs.js +9 -9
  22. package/dist/src/commands/collection-type-configs.js.map +1 -1
  23. package/dist/src/commands/collections.d.ts +2 -0
  24. package/dist/src/commands/collections.js +33 -36
  25. package/dist/src/commands/collections.js.map +1 -1
  26. package/dist/src/commands/comparisons.d.ts +2 -0
  27. package/dist/src/commands/cron-triggers.d.ts +2 -0
  28. package/dist/src/commands/cron-triggers.js +8 -15
  29. package/dist/src/commands/cron-triggers.js.map +1 -1
  30. package/dist/src/commands/database-types.d.ts +2 -0
  31. package/dist/src/commands/database-types.js +17 -18
  32. package/dist/src/commands/database-types.js.map +1 -1
  33. package/dist/src/commands/databases.d.ts +2 -0
  34. package/dist/src/commands/databases.js +72 -45
  35. package/dist/src/commands/databases.js.map +1 -1
  36. package/dist/src/commands/documents.d.ts +2 -0
  37. package/dist/src/commands/documents.js +17 -18
  38. package/dist/src/commands/documents.js.map +1 -1
  39. package/dist/src/commands/email-templates.d.ts +2 -0
  40. package/dist/src/commands/email-templates.js +9 -9
  41. package/dist/src/commands/email-templates.js.map +1 -1
  42. package/dist/src/commands/env.d.ts +12 -0
  43. package/dist/src/commands/group-type-configs.d.ts +2 -0
  44. package/dist/src/commands/group-type-configs.js +9 -9
  45. package/dist/src/commands/group-type-configs.js.map +1 -1
  46. package/dist/src/commands/groups.d.ts +2 -0
  47. package/dist/src/commands/groups.js +17 -18
  48. package/dist/src/commands/groups.js.map +1 -1
  49. package/dist/src/commands/guides.d.ts +84 -0
  50. package/dist/src/commands/guides.js +201 -24
  51. package/dist/src/commands/guides.js.map +1 -1
  52. package/dist/src/commands/init.d.ts +17 -0
  53. package/dist/src/commands/init.js +63 -25
  54. package/dist/src/commands/init.js.map +1 -1
  55. package/dist/src/commands/integrations.d.ts +2 -0
  56. package/dist/src/commands/integrations.js +39 -23
  57. package/dist/src/commands/integrations.js.map +1 -1
  58. package/dist/src/commands/llm.d.ts +2 -0
  59. package/dist/src/commands/llm.js +4 -2
  60. package/dist/src/commands/llm.js.map +1 -1
  61. package/dist/src/commands/prompts.d.ts +2 -0
  62. package/dist/src/commands/prompts.js +33 -36
  63. package/dist/src/commands/prompts.js.map +1 -1
  64. package/dist/src/commands/rule-sets.d.ts +2 -0
  65. package/dist/src/commands/rule-sets.js +9 -9
  66. package/dist/src/commands/rule-sets.js.map +1 -1
  67. package/dist/src/commands/secrets.d.ts +2 -0
  68. package/dist/src/commands/skill.d.ts +2 -0
  69. package/dist/src/commands/sync.d.ts +99 -0
  70. package/dist/src/commands/sync.js +437 -31
  71. package/dist/src/commands/sync.js.map +1 -1
  72. package/dist/src/commands/tokens.d.ts +2 -0
  73. package/dist/src/commands/tokens.js +113 -10
  74. package/dist/src/commands/tokens.js.map +1 -1
  75. package/dist/src/commands/users.d.ts +2 -0
  76. package/dist/src/commands/users.js +41 -45
  77. package/dist/src/commands/users.js.map +1 -1
  78. package/dist/src/commands/waitlist.d.ts +2 -0
  79. package/dist/src/commands/waitlist.js +10 -10
  80. package/dist/src/commands/waitlist.js.map +1 -1
  81. package/dist/src/commands/webhooks.d.ts +2 -0
  82. package/dist/src/commands/webhooks.js +9 -9
  83. package/dist/src/commands/webhooks.js.map +1 -1
  84. package/dist/src/commands/workflows.d.ts +49 -0
  85. package/dist/src/commands/workflows.js +136 -57
  86. package/dist/src/commands/workflows.js.map +1 -1
  87. package/dist/src/lib/api-client.d.ts +1229 -0
  88. package/dist/src/lib/api-client.js +44 -11
  89. package/dist/src/lib/api-client.js.map +1 -1
  90. package/dist/src/lib/auth-flow.d.ts +8 -0
  91. package/dist/src/lib/cli-manifest.d.ts +60 -0
  92. package/dist/src/lib/cli-manifest.js +70 -0
  93. package/dist/src/lib/cli-manifest.js.map +1 -0
  94. package/dist/src/lib/config.d.ts +37 -0
  95. package/dist/src/lib/confirm-prompt.d.ts +83 -0
  96. package/dist/src/lib/confirm-prompt.js +110 -0
  97. package/dist/src/lib/confirm-prompt.js.map +1 -0
  98. package/dist/src/lib/constants.d.ts +2 -0
  99. package/dist/src/lib/crash-handlers.d.ts +20 -0
  100. package/dist/src/lib/crash-handlers.js +49 -0
  101. package/dist/src/lib/crash-handlers.js.map +1 -0
  102. package/dist/src/lib/credentials-store.d.ts +79 -0
  103. package/dist/src/lib/csv.d.ts +48 -0
  104. package/dist/src/lib/db-codegen/dbFingerprint.d.ts +10 -0
  105. package/dist/src/lib/db-codegen/dbGenerator.d.ts +111 -0
  106. package/dist/src/lib/db-codegen/dbNaming.d.ts +45 -0
  107. package/dist/src/lib/db-codegen/dbTemplates.d.ts +97 -0
  108. package/dist/src/lib/db-codegen/dbTemplates.js +31 -10
  109. package/dist/src/lib/db-codegen/dbTemplates.js.map +1 -1
  110. package/dist/src/lib/db-codegen/dbTsTypes.d.ts +78 -0
  111. package/dist/src/lib/db-codegen/dbTsTypes.js +2 -2
  112. package/dist/src/lib/db-codegen/dbTsTypes.js.map +1 -1
  113. package/dist/src/lib/env-resolver.d.ts +62 -0
  114. package/dist/src/lib/fetch.d.ts +5 -0
  115. package/dist/src/lib/generated-allowlist.d.ts +28 -0
  116. package/dist/src/lib/generated-allowlist.js +181 -0
  117. package/dist/src/lib/generated-allowlist.js.map +1 -0
  118. package/dist/src/lib/init-config.d.ts +46 -0
  119. package/dist/src/lib/init-config.js +7 -0
  120. package/dist/src/lib/init-config.js.map +1 -1
  121. package/dist/src/lib/migration-nag.d.ts +49 -0
  122. package/dist/src/lib/output.d.ts +49 -0
  123. package/dist/src/lib/output.js +25 -1
  124. package/dist/src/lib/output.js.map +1 -1
  125. package/dist/src/lib/paginate.d.ts +33 -0
  126. package/dist/src/lib/project-config.d.ts +97 -0
  127. package/dist/src/lib/refresh-admin-credentials.d.ts +65 -0
  128. package/dist/src/lib/resolve-platform.d.ts +45 -0
  129. package/dist/src/lib/resolve-platform.js +43 -0
  130. package/dist/src/lib/resolve-platform.js.map +1 -0
  131. package/dist/src/lib/skill-installer.d.ts +23 -0
  132. package/dist/src/lib/snapshots.d.ts +99 -0
  133. package/dist/src/lib/snapshots.js +357 -0
  134. package/dist/src/lib/snapshots.js.map +1 -0
  135. package/dist/src/lib/sync-paths.d.ts +72 -0
  136. package/dist/src/lib/sync-paths.js +29 -1
  137. package/dist/src/lib/sync-paths.js.map +1 -1
  138. package/dist/src/lib/template.d.ts +93 -0
  139. package/dist/src/lib/token-inject.d.ts +56 -0
  140. package/dist/src/lib/token-inject.js +204 -0
  141. package/dist/src/lib/token-inject.js.map +1 -0
  142. package/dist/src/lib/toml-database-config.d.ts +132 -0
  143. package/dist/src/lib/toml-params-validator.d.ts +95 -0
  144. package/dist/src/lib/version-check.d.ts +10 -0
  145. package/dist/src/lib/workflow-fragments.d.ts +41 -0
  146. package/dist/src/lib/workflow-toml-validator.d.ts +95 -0
  147. package/dist/src/lib/workflow-toml-validator.js +71 -130
  148. package/dist/src/lib/workflow-toml-validator.js.map +1 -1
  149. package/dist/src/types/index.d.ts +513 -0
  150. package/dist/src/validators.d.ts +64 -0
  151. package/dist/src/validators.js +63 -0
  152. package/dist/src/validators.js.map +1 -0
  153. package/package.json +10 -1
@@ -0,0 +1,83 @@
1
+ /**
2
+ * Shared interactive-confirmation helper for destructive CLI commands.
3
+ *
4
+ * Background (issue #972): every destructive command used the legacy
5
+ * `inquirer.default.prompt([{ type: "confirm", ... }])` pattern directly. When
6
+ * stdin is not a usable TTY (closed/piped/EOF/Ctrl-D), inquirer v9.3.8 throws
7
+ * `ERR_USE_AFTER_CLOSE: readline was closed` from inside a Node event-emitter
8
+ * callback (`PromptUI.onForceClose` -> `process.emit`). Because that throw
9
+ * originates in event-dispatch, it escapes both the command's own `try/catch`
10
+ * AND the top-level `.catch` in `cli/bin/primitive.ts` — and with no
11
+ * `uncaughtException` handler, Node printed only the bare `Node.js vX` banner
12
+ * and died before the API call. `cron-triggers delete` / `workflows delete`
13
+ * crashed without deleting anything.
14
+ *
15
+ * Migration (issue #999): the helper's internals are now built on the modern
16
+ * `@inquirer/prompts` `confirm()` API instead of legacy `inquirer.default.prompt`.
17
+ * The modern API rejects its promise with a *catchable* `ExitPromptError`
18
+ * (Ctrl-C / SIGINT / force-close) / `AbortPromptError` / `CancelPromptError`
19
+ * rather than crashing via `process.emit`, so the root-cause crash class is
20
+ * removed. Crucially, `@inquirer/prompts` `confirm()` has **no built-in non-TTY
21
+ * guard** — so the #972 guard below MUST stay, or the crash class regresses into
22
+ * a hang/opaque-reject on piped/closed stdin. The migration changes only the
23
+ * default prompt fn's internals; the guard, the `ConfirmPromptError` contract,
24
+ * and the public `confirmPrompt()` signature are preserved.
25
+ *
26
+ * `confirmPrompt()` fixes the cause:
27
+ * 1. Non-TTY guard — if stdin is not a TTY, it never touches the prompt.
28
+ * Instead it throws a catchable `ConfirmPromptError` with an actionable
29
+ * message ("re-run with --yes ..."). Per the maintainer decision on #972
30
+ * this is a hard error, never a silent mutate or silent abort.
31
+ * 2. Local try/catch — in a real TTY the prompt is awaited inside a try/catch
32
+ * so an abort/EOF (`ExitPromptError` / `AbortPromptError` /
33
+ * `CancelPromptError`, or legacy `ERR_USE_AFTER_CLOSE`) surfaces as a clean
34
+ * `ConfirmPromptError`, not a stack trace or crash.
35
+ *
36
+ * Commands should call this instead of inquirer directly, e.g.:
37
+ *
38
+ * if (!options.yes) {
39
+ * try {
40
+ * const ok = await confirmPrompt(`Delete cron trigger ${id}?`);
41
+ * if (!ok) { info("Cancelled."); return; }
42
+ * } catch (err) {
43
+ * error(err.message);
44
+ * process.exit(1);
45
+ * }
46
+ * }
47
+ */
48
+ /** Error thrown when a confirmation can't be obtained interactively. */
49
+ export declare class ConfirmPromptError extends Error {
50
+ constructor(message: string);
51
+ }
52
+ /**
53
+ * Shape of the `@inquirer/prompts` `confirm()` function. Injectable for testing.
54
+ *
55
+ * Note (#999): this shifted from the legacy inquirer questions-array →
56
+ * `{ confirm: boolean }` shape to the modern bare `{ message, default }` →
57
+ * `Promise<boolean>` shape.
58
+ */
59
+ export type PromptFn = (config: {
60
+ message: string;
61
+ default?: boolean;
62
+ }) => Promise<boolean>;
63
+ export interface ConfirmPromptOptions {
64
+ /** Default answer when the user just presses enter. Defaults to false. */
65
+ defaultValue?: boolean;
66
+ /**
67
+ * Override the underlying prompt function (for tests). Defaults to lazily
68
+ * importing `@inquirer/prompts` `confirm`.
69
+ */
70
+ promptFn?: PromptFn;
71
+ /**
72
+ * Override TTY detection (for tests). Defaults to `process.stdin.isTTY`.
73
+ */
74
+ isTTY?: boolean;
75
+ }
76
+ /**
77
+ * Ask the user a yes/no confirmation question.
78
+ *
79
+ * @returns `true` if the user confirms, `false` if they decline.
80
+ * @throws {ConfirmPromptError} when stdin is not an interactive TTY (so the
81
+ * prompt can't be shown), or when the underlying prompt aborts/crashes.
82
+ */
83
+ export declare function confirmPrompt(message: string, options?: ConfirmPromptOptions): Promise<boolean>;
@@ -0,0 +1,110 @@
1
+ /**
2
+ * Shared interactive-confirmation helper for destructive CLI commands.
3
+ *
4
+ * Background (issue #972): every destructive command used the legacy
5
+ * `inquirer.default.prompt([{ type: "confirm", ... }])` pattern directly. When
6
+ * stdin is not a usable TTY (closed/piped/EOF/Ctrl-D), inquirer v9.3.8 throws
7
+ * `ERR_USE_AFTER_CLOSE: readline was closed` from inside a Node event-emitter
8
+ * callback (`PromptUI.onForceClose` -> `process.emit`). Because that throw
9
+ * originates in event-dispatch, it escapes both the command's own `try/catch`
10
+ * AND the top-level `.catch` in `cli/bin/primitive.ts` — and with no
11
+ * `uncaughtException` handler, Node printed only the bare `Node.js vX` banner
12
+ * and died before the API call. `cron-triggers delete` / `workflows delete`
13
+ * crashed without deleting anything.
14
+ *
15
+ * Migration (issue #999): the helper's internals are now built on the modern
16
+ * `@inquirer/prompts` `confirm()` API instead of legacy `inquirer.default.prompt`.
17
+ * The modern API rejects its promise with a *catchable* `ExitPromptError`
18
+ * (Ctrl-C / SIGINT / force-close) / `AbortPromptError` / `CancelPromptError`
19
+ * rather than crashing via `process.emit`, so the root-cause crash class is
20
+ * removed. Crucially, `@inquirer/prompts` `confirm()` has **no built-in non-TTY
21
+ * guard** — so the #972 guard below MUST stay, or the crash class regresses into
22
+ * a hang/opaque-reject on piped/closed stdin. The migration changes only the
23
+ * default prompt fn's internals; the guard, the `ConfirmPromptError` contract,
24
+ * and the public `confirmPrompt()` signature are preserved.
25
+ *
26
+ * `confirmPrompt()` fixes the cause:
27
+ * 1. Non-TTY guard — if stdin is not a TTY, it never touches the prompt.
28
+ * Instead it throws a catchable `ConfirmPromptError` with an actionable
29
+ * message ("re-run with --yes ..."). Per the maintainer decision on #972
30
+ * this is a hard error, never a silent mutate or silent abort.
31
+ * 2. Local try/catch — in a real TTY the prompt is awaited inside a try/catch
32
+ * so an abort/EOF (`ExitPromptError` / `AbortPromptError` /
33
+ * `CancelPromptError`, or legacy `ERR_USE_AFTER_CLOSE`) surfaces as a clean
34
+ * `ConfirmPromptError`, not a stack trace or crash.
35
+ *
36
+ * Commands should call this instead of inquirer directly, e.g.:
37
+ *
38
+ * if (!options.yes) {
39
+ * try {
40
+ * const ok = await confirmPrompt(`Delete cron trigger ${id}?`);
41
+ * if (!ok) { info("Cancelled."); return; }
42
+ * } catch (err) {
43
+ * error(err.message);
44
+ * process.exit(1);
45
+ * }
46
+ * }
47
+ */
48
+ /** Error thrown when a confirmation can't be obtained interactively. */
49
+ export class ConfirmPromptError extends Error {
50
+ constructor(message) {
51
+ super(message);
52
+ this.name = "ConfirmPromptError";
53
+ }
54
+ }
55
+ async function defaultPromptFn(config) {
56
+ const { confirm } = await import("@inquirer/prompts");
57
+ return confirm(config);
58
+ }
59
+ /**
60
+ * Names of the catchable abort errors thrown by `@inquirer/prompts` when the
61
+ * user interrupts a prompt (Ctrl-C / SIGINT / force-close / abort signal /
62
+ * cancel). We match on `name` (not just `instanceof`) so detection is robust to
63
+ * the error class being duplicated across module realms.
64
+ */
65
+ const ABORT_ERROR_NAMES = new Set([
66
+ "ExitPromptError",
67
+ "AbortPromptError",
68
+ "CancelPromptError",
69
+ ]);
70
+ /**
71
+ * Ask the user a yes/no confirmation question.
72
+ *
73
+ * @returns `true` if the user confirms, `false` if they decline.
74
+ * @throws {ConfirmPromptError} when stdin is not an interactive TTY (so the
75
+ * prompt can't be shown), or when the underlying prompt aborts/crashes.
76
+ */
77
+ export async function confirmPrompt(message, options = {}) {
78
+ const isTTY = options.isTTY ?? Boolean(process.stdin.isTTY);
79
+ if (!isTTY) {
80
+ throw new ConfirmPromptError("No interactive terminal available to confirm this action. " +
81
+ "Re-run with --yes (-y) to skip the confirmation prompt in " +
82
+ "non-interactive contexts (CI, pipes, agents).");
83
+ }
84
+ const promptFn = options.promptFn ?? defaultPromptFn;
85
+ try {
86
+ const confirmed = await promptFn({
87
+ message,
88
+ default: options.defaultValue ?? false,
89
+ });
90
+ return Boolean(confirmed);
91
+ }
92
+ catch (err) {
93
+ // A deliberate abort/EOF/Ctrl-C/Ctrl-D — either the modern
94
+ // `@inquirer/prompts` `ExitPromptError`/`AbortPromptError`/`CancelPromptError`
95
+ // (matched by name to survive cross-realm `instanceof` mismatches), or the
96
+ // legacy inquirer `ERR_USE_AFTER_CLOSE`. Surface as a clean, actionable
97
+ // error so the caller exits 1 with no stack trace / bare `Node.js vX` banner.
98
+ if (ABORT_ERROR_NAMES.has(err?.name) ||
99
+ err?.code === "ERR_USE_AFTER_CLOSE") {
100
+ throw new ConfirmPromptError("Confirmation cancelled (no interactive input). " +
101
+ "Re-run with --yes (-y) to skip confirmation.");
102
+ }
103
+ // Any other prompt failure: still surface it as a clean, catchable error
104
+ // rather than letting it escape as an uncaught crash.
105
+ throw new ConfirmPromptError(err?.message
106
+ ? `Confirmation prompt failed: ${err.message}`
107
+ : "Confirmation prompt failed.");
108
+ }
109
+ }
110
+ //# sourceMappingURL=confirm-prompt.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"confirm-prompt.js","sourceRoot":"","sources":["../../../src/lib/confirm-prompt.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8CG;AAEH,wEAAwE;AACxE,MAAM,OAAO,kBAAmB,SAAQ,KAAK;IAC3C,YAAY,OAAe;QACzB,KAAK,CAAC,OAAO,CAAC,CAAC;QACf,IAAI,CAAC,IAAI,GAAG,oBAAoB,CAAC;IACnC,CAAC;CACF;AA4BD,KAAK,UAAU,eAAe,CAAC,MAG9B;IACC,MAAM,EAAE,OAAO,EAAE,GAAG,MAAM,MAAM,CAAC,mBAAmB,CAAC,CAAC;IACtD,OAAO,OAAO,CAAC,MAAM,CAAC,CAAC;AACzB,CAAC;AAED;;;;;GAKG;AACH,MAAM,iBAAiB,GAAG,IAAI,GAAG,CAAC;IAChC,iBAAiB;IACjB,kBAAkB;IAClB,mBAAmB;CACpB,CAAC,CAAC;AAEH;;;;;;GAMG;AACH,MAAM,CAAC,KAAK,UAAU,aAAa,CACjC,OAAe,EACf,UAAgC,EAAE;IAElC,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,IAAI,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IAE5D,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,MAAM,IAAI,kBAAkB,CAC1B,4DAA4D;YAC1D,4DAA4D;YAC5D,+CAA+C,CAClD,CAAC;IACJ,CAAC;IAED,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,IAAI,eAAe,CAAC;IAErD,IAAI,CAAC;QACH,MAAM,SAAS,GAAG,MAAM,QAAQ,CAAC;YAC/B,OAAO;YACP,OAAO,EAAE,OAAO,CAAC,YAAY,IAAI,KAAK;SACvC,CAAC,CAAC;QACH,OAAO,OAAO,CAAC,SAAS,CAAC,CAAC;IAC5B,CAAC;IAAC,OAAO,GAAQ,EAAE,CAAC;QAClB,2DAA2D;QAC3D,+EAA+E;QAC/E,2EAA2E;QAC3E,wEAAwE;QACxE,8EAA8E;QAC9E,IACE,iBAAiB,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC;YAChC,GAAG,EAAE,IAAI,KAAK,qBAAqB,EACnC,CAAC;YACD,MAAM,IAAI,kBAAkB,CAC1B,iDAAiD;gBAC/C,8CAA8C,CACjD,CAAC;QACJ,CAAC;QACD,yEAAyE;QACzE,sDAAsD;QACtD,MAAM,IAAI,kBAAkB,CAC1B,GAAG,EAAE,OAAO;YACV,CAAC,CAAC,+BAA+B,GAAG,CAAC,OAAO,EAAE;YAC9C,CAAC,CAAC,6BAA6B,CAClC,CAAC;IACJ,CAAC;AACH,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare const PRODUCTION_SERVER_URL = "https://primitiveapi.com";
2
+ export declare const DEFAULT_SERVER_URL: string;
@@ -0,0 +1,20 @@
1
+ /**
2
+ * Global crash handlers for the CLI entrypoint (issue #972, Option A).
3
+ *
4
+ * The CLI previously registered no `uncaughtException` / `unhandledRejection`
5
+ * handlers. When an error escaped the top-level `.catch` in
6
+ * `cli/bin/primitive.ts` — e.g. the `ERR_USE_AFTER_CLOSE` thrown from inside an
7
+ * inquirer event-emitter callback when stdin isn't a usable TTY — Node printed
8
+ * only its bare `Node.js vX` version banner and died. That was the exact
9
+ * symptom in #972: a delete command crashed with no error message.
10
+ *
11
+ * These handlers are a narrow safety net: print a clean `✗ <message>` to stderr
12
+ * (matching `error()` in output.ts) and exit non-zero, so NO command can ever
13
+ * bare-crash again. They intentionally do not try to recover — they only make
14
+ * an already-fatal error legible.
15
+ */
16
+ /**
17
+ * Register process-level handlers that convert otherwise-fatal escaped errors
18
+ * into a clean stderr message + non-zero exit. Idempotent.
19
+ */
20
+ export declare function installGlobalCrashHandlers(): void;
@@ -0,0 +1,49 @@
1
+ /**
2
+ * Global crash handlers for the CLI entrypoint (issue #972, Option A).
3
+ *
4
+ * The CLI previously registered no `uncaughtException` / `unhandledRejection`
5
+ * handlers. When an error escaped the top-level `.catch` in
6
+ * `cli/bin/primitive.ts` — e.g. the `ERR_USE_AFTER_CLOSE` thrown from inside an
7
+ * inquirer event-emitter callback when stdin isn't a usable TTY — Node printed
8
+ * only its bare `Node.js vX` version banner and died. That was the exact
9
+ * symptom in #972: a delete command crashed with no error message.
10
+ *
11
+ * These handlers are a narrow safety net: print a clean `✗ <message>` to stderr
12
+ * (matching `error()` in output.ts) and exit non-zero, so NO command can ever
13
+ * bare-crash again. They intentionally do not try to recover — they only make
14
+ * an already-fatal error legible.
15
+ */
16
+ import { error } from "./output.js";
17
+ function describe(err) {
18
+ if (err && typeof err === "object") {
19
+ const e = err;
20
+ // The signature crash from a non-TTY confirm prompt — give actionable help.
21
+ if (e.code === "ERR_USE_AFTER_CLOSE") {
22
+ return ("No interactive terminal available to confirm this action. " +
23
+ "Re-run with --yes (-y) to skip the confirmation prompt in " +
24
+ "non-interactive contexts (CI, pipes, agents).");
25
+ }
26
+ if (e.message)
27
+ return e.message;
28
+ }
29
+ return "An unexpected error occurred.";
30
+ }
31
+ let installed = false;
32
+ /**
33
+ * Register process-level handlers that convert otherwise-fatal escaped errors
34
+ * into a clean stderr message + non-zero exit. Idempotent.
35
+ */
36
+ export function installGlobalCrashHandlers() {
37
+ if (installed)
38
+ return;
39
+ installed = true;
40
+ process.on("uncaughtException", (err) => {
41
+ error(describe(err));
42
+ process.exit(1);
43
+ });
44
+ process.on("unhandledRejection", (reason) => {
45
+ error(describe(reason));
46
+ process.exit(1);
47
+ });
48
+ }
49
+ //# sourceMappingURL=crash-handlers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"crash-handlers.js","sourceRoot":"","sources":["../../../src/lib/crash-handlers.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAEH,OAAO,EAAE,KAAK,EAAE,MAAM,aAAa,CAAC;AAEpC,SAAS,QAAQ,CAAC,GAAY;IAC5B,IAAI,GAAG,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE,CAAC;QACnC,MAAM,CAAC,GAAG,GAA0C,CAAC;QACrD,4EAA4E;QAC5E,IAAI,CAAC,CAAC,IAAI,KAAK,qBAAqB,EAAE,CAAC;YACrC,OAAO,CACL,4DAA4D;gBAC5D,4DAA4D;gBAC5D,+CAA+C,CAChD,CAAC;QACJ,CAAC;QACD,IAAI,CAAC,CAAC,OAAO;YAAE,OAAO,CAAC,CAAC,OAAO,CAAC;IAClC,CAAC;IACD,OAAO,+BAA+B,CAAC;AACzC,CAAC;AAED,IAAI,SAAS,GAAG,KAAK,CAAC;AAEtB;;;GAGG;AACH,MAAM,UAAU,0BAA0B;IACxC,IAAI,SAAS;QAAE,OAAO;IACtB,SAAS,GAAG,IAAI,CAAC;IAEjB,OAAO,CAAC,EAAE,CAAC,mBAAmB,EAAE,CAAC,GAAG,EAAE,EAAE;QACtC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC;QACrB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC,CAAC,CAAC;IAEH,OAAO,CAAC,EAAE,CAAC,oBAAoB,EAAE,CAAC,MAAM,EAAE,EAAE;QAC1C,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;QACxB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,79 @@
1
+ /**
2
+ * Credentials storage with dual-mode support.
3
+ *
4
+ * Mode A (project mode): credentials live at
5
+ * <projectRoot>/.primitive/credentials.json
6
+ * with per-environment slots:
7
+ * {
8
+ * "environments": {
9
+ * "dev": { "accessToken": "...", "refreshToken": "...", ... },
10
+ * "prod": { ... }
11
+ * }
12
+ * }
13
+ *
14
+ * Mode B (legacy mode): no .primitive/config.json in scope, so we fall back to
15
+ * ~/.primitive/credentials.json
16
+ * with the flat Credentials shape the CLI has always used.
17
+ *
18
+ * The legacy mode is still the "global default" for users who haven't
19
+ * run `primitive init` yet.
20
+ */
21
+ import type { Credentials } from "../types/index.js";
22
+ export declare const PROJECT_LOCAL_DIR = ".primitive";
23
+ export declare const PROJECT_CREDENTIALS_FILENAME = "credentials.json";
24
+ /** Returns true when a .primitive/config.json is in scope (project mode active). */
25
+ export declare function isProjectMode(): boolean;
26
+ /**
27
+ * Returns the path to the credentials file that will be used, based on
28
+ * whether we're in project mode or legacy mode.
29
+ */
30
+ export declare function getCredentialsFilePath(): string;
31
+ /**
32
+ * Reads credentials for the current environment (project mode) or the
33
+ * legacy global file. Returns the standard Credentials shape so existing
34
+ * callers don't need to change.
35
+ *
36
+ * In project mode, we blend the per-env stored creds with the serverUrl
37
+ * and appId from .primitive/config.json to produce the Credentials the rest of
38
+ * the CLI expects.
39
+ */
40
+ export declare function loadCredentialsStore(): Credentials | null;
41
+ /**
42
+ * Saves credentials. In project mode, writes to the per-env slot in the
43
+ * project credentials file. In legacy mode, writes the global file.
44
+ *
45
+ * The serverUrl on the incoming Credentials is used as-is for legacy
46
+ * mode, and ignored in project mode (the env's apiUrl is canonical).
47
+ */
48
+ export declare function saveCredentialsStore(credentials: Credentials): void;
49
+ /**
50
+ * Clears credentials for the current environment (project mode) or
51
+ * deletes the legacy file (legacy mode).
52
+ */
53
+ export declare function clearCredentialsStore(): void;
54
+ /**
55
+ * Removes the credentials slot for the named environment in a specific
56
+ * project's credentials file. Used by `env remove` so stale tokens from a
57
+ * previously-removed environment can never be silently re-used if the same
58
+ * name is later added back. Never throws.
59
+ */
60
+ export declare function clearCredentialsForEnvInProject(projectRoot: string, envName: string): void;
61
+ /**
62
+ * Lists all environments that have stored credentials in project mode.
63
+ * Returns an empty array in legacy mode.
64
+ */
65
+ export declare function listAuthenticatedEnvironments(): string[];
66
+ /**
67
+ * Updates the "current app" in credential storage. In project mode with a
68
+ * project-level appId this is a no-op (the project config is canonical).
69
+ */
70
+ export declare function setCurrentAppStore(appId: string, appName?: string): void;
71
+ /**
72
+ * Clears the "current app" for the current environment.
73
+ */
74
+ export declare function clearCurrentAppStore(): void;
75
+ /**
76
+ * Returns the legacy credentials file, if one exists. Used by migration
77
+ * prompts that want to detect a pre-project-config install.
78
+ */
79
+ export declare function peekLegacyCredentials(): Credentials | null;
@@ -0,0 +1,48 @@
1
+ /**
2
+ * Parse a CSV string into an array of row objects keyed by header.
3
+ *
4
+ * Quote-aware: respects double-quoted fields that may contain the delimiter,
5
+ * embedded newlines, and escaped double-quotes (`""`). Empty-string cells are
6
+ * omitted from the row object (matching the client). Headers-only / empty
7
+ * input yields `[]`.
8
+ */
9
+ export declare function parseCsv(csv: string, delimiter?: string): Record<string, string>[];
10
+ /**
11
+ * Coerce a string cell to a target type. Mirrors the client's `coerceValue`:
12
+ * a failed number coercion returns the original string (so the server's
13
+ * schema validation surfaces a precise error rather than a silent bad write).
14
+ */
15
+ export declare function coerceValue(value: string, type: string): any;
16
+ /**
17
+ * Rename row keys per a `{ "CSV Header": "fieldName" }` map. Keys not present
18
+ * in the map are preserved. No-op when `map` is undefined.
19
+ */
20
+ export declare function applyColumnMap(rows: Record<string, any>[], map: Record<string, string> | undefined): Record<string, any>[];
21
+ export interface BuildRowsOptions {
22
+ /** Model name passed in each batch item's params. */
23
+ modelName: string;
24
+ /** Type coercion overrides keyed by (post-column-map) field name. */
25
+ types?: Record<string, string>;
26
+ /** CSV column whose value to use as the record id (else a ULID is generated). */
27
+ idColumn?: string;
28
+ }
29
+ export interface BatchItem {
30
+ params: {
31
+ modelName: string;
32
+ id: string;
33
+ data: Record<string, any>;
34
+ };
35
+ }
36
+ export interface BuildRowsResult {
37
+ batch: BatchItem[];
38
+ rowCount: number;
39
+ }
40
+ /**
41
+ * Apply type coercion, assign each row an id (from `idColumn` or a generated
42
+ * ULID), and shape rows into `executeBatch` items
43
+ * (`{ params: { modelName, id, data } }`). Mirrors steps 5–7 of the client's
44
+ * `importCsv`.
45
+ */
46
+ export declare function buildRows(rows: Record<string, any>[], options: BuildRowsOptions): BuildRowsResult;
47
+ /** Split an array into fixed-size chunks. */
48
+ export declare function chunk<T>(items: T[], size: number): T[][];
@@ -0,0 +1,10 @@
1
+ /**
2
+ * Short fingerprint for a database-type TOML string (issue #814).
3
+ *
4
+ * Stamped in the header of every generated file so a human (or the
5
+ * `--check` CI guard) can tell at a glance whether the source TOML and the
6
+ * generated output are in sync. Mirrors codegen-v2's `fingerprintToml`
7
+ * (`packages/js-bao/src/cli/v2/fingerprint.ts`): 16 hex chars of SHA-256 —
8
+ * enough for a sanity check, short enough to fit on one comment line.
9
+ */
10
+ export declare function fingerprintToml(tomlContent: string): string;
@@ -0,0 +1,111 @@
1
+ /**
2
+ * `databases codegen` orchestrator (issue #814).
3
+ *
4
+ * Pure-ish core: given a set of database-type TOML strings + a target
5
+ * output dir, decides what `*.generated.ts` files to write (in memory),
6
+ * then either (a) writes them, or (b) compares them against on-disk for
7
+ * `--check`. No commander dependency, no `process.exit`.
8
+ *
9
+ * Reuses the codegen-v2 machinery:
10
+ * - `loadSchemaFromTomlString` (the shared `[models.*]` parser) for record
11
+ * interfaces — the database-type `[models.*]` blocks round-trip through
12
+ * the same loader and the same `FieldType` vocabulary.
13
+ * - The banner + fingerprint header pattern and stale-output cleanup.
14
+ *
15
+ * Adds the OPERATIONS half (op-params interfaces + per-op result aliases)
16
+ * that codegen-v2 has no analog for. Op return contracts are NOT stored in
17
+ * the TOML, so result aliases are generic shapes keyed off `op.type`.
18
+ */
19
+ export interface DbCodegenInput {
20
+ /** Database type name (drives the output filename). */
21
+ databaseType: string;
22
+ /** Path to the source `database-types/<type>.toml` file. */
23
+ tomlPath: string;
24
+ /** Raw TOML content (already read from disk). */
25
+ tomlContent: string;
26
+ }
27
+ export interface GenerateDbTypesOptions {
28
+ /** One entry per database-type TOML file to generate from. */
29
+ inputs: DbCodegenInput[];
30
+ /** Directory where `<type>.generated.ts` files are written. */
31
+ outputDir: string;
32
+ /**
33
+ * If true, do not write to disk. Compare the would-be output to what's on
34
+ * disk and return a non-empty `mismatches` list if any file would change.
35
+ */
36
+ check?: boolean;
37
+ /**
38
+ * Set when the run was filtered to a single database type (the
39
+ * `primitive databases codegen <type>` argument). The `inputs` set is then
40
+ * PARTIAL — it does not represent the full set of owned generated files — so
41
+ * stale-output cleanup (and `--check` stale detection) must be scoped to the
42
+ * filtered type's own file and must NOT touch / flag sibling types'
43
+ * `*.generated.ts` files. The unfiltered "generate all" run leaves this unset
44
+ * and keeps full orphan cleanup.
45
+ */
46
+ singleType?: boolean;
47
+ }
48
+ export interface DbCodegenResult {
49
+ /** Files written (or that would be written under `--check`). */
50
+ writtenFiles: string[];
51
+ /** Stale `*.generated.ts` files deleted (or flagged stale under `--check`). */
52
+ deletedFiles: string[];
53
+ /** Only populated under `--check`: files whose on-disk content is out of date. */
54
+ mismatches: DbCheckMismatch[];
55
+ }
56
+ export interface DbCheckMismatch {
57
+ filePath: string;
58
+ reason: "missing" | "differs" | "stale";
59
+ }
60
+ /**
61
+ * Render the in-memory `.generated.ts` content for a single database-type
62
+ * TOML. Exported so tests can assert on the rendered string directly.
63
+ */
64
+ export declare function renderDbTypeFile(input: DbCodegenInput): string;
65
+ /**
66
+ * Per-model result of {@link inferRequiredRecordFields}. Shaped as an object
67
+ * (rather than a bare `Set`) so it can grow additional inferred metadata later
68
+ * without changing the call-site contract — e.g. future alignment with #845's
69
+ * server-side op-params analysis. For now it carries only the set of record
70
+ * fields the operation params prove are always supplied at create time.
71
+ */
72
+ export interface InferredRecordModelInfo {
73
+ /** Record field names promoted to non-optional by op-params inference. */
74
+ fields: Set<string>;
75
+ }
76
+ /**
77
+ * Infer which record fields are always supplied at create time, by reading the
78
+ * `save` operation params (issue #842).
79
+ *
80
+ * A field is promoted to required for a model only if:
81
+ * - at least one `save` (create) op writes it from a bare `$params.X`
82
+ * binding, AND
83
+ * - EVERY `save` op that writes the field binds it to a param declared
84
+ * `required: true` (the "ALL writers agree" rule).
85
+ *
86
+ * Deliberately conservative:
87
+ * - `patch` (and every non-`save` inner op) is excluded — partial updates
88
+ * don't guarantee creation-time presence.
89
+ * - whole-object saves (`data = "$params.X"`, a string rather than a
90
+ * per-field map) carry no field-level mapping, so no field can be promoted
91
+ * from them (Q4 skip).
92
+ * - only an EXACT top-level bare `$params.X` value counts (the whole binding
93
+ * value must match `^\$params\.X$`, and `X` must be a declared top-level
94
+ * param). A literal, computed, partially-interpolated value, or a nested
95
+ * path (`$params.obj.sub`) never promotes a field — a nested sub-property
96
+ * may be absent even when the top-level object param is required. This is
97
+ * deliberately stricter than the CLI's lenient `collectParamRefs`
98
+ * first-segment rule, which is fine for its validation use but would
99
+ * wrongly promote nested bindings here.
100
+ * - inner ops are matched to the record model by the inner op's own
101
+ * `modelName` when present, falling back to the enclosing operation's
102
+ * `modelName`.
103
+ *
104
+ * Pure: no I/O, no mutation of the input. The caller OR's the result with the
105
+ * schema-level `required` flag, so inference only ever ADDS requiredness.
106
+ */
107
+ export declare function inferRequiredRecordFields(operations: any[]): Map<string, InferredRecordModelInfo>;
108
+ /**
109
+ * Run codegen end-to-end across one or more database-type TOML inputs.
110
+ */
111
+ export declare function generateDbTypes(options: GenerateDbTypesOptions): Promise<DbCodegenResult>;
@@ -0,0 +1,45 @@
1
+ /**
2
+ * Identifier-naming helpers for the `databases codegen` generator (issue #814).
3
+ *
4
+ * Record interfaces mirror codegen-v2's class-name resolution
5
+ * (`packages/js-bao/src/cli/v2/pluralization.ts`): honor a `class_name`
6
+ * override, else singularize a snake_case plural to PascalCase. The
7
+ * singularization algorithm below is a faithful copy of codegen-v2's
8
+ * `singularizeToPascalCase`; it is inlined rather than imported because the
9
+ * CLI builds standalone (`tsc -p cli/tsconfig.json`, `rootDir: "."`) and
10
+ * cannot reach into the vendored `packages/js-bao/src` tree.
11
+ *
12
+ * It DIVERGES in the fallback: codegen-v2 THROWS when a model name doesn't
13
+ * look like a recognizable plural (ORM models are expected to be plural).
14
+ * Database-type model names are author-curated and frequently already
15
+ * singular / PascalCase (e.g. `[models.User]`), so we PascalCase the name
16
+ * verbatim rather than failing the whole run.
17
+ *
18
+ * Op interfaces/aliases derive from the operation name (PascalCase) +
19
+ * `Params` / `Result` suffix (e.g. `saveAccount` → `SaveAccountParams`,
20
+ * `SaveAccountResult`).
21
+ */
22
+ /**
23
+ * Resolve the record-interface name for a model. Honors a `class_name`
24
+ * override, then tries plural→singular Pascal, then falls back to a verbatim
25
+ * PascalCase of the model name (no throw).
26
+ */
27
+ export declare function resolveRecordInterfaceName(modelName: string, classNameOverride: string | undefined): string;
28
+ /** `saveAccount` → `SaveAccountParams`. */
29
+ export declare function opParamsInterfaceName(opName: string): string;
30
+ /** `saveAccount` → `SaveAccountResult`. */
31
+ export declare function opResultAliasName(opName: string): string;
32
+ /**
33
+ * Convert a snake_case plural model name (`user_prefs`, `categories`) into a
34
+ * PascalCase singular interface name (`UserPref`, `Category`). Returns null
35
+ * if no suffix rule matches the input. Faithful copy of codegen-v2's
36
+ * `singularizeToPascalCase`.
37
+ */
38
+ export declare function singularizeToPascalCase(snakePlural: string): string | null;
39
+ /**
40
+ * PascalCase an identifier: splits on non-alphanumeric separators
41
+ * (`_`, `-`, spaces) and capitalizes each segment, preserving any existing
42
+ * internal casing (`saveAccount` → `SaveAccount`, `save_account` →
43
+ * `SaveAccount`).
44
+ */
45
+ export declare function pascalCase(name: string): string;