@lunora/cli 0.0.0 → 1.0.0-alpha.10

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 (75) hide show
  1. package/LICENSE.md +105 -0
  2. package/README.md +109 -9
  3. package/__assets__/package-og.svg +14 -0
  4. package/dist/bin.mjs +11 -0
  5. package/dist/index.d.mts +956 -0
  6. package/dist/index.d.ts +956 -0
  7. package/dist/index.mjs +19 -0
  8. package/dist/packem_chunks/handler.mjs +150 -0
  9. package/dist/packem_chunks/handler10.mjs +22 -0
  10. package/dist/packem_chunks/handler11.mjs +192 -0
  11. package/dist/packem_chunks/handler12.mjs +131 -0
  12. package/dist/packem_chunks/handler13.mjs +65 -0
  13. package/dist/packem_chunks/handler14.mjs +58 -0
  14. package/dist/packem_chunks/handler15.mjs +79 -0
  15. package/dist/packem_chunks/handler16.mjs +43 -0
  16. package/dist/packem_chunks/handler17.mjs +105 -0
  17. package/dist/packem_chunks/handler18.mjs +170 -0
  18. package/dist/packem_chunks/handler19.mjs +89 -0
  19. package/dist/packem_chunks/handler2.mjs +114 -0
  20. package/dist/packem_chunks/handler20.mjs +94 -0
  21. package/dist/packem_chunks/handler21.mjs +311 -0
  22. package/dist/packem_chunks/handler3.mjs +204 -0
  23. package/dist/packem_chunks/handler4.mjs +33 -0
  24. package/dist/packem_chunks/handler5.mjs +49 -0
  25. package/dist/packem_chunks/handler6.mjs +91 -0
  26. package/dist/packem_chunks/handler7.mjs +42 -0
  27. package/dist/packem_chunks/handler8.mjs +174 -0
  28. package/dist/packem_chunks/handler9.mjs +16 -0
  29. package/dist/packem_chunks/planDevCommand.mjs +500 -0
  30. package/dist/packem_chunks/runCodegenCommand.mjs +52 -0
  31. package/dist/packem_chunks/runDeployCommand.mjs +504 -0
  32. package/dist/packem_chunks/runInitCommand.mjs +1498 -0
  33. package/dist/packem_chunks/runMigrateGenerateCommand.mjs +397 -0
  34. package/dist/packem_chunks/runResetCommand.mjs +41 -0
  35. package/dist/packem_chunks/runRpcCommand.mjs +68 -0
  36. package/dist/packem_shared/COMMANDS-D3h9Iwvl.mjs +944 -0
  37. package/dist/packem_shared/DEFAULT_IMPORT_BATCH_SIZE-Ck-2bU08.mjs +244 -0
  38. package/dist/packem_shared/admin-url-4UzT-CI4.mjs +19 -0
  39. package/dist/packem_shared/api-spec-CtA6ilu4.mjs +13 -0
  40. package/dist/packem_shared/buildRegistryIndex-BcYe607_.mjs +38 -0
  41. package/dist/packem_shared/command-BC30oSBW.mjs +14 -0
  42. package/dist/packem_shared/commands-DPKWlqqX.mjs +812 -0
  43. package/dist/packem_shared/createLogger-B40gPzQo.mjs +78 -0
  44. package/dist/packem_shared/createRecordingSpawner-DxI3mebw.mjs +43 -0
  45. package/dist/packem_shared/detect-package-manager-DYp7n3mJ.mjs +61 -0
  46. package/dist/packem_shared/diffSnapshots-BeDvvNiF.mjs +161 -0
  47. package/dist/packem_shared/docker-hMQ97KSQ.mjs +21 -0
  48. package/dist/packem_shared/insertSchemaExtension-BuzF6-t2.mjs +59 -0
  49. package/dist/packem_shared/open-url-Dfq6fAyT.mjs +41 -0
  50. package/dist/packem_shared/output-format-wUvAN6AL.mjs +17 -0
  51. package/dist/packem_shared/parseArgs-YXFuKdEk.mjs +56 -0
  52. package/dist/packem_shared/parseManifest--vZf2FY1.mjs +94 -0
  53. package/dist/packem_shared/resolve-target-qbsJ_5sF.mjs +16 -0
  54. package/dist/packem_shared/runAddCommand-CTRA_JlL.mjs +4 -0
  55. package/dist/packem_shared/schema-drift-gate-BtBt0as0.mjs +79 -0
  56. package/dist/packem_shared/schemaIrToSnapshot-DdsljJT-.mjs +43 -0
  57. package/dist/packem_shared/storage-2RJBhUC4.mjs +84 -0
  58. package/dist/packem_shared/tui-prompts-DEiPCKV-.mjs +661 -0
  59. package/dist/packem_shared/wrangler-name-cy4yhm9j.mjs +12 -0
  60. package/package.json +62 -18
  61. package/skills/README.md +29 -0
  62. package/skills/lunora/SKILL.md +83 -0
  63. package/skills/lunora-create-package/SKILL.md +129 -0
  64. package/skills/lunora-deploy/SKILL.md +150 -0
  65. package/skills/lunora-functions/SKILL.md +182 -0
  66. package/skills/lunora-migration-helper/SKILL.md +194 -0
  67. package/skills/lunora-performance-audit/SKILL.md +143 -0
  68. package/skills/lunora-quickstart/SKILL.md +240 -0
  69. package/skills/lunora-realtime/SKILL.md +177 -0
  70. package/skills/lunora-setup-auth/SKILL.md +170 -0
  71. package/skills/lunora-setup-hyperdrive/SKILL.md +154 -0
  72. package/skills/lunora-setup-hyperdrive-global/SKILL.md +171 -0
  73. package/skills/lunora-setup-mail/SKILL.md +151 -0
  74. package/skills/lunora-setup-scheduler/SKILL.md +157 -0
  75. package/skills/lunora-setup-storage/SKILL.md +158 -0
@@ -0,0 +1,79 @@
1
+ import { existsSync, readFileSync, writeFileSync } from 'node:fs';
2
+ import { evaluateSchemaDrift, parseSchemaSnapshot, SchemaSnapshotParseError, serializeSchemaSnapshot } from '@lunora/codegen';
3
+
4
+ const readBaseline = (snapshotPath) => {
5
+ if (!existsSync(snapshotPath)) {
6
+ return { status: "absent" };
7
+ }
8
+ let content;
9
+ try {
10
+ content = readFileSync(snapshotPath, "utf8");
11
+ } catch {
12
+ return { status: "corrupt" };
13
+ }
14
+ try {
15
+ const snapshot = parseSchemaSnapshot(content);
16
+ return snapshot === void 0 ? { status: "corrupt" } : { snapshot, status: "ok" };
17
+ } catch (error) {
18
+ if (error instanceof SchemaSnapshotParseError) {
19
+ return { status: "corrupt" };
20
+ }
21
+ throw error;
22
+ }
23
+ };
24
+ const writeBaseline = (snapshotPath, snapshot) => {
25
+ writeFileSync(snapshotPath, serializeSchemaSnapshot(snapshot), "utf8");
26
+ };
27
+ const corruptBaselineResult = (context) => {
28
+ const reason = `schema-drift gate: the committed baseline ${context.snapshotPath} is unreadable or malformed, so schema drift cannot be checked. Fix it (e.g. resolve a merge conflict in lunora/.lunora-schema.json), or pass --update-schema-baseline to regenerate it from the current schema.`;
29
+ if (context.updateBaseline && !context.readOnly) {
30
+ context.logger.warn(
31
+ `schema baseline was unreadable; regenerating from the current schema on success (--update-schema-baseline): ${context.snapshotPath}`
32
+ );
33
+ return { blocked: false, changes: [], reason, rebless: context.rebless };
34
+ }
35
+ if (!context.readOnly) {
36
+ context.logger.error(reason);
37
+ }
38
+ return { blocked: true, changes: [], reason };
39
+ };
40
+ const blockedDecisionResult = (decision, context) => {
41
+ if (!context.readOnly) {
42
+ context.logger.error(decision.reason);
43
+ }
44
+ if (context.updateBaseline && !context.readOnly) {
45
+ context.logger.warn(`schema baseline will be re-blessed despite breaking drift on success (--update-schema-baseline): ${context.snapshotPath}`);
46
+ return { blocked: false, changes: decision.changes, reason: decision.reason, rebless: context.rebless };
47
+ }
48
+ return { blocked: true, changes: decision.changes, reason: decision.reason };
49
+ };
50
+ const runSchemaDriftGate = (options) => {
51
+ const { allowDrift, codegen, logger, readOnly = false, updateBaseline = false } = options;
52
+ const snapshotPath = codegen.schemaSnapshotPath;
53
+ const baseline = readBaseline(snapshotPath);
54
+ const context = {
55
+ logger,
56
+ readOnly,
57
+ rebless: () => {
58
+ writeBaseline(snapshotPath, codegen.schemaSnapshot);
59
+ },
60
+ snapshotPath,
61
+ updateBaseline
62
+ };
63
+ if (baseline.status === "corrupt") {
64
+ return corruptBaselineResult(context);
65
+ }
66
+ const decision = evaluateSchemaDrift({ allowDrift, baseline: baseline.status === "ok" ? baseline.snapshot : void 0, current: codegen.schemaSnapshot });
67
+ if (decision.blocked) {
68
+ return blockedDecisionResult(decision, context);
69
+ }
70
+ if (decision.changes.length > 0) {
71
+ if (!readOnly) {
72
+ logger.info(decision.reason);
73
+ }
74
+ return { blocked: false, changes: decision.changes, reason: decision.reason, rebless: readOnly ? void 0 : context.rebless };
75
+ }
76
+ return { blocked: false, changes: decision.changes, reason: decision.reason };
77
+ };
78
+
79
+ export { runSchemaDriftGate as r };
@@ -0,0 +1,43 @@
1
+ import { validatorKindToSqlType } from './diffSnapshots-BeDvvNiF.mjs';
2
+
3
+ const validatorToColumn = (validator) => {
4
+ if (validator.kind === "optional" && validator.inner) {
5
+ return {
6
+ nullable: true,
7
+ sqlType: validatorKindToSqlType(validator.inner.kind)
8
+ };
9
+ }
10
+ return {
11
+ nullable: false,
12
+ sqlType: validatorKindToSqlType(validator.kind)
13
+ };
14
+ };
15
+ const isGlobal = (mode) => mode === "global";
16
+ const schemaIrToSnapshot = (ir) => {
17
+ const tables = {};
18
+ for (const table of ir.tables) {
19
+ if (!isGlobal(table.shardMode)) {
20
+ continue;
21
+ }
22
+ const columns = {};
23
+ for (const [columnName, validator] of Object.entries(table.shape)) {
24
+ columns[columnName] = validatorToColumn(validator);
25
+ }
26
+ const indexes = {};
27
+ for (const index of table.indexes) {
28
+ indexes[index.name] = {
29
+ fields: [...index.fields],
30
+ name: index.name,
31
+ unique: index.unique ?? false
32
+ };
33
+ }
34
+ tables[table.name] = {
35
+ columns,
36
+ indexes,
37
+ name: table.name
38
+ };
39
+ }
40
+ return { tables, version: 1 };
41
+ };
42
+
43
+ export { schemaIrToSnapshot as default };
@@ -0,0 +1,84 @@
1
+ import { s as setBindingField } from './commands-DPKWlqqX.mjs';
2
+
3
+ const INVALID_SLUG_CHARS = /[^a-z0-9]+/u;
4
+ const toKebabSlug = (input, min, max) => {
5
+ let slug = input.toLowerCase().split(INVALID_SLUG_CHARS).filter(Boolean).join("-").slice(0, max);
6
+ if (slug.endsWith("-")) {
7
+ slug = slug.slice(0, -1);
8
+ }
9
+ return slug.length >= min ? slug : void 0;
10
+ };
11
+
12
+ const DB_BINDING = "DB";
13
+ const AUTH_DB_PROMPT = "Name your D1 database (run `wrangler d1 create` to get its id, then put it in wrangler.jsonc)";
14
+ const sanitizeDatabaseName = (input) => toKebabSlug(input, 1, 64);
15
+ const FALLBACK_DATABASE_NAME = "lunora-db";
16
+ const deriveDatabaseName = (projectName) => sanitizeDatabaseName(`${projectName}-db`) ?? FALLBACK_DATABASE_NAME;
17
+ const promptDatabaseName = async (text, projectName) => {
18
+ const fallback = deriveDatabaseName(projectName);
19
+ return sanitizeDatabaseName(await text(AUTH_DB_PROMPT, { default: fallback, placeholder: fallback })) ?? fallback;
20
+ };
21
+ const withAuthDatabaseName = (manifest, name) => setBindingField(manifest, "d1_databases", { key: "binding", value: DB_BINDING }, "database_name", name);
22
+
23
+ const AUTH_PROVIDER_OPTIONS = [
24
+ { description: "Email + password on better-auth (default)", label: "Email & password", value: "auth" },
25
+ { description: "Clerk-hosted auth", label: "Clerk", value: "auth-clerk" },
26
+ { description: "Auth0 (OIDC)", label: "Auth0", value: "auth-auth0" }
27
+ ];
28
+ const DEFAULT_AUTH_ITEM = "auth";
29
+ const promptAuthProvider = async (select) => await select("Which auth provider?", AUTH_PROVIDER_OPTIONS, { default: DEFAULT_AUTH_ITEM }) ?? DEFAULT_AUTH_ITEM;
30
+ const EMAIL_ITEM = "mail";
31
+ const normalizeFeature = (raw) => {
32
+ const value = raw.trim();
33
+ if (value === "") {
34
+ return void 0;
35
+ }
36
+ const lower = value.toLowerCase();
37
+ if (lower === "auth") {
38
+ return { kind: "auth" };
39
+ }
40
+ if (lower === "email" || lower === "mail") {
41
+ return { kind: "email" };
42
+ }
43
+ return { item: lower, kind: "item" };
44
+ };
45
+
46
+ const SEND_EMAIL_BINDING = "SEND_EMAIL";
47
+ const MAIL_DESTINATION_PROMPT = "Verified destination email for production delivery (blank = set it later in wrangler.jsonc)";
48
+ const isValidEmail = (value) => {
49
+ const trimmed = value.trim();
50
+ if (trimmed.length === 0 || trimmed.includes(" ")) {
51
+ return false;
52
+ }
53
+ const at = trimmed.indexOf("@");
54
+ if (at <= 0 || at !== trimmed.lastIndexOf("@")) {
55
+ return false;
56
+ }
57
+ const domain = trimmed.slice(at + 1);
58
+ return domain.length >= 3 && domain.includes(".") && !domain.startsWith(".") && !domain.endsWith(".");
59
+ };
60
+ const resolveTypedDestination = (entered, warn) => {
61
+ const trimmed = entered.trim();
62
+ if (trimmed === "") {
63
+ return void 0;
64
+ }
65
+ if (!isValidEmail(trimmed)) {
66
+ warn(`"${trimmed}" doesn't look like an email — leaving the placeholder; set destination_address in wrangler.jsonc.`);
67
+ return void 0;
68
+ }
69
+ return trimmed;
70
+ };
71
+ const withMailDestination = (manifest, address) => setBindingField(manifest, "send_email", { key: "name", value: SEND_EMAIL_BINDING }, "destination_address", address);
72
+
73
+ const UPLOADS_BINDING = "UPLOADS";
74
+ const STORAGE_BUCKET_PROMPT = "Name your R2 bucket (you can rename it in wrangler.jsonc later)";
75
+ const sanitizeBucketName = (input) => toKebabSlug(input, 3, 63);
76
+ const FALLBACK_BUCKET_NAME = "lunora-uploads";
77
+ const deriveBucketName = (projectName) => sanitizeBucketName(`${projectName}-uploads`) ?? FALLBACK_BUCKET_NAME;
78
+ const promptBucketName = async (text, projectName) => {
79
+ const fallback = deriveBucketName(projectName);
80
+ return sanitizeBucketName(await text(STORAGE_BUCKET_PROMPT, { default: fallback, placeholder: fallback })) ?? fallback;
81
+ };
82
+ const withStorageBucketName = (manifest, bucketName) => setBindingField(manifest, "r2_buckets", { key: "binding", value: UPLOADS_BINDING }, "bucket_name", bucketName);
83
+
84
+ export { AUTH_PROVIDER_OPTIONS as A, DEFAULT_AUTH_ITEM as D, EMAIL_ITEM as E, MAIL_DESTINATION_PROMPT as M, sanitizeDatabaseName as a, deriveDatabaseName as b, promptDatabaseName as c, deriveBucketName as d, promptAuthProvider as e, withMailDestination as f, withAuthDatabaseName as g, normalizeFeature as n, promptBucketName as p, resolveTypedDestination as r, sanitizeBucketName as s, withStorageBucketName as w };