Merge pull request #8 from bipprojectbali/tasks/fix-docker-build/optimize-config-and-prisma-handlers/02-04-2026-15-00
Tasks/fix docker build/optimize config and prisma handlers/02 04 2026 15 00
This commit is contained in:
47
.dockerignore
Normal file
47
.dockerignore
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
node_modules
|
||||||
|
.next
|
||||||
|
.git
|
||||||
|
.env
|
||||||
|
.env.local
|
||||||
|
.env.development.local
|
||||||
|
.env.test.local
|
||||||
|
.env.production.local
|
||||||
|
*.log
|
||||||
|
npm-debug.log*
|
||||||
|
yarn-debug.log*
|
||||||
|
yarn-error.log*
|
||||||
|
bun-debug.log*
|
||||||
|
|
||||||
|
# Docker files
|
||||||
|
Dockerfile
|
||||||
|
.dockerignore
|
||||||
|
|
||||||
|
# OS files
|
||||||
|
.DS_Store
|
||||||
|
Thumbs.db
|
||||||
|
|
||||||
|
# Markdown/Documentation
|
||||||
|
README.md
|
||||||
|
GEMINI.md
|
||||||
|
AGENTS.md
|
||||||
|
AUDIT_REPORT.md
|
||||||
|
QWEN.md
|
||||||
|
NOTE.md
|
||||||
|
task-project-apbdes.md
|
||||||
|
MUSIK_CREATE_ANALYSIS.md
|
||||||
|
darkMode.md
|
||||||
|
/test-results
|
||||||
|
/playwright-report
|
||||||
|
/tmp_assets
|
||||||
|
/foldergambar
|
||||||
|
/googleapi
|
||||||
|
/xx
|
||||||
|
/xx.ts
|
||||||
|
/xx.txt
|
||||||
|
/test.txt
|
||||||
|
/x.json
|
||||||
|
/x.sh
|
||||||
|
/xcoba.ts
|
||||||
|
/xcoba2.ts
|
||||||
|
/gambar.ttx
|
||||||
|
/test-berita-state.ts
|
||||||
22
Dockerfile
22
Dockerfile
@@ -1,5 +1,5 @@
|
|||||||
# Stage 1: Build
|
# Stage 1: Build
|
||||||
FROM oven/bun:1.3 AS build
|
FROM oven/bun:1.1 AS build
|
||||||
|
|
||||||
# Install build dependencies for native modules
|
# Install build dependencies for native modules
|
||||||
RUN apt-get update && apt-get install -y \
|
RUN apt-get update && apt-get install -y \
|
||||||
@@ -11,10 +11,18 @@ RUN apt-get update && apt-get install -y \
|
|||||||
# Set the working directory
|
# Set the working directory
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
|
# Disable telemetry and set build-time environment
|
||||||
|
ENV NEXT_TELEMETRY_DISABLED=1
|
||||||
|
ENV NODE_ENV=production
|
||||||
|
ENV NODE_OPTIONS="--max-old-space-size=4096"
|
||||||
|
# Critical ENV for API route evaluation during build
|
||||||
|
ENV WIBU_UPLOAD_DIR=uploads
|
||||||
|
ENV DATABASE_URL="postgresql://bip:Production_123@pgbouncer:5432/desa-darmasaba-staging?pgbouncer=true"
|
||||||
|
|
||||||
# Copy package files
|
# Copy package files
|
||||||
COPY package.json bun.lock* ./
|
COPY package.json bun.lock* ./
|
||||||
|
|
||||||
# Install dependencies
|
# Install dependencies with frozen lockfile
|
||||||
RUN bun install --frozen-lockfile
|
RUN bun install --frozen-lockfile
|
||||||
|
|
||||||
# Copy the rest of the application code
|
# Copy the rest of the application code
|
||||||
@@ -27,14 +35,16 @@ RUN cp .env.example .env
|
|||||||
RUN bun x prisma generate
|
RUN bun x prisma generate
|
||||||
|
|
||||||
# Build the application frontend
|
# Build the application frontend
|
||||||
ENV NODE_ENV=production
|
|
||||||
RUN bun run build
|
RUN bun run build
|
||||||
|
|
||||||
# Stage 2: Runtime
|
# Stage 2: Runtime
|
||||||
FROM oven/bun:1.3-slim AS runtime
|
FROM oven/bun:1.1-slim AS runtime
|
||||||
|
|
||||||
# Set environment variables
|
# Set environment variables
|
||||||
ENV NODE_ENV=production
|
ENV NODE_ENV=production
|
||||||
|
ENV NEXT_TELEMETRY_DISABLED=1
|
||||||
|
# Ensure runtime also has critical envs if they are checked at startup
|
||||||
|
ENV WIBU_UPLOAD_DIR=uploads
|
||||||
|
|
||||||
# Install runtime dependencies
|
# Install runtime dependencies
|
||||||
RUN apt-get update && apt-get install -y \
|
RUN apt-get update && apt-get install -y \
|
||||||
@@ -46,10 +56,12 @@ WORKDIR /app
|
|||||||
|
|
||||||
# Copy necessary files from build stage
|
# Copy necessary files from build stage
|
||||||
COPY --from=build /app/package.json ./
|
COPY --from=build /app/package.json ./
|
||||||
|
COPY --from=build /app/bun.lock* ./
|
||||||
|
COPY --from=build /app/next.config.ts ./
|
||||||
|
COPY --from=build /app/postcss.config.cjs ./
|
||||||
COPY --from=build /app/tsconfig.json ./
|
COPY --from=build /app/tsconfig.json ./
|
||||||
COPY --from=build /app/.next ./.next
|
COPY --from=build /app/.next ./.next
|
||||||
COPY --from=build /app/public ./public
|
COPY --from=build /app/public ./public
|
||||||
COPY --from=build /app/src ./src
|
|
||||||
COPY --from=build /app/node_modules ./node_modules
|
COPY --from=build /app/node_modules ./node_modules
|
||||||
COPY --from=build /app/prisma ./prisma
|
COPY --from=build /app/prisma ./prisma
|
||||||
|
|
||||||
|
|||||||
@@ -120,7 +120,7 @@
|
|||||||
"@types/react-dom": "^19",
|
"@types/react-dom": "^19",
|
||||||
"@vitest/ui": "^4.0.18",
|
"@vitest/ui": "^4.0.18",
|
||||||
"eslint": "^9",
|
"eslint": "^9",
|
||||||
"eslint-config-next": "15.1.6",
|
"eslint-config-next": "15.5.12",
|
||||||
"jsdom": "^28.0.0",
|
"jsdom": "^28.0.0",
|
||||||
"msw": "^2.12.9",
|
"msw": "^2.12.9",
|
||||||
"parcel": "^2.6.2",
|
"parcel": "^2.6.2",
|
||||||
|
|||||||
@@ -43,7 +43,6 @@ export async function POST(req: Request) {
|
|||||||
try {
|
try {
|
||||||
const res = await fetch(waUrl);
|
const res = await fetch(waUrl);
|
||||||
if (!res.ok) {
|
if (!res.ok) {
|
||||||
const errorText = await res.text();
|
|
||||||
console.error(`⚠️ WA Service HTTP Error: ${res.status} ${res.statusText}. Continuing since OTP is logged.`);
|
console.error(`⚠️ WA Service HTTP Error: ${res.status} ${res.statusText}. Continuing since OTP is logged.`);
|
||||||
console.log(`💡 Use this OTP to login: ${codeOtp}`);
|
console.log(`💡 Use this OTP to login: ${codeOtp}`);
|
||||||
} else {
|
} else {
|
||||||
@@ -53,8 +52,9 @@ export async function POST(req: Request) {
|
|||||||
console.error("⚠️ WA Service Logic Error:", sendWa);
|
console.error("⚠️ WA Service Logic Error:", sendWa);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (waError: any) {
|
} catch (waError: unknown) {
|
||||||
console.error("⚠️ WA Connection Exception. Continuing since OTP is logged.", waError.message);
|
const errorMessage = waError instanceof Error ? waError.message : String(waError);
|
||||||
|
console.error("⚠️ WA Connection Exception. Continuing since OTP is logged.", errorMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
const createOtpId = await prisma.kodeOtp.create({
|
const createOtpId = await prisma.kodeOtp.create({
|
||||||
|
|||||||
@@ -35,8 +35,9 @@ export async function POST(req: Request) {
|
|||||||
const waData = await waRes.json();
|
const waData = await waRes.json();
|
||||||
console.log("📱 WA Response (Register):", waData);
|
console.log("📱 WA Response (Register):", waData);
|
||||||
}
|
}
|
||||||
} catch (waError: any) {
|
} catch (waError: unknown) {
|
||||||
console.warn("⚠️ WA Connection Exception (Register). Continuing since OTP is logged.", waError.message);
|
const errorMessage = waError instanceof Error ? waError.message : String(waError);
|
||||||
|
console.warn("⚠️ WA Connection Exception (Register). Continuing since OTP is logged.", errorMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
// ✅ Simpan OTP ke database
|
// ✅ Simpan OTP ke database
|
||||||
|
|||||||
@@ -31,8 +31,9 @@ export async function POST(req: Request) {
|
|||||||
const waData = await waRes.json();
|
const waData = await waRes.json();
|
||||||
console.log("📱 WA Response (Resend):", waData);
|
console.log("📱 WA Response (Resend):", waData);
|
||||||
}
|
}
|
||||||
} catch (waError: any) {
|
} catch (waError: unknown) {
|
||||||
console.warn("⚠️ WA Connection Exception (Resend). Continuing since OTP is logged.", waError.message);
|
const errorMessage = waError instanceof Error ? waError.message : String(waError);
|
||||||
|
console.warn("⚠️ WA Connection Exception (Resend). Continuing since OTP is logged.", errorMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Simpan OTP ke database
|
// Simpan OTP ke database
|
||||||
|
|||||||
@@ -34,8 +34,9 @@ export async function POST(req: Request) {
|
|||||||
const sendWa = await res.json();
|
const sendWa = await res.json();
|
||||||
console.log("📱 WA Response (SendOTPRegister):", sendWa);
|
console.log("📱 WA Response (SendOTPRegister):", sendWa);
|
||||||
}
|
}
|
||||||
} catch (waError: any) {
|
} catch (waError: unknown) {
|
||||||
console.warn("⚠️ WA Connection Exception (SendOTPRegister). Continuing since OTP is logged.", waError.message);
|
const errorMessage = waError instanceof Error ? waError.message : String(waError);
|
||||||
|
console.warn("⚠️ WA Connection Exception (SendOTPRegister). Continuing since OTP is logged.", errorMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Simpan OTP
|
// Simpan OTP
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
/* eslint-disable @typescript-eslint/no-explicit-any */
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||||
import { Paper, Title, Progress, Stack, Text, Group, Box, rem } from '@mantine/core'
|
import { Paper, Title, Progress, Stack, Text, Group, Box } from '@mantine/core'
|
||||||
import { IconArrowUpRight, IconArrowDownRight } from '@tabler/icons-react'
|
import { IconArrowUpRight, IconArrowDownRight } from '@tabler/icons-react'
|
||||||
import { APBDes, APBDesItem, SummaryData } from '../types/apbdes'
|
import { APBDes, APBDesItem } from '../types/apbdes'
|
||||||
|
|
||||||
interface SummaryProps {
|
interface SummaryProps {
|
||||||
title: string
|
title: string
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
||||||
import { Paper, Table, Title, Box, ScrollArea, Badge } from '@mantine/core'
|
import { Paper, Table, Title, Box, ScrollArea, Badge } from '@mantine/core'
|
||||||
import { APBDes, APBDesItem } from '../types/apbdes'
|
import { APBDes, APBDesItem } from '../types/apbdes'
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
||||||
import { Paper, Table, Title, Badge, Text, Box, ScrollArea } from '@mantine/core'
|
import { Paper, Table, Title, Badge, Text, Box, ScrollArea } from '@mantine/core'
|
||||||
import { APBDes, APBDesItem, RealisasiItem } from '../types/apbdes'
|
import { APBDes, APBDesItem, RealisasiItem } from '../types/apbdes'
|
||||||
|
|
||||||
|
|||||||
@@ -29,16 +29,18 @@ process.on('unhandledRejection', async (error) => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Handle graceful shutdown
|
// Handle graceful shutdown
|
||||||
process.on('SIGINT', async () => {
|
if (process.env.NODE_ENV === 'production' && !process.env.NEXT_PHASE) {
|
||||||
|
process.on('SIGINT', async () => {
|
||||||
console.log('Received SIGINT signal. Closing database connections...');
|
console.log('Received SIGINT signal. Closing database connections...');
|
||||||
await prisma.$disconnect();
|
await prisma.$disconnect();
|
||||||
process.exit(0);
|
// Allow natural exit
|
||||||
});
|
});
|
||||||
|
|
||||||
process.on('SIGTERM', async () => {
|
process.on('SIGTERM', async () => {
|
||||||
console.log('Received SIGTERM signal. Closing database connections...');
|
console.log('Received SIGTERM signal. Closing database connections...');
|
||||||
await prisma.$disconnect();
|
await prisma.$disconnect();
|
||||||
process.exit(0);
|
// Allow natural exit
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
export default prisma;
|
export default prisma;
|
||||||
Reference in New Issue
Block a user