Compare commits

...

4 Commits

Author SHA1 Message Date
1c9459dcf3 Fix comment forum
Forum API (Mobile)
- src/app/api/mobile/forum/[id]/comment/route.ts

### No Issue
2026-01-29 17:41:21 +08:00
8b54f5ca65 Fix send whatsapp
Auth API
- src/app/api/auth/login/route.ts
- src/app/api/auth/mobile-login/route.ts
- src/app/api/auth/mobile-register/route.ts
- src/app/api/auth/resend/route.ts

User API (Mobile)
- src/app/api/mobile/user/route.ts
- src/app/api/mobile/admin/user/[id]/route.ts

Utility
- src/lib/code-otp-sender.ts

### No issue
2026-01-29 15:04:40 +08:00
c94da645f3 API – Donation (Admin & User)
- src/app/api/mobile/admin/donation/[id]/disbursement/route.ts
- src/app/api/mobile/donation/[id]/news/route.ts
- src/app/api/mobile/donation/route.ts

Donation Helper / Logic
- src/lib/mobile/donation/

### No Issue
2026-01-27 16:59:31 +08:00
da0477102e chore(release): 1.5.38 2026-01-27 16:57:22 +08:00
14 changed files with 191 additions and 51 deletions

View File

@@ -2,6 +2,8 @@
All notable changes to this project will be documented in this file. See [commit-and-tag-version](https://github.com/absolute-version/commit-and-tag-version) for commit guidelines.
## [1.5.38](https://wibugit.wibudev.com/wibu/hipmi/compare/v1.5.37...v1.5.38) (2026-01-27)
## [1.5.37](https://wibugit.wibudev.com/wibu/hipmi/compare/v1.5.36...v1.5.37) (2026-01-23)
## [1.5.36](https://wibugit.wibudev.com/wibu/hipmi/compare/v1.5.35...v1.5.36) (2026-01-13)

View File

@@ -1,6 +1,6 @@
{
"name": "hipmi",
"version": "1.5.37",
"version": "1.5.38",
"private": true,
"prisma": {
"seed": "bun prisma/seed.ts"

View File

@@ -2,7 +2,7 @@ import { prisma } from "@/lib";
import { randomOTP } from "@/app_modules/auth/fun/rondom_otp";
import backendLogger from "@/util/backendLogger";
import { NextResponse } from "next/server";
import { sendCodeOtp } from "@/lib/code-otp-sender";
import { funSendToWhatsApp } from "@/lib/code-otp-sender";
export async function POST(req: Request) {
if (req.method !== "POST") {
@@ -30,7 +30,7 @@ export async function POST(req: Request) {
{ status: 400 },
);
const resSendCode = await sendCodeOtp({
const resSendCode = await funSendToWhatsApp({
nomor,
codeOtp: codeOtp.toString(),
});

View File

@@ -1,7 +1,7 @@
import { prisma } from "@/lib";
import { randomOTP } from "@/app_modules/auth/fun/rondom_otp";
import { NextResponse } from "next/server";
import { sendCodeOtp } from "@/lib/code-otp-sender";
import { funSendToWhatsApp } from "@/lib/code-otp-sender";
export async function POST(req: Request) {
try {
@@ -35,7 +35,7 @@ export async function POST(req: Request) {
{ status: 400 },
);
const resSendCode = await sendCodeOtp({
const resSendCode = await funSendToWhatsApp({
nomor,
codeOtp: codeOtp.toString(),
});

View File

@@ -7,7 +7,7 @@ import {
NotificationMobileBodyType,
NotificationMobileTitleType,
} from "../../../../../types/type-mobile-notification";
import { sendCodeOtp } from "@/lib/code-otp-sender";
import { funSendToWhatsApp } from "@/lib/code-otp-sender";
export async function POST(req: Request) {
if (req.method !== "POST") {
@@ -70,7 +70,7 @@ export async function POST(req: Request) {
{ status: 400 }
);
const resSendCode = await sendCodeOtp({
const resSendCode = await funSendToWhatsApp({
nomor: data.nomor,
codeOtp: codeOtp.toString(),
});

View File

@@ -2,7 +2,7 @@ import { prisma } from "@/lib";
import { randomOTP } from "@/app_modules/auth/fun/rondom_otp";
import backendLogger from "@/util/backendLogger";
import { NextResponse } from "next/server";
import { sendCodeOtp } from "@/lib/code-otp-sender";
import { funSendToWhatsApp } from "@/lib/code-otp-sender";
export async function POST(req: Request) {
if (req.method !== "POST") {
@@ -17,7 +17,7 @@ export async function POST(req: Request) {
const body = await req.json();
const { nomor } = body;
const resSendCode = await sendCodeOtp({
const resSendCode = await funSendToWhatsApp({
nomor,
codeOtp: codeOtp.toString(),
});

View File

@@ -1,11 +1,22 @@
import { funFindDonaturList } from "@/lib/mobile/donation/find-donatur-list";
import {
sendNotificationMobileToManyUser,
sendNotificationMobileToOneUser,
} from "@/lib/mobile/notification/send-notification";
import prisma from "@/lib/prisma";
import { NextResponse } from "next/server";
import {
NotificationMobileBodyType,
NotificationMobileTitleType,
} from "../../../../../../../../types/type-mobile-notification";
import { routeUserMobile } from "@/lib/mobile/route-page-mobile";
export { POST, GET };
async function POST(request: Request, { params }: { params: { id: string } }) {
const { id } = params;
const { data } = await request.json();
const { title, nominalCair, deskripsi, imageId, authorId } = data;
try {
const dataDonasi = await prisma.donasi.findUnique({
@@ -22,19 +33,19 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
return NextResponse.json(
{
success: false,
message: "Pencarian Donasi Gagal",
reason: "Pencarian Donasi Gagal",
message: "DataPencarian Donasi Gagal",
reason: "Data Pencarian Donasi Gagal",
},
{ status: 400 }
{ status: 400 },
);
const createPencairan = await prisma.donasi_PencairanDana.create({
data: {
donasiId: id,
nominalCair: +data.nominalCair,
deskripsi: data.deskripsi,
title: data.title,
imageId: data.imageId,
nominalCair: +nominalCair,
deskripsi: deskripsi,
title: title,
imageId: imageId,
},
});
@@ -45,11 +56,11 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
message: "Pencairan Dana Gagal",
reason: "Pencairan Dana Gagal",
},
{ status: 400 }
{ status: 400 },
);
const hasilTotalPencairan =
Number(dataDonasi.totalPencairan) + Number(data.nominalCair);
Number(dataDonasi.totalPencairan) + Number(nominalCair);
// const hasilAkumulasiPencairan = Number(dataDonasi.akumulasiPencairan) + 1;
const countPencairan = await prisma.donasi_PencairanDana.count({
@@ -66,8 +77,47 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
akumulasiPencairan: countPencairan,
totalPencairan: hasilTotalPencairan,
},
select: {
authorId: true,
title: true,
},
});
// ================= START SEND NOTIFICATION =================
await sendNotificationMobileToOneUser({
recipientId: updateDonasi?.authorId!,
senderId: authorId,
payload: {
title: "Pencairan Dana Berhasil" as NotificationMobileTitleType,
body: `Telah dilaksanakan pencairan dana untuk ${updateDonasi?.title}` as NotificationMobileBodyType,
type: "announcement",
kategoriApp: "DONASI",
deepLink: routeUserMobile.donationDetailPublish({
id: id,
}),
},
});
const recipientIds = await funFindDonaturList(id);
if (recipientIds.length > 0) {
await sendNotificationMobileToManyUser({
recipientIds,
senderId: authorId,
payload: {
title: "Pencarian Dana" as NotificationMobileTitleType,
body: `Update pencarian dana pada ${updateDonasi?.title}` as NotificationMobileBodyType,
type: "announcement",
kategoriApp: "DONASI",
deepLink: routeUserMobile.donationDetailPublish({
id: id,
}),
},
});
}
// ================= END SEND NOTIFICATION =================
if (!updateDonasi)
return NextResponse.json(
{
@@ -75,7 +125,7 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
message: "Update Donasi Gagal",
reason: "Update Donasi Gagal",
},
{ status: 400 }
{ status: 400 },
);
return NextResponse.json(
@@ -84,7 +134,7 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
message: "Pencairan Dana Berhasil",
// data: data,
},
{ status: 200 }
{ status: 200 },
);
} catch (error) {
console.error("[ERROR]", error);
@@ -94,7 +144,7 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
message: "Pencairan Dana Gagal",
reason: (error as Error).message,
},
{ status: 500 }
{ status: 500 },
);
}
}
@@ -110,7 +160,6 @@ async function GET(request: Request, { params }: { params: { id: string } }) {
console.log("[CATEGORY]", category);
let fixData;
try {
if (category === "get-all") {
fixData = await prisma.donasi_PencairanDana.findMany({
take: page ? takeData : undefined,
@@ -140,7 +189,7 @@ async function GET(request: Request, { params }: { params: { id: string } }) {
message: "Category tidak ditemukan",
reason: "Category tidak ditemukan",
},
{ status: 400 }
{ status: 400 },
);
}
@@ -150,7 +199,7 @@ async function GET(request: Request, { params }: { params: { id: string } }) {
message: "Success get data disbursement",
data: fixData,
},
{ status: 200 }
{ status: 200 },
);
} catch (error) {
console.error("[ERROR]", error);
@@ -160,7 +209,7 @@ async function GET(request: Request, { params }: { params: { id: string } }) {
message: "Gagal mendapatkan data disbursement",
reason: (error as Error).message,
},
{ status: 500 }
{ status: 500 },
);
}
}

View File

@@ -1,4 +1,5 @@
import { prisma } from "@/lib";
import { funSendToWhatsApp } from "@/lib/code-otp-sender";
import _ from "lodash";
import { NextResponse } from "next/server";
@@ -50,8 +51,25 @@ async function PUT(request: Request, { params }: { params: { id: string } }) {
data: {
active: data.active,
},
select: {
nomor: true,
},
});
if (data.active) {
const resSendCode = await funSendToWhatsApp({
nomor: updateData.nomor,
newMessage:
"Halo sahabat HIConnect, \nSelamat akun anda telah aktif ! \n\n*Pesan ini di kirim secara otomatis, tidak perlu di balas.",
});
} else {
const resSendCode = await funSendToWhatsApp({
nomor: updateData.nomor,
newMessage:
"Halo sahabat HIConnect, \nMohon maaf akun anda telah dinonaktifkan ! Hubungi admin untuk informasi lebih lanjut. \n\n*Pesan ini di kirim secara otomatis, tidak perlu di balas.",
});
}
console.log("[Update Active Berhasil]", updateData);
} else if (category === "role") {
const fixName = _.startCase(data.role.replace(/_/g, " "));

View File

@@ -1,25 +1,39 @@
import { NextRequest, NextResponse } from "next/server";
import { prisma } from "@/lib";
import _ from "lodash";
import { sendNotificationMobileToManyUser } from "@/lib/mobile/notification/send-notification";
import {
NotificationMobileBodyType,
NotificationMobileTitleType,
} from "../../../../../../../types/type-mobile-notification";
import { routeUserMobile } from "@/lib/mobile/route-page-mobile";
import { funFindDonaturList } from "@/lib/mobile/donation/find-donatur-list";
export { POST, GET, PUT, DELETE };
async function POST(
request: NextRequest,
{ params }: { params: { id: string } }
{ params }: { params: { id: string } },
) {
const { id } = params;
const { data } = await request.json();
const { title, deskripsi, imageId } = data;
const senderId = await prisma.donasi.findUnique({
where: { id: id },
select: {
authorId: true,
},
});
try {
if (data && data?.imageId) {
const createWithFile = await prisma.donasi_Kabar.create({
data: {
title: data.title,
deskripsi: data.deskripsi,
title: title,
deskripsi: deskripsi,
donasiId: id,
imageId: data.imageId,
imageId: imageId,
},
});
@@ -28,8 +42,8 @@ async function POST(
} else {
const create = await prisma.donasi_Kabar.create({
data: {
title: data.title,
deskripsi: data.deskripsi,
title: title,
deskripsi: deskripsi,
donasiId: id,
},
});
@@ -38,6 +52,25 @@ async function POST(
return NextResponse.json({ status: 400, message: "Gagal disimpan" });
}
const recipientIds = await funFindDonaturList(id);
// SEND NOTIFICATION
if (recipientIds.length > 0) {
await sendNotificationMobileToManyUser({
recipientIds,
senderId: senderId?.authorId!,
payload: {
title: "Berita terbaru" as NotificationMobileTitleType,
body: `Ada berita terupdate pada ${title}` as NotificationMobileBodyType,
type: "announcement",
kategoriApp: "DONASI",
deepLink: routeUserMobile.donationDetailPublish({
id: id,
}),
},
});
}
return NextResponse.json({
status: 200,
success: true,
@@ -56,7 +89,7 @@ async function POST(
async function GET(
request: NextRequest,
{ params }: { params: { id: string } }
{ params }: { params: { id: string } },
) {
const { id } = params;
const { searchParams } = new URL(request.url);
@@ -178,7 +211,7 @@ async function PUT(request: Request, { params }: { params: { id: string } }) {
async function DELETE(
request: Request,
{ params }: { params: { id: string } }
{ params }: { params: { id: string } },
) {
const { id } = params;
try {
@@ -198,7 +231,7 @@ async function DELETE(
headers: {
Authorization: `Bearer ${process.env.WS_APIKEY}`,
},
}
},
);
if (!deleteImage) {

View File

@@ -5,7 +5,7 @@ import { NextResponse } from "next/server";
import { NotificationMobileBodyType } from "../../../../../types/type-mobile-notification";
import { routeAdminMobile } from "@/lib/mobile/route-page-mobile";
export { POST };
export { POST, GET };
async function POST(request: Request) {
const { data } = await request.json();
@@ -121,7 +121,7 @@ async function POST(request: Request) {
}
// GET ALL DATA DONASI
export async function GET(request: Request) {
async function GET(request: Request) {
const { searchParams } = new URL(request.url);
const category = searchParams.get("category");
const authorId = searchParams.get("authorId");

View File

@@ -90,9 +90,15 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
async function GET(request: Request, { params }: { params: { id: string } }) {
const { id } = params;
const { searchParams } = new URL(request.url);
const page = Number(searchParams.get("page"));
const takeData = 5
const skipData = page * takeData - takeData;
try {
const data = await prisma.forum_Komentar.findMany({
take: page ? takeData : undefined,
skip: page ? skipData : undefined,
orderBy: {
createdAt: "desc",
},

View File

@@ -5,8 +5,16 @@ export async function GET(request: Request) {
try {
const { searchParams } = new URL(request.url);
const search = searchParams.get("search");
const page = Number(searchParams.get("page"));
const takeData = 10;
const skipData = page * takeData - takeData;
console.log("SEARCH", search);
console.log("PAGE", page);
const data = await prisma.user.findMany({
take: page ? takeData : undefined,
skip: page ? skipData : undefined,
orderBy: {
username: "asc",
},
@@ -43,16 +51,12 @@ export async function GET(request: Request) {
},
});
return NextResponse.json(
{
success: true,
message: "Berhasil mendapatkan data",
data: data,
},
{
status: 200,
}
);
return NextResponse.json({
status: 200,
success: true,
message: "Berhasil mendapatkan data",
data: data,
});
} catch (error) {
return NextResponse.json(
{
@@ -62,7 +66,7 @@ export async function GET(request: Request) {
},
{
status: 500,
}
},
);
}

View File

@@ -1,13 +1,16 @@
const sendCodeOtp = async ({
nomor,
codeOtp,
newMessage,
}: {
nomor: string;
codeOtp: string;
codeOtp?: string;
newMessage?: string;
}) => {
const msg = `HIPMI%20-%20Kode%20ini%20bersifat%20RAHASIA%20dan%20JANGAN%20DI%20BAGIKAN%20KEPADA%20SIAPAPUN%2C%20termasuk%20anggota%20ataupun%20pengurus%20HIPMI%20lainnya.%20Kode%20OTP%20anda%3A%20${codeOtp}.`;
const msg = newMessage || `HIPMI - Kode ini bersifat RAHASIA dan JANGAN DI BAGIKAN KEPADA SIAPAPUN, termasuk anggota ataupun pengurus HIPMI lainnya.\n\n>> Kode OTP anda: ${codeOtp}.`;
const enCode = encodeURIComponent(msg);
const res = await fetch(
`https://cld-dkr-prod-wajs-server.wibudev.com/api/wa/code?nom=${nomor}&text=${msg}`,
`https://cld-dkr-prod-wajs-server.wibudev.com/api/wa/code?nom=${nomor}&text=${enCode}`,
{
cache: "no-cache",
headers: {
@@ -25,4 +28,4 @@ const sendCodeOtp = async ({
return res;
};
export { sendCodeOtp };
export { sendCodeOtp as funSendToWhatsApp };

View File

@@ -0,0 +1,25 @@
import prisma from "@/lib/prisma";
export const funFindDonaturList = async (donasiId: string) => {
const finDonatur = await prisma.donasi_Invoice.findMany({
where: {
donasiId: donasiId,
DonasiMaster_StatusInvoice: {
name: "Berhasil",
},
},
select: {
authorId: true,
},
distinct: ["authorId"], // Ambil hanya authorId unik dari DB
});
// Filter null safety (jika diperlukan)
const recipientIds = finDonatur
.map((e) => e.authorId)
.filter((id): id is string => id !== null && id !== undefined);
console.log("[FIND DONATUR UNIK]", recipientIds);
return recipientIds;
};