@monkilab/backbone 0.1.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.
Files changed (181) hide show
  1. package/README.md +86 -0
  2. package/bin/cli.mjs +2 -0
  3. package/dist/__tests__/create.test.d.ts +2 -0
  4. package/dist/__tests__/create.test.d.ts.map +1 -0
  5. package/dist/__tests__/create.test.js +109 -0
  6. package/dist/__tests__/create.test.js.map +1 -0
  7. package/dist/__tests__/generators/convex.test.d.ts +2 -0
  8. package/dist/__tests__/generators/convex.test.d.ts.map +1 -0
  9. package/dist/__tests__/generators/convex.test.js +101 -0
  10. package/dist/__tests__/generators/convex.test.js.map +1 -0
  11. package/dist/__tests__/generators/github-actions.test.d.ts +2 -0
  12. package/dist/__tests__/generators/github-actions.test.d.ts.map +1 -0
  13. package/dist/__tests__/generators/github-actions.test.js +92 -0
  14. package/dist/__tests__/generators/github-actions.test.js.map +1 -0
  15. package/dist/__tests__/generators/ionic.test.d.ts +2 -0
  16. package/dist/__tests__/generators/ionic.test.d.ts.map +1 -0
  17. package/dist/__tests__/generators/ionic.test.js +94 -0
  18. package/dist/__tests__/generators/ionic.test.js.map +1 -0
  19. package/dist/__tests__/generators/llm-lib.test.d.ts +2 -0
  20. package/dist/__tests__/generators/llm-lib.test.d.ts.map +1 -0
  21. package/dist/__tests__/generators/llm-lib.test.js +67 -0
  22. package/dist/__tests__/generators/llm-lib.test.js.map +1 -0
  23. package/dist/__tests__/generators/nextjs.test.d.ts +2 -0
  24. package/dist/__tests__/generators/nextjs.test.d.ts.map +1 -0
  25. package/dist/__tests__/generators/nextjs.test.js +88 -0
  26. package/dist/__tests__/generators/nextjs.test.js.map +1 -0
  27. package/dist/__tests__/generators/nx.test.d.ts +2 -0
  28. package/dist/__tests__/generators/nx.test.d.ts.map +1 -0
  29. package/dist/__tests__/generators/nx.test.js +87 -0
  30. package/dist/__tests__/generators/nx.test.js.map +1 -0
  31. package/dist/__tests__/generators/supabase.test.d.ts +2 -0
  32. package/dist/__tests__/generators/supabase.test.d.ts.map +1 -0
  33. package/dist/__tests__/generators/supabase.test.js +98 -0
  34. package/dist/__tests__/generators/supabase.test.js.map +1 -0
  35. package/dist/__tests__/generators/ui-lib.test.d.ts +2 -0
  36. package/dist/__tests__/generators/ui-lib.test.d.ts.map +1 -0
  37. package/dist/__tests__/generators/ui-lib.test.js +67 -0
  38. package/dist/__tests__/generators/ui-lib.test.js.map +1 -0
  39. package/dist/__tests__/index.test.d.ts +2 -0
  40. package/dist/__tests__/index.test.d.ts.map +1 -0
  41. package/dist/__tests__/index.test.js +117 -0
  42. package/dist/__tests__/index.test.js.map +1 -0
  43. package/dist/__tests__/prompts.test.d.ts +2 -0
  44. package/dist/__tests__/prompts.test.d.ts.map +1 -0
  45. package/dist/__tests__/prompts.test.js +78 -0
  46. package/dist/__tests__/prompts.test.js.map +1 -0
  47. package/dist/cli/create.d.ts +3 -0
  48. package/dist/cli/create.d.ts.map +1 -0
  49. package/dist/cli/create.js +58 -0
  50. package/dist/cli/create.js.map +1 -0
  51. package/dist/cli/generators/convex.d.ts +8 -0
  52. package/dist/cli/generators/convex.d.ts.map +1 -0
  53. package/dist/cli/generators/convex.js +51 -0
  54. package/dist/cli/generators/convex.js.map +1 -0
  55. package/dist/cli/generators/github-actions.d.ts +6 -0
  56. package/dist/cli/generators/github-actions.d.ts.map +1 -0
  57. package/dist/cli/generators/github-actions.js +16 -0
  58. package/dist/cli/generators/github-actions.js.map +1 -0
  59. package/dist/cli/generators/ionic.d.ts +13 -0
  60. package/dist/cli/generators/ionic.d.ts.map +1 -0
  61. package/dist/cli/generators/ionic.js +57 -0
  62. package/dist/cli/generators/ionic.js.map +1 -0
  63. package/dist/cli/generators/llm-lib.d.ts +10 -0
  64. package/dist/cli/generators/llm-lib.d.ts.map +1 -0
  65. package/dist/cli/generators/llm-lib.js +38 -0
  66. package/dist/cli/generators/llm-lib.js.map +1 -0
  67. package/dist/cli/generators/nextjs.d.ts +16 -0
  68. package/dist/cli/generators/nextjs.d.ts.map +1 -0
  69. package/dist/cli/generators/nextjs.js +59 -0
  70. package/dist/cli/generators/nextjs.js.map +1 -0
  71. package/dist/cli/generators/nx.d.ts +7 -0
  72. package/dist/cli/generators/nx.d.ts.map +1 -0
  73. package/dist/cli/generators/nx.js +59 -0
  74. package/dist/cli/generators/nx.js.map +1 -0
  75. package/dist/cli/generators/supabase.d.ts +8 -0
  76. package/dist/cli/generators/supabase.d.ts.map +1 -0
  77. package/dist/cli/generators/supabase.js +54 -0
  78. package/dist/cli/generators/supabase.js.map +1 -0
  79. package/dist/cli/generators/ui-lib.d.ts +10 -0
  80. package/dist/cli/generators/ui-lib.d.ts.map +1 -0
  81. package/dist/cli/generators/ui-lib.js +38 -0
  82. package/dist/cli/generators/ui-lib.js.map +1 -0
  83. package/dist/cli/index.d.ts +2 -0
  84. package/dist/cli/index.d.ts.map +1 -0
  85. package/dist/cli/index.js +22 -0
  86. package/dist/cli/index.js.map +1 -0
  87. package/dist/cli/prompts.d.ts +6 -0
  88. package/dist/cli/prompts.d.ts.map +1 -0
  89. package/dist/cli/prompts.js +18 -0
  90. package/dist/cli/prompts.js.map +1 -0
  91. package/dist/cli/templates/packages/llm/src/index.d.ts +2 -0
  92. package/dist/cli/templates/packages/llm/src/index.d.ts.map +1 -0
  93. package/dist/cli/templates/packages/llm/src/index.js +2 -0
  94. package/dist/cli/templates/packages/llm/src/index.js.map +1 -0
  95. package/dist/cli/templates/packages/ui/src/index.d.ts +2 -0
  96. package/dist/cli/templates/packages/ui/src/index.d.ts.map +1 -0
  97. package/dist/cli/templates/packages/ui/src/index.js +2 -0
  98. package/dist/cli/templates/packages/ui/src/index.js.map +1 -0
  99. package/dist/cli/types.d.ts +11 -0
  100. package/dist/cli/types.d.ts.map +1 -0
  101. package/dist/cli/types.js +2 -0
  102. package/dist/cli/types.js.map +1 -0
  103. package/dist/index.d.ts +2 -0
  104. package/dist/index.d.ts.map +1 -0
  105. package/dist/index.js +46 -0
  106. package/dist/index.js.map +1 -0
  107. package/package.json +30 -0
  108. package/src/cli/templates/.github/workflows/deploy.yml +26 -0
  109. package/src/cli/templates/apps/mobile/capacitor.config.ts.tpl +12 -0
  110. package/src/cli/templates/apps/mobile/index.html +12 -0
  111. package/src/cli/templates/apps/mobile/package.json +30 -0
  112. package/src/cli/templates/apps/mobile/project.json +18 -0
  113. package/src/cli/templates/apps/mobile/src/App.tsx.tpl +21 -0
  114. package/src/cli/templates/apps/mobile/src/main.tsx.tpl +11 -0
  115. package/src/cli/templates/apps/mobile/src/pages/Home.tsx.tpl +22 -0
  116. package/src/cli/templates/apps/mobile/src/theme/tailwind.css +1 -0
  117. package/src/cli/templates/apps/mobile/tsconfig.json.tpl +17 -0
  118. package/src/cli/templates/apps/mobile/vite.config.ts.tpl +9 -0
  119. package/src/cli/templates/apps/web/app/globals.css +1 -0
  120. package/src/cli/templates/apps/web/app/layout.tsx.tpl +19 -0
  121. package/src/cli/templates/apps/web/app/page.tsx.tpl +3 -0
  122. package/src/cli/templates/apps/web/jest.config.ts.tpl +11 -0
  123. package/src/cli/templates/apps/web/next.config.ts.tpl +5 -0
  124. package/src/cli/templates/apps/web/package.json +29 -0
  125. package/src/cli/templates/apps/web/playwright.config.ts.tpl +13 -0
  126. package/src/cli/templates/apps/web/postcss.config.js +7 -0
  127. package/src/cli/templates/apps/web/project.json +39 -0
  128. package/src/cli/templates/apps/web/tsconfig.json +23 -0
  129. package/src/cli/templates/nx/README.md +19 -0
  130. package/src/cli/templates/nx/nx.json +25 -0
  131. package/src/cli/templates/nx/package.json +22 -0
  132. package/src/cli/templates/nx/tsconfig.base.json +21 -0
  133. package/src/cli/templates/packages/llm/package.json +17 -0
  134. package/src/cli/templates/packages/llm/src/index.ts +2 -0
  135. package/src/cli/templates/packages/llm/tsconfig.json +8 -0
  136. package/src/cli/templates/packages/ui/package.json +19 -0
  137. package/src/cli/templates/packages/ui/src/index.ts +2 -0
  138. package/src/cli/templates/packages/ui/tsconfig.json +8 -0
  139. package/templates/.github/workflows/deploy.yml +20 -0
  140. package/templates/apps/mobile/capacitor.config.ts +7 -0
  141. package/templates/apps/mobile/ionic.config.json +5 -0
  142. package/templates/apps/mobile/jest.config.ts +7 -0
  143. package/templates/apps/mobile/project.json +13 -0
  144. package/templates/apps/mobile/src/app/globals.css +1 -0
  145. package/templates/apps/mobile/src/app/layout.tsx +7 -0
  146. package/templates/apps/mobile/src/app/page.tsx +8 -0
  147. package/templates/apps/mobile/tailwind.config.js +6 -0
  148. package/templates/apps/mobile/tsconfig.json +6 -0
  149. package/templates/apps/web/jest.config.ts +7 -0
  150. package/templates/apps/web/next.config.ts +3 -0
  151. package/templates/apps/web/postcss.config.js +1 -0
  152. package/templates/apps/web/project.json +12 -0
  153. package/templates/apps/web/src/app/globals.css +1 -0
  154. package/templates/apps/web/src/app/layout.tsx +10 -0
  155. package/templates/apps/web/src/app/page.tsx +3 -0
  156. package/templates/apps/web/tailwind.config.js +6 -0
  157. package/templates/apps/web/tsconfig.json +6 -0
  158. package/templates/backend/convex/apps/web/src/lib/backend.ts +7 -0
  159. package/templates/backend/convex/convex/.gitkeep +0 -0
  160. package/templates/backend/convex/convex.config.ts +2 -0
  161. package/templates/backend/supabase/apps/web/src/lib/backend.ts +8 -0
  162. package/templates/backend/supabase/supabase/config.toml +16 -0
  163. package/templates/backend/supabase/supabase/migrations/.gitkeep +0 -0
  164. package/templates/docs/README.md +3 -0
  165. package/templates/nx/.env.example +9 -0
  166. package/templates/nx/.prettierrc +1 -0
  167. package/templates/nx/jest.preset.js +2 -0
  168. package/templates/nx/nx.json +25 -0
  169. package/templates/nx/package.json +33 -0
  170. package/templates/nx/playwright.config.ts +19 -0
  171. package/templates/nx/tsconfig.base.json +20 -0
  172. package/templates/packages/llm/jest.config.ts +6 -0
  173. package/templates/packages/llm/project.json +11 -0
  174. package/templates/packages/llm/src/email/resend.ts +4 -0
  175. package/templates/packages/llm/src/index.ts +2 -0
  176. package/templates/packages/llm/tsconfig.json +6 -0
  177. package/templates/packages/ui/jest.config.ts +6 -0
  178. package/templates/packages/ui/project.json +11 -0
  179. package/templates/packages/ui/src/index.ts +2 -0
  180. package/templates/packages/ui/tsconfig.json +6 -0
  181. package/templates/templates/README.md +3 -0
@@ -0,0 +1,54 @@
1
+ import { mkdir, readdir, readFile, stat, writeFile } from 'node:fs/promises';
2
+ import { dirname, join } from 'node:path';
3
+ import { fileURLToPath } from 'node:url';
4
+ const TEMPLATE_DIR = join(dirname(fileURLToPath(import.meta.url)), '..', 'templates', 'backend', 'supabase');
5
+ const SUPABASE_SHIM = `// Auto-generated — do not edit
6
+ export { createClient } from '@supabase/supabase-js';
7
+ `;
8
+ function interpolate(content, projectName) {
9
+ return content.replaceAll('{{projectName}}', projectName);
10
+ }
11
+ async function copyInterpolated(src, dest, projectName) {
12
+ await mkdir(dest, { recursive: true });
13
+ const entries = await readdir(src, { withFileTypes: true });
14
+ for (const entry of entries) {
15
+ const srcPath = join(src, entry.name);
16
+ const destPath = join(dest, entry.name);
17
+ if (entry.isDirectory()) {
18
+ await copyInterpolated(srcPath, destPath, projectName);
19
+ }
20
+ else {
21
+ const resolvedDest = destPath.endsWith('.tpl')
22
+ ? destPath.slice(0, -4)
23
+ : destPath;
24
+ const raw = await readFile(srcPath, 'utf8');
25
+ await writeFile(resolvedDest, interpolate(raw, projectName), 'utf8');
26
+ }
27
+ }
28
+ }
29
+ /**
30
+ * Generates a Supabase backend at `outDir/supabase/` and injects the Supabase
31
+ * client shim into `outDir/apps/web/src/lib/backend.ts`.
32
+ *
33
+ * Also adds `@supabase/supabase-js` to `outDir/apps/web/package.json` dependencies.
34
+ */
35
+ export async function generateSupabase(projectName, outDir) {
36
+ const templateStat = await stat(TEMPLATE_DIR).catch(() => null);
37
+ if (!templateStat?.isDirectory()) {
38
+ throw new Error(`Supabase backend template directory not found: ${TEMPLATE_DIR}`);
39
+ }
40
+ const destDir = join(outDir, 'supabase');
41
+ await copyInterpolated(TEMPLATE_DIR, destDir, projectName);
42
+ const libDir = join(outDir, 'apps', 'web', 'src', 'lib');
43
+ await mkdir(libDir, { recursive: true });
44
+ await writeFile(join(libDir, 'backend.ts'), SUPABASE_SHIM, 'utf8');
45
+ const pkgPath = join(outDir, 'apps', 'web', 'package.json');
46
+ const pkgRaw = await readFile(pkgPath, 'utf8');
47
+ const pkg = JSON.parse(pkgRaw);
48
+ pkg.dependencies = {
49
+ ...pkg.dependencies,
50
+ '@supabase/supabase-js': 'latest',
51
+ };
52
+ await writeFile(pkgPath, JSON.stringify(pkg, null, 2) + '\n', 'utf8');
53
+ }
54
+ //# sourceMappingURL=supabase.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"supabase.js","sourceRoot":"","sources":["../../../src/cli/generators/supabase.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ /**
2
+ * Generate the UI library package under `outDir/packages/ui/`.
3
+ *
4
+ * Copies templates from `src/cli/templates/packages/ui/`, replacing
5
+ * `{{projectName}}` and `{{scope}}` tokens in all file contents.
6
+ * The output `package.json` will declare `tailwindcss@^4.0.0` and use the
7
+ * scoped name `@monkilab/<projectName>-ui`.
8
+ */
9
+ export declare function generateUiLib(projectName: string, outDir: string): Promise<void>;
10
+ //# sourceMappingURL=ui-lib.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ui-lib.d.ts","sourceRoot":"","sources":["../../../src/cli/generators/ui-lib.ts"],"names":[],"mappings":"AA0CA;;;;;;;GAOG;AACH,wBAAsB,aAAa,CACjC,WAAW,EAAE,MAAM,EACnB,MAAM,EAAE,MAAM,GACb,OAAO,CAAC,IAAI,CAAC,CAGf"}
@@ -0,0 +1,38 @@
1
+ import { mkdir, readdir, readFile, writeFile } from 'node:fs/promises';
2
+ import { join, dirname } from 'node:path';
3
+ import { fileURLToPath } from 'node:url';
4
+ const TEMPLATE_DIR = join(dirname(fileURLToPath(import.meta.url)), '..', 'templates', 'packages', 'ui');
5
+ const SCOPE = '@monkilab';
6
+ function interpolate(content, projectName) {
7
+ return content
8
+ .replaceAll('{{projectName}}', projectName)
9
+ .replaceAll('{{scope}}', SCOPE);
10
+ }
11
+ async function copyInterpolated(src, dest, projectName) {
12
+ await mkdir(dest, { recursive: true });
13
+ const entries = await readdir(src, { withFileTypes: true });
14
+ for (const entry of entries) {
15
+ const srcPath = join(src, entry.name);
16
+ const destPath = join(dest, entry.name);
17
+ if (entry.isDirectory()) {
18
+ await copyInterpolated(srcPath, destPath, projectName);
19
+ }
20
+ else {
21
+ const raw = await readFile(srcPath, 'utf8');
22
+ await writeFile(destPath, interpolate(raw, projectName), 'utf8');
23
+ }
24
+ }
25
+ }
26
+ /**
27
+ * Generate the UI library package under `outDir/packages/ui/`.
28
+ *
29
+ * Copies templates from `src/cli/templates/packages/ui/`, replacing
30
+ * `{{projectName}}` and `{{scope}}` tokens in all file contents.
31
+ * The output `package.json` will declare `tailwindcss@^4.0.0` and use the
32
+ * scoped name `@monkilab/<projectName>-ui`.
33
+ */
34
+ export async function generateUiLib(projectName, outDir) {
35
+ const dest = join(outDir, 'packages', 'ui');
36
+ await copyInterpolated(TEMPLATE_DIR, dest, projectName);
37
+ }
38
+ //# sourceMappingURL=ui-lib.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ui-lib.js","sourceRoot":"","sources":["../../../src/cli/generators/ui-lib.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAC;AACvE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AAEzC,MAAM,YAAY,GAAG,IAAI,CACvB,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EACvC,IAAI,EACJ,WAAW,EACX,UAAU,EACV,IAAI,CACL,CAAC;AAEF,MAAM,KAAK,GAAG,WAAW,CAAC;AAE1B,SAAS,WAAW,CAAC,OAAe,EAAE,WAAmB;IACvD,OAAO,OAAO;SACX,UAAU,CAAC,iBAAiB,EAAE,WAAW,CAAC;SAC1C,UAAU,CAAC,WAAW,EAAE,KAAK,CAAC,CAAC;AACpC,CAAC;AAED,KAAK,UAAU,gBAAgB,CAC7B,GAAW,EACX,IAAY,EACZ,WAAmB;IAEnB,MAAM,KAAK,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAEvC,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,GAAG,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;IAE5D,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;QAC5B,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;QACtC,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;QAExC,IAAI,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC;YACxB,MAAM,gBAAgB,CAAC,OAAO,EAAE,QAAQ,EAAE,WAAW,CAAC,CAAC;QACzD,CAAC;aAAM,CAAC;YACN,MAAM,GAAG,GAAG,MAAM,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;YAC5C,MAAM,SAAS,CAAC,QAAQ,EAAE,WAAW,CAAC,GAAG,EAAE,WAAW,CAAC,EAAE,MAAM,CAAC,CAAC;QACnE,CAAC;IACH,CAAC;AACH,CAAC;AAED;;;;;;;GAOG;AACH,MAAM,CAAC,KAAK,UAAU,aAAa,CACjC,WAAmB,EACnB,MAAc;IAEd,MAAM,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,UAAU,EAAE,IAAI,CAAC,CAAC;IAC5C,MAAM,gBAAgB,CAAC,YAAY,EAAE,IAAI,EAAE,WAAW,CAAC,CAAC;AAC1D,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function main(): Promise<void>;
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":"AAGA,wBAAsB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC,CAoB1C"}
@@ -0,0 +1,22 @@
1
+ import { collectOptions } from './prompts.js';
2
+ import { create } from './create.js';
3
+ export async function main() {
4
+ const projectName = process.argv[2];
5
+ if (!projectName) {
6
+ console.error('Error: project name is required.');
7
+ process.exit(1);
8
+ }
9
+ if (!/^[a-z][a-z0-9-]*$/.test(projectName)) {
10
+ console.error(`Error: invalid project name "${projectName}". Must match /^[a-z][a-z0-9-]*$/`);
11
+ process.exit(1);
12
+ }
13
+ try {
14
+ const options = await collectOptions(projectName);
15
+ await create(options);
16
+ }
17
+ catch (err) {
18
+ console.error(err);
19
+ process.exit(1);
20
+ }
21
+ }
22
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,cAAc,CAAC;AAC9C,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAErC,MAAM,CAAC,KAAK,UAAU,IAAI;IACxB,MAAM,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAEpC,IAAI,CAAC,WAAW,EAAE,CAAC;QACjB,OAAO,CAAC,KAAK,CAAC,kCAAkC,CAAC,CAAC;QAClD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC;QAC3C,OAAO,CAAC,KAAK,CAAC,gCAAgC,WAAW,mCAAmC,CAAC,CAAC;QAC9F,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,MAAM,cAAc,CAAC,WAAW,CAAC,CAAC;QAClD,MAAM,MAAM,CAAC,OAAO,CAAC,CAAC;IACxB,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACnB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC"}
@@ -0,0 +1,6 @@
1
+ export interface CliOptions {
2
+ projectName: string;
3
+ backend: 'convex' | 'supabase';
4
+ }
5
+ export declare function collectOptions(projectName: string): Promise<CliOptions>;
6
+ //# sourceMappingURL=prompts.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prompts.d.ts","sourceRoot":"","sources":["../../src/cli/prompts.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,UAAU;IACzB,WAAW,EAAE,MAAM,CAAC;IACpB,OAAO,EAAE,QAAQ,GAAG,UAAU,CAAC;CAChC;AAED,wBAAsB,cAAc,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,UAAU,CAAC,CAmB7E"}
@@ -0,0 +1,18 @@
1
+ import * as clack from '@clack/prompts';
2
+ export async function collectOptions(projectName) {
3
+ clack.intro('🏗 BACKBONE — Nx monorepo scaffolding');
4
+ const backend = await clack.select({
5
+ message: 'Select a backend:',
6
+ options: [
7
+ { value: 'convex', label: 'Convex', hint: 'Real-time serverless backend' },
8
+ { value: 'supabase', label: 'Supabase', hint: 'Postgres + Auth + Storage' },
9
+ ],
10
+ });
11
+ if (clack.isCancel(backend)) {
12
+ clack.cancel('Setup cancelled.');
13
+ process.exit(0);
14
+ }
15
+ clack.outro('Options collected — scaffolding your project…');
16
+ return { projectName, backend };
17
+ }
18
+ //# sourceMappingURL=prompts.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prompts.js","sourceRoot":"","sources":["../../src/cli/prompts.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,KAAK,MAAM,gBAAgB,CAAC;AAOxC,MAAM,CAAC,KAAK,UAAU,cAAc,CAAC,WAAmB;IACtD,KAAK,CAAC,KAAK,CAAC,wCAAwC,CAAC,CAAC;IAEtD,MAAM,OAAO,GAAG,MAAM,KAAK,CAAC,MAAM,CAAwB;QACxD,OAAO,EAAE,mBAAmB;QAC5B,OAAO,EAAE;YACP,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,8BAA8B,EAAE;YAC1E,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,2BAA2B,EAAE;SAC5E;KACF,CAAC,CAAC;IAEH,IAAI,KAAK,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;QAC5B,KAAK,CAAC,MAAM,CAAC,kBAAkB,CAAC,CAAC;QACjC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,KAAK,CAAC,KAAK,CAAC,+CAA+C,CAAC,CAAC;IAE7D,OAAO,EAAE,WAAW,EAAE,OAAO,EAAE,CAAC;AAClC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../../../src/cli/templates/packages/llm/src/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../../src/cli/templates/packages/llm/src/index.ts"],"names":[],"mappings":""}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../../../src/cli/templates/packages/ui/src/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../../src/cli/templates/packages/ui/src/index.ts"],"names":[],"mappings":""}
@@ -0,0 +1,11 @@
1
+ export type BackendChoice = 'convex' | 'supabase';
2
+ export interface ScaffoldOptions {
3
+ projectName: string;
4
+ backend: BackendChoice;
5
+ outputDir: string;
6
+ }
7
+ export interface TemplateFile {
8
+ src: string;
9
+ dest: string;
10
+ }
11
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/cli/types.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,aAAa,GAAG,QAAQ,GAAG,UAAU,CAAC;AAElD,MAAM,WAAW,eAAe;IAC9B,WAAW,EAAE,MAAM,CAAC;IACpB,OAAO,EAAE,aAAa,CAAC;IACvB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,YAAY;IAC3B,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;CACd"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/cli/types.ts"],"names":[],"mappings":""}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":""}
package/dist/index.js ADDED
@@ -0,0 +1,46 @@
1
+ import * as p from '@clack/prompts';
2
+ async function main() {
3
+ p.intro('backbone — Nx monorepo scaffolder');
4
+ const projectName = await p.text({
5
+ message: 'Project name',
6
+ placeholder: 'my-project',
7
+ validate: (v) => (v.trim().length === 0 ? 'Project name is required' : undefined),
8
+ });
9
+ if (p.isCancel(projectName)) {
10
+ p.cancel('Cancelled.');
11
+ process.exit(0);
12
+ }
13
+ const backend = await p.select({
14
+ message: 'Backend',
15
+ options: [
16
+ { value: 'convex', label: 'Convex' },
17
+ { value: 'supabase', label: 'Supabase' },
18
+ ],
19
+ });
20
+ if (p.isCancel(backend)) {
21
+ p.cancel('Cancelled.');
22
+ process.exit(0);
23
+ }
24
+ const outputDir = await p.text({
25
+ message: 'Output directory',
26
+ placeholder: `./${projectName}`,
27
+ defaultValue: `./${projectName}`,
28
+ });
29
+ if (p.isCancel(outputDir)) {
30
+ p.cancel('Cancelled.');
31
+ process.exit(0);
32
+ }
33
+ const options = {
34
+ projectName: projectName,
35
+ backend: backend,
36
+ outputDir: outputDir,
37
+ };
38
+ p.log.info(`Scaffolding ${options.projectName} with ${options.backend} backend into ${options.outputDir}`);
39
+ // TODO: invoke scaffold runner with options
40
+ p.outro('Done! Your Nx monorepo is ready.');
41
+ }
42
+ main().catch((err) => {
43
+ console.error(err);
44
+ process.exit(1);
45
+ });
46
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,CAAC,MAAM,gBAAgB,CAAC;AAGpC,KAAK,UAAU,IAAI;IACjB,CAAC,CAAC,KAAK,CAAC,mCAAmC,CAAC,CAAC;IAE7C,MAAM,WAAW,GAAG,MAAM,CAAC,CAAC,IAAI,CAAC;QAC/B,OAAO,EAAE,cAAc;QACvB,WAAW,EAAE,YAAY;QACzB,QAAQ,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,0BAA0B,CAAC,CAAC,CAAC,SAAS,CAAC;KAClF,CAAC,CAAC;IAEH,IAAI,CAAC,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC;QAC5B,CAAC,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;QACvB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,MAAM,OAAO,GAAG,MAAM,CAAC,CAAC,MAAM,CAAgB;QAC5C,OAAO,EAAE,SAAS;QAClB,OAAO,EAAE;YACP,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE;YACpC,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAE;SACzC;KACF,CAAC,CAAC;IAEH,IAAI,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;QACxB,CAAC,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;QACvB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,MAAM,SAAS,GAAG,MAAM,CAAC,CAAC,IAAI,CAAC;QAC7B,OAAO,EAAE,kBAAkB;QAC3B,WAAW,EAAE,KAAK,WAAW,EAAE;QAC/B,YAAY,EAAE,KAAK,WAAW,EAAE;KACjC,CAAC,CAAC;IAEH,IAAI,CAAC,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;QAC1B,CAAC,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;QACvB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,MAAM,OAAO,GAAoB;QAC/B,WAAW,EAAE,WAAqB;QAClC,OAAO,EAAE,OAAwB;QACjC,SAAS,EAAE,SAAmB;KAC/B,CAAC;IAEF,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,eAAe,OAAO,CAAC,WAAW,SAAS,OAAO,CAAC,OAAO,iBAAiB,OAAO,CAAC,SAAS,EAAE,CAAC,CAAC;IAE3G,4CAA4C;IAE5C,CAAC,CAAC,KAAK,CAAC,kCAAkC,CAAC,CAAC;AAC9C,CAAC;AAED,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;IACnB,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACnB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC"}
package/package.json ADDED
@@ -0,0 +1,30 @@
1
+ {
2
+ "name": "@monkilab/backbone",
3
+ "version": "0.1.0",
4
+ "description": "Scaffold a production-ready Nx monorepo for MonkiLabs projects",
5
+ "type": "module",
6
+ "bin": { "backbone": "bin/cli.mjs" },
7
+ "exports": { ".": "./dist/index.js" },
8
+ "scripts": {
9
+ "build": "tsc",
10
+ "dev": "tsc --watch",
11
+ "test": "vitest run",
12
+ "test:watch": "vitest",
13
+ "prepublishOnly": "npm run build"
14
+ },
15
+ "dependencies": { "@clack/prompts": "^0.9.0" },
16
+ "devDependencies": {
17
+ "@types/node": "^22.0.0",
18
+ "memfs": "^4.56.11",
19
+ "typescript": "^5.0.0",
20
+ "vitest": "^4.0.0"
21
+ },
22
+ "engines": { "node": ">=22.5.0" },
23
+ "files": [
24
+ "bin/",
25
+ "dist/",
26
+ "src/cli/templates/",
27
+ "templates/",
28
+ "README.md"
29
+ ]
30
+ }
@@ -0,0 +1,26 @@
1
+ name: Deploy
2
+
3
+ on:
4
+ push:
5
+ branches:
6
+ - main
7
+
8
+ jobs:
9
+ deploy:
10
+ name: Deploy {{projectName}} to Coolify
11
+ runs-on: ubuntu-latest
12
+
13
+ steps:
14
+ - name: Checkout
15
+ uses: actions/checkout@v4
16
+
17
+ - name: Setup Node.js
18
+ uses: actions/setup-node@v4
19
+ with:
20
+ node-version: 20
21
+
22
+ - name: Trigger Coolify deployment
23
+ run: |
24
+ curl --silent --fail --show-error \
25
+ -X GET \
26
+ "${{ secrets.COOLIFY_WEBHOOK_URL }}"
@@ -0,0 +1,12 @@
1
+ import type { CapacitorConfig } from '@capacitor/cli';
2
+
3
+ const config: CapacitorConfig = {
4
+ appId: 'com.monkilab.{{projectName}}',
5
+ appName: '{{projectName}}',
6
+ webDir: 'dist',
7
+ server: {
8
+ androidScheme: 'https',
9
+ },
10
+ };
11
+
12
+ export default config;
@@ -0,0 +1,12 @@
1
+ <!doctype html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="UTF-8" />
5
+ <meta name="viewport" content="width=device-width, initial-scale=1.0, viewport-fit=cover" />
6
+ <title>{{projectName}}</title>
7
+ </head>
8
+ <body>
9
+ <div id="root"></div>
10
+ <script type="module" src="/src/main.tsx"></script>
11
+ </body>
12
+ </html>
@@ -0,0 +1,30 @@
1
+ {
2
+ "name": "{{scope}}/{{projectName}}-mobile",
3
+ "version": "0.0.1",
4
+ "private": true,
5
+ "type": "module",
6
+ "scripts": {
7
+ "start": "vite",
8
+ "build": "tsc && vite build",
9
+ "preview": "vite preview",
10
+ "test": "vitest run"
11
+ },
12
+ "dependencies": {
13
+ "@capacitor/core": "^7.0.0",
14
+ "@ionic/react": "^8.0.0",
15
+ "@nxext/ionic-react": "^20.0.0",
16
+ "react": "^18.2.0",
17
+ "react-dom": "^18.2.0",
18
+ "react-router-dom": "^6.0.0",
19
+ "tailwindcss": "^4.0.0"
20
+ },
21
+ "devDependencies": {
22
+ "@capacitor/cli": "^7.0.0",
23
+ "@types/react": "^18.2.0",
24
+ "@types/react-dom": "^18.2.0",
25
+ "@vitejs/plugin-react": "^4.0.0",
26
+ "typescript": "~5.4.0",
27
+ "vite": "^5.0.0",
28
+ "vitest": "^1.0.0"
29
+ }
30
+ }
@@ -0,0 +1,18 @@
1
+ {
2
+ "name": "{{projectName}}-mobile",
3
+ "$schema": "../../node_modules/nx/schemas/project-schema.json",
4
+ "projectType": "application",
5
+ "sourceRoot": "apps/mobile/src",
6
+ "targets": {
7
+ "build": {
8
+ "executor": "@nxext/ionic-react:build"
9
+ },
10
+ "serve": {
11
+ "executor": "@nxext/ionic-react:serve"
12
+ },
13
+ "test": {
14
+ "executor": "@nx/vite:test"
15
+ }
16
+ },
17
+ "tags": ["scope:mobile", "type:app"]
18
+ }
@@ -0,0 +1,21 @@
1
+ import { IonApp, IonRouterOutlet, setupIonicReact } from '@ionic/react';
2
+ import { IonReactRouter } from '@ionic/react-router';
3
+ import { Route, Redirect } from 'react-router-dom';
4
+ import Home from './pages/Home';
5
+
6
+ setupIonicReact();
7
+
8
+ const App: React.FC = () => (
9
+ <IonApp>
10
+ <IonReactRouter>
11
+ <IonRouterOutlet>
12
+ <Route exact path="/home" component={Home} />
13
+ <Route exact path="/">
14
+ <Redirect to="/home" />
15
+ </Route>
16
+ </IonRouterOutlet>
17
+ </IonReactRouter>
18
+ </IonApp>
19
+ );
20
+
21
+ export default App;
@@ -0,0 +1,11 @@
1
+ import React from 'react';
2
+ import { createRoot } from 'react-dom/client';
3
+ import App from './App';
4
+ import './theme/tailwind.css';
5
+
6
+ const root = createRoot(document.getElementById('root') as HTMLElement);
7
+ root.render(
8
+ <React.StrictMode>
9
+ <App />
10
+ </React.StrictMode>,
11
+ );
@@ -0,0 +1,22 @@
1
+ import {
2
+ IonContent,
3
+ IonHeader,
4
+ IonPage,
5
+ IonTitle,
6
+ IonToolbar,
7
+ } from '@ionic/react';
8
+
9
+ const Home: React.FC = () => (
10
+ <IonPage>
11
+ <IonHeader>
12
+ <IonToolbar>
13
+ <IonTitle>{{projectName}}</IonTitle>
14
+ </IonToolbar>
15
+ </IonHeader>
16
+ <IonContent className="ion-padding">
17
+ <p className="text-base">Welcome to {{projectName}}</p>
18
+ </IonContent>
19
+ </IonPage>
20
+ );
21
+
22
+ export default Home;
@@ -0,0 +1 @@
1
+ @import 'tailwindcss';
@@ -0,0 +1,17 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ES2022",
4
+ "lib": ["ES2022", "DOM", "DOM.Iterable"],
5
+ "module": "ESNext",
6
+ "moduleResolution": "Bundler",
7
+ "jsx": "react-jsx",
8
+ "strict": true,
9
+ "esModuleInterop": true,
10
+ "skipLibCheck": true,
11
+ "forceConsistentCasingInFileNames": true,
12
+ "resolveJsonModule": true,
13
+ "isolatedModules": true,
14
+ "noEmit": true
15
+ },
16
+ "include": ["src"]
17
+ }
@@ -0,0 +1,9 @@
1
+ import { defineConfig } from 'vite';
2
+ import react from '@vitejs/plugin-react';
3
+
4
+ export default defineConfig({
5
+ plugins: [react()],
6
+ test: {
7
+ environment: 'jsdom',
8
+ },
9
+ });
@@ -0,0 +1 @@
1
+ @import "tailwindcss";
@@ -0,0 +1,19 @@
1
+ import type { Metadata } from 'next';
2
+ import './globals.css';
3
+
4
+ export const metadata: Metadata = {
5
+ title: '{{projectName}}',
6
+ description: '{{projectName}} web application',
7
+ };
8
+
9
+ export default function RootLayout({
10
+ children,
11
+ }: {
12
+ children: React.ReactNode;
13
+ }) {
14
+ return (
15
+ <html lang="en">
16
+ <body>{children}</body>
17
+ </html>
18
+ );
19
+ }
@@ -0,0 +1,3 @@
1
+ export default function Page() {
2
+ return <main>{{projectName}}</main>;
3
+ }
@@ -0,0 +1,11 @@
1
+ import type { Config } from 'jest';
2
+
3
+ const config: Config = {
4
+ preset: 'ts-jest',
5
+ testEnvironment: 'node',
6
+ moduleNameMapper: {
7
+ '^@/(.*)$': '<rootDir>/src/$1',
8
+ },
9
+ };
10
+
11
+ export default config;
@@ -0,0 +1,5 @@
1
+ import type { NextConfig } from 'next';
2
+
3
+ const nextConfig: NextConfig = {};
4
+
5
+ export default nextConfig;
@@ -0,0 +1,29 @@
1
+ {
2
+ "name": "{{scope}}/{{projectName}}-web",
3
+ "version": "0.0.1",
4
+ "private": true,
5
+ "scripts": {
6
+ "dev": "next dev",
7
+ "build": "next build",
8
+ "start": "next start",
9
+ "test": "jest",
10
+ "test:e2e": "playwright test"
11
+ },
12
+ "dependencies": {
13
+ "next": "^15.0.0",
14
+ "react": "^19.0.0",
15
+ "react-dom": "^19.0.0"
16
+ },
17
+ "devDependencies": {
18
+ "@playwright/test": "^1.40.0",
19
+ "@tailwindcss/postcss": "^4.0.0",
20
+ "@types/jest": "^29.0.0",
21
+ "@types/node": "^22.0.0",
22
+ "@types/react": "^19.0.0",
23
+ "@types/react-dom": "^19.0.0",
24
+ "jest": "^29.0.0",
25
+ "tailwindcss": "^4.0.0",
26
+ "ts-jest": "^29.0.0",
27
+ "typescript": "~5.4.0"
28
+ }
29
+ }
@@ -0,0 +1,13 @@
1
+ import { defineConfig } from '@playwright/test';
2
+
3
+ export default defineConfig({
4
+ testDir: './e2e',
5
+ use: {
6
+ baseURL: 'http://localhost:3000',
7
+ },
8
+ webServer: {
9
+ command: 'npm run dev',
10
+ url: 'http://localhost:3000',
11
+ reuseExistingServer: !process.env['CI'],
12
+ },
13
+ });
@@ -0,0 +1,7 @@
1
+ const config = {
2
+ plugins: {
3
+ '@tailwindcss/postcss': {},
4
+ },
5
+ };
6
+
7
+ export default config;