hatchkit 0.1.1

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 (149) hide show
  1. package/dist/config.d.ts +131 -0
  2. package/dist/config.d.ts.map +1 -0
  3. package/dist/config.js +629 -0
  4. package/dist/config.js.map +1 -0
  5. package/dist/deploy/coolify.d.ts +4 -0
  6. package/dist/deploy/coolify.d.ts.map +1 -0
  7. package/dist/deploy/coolify.js +20 -0
  8. package/dist/deploy/coolify.js.map +1 -0
  9. package/dist/deploy/github.d.ts +4 -0
  10. package/dist/deploy/github.d.ts.map +1 -0
  11. package/dist/deploy/github.js +39 -0
  12. package/dist/deploy/github.js.map +1 -0
  13. package/dist/deploy/gpu.d.ts +4 -0
  14. package/dist/deploy/gpu.d.ts.map +1 -0
  15. package/dist/deploy/gpu.js +97 -0
  16. package/dist/deploy/gpu.js.map +1 -0
  17. package/dist/deploy/keys.d.ts +9 -0
  18. package/dist/deploy/keys.d.ts.map +1 -0
  19. package/dist/deploy/keys.js +73 -0
  20. package/dist/deploy/keys.js.map +1 -0
  21. package/dist/deploy/terraform.d.ts +4 -0
  22. package/dist/deploy/terraform.d.ts.map +1 -0
  23. package/dist/deploy/terraform.js +55 -0
  24. package/dist/deploy/terraform.js.map +1 -0
  25. package/dist/index.d.ts +3 -0
  26. package/dist/index.d.ts.map +1 -0
  27. package/dist/index.js +599 -0
  28. package/dist/index.js.map +1 -0
  29. package/dist/prompts.d.ts +52 -0
  30. package/dist/prompts.d.ts.map +1 -0
  31. package/dist/prompts.js +313 -0
  32. package/dist/prompts.js.map +1 -0
  33. package/dist/provision/glitchtip.d.ts +6 -0
  34. package/dist/provision/glitchtip.d.ts.map +1 -0
  35. package/dist/provision/glitchtip.js +46 -0
  36. package/dist/provision/glitchtip.js.map +1 -0
  37. package/dist/provision/index.d.ts +9 -0
  38. package/dist/provision/index.d.ts.map +1 -0
  39. package/dist/provision/index.js +108 -0
  40. package/dist/provision/index.js.map +1 -0
  41. package/dist/provision/openpanel.d.ts +8 -0
  42. package/dist/provision/openpanel.d.ts.map +1 -0
  43. package/dist/provision/openpanel.js +66 -0
  44. package/dist/provision/openpanel.js.map +1 -0
  45. package/dist/provision/resend.d.ts +16 -0
  46. package/dist/provision/resend.d.ts.map +1 -0
  47. package/dist/provision/resend.js +43 -0
  48. package/dist/provision/resend.js.map +1 -0
  49. package/dist/scaffold/app.d.ts +13 -0
  50. package/dist/scaffold/app.d.ts.map +1 -0
  51. package/dist/scaffold/app.js +340 -0
  52. package/dist/scaffold/app.js.map +1 -0
  53. package/dist/scaffold/dotenvx.d.ts +30 -0
  54. package/dist/scaffold/dotenvx.d.ts.map +1 -0
  55. package/dist/scaffold/dotenvx.js +142 -0
  56. package/dist/scaffold/dotenvx.js.map +1 -0
  57. package/dist/scaffold/infra.d.ts +17 -0
  58. package/dist/scaffold/infra.d.ts.map +1 -0
  59. package/dist/scaffold/infra.js +200 -0
  60. package/dist/scaffold/infra.js.map +1 -0
  61. package/dist/scaffold/manifest.d.ts +50 -0
  62. package/dist/scaffold/manifest.d.ts.map +1 -0
  63. package/dist/scaffold/manifest.js +83 -0
  64. package/dist/scaffold/manifest.js.map +1 -0
  65. package/dist/scaffold/ml-client.d.ts +20 -0
  66. package/dist/scaffold/ml-client.d.ts.map +1 -0
  67. package/dist/scaffold/ml-client.js +38 -0
  68. package/dist/scaffold/ml-client.js.map +1 -0
  69. package/dist/scaffold/pkg-json.d.ts +20 -0
  70. package/dist/scaffold/pkg-json.d.ts.map +1 -0
  71. package/dist/scaffold/pkg-json.js +113 -0
  72. package/dist/scaffold/pkg-json.js.map +1 -0
  73. package/dist/scaffold/starter-files.d.ts +40 -0
  74. package/dist/scaffold/starter-files.d.ts.map +1 -0
  75. package/dist/scaffold/starter-files.js +197 -0
  76. package/dist/scaffold/starter-files.js.map +1 -0
  77. package/dist/scaffold/update.d.ts +8 -0
  78. package/dist/scaffold/update.d.ts.map +1 -0
  79. package/dist/scaffold/update.js +255 -0
  80. package/dist/scaffold/update.js.map +1 -0
  81. package/dist/templates/addons/analytics/middleware.ts.hbs +13 -0
  82. package/dist/templates/addons/analytics/sentry.ts.hbs +16 -0
  83. package/dist/templates/addons/storage/s3.ts.hbs +40 -0
  84. package/dist/templates/addons/storage/upload.ts.hbs +23 -0
  85. package/dist/templates/addons/stripe/checkout.ts.hbs +27 -0
  86. package/dist/templates/addons/stripe/client.ts.hbs +6 -0
  87. package/dist/templates/addons/stripe/webhook.ts.hbs +39 -0
  88. package/dist/templates/addons/websocket/redis.ts.hbs +25 -0
  89. package/dist/templates/addons/websocket/ws.ts.hbs +32 -0
  90. package/dist/templates/base/.dockerignore.hbs +7 -0
  91. package/dist/templates/base/Dockerfile.hbs +18 -0
  92. package/dist/templates/base/env.example.hbs +60 -0
  93. package/dist/templates/base/github-actions.yml.hbs +35 -0
  94. package/dist/templates/base/gitignore.hbs +5 -0
  95. package/dist/templates/base/package.json.hbs +36 -0
  96. package/dist/templates/base/src/auth/auth.ts.hbs +13 -0
  97. package/dist/templates/base/src/auth/routes.ts.hbs +19 -0
  98. package/dist/templates/base/src/config.ts.hbs +36 -0
  99. package/dist/templates/base/src/db.ts.hbs +12 -0
  100. package/dist/templates/base/src/index.ts.hbs +80 -0
  101. package/dist/templates/base/src/routes/health.ts.hbs +12 -0
  102. package/dist/templates/base/tsconfig.json.hbs +18 -0
  103. package/dist/templates/ml-clients/3d-extraction.ts.hbs +20 -0
  104. package/dist/templates/ml-clients/background-removal.ts.hbs +17 -0
  105. package/dist/templates/ml-clients/custom-hf.ts.hbs +20 -0
  106. package/dist/templates/ml-clients/image-recognition.ts.hbs +22 -0
  107. package/dist/templates/ml-clients/subtitles.ts.hbs +26 -0
  108. package/dist/utils/coolify-api.d.ts +45 -0
  109. package/dist/utils/coolify-api.d.ts.map +1 -0
  110. package/dist/utils/coolify-api.js +72 -0
  111. package/dist/utils/coolify-api.js.map +1 -0
  112. package/dist/utils/errors.d.ts +2 -0
  113. package/dist/utils/errors.d.ts.map +1 -0
  114. package/dist/utils/errors.js +31 -0
  115. package/dist/utils/errors.js.map +1 -0
  116. package/dist/utils/exec.d.ts +23 -0
  117. package/dist/utils/exec.d.ts.map +1 -0
  118. package/dist/utils/exec.js +47 -0
  119. package/dist/utils/exec.js.map +1 -0
  120. package/dist/utils/flags.d.ts +17 -0
  121. package/dist/utils/flags.d.ts.map +1 -0
  122. package/dist/utils/flags.js +116 -0
  123. package/dist/utils/flags.js.map +1 -0
  124. package/dist/utils/hf-api.d.ts +13 -0
  125. package/dist/utils/hf-api.d.ts.map +1 -0
  126. package/dist/utils/hf-api.js +30 -0
  127. package/dist/utils/hf-api.js.map +1 -0
  128. package/dist/utils/ports.d.ts +29 -0
  129. package/dist/utils/ports.d.ts.map +1 -0
  130. package/dist/utils/ports.js +86 -0
  131. package/dist/utils/ports.js.map +1 -0
  132. package/dist/utils/secrets.d.ts +25 -0
  133. package/dist/utils/secrets.d.ts.map +1 -0
  134. package/dist/utils/secrets.js +51 -0
  135. package/dist/utils/secrets.js.map +1 -0
  136. package/dist/utils/template.d.ts +7 -0
  137. package/dist/utils/template.d.ts.map +1 -0
  138. package/dist/utils/template.js +35 -0
  139. package/dist/utils/template.js.map +1 -0
  140. package/dist/utils/validate.d.ts +16 -0
  141. package/dist/utils/validate.d.ts.map +1 -0
  142. package/dist/utils/validate.js +60 -0
  143. package/dist/utils/validate.js.map +1 -0
  144. package/dist/utils/version.d.ts +2 -0
  145. package/dist/utils/version.d.ts.map +1 -0
  146. package/dist/utils/version.js +21 -0
  147. package/dist/utils/version.js.map +1 -0
  148. package/package.json +48 -0
  149. package/scripts/copy-templates.mjs +20 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"update.js","sourceRoot":"","sources":["../../src/scaffold/update.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,13 @@
1
+ import * as Sentry from "@sentry/node";
2
+ import type { Request, Response, NextFunction } from "express";
3
+
4
+ export function errorTrackingMiddleware(
5
+ err: Error,
6
+ _req: Request,
7
+ res: Response,
8
+ _next: NextFunction,
9
+ ): void {
10
+ Sentry.captureException(err);
11
+ console.error(err);
12
+ res.status(500).json({ error: "Internal server error" });
13
+ }
@@ -0,0 +1,16 @@
1
+ import * as Sentry from "@sentry/node";
2
+ import type { Express } from "express";
3
+ import { config } from "../config.js";
4
+
5
+ export function initSentry(app: Express): void {
6
+ if (!config.GLITCHTIP_DSN) return;
7
+
8
+ Sentry.init({
9
+ dsn: config.GLITCHTIP_DSN,
10
+ integrations: [
11
+ Sentry.httpIntegration(),
12
+ Sentry.expressIntegration({ app }),
13
+ ],
14
+ tracesSampleRate: 0.1,
15
+ });
16
+ }
@@ -0,0 +1,40 @@
1
+ import { S3Client, PutObjectCommand, GetObjectCommand } from "@aws-sdk/client-s3";
2
+ import { getSignedUrl } from "@aws-sdk/s3-request-presigner";
3
+ import { config } from "../config.js";
4
+
5
+ export const s3 = new S3Client({
6
+ endpoint: config.S3_ENDPOINT,
7
+ region: config.S3_REGION,
8
+ credentials: {
9
+ accessKeyId: config.S3_ACCESS_KEY,
10
+ secretAccessKey: config.S3_SECRET_KEY,
11
+ },
12
+ forcePathStyle: true,
13
+ });
14
+
15
+ export async function uploadToS3(
16
+ key: string,
17
+ body: Buffer,
18
+ contentType: string,
19
+ ): Promise<string> {
20
+ await s3.send(
21
+ new PutObjectCommand({
22
+ Bucket: config.S3_BUCKET,
23
+ Key: key,
24
+ Body: body,
25
+ ContentType: contentType,
26
+ }),
27
+ );
28
+ return key;
29
+ }
30
+
31
+ export async function getPresignedUrl(
32
+ key: string,
33
+ expiresIn = 3600,
34
+ ): Promise<string> {
35
+ return getSignedUrl(
36
+ s3,
37
+ new GetObjectCommand({ Bucket: config.S3_BUCKET, Key: key }),
38
+ { expiresIn },
39
+ );
40
+ }
@@ -0,0 +1,23 @@
1
+ import { Router } from "express";
2
+ import multer from "multer";
3
+ import { randomUUID } from "node:crypto";
4
+ import { uploadToS3, getPresignedUrl } from "./s3.js";
5
+
6
+ const upload = multer({ limits: { fileSize: 50 * 1024 * 1024 } }); // 50 MB
7
+
8
+ export const uploadRouter = Router();
9
+
10
+ uploadRouter.post("/", upload.single("file"), async (req, res) => {
11
+ if (!req.file) {
12
+ res.status(400).json({ error: "No file uploaded" });
13
+ return;
14
+ }
15
+
16
+ const ext = req.file.originalname.split(".").pop();
17
+ const key = `uploads/${randomUUID()}.${ext}`;
18
+
19
+ await uploadToS3(key, req.file.buffer, req.file.mimetype);
20
+ const url = await getPresignedUrl(key);
21
+
22
+ res.json({ key, url });
23
+ });
@@ -0,0 +1,27 @@
1
+ import { Router } from "express";
2
+ import { stripe } from "./client.js";
3
+ import { config } from "../config.js";
4
+
5
+ export const checkoutRouter = Router();
6
+
7
+ checkoutRouter.post("/create-checkout-session", async (req, res) => {
8
+ try {
9
+ const session = await stripe.checkout.sessions.create({
10
+ mode: "subscription",
11
+ payment_method_types: ["card"],
12
+ line_items: [
13
+ {
14
+ price: req.body.priceId,
15
+ quantity: 1,
16
+ },
17
+ ],
18
+ success_url: `${config.FRONTEND_URL}/success?session_id={CHECKOUT_SESSION_ID}`,
19
+ cancel_url: `${config.FRONTEND_URL}/cancel`,
20
+ });
21
+
22
+ res.json({ url: session.url });
23
+ } catch (error) {
24
+ console.error("Checkout error:", error);
25
+ res.status(500).json({ error: "Failed to create checkout session" });
26
+ }
27
+ });
@@ -0,0 +1,6 @@
1
+ import Stripe from "stripe";
2
+ import { config } from "../config.js";
3
+
4
+ export const stripe = new Stripe(config.STRIPE_SECRET_KEY, {
5
+ apiVersion: "2024-12-18.acacia",
6
+ });
@@ -0,0 +1,39 @@
1
+ import { Router, raw } from "express";
2
+ import { stripe } from "./client.js";
3
+ import { config } from "../config.js";
4
+
5
+ export const stripeWebhookRouter = Router();
6
+
7
+ stripeWebhookRouter.post("/", raw({ type: "application/json" }), async (req, res) => {
8
+ const signature = req.headers["stripe-signature"];
9
+ if (!signature) {
10
+ res.status(400).json({ error: "Missing stripe-signature header" });
11
+ return;
12
+ }
13
+
14
+ let event;
15
+ try {
16
+ event = stripe.webhooks.constructEvent(req.body, signature, config.STRIPE_WEBHOOK_SECRET);
17
+ } catch (err) {
18
+ console.error("Webhook signature verification failed:", err);
19
+ res.status(400).json({ error: "Invalid signature" });
20
+ return;
21
+ }
22
+
23
+ switch (event.type) {
24
+ case "checkout.session.completed":
25
+ console.log("Checkout session completed:", event.data.object.id);
26
+ // TODO: Fulfill the order
27
+ break;
28
+ case "customer.subscription.updated":
29
+ console.log("Subscription updated:", event.data.object.id);
30
+ break;
31
+ case "customer.subscription.deleted":
32
+ console.log("Subscription canceled:", event.data.object.id);
33
+ break;
34
+ default:
35
+ console.log(`Unhandled event type: ${event.type}`);
36
+ }
37
+
38
+ res.json({ received: true });
39
+ });
@@ -0,0 +1,25 @@
1
+ import Redis from "ioredis";
2
+ import { config } from "../config.js";
3
+
4
+ export const redis = new Redis(config.REDIS_URL);
5
+
6
+ // Separate connection for pub/sub (Redis requires dedicated connections for subscribers)
7
+ const subscriber = new Redis(config.REDIS_URL);
8
+ const publisher = new Redis(config.REDIS_URL);
9
+
10
+ const handlers = new Map<string, (message: string) => void>();
11
+
12
+ subscriber.on("message", (channel, message) => {
13
+ const handler = handlers.get(channel);
14
+ if (handler) handler(message);
15
+ });
16
+
17
+ export const pubsub = {
18
+ publish(channel: string, message: string) {
19
+ publisher.publish(channel, message);
20
+ },
21
+ subscribe(channel: string, handler: (message: string) => void) {
22
+ handlers.set(channel, handler);
23
+ subscriber.subscribe(channel);
24
+ },
25
+ };
@@ -0,0 +1,32 @@
1
+ import { WebSocketServer, type WebSocket } from "ws";
2
+ import type { Server } from "node:http";
3
+ import { pubsub } from "./redis.js";
4
+
5
+ export function setupWebSocket(server: Server): WebSocketServer {
6
+ const wss = new WebSocketServer({ server, path: "/api/ws" });
7
+
8
+ wss.on("connection", (ws: WebSocket) => {
9
+ console.log("WebSocket client connected");
10
+
11
+ ws.on("message", (data) => {
12
+ const message = data.toString();
13
+ // Publish to Redis for multi-replica fanout
14
+ pubsub.publish("ws:broadcast", message);
15
+ });
16
+
17
+ ws.on("close", () => {
18
+ console.log("WebSocket client disconnected");
19
+ });
20
+ });
21
+
22
+ // Subscribe to Redis for messages from other replicas
23
+ pubsub.subscribe("ws:broadcast", (message) => {
24
+ for (const client of wss.clients) {
25
+ if (client.readyState === 1) {
26
+ client.send(message);
27
+ }
28
+ }
29
+ });
30
+
31
+ return wss;
32
+ }
@@ -0,0 +1,7 @@
1
+ node_modules
2
+ dist
3
+ .git
4
+ .gitignore
5
+ .env
6
+ .env.*
7
+ *.md
@@ -0,0 +1,18 @@
1
+ FROM node:20-bookworm-slim AS deps
2
+ WORKDIR /app
3
+ COPY package*.json ./
4
+ RUN npm ci
5
+
6
+ FROM deps AS build
7
+ COPY . .
8
+ RUN npm run build
9
+
10
+ FROM node:20-bookworm-slim AS runtime
11
+ WORKDIR /app
12
+ ENV NODE_ENV=production
13
+ ENV PORT=3000
14
+ COPY package*.json ./
15
+ RUN npm ci --omit=dev
16
+ COPY --from=build /app/dist ./dist
17
+ EXPOSE 3000
18
+ CMD ["node", "dist/index.js"]
@@ -0,0 +1,60 @@
1
+ NODE_ENV=production
2
+ PORT=3000
3
+ FRONTEND_URL=https://{{domain}}
4
+
5
+ # MongoDB
6
+ MONGODB_URI=mongodb://localhost:27017/{{name}}
7
+
8
+ # Auth (better-auth)
9
+ AUTH_SECRET=change-me-to-a-random-string
10
+ AUTH_URL=https://{{domain}}
11
+ {{#if websocket}}
12
+
13
+ # Redis
14
+ REDIS_URL=redis://localhost:6379
15
+ {{/if}}
16
+ {{#if stripe}}
17
+
18
+ # Stripe
19
+ STRIPE_SECRET_KEY=sk_test_...
20
+ STRIPE_PUBLISHABLE_KEY=pk_test_...
21
+ STRIPE_WEBHOOK_SECRET=whsec_...
22
+ {{/if}}
23
+ {{#if analytics}}
24
+
25
+ # Error tracking
26
+ GLITCHTIP_DSN=
27
+ # Analytics (OpenPanel)
28
+ OPENPANEL_API_URL=
29
+ OPENPANEL_CLIENT_ID=
30
+ OPENPANEL_CLIENT_SECRET=
31
+ {{/if}}
32
+ {{#if s3}}
33
+
34
+ # S3 Storage
35
+ S3_ENDPOINT=
36
+ S3_BUCKET=
37
+ S3_ACCESS_KEY=
38
+ S3_SECRET_KEY=
39
+ S3_REGION=
40
+ {{/if}}
41
+ {{#if has3d}}
42
+
43
+ # ML: 3D Model Extraction
44
+ ML_3D_EXTRACTION_ENDPOINT=
45
+ {{/if}}
46
+ {{#if hasSubtitles}}
47
+
48
+ # ML: Subtitle Generation
49
+ ML_SUBTITLES_ENDPOINT=
50
+ {{/if}}
51
+ {{#if hasImageRecognition}}
52
+
53
+ # ML: Image Recognition
54
+ ML_IMAGE_RECOGNITION_ENDPOINT=
55
+ {{/if}}
56
+ {{#if hasBgRemoval}}
57
+
58
+ # ML: Background Removal
59
+ ML_BACKGROUND_REMOVAL_ENDPOINT=
60
+ {{/if}}
@@ -0,0 +1,35 @@
1
+ name: Deploy
2
+
3
+ on:
4
+ push:
5
+ branches: [main]
6
+
7
+ jobs:
8
+ deploy:
9
+ runs-on: ubuntu-latest
10
+ permissions:
11
+ contents: read
12
+ packages: write
13
+
14
+ steps:
15
+ - uses: actions/checkout@v4
16
+
17
+ - uses: docker/login-action@v3
18
+ with:
19
+ registry: ghcr.io
20
+ username: $\{{ github.actor }}
21
+ password: $\{{ secrets.GITHUB_TOKEN }}
22
+
23
+ - uses: docker/build-push-action@v5
24
+ with:
25
+ context: .
26
+ push: true
27
+ tags: |
28
+ ghcr.io/$\{{ github.repository }}:latest
29
+ ghcr.io/$\{{ github.repository }}:$\{{ github.sha }}
30
+
31
+ - name: Trigger Coolify deploy
32
+ run: |
33
+ curl -f -X POST "$\{{ secrets.COOLIFY_WEBHOOK_URL }}" \
34
+ -H "Content-Type: application/json" \
35
+ -d '{"sha": "$\{{ github.sha }}"}'
@@ -0,0 +1,5 @@
1
+ node_modules/
2
+ dist/
3
+ .env
4
+ .env.local
5
+ *.log
@@ -0,0 +1,36 @@
1
+ {
2
+ "name": "{{name}}",
3
+ "version": "0.1.0",
4
+ "private": true,
5
+ "type": "module",
6
+ "scripts": {
7
+ "dev": "tsx watch src/index.ts",
8
+ "build": "tsc",
9
+ "start": "node dist/index.js",
10
+ "typecheck": "tsc --noEmit"
11
+ },
12
+ "dependencies": {
13
+ "express": "^4.21.0",
14
+ "mongoose": "^8.0.0",
15
+ "better-auth": "^1.0.0",
16
+ "dotenv": "^16.4.0",
17
+ "cors": "^2.8.5",
18
+ "helmet": "^7.1.0"{{#if websocket}},
19
+ "ws": "^8.16.0",
20
+ "ioredis": "^5.3.0"{{/if}}{{#if stripe}},
21
+ "stripe": "^15.0.0"{{/if}}{{#if analytics}},
22
+ "@sentry/node": "^8.0.0"{{/if}}{{#if s3}},
23
+ "@aws-sdk/client-s3": "^3.500.0",
24
+ "@aws-sdk/s3-request-presigner": "^3.500.0",
25
+ "multer": "^1.4.5-lts.1"{{/if}}
26
+ },
27
+ "devDependencies": {
28
+ "@types/express": "^4.17.21",
29
+ "@types/cors": "^2.8.17",
30
+ "@types/node": "^22.0.0",{{#if websocket}}
31
+ "@types/ws": "^8.5.10",{{/if}}{{#if s3}}
32
+ "@types/multer": "^1.4.11",{{/if}}
33
+ "tsx": "^4.0.0",
34
+ "typescript": "^5.6.0"
35
+ }
36
+ }
@@ -0,0 +1,13 @@
1
+ import { betterAuth } from "better-auth";
2
+ import { mongodbAdapter } from "better-auth/adapters/mongodb";
3
+ import mongoose from "mongoose";
4
+ import { config } from "../config.js";
5
+
6
+ export const auth = betterAuth({
7
+ database: mongodbAdapter(mongoose.connection),
8
+ secret: config.AUTH_SECRET,
9
+ baseURL: config.AUTH_URL,
10
+ emailAndPassword: {
11
+ enabled: true,
12
+ },
13
+ });
@@ -0,0 +1,19 @@
1
+ import { Router } from "express";
2
+ import { auth } from "./auth.js";
3
+
4
+ export const authRouter = Router();
5
+
6
+ // better-auth handles all auth routes via its handler
7
+ authRouter.all("/*splat", async (req, res) => {
8
+ const response = await auth.handler(req);
9
+ res.status(response.status);
10
+ for (const [key, value] of response.headers.entries()) {
11
+ res.setHeader(key, value);
12
+ }
13
+ if (response.body) {
14
+ const text = await response.text();
15
+ res.send(text);
16
+ } else {
17
+ res.end();
18
+ }
19
+ });
@@ -0,0 +1,36 @@
1
+ export const config = {
2
+ PORT: parseInt(process.env.PORT || "3000", 10),
3
+ FRONTEND_URL: process.env.FRONTEND_URL || "http://localhost:3000",
4
+ MONGODB_URI: process.env.MONGODB_URI || "mongodb://localhost:27017/{{name}}",
5
+ AUTH_SECRET: process.env.AUTH_SECRET || "change-me",
6
+ AUTH_URL: process.env.AUTH_URL || "http://localhost:3000",
7
+ {{#if websocket}}
8
+ REDIS_URL: process.env.REDIS_URL || "redis://localhost:6379",
9
+ {{/if}}
10
+ {{#if stripe}}
11
+ STRIPE_SECRET_KEY: process.env.STRIPE_SECRET_KEY || "",
12
+ STRIPE_WEBHOOK_SECRET: process.env.STRIPE_WEBHOOK_SECRET || "",
13
+ {{/if}}
14
+ {{#if analytics}}
15
+ GLITCHTIP_DSN: process.env.GLITCHTIP_DSN || "",
16
+ {{/if}}
17
+ {{#if s3}}
18
+ S3_ENDPOINT: process.env.S3_ENDPOINT || "",
19
+ S3_BUCKET: process.env.S3_BUCKET || "",
20
+ S3_ACCESS_KEY: process.env.S3_ACCESS_KEY || "",
21
+ S3_SECRET_KEY: process.env.S3_SECRET_KEY || "",
22
+ S3_REGION: process.env.S3_REGION || "us-east-1",
23
+ {{/if}}
24
+ {{#if has3d}}
25
+ ML_3D_EXTRACTION_ENDPOINT: process.env.ML_3D_EXTRACTION_ENDPOINT || "",
26
+ {{/if}}
27
+ {{#if hasSubtitles}}
28
+ ML_SUBTITLES_ENDPOINT: process.env.ML_SUBTITLES_ENDPOINT || "",
29
+ {{/if}}
30
+ {{#if hasImageRecognition}}
31
+ ML_IMAGE_RECOGNITION_ENDPOINT: process.env.ML_IMAGE_RECOGNITION_ENDPOINT || "",
32
+ {{/if}}
33
+ {{#if hasBgRemoval}}
34
+ ML_BACKGROUND_REMOVAL_ENDPOINT: process.env.ML_BACKGROUND_REMOVAL_ENDPOINT || "",
35
+ {{/if}}
36
+ } as const;
@@ -0,0 +1,12 @@
1
+ import mongoose from "mongoose";
2
+ import { config } from "./config.js";
3
+
4
+ export async function connectDb(): Promise<void> {
5
+ try {
6
+ await mongoose.connect(config.MONGODB_URI);
7
+ console.log("Connected to MongoDB");
8
+ } catch (error) {
9
+ console.error("MongoDB connection error:", error);
10
+ process.exit(1);
11
+ }
12
+ }
@@ -0,0 +1,80 @@
1
+ import "dotenv/config";
2
+ import express from "express";
3
+ import cors from "cors";
4
+ import helmet from "helmet";
5
+ {{#if websocket}}
6
+ import { createServer } from "node:http";
7
+ import { setupWebSocket } from "./ws.js";
8
+ {{/if}}
9
+ {{#if analytics}}
10
+ import { initSentry } from "./analytics/sentry.js";
11
+ import { errorTrackingMiddleware } from "./analytics/middleware.js";
12
+ {{/if}}
13
+ import { connectDb } from "./db.js";
14
+ import { healthRouter } from "./routes/health.js";
15
+ import { authRouter } from "./auth/routes.js";
16
+ {{#if stripe}}
17
+ import { stripeWebhookRouter } from "./stripe/webhook.js";
18
+ import { checkoutRouter } from "./stripe/checkout.js";
19
+ {{/if}}
20
+ import { config } from "./config.js";
21
+
22
+ const app = express();
23
+ {{#if analytics}}
24
+ initSentry(app);
25
+ {{/if}}
26
+
27
+ {{#if stripe}}
28
+ // Stripe webhook needs raw body — mount BEFORE json parser
29
+ app.use("/api/stripe/webhook", stripeWebhookRouter);
30
+ {{/if}}
31
+
32
+ app.use(helmet());
33
+ app.use(cors({ origin: config.FRONTEND_URL, credentials: true }));
34
+ app.use(express.json());
35
+
36
+ // Routes
37
+ app.use("/api/health", healthRouter);
38
+ app.use("/api/auth", authRouter);
39
+ {{#if stripe}}
40
+ app.use("/api/stripe", checkoutRouter);
41
+ {{/if}}
42
+
43
+ {{#if analytics}}
44
+ app.use(errorTrackingMiddleware);
45
+ {{/if}}
46
+
47
+ async function start() {
48
+ await connectDb();
49
+ {{#if websocket}}
50
+
51
+ const server = createServer(app);
52
+ setupWebSocket(server);
53
+
54
+ server.listen(config.PORT, () => {
55
+ console.log(`Server running on port ${config.PORT}`);
56
+ });
57
+
58
+ // Graceful shutdown
59
+ process.on("SIGTERM", () => {
60
+ console.log("SIGTERM received, shutting down gracefully...");
61
+ server.close(() => process.exit(0));
62
+ setTimeout(() => process.exit(1), 10_000);
63
+ });
64
+ {{else}}
65
+
66
+ app.listen(config.PORT, () => {
67
+ console.log(`Server running on port ${config.PORT}`);
68
+ });
69
+
70
+ process.on("SIGTERM", () => {
71
+ console.log("SIGTERM received, shutting down...");
72
+ process.exit(0);
73
+ });
74
+ {{/if}}
75
+ }
76
+
77
+ start().catch((err) => {
78
+ console.error("Failed to start:", err);
79
+ process.exit(1);
80
+ });
@@ -0,0 +1,12 @@
1
+ import { Router } from "express";
2
+ import mongoose from "mongoose";
3
+
4
+ export const healthRouter = Router();
5
+
6
+ healthRouter.get("/", (_req, res) => {
7
+ res.json({
8
+ status: "ok",
9
+ uptime: process.uptime(),
10
+ mongo: mongoose.connection.readyState === 1 ? "connected" : "disconnected",
11
+ });
12
+ });
@@ -0,0 +1,18 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ES2022",
4
+ "module": "ESNext",
5
+ "moduleResolution": "bundler",
6
+ "outDir": "./dist",
7
+ "rootDir": "./src",
8
+ "strict": true,
9
+ "esModuleInterop": true,
10
+ "skipLibCheck": true,
11
+ "forceConsistentCasingInFileNames": true,
12
+ "resolveJsonModule": true,
13
+ "declaration": true,
14
+ "sourceMap": true
15
+ },
16
+ "include": ["src/**/*"],
17
+ "exclude": ["node_modules", "dist"]
18
+ }
@@ -0,0 +1,20 @@
1
+ import { config } from "../config.js";
2
+
3
+ export async function generate3dModel(imageBuffer: Buffer): Promise<{
4
+ glbBuffer: Buffer;
5
+ }> {
6
+ const formData = new FormData();
7
+ formData.append("file", new Blob([imageBuffer]));
8
+
9
+ const res = await fetch(config.ML_3D_EXTRACTION_ENDPOINT, {
10
+ method: "POST",
11
+ body: formData,
12
+ });
13
+
14
+ if (!res.ok) {
15
+ throw new Error(`3D extraction failed: ${res.status} ${res.statusText}`);
16
+ }
17
+
18
+ const glbBuffer = Buffer.from(await res.arrayBuffer());
19
+ return { glbBuffer };
20
+ }
@@ -0,0 +1,17 @@
1
+ import { config } from "../config.js";
2
+
3
+ export async function removeBackground(imageBuffer: Buffer): Promise<Buffer> {
4
+ const formData = new FormData();
5
+ formData.append("file", new Blob([imageBuffer]));
6
+
7
+ const res = await fetch(config.ML_BACKGROUND_REMOVAL_ENDPOINT, {
8
+ method: "POST",
9
+ body: formData,
10
+ });
11
+
12
+ if (!res.ok) {
13
+ throw new Error(`Background removal failed: ${res.status} ${res.statusText}`);
14
+ }
15
+
16
+ return Buffer.from(await res.arrayBuffer());
17
+ }