@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 +8 -1
- package/dist/deps.js +112 -89
- package/dist/deps.js.map +1 -1
- package/dist/generators/backend.d.ts +2 -2
- package/dist/generators/backend.js +15 -15
- package/dist/generators/backend.js.map +1 -1
- package/dist/generators/docker.d.ts +1 -1
- package/dist/generators/docker.js +9 -7
- package/dist/generators/docker.js.map +1 -1
- package/dist/index.js +28 -14
- package/dist/index.js.map +1 -1
- package/dist/prompts.d.ts +2 -0
- package/dist/prompts.js +39 -16
- package/dist/prompts.js.map +1 -1
- package/dist/scaffold.d.ts +1 -1
- package/dist/scaffold.js +51 -42
- package/dist/scaffold.js.map +1 -1
- package/dist/utils/template.d.ts +1 -0
- package/dist/utils/template.js +7 -7
- package/dist/utils/template.js.map +1 -1
- package/package.json +1 -1
- package/templates/backend/app.module.ts.ejs +1 -1
- package/templates/backend/bff.controller.ts.ejs +1 -1
- package/templates/backend/main.ts.ejs +1 -1
- package/templates/claude/CLAUDE.md.ejs +18 -2
package/dist/deps.d.ts
CHANGED
|
@@ -1,4 +1,11 @@
|
|
|
1
|
-
export
|
|
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
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
typeorm:
|
|
14
|
-
pg:
|
|
15
|
-
ioredis:
|
|
16
|
-
|
|
17
|
-
passport:
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
zod:
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
rxjs:
|
|
29
|
-
|
|
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
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
typescript:
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
jest:
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
supertest:
|
|
47
|
-
|
|
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:
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
zustand:
|
|
55
|
-
zod:
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
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:
|
|
82
|
-
|
|
83
|
-
typescript:
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
tailwindcss:
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
vitest:
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
jsdom:
|
|
94
|
-
|
|
95
|
-
|
|
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":"
|
|
1
|
+
{"version":3,"file":"deps.js","sourceRoot":"","sources":["../src/deps.ts"],"names":[],"mappings":"AAMA,MAAM,CAAC,MAAM,iBAAiB,GAA2B;IACvD,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,SAAS;IAC5B,kBAAkB,EAAE,QAAQ;IAC5B,sBAAsB,EAAE,QAAQ;IAChC,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,yBAAyB,GAAsB;IAC1D,EAAE,IAAI,EAAE,qBAAqB,EAAE,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,iBAAiB,EAAE;IAC5E,EAAE,IAAI,EAAE,mBAAmB,EAAE,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,gBAAgB,EAAE;IACzE,EAAE,IAAI,EAAE,kBAAkB,EAAE,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,iBAAiB,EAAE;IACzE;QACE,IAAI,EAAE,oBAAoB;QAC1B,OAAO,EAAE,QAAQ;QACjB,KAAK,EAAE,oBAAoB;KAC5B;IACD;QACE,IAAI,EAAE,0BAA0B;QAChC,OAAO,EAAE,QAAQ;QACjB,KAAK,EAAE,yBAAyB;KACjC;CACF,CAAC;AAEF,MAAM,UAAU,gBAAgB,CAC9B,gBAA0B;IAE1B,MAAM,IAAI,GAAG,EAAE,GAAG,iBAAiB,EAAE,CAAC;IACtC,KAAK,MAAM,GAAG,IAAI,yBAAyB,EAAE,CAAC;QAC5C,IAAI,gBAAgB,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC;YACxC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,GAAG,CAAC,OAAO,CAAC;QAC/B,CAAC;IACH,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED,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,+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,2 +1,2 @@
|
|
|
1
|
-
import { type TemplateContext } from
|
|
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
|
|
2
|
-
import { renderAndWrite, renderTemplate, writeFile } from
|
|
3
|
-
import { toPascalCase } from
|
|
4
|
-
import {
|
|
5
|
-
export function generateBackend(projectDir, ctx) {
|
|
6
|
-
const rendered = renderTemplate(
|
|
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 =
|
|
8
|
+
pkgJson.dependencies = buildBackendDeps(selectedPackages);
|
|
9
9
|
pkgJson.devDependencies = BACKEND_DEV_DEPS;
|
|
10
|
-
writeFile(join(projectDir,
|
|
11
|
-
renderAndWrite(
|
|
12
|
-
renderAndWrite(
|
|
13
|
-
renderAndWrite(
|
|
14
|
-
renderAndWrite(
|
|
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(
|
|
17
|
-
renderAndWrite(
|
|
18
|
-
renderAndWrite(
|
|
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;
|
|
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
|
|
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
|
|
2
|
-
import { renderAndWrite } from
|
|
1
|
+
import { join } from "node:path";
|
|
2
|
+
import { renderAndWrite } from "../utils/template.js";
|
|
3
3
|
export function generateDocker(projectDir, ctx) {
|
|
4
|
-
renderAndWrite(
|
|
5
|
-
renderAndWrite(
|
|
6
|
-
renderAndWrite(
|
|
7
|
-
|
|
8
|
-
|
|
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,
|
|
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
|
|
3
|
-
import { promptForOptions } from
|
|
4
|
-
import { scaffold } from
|
|
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(
|
|
8
|
-
.description(
|
|
9
|
-
.version(
|
|
10
|
-
.argument(
|
|
11
|
-
.option(
|
|
12
|
-
.option(
|
|
13
|
-
.option(
|
|
14
|
-
.option(
|
|
15
|
-
.option(
|
|
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
|
|
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(
|
|
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,
|
|
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
package/dist/prompts.js
CHANGED
|
@@ -1,31 +1,32 @@
|
|
|
1
|
-
import inquirer from
|
|
2
|
-
import { validateProjectName, validateModuleName } from
|
|
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:
|
|
7
|
-
name:
|
|
8
|
-
message:
|
|
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:
|
|
15
|
-
name:
|
|
16
|
-
message:
|
|
17
|
-
default:
|
|
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:
|
|
22
|
-
name:
|
|
23
|
-
message:
|
|
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
|
|
28
|
-
const mods = input.split(
|
|
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 ??
|
|
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
|
};
|
package/dist/prompts.js.map
CHANGED
|
@@ -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;
|
|
1
|
+
{"version":3,"file":"prompts.js","sourceRoot":"","sources":["../src/prompts.ts"],"names":[],"mappings":"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"}
|
package/dist/scaffold.d.ts
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import type { ProjectOptions } from
|
|
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
|
|
2
|
-
import { existsSync } from
|
|
3
|
-
import ora from
|
|
4
|
-
import { toUnderscored, toPascalCase } from
|
|
5
|
-
import { run } from
|
|
6
|
-
import { generateRoot } from
|
|
7
|
-
import { generateBackend } from
|
|
8
|
-
import { generateBff } from
|
|
9
|
-
import { generateModules } from
|
|
10
|
-
import { generateFrontend } from
|
|
11
|
-
import { generateDocker } from
|
|
12
|
-
import { generateScripts } from
|
|
13
|
-
import { generateE2e } from
|
|
14
|
-
import { generateClaude } from
|
|
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(
|
|
29
|
+
spinner.start("Generating root files");
|
|
29
30
|
generateRoot(projectDir, ctx);
|
|
30
31
|
spinner.succeed();
|
|
31
|
-
spinner.start(
|
|
32
|
-
generateBackend(projectDir, ctx);
|
|
32
|
+
spinner.start("Generating backend");
|
|
33
|
+
generateBackend(projectDir, ctx, options.backendPackages);
|
|
33
34
|
spinner.succeed();
|
|
34
|
-
spinner.start(
|
|
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
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
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(
|
|
49
|
+
spinner.start("Generating scripts");
|
|
47
50
|
generateScripts(projectDir, ctx);
|
|
48
51
|
spinner.succeed();
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
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(
|
|
57
|
-
run(
|
|
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(
|
|
65
|
-
run(
|
|
66
|
-
run(
|
|
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(
|
|
79
|
+
console.log("");
|
|
80
|
+
console.log(" Next steps:");
|
|
74
81
|
console.log(` cd ${options.projectName}`);
|
|
75
|
-
console.log(
|
|
76
|
-
|
|
77
|
-
|
|
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
|
package/dist/scaffold.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"scaffold.js","sourceRoot":"","sources":["../src/scaffold.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACrC,OAAO,GAAG,MAAM,KAAK,CAAC;AACtB,OAAO,EAAE,aAAa,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAClE,OAAO,EAAE,GAAG,EAAU,MAAM,iBAAiB,CAAC;AAC9C,OAAO,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAC;AACpD,OAAO,EAAE,eAAe,EAAE,MAAM,yBAAyB,CAAC;AAC1D,OAAO,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAC;AAClD,OAAO,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AACzD,OAAO,EAAE,gBAAgB,EAAE,MAAM,0BAA0B,CAAC;AAC5D,OAAO,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AACxD,OAAO,EAAE,eAAe,EAAE,MAAM,yBAAyB,CAAC;AAC1D,OAAO,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAC;AAClD,OAAO,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AAIxD,MAAM,CAAC,KAAK,UAAU,QAAQ,
|
|
1
|
+
{"version":3,"file":"scaffold.js","sourceRoot":"","sources":["../src/scaffold.ts"],"names":[],"mappings":"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"}
|
package/dist/utils/template.d.ts
CHANGED
|
@@ -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;
|
package/dist/utils/template.js
CHANGED
|
@@ -1,17 +1,17 @@
|
|
|
1
|
-
import { readFileSync, writeFileSync, mkdirSync } from
|
|
2
|
-
import { dirname, join } from
|
|
3
|
-
import { fileURLToPath } from
|
|
4
|
-
import ejs from
|
|
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,
|
|
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,
|
|
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,
|
|
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;
|
|
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
|
@@ -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 '@
|
|
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 '@
|
|
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 '@
|
|
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
|
```
|