Compare commits
5 Commits
tasks/api/
...
nico/2-apr
| Author | SHA1 | Date | |
|---|---|---|---|
| 721357adcf | |||
| 50a7356618 | |||
| 970949a68b | |||
| 8777c45a44 | |||
| b751f031cd |
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
|
||||||
84
Dockerfile
84
Dockerfile
@@ -1,3 +1,77 @@
|
|||||||
|
# # Stage 1: Build
|
||||||
|
# FROM oven/bun:1.1 AS build
|
||||||
|
|
||||||
|
# # Install build dependencies for native modules
|
||||||
|
# RUN apt-get update && apt-get install -y \
|
||||||
|
# python3 \
|
||||||
|
# make \
|
||||||
|
# g++ \
|
||||||
|
# && rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
# # Set the working directory
|
||||||
|
# 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.json bun.lock* ./
|
||||||
|
|
||||||
|
# # Install dependencies with frozen lockfile
|
||||||
|
# RUN bun install --frozen-lockfile
|
||||||
|
|
||||||
|
# # Copy the rest of the application code
|
||||||
|
# COPY . .
|
||||||
|
|
||||||
|
# # Use .env.example as default env for build
|
||||||
|
# RUN cp .env.example .env
|
||||||
|
|
||||||
|
# # Generate Prisma client
|
||||||
|
# RUN bun x prisma generate
|
||||||
|
|
||||||
|
# # Build the application frontend
|
||||||
|
# RUN bun run build
|
||||||
|
|
||||||
|
# # Stage 2: Runtime
|
||||||
|
# FROM oven/bun:1.1-slim AS runtime
|
||||||
|
|
||||||
|
# # Set environment variables
|
||||||
|
# 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
|
||||||
|
# RUN apt-get update && apt-get install -y \
|
||||||
|
# postgresql-client \
|
||||||
|
# && rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
# # Set the working directory
|
||||||
|
# WORKDIR /app
|
||||||
|
|
||||||
|
# # Copy necessary files from build stage
|
||||||
|
# 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/.next ./.next
|
||||||
|
# COPY --from=build /app/public ./public
|
||||||
|
# COPY --from=build /app/node_modules ./node_modules
|
||||||
|
# COPY --from=build /app/prisma ./prisma
|
||||||
|
|
||||||
|
# # Expose the port
|
||||||
|
# EXPOSE 3000
|
||||||
|
|
||||||
|
# # Start the application
|
||||||
|
# CMD ["bun", "start"]
|
||||||
|
|
||||||
|
|
||||||
# Stage 1: Build
|
# Stage 1: Build
|
||||||
FROM oven/bun:1.3 AS build
|
FROM oven/bun:1.3 AS build
|
||||||
|
|
||||||
@@ -26,8 +100,10 @@ RUN cp .env.example .env
|
|||||||
# Generate Prisma client
|
# Generate Prisma client
|
||||||
RUN bun x prisma generate
|
RUN bun x prisma generate
|
||||||
|
|
||||||
|
# Generate API types
|
||||||
|
RUN bun run gen:api
|
||||||
|
|
||||||
# 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
|
||||||
@@ -47,8 +123,8 @@ 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/tsconfig.json ./
|
COPY --from=build /app/tsconfig.json ./
|
||||||
COPY --from=build /app/.next ./.next
|
COPY --from=build /app/dist ./dist
|
||||||
COPY --from=build /app/public ./public
|
COPY --from=build /app/generated ./generated
|
||||||
COPY --from=build /app/src ./src
|
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
|
||||||
@@ -57,4 +133,4 @@ COPY --from=build /app/prisma ./prisma
|
|||||||
EXPOSE 3000
|
EXPOSE 3000
|
||||||
|
|
||||||
# Start the application
|
# Start the application
|
||||||
CMD ["bun", "start"]
|
CMD ["bun", "start"]
|
||||||
@@ -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",
|
||||||
|
|||||||
@@ -81,7 +81,7 @@ const Utils = new Elysia({
|
|||||||
if (!process.env.WIBU_UPLOAD_DIR)
|
if (!process.env.WIBU_UPLOAD_DIR)
|
||||||
throw new Error("WIBU_UPLOAD_DIR is not defined");
|
throw new Error("WIBU_UPLOAD_DIR is not defined");
|
||||||
|
|
||||||
const ApiServer = new Elysia()
|
const ApiServer = new Elysia({ prefix: "/api" })
|
||||||
.use(
|
.use(
|
||||||
staticPlugin({
|
staticPlugin({
|
||||||
assets: UPLOAD_DIR,
|
assets: UPLOAD_DIR,
|
||||||
@@ -89,6 +89,17 @@ const ApiServer = new Elysia()
|
|||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
.use(cors(corsConfig))
|
.use(cors(corsConfig))
|
||||||
|
.use(
|
||||||
|
swagger({
|
||||||
|
path: "/docs",
|
||||||
|
documentation: {
|
||||||
|
info: {
|
||||||
|
title: "Desa Darmasaba API Documentation",
|
||||||
|
version: "1.0.0",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
.onError(({ code }) => {
|
.onError(({ code }) => {
|
||||||
if (code === "NOT_FOUND") {
|
if (code === "NOT_FOUND") {
|
||||||
return {
|
return {
|
||||||
@@ -97,142 +108,128 @@ const ApiServer = new Elysia()
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.group("/api", (app) =>
|
.use(Utils)
|
||||||
app
|
.use(FileStorage)
|
||||||
.use(
|
.use(LandingPage)
|
||||||
swagger({
|
.use(PPID)
|
||||||
path: "/docs",
|
.use(Desa)
|
||||||
documentation: {
|
.use(Kesehatan)
|
||||||
info: {
|
.use(Keamanan)
|
||||||
title: "Desa Darmasaba API Documentation",
|
.use(Ekonomi)
|
||||||
version: "1.0.0",
|
.use(Inovasi)
|
||||||
},
|
.use(Lingkungan)
|
||||||
},
|
.use(Pendidikan)
|
||||||
|
.use(User)
|
||||||
|
.use(Role)
|
||||||
|
.use(Search)
|
||||||
|
.get("/layanan", layanan)
|
||||||
|
.get("/potensi", getPotensi)
|
||||||
|
.get(
|
||||||
|
"/img/:name",
|
||||||
|
({ params, query }) => {
|
||||||
|
return img({
|
||||||
|
name: params.name,
|
||||||
|
UPLOAD_DIR_IMAGE,
|
||||||
|
ROOT,
|
||||||
|
size: query.size,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
{
|
||||||
|
params: t.Object({
|
||||||
|
name: t.String(),
|
||||||
|
}),
|
||||||
|
query: t.Optional(
|
||||||
|
t.Object({
|
||||||
|
size: t.Optional(t.Number()),
|
||||||
}),
|
}),
|
||||||
)
|
|
||||||
.use(Utils)
|
|
||||||
.use(FileStorage)
|
|
||||||
.use(LandingPage)
|
|
||||||
.use(PPID)
|
|
||||||
.use(Desa)
|
|
||||||
.use(Kesehatan)
|
|
||||||
.use(Keamanan)
|
|
||||||
.use(Ekonomi)
|
|
||||||
.use(Inovasi)
|
|
||||||
.use(Lingkungan)
|
|
||||||
.use(Pendidikan)
|
|
||||||
.use(User)
|
|
||||||
.use(Role)
|
|
||||||
.use(Search)
|
|
||||||
.get("/layanan", layanan)
|
|
||||||
.get("/potensi", getPotensi)
|
|
||||||
.get(
|
|
||||||
"/img/:name",
|
|
||||||
({ params, query }) => {
|
|
||||||
return img({
|
|
||||||
name: params.name,
|
|
||||||
UPLOAD_DIR_IMAGE,
|
|
||||||
ROOT,
|
|
||||||
size: query.size,
|
|
||||||
});
|
|
||||||
},
|
|
||||||
{
|
|
||||||
params: t.Object({
|
|
||||||
name: t.String(),
|
|
||||||
}),
|
|
||||||
query: t.Optional(
|
|
||||||
t.Object({
|
|
||||||
size: t.Optional(t.Number()),
|
|
||||||
}),
|
|
||||||
),
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.delete(
|
|
||||||
"/img/:name",
|
|
||||||
({ params }) => {
|
|
||||||
return imgDel({
|
|
||||||
name: params.name,
|
|
||||||
UPLOAD_DIR_IMAGE,
|
|
||||||
});
|
|
||||||
},
|
|
||||||
{
|
|
||||||
params: t.Object({
|
|
||||||
name: t.String(),
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.get(
|
|
||||||
"/imgs",
|
|
||||||
({ query }) => {
|
|
||||||
return imgs({
|
|
||||||
search: query.search,
|
|
||||||
page: query.page,
|
|
||||||
count: query.count,
|
|
||||||
UPLOAD_DIR_IMAGE,
|
|
||||||
});
|
|
||||||
},
|
|
||||||
{
|
|
||||||
query: t.Optional(
|
|
||||||
t.Object({
|
|
||||||
page: t.Number({ default: 1 }),
|
|
||||||
count: t.Number({ default: 10 }),
|
|
||||||
search: t.String({ default: "" }),
|
|
||||||
}),
|
|
||||||
),
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.post(
|
|
||||||
"/upl-img",
|
|
||||||
({ body }) => {
|
|
||||||
console.log(body.title);
|
|
||||||
return uplImg({ files: body.files, UPLOAD_DIR_IMAGE });
|
|
||||||
},
|
|
||||||
{
|
|
||||||
body: t.Object({
|
|
||||||
title: t.String(),
|
|
||||||
files: t.Files({ multiple: true }),
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.post(
|
|
||||||
"/upl-img-single",
|
|
||||||
({ body }) => {
|
|
||||||
return uplImgSingle({
|
|
||||||
fileName: body.name,
|
|
||||||
file: body.file,
|
|
||||||
UPLOAD_DIR_IMAGE,
|
|
||||||
});
|
|
||||||
},
|
|
||||||
{
|
|
||||||
body: t.Object({
|
|
||||||
name: t.String(),
|
|
||||||
file: t.File(),
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.post(
|
|
||||||
"/upl-csv-single",
|
|
||||||
({ body }) => {
|
|
||||||
return uplCsvSingle({ fileName: body.name, file: body.file });
|
|
||||||
},
|
|
||||||
{
|
|
||||||
body: t.Object({
|
|
||||||
name: t.String(),
|
|
||||||
file: t.File(),
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.post(
|
|
||||||
"/upl-csv",
|
|
||||||
({ body }) => {
|
|
||||||
return uplCsv({ files: body.files });
|
|
||||||
},
|
|
||||||
{
|
|
||||||
body: t.Object({
|
|
||||||
files: t.Files(),
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
),
|
),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.delete(
|
||||||
|
"/img/:name",
|
||||||
|
({ params }) => {
|
||||||
|
return imgDel({
|
||||||
|
name: params.name,
|
||||||
|
UPLOAD_DIR_IMAGE,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
{
|
||||||
|
params: t.Object({
|
||||||
|
name: t.String(),
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.get(
|
||||||
|
"/imgs",
|
||||||
|
({ query }) => {
|
||||||
|
return imgs({
|
||||||
|
search: query.search,
|
||||||
|
page: query.page,
|
||||||
|
count: query.count,
|
||||||
|
UPLOAD_DIR_IMAGE,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
{
|
||||||
|
query: t.Optional(
|
||||||
|
t.Object({
|
||||||
|
page: t.Number({ default: 1 }),
|
||||||
|
count: t.Number({ default: 10 }),
|
||||||
|
search: t.String({ default: "" }),
|
||||||
|
}),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.post(
|
||||||
|
"/upl-img",
|
||||||
|
({ body }) => {
|
||||||
|
console.log(body.title);
|
||||||
|
return uplImg({ files: body.files, UPLOAD_DIR_IMAGE });
|
||||||
|
},
|
||||||
|
{
|
||||||
|
body: t.Object({
|
||||||
|
title: t.String(),
|
||||||
|
files: t.Files({ multiple: true }),
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.post(
|
||||||
|
"/upl-img-single",
|
||||||
|
({ body }) => {
|
||||||
|
return uplImgSingle({
|
||||||
|
fileName: body.name,
|
||||||
|
file: body.file,
|
||||||
|
UPLOAD_DIR_IMAGE,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
{
|
||||||
|
body: t.Object({
|
||||||
|
name: t.String(),
|
||||||
|
file: t.File(),
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.post(
|
||||||
|
"/upl-csv-single",
|
||||||
|
({ body }) => {
|
||||||
|
return uplCsvSingle({ fileName: body.name, file: body.file });
|
||||||
|
},
|
||||||
|
{
|
||||||
|
body: t.Object({
|
||||||
|
name: t.String(),
|
||||||
|
file: t.File(),
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.post(
|
||||||
|
"/upl-csv",
|
||||||
|
({ body }) => {
|
||||||
|
return uplCsv({ files: body.files });
|
||||||
|
},
|
||||||
|
{
|
||||||
|
body: t.Object({
|
||||||
|
files: t.Files(),
|
||||||
|
}),
|
||||||
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
export const GET = ApiServer.handle;
|
export const GET = ApiServer.handle;
|
||||||
|
|||||||
@@ -33,6 +33,8 @@ export async function POST(req: Request) {
|
|||||||
const codeOtp = randomOTP();
|
const codeOtp = randomOTP();
|
||||||
const otpNumber = Number(codeOtp);
|
const otpNumber = Number(codeOtp);
|
||||||
|
|
||||||
|
console.log(`🔑 DEBUG OTP [${nomor}]: ${codeOtp}`);
|
||||||
|
|
||||||
const waMessage = `Website Desa Darmasaba - Kode ini bersifat RAHASIA dan JANGAN DI BAGIKAN KEPADA SIAPAPUN, termasuk anggota ataupun Admin lainnya.\n\n>> Kode OTP anda: ${codeOtp}.`;
|
const waMessage = `Website Desa Darmasaba - Kode ini bersifat RAHASIA dan JANGAN DI BAGIKAN KEPADA SIAPAPUN, termasuk anggota ataupun Admin lainnya.\n\n>> Kode OTP anda: ${codeOtp}.`;
|
||||||
const waUrl = `https://wa.wibudev.com/code?nom=${encodeURIComponent(nomor)}&text=${encodeURIComponent(waMessage)}`;
|
const waUrl = `https://wa.wibudev.com/code?nom=${encodeURIComponent(nomor)}&text=${encodeURIComponent(waMessage)}`;
|
||||||
|
|
||||||
@@ -40,26 +42,19 @@ export async function POST(req: Request) {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const res = await fetch(waUrl);
|
const res = await fetch(waUrl);
|
||||||
const sendWa = await res.json();
|
if (!res.ok) {
|
||||||
console.log("📱 WA Response:", sendWa);
|
console.error(`⚠️ WA Service HTTP Error: ${res.status} ${res.statusText}. Continuing since OTP is logged.`);
|
||||||
|
console.log(`💡 Use this OTP to login: ${codeOtp}`);
|
||||||
if (sendWa.status !== "success") {
|
} else {
|
||||||
console.error("❌ WA Service Error:", sendWa);
|
const sendWa = await res.json();
|
||||||
return NextResponse.json(
|
console.log("📱 WA Response:", sendWa);
|
||||||
{
|
if (sendWa.status !== "success") {
|
||||||
success: false,
|
console.error("⚠️ WA Service Logic Error:", sendWa);
|
||||||
message: "Gagal mengirim OTP via WhatsApp",
|
}
|
||||||
debug: sendWa
|
|
||||||
},
|
|
||||||
{ status: 400 }
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
} catch (waError) {
|
} catch (waError: unknown) {
|
||||||
console.error("❌ Fetch WA Error:", waError);
|
const errorMessage = waError instanceof Error ? waError.message : String(waError);
|
||||||
return NextResponse.json(
|
console.error("⚠️ WA Connection Exception. Continuing since OTP is logged.", errorMessage);
|
||||||
{ success: false, message: "Terjadi kesalahan saat mengirim WA" },
|
|
||||||
{ status: 500 }
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const createOtpId = await prisma.kodeOtp.create({
|
const createOtpId = await prisma.kodeOtp.create({
|
||||||
|
|||||||
@@ -22,14 +22,22 @@ export async function POST(req: Request) {
|
|||||||
// ✅ Generate dan kirim OTP
|
// ✅ Generate dan kirim OTP
|
||||||
const codeOtp = randomOTP();
|
const codeOtp = randomOTP();
|
||||||
const otpNumber = Number(codeOtp);
|
const otpNumber = Number(codeOtp);
|
||||||
|
console.log(`🔑 DEBUG REGISTER OTP [${nomor}]: ${codeOtp}`);
|
||||||
|
|
||||||
const waMessage = `Website Desa Darmasaba - Kode verifikasi Anda: ${codeOtp}`;
|
const waMessage = `Website Desa Darmasaba - Kode verifikasi Anda: ${codeOtp}`;
|
||||||
const waUrl = `https://wa.wibudev.com/code?nom=${encodeURIComponent(nomor)}&text=${encodeURIComponent(waMessage)}`;
|
const waUrl = `https://wa.wibudev.com/code?nom=${encodeURIComponent(nomor)}&text=${encodeURIComponent(waMessage)}`;
|
||||||
const waRes = await fetch(waUrl);
|
|
||||||
const waData = await waRes.json();
|
try {
|
||||||
|
const waRes = await fetch(waUrl);
|
||||||
if (waData.status !== "success") {
|
if (!waRes.ok) {
|
||||||
return NextResponse.json({ success: false, message: 'Gagal mengirim OTP via WhatsApp' }, { status: 400 });
|
console.warn(`⚠️ WA Service HTTP Error (Register): ${waRes.status} ${waRes.statusText}. Continuing since OTP is logged.`);
|
||||||
|
} else {
|
||||||
|
const waData = await waRes.json();
|
||||||
|
console.log("📱 WA Response (Register):", waData);
|
||||||
|
}
|
||||||
|
} catch (waError: unknown) {
|
||||||
|
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
|
||||||
|
|||||||
@@ -17,18 +17,23 @@ export async function POST(req: Request) {
|
|||||||
|
|
||||||
const codeOtp = randomOTP();
|
const codeOtp = randomOTP();
|
||||||
const otpNumber = Number(codeOtp);
|
const otpNumber = Number(codeOtp);
|
||||||
|
console.log(`🔑 DEBUG RESEND OTP [${nomor}]: ${codeOtp}`);
|
||||||
|
|
||||||
// Kirim OTP via WhatsApp
|
// Kirim OTP via WhatsApp
|
||||||
const waMessage = `Website Desa Darmasaba - Kode ini bersifat RAHASIA dan JANGAN DI BAGIKAN KEPADA SIAPAPUN, termasuk anggota ataupun Admin lainnya.\n\n>> Kode OTP anda: ${codeOtp}.`;
|
const waMessage = `Website Desa Darmasaba - Kode ini bersifat RAHASIA dan JANGAN DI BAGIKAN KEPADA SIAPAPUN, termasuk anggota ataupun Admin lainnya.\n\n>> Kode OTP anda: ${codeOtp}.`;
|
||||||
const waUrl = `https://wa.wibudev.com/code?nom=${encodeURIComponent(nomor)}&text=${encodeURIComponent(waMessage)}`;
|
const waUrl = `https://wa.wibudev.com/code?nom=${encodeURIComponent(nomor)}&text=${encodeURIComponent(waMessage)}`;
|
||||||
const waRes = await fetch(waUrl);
|
|
||||||
const waData = await waRes.json();
|
try {
|
||||||
|
const waRes = await fetch(waUrl);
|
||||||
if (waData.status !== "success") {
|
if (!waRes.ok) {
|
||||||
return NextResponse.json(
|
console.warn(`⚠️ WA Service HTTP Error (Resend): ${waRes.status} ${waRes.statusText}. Continuing since OTP is logged.`);
|
||||||
{ success: false, message: "Gagal mengirim OTP via WhatsApp" },
|
} else {
|
||||||
{ status: 400 }
|
const waData = await waRes.json();
|
||||||
);
|
console.log("📱 WA Response (Resend):", waData);
|
||||||
|
}
|
||||||
|
} catch (waError: unknown) {
|
||||||
|
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
|
||||||
|
|||||||
@@ -21,14 +21,22 @@ export async function POST(req: Request) {
|
|||||||
// Generate OTP
|
// Generate OTP
|
||||||
const codeOtp = randomOTP();
|
const codeOtp = randomOTP();
|
||||||
const otpNumber = Number(codeOtp);
|
const otpNumber = Number(codeOtp);
|
||||||
|
console.log(`🔑 DEBUG SEND-OTP-REGISTER [${nomor}]: ${codeOtp}`);
|
||||||
|
|
||||||
// Kirim WA
|
// Kirim WA
|
||||||
const waUrl = `https://wa.wibudev.com/code?nom=${encodeURIComponent(nomor)}&text=Website Desa Darmasaba - Kode verifikasi Anda: ${codeOtp}`;
|
const waUrl = `https://wa.wibudev.com/code?nom=${encodeURIComponent(nomor)}&text=Website Desa Darmasaba - Kode verifikasi Anda: ${codeOtp}`;
|
||||||
const res = await fetch(waUrl);
|
|
||||||
const sendWa = await res.json();
|
try {
|
||||||
|
const res = await fetch(waUrl);
|
||||||
if (sendWa.status !== "success") {
|
if (!res.ok) {
|
||||||
return NextResponse.json({ success: false, message: 'Gagal mengirim OTP' }, { status: 400 });
|
console.warn(`⚠️ WA Service HTTP Error (SendOTPRegister): ${res.status} ${res.statusText}. Continuing since OTP is logged.`);
|
||||||
|
} else {
|
||||||
|
const sendWa = await res.json();
|
||||||
|
console.log("📱 WA Response (SendOTPRegister):", sendWa);
|
||||||
|
}
|
||||||
|
} catch (waError: unknown) {
|
||||||
|
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,6 @@
|
|||||||
/* eslint-disable @typescript-eslint/no-explicit-any */
|
import { Paper, Title, Progress, Stack, Text, Group, Box } from '@mantine/core'
|
||||||
import { Paper, Title, Progress, Stack, Text, Group, Box, rem } 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
|
||||||
@@ -107,7 +106,7 @@ function Summary({ title, data, icon }: SummaryProps) {
|
|||||||
|
|
||||||
<Text
|
<Text
|
||||||
fz="xs"
|
fz="xs"
|
||||||
c={statusMessage.color as any}
|
c={statusMessage.color}
|
||||||
fw={600}
|
fw={600}
|
||||||
style={{
|
style={{
|
||||||
backgroundColor: `var(--mantine-color-${statusMessage.color}-0)`,
|
backgroundColor: `var(--mantine-color-${statusMessage.color}-0)`,
|
||||||
|
|||||||
@@ -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) {
|
||||||
console.log('Received SIGINT signal. Closing database connections...');
|
process.on('SIGINT', async () => {
|
||||||
await prisma.$disconnect();
|
console.log('Received SIGINT signal. Closing database connections...');
|
||||||
process.exit(0);
|
await prisma.$disconnect();
|
||||||
});
|
// 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