Merge pull request 'upd: api noc' (#25) from amalia/12-mar-26 into join
Reviewed-on: #25
This commit is contained in:
@@ -1,7 +1,8 @@
|
|||||||
|
import { prisma } from "@/module/_global";
|
||||||
import cors from "@elysiajs/cors";
|
import cors from "@elysiajs/cors";
|
||||||
import { swagger } from "@elysiajs/swagger";
|
import { swagger } from "@elysiajs/swagger";
|
||||||
import Elysia, { t } from "elysia";
|
import Elysia, { t } from "elysia";
|
||||||
import { prisma } from "@/module/_global";
|
import _ from "lodash";
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
import "moment/locale/id";
|
import "moment/locale/id";
|
||||||
|
|
||||||
@@ -231,141 +232,11 @@ const NocServer = new Elysia({ prefix: "/api/noc" })
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
// ── GET /api/noc/village-summary ───────────────────────────────────────────
|
|
||||||
.get(
|
|
||||||
"/village-summary",
|
|
||||||
async ({ query, set }) => {
|
|
||||||
const { idDesa } = query;
|
|
||||||
|
|
||||||
if (!idDesa) {
|
|
||||||
set.status = 400;
|
|
||||||
return { success: false, message: "Parameter idDesa wajib diisi", data: null };
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const counts = await prisma.village.findUnique({
|
|
||||||
where: { id: idDesa },
|
|
||||||
select: {
|
|
||||||
name: true,
|
|
||||||
_count: {
|
|
||||||
select: {
|
|
||||||
User: true,
|
|
||||||
Group: true,
|
|
||||||
Division: true,
|
|
||||||
Project: true,
|
|
||||||
Announcement: true,
|
|
||||||
Discussion: true,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!counts) {
|
|
||||||
set.status = 404;
|
|
||||||
return { success: false, message: "Desa tidak ditemukan", data: null };
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
success: true,
|
|
||||||
message: "Berhasil mendapatkan ringkasan desa",
|
|
||||||
data: {
|
|
||||||
idDesa,
|
|
||||||
namaDesa: counts.name,
|
|
||||||
summary: {
|
|
||||||
totalWarga: counts._count.User,
|
|
||||||
totalGrup: counts._count.Group,
|
|
||||||
totalDivisi: counts._count.Division,
|
|
||||||
totalProyek: counts._count.Project,
|
|
||||||
totalPengumuman: counts._count.Announcement,
|
|
||||||
totalDiskusi: counts._count.Discussion,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
} catch (error) {
|
|
||||||
console.error("[NOC] village-summary error:", error);
|
|
||||||
set.status = 500;
|
|
||||||
return { success: false, message: "Terjadi kesalahan pada server", data: null };
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
query: t.Object({ idDesa: t.String() }),
|
|
||||||
detail: { summary: "Village Summary Statistics", tags: ["NOC"] }
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
// ── GET /api/noc/recent-activity ───────────────────────────────────────────
|
|
||||||
.get(
|
|
||||||
"/recent-activity",
|
|
||||||
async ({ query, set }) => {
|
|
||||||
const { idDesa, limit } = query;
|
|
||||||
|
|
||||||
if (!idDesa) {
|
|
||||||
set.status = 400;
|
|
||||||
return { success: false, message: "Parameter idDesa wajib diisi", data: null };
|
|
||||||
}
|
|
||||||
|
|
||||||
const maxResults = Math.min(Number(limit ?? 10), 50);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const logs = await prisma.userLog.findMany({
|
|
||||||
where: {
|
|
||||||
User: {
|
|
||||||
idVillage: idDesa
|
|
||||||
}
|
|
||||||
},
|
|
||||||
select: {
|
|
||||||
id: true,
|
|
||||||
action: true,
|
|
||||||
desc: true,
|
|
||||||
createdAt: true,
|
|
||||||
User: {
|
|
||||||
select: {
|
|
||||||
name: true,
|
|
||||||
img: true,
|
|
||||||
Group: { select: { name: true } }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
orderBy: { createdAt: "desc" },
|
|
||||||
take: maxResults
|
|
||||||
});
|
|
||||||
|
|
||||||
const mapped = logs.map(l => ({
|
|
||||||
id: l.id,
|
|
||||||
userName: l.User.name,
|
|
||||||
userGroup: l.User.Group.name,
|
|
||||||
userImg: l.User.img,
|
|
||||||
action: l.action,
|
|
||||||
description: l.desc,
|
|
||||||
time: moment(l.createdAt).fromNow(),
|
|
||||||
date: moment(l.createdAt).format("YYYY-MM-DD HH:mm:ss")
|
|
||||||
}));
|
|
||||||
|
|
||||||
return {
|
|
||||||
success: true,
|
|
||||||
message: "Berhasil mendapatkan aktivitas terbaru",
|
|
||||||
data: { idDesa, total: mapped.length, activities: mapped }
|
|
||||||
};
|
|
||||||
} catch (error) {
|
|
||||||
console.error("[NOC] recent-activity error:", error);
|
|
||||||
set.status = 500;
|
|
||||||
return { success: false, message: "Terjadi kesalahan pada server", data: null };
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
query: t.Object({
|
|
||||||
idDesa: t.String(),
|
|
||||||
limit: t.Optional(t.String())
|
|
||||||
}),
|
|
||||||
detail: { summary: "Recent User Activity Logs", tags: ["NOC"] }
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
// ── GET /api/noc/upcoming-events ───────────────────────────────────────────
|
// ── GET /api/noc/upcoming-events ───────────────────────────────────────────
|
||||||
.get(
|
.get(
|
||||||
"/upcoming-events",
|
"/upcoming-events",
|
||||||
async ({ query, set }) => {
|
async ({ query, set }) => {
|
||||||
const { idDesa, limit } = query;
|
const { idDesa, limit, filter } = query;
|
||||||
|
|
||||||
if (!idDesa) {
|
if (!idDesa) {
|
||||||
set.status = 400;
|
set.status = 400;
|
||||||
@@ -443,7 +314,8 @@ const NocServer = new Elysia({ prefix: "/api/noc" })
|
|||||||
take: maxResults,
|
take: maxResults,
|
||||||
});
|
});
|
||||||
|
|
||||||
const mapped = events.map((e) => ({
|
const todayMoment = moment().startOf("day");
|
||||||
|
const mapper = (e: any) => ({
|
||||||
id: e.id,
|
id: e.id,
|
||||||
idCalendar: e.idCalendar,
|
idCalendar: e.idCalendar,
|
||||||
title: e.DivisionCalendar.title,
|
title: e.DivisionCalendar.title,
|
||||||
@@ -462,17 +334,29 @@ const NocServer = new Elysia({ prefix: "/api/noc" })
|
|||||||
id: e.Division.id,
|
id: e.Division.id,
|
||||||
name: e.Division.name,
|
name: e.Division.name,
|
||||||
},
|
},
|
||||||
}));
|
});
|
||||||
|
|
||||||
|
const todayEvents = events.filter(e => moment(e.dateStart).isSame(todayMoment, 'day')).map(mapper);
|
||||||
|
const upcomingEvents = events.filter(e => moment(e.dateStart).isAfter(todayMoment, 'day')).map(mapper);
|
||||||
|
|
||||||
|
let data: any = {
|
||||||
|
idDesa: village.id,
|
||||||
|
namaDesa: village.name,
|
||||||
|
};
|
||||||
|
|
||||||
|
if (filter === "today") {
|
||||||
|
data.events = todayEvents;
|
||||||
|
} else if (filter === "upcoming") {
|
||||||
|
data.events = upcomingEvents;
|
||||||
|
} else {
|
||||||
|
data.today = todayEvents;
|
||||||
|
data.upcoming = upcomingEvents;
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
success: true,
|
success: true,
|
||||||
message: "Berhasil mendapatkan upcoming events",
|
message: "Berhasil mendapatkan events",
|
||||||
data: {
|
data: data,
|
||||||
idDesa: village.id,
|
|
||||||
namaDesa: village.name,
|
|
||||||
total: mapped.length,
|
|
||||||
events: mapped,
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("[NOC] upcoming-events error:", error);
|
console.error("[NOC] upcoming-events error:", error);
|
||||||
@@ -490,14 +374,314 @@ const NocServer = new Elysia({ prefix: "/api/noc" })
|
|||||||
limit: t.Optional(
|
limit: t.Optional(
|
||||||
t.String({ description: "Jumlah maksimal event (default: 10, maks: 50)" })
|
t.String({ description: "Jumlah maksimal event (default: 10, maks: 50)" })
|
||||||
),
|
),
|
||||||
|
filter: t.Optional(
|
||||||
|
t.String({ description: "Filter event: 'today' atau 'upcoming'" })
|
||||||
|
),
|
||||||
}),
|
}),
|
||||||
detail: {
|
detail: {
|
||||||
summary: "Upcoming Events",
|
summary: "Events (Today & Upcoming)",
|
||||||
description: "Mendapatkan daftar event yang akan datang untuk semua divisi pada desa tertentu.",
|
description: "Mendapatkan daftar event pada hari ini dan yang akan datang untuk semua divisi pada desa tertentu.",
|
||||||
|
tags: ["NOC"],
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// ── GET /api/noc/diagram-jumlah-document ───────────────────────────────────────────────
|
||||||
|
.get(
|
||||||
|
"/diagram-jumlah-document",
|
||||||
|
async ({ query, set }) => {
|
||||||
|
const { idDesa } = query;
|
||||||
|
|
||||||
|
if (!idDesa) {
|
||||||
|
set.status = 400;
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "Parameter idDesa wajib diisi",
|
||||||
|
data: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const village = await prisma.village.findUnique({
|
||||||
|
where: { id: idDesa },
|
||||||
|
select: { id: true, name: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!village) {
|
||||||
|
set.status = 404;
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "Desa tidak ditemukan",
|
||||||
|
data: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const documents = await prisma.divisionDocumentFolderFile.findMany({
|
||||||
|
where: {
|
||||||
|
isActive: true,
|
||||||
|
category: 'FILE',
|
||||||
|
Division: {
|
||||||
|
isActive: true,
|
||||||
|
idVillage: idDesa,
|
||||||
|
Group: {
|
||||||
|
isActive: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const groupData = _.map(_.groupBy(documents, "extension"), (v: any) => ({
|
||||||
|
file: v[0].extension,
|
||||||
|
jumlah: v.length,
|
||||||
|
}))
|
||||||
|
|
||||||
|
const image = ['jpg', 'jpeg', 'png', 'heic']
|
||||||
|
|
||||||
|
|
||||||
|
let hasilImage = {
|
||||||
|
label: 'Gambar',
|
||||||
|
value: 0,
|
||||||
|
color: '#fac858'
|
||||||
|
}
|
||||||
|
|
||||||
|
let hasilFile = {
|
||||||
|
label: 'Dokumen',
|
||||||
|
value: 0,
|
||||||
|
color: '#92cc76'
|
||||||
|
}
|
||||||
|
|
||||||
|
groupData.map((v: any) => {
|
||||||
|
if (image.some((i: any) => i == v.file)) {
|
||||||
|
hasilImage = {
|
||||||
|
label: 'Gambar',
|
||||||
|
value: hasilImage.value + v.jumlah,
|
||||||
|
color: '#fac858'
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
hasilFile = {
|
||||||
|
label: 'Dokumen',
|
||||||
|
value: hasilFile.value + v.jumlah,
|
||||||
|
color: '#92cc76'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const allData = [hasilImage, hasilFile]
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
message: "Berhasil mendapatkan jumlah document",
|
||||||
|
data: allData
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error("[NOC] jumlah-document error:", error);
|
||||||
|
set.status = 500;
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "Terjadi kesalahan pada server",
|
||||||
|
data: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
query: t.Object({
|
||||||
|
idDesa: t.String({ description: "ID Desa yang ingin dicari" }),
|
||||||
|
}),
|
||||||
|
detail: {
|
||||||
|
summary: "Diagram Jumlah Document",
|
||||||
|
description: "Mendapatkan diagram jumlah document pada desa tertentu.",
|
||||||
|
tags: ["NOC"],
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// -- GET /api/noc/diagram-progres-kegiatan
|
||||||
|
.get(
|
||||||
|
"/diagram-progres-kegiatan",
|
||||||
|
async ({ query, set }) => {
|
||||||
|
const { idDesa } = query;
|
||||||
|
|
||||||
|
if (!idDesa) {
|
||||||
|
set.status = 400;
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "Parameter idDesa wajib diisi",
|
||||||
|
data: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const village = await prisma.village.findUnique({
|
||||||
|
where: { id: idDesa },
|
||||||
|
select: { id: true, name: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!village) {
|
||||||
|
set.status = 404;
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "Desa tidak ditemukan",
|
||||||
|
data: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await prisma.project.groupBy({
|
||||||
|
where: {
|
||||||
|
isActive: true,
|
||||||
|
idVillage: idDesa,
|
||||||
|
Group: {
|
||||||
|
isActive: true,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
by: ["status"],
|
||||||
|
_count: true
|
||||||
|
})
|
||||||
|
|
||||||
|
const dataStatus = [{ name: 'Segera dikerjakan', status: 0, color: '#177AD5' }, { name: 'Dikerjakan', status: 1, color: '#fac858' }, { name: 'Selesai dikerjakan', status: 2, color: '#92cc76' }, { name: 'Dibatalkan', status: 3, color: '#ED6665' }]
|
||||||
|
const hasil: any[] = []
|
||||||
|
let input
|
||||||
|
for (let index = 0; index < dataStatus.length; index++) {
|
||||||
|
const cek = data.some((i: any) => i.status == dataStatus[index].status)
|
||||||
|
if (cek) {
|
||||||
|
const find = ((Number(data.find((i: any) => i.status == dataStatus[index].status)?._count) * 100) / data.reduce((n, { _count }) => n + _count, 0)).toFixed(2)
|
||||||
|
const fix = find != "100.00" ? find.substr(-2, 2) == "00" ? find.substr(0, 2) : find : "100"
|
||||||
|
input = {
|
||||||
|
text: fix + '%',
|
||||||
|
value: fix,
|
||||||
|
color: dataStatus[index].color
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
input = {
|
||||||
|
text: '0%',
|
||||||
|
value: 0,
|
||||||
|
color: dataStatus[index].color
|
||||||
|
}
|
||||||
|
}
|
||||||
|
hasil.push(input)
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
message: "Berhasil mendapatkan progres kegiatan",
|
||||||
|
data: hasil
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error("[NOC] progres-kegiatan error:", error);
|
||||||
|
set.status = 500;
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "Terjadi kesalahan pada server",
|
||||||
|
data: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
query: t.Object({
|
||||||
|
idDesa: t.String({ description: "ID Desa yang ingin dicari" }),
|
||||||
|
}),
|
||||||
|
detail: {
|
||||||
|
summary: "Diagram Progres Kegiatan",
|
||||||
|
description: "Mendapatkan diagram progres kegiatan pada desa tertentu.",
|
||||||
|
tags: ["NOC"],
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// -- GET /api/noc/latest-discussion
|
||||||
|
.get(
|
||||||
|
"/latest-discussion",
|
||||||
|
async ({ query, set }) => {
|
||||||
|
const { idDesa, limit } = query;
|
||||||
|
|
||||||
|
const maxResults = Math.min(Number(limit ?? 5), 50);
|
||||||
|
|
||||||
|
if (!idDesa) {
|
||||||
|
set.status = 400;
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "Parameter idDesa wajib diisi",
|
||||||
|
data: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const village = await prisma.village.findUnique({
|
||||||
|
where: { id: idDesa },
|
||||||
|
select: { id: true, name: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!village) {
|
||||||
|
set.status = 404;
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "Desa tidak ditemukan",
|
||||||
|
data: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await prisma.discussion.findMany({
|
||||||
|
take: maxResults,
|
||||||
|
where: {
|
||||||
|
idVillage: idDesa,
|
||||||
|
isActive: true,
|
||||||
|
status: 1,
|
||||||
|
},
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
title: true,
|
||||||
|
desc: true,
|
||||||
|
createdAt: true,
|
||||||
|
User: {
|
||||||
|
select: {
|
||||||
|
name: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Group: {
|
||||||
|
select: {
|
||||||
|
name: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
orderBy: {
|
||||||
|
createdAt: "desc"
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const allData = data.map((v: any) => ({
|
||||||
|
..._.omit(v, ["createdAt", "User", "Group"]),
|
||||||
|
date: moment(v.createdAt).format("ll"),
|
||||||
|
user: v.User.name,
|
||||||
|
group: v.Group.name
|
||||||
|
}))
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
message: "Berhasil mendapatkan latest discussion",
|
||||||
|
data: allData
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error("[NOC] latest-discussion error:", error);
|
||||||
|
set.status = 500;
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "Terjadi kesalahan pada server",
|
||||||
|
data: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
query: t.Object({
|
||||||
|
idDesa: t.String({ description: "ID Desa yang ingin dicari" }),
|
||||||
|
limit: t.Optional(t.String({ description: "Limit data" })),
|
||||||
|
}),
|
||||||
|
detail: {
|
||||||
|
summary: "Latest Discussion",
|
||||||
|
description: "Mendapatkan latest discussion pada desa tertentu.",
|
||||||
tags: ["NOC"],
|
tags: ["NOC"],
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
||||||
export const GET = NocServer.handle;
|
export const GET = NocServer.handle;
|
||||||
export const POST = NocServer.handle;
|
export const POST = NocServer.handle;
|
||||||
|
|||||||
Reference in New Issue
Block a user