Compare commits

...

5 Commits

Author SHA1 Message Date
f103ae93ad chore(release): 1.5.39 2026-01-30 17:16:05 +08:00
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 193 additions and 51 deletions

View File

@@ -2,6 +2,10 @@
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. 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.39](https://wibugit.wibudev.com/wibu/hipmi/compare/v1.5.38...v1.5.39) (2026-01-30)
## [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.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) ## [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", "name": "hipmi",
"version": "1.5.37", "version": "1.5.39",
"private": true, "private": true,
"prisma": { "prisma": {
"seed": "bun prisma/seed.ts" "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 { randomOTP } from "@/app_modules/auth/fun/rondom_otp";
import backendLogger from "@/util/backendLogger"; import backendLogger from "@/util/backendLogger";
import { NextResponse } from "next/server"; 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) { export async function POST(req: Request) {
if (req.method !== "POST") { if (req.method !== "POST") {
@@ -30,7 +30,7 @@ export async function POST(req: Request) {
{ status: 400 }, { status: 400 },
); );
const resSendCode = await sendCodeOtp({ const resSendCode = await funSendToWhatsApp({
nomor, nomor,
codeOtp: codeOtp.toString(), codeOtp: codeOtp.toString(),
}); });

View File

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

View File

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

View File

@@ -2,7 +2,7 @@ import { prisma } from "@/lib";
import { randomOTP } from "@/app_modules/auth/fun/rondom_otp"; import { randomOTP } from "@/app_modules/auth/fun/rondom_otp";
import backendLogger from "@/util/backendLogger"; import backendLogger from "@/util/backendLogger";
import { NextResponse } from "next/server"; 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) { export async function POST(req: Request) {
if (req.method !== "POST") { if (req.method !== "POST") {
@@ -17,7 +17,7 @@ export async function POST(req: Request) {
const body = await req.json(); const body = await req.json();
const { nomor } = body; const { nomor } = body;
const resSendCode = await sendCodeOtp({ const resSendCode = await funSendToWhatsApp({
nomor, nomor,
codeOtp: codeOtp.toString(), 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 prisma from "@/lib/prisma";
import { NextResponse } from "next/server"; import { NextResponse } from "next/server";
import {
NotificationMobileBodyType,
NotificationMobileTitleType,
} from "../../../../../../../../types/type-mobile-notification";
import { routeUserMobile } from "@/lib/mobile/route-page-mobile";
export { POST, GET }; export { POST, GET };
async function POST(request: Request, { params }: { params: { id: string } }) { async function POST(request: Request, { params }: { params: { id: string } }) {
const { id } = params; const { id } = params;
const { data } = await request.json(); const { data } = await request.json();
const { title, nominalCair, deskripsi, imageId, authorId } = data;
try { try {
const dataDonasi = await prisma.donasi.findUnique({ const dataDonasi = await prisma.donasi.findUnique({
@@ -22,19 +33,19 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
return NextResponse.json( return NextResponse.json(
{ {
success: false, success: false,
message: "Pencarian Donasi Gagal", message: "DataPencarian Donasi Gagal",
reason: "Pencarian Donasi Gagal", reason: "Data Pencarian Donasi Gagal",
}, },
{ status: 400 } { status: 400 },
); );
const createPencairan = await prisma.donasi_PencairanDana.create({ const createPencairan = await prisma.donasi_PencairanDana.create({
data: { data: {
donasiId: id, donasiId: id,
nominalCair: +data.nominalCair, nominalCair: +nominalCair,
deskripsi: data.deskripsi, deskripsi: deskripsi,
title: data.title, title: title,
imageId: data.imageId, imageId: imageId,
}, },
}); });
@@ -45,11 +56,11 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
message: "Pencairan Dana Gagal", message: "Pencairan Dana Gagal",
reason: "Pencairan Dana Gagal", reason: "Pencairan Dana Gagal",
}, },
{ status: 400 } { status: 400 },
); );
const hasilTotalPencairan = const hasilTotalPencairan =
Number(dataDonasi.totalPencairan) + Number(data.nominalCair); Number(dataDonasi.totalPencairan) + Number(nominalCair);
// const hasilAkumulasiPencairan = Number(dataDonasi.akumulasiPencairan) + 1; // const hasilAkumulasiPencairan = Number(dataDonasi.akumulasiPencairan) + 1;
const countPencairan = await prisma.donasi_PencairanDana.count({ const countPencairan = await prisma.donasi_PencairanDana.count({
@@ -66,8 +77,47 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
akumulasiPencairan: countPencairan, akumulasiPencairan: countPencairan,
totalPencairan: hasilTotalPencairan, 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) if (!updateDonasi)
return NextResponse.json( return NextResponse.json(
{ {
@@ -75,7 +125,7 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
message: "Update Donasi Gagal", message: "Update Donasi Gagal",
reason: "Update Donasi Gagal", reason: "Update Donasi Gagal",
}, },
{ status: 400 } { status: 400 },
); );
return NextResponse.json( return NextResponse.json(
@@ -84,7 +134,7 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
message: "Pencairan Dana Berhasil", message: "Pencairan Dana Berhasil",
// data: data, // data: data,
}, },
{ status: 200 } { status: 200 },
); );
} catch (error) { } catch (error) {
console.error("[ERROR]", error); console.error("[ERROR]", error);
@@ -94,7 +144,7 @@ async function POST(request: Request, { params }: { params: { id: string } }) {
message: "Pencairan Dana Gagal", message: "Pencairan Dana Gagal",
reason: (error as Error).message, 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); console.log("[CATEGORY]", category);
let fixData; let fixData;
try { try {
if (category === "get-all") { if (category === "get-all") {
fixData = await prisma.donasi_PencairanDana.findMany({ fixData = await prisma.donasi_PencairanDana.findMany({
take: page ? takeData : undefined, take: page ? takeData : undefined,
@@ -140,7 +189,7 @@ async function GET(request: Request, { params }: { params: { id: string } }) {
message: "Category tidak ditemukan", message: "Category tidak ditemukan",
reason: "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", message: "Success get data disbursement",
data: fixData, data: fixData,
}, },
{ status: 200 } { status: 200 },
); );
} catch (error) { } catch (error) {
console.error("[ERROR]", error); console.error("[ERROR]", error);
@@ -160,7 +209,7 @@ async function GET(request: Request, { params }: { params: { id: string } }) {
message: "Gagal mendapatkan data disbursement", message: "Gagal mendapatkan data disbursement",
reason: (error as Error).message, reason: (error as Error).message,
}, },
{ status: 500 } { status: 500 },
); );
} }
} }

View File

@@ -1,4 +1,5 @@
import { prisma } from "@/lib"; import { prisma } from "@/lib";
import { funSendToWhatsApp } from "@/lib/code-otp-sender";
import _ from "lodash"; import _ from "lodash";
import { NextResponse } from "next/server"; import { NextResponse } from "next/server";
@@ -50,8 +51,25 @@ async function PUT(request: Request, { params }: { params: { id: string } }) {
data: { data: {
active: data.active, 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); console.log("[Update Active Berhasil]", updateData);
} else if (category === "role") { } else if (category === "role") {
const fixName = _.startCase(data.role.replace(/_/g, " ")); const fixName = _.startCase(data.role.replace(/_/g, " "));

View File

@@ -1,25 +1,39 @@
import { NextRequest, NextResponse } from "next/server"; import { NextRequest, NextResponse } from "next/server";
import { prisma } from "@/lib"; import { prisma } from "@/lib";
import _ from "lodash"; 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 }; export { POST, GET, PUT, DELETE };
async function POST( async function POST(
request: NextRequest, request: NextRequest,
{ params }: { params: { id: string } } { params }: { params: { id: string } },
) { ) {
const { id } = params; const { id } = params;
const { data } = await request.json(); const { data } = await request.json();
const { title, deskripsi, imageId } = data;
const senderId = await prisma.donasi.findUnique({
where: { id: id },
select: {
authorId: true,
},
});
try { try {
if (data && data?.imageId) { if (data && data?.imageId) {
const createWithFile = await prisma.donasi_Kabar.create({ const createWithFile = await prisma.donasi_Kabar.create({
data: { data: {
title: data.title, title: title,
deskripsi: data.deskripsi, deskripsi: deskripsi,
donasiId: id, donasiId: id,
imageId: data.imageId, imageId: imageId,
}, },
}); });
@@ -28,8 +42,8 @@ async function POST(
} else { } else {
const create = await prisma.donasi_Kabar.create({ const create = await prisma.donasi_Kabar.create({
data: { data: {
title: data.title, title: title,
deskripsi: data.deskripsi, deskripsi: deskripsi,
donasiId: id, donasiId: id,
}, },
}); });
@@ -38,6 +52,25 @@ async function POST(
return NextResponse.json({ status: 400, message: "Gagal disimpan" }); 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({ return NextResponse.json({
status: 200, status: 200,
success: true, success: true,
@@ -56,7 +89,7 @@ async function POST(
async function GET( async function GET(
request: NextRequest, request: NextRequest,
{ params }: { params: { id: string } } { params }: { params: { id: string } },
) { ) {
const { id } = params; const { id } = params;
const { searchParams } = new URL(request.url); const { searchParams } = new URL(request.url);
@@ -178,7 +211,7 @@ async function PUT(request: Request, { params }: { params: { id: string } }) {
async function DELETE( async function DELETE(
request: Request, request: Request,
{ params }: { params: { id: string } } { params }: { params: { id: string } },
) { ) {
const { id } = params; const { id } = params;
try { try {
@@ -198,7 +231,7 @@ async function DELETE(
headers: { headers: {
Authorization: `Bearer ${process.env.WS_APIKEY}`, Authorization: `Bearer ${process.env.WS_APIKEY}`,
}, },
} },
); );
if (!deleteImage) { if (!deleteImage) {

View File

@@ -5,7 +5,7 @@ import { NextResponse } from "next/server";
import { NotificationMobileBodyType } from "../../../../../types/type-mobile-notification"; import { NotificationMobileBodyType } from "../../../../../types/type-mobile-notification";
import { routeAdminMobile } from "@/lib/mobile/route-page-mobile"; import { routeAdminMobile } from "@/lib/mobile/route-page-mobile";
export { POST }; export { POST, GET };
async function POST(request: Request) { async function POST(request: Request) {
const { data } = await request.json(); const { data } = await request.json();
@@ -121,7 +121,7 @@ async function POST(request: Request) {
} }
// GET ALL DATA DONASI // GET ALL DATA DONASI
export async function GET(request: Request) { async function GET(request: Request) {
const { searchParams } = new URL(request.url); const { searchParams } = new URL(request.url);
const category = searchParams.get("category"); const category = searchParams.get("category");
const authorId = searchParams.get("authorId"); 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 } }) { async function GET(request: Request, { params }: { params: { id: string } }) {
const { id } = params; const { id } = params;
const { searchParams } = new URL(request.url);
const page = Number(searchParams.get("page"));
const takeData = 5
const skipData = page * takeData - takeData;
try { try {
const data = await prisma.forum_Komentar.findMany({ const data = await prisma.forum_Komentar.findMany({
take: page ? takeData : undefined,
skip: page ? skipData : undefined,
orderBy: { orderBy: {
createdAt: "desc", createdAt: "desc",
}, },

View File

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

View File

@@ -1,13 +1,16 @@
const sendCodeOtp = async ({ const sendCodeOtp = async ({
nomor, nomor,
codeOtp, codeOtp,
newMessage,
}: { }: {
nomor: string; 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( 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", cache: "no-cache",
headers: { headers: {
@@ -25,4 +28,4 @@ const sendCodeOtp = async ({
return res; 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;
};