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.
- package/dist/config.d.ts +131 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +629 -0
- package/dist/config.js.map +1 -0
- package/dist/deploy/coolify.d.ts +4 -0
- package/dist/deploy/coolify.d.ts.map +1 -0
- package/dist/deploy/coolify.js +20 -0
- package/dist/deploy/coolify.js.map +1 -0
- package/dist/deploy/github.d.ts +4 -0
- package/dist/deploy/github.d.ts.map +1 -0
- package/dist/deploy/github.js +39 -0
- package/dist/deploy/github.js.map +1 -0
- package/dist/deploy/gpu.d.ts +4 -0
- package/dist/deploy/gpu.d.ts.map +1 -0
- package/dist/deploy/gpu.js +97 -0
- package/dist/deploy/gpu.js.map +1 -0
- package/dist/deploy/keys.d.ts +9 -0
- package/dist/deploy/keys.d.ts.map +1 -0
- package/dist/deploy/keys.js +73 -0
- package/dist/deploy/keys.js.map +1 -0
- package/dist/deploy/terraform.d.ts +4 -0
- package/dist/deploy/terraform.d.ts.map +1 -0
- package/dist/deploy/terraform.js +55 -0
- package/dist/deploy/terraform.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +599 -0
- package/dist/index.js.map +1 -0
- package/dist/prompts.d.ts +52 -0
- package/dist/prompts.d.ts.map +1 -0
- package/dist/prompts.js +313 -0
- package/dist/prompts.js.map +1 -0
- package/dist/provision/glitchtip.d.ts +6 -0
- package/dist/provision/glitchtip.d.ts.map +1 -0
- package/dist/provision/glitchtip.js +46 -0
- package/dist/provision/glitchtip.js.map +1 -0
- package/dist/provision/index.d.ts +9 -0
- package/dist/provision/index.d.ts.map +1 -0
- package/dist/provision/index.js +108 -0
- package/dist/provision/index.js.map +1 -0
- package/dist/provision/openpanel.d.ts +8 -0
- package/dist/provision/openpanel.d.ts.map +1 -0
- package/dist/provision/openpanel.js +66 -0
- package/dist/provision/openpanel.js.map +1 -0
- package/dist/provision/resend.d.ts +16 -0
- package/dist/provision/resend.d.ts.map +1 -0
- package/dist/provision/resend.js +43 -0
- package/dist/provision/resend.js.map +1 -0
- package/dist/scaffold/app.d.ts +13 -0
- package/dist/scaffold/app.d.ts.map +1 -0
- package/dist/scaffold/app.js +340 -0
- package/dist/scaffold/app.js.map +1 -0
- package/dist/scaffold/dotenvx.d.ts +30 -0
- package/dist/scaffold/dotenvx.d.ts.map +1 -0
- package/dist/scaffold/dotenvx.js +142 -0
- package/dist/scaffold/dotenvx.js.map +1 -0
- package/dist/scaffold/infra.d.ts +17 -0
- package/dist/scaffold/infra.d.ts.map +1 -0
- package/dist/scaffold/infra.js +200 -0
- package/dist/scaffold/infra.js.map +1 -0
- package/dist/scaffold/manifest.d.ts +50 -0
- package/dist/scaffold/manifest.d.ts.map +1 -0
- package/dist/scaffold/manifest.js +83 -0
- package/dist/scaffold/manifest.js.map +1 -0
- package/dist/scaffold/ml-client.d.ts +20 -0
- package/dist/scaffold/ml-client.d.ts.map +1 -0
- package/dist/scaffold/ml-client.js +38 -0
- package/dist/scaffold/ml-client.js.map +1 -0
- package/dist/scaffold/pkg-json.d.ts +20 -0
- package/dist/scaffold/pkg-json.d.ts.map +1 -0
- package/dist/scaffold/pkg-json.js +113 -0
- package/dist/scaffold/pkg-json.js.map +1 -0
- package/dist/scaffold/starter-files.d.ts +40 -0
- package/dist/scaffold/starter-files.d.ts.map +1 -0
- package/dist/scaffold/starter-files.js +197 -0
- package/dist/scaffold/starter-files.js.map +1 -0
- package/dist/scaffold/update.d.ts +8 -0
- package/dist/scaffold/update.d.ts.map +1 -0
- package/dist/scaffold/update.js +255 -0
- package/dist/scaffold/update.js.map +1 -0
- package/dist/templates/addons/analytics/middleware.ts.hbs +13 -0
- package/dist/templates/addons/analytics/sentry.ts.hbs +16 -0
- package/dist/templates/addons/storage/s3.ts.hbs +40 -0
- package/dist/templates/addons/storage/upload.ts.hbs +23 -0
- package/dist/templates/addons/stripe/checkout.ts.hbs +27 -0
- package/dist/templates/addons/stripe/client.ts.hbs +6 -0
- package/dist/templates/addons/stripe/webhook.ts.hbs +39 -0
- package/dist/templates/addons/websocket/redis.ts.hbs +25 -0
- package/dist/templates/addons/websocket/ws.ts.hbs +32 -0
- package/dist/templates/base/.dockerignore.hbs +7 -0
- package/dist/templates/base/Dockerfile.hbs +18 -0
- package/dist/templates/base/env.example.hbs +60 -0
- package/dist/templates/base/github-actions.yml.hbs +35 -0
- package/dist/templates/base/gitignore.hbs +5 -0
- package/dist/templates/base/package.json.hbs +36 -0
- package/dist/templates/base/src/auth/auth.ts.hbs +13 -0
- package/dist/templates/base/src/auth/routes.ts.hbs +19 -0
- package/dist/templates/base/src/config.ts.hbs +36 -0
- package/dist/templates/base/src/db.ts.hbs +12 -0
- package/dist/templates/base/src/index.ts.hbs +80 -0
- package/dist/templates/base/src/routes/health.ts.hbs +12 -0
- package/dist/templates/base/tsconfig.json.hbs +18 -0
- package/dist/templates/ml-clients/3d-extraction.ts.hbs +20 -0
- package/dist/templates/ml-clients/background-removal.ts.hbs +17 -0
- package/dist/templates/ml-clients/custom-hf.ts.hbs +20 -0
- package/dist/templates/ml-clients/image-recognition.ts.hbs +22 -0
- package/dist/templates/ml-clients/subtitles.ts.hbs +26 -0
- package/dist/utils/coolify-api.d.ts +45 -0
- package/dist/utils/coolify-api.d.ts.map +1 -0
- package/dist/utils/coolify-api.js +72 -0
- package/dist/utils/coolify-api.js.map +1 -0
- package/dist/utils/errors.d.ts +2 -0
- package/dist/utils/errors.d.ts.map +1 -0
- package/dist/utils/errors.js +31 -0
- package/dist/utils/errors.js.map +1 -0
- package/dist/utils/exec.d.ts +23 -0
- package/dist/utils/exec.d.ts.map +1 -0
- package/dist/utils/exec.js +47 -0
- package/dist/utils/exec.js.map +1 -0
- package/dist/utils/flags.d.ts +17 -0
- package/dist/utils/flags.d.ts.map +1 -0
- package/dist/utils/flags.js +116 -0
- package/dist/utils/flags.js.map +1 -0
- package/dist/utils/hf-api.d.ts +13 -0
- package/dist/utils/hf-api.d.ts.map +1 -0
- package/dist/utils/hf-api.js +30 -0
- package/dist/utils/hf-api.js.map +1 -0
- package/dist/utils/ports.d.ts +29 -0
- package/dist/utils/ports.d.ts.map +1 -0
- package/dist/utils/ports.js +86 -0
- package/dist/utils/ports.js.map +1 -0
- package/dist/utils/secrets.d.ts +25 -0
- package/dist/utils/secrets.d.ts.map +1 -0
- package/dist/utils/secrets.js +51 -0
- package/dist/utils/secrets.js.map +1 -0
- package/dist/utils/template.d.ts +7 -0
- package/dist/utils/template.d.ts.map +1 -0
- package/dist/utils/template.js +35 -0
- package/dist/utils/template.js.map +1 -0
- package/dist/utils/validate.d.ts +16 -0
- package/dist/utils/validate.d.ts.map +1 -0
- package/dist/utils/validate.js +60 -0
- package/dist/utils/validate.js.map +1 -0
- package/dist/utils/version.d.ts +2 -0
- package/dist/utils/version.d.ts.map +1 -0
- package/dist/utils/version.js +21 -0
- package/dist/utils/version.js.map +1 -0
- package/package.json +48 -0
- 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,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,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,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
|
+
}
|