Merge pull request #467 from bipproduction/amalia/26-jun-25
Amalia/26 jun 25
This commit is contained in:
@@ -39,8 +39,10 @@
|
|||||||
"@tiptap/pm": "^2.11.7",
|
"@tiptap/pm": "^2.11.7",
|
||||||
"@tiptap/react": "^2.4.0",
|
"@tiptap/react": "^2.4.0",
|
||||||
"@tiptap/starter-kit": "^2.4.0",
|
"@tiptap/starter-kit": "^2.4.0",
|
||||||
|
"@types/bun": "^1.2.17",
|
||||||
"@types/busboy": "^1.5.4",
|
"@types/busboy": "^1.5.4",
|
||||||
"@types/lodash": "^4.17.6",
|
"@types/lodash": "^4.17.6",
|
||||||
|
"@types/minimist": "^1.2.5",
|
||||||
"@types/multer": "^1.4.12",
|
"@types/multer": "^1.4.12",
|
||||||
"@types/web-push": "^3.6.3",
|
"@types/web-push": "^3.6.3",
|
||||||
"busboy": "^1.6.0",
|
"busboy": "^1.6.0",
|
||||||
@@ -56,6 +58,7 @@
|
|||||||
"iron-session": "^8.0.2",
|
"iron-session": "^8.0.2",
|
||||||
"jose": "^5.9.2",
|
"jose": "^5.9.2",
|
||||||
"lodash": "^4.17.21",
|
"lodash": "^4.17.21",
|
||||||
|
"minimist": "^1.2.8",
|
||||||
"moment": "^2.30.1",
|
"moment": "^2.30.1",
|
||||||
"multer": "^1.4.5-lts.2",
|
"multer": "^1.4.5-lts.2",
|
||||||
"next": "14.2.4",
|
"next": "14.2.4",
|
||||||
@@ -71,6 +74,7 @@
|
|||||||
"rrule": "^2.8.1",
|
"rrule": "^2.8.1",
|
||||||
"sharp": "^0.33.5",
|
"sharp": "^0.33.5",
|
||||||
"supabase": "^1.192.5",
|
"supabase": "^1.192.5",
|
||||||
|
"tsx": "^4.20.3",
|
||||||
"web-push": "^3.6.7",
|
"web-push": "^3.6.7",
|
||||||
"wibu-cli": "^1.0.91",
|
"wibu-cli": "^1.0.91",
|
||||||
"wibu-pkg": "^1.0.63",
|
"wibu-pkg": "^1.0.63",
|
||||||
|
|||||||
@@ -1,8 +0,0 @@
|
|||||||
import { NextResponse } from "next/server";
|
|
||||||
import { sendFCM } from "../../../../xfcm/send";
|
|
||||||
|
|
||||||
export const GET = () => {
|
|
||||||
const token = "cRz96GHKTRaQaRJ35e8Hxa:APA91bEUSxE0VPbqKSzseQ_zGhbYsDofMexKykRw7o_3z2aPM9YFmZbeA2enrmb3qjdZ2g4-QQtiNHAyaZqAT1ITOrwo9jVJlShTeABmEFYP5GLEUZ3dlLc";
|
|
||||||
sendFCM(token, "Test dari Local", "Ini hanya percobaan notifikasi dari script.");
|
|
||||||
return NextResponse.json({ success: true, message: "Sukses" }, { status: 200 });
|
|
||||||
}
|
|
||||||
21
src/app/api/mobile/fcm/[[...slug]]/route.ts
Normal file
21
src/app/api/mobile/fcm/[[...slug]]/route.ts
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
import { sendFCM } from "@/lib/firebase/fcm";
|
||||||
|
import elysia from "elysia";
|
||||||
|
|
||||||
|
|
||||||
|
const ApiV2 = new elysia({
|
||||||
|
prefix: "/api/mobile/fcm"
|
||||||
|
})
|
||||||
|
.get("/", async () => {
|
||||||
|
const token = [
|
||||||
|
'c89yuexsS_uc1tOErVPu5a:APA91bEb6tEKXAfReZjFVJ2mMyOzoW_RXryLSnSJTpbIVV3G0L_DCNkLuRvJ02Ip-Erz88QCQBAt-C2SN8eCRxu3-v1sBzXzKPtDv-huXpkjXsyrkifqvUo',
|
||||||
|
'cRz96GHKTRaQaRJ35e8Hxa:APA91bEUSxE0VPbqKSzseQ_zGhbYsDofMexKykRw7o_3z2aPM9YFmZbeA2enrmb3qjdZ2g4-QQtiNHAyaZqAT1ITOrwo9jVJlShTeABmEFYP5GLEUZ3dlLc'
|
||||||
|
]
|
||||||
|
|
||||||
|
sendFCM(token, "Test dari Production STG", "Ini hanya percobaan notifikasi dari script.");
|
||||||
|
return {
|
||||||
|
data: "success elysia api"
|
||||||
|
};
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
export const GET = ApiV2.handle;
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import { prisma } from "@/module/_global";
|
import { prisma } from "@/module/_global";
|
||||||
import { funGetUserByCookies } from "@/module/auth";
|
import { funGetUserById } from "@/module/auth";
|
||||||
import { createLogUser } from "@/module/user";
|
import { createLogUserMobile } from "@/module/user";
|
||||||
import _ from "lodash";
|
import _ from "lodash";
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
import "moment/locale/id";
|
import "moment/locale/id";
|
||||||
@@ -9,21 +9,22 @@ import { NextResponse } from "next/server";
|
|||||||
// GET ALL NOTIFIKASI
|
// GET ALL NOTIFIKASI
|
||||||
export async function GET(request: Request) {
|
export async function GET(request: Request) {
|
||||||
try {
|
try {
|
||||||
const user = await funGetUserByCookies();
|
|
||||||
if (user.id == undefined) {
|
|
||||||
return NextResponse.json({ success: false, message: "Anda harus login untuk mengakses ini" }, { status: 401 });
|
|
||||||
}
|
|
||||||
|
|
||||||
const { searchParams } = new URL(request.url);
|
const { searchParams } = new URL(request.url);
|
||||||
|
const user = searchParams.get('user');
|
||||||
const page = searchParams.get('page');
|
const page = searchParams.get('page');
|
||||||
const dataSkip = Number(page) * 10 - 10;
|
const dataSkip = Number(page) * 10 - 10;
|
||||||
|
|
||||||
|
const userMobile = await funGetUserById({ id: String(user) });
|
||||||
|
if (userMobile.id == "null" || userMobile.id == undefined || userMobile.id == "") {
|
||||||
|
return NextResponse.json({ success: false, message: "Anda harus login untuk mengakses ini" }, { status: 200 });
|
||||||
|
}
|
||||||
|
|
||||||
const announcements = await prisma.notifications.findMany({
|
const announcements = await prisma.notifications.findMany({
|
||||||
skip: dataSkip,
|
skip: dataSkip,
|
||||||
take: 10,
|
take: 10,
|
||||||
where: {
|
where: {
|
||||||
isActive: true,
|
isActive: true,
|
||||||
idUserTo: user.id
|
idUserTo: userMobile.id
|
||||||
},
|
},
|
||||||
orderBy: [
|
orderBy: [
|
||||||
{
|
{
|
||||||
@@ -53,11 +54,13 @@ export async function GET(request: Request) {
|
|||||||
// UPDATE READ NOTIFIKASI
|
// UPDATE READ NOTIFIKASI
|
||||||
export async function PUT(request: Request, context: { params: { id: string } }) {
|
export async function PUT(request: Request, context: { params: { id: string } }) {
|
||||||
try {
|
try {
|
||||||
const user = await funGetUserByCookies()
|
const { id, user } = await request.json();
|
||||||
if (user.id == undefined) {
|
const userMobile = await funGetUserById({ id: String(user) });
|
||||||
return NextResponse.json({ success: false, message: "Anda harus login untuk mengakses ini" }, { status: 401 });
|
|
||||||
|
if (userMobile.id == "null" || userMobile.id == undefined || userMobile.id == "") {
|
||||||
|
return NextResponse.json({ success: false, message: "Anda harus login untuk mengakses ini" }, { status: 200 });
|
||||||
}
|
}
|
||||||
const { id } = await request.json();
|
|
||||||
const data = await prisma.notifications.count({
|
const data = await prisma.notifications.count({
|
||||||
where: {
|
where: {
|
||||||
id: id,
|
id: id,
|
||||||
@@ -70,7 +73,7 @@ export async function PUT(request: Request, context: { params: { id: string } })
|
|||||||
success: false,
|
success: false,
|
||||||
message: "Gagal mendapatkan data, data tidak ditemukan",
|
message: "Gagal mendapatkan data, data tidak ditemukan",
|
||||||
},
|
},
|
||||||
{ status: 404 }
|
{ status: 200 }
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -84,13 +87,13 @@ export async function PUT(request: Request, context: { params: { id: string } })
|
|||||||
});
|
});
|
||||||
|
|
||||||
// create log user
|
// create log user
|
||||||
const log = await createLogUser({ act: 'UPDATE', desc: 'User membaca notifikasi', table: 'notifications', data: id })
|
const log = await createLogUserMobile({ act: 'UPDATE', desc: 'User membaca notifikasi', table: 'notifications', data: id, user: userMobile.id })
|
||||||
|
|
||||||
return NextResponse.json({ success: true, message: "Berhasil mendapatkan notifikasi", }, { status: 200 });
|
return NextResponse.json({ success: true, message: "Berhasil membaca notifikasi", }, { status: 200 });
|
||||||
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
return NextResponse.json({ success: false, message: "Gagal mendapatkan notifikasi, coba lagi nanti (error: 500)", reason: (error as Error).message, }, { status: 500 });
|
return NextResponse.json({ success: false, message: "Gagal membaca notifikasi, coba lagi nanti (error: 500)", reason: (error as Error).message, }, { status: 500 });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -98,14 +101,16 @@ export async function PUT(request: Request, context: { params: { id: string } })
|
|||||||
// UPDATE READ ALL NOTIFICATION
|
// UPDATE READ ALL NOTIFICATION
|
||||||
export async function POST(request: Request) {
|
export async function POST(request: Request) {
|
||||||
try {
|
try {
|
||||||
const user = await funGetUserByCookies()
|
const { user } = await request.json();
|
||||||
if (user.id == undefined) {
|
const userMobile = await funGetUserById({ id: String(user) });
|
||||||
return NextResponse.json({ success: false, message: "Anda harus login untuk mengakses ini" }, { status: 401 });
|
|
||||||
|
if (userMobile.id == "null" || userMobile.id == undefined || userMobile.id == "") {
|
||||||
|
return NextResponse.json({ success: false, message: "Anda harus login untuk mengakses ini" }, { status: 200 });
|
||||||
}
|
}
|
||||||
|
|
||||||
const upd = await prisma.notifications.updateMany({
|
const upd = await prisma.notifications.updateMany({
|
||||||
where: {
|
where: {
|
||||||
idUserTo: user.id,
|
idUserTo: userMobile.id,
|
||||||
isRead: false
|
isRead: false
|
||||||
},
|
},
|
||||||
data: {
|
data: {
|
||||||
@@ -114,7 +119,7 @@ export async function POST(request: Request) {
|
|||||||
})
|
})
|
||||||
|
|
||||||
// create log user
|
// create log user
|
||||||
const log = await createLogUser({ act: 'UPDATE', desc: 'User menandai semua notifikasi', table: 'notifications', data: '' })
|
const log = await createLogUserMobile({ act: 'UPDATE', desc: 'User menandai semua notifikasi', table: 'notifications', data: '', user: userMobile.id })
|
||||||
|
|
||||||
return NextResponse.json({ success: true, message: "Berhasil mengupdate notifikasi", }, { status: 200 });
|
return NextResponse.json({ success: true, message: "Berhasil mengupdate notifikasi", }, { status: 200 });
|
||||||
|
|
||||||
|
|||||||
@@ -1,75 +0,0 @@
|
|||||||
import prisma from '@/lib/prisma';
|
|
||||||
import admin from 'firebase-admin';
|
|
||||||
import { NextResponse } from "next/server";
|
|
||||||
|
|
||||||
export async function GET(request: Request) {
|
|
||||||
try {
|
|
||||||
if (!admin.apps.length) {
|
|
||||||
const serviceAccount = require("../../../../service_key.json");
|
|
||||||
admin.initializeApp({
|
|
||||||
credential: admin.credential.cert(serviceAccount),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const token = [
|
|
||||||
// 'fPfBeTn4R4KsHhDw-8Edj0:APA91bGBjguuZsMhisJua_Wa3m7z7vBCE08vjyDBScmN0eIRgpfINlDx4SI6-upn-rr6tTAcoPxeQkxxbEsohcUbbV5DjUyIG4xR6wuvKOJMp3Mr4rNWFv8',
|
|
||||||
// 'c89yuexsS_uc1tOErVPu5a:APA91bEb6tEKXAfReZjFVJ2mMyOzoW_RXryLSnSJTpbIVV3G0L_DCNkLuRvJ02Ip-Erz88QCQBAt-C2SN8eCRxu3-v1sBzXzKPtDv-huXpkjXsyrkifqvUo',
|
|
||||||
'cRz96GHKTRaQaRJ35e8Hxa:APA91bEUSxE0VPbqKSzseQ_zGhbYsDofMexKykRw7o_3z2aPM9YFmZbeA2enrmb3qjdZ2g4-QQtiNHAyaZqAT1ITOrwo9jVJlShTeABmEFYP5GLEUZ3dlLc'
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
const message = {
|
|
||||||
notification: {
|
|
||||||
title: 'New Update :)',
|
|
||||||
body: 'Check out the latest features in our app.',
|
|
||||||
},
|
|
||||||
data: {
|
|
||||||
category: 'value1',
|
|
||||||
content: 'value2',
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
admin.messaging().sendEachForMulticast({
|
|
||||||
tokens: token,
|
|
||||||
...message,
|
|
||||||
})
|
|
||||||
.then(async (response) => {
|
|
||||||
// Handle successful sends and failed tokens
|
|
||||||
if (response.failureCount > 0) {
|
|
||||||
const failedTokens: any[] = [];
|
|
||||||
response.responses.forEach((resp, idx) => {
|
|
||||||
if (!resp.success) {
|
|
||||||
failedTokens.push(token[idx]);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
const deleteToken = await prisma.tokenDeviceUser.deleteMany({
|
|
||||||
where: {
|
|
||||||
token: {
|
|
||||||
in: failedTokens
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
console.error('Error sending message:', error);
|
|
||||||
});
|
|
||||||
|
|
||||||
// const message = {
|
|
||||||
// notification: {
|
|
||||||
// title: 'Dari API Localhost Coba',
|
|
||||||
// body: 'Hello World Coba',
|
|
||||||
// },
|
|
||||||
|
|
||||||
// token: 'c89yuexsS_uc1tOErVPu5a:APA91bEb6tEKXAfReZjFVJ2mMyOzoW_RXryLSnSJTpbIVV3G0L_DCNkLuRvJ02Ip-Erz88QCQBAt-C2SN8eCRxu3-v1sBzXzKPtDv-huXpkjXsyrkifqvUo',
|
|
||||||
|
|
||||||
// };
|
|
||||||
|
|
||||||
// const response = await admin.messaging().send(message);
|
|
||||||
|
|
||||||
return NextResponse.json({ success: true, message: `Sukses ${admin.apps.length}` }, { status: 200 });
|
|
||||||
} catch (error) {
|
|
||||||
console.error(error);
|
|
||||||
return NextResponse.json({ success: false, message: "Gagal", reason: (error as Error).message, }, { status: 500 });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -42,6 +42,11 @@ export async function GET(request: Request, context: { params: { id: string } })
|
|||||||
id: true
|
id: true
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
Village:{
|
||||||
|
select:{
|
||||||
|
name:true
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -51,10 +56,11 @@ export async function GET(request: Request, context: { params: { id: string } })
|
|||||||
const idUserRole = users?.UserRole.id
|
const idUserRole = users?.UserRole.id
|
||||||
const phone = users?.phone.substr(2)
|
const phone = users?.phone.substr(2)
|
||||||
const role = users?.UserRole.name
|
const role = users?.UserRole.name
|
||||||
|
const village = users?.Village.name
|
||||||
|
|
||||||
const result = { ...userData, group, position, idUserRole, phone, role };
|
const result = { ...userData, group, position, idUserRole, phone, role, village };
|
||||||
|
|
||||||
const omitData = _.omit(result, ["Group", "Position", "UserRole"]);
|
const omitData = _.omit(result, ["Group", "Position", "UserRole", "Village"]);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import { NextResponse } from "next/server";
|
|||||||
|
|
||||||
export async function GET(request: Request) {
|
export async function GET(request: Request) {
|
||||||
try {
|
try {
|
||||||
return NextResponse.json({ success: true, version: "1.5.0", tahap: "beta", update: "- update api mobile; - firebase admin" }, { status: 200 });
|
return NextResponse.json({ success: true, version: "1.5.1", tahap: "beta", update: "-percobaan firebase admin fcm" }, { status: 200 });
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
return NextResponse.json({ success: false, version: "Gagal mendapatkan version, coba lagi nanti (error: 500)", reason: (error as Error).message, }, { status: 500 });
|
return NextResponse.json({ success: false, version: "Gagal mendapatkan version, coba lagi nanti (error: 500)", reason: (error as Error).message, }, { status: 500 });
|
||||||
|
|||||||
66
src/lib/firebase/fcm.ts
Normal file
66
src/lib/firebase/fcm.ts
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
import admin from "firebase-admin";
|
||||||
|
|
||||||
|
// Inisialisasi admin (hanya sekali)
|
||||||
|
if (!admin.apps.length) {
|
||||||
|
admin.initializeApp({
|
||||||
|
credential: admin.credential.cert({
|
||||||
|
projectId: process.env.GOOGLE_PROJECT_ID,
|
||||||
|
privateKey: process.env.GOOGLE_PRIVATE_KEY?.replace(/\\n/g, '\n'),
|
||||||
|
clientEmail: process.env.GOOGLE_CLIENT_EMAIL,
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Kirim push notifikasi ke FCM token
|
||||||
|
*/
|
||||||
|
export async function sendFCM(token: string[], title: string, body: string) {
|
||||||
|
const message = {
|
||||||
|
notification: {
|
||||||
|
title,
|
||||||
|
body,
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
id: 'cm1eqwkwj00067j2bzejyh9u7',
|
||||||
|
category: 'division',
|
||||||
|
content: 'cm1eqwkvu00017j2b8qnisp9g',
|
||||||
|
},
|
||||||
|
apns: {
|
||||||
|
payload: {
|
||||||
|
aps: {
|
||||||
|
contentAvailable: true,
|
||||||
|
alert: {
|
||||||
|
title,
|
||||||
|
body,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
token,
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await admin.messaging().sendEachForMulticast({
|
||||||
|
tokens: token,
|
||||||
|
...message,
|
||||||
|
})
|
||||||
|
console.log("✅ FCM berhasil dikirim:", result);
|
||||||
|
if (result.failureCount > 0) {
|
||||||
|
const failedTokens: any[] = [];
|
||||||
|
result.responses.forEach((resp, idx) => {
|
||||||
|
if (!resp.success) {
|
||||||
|
failedTokens.push(token[idx]);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// const deleteToken = await prisma.tokenDeviceUser.deleteMany({
|
||||||
|
// where: {
|
||||||
|
// token: {
|
||||||
|
// in: failedTokens
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// });
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
console.error("❌ Gagal mengirim FCM:", err);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
import admin from "firebase-admin";
|
|
||||||
|
|
||||||
if (!admin.apps.length) {
|
|
||||||
admin.initializeApp({
|
|
||||||
credential: admin.credential.cert({
|
|
||||||
projectId: process.env.GOOGLE_PROJECT_ID,
|
|
||||||
privateKey: process.env.GOOGLE_PRIVATE_KEY?.replace(/\\n/g, '\n'),
|
|
||||||
clientEmail: process.env.GOOGLE_CLIENT_EMAIL,
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export default admin;
|
|
||||||
38
xfcm/send.ts
38
xfcm/send.ts
@@ -1,38 +0,0 @@
|
|||||||
import admin from "firebase-admin";
|
|
||||||
|
|
||||||
// Inisialisasi admin (hanya sekali)
|
|
||||||
if (!admin.apps.length) {
|
|
||||||
admin.initializeApp({
|
|
||||||
credential: admin.credential.cert({
|
|
||||||
projectId: process.env.GOOGLE_PROJECT_ID,
|
|
||||||
privateKey: process.env.GOOGLE_PRIVATE_KEY?.replace(/\\n/g, '\n'),
|
|
||||||
clientEmail: process.env.GOOGLE_CLIENT_EMAIL,
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Kirim push notifikasi ke FCM token
|
|
||||||
*/
|
|
||||||
export async function sendFCM(token: string, title: string, body: string) {
|
|
||||||
const message = {
|
|
||||||
token,
|
|
||||||
notification: {
|
|
||||||
title,
|
|
||||||
body,
|
|
||||||
},
|
|
||||||
data: {
|
|
||||||
tested: "true"
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
try {
|
|
||||||
const result = await admin.messaging().send(message);
|
|
||||||
console.log("✅ FCM berhasil dikirim:", result);
|
|
||||||
} catch (err) {
|
|
||||||
console.error("❌ Gagal mengirim FCM:", err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// const token = "cRz96GHKTRaQaRJ35e8Hxa:APA91bEUSxE0VPbqKSzseQ_zGhbYsDofMexKykRw7o_3z2aPM9YFmZbeA2enrmb3qjdZ2g4-QQtiNHAyaZqAT1ITOrwo9jVJlShTeABmEFYP5GLEUZ3dlLc";
|
|
||||||
// sendFCM(token, "Test dari Local", "Ini hanya percobaan notifikasi dari script.");
|
|
||||||
Reference in New Issue
Block a user