@quanticjs/create-app 0.1.1 → 0.1.2

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/deps.d.ts CHANGED
@@ -1,4 +1,11 @@
1
- export declare const BACKEND_DEPS: Record<string, string>;
1
+ export interface OptionalPackage {
2
+ name: string;
3
+ version: string;
4
+ label: string;
5
+ }
6
+ export declare const BACKEND_CORE_DEPS: Record<string, string>;
7
+ export declare const OPTIONAL_BACKEND_PACKAGES: OptionalPackage[];
8
+ export declare function buildBackendDeps(selectedPackages: string[]): Record<string, string>;
2
9
  export declare const BACKEND_DEV_DEPS: Record<string, string>;
3
10
  export declare const FRONTEND_DEPS: Record<string, string>;
4
11
  export declare const FRONTEND_DEV_DEPS: Record<string, string>;
package/dist/deps.js CHANGED
@@ -1,97 +1,120 @@
1
- export const BACKEND_DEPS = {
2
- '@nestjs/core': '^10.4.0',
3
- '@nestjs/common': '^10.4.0',
4
- '@nestjs/platform-express': '^10.4.0',
5
- '@nestjs/config': '^3.2.0',
6
- '@nestjs/cqrs': '^10.2.0',
7
- '@nestjs/typeorm': '^10.0.0',
8
- '@nestjs/passport': '^10.0.0',
9
- '@nestjs/swagger': '^7.4.0',
10
- '@nestjs/schedule': '^4.1.0',
11
- '@nestjs-cqrs/quanticjs': '^1.0.0',
12
- '@quanticjs/core': '^1.0.0',
13
- typeorm: '^0.3.20',
14
- pg: '^8.12.0',
15
- ioredis: '^5.4.0',
16
- '@nestjs-modules/ioredis': '^2.0.0',
17
- passport: '^0.7.0',
18
- 'passport-jwt': '^4.0.1',
19
- 'cookie-parser': '^1.4.6',
20
- 'class-validator': '^0.14.1',
21
- 'class-transformer': '^0.5.1',
22
- zod: '^3.23.0',
23
- 'nestjs-pino': '^4.1.0',
24
- 'pino-http': '^10.2.0',
25
- '@sentry/node': '^8.0.0',
26
- '@sentry/tracing': '^7.0.0',
27
- 'prom-client': '^15.1.0',
28
- rxjs: '^7.8.0',
29
- 'reflect-metadata': '^0.2.2',
1
+ export const BACKEND_CORE_DEPS = {
2
+ "@nestjs/core": "^11.0.0",
3
+ "@nestjs/common": "^11.0.0",
4
+ "@nestjs/platform-express": "^11.0.0",
5
+ "@nestjs/config": "^4.0.0",
6
+ "@nestjs/cqrs": "^11.0.3",
7
+ "@nestjs/typeorm": "^11.0.0",
8
+ "@nestjs/passport": "^11.0.0",
9
+ "@nestjs/swagger": "^11.0.0",
10
+ "@nestjs/schedule": "^5.0.0",
11
+ "@quanticjs/quanticjs": "^3.0.0",
12
+ "@quanticjs/core": "^3.0.0",
13
+ typeorm: "^0.3.20",
14
+ pg: "^8.12.0",
15
+ ioredis: "^5.4.0",
16
+ "@nestjs-modules/ioredis": "^2.0.0",
17
+ passport: "^0.7.0",
18
+ "passport-jwt": "^4.0.1",
19
+ "cookie-parser": "^1.4.6",
20
+ "class-validator": "^0.14.1",
21
+ "class-transformer": "^0.5.1",
22
+ zod: "^3.23.0",
23
+ "nestjs-pino": "^4.1.0",
24
+ "pino-http": "^11.0.0",
25
+ "@sentry/node": "^8.0.0",
26
+ "@sentry/tracing": "^7.0.0",
27
+ "prom-client": "^15.1.0",
28
+ rxjs: "^7.8.0",
29
+ "reflect-metadata": "^0.2.2",
30
30
  };
31
+ export const OPTIONAL_BACKEND_PACKAGES = [
32
+ { name: "@quanticjs/workflow", version: "^3.0.0", label: "Workflow engine" },
33
+ { name: "@quanticjs/events", version: "^3.0.0", label: "Event sourcing" },
34
+ { name: "@quanticjs/redis", version: "^3.0.0", label: "Redis utilities" },
35
+ {
36
+ name: "@quanticjs/metrics",
37
+ version: "^3.0.0",
38
+ label: "Prometheus metrics",
39
+ },
40
+ {
41
+ name: "@quanticjs/feature-flags",
42
+ version: "^3.0.0",
43
+ label: "Feature flags (Unleash)",
44
+ },
45
+ ];
46
+ export function buildBackendDeps(selectedPackages) {
47
+ const deps = { ...BACKEND_CORE_DEPS };
48
+ for (const pkg of OPTIONAL_BACKEND_PACKAGES) {
49
+ if (selectedPackages.includes(pkg.name)) {
50
+ deps[pkg.name] = pkg.version;
51
+ }
52
+ }
53
+ return deps;
54
+ }
31
55
  export const BACKEND_DEV_DEPS = {
32
- '@types/node': '^20.14.0',
33
- '@types/express': '^4.17.0',
34
- '@types/passport-jwt': '^4.0.0',
35
- '@types/cookie-parser': '^1.4.0',
36
- typescript: '^5.5.0',
37
- 'ts-node': '^10.9.0',
38
- 'tsconfig-paths': '^4.2.0',
39
- '@nestjs/cli': '^10.4.0',
40
- '@nestjs/schematics': '^10.1.0',
41
- '@nestjs/testing': '^10.4.0',
42
- 'pino-pretty': '^11.2.0',
43
- jest: '^29.7.0',
44
- 'ts-jest': '^29.2.0',
45
- '@types/jest': '^29.5.0',
46
- supertest: '^7.0.0',
47
- '@types/supertest': '^6.0.0',
56
+ "@types/node": "^20.14.0",
57
+ "@types/express": "^4.17.0",
58
+ "@types/passport-jwt": "^4.0.0",
59
+ "@types/cookie-parser": "^1.4.0",
60
+ typescript: "^5.5.0",
61
+ "ts-node": "^10.9.0",
62
+ "tsconfig-paths": "^4.2.0",
63
+ "@nestjs/cli": "^11.0.0",
64
+ "@nestjs/schematics": "^11.0.0",
65
+ "@nestjs/testing": "^11.0.0",
66
+ "pino-pretty": "^11.2.0",
67
+ jest: "^29.7.0",
68
+ "ts-jest": "^29.2.0",
69
+ "@types/jest": "^29.5.0",
70
+ supertest: "^7.0.0",
71
+ "@types/supertest": "^6.0.0",
48
72
  };
49
73
  export const FRONTEND_DEPS = {
50
- react: '^18.3.0',
51
- 'react-dom': '^18.3.0',
52
- 'react-router-dom': '^6.26.0',
53
- '@tanstack/react-query': '^5.51.0',
54
- zustand: '^4.5.0',
55
- zod: '^3.23.0',
56
- '@quanticjs/react-core': '^1.0.0',
57
- '@quanticjs/react-query': '^1.0.0',
58
- '@quanticjs/react-forms': '^1.0.0',
59
- '@quanticjs/react-ui': '^1.0.0',
60
- '@quanticjs/workflow-ui': '^1.0.0',
61
- '@radix-ui/react-dialog': '^1.1.0',
62
- '@radix-ui/react-dropdown-menu': '^2.1.0',
63
- '@radix-ui/react-label': '^2.1.0',
64
- '@radix-ui/react-select': '^2.1.0',
65
- '@radix-ui/react-slot': '^1.1.0',
66
- '@radix-ui/react-tabs': '^1.1.0',
67
- '@radix-ui/react-toast': '^1.2.0',
68
- '@radix-ui/react-tooltip': '^1.1.0',
69
- '@radix-ui/react-avatar': '^1.1.0',
70
- '@radix-ui/react-checkbox': '^1.1.0',
71
- '@radix-ui/react-popover': '^1.1.0',
72
- '@radix-ui/react-separator': '^1.1.0',
73
- '@radix-ui/react-switch': '^1.1.0',
74
- 'lucide-react': '^0.400.0',
75
- 'class-variance-authority': '^0.7.0',
76
- clsx: '^2.1.0',
77
- 'tailwind-merge': '^2.4.0',
78
- '@sentry/react': '^8.0.0',
74
+ react: "^18.3.0",
75
+ "react-dom": "^18.3.0",
76
+ "react-router-dom": "^6.26.0",
77
+ "@tanstack/react-query": "^5.51.0",
78
+ zustand: "^4.5.0",
79
+ zod: "^3.23.0",
80
+ "@quanticjs/react-core": "^0.1.0",
81
+ "@quanticjs/react-query": "^0.1.0",
82
+ "@quanticjs/react-forms": "^0.1.0",
83
+ "@quanticjs/react-ui": "^0.1.0",
84
+ "@radix-ui/react-dialog": "^1.1.0",
85
+ "@radix-ui/react-dropdown-menu": "^2.1.0",
86
+ "@radix-ui/react-label": "^2.1.0",
87
+ "@radix-ui/react-select": "^2.1.0",
88
+ "@radix-ui/react-slot": "^1.1.0",
89
+ "@radix-ui/react-tabs": "^1.1.0",
90
+ "@radix-ui/react-toast": "^1.2.0",
91
+ "@radix-ui/react-tooltip": "^1.1.0",
92
+ "@radix-ui/react-avatar": "^1.1.0",
93
+ "@radix-ui/react-checkbox": "^1.1.0",
94
+ "@radix-ui/react-popover": "^1.1.0",
95
+ "@radix-ui/react-separator": "^1.1.0",
96
+ "@radix-ui/react-switch": "^1.1.0",
97
+ "lucide-react": "^0.400.0",
98
+ "class-variance-authority": "^0.7.0",
99
+ clsx: "^2.1.0",
100
+ "tailwind-merge": "^2.4.0",
101
+ "@sentry/react": "^8.0.0",
79
102
  };
80
103
  export const FRONTEND_DEV_DEPS = {
81
- vite: '^5.4.0',
82
- '@vitejs/plugin-react': '^4.3.0',
83
- typescript: '^5.5.0',
84
- '@types/react': '^18.3.0',
85
- '@types/react-dom': '^18.3.0',
86
- tailwindcss: '^4.0.0',
87
- '@tailwindcss/vite': '^4.0.0',
88
- '@tanstack/react-query-devtools': '^5.51.0',
89
- vitest: '^2.0.0',
90
- '@testing-library/react': '^16.0.0',
91
- '@testing-library/jest-dom': '^6.4.0',
92
- '@testing-library/user-event': '^14.5.0',
93
- jsdom: '^24.1.0',
94
- '@vitest/coverage-v8': '^2.0.0',
95
- '@playwright/test': '^1.45.0',
104
+ vite: "^5.4.0",
105
+ "@vitejs/plugin-react": "^4.3.0",
106
+ typescript: "^5.5.0",
107
+ "@types/react": "^18.3.0",
108
+ "@types/react-dom": "^18.3.0",
109
+ tailwindcss: "^4.0.0",
110
+ "@tailwindcss/vite": "^4.0.0",
111
+ "@tanstack/react-query-devtools": "^5.51.0",
112
+ vitest: "^2.0.0",
113
+ "@testing-library/react": "^16.0.0",
114
+ "@testing-library/jest-dom": "^6.4.0",
115
+ "@testing-library/user-event": "^14.5.0",
116
+ jsdom: "^24.1.0",
117
+ "@vitest/coverage-v8": "^2.0.0",
118
+ "@playwright/test": "^1.45.0",
96
119
  };
97
120
  //# sourceMappingURL=deps.js.map
package/dist/deps.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"deps.js","sourceRoot":"","sources":["../src/deps.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,YAAY,GAA2B;IAClD,cAAc,EAAE,SAAS;IACzB,gBAAgB,EAAE,SAAS;IAC3B,0BAA0B,EAAE,SAAS;IACrC,gBAAgB,EAAE,QAAQ;IAC1B,cAAc,EAAE,SAAS;IACzB,iBAAiB,EAAE,SAAS;IAC5B,kBAAkB,EAAE,SAAS;IAC7B,iBAAiB,EAAE,QAAQ;IAC3B,kBAAkB,EAAE,QAAQ;IAC5B,wBAAwB,EAAE,QAAQ;IAClC,iBAAiB,EAAE,QAAQ;IAC3B,OAAO,EAAE,SAAS;IAClB,EAAE,EAAE,SAAS;IACb,OAAO,EAAE,QAAQ;IACjB,yBAAyB,EAAE,QAAQ;IACnC,QAAQ,EAAE,QAAQ;IAClB,cAAc,EAAE,QAAQ;IACxB,eAAe,EAAE,QAAQ;IACzB,iBAAiB,EAAE,SAAS;IAC5B,mBAAmB,EAAE,QAAQ;IAC7B,GAAG,EAAE,SAAS;IACd,aAAa,EAAE,QAAQ;IACvB,WAAW,EAAE,SAAS;IACtB,cAAc,EAAE,QAAQ;IACxB,iBAAiB,EAAE,QAAQ;IAC3B,aAAa,EAAE,SAAS;IACxB,IAAI,EAAE,QAAQ;IACd,kBAAkB,EAAE,QAAQ;CAC7B,CAAC;AAEF,MAAM,CAAC,MAAM,gBAAgB,GAA2B;IACtD,aAAa,EAAE,UAAU;IACzB,gBAAgB,EAAE,SAAS;IAC3B,qBAAqB,EAAE,QAAQ;IAC/B,sBAAsB,EAAE,QAAQ;IAChC,UAAU,EAAE,QAAQ;IACpB,SAAS,EAAE,SAAS;IACpB,gBAAgB,EAAE,QAAQ;IAC1B,aAAa,EAAE,SAAS;IACxB,oBAAoB,EAAE,SAAS;IAC/B,iBAAiB,EAAE,SAAS;IAC5B,aAAa,EAAE,SAAS;IACxB,IAAI,EAAE,SAAS;IACf,SAAS,EAAE,SAAS;IACpB,aAAa,EAAE,SAAS;IACxB,SAAS,EAAE,QAAQ;IACnB,kBAAkB,EAAE,QAAQ;CAC7B,CAAC;AAEF,MAAM,CAAC,MAAM,aAAa,GAA2B;IACnD,KAAK,EAAE,SAAS;IAChB,WAAW,EAAE,SAAS;IACtB,kBAAkB,EAAE,SAAS;IAC7B,uBAAuB,EAAE,SAAS;IAClC,OAAO,EAAE,QAAQ;IACjB,GAAG,EAAE,SAAS;IACd,uBAAuB,EAAE,QAAQ;IACjC,wBAAwB,EAAE,QAAQ;IAClC,wBAAwB,EAAE,QAAQ;IAClC,qBAAqB,EAAE,QAAQ;IAC/B,wBAAwB,EAAE,QAAQ;IAClC,wBAAwB,EAAE,QAAQ;IAClC,+BAA+B,EAAE,QAAQ;IACzC,uBAAuB,EAAE,QAAQ;IACjC,wBAAwB,EAAE,QAAQ;IAClC,sBAAsB,EAAE,QAAQ;IAChC,sBAAsB,EAAE,QAAQ;IAChC,uBAAuB,EAAE,QAAQ;IACjC,yBAAyB,EAAE,QAAQ;IACnC,wBAAwB,EAAE,QAAQ;IAClC,0BAA0B,EAAE,QAAQ;IACpC,yBAAyB,EAAE,QAAQ;IACnC,2BAA2B,EAAE,QAAQ;IACrC,wBAAwB,EAAE,QAAQ;IAClC,cAAc,EAAE,UAAU;IAC1B,0BAA0B,EAAE,QAAQ;IACpC,IAAI,EAAE,QAAQ;IACd,gBAAgB,EAAE,QAAQ;IAC1B,eAAe,EAAE,QAAQ;CAC1B,CAAC;AAEF,MAAM,CAAC,MAAM,iBAAiB,GAA2B;IACvD,IAAI,EAAE,QAAQ;IACd,sBAAsB,EAAE,QAAQ;IAChC,UAAU,EAAE,QAAQ;IACpB,cAAc,EAAE,SAAS;IACzB,kBAAkB,EAAE,SAAS;IAC7B,WAAW,EAAE,QAAQ;IACrB,mBAAmB,EAAE,QAAQ;IAC7B,gCAAgC,EAAE,SAAS;IAC3C,MAAM,EAAE,QAAQ;IAChB,wBAAwB,EAAE,SAAS;IACnC,2BAA2B,EAAE,QAAQ;IACrC,6BAA6B,EAAE,SAAS;IACxC,KAAK,EAAE,SAAS;IAChB,qBAAqB,EAAE,QAAQ;IAC/B,kBAAkB,EAAE,SAAS;CAC9B,CAAC"}
1
+ {"version":3,"file":"deps.js","sourceRoot":"","sources":["../src/deps.ts"],"names":[],"mappings":"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"}
@@ -1,2 +1,2 @@
1
- import { type TemplateContext } from '../utils/template.js';
2
- export declare function generateBackend(projectDir: string, ctx: TemplateContext): void;
1
+ import { type TemplateContext } from "../utils/template.js";
2
+ export declare function generateBackend(projectDir: string, ctx: TemplateContext, selectedPackages: string[]): void;
@@ -1,20 +1,20 @@
1
- import { join } from 'node:path';
2
- import { renderAndWrite, renderTemplate, writeFile } from '../utils/template.js';
3
- import { toPascalCase } from '../utils/validate.js';
4
- import { BACKEND_DEPS, BACKEND_DEV_DEPS } from '../deps.js';
5
- export function generateBackend(projectDir, ctx) {
6
- const rendered = renderTemplate('backend/package.json.ejs', ctx);
1
+ import { join } from "node:path";
2
+ import { renderAndWrite, renderTemplate, writeFile, } from "../utils/template.js";
3
+ import { toPascalCase } from "../utils/validate.js";
4
+ import { buildBackendDeps, BACKEND_DEV_DEPS } from "../deps.js";
5
+ export function generateBackend(projectDir, ctx, selectedPackages) {
6
+ const rendered = renderTemplate("backend/package.json.ejs", ctx);
7
7
  const pkgJson = JSON.parse(rendered);
8
- pkgJson.dependencies = BACKEND_DEPS;
8
+ pkgJson.dependencies = buildBackendDeps(selectedPackages);
9
9
  pkgJson.devDependencies = BACKEND_DEV_DEPS;
10
- writeFile(join(projectDir, 'package.json'), JSON.stringify(pkgJson, null, 2) + '\n');
11
- renderAndWrite('backend/tsconfig.json.ejs', join(projectDir, 'tsconfig.json'), ctx);
12
- renderAndWrite('backend/tsconfig.build.json.ejs', join(projectDir, 'tsconfig.build.json'), ctx);
13
- renderAndWrite('backend/nest-cli.json.ejs', join(projectDir, 'nest-cli.json'), ctx);
14
- renderAndWrite('backend/env.example.ejs', join(projectDir, '.env.example'), ctx);
10
+ writeFile(join(projectDir, "package.json"), JSON.stringify(pkgJson, null, 2) + "\n");
11
+ renderAndWrite("backend/tsconfig.json.ejs", join(projectDir, "tsconfig.json"), ctx);
12
+ renderAndWrite("backend/tsconfig.build.json.ejs", join(projectDir, "tsconfig.build.json"), ctx);
13
+ renderAndWrite("backend/nest-cli.json.ejs", join(projectDir, "nest-cli.json"), ctx);
14
+ renderAndWrite("backend/env.example.ejs", join(projectDir, ".env.example"), ctx);
15
15
  const extendedCtx = { ...ctx, toPascalCase };
16
- renderAndWrite('backend/main.ts.ejs', join(projectDir, 'src', 'main.ts'), extendedCtx);
17
- renderAndWrite('backend/app.module.ts.ejs', join(projectDir, 'src', 'app.module.ts'), extendedCtx);
18
- renderAndWrite('backend/data-source.ts.ejs', join(projectDir, 'src', 'data-source.ts'), ctx);
16
+ renderAndWrite("backend/main.ts.ejs", join(projectDir, "src", "main.ts"), extendedCtx);
17
+ renderAndWrite("backend/app.module.ts.ejs", join(projectDir, "src", "app.module.ts"), extendedCtx);
18
+ renderAndWrite("backend/data-source.ts.ejs", join(projectDir, "src", "data-source.ts"), ctx);
19
19
  }
20
20
  //# sourceMappingURL=backend.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"backend.js","sourceRoot":"","sources":["../../src/generators/backend.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAEjC,OAAO,EAAE,cAAc,EAAE,cAAc,EAAE,SAAS,EAAwB,MAAM,sBAAsB,CAAC;AACvG,OAAO,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAC;AACpD,OAAO,EAAE,YAAY,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAC;AAE5D,MAAM,UAAU,eAAe,CAAC,UAAkB,EAAE,GAAoB;IACtE,MAAM,QAAQ,GAAG,cAAc,CAAC,0BAA0B,EAAE,GAAG,CAAC,CAAC;IACjE,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;IACrC,OAAO,CAAC,YAAY,GAAG,YAAY,CAAC;IACpC,OAAO,CAAC,eAAe,GAAG,gBAAgB,CAAC;IAC3C,SAAS,CAAC,IAAI,CAAC,UAAU,EAAE,cAAc,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC;IAErF,cAAc,CAAC,2BAA2B,EAAE,IAAI,CAAC,UAAU,EAAE,eAAe,CAAC,EAAE,GAAG,CAAC,CAAC;IACpF,cAAc,CAAC,iCAAiC,EAAE,IAAI,CAAC,UAAU,EAAE,qBAAqB,CAAC,EAAE,GAAG,CAAC,CAAC;IAChG,cAAc,CAAC,2BAA2B,EAAE,IAAI,CAAC,UAAU,EAAE,eAAe,CAAC,EAAE,GAAG,CAAC,CAAC;IACpF,cAAc,CAAC,yBAAyB,EAAE,IAAI,CAAC,UAAU,EAAE,cAAc,CAAC,EAAE,GAAG,CAAC,CAAC;IAEjF,MAAM,WAAW,GAAG,EAAE,GAAG,GAAG,EAAE,YAAY,EAAE,CAAC;IAC7C,cAAc,CAAC,qBAAqB,EAAE,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE,SAAS,CAAC,EAAE,WAAW,CAAC,CAAC;IACvF,cAAc,CAAC,2BAA2B,EAAE,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE,eAAe,CAAC,EAAE,WAAW,CAAC,CAAC;IACnG,cAAc,CAAC,4BAA4B,EAAE,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE,gBAAgB,CAAC,EAAE,GAAG,CAAC,CAAC;AAC/F,CAAC"}
1
+ {"version":3,"file":"backend.js","sourceRoot":"","sources":["../../src/generators/backend.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EACL,cAAc,EACd,cAAc,EACd,SAAS,GAEV,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAC;AACpD,OAAO,EAAE,gBAAgB,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAC;AAEhE,MAAM,UAAU,eAAe,CAC7B,UAAkB,EAClB,GAAoB,EACpB,gBAA0B;IAE1B,MAAM,QAAQ,GAAG,cAAc,CAAC,0BAA0B,EAAE,GAAG,CAAC,CAAC;IACjE,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;IACrC,OAAO,CAAC,YAAY,GAAG,gBAAgB,CAAC,gBAAgB,CAAC,CAAC;IAC1D,OAAO,CAAC,eAAe,GAAG,gBAAgB,CAAC;IAC3C,SAAS,CACP,IAAI,CAAC,UAAU,EAAE,cAAc,CAAC,EAChC,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,GAAG,IAAI,CACxC,CAAC;IAEF,cAAc,CACZ,2BAA2B,EAC3B,IAAI,CAAC,UAAU,EAAE,eAAe,CAAC,EACjC,GAAG,CACJ,CAAC;IACF,cAAc,CACZ,iCAAiC,EACjC,IAAI,CAAC,UAAU,EAAE,qBAAqB,CAAC,EACvC,GAAG,CACJ,CAAC;IACF,cAAc,CACZ,2BAA2B,EAC3B,IAAI,CAAC,UAAU,EAAE,eAAe,CAAC,EACjC,GAAG,CACJ,CAAC;IACF,cAAc,CACZ,yBAAyB,EACzB,IAAI,CAAC,UAAU,EAAE,cAAc,CAAC,EAChC,GAAG,CACJ,CAAC;IAEF,MAAM,WAAW,GAAG,EAAE,GAAG,GAAG,EAAE,YAAY,EAAE,CAAC;IAC7C,cAAc,CACZ,qBAAqB,EACrB,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE,SAAS,CAAC,EAClC,WAAW,CACZ,CAAC;IACF,cAAc,CACZ,2BAA2B,EAC3B,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE,eAAe,CAAC,EACxC,WAAW,CACZ,CAAC;IACF,cAAc,CACZ,4BAA4B,EAC5B,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE,gBAAgB,CAAC,EACzC,GAAG,CACJ,CAAC;AACJ,CAAC"}
@@ -1,2 +1,2 @@
1
- import { type TemplateContext } from '../utils/template.js';
1
+ import { type TemplateContext } from "../utils/template.js";
2
2
  export declare function generateDocker(projectDir: string, ctx: TemplateContext): void;
@@ -1,10 +1,12 @@
1
- import { join } from 'node:path';
2
- import { renderAndWrite } from '../utils/template.js';
1
+ import { join } from "node:path";
2
+ import { renderAndWrite } from "../utils/template.js";
3
3
  export function generateDocker(projectDir, ctx) {
4
- renderAndWrite('docker/Dockerfile.ejs', join(projectDir, 'Dockerfile'), ctx);
5
- renderAndWrite('docker/Dockerfile.client.ejs', join(projectDir, 'client', 'Dockerfile'), ctx);
6
- renderAndWrite('docker/docker-compose.yml.ejs', join(projectDir, 'docker-compose.yml'), ctx);
7
- renderAndWrite('docker/docker-compose.test.yml.ejs', join(projectDir, 'docker-compose.test.yml'), ctx);
8
- renderAndWrite('docker/nginx.conf.ejs', join(projectDir, 'client', 'nginx.conf'), ctx);
4
+ renderAndWrite("docker/Dockerfile.ejs", join(projectDir, "Dockerfile"), ctx);
5
+ renderAndWrite("docker/docker-compose.yml.ejs", join(projectDir, "docker-compose.yml"), ctx);
6
+ renderAndWrite("docker/docker-compose.test.yml.ejs", join(projectDir, "docker-compose.test.yml"), ctx);
7
+ if (ctx.includeFrontend) {
8
+ renderAndWrite("docker/Dockerfile.client.ejs", join(projectDir, "client", "Dockerfile"), ctx);
9
+ renderAndWrite("docker/nginx.conf.ejs", join(projectDir, "client", "nginx.conf"), ctx);
10
+ }
9
11
  }
10
12
  //# sourceMappingURL=docker.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"docker.js","sourceRoot":"","sources":["../../src/generators/docker.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,cAAc,EAAwB,MAAM,sBAAsB,CAAC;AAE5E,MAAM,UAAU,cAAc,CAAC,UAAkB,EAAE,GAAoB;IACrE,cAAc,CAAC,uBAAuB,EAAE,IAAI,CAAC,UAAU,EAAE,YAAY,CAAC,EAAE,GAAG,CAAC,CAAC;IAC7E,cAAc,CAAC,8BAA8B,EAAE,IAAI,CAAC,UAAU,EAAE,QAAQ,EAAE,YAAY,CAAC,EAAE,GAAG,CAAC,CAAC;IAC9F,cAAc,CAAC,+BAA+B,EAAE,IAAI,CAAC,UAAU,EAAE,oBAAoB,CAAC,EAAE,GAAG,CAAC,CAAC;IAC7F,cAAc,CAAC,oCAAoC,EAAE,IAAI,CAAC,UAAU,EAAE,yBAAyB,CAAC,EAAE,GAAG,CAAC,CAAC;IACvG,cAAc,CAAC,uBAAuB,EAAE,IAAI,CAAC,UAAU,EAAE,QAAQ,EAAE,YAAY,CAAC,EAAE,GAAG,CAAC,CAAC;AACzF,CAAC"}
1
+ {"version":3,"file":"docker.js","sourceRoot":"","sources":["../../src/generators/docker.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,cAAc,EAAwB,MAAM,sBAAsB,CAAC;AAE5E,MAAM,UAAU,cAAc,CAAC,UAAkB,EAAE,GAAoB;IACrE,cAAc,CAAC,uBAAuB,EAAE,IAAI,CAAC,UAAU,EAAE,YAAY,CAAC,EAAE,GAAG,CAAC,CAAC;IAC7E,cAAc,CACZ,+BAA+B,EAC/B,IAAI,CAAC,UAAU,EAAE,oBAAoB,CAAC,EACtC,GAAG,CACJ,CAAC;IACF,cAAc,CACZ,oCAAoC,EACpC,IAAI,CAAC,UAAU,EAAE,yBAAyB,CAAC,EAC3C,GAAG,CACJ,CAAC;IAEF,IAAI,GAAG,CAAC,eAAe,EAAE,CAAC;QACxB,cAAc,CACZ,8BAA8B,EAC9B,IAAI,CAAC,UAAU,EAAE,QAAQ,EAAE,YAAY,CAAC,EACxC,GAAG,CACJ,CAAC;QACF,cAAc,CACZ,uBAAuB,EACvB,IAAI,CAAC,UAAU,EAAE,QAAQ,EAAE,YAAY,CAAC,EACxC,GAAG,CACJ,CAAC;IACJ,CAAC;AACH,CAAC"}
package/dist/index.js CHANGED
@@ -1,31 +1,45 @@
1
1
  #!/usr/bin/env node
2
- import { Command } from 'commander';
3
- import { promptForOptions } from './prompts.js';
4
- import { scaffold } from './scaffold.js';
2
+ import { Command } from "commander";
3
+ import { promptForOptions } from "./prompts.js";
4
+ import { scaffold } from "./scaffold.js";
5
5
  const program = new Command();
6
6
  program
7
- .name('create-quanticjs-app')
8
- .description('Scaffold a QuanticJS project — NestJS + React + Docker')
9
- .version('0.1.0')
10
- .argument('[project-name]', 'Project name (kebab-case)')
11
- .option('-d, --description <desc>', 'Project description')
12
- .option('-m, --modules <modules>', 'Comma-separated module names')
13
- .option('--no-install', 'Skip npm install')
14
- .option('--no-git', 'Skip git init')
15
- .option('--dry-run', 'Print what would be created without writing')
7
+ .name("create-quanticjs-app")
8
+ .description("Scaffold a QuanticJS project — NestJS + React + Docker")
9
+ .version("0.1.0")
10
+ .argument("[project-name]", "Project name (kebab-case)")
11
+ .option("-d, --description <desc>", "Project description")
12
+ .option("-m, --modules <modules>", "Comma-separated module names")
13
+ .option("--frontend", "Include frontend (default)")
14
+ .option("--no-frontend", "Skip frontend generation")
15
+ .option("--packages <packages>", "Comma-separated optional @quanticjs backend packages")
16
+ .option("--no-install", "Skip npm install")
17
+ .option("--no-git", "Skip git init")
18
+ .option("--dry-run", "Print what would be created without writing")
16
19
  .action(async (nameArg, opts) => {
17
20
  try {
18
21
  const cliModules = opts.modules
19
- ? opts.modules.split(',').map((m) => m.trim()).filter(Boolean)
22
+ ? opts.modules
23
+ .split(",")
24
+ .map((m) => m.trim())
25
+ .filter(Boolean)
26
+ : undefined;
27
+ const cliPackages = opts.packages
28
+ ? opts.packages
29
+ .split(",")
30
+ .map((p) => p.trim())
31
+ .filter(Boolean)
20
32
  : undefined;
21
33
  const options = await promptForOptions(nameArg, {
22
34
  projectDescription: opts.description,
23
35
  modules: cliModules,
36
+ includeFrontend: opts.frontend ?? (opts.dryRun ? true : undefined),
37
+ backendPackages: cliPackages ?? (opts.dryRun ? [] : undefined),
24
38
  skipInstall: !opts.install,
25
39
  skipGit: !opts.git,
26
40
  });
27
41
  if (opts.dryRun) {
28
- console.log('Dry run — would create project with:');
42
+ console.log("Dry run — would create project with:");
29
43
  console.log(JSON.stringify(options, null, 2));
30
44
  return;
31
45
  }
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AAEA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,gBAAgB,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAEzC,MAAM,OAAO,GAAG,IAAI,OAAO,EAAE,CAAC;AAE9B,OAAO;KACJ,IAAI,CAAC,sBAAsB,CAAC;KAC5B,WAAW,CAAC,wDAAwD,CAAC;KACrE,OAAO,CAAC,OAAO,CAAC;KAChB,QAAQ,CAAC,gBAAgB,EAAE,2BAA2B,CAAC;KACvD,MAAM,CAAC,0BAA0B,EAAE,qBAAqB,CAAC;KACzD,MAAM,CAAC,yBAAyB,EAAE,8BAA8B,CAAC;KACjE,MAAM,CAAC,cAAc,EAAE,kBAAkB,CAAC;KAC1C,MAAM,CAAC,UAAU,EAAE,eAAe,CAAC;KACnC,MAAM,CAAC,WAAW,EAAE,6CAA6C,CAAC;KAClE,MAAM,CAAC,KAAK,EAAE,OAA2B,EAAE,IAAI,EAAE,EAAE;IAClD,IAAI,CAAC;QACH,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO;YAC7B,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC;YACtE,CAAC,CAAC,SAAS,CAAC;QAEd,MAAM,OAAO,GAAG,MAAM,gBAAgB,CAAC,OAAO,EAAE;YAC9C,kBAAkB,EAAE,IAAI,CAAC,WAAW;YACpC,OAAO,EAAE,UAAU;YACnB,WAAW,EAAE,CAAC,IAAI,CAAC,OAAO;YAC1B,OAAO,EAAE,CAAC,IAAI,CAAC,GAAG;SACnB,CAAC,CAAC;QAEH,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,OAAO,CAAC,GAAG,CAAC,sCAAsC,CAAC,CAAC;YACpD,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;YAC9C,OAAO;QACT,CAAC;QAED,MAAM,QAAQ,CAAC,OAAO,EAAE,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;IACzC,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,OAAO,CAAC,KAAK,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;QACxD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC,CAAC,CAAC;AAEL,OAAO,CAAC,KAAK,EAAE,CAAC"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AAEA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,gBAAgB,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAEzC,MAAM,OAAO,GAAG,IAAI,OAAO,EAAE,CAAC;AAE9B,OAAO;KACJ,IAAI,CAAC,sBAAsB,CAAC;KAC5B,WAAW,CAAC,wDAAwD,CAAC;KACrE,OAAO,CAAC,OAAO,CAAC;KAChB,QAAQ,CAAC,gBAAgB,EAAE,2BAA2B,CAAC;KACvD,MAAM,CAAC,0BAA0B,EAAE,qBAAqB,CAAC;KACzD,MAAM,CAAC,yBAAyB,EAAE,8BAA8B,CAAC;KACjE,MAAM,CAAC,YAAY,EAAE,4BAA4B,CAAC;KAClD,MAAM,CAAC,eAAe,EAAE,0BAA0B,CAAC;KACnD,MAAM,CACL,uBAAuB,EACvB,sDAAsD,CACvD;KACA,MAAM,CAAC,cAAc,EAAE,kBAAkB,CAAC;KAC1C,MAAM,CAAC,UAAU,EAAE,eAAe,CAAC;KACnC,MAAM,CAAC,WAAW,EAAE,6CAA6C,CAAC;KAClE,MAAM,CAAC,KAAK,EAAE,OAA2B,EAAE,IAAI,EAAE,EAAE;IAClD,IAAI,CAAC;QACH,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO;YAC7B,CAAC,CAAC,IAAI,CAAC,OAAO;iBACT,KAAK,CAAC,GAAG,CAAC;iBACV,GAAG,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;iBAC5B,MAAM,CAAC,OAAO,CAAC;YACpB,CAAC,CAAC,SAAS,CAAC;QAEd,MAAM,WAAW,GAAG,IAAI,CAAC,QAAQ;YAC/B,CAAC,CAAC,IAAI,CAAC,QAAQ;iBACV,KAAK,CAAC,GAAG,CAAC;iBACV,GAAG,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;iBAC5B,MAAM,CAAC,OAAO,CAAC;YACpB,CAAC,CAAC,SAAS,CAAC;QAEd,MAAM,OAAO,GAAG,MAAM,gBAAgB,CAAC,OAAO,EAAE;YAC9C,kBAAkB,EAAE,IAAI,CAAC,WAAW;YACpC,OAAO,EAAE,UAAU;YACnB,eAAe,EAAE,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC;YAClE,eAAe,EAAE,WAAW,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;YAC9D,WAAW,EAAE,CAAC,IAAI,CAAC,OAAO;YAC1B,OAAO,EAAE,CAAC,IAAI,CAAC,GAAG;SACnB,CAAC,CAAC;QAEH,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,OAAO,CAAC,GAAG,CAAC,sCAAsC,CAAC,CAAC;YACpD,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;YAC9C,OAAO;QACT,CAAC;QAED,MAAM,QAAQ,CAAC,OAAO,EAAE,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;IACzC,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,OAAO,CAAC,KAAK,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;QACxD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC,CAAC,CAAC;AAEL,OAAO,CAAC,KAAK,EAAE,CAAC"}
package/dist/prompts.d.ts CHANGED
@@ -2,6 +2,8 @@ export interface ProjectOptions {
2
2
  projectName: string;
3
3
  projectDescription: string;
4
4
  modules: string[];
5
+ includeFrontend: boolean;
6
+ backendPackages: string[];
5
7
  skipInstall: boolean;
6
8
  skipGit: boolean;
7
9
  }
package/dist/prompts.js CHANGED
@@ -1,31 +1,32 @@
1
- import inquirer from 'inquirer';
2
- import { validateProjectName, validateModuleName } from './utils/validate.js';
1
+ import inquirer from "inquirer";
2
+ import { validateProjectName, validateModuleName } from "./utils/validate.js";
3
+ import { OPTIONAL_BACKEND_PACKAGES } from "./deps.js";
3
4
  export async function promptForOptions(nameArg, cliOpts) {
4
5
  const answers = await inquirer.prompt([
5
6
  {
6
- type: 'input',
7
- name: 'projectName',
8
- message: 'Project name (kebab-case):',
7
+ type: "input",
8
+ name: "projectName",
9
+ message: "Project name (kebab-case):",
9
10
  default: nameArg,
10
11
  when: !nameArg,
11
12
  validate: validateProjectName,
12
13
  },
13
14
  {
14
- type: 'input',
15
- name: 'projectDescription',
16
- message: 'Project description:',
17
- default: 'A QuanticJS application',
15
+ type: "input",
16
+ name: "projectDescription",
17
+ message: "Project description:",
18
+ default: "A QuanticJS application",
18
19
  when: !cliOpts?.projectDescription,
19
20
  },
20
21
  {
21
- type: 'input',
22
- name: 'modulesRaw',
23
- message: 'Initial modules (comma-separated, e.g. change-request,estimation):',
22
+ type: "input",
23
+ name: "modulesRaw",
24
+ message: "Initial modules (comma-separated, e.g. change-request,estimation):",
24
25
  when: !cliOpts?.modules,
25
26
  validate: (input) => {
26
27
  if (!input.trim())
27
- return 'At least one module is required';
28
- const mods = input.split(',').map((m) => m.trim());
28
+ return "At least one module is required";
29
+ const mods = input.split(",").map((m) => m.trim());
29
30
  for (const mod of mods) {
30
31
  const result = validateModuleName(mod);
31
32
  if (result !== true)
@@ -34,18 +35,40 @@ export async function promptForOptions(nameArg, cliOpts) {
34
35
  return true;
35
36
  },
36
37
  },
38
+ {
39
+ type: "confirm",
40
+ name: "includeFrontend",
41
+ message: "Include frontend (React + Vite)?",
42
+ default: true,
43
+ when: cliOpts?.includeFrontend === undefined,
44
+ },
45
+ {
46
+ type: "checkbox",
47
+ name: "backendPackages",
48
+ message: "Which backend QuanticJS packages?",
49
+ choices: OPTIONAL_BACKEND_PACKAGES.map((pkg) => ({
50
+ name: `${pkg.name} — ${pkg.label}`,
51
+ value: pkg.name,
52
+ checked: true,
53
+ })),
54
+ when: cliOpts?.backendPackages === undefined,
55
+ },
37
56
  ]);
38
57
  const projectName = nameArg ?? answers.projectName;
39
- const projectDescription = cliOpts?.projectDescription ?? answers.projectDescription ?? 'A QuanticJS application';
58
+ const projectDescription = cliOpts?.projectDescription ??
59
+ answers.projectDescription ??
60
+ "A QuanticJS application";
40
61
  const modules = cliOpts?.modules ??
41
62
  answers.modulesRaw
42
- .split(',')
63
+ .split(",")
43
64
  .map((m) => m.trim())
44
65
  .filter(Boolean);
45
66
  return {
46
67
  projectName,
47
68
  projectDescription,
48
69
  modules,
70
+ includeFrontend: cliOpts?.includeFrontend ?? answers.includeFrontend ?? true,
71
+ backendPackages: cliOpts?.backendPackages ?? answers.backendPackages ?? [],
49
72
  skipInstall: cliOpts?.skipInstall ?? false,
50
73
  skipGit: cliOpts?.skipGit ?? false,
51
74
  };
@@ -1 +1 @@
1
- {"version":3,"file":"prompts.js","sourceRoot":"","sources":["../src/prompts.ts"],"names":[],"mappings":"AAAA,OAAO,QAAQ,MAAM,UAAU,CAAC;AAChC,OAAO,EAAE,mBAAmB,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AAU9E,MAAM,CAAC,KAAK,UAAU,gBAAgB,CACpC,OAAgB,EAChB,OAAiC;IAEjC,MAAM,OAAO,GAAG,MAAM,QAAQ,CAAC,MAAM,CAAC;QACpC;YACE,IAAI,EAAE,OAAO;YACb,IAAI,EAAE,aAAa;YACnB,OAAO,EAAE,4BAA4B;YACrC,OAAO,EAAE,OAAO;YAChB,IAAI,EAAE,CAAC,OAAO;YACd,QAAQ,EAAE,mBAAmB;SAC9B;QACD;YACE,IAAI,EAAE,OAAO;YACb,IAAI,EAAE,oBAAoB;YAC1B,OAAO,EAAE,sBAAsB;YAC/B,OAAO,EAAE,yBAAyB;YAClC,IAAI,EAAE,CAAC,OAAO,EAAE,kBAAkB;SACnC;QACD;YACE,IAAI,EAAE,OAAO;YACb,IAAI,EAAE,YAAY;YAClB,OAAO,EAAE,oEAAoE;YAC7E,IAAI,EAAE,CAAC,OAAO,EAAE,OAAO;YACvB,QAAQ,EAAE,CAAC,KAAa,EAAE,EAAE;gBAC1B,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE;oBAAE,OAAO,iCAAiC,CAAC;gBAC5D,MAAM,IAAI,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;gBACnD,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;oBACvB,MAAM,MAAM,GAAG,kBAAkB,CAAC,GAAG,CAAC,CAAC;oBACvC,IAAI,MAAM,KAAK,IAAI;wBAAE,OAAO,WAAW,GAAG,MAAM,MAAM,EAAE,CAAC;gBAC3D,CAAC;gBACD,OAAO,IAAI,CAAC;YACd,CAAC;SACF;KACF,CAAC,CAAC;IAEH,MAAM,WAAW,GAAG,OAAO,IAAI,OAAO,CAAC,WAAW,CAAC;IACnD,MAAM,kBAAkB,GACtB,OAAO,EAAE,kBAAkB,IAAI,OAAO,CAAC,kBAAkB,IAAI,yBAAyB,CAAC;IACzF,MAAM,OAAO,GACX,OAAO,EAAE,OAAO;QACf,OAAO,CAAC,UAAqB;aAC3B,KAAK,CAAC,GAAG,CAAC;aACV,GAAG,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;aAC5B,MAAM,CAAC,OAAO,CAAC,CAAC;IAErB,OAAO;QACL,WAAW;QACX,kBAAkB;QAClB,OAAO;QACP,WAAW,EAAE,OAAO,EAAE,WAAW,IAAI,KAAK;QAC1C,OAAO,EAAE,OAAO,EAAE,OAAO,IAAI,KAAK;KACnC,CAAC;AACJ,CAAC"}
1
+ {"version":3,"file":"prompts.js","sourceRoot":"","sources":["../src/prompts.ts"],"names":[],"mappings":"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"}
@@ -1,2 +1,2 @@
1
- import type { ProjectOptions } from './prompts.js';
1
+ import type { ProjectOptions } from "./prompts.js";
2
2
  export declare function scaffold(options: ProjectOptions, outputDir: string): Promise<void>;
package/dist/scaffold.js CHANGED
@@ -1,17 +1,17 @@
1
- import { join } from 'node:path';
2
- import { existsSync } from 'node:fs';
3
- import ora from 'ora';
4
- import { toUnderscored, toPascalCase } from './utils/validate.js';
5
- import { run } from './utils/exec.js';
6
- import { generateRoot } from './generators/root.js';
7
- import { generateBackend } from './generators/backend.js';
8
- import { generateBff } from './generators/bff.js';
9
- import { generateModules } from './generators/module.js';
10
- import { generateFrontend } from './generators/frontend.js';
11
- import { generateDocker } from './generators/docker.js';
12
- import { generateScripts } from './generators/scripts.js';
13
- import { generateE2e } from './generators/e2e.js';
14
- import { generateClaude } from './generators/claude.js';
1
+ import { join } from "node:path";
2
+ import { existsSync } from "node:fs";
3
+ import ora from "ora";
4
+ import { toUnderscored, toPascalCase } from "./utils/validate.js";
5
+ import { run } from "./utils/exec.js";
6
+ import { generateRoot } from "./generators/root.js";
7
+ import { generateBackend } from "./generators/backend.js";
8
+ import { generateBff } from "./generators/bff.js";
9
+ import { generateModules } from "./generators/module.js";
10
+ import { generateFrontend } from "./generators/frontend.js";
11
+ import { generateDocker } from "./generators/docker.js";
12
+ import { generateScripts } from "./generators/scripts.js";
13
+ import { generateE2e } from "./generators/e2e.js";
14
+ import { generateClaude } from "./generators/claude.js";
15
15
  export async function scaffold(options, outputDir) {
16
16
  const projectDir = join(outputDir, options.projectName);
17
17
  if (existsSync(projectDir)) {
@@ -23,57 +23,66 @@ export async function scaffold(options, outputDir) {
23
23
  projectNameUnderscored: toUnderscored(options.projectName),
24
24
  projectNamePascal: toPascalCase(options.projectName),
25
25
  modules: options.modules,
26
+ includeFrontend: options.includeFrontend,
26
27
  };
27
28
  const spinner = ora();
28
- spinner.start('Generating root files');
29
+ spinner.start("Generating root files");
29
30
  generateRoot(projectDir, ctx);
30
31
  spinner.succeed();
31
- spinner.start('Generating backend');
32
- generateBackend(projectDir, ctx);
32
+ spinner.start("Generating backend");
33
+ generateBackend(projectDir, ctx, options.backendPackages);
33
34
  spinner.succeed();
34
- spinner.start('Generating BFF auth module');
35
+ spinner.start("Generating BFF auth module");
35
36
  generateBff(projectDir, ctx);
36
37
  spinner.succeed();
37
- spinner.start(`Generating modules: ${options.modules.join(', ')}`);
38
+ spinner.start(`Generating modules: ${options.modules.join(", ")}`);
38
39
  generateModules(projectDir, ctx);
39
40
  spinner.succeed();
40
- spinner.start('Generating frontend');
41
- generateFrontend(projectDir, ctx);
42
- spinner.succeed();
43
- spinner.start('Generating Docker files');
41
+ if (options.includeFrontend) {
42
+ spinner.start("Generating frontend");
43
+ generateFrontend(projectDir, ctx);
44
+ spinner.succeed();
45
+ }
46
+ spinner.start("Generating Docker files");
44
47
  generateDocker(projectDir, ctx);
45
48
  spinner.succeed();
46
- spinner.start('Generating scripts');
49
+ spinner.start("Generating scripts");
47
50
  generateScripts(projectDir, ctx);
48
51
  spinner.succeed();
49
- spinner.start('Generating E2E setup');
50
- generateE2e(projectDir, ctx);
51
- spinner.succeed();
52
- spinner.start('Generating .claude config (rules, skills, hooks)');
52
+ if (options.includeFrontend) {
53
+ spinner.start("Generating E2E setup");
54
+ generateE2e(projectDir, ctx);
55
+ spinner.succeed();
56
+ }
57
+ spinner.start("Generating .claude config (rules, skills, hooks)");
53
58
  generateClaude(projectDir, ctx);
54
59
  spinner.succeed();
55
60
  if (!options.skipInstall) {
56
- spinner.start('Installing backend dependencies');
57
- run('npm install', projectDir);
58
- spinner.succeed();
59
- spinner.start('Installing frontend dependencies');
60
- run('npm install', join(projectDir, 'client'));
61
+ spinner.start("Installing backend dependencies");
62
+ run("npm install", projectDir);
61
63
  spinner.succeed();
64
+ if (options.includeFrontend) {
65
+ spinner.start("Installing frontend dependencies");
66
+ run("npm install", join(projectDir, "client"));
67
+ spinner.succeed();
68
+ }
62
69
  }
63
70
  if (!options.skipGit) {
64
- spinner.start('Initializing git repository');
65
- run('git init', projectDir);
66
- run('git add -A', projectDir);
71
+ spinner.start("Initializing git repository");
72
+ run("git init", projectDir);
73
+ run("git add -A", projectDir);
67
74
  run('git commit -m "chore: scaffold QuanticJS project"', projectDir);
68
75
  spinner.succeed();
69
76
  }
70
- console.log('');
77
+ console.log("");
71
78
  console.log(` Project created at ${projectDir}`);
72
- console.log('');
73
- console.log(' Next steps:');
79
+ console.log("");
80
+ console.log(" Next steps:");
74
81
  console.log(` cd ${options.projectName}`);
75
- console.log(' docker compose up # Start infrastructure');
76
- console.log(' cd client && npm run dev # Start frontend');
77
- console.log('');
82
+ console.log(" docker compose up # Start infrastructure");
83
+ if (options.includeFrontend) {
84
+ console.log(" cd client && npm run dev # Start frontend");
85
+ }
86
+ console.log("");
78
87
  }
79
88
  //# sourceMappingURL=scaffold.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"scaffold.js","sourceRoot":"","sources":["../src/scaffold.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"scaffold.js","sourceRoot":"","sources":["../src/scaffold.ts"],"names":[],"mappings":"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"}
@@ -4,6 +4,7 @@ export interface TemplateContext {
4
4
  projectNameUnderscored: string;
5
5
  projectNamePascal: string;
6
6
  modules: string[];
7
+ includeFrontend: boolean;
7
8
  }
8
9
  export declare function renderTemplate(templatePath: string, ctx: TemplateContext): string;
9
10
  export declare function writeFile(filePath: string, content: string): void;
@@ -1,17 +1,17 @@
1
- import { readFileSync, writeFileSync, mkdirSync } from 'node:fs';
2
- import { dirname, join } from 'node:path';
3
- import { fileURLToPath } from 'node:url';
4
- import ejs from 'ejs';
1
+ import { readFileSync, writeFileSync, mkdirSync } from "node:fs";
2
+ import { dirname, join } from "node:path";
3
+ import { fileURLToPath } from "node:url";
4
+ import ejs from "ejs";
5
5
  const __dirname = dirname(fileURLToPath(import.meta.url));
6
- const TEMPLATES_DIR = join(__dirname, '..', '..', 'templates');
6
+ const TEMPLATES_DIR = join(__dirname, "..", "..", "templates");
7
7
  export function renderTemplate(templatePath, ctx) {
8
8
  const fullPath = join(TEMPLATES_DIR, templatePath);
9
- const content = readFileSync(fullPath, 'utf-8');
9
+ const content = readFileSync(fullPath, "utf-8");
10
10
  return ejs.render(content, ctx);
11
11
  }
12
12
  export function writeFile(filePath, content) {
13
13
  mkdirSync(dirname(filePath), { recursive: true });
14
- writeFileSync(filePath, content, 'utf-8');
14
+ writeFileSync(filePath, content, "utf-8");
15
15
  }
16
16
  export function renderAndWrite(templatePath, outputPath, ctx) {
17
17
  const content = renderTemplate(templatePath, ctx);
@@ -1 +1 @@
1
- {"version":3,"file":"template.js","sourceRoot":"","sources":["../../src/utils/template.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,SAAS,EAAE,MAAM,SAAS,CAAC;AACjE,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACzC,OAAO,GAAG,MAAM,KAAK,CAAC;AAEtB,MAAM,SAAS,GAAG,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;AAC1D,MAAM,aAAa,GAAG,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,WAAW,CAAC,CAAC;AAU/D,MAAM,UAAU,cAAc,CAAC,YAAoB,EAAE,GAAoB;IACvE,MAAM,QAAQ,GAAG,IAAI,CAAC,aAAa,EAAE,YAAY,CAAC,CAAC;IACnD,MAAM,OAAO,GAAG,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;IAChD,OAAO,GAAG,CAAC,MAAM,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;AAClC,CAAC;AAED,MAAM,UAAU,SAAS,CAAC,QAAgB,EAAE,OAAe;IACzD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAClD,aAAa,CAAC,QAAQ,EAAE,OAAO,EAAE,OAAO,CAAC,CAAC;AAC5C,CAAC;AAED,MAAM,UAAU,cAAc,CAC5B,YAAoB,EACpB,UAAkB,EAClB,GAAoB;IAEpB,MAAM,OAAO,GAAG,cAAc,CAAC,YAAY,EAAE,GAAG,CAAC,CAAC;IAClD,SAAS,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;AACjC,CAAC"}
1
+ {"version":3,"file":"template.js","sourceRoot":"","sources":["../../src/utils/template.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,SAAS,EAAE,MAAM,SAAS,CAAC;AACjE,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACzC,OAAO,GAAG,MAAM,KAAK,CAAC;AAEtB,MAAM,SAAS,GAAG,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;AAC1D,MAAM,aAAa,GAAG,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,WAAW,CAAC,CAAC;AAW/D,MAAM,UAAU,cAAc,CAC5B,YAAoB,EACpB,GAAoB;IAEpB,MAAM,QAAQ,GAAG,IAAI,CAAC,aAAa,EAAE,YAAY,CAAC,CAAC;IACnD,MAAM,OAAO,GAAG,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;IAChD,OAAO,GAAG,CAAC,MAAM,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;AAClC,CAAC;AAED,MAAM,UAAU,SAAS,CAAC,QAAgB,EAAE,OAAe;IACzD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAClD,aAAa,CAAC,QAAQ,EAAE,OAAO,EAAE,OAAO,CAAC,CAAC;AAC5C,CAAC;AAED,MAAM,UAAU,cAAc,CAC5B,YAAoB,EACpB,UAAkB,EAClB,GAAoB;IAEpB,MAAM,OAAO,GAAG,cAAc,CAAC,YAAY,EAAE,GAAG,CAAC,CAAC;IAClD,SAAS,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;AACjC,CAAC"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@quanticjs/create-app",
3
- "version": "0.1.1",
3
+ "version": "0.1.2",
4
4
  "description": "Scaffold a QuanticJS project — NestJS modular monolith + React frontend",
5
5
  "bin": {
6
6
  "create-quanticjs-app": "./dist/index.js"
@@ -4,7 +4,7 @@ import { TypeOrmModule } from '@nestjs/typeorm';
4
4
  import { CqrsModule } from '@nestjs/cqrs';
5
5
  import { ScheduleModule } from '@nestjs/schedule';
6
6
  import { LoggerModule } from 'nestjs-pino';
7
- import { QuanticModule } from '@nestjs-cqrs/quanticjs';
7
+ import { QuanticModule } from '@quanticjs/quanticjs';
8
8
  import { QuanticHealthModule } from '@quanticjs/core';
9
9
  import { BffModule } from './bff/bff.module';
10
10
  <% modules.forEach(function(mod) { -%>
@@ -1,7 +1,7 @@
1
1
  import { Controller, Get, Post, Req, Res, Query } from '@nestjs/common';
2
2
  import { ApiTags, ApiOperation } from '@nestjs/swagger';
3
3
  import { Request, Response } from 'express';
4
- import { Public } from '@nestjs-cqrs/quanticjs';
4
+ import { Public } from '@quanticjs/quanticjs';
5
5
  import { BffService } from './bff.service';
6
6
 
7
7
  @ApiTags('auth')
@@ -3,7 +3,7 @@ import { ValidationPipe } from '@nestjs/common';
3
3
  import { SwaggerModule, DocumentBuilder } from '@nestjs/swagger';
4
4
  import { Logger } from 'nestjs-pino';
5
5
  import * as cookieParser from 'cookie-parser';
6
- import { GlobalExceptionFilter, ResultInterceptor } from '@nestjs-cqrs/quanticjs';
6
+ import { GlobalExceptionFilter, ResultInterceptor } from '@quanticjs/quanticjs';
7
7
  import { AppModule } from './app.module';
8
8
 
9
9
  async function bootstrap() {
@@ -3,12 +3,18 @@
3
3
  ## Stack
4
4
 
5
5
  - **Backend:** NestJS (modular monolith), CQRS with pipeline behaviors, TypeORM, PostgreSQL, Redis, Keycloak BFF auth
6
+ <% if (includeFrontend) { -%>
6
7
  - **Frontend:** React, Vite, TanStack Query, Zustand, React Hook Form + Zod, Tailwind CSS, shadcn/ui
8
+ <% } -%>
7
9
  - **Infrastructure:** Docker Compose (local dev), Kubernetes + Helm + ArgoCD (production)
8
10
 
9
11
  ## Architecture
10
12
 
13
+ <% if (includeFrontend) { -%>
11
14
  **Modular monolith.** One backend Docker image, one frontend Docker image. Modules communicate via CommandBus/QueryBus. Each module owns its own PostgreSQL schema.
15
+ <% } else { -%>
16
+ **Modular monolith.** One backend Docker image. Modules communicate via CommandBus/QueryBus. Each module owns its own PostgreSQL schema.
17
+ <% } -%>
12
18
 
13
19
  **CQRS.** Every operation is a Command/Query class + Handler. Controllers are thin — they only parse requests and dispatch to the bus.
14
20
 
@@ -19,15 +25,20 @@
19
25
  ```bash
20
26
  # Start backend + infrastructure
21
27
  docker compose up
22
-
28
+ <% if (includeFrontend) { %>
23
29
  # Start frontend (separate terminal)
24
30
  cd client && npm run dev
25
31
 
26
32
  # Browser
27
33
  open http://localhost:5173
34
+ <% } %>
28
35
  ```
29
36
 
37
+ <% if (includeFrontend) { -%>
30
38
  **Ports:** 5173 (frontend/Vite), 3000 (backend API), 8080 (Keycloak)
39
+ <% } else { -%>
40
+ **Ports:** 3000 (backend API), 8080 (Keycloak)
41
+ <% } -%>
31
42
 
32
43
  ## Key Conventions
33
44
 
@@ -50,16 +61,20 @@ Use the appropriate skill for each task. Skills enforce architectural rules auto
50
61
  | Wire a handler to an HTTP endpoint | `/add-api-endpoint` |
51
62
  | Add an authenticated/role-gated endpoint | `/add-auth-endpoint` |
52
63
  | Create a new bounded context module | `/add-module` |
64
+ <% if (includeFrontend) { -%>
53
65
  | Add a React page with data fetching | `/add-frontend-page` |
66
+ <% } -%>
54
67
  | Integrate an external API (AI, payment, etc.) | `/add-integration` |
55
68
  | Add inter-module async events | `/add-event` |
56
69
  | Add live WebSocket updates | `/add-realtime` |
57
70
  | Write backend tests | `/write-backend-tests` |
71
+ <% if (includeFrontend) { -%>
58
72
  | Write Playwright E2E tests | `/write-ui-tests` |
59
73
  | Find missing E2E coverage | `/e2e-scan` |
60
74
  | Walk through journeys via MCP browser | `/e2e-verify` |
61
75
  | Audit specs against real pages via MCP | `/e2e-audit` |
62
76
  | Run full E2E suite (scan + audit + verify) | `/e2e-full` |
77
+ <% } -%>
63
78
  | Run the test suite | `/run-tests` |
64
79
  | Fix a bug (TDD workflow) | `/fix-bug` |
65
80
  | Review code before merge | `/review-code` |
@@ -77,10 +92,11 @@ Use the appropriate skill for each task. Skills enforce architectural rules auto
77
92
  ```bash
78
93
  # Backend unit + integration tests
79
94
  npm test
80
-
95
+ <% if (includeFrontend) { %>
81
96
  # Frontend tests
82
97
  cd client && npm test
83
98
 
84
99
  # E2E tests
85
100
  cd client && npx playwright test
101
+ <% } %>
86
102
  ```