API Menu Kesehatan, Sub Menu Posyandu
This commit is contained in:
@@ -62,6 +62,8 @@ model FileStorage {
|
|||||||
link String
|
link String
|
||||||
Berita Berita[]
|
Berita Berita[]
|
||||||
PotensiDesa PotensiDesa[]
|
PotensiDesa PotensiDesa[]
|
||||||
|
|
||||||
|
Posyandu Posyandu[]
|
||||||
}
|
}
|
||||||
|
|
||||||
//========================================= MENU PPID ========================================= //
|
//========================================= MENU PPID ========================================= //
|
||||||
@@ -615,3 +617,17 @@ model DoctorSign {
|
|||||||
deletedAt DateTime @default(now())
|
deletedAt DateTime @default(now())
|
||||||
isActive Boolean @default(true)
|
isActive Boolean @default(true)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ========================================= POSYANDU ========================================= //
|
||||||
|
model Posyandu {
|
||||||
|
id String @id @default(cuid())
|
||||||
|
name String
|
||||||
|
nomor String
|
||||||
|
deskripsi String
|
||||||
|
image FileStorage @relation(fields: [imageId], references: [id])
|
||||||
|
imageId String
|
||||||
|
createdAt DateTime @default(now())
|
||||||
|
updatedAt DateTime @updatedAt
|
||||||
|
deletedAt DateTime @default(now())
|
||||||
|
isActive Boolean @default(true)
|
||||||
|
}
|
||||||
|
|||||||
@@ -12,11 +12,11 @@ function CreatePosyandu() {
|
|||||||
<Box>
|
<Box>
|
||||||
<Box mb={10}>
|
<Box mb={10}>
|
||||||
<Button onClick={() => router.back()} variant='subtle' color={'blue'}>
|
<Button onClick={() => router.back()} variant='subtle' color={'blue'}>
|
||||||
<IconArrowBack color={colors['blue-button']} size={25}/>
|
<IconArrowBack color={colors['blue-button']} size={25} />
|
||||||
</Button>
|
</Button>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
<Paper w={{base: '100%', md: '50%'}} bg={colors['white-1']} p={'md'}>
|
<Paper w={{ base: '100%', md: '50%' }} bg={colors['white-1']} p={'md'}>
|
||||||
<Stack gap={"xs"}>
|
<Stack gap={"xs"}>
|
||||||
<Title order={4}>Create Posyandu</Title>
|
<Title order={4}>Create Posyandu</Title>
|
||||||
<Box>
|
<Box>
|
||||||
@@ -24,8 +24,12 @@ function CreatePosyandu() {
|
|||||||
<IconImageInPicture size={50} />
|
<IconImageInPicture size={50} />
|
||||||
</Box>
|
</Box>
|
||||||
<TextInput
|
<TextInput
|
||||||
label={<Text fw={"bold"} fz={"sm"}>Nama Posyandu</Text>}
|
label={<Text fw={"bold"} fz={"sm"}>Nama Posyandu</Text>}
|
||||||
placeholder='Masukkan nama posyandu'
|
placeholder='Masukkan nama posyandu'
|
||||||
|
/>
|
||||||
|
<TextInput
|
||||||
|
label={<Text fw={"bold"} fz={"sm"}>Nomor Posyandu</Text>}
|
||||||
|
placeholder='Masukkan nomor posyandu'
|
||||||
/>
|
/>
|
||||||
<Box>
|
<Box>
|
||||||
<Text fw={"bold"} fz={"sm"}>Deskripsi Posyandu</Text>
|
<Text fw={"bold"} fz={"sm"}>Deskripsi Posyandu</Text>
|
||||||
@@ -36,9 +40,9 @@ function CreatePosyandu() {
|
|||||||
<Group>
|
<Group>
|
||||||
<Button bg={colors['blue-button']}>Submit</Button>
|
<Button bg={colors['blue-button']}>Submit</Button>
|
||||||
</Group>
|
</Group>
|
||||||
</Stack>
|
</Stack>
|
||||||
</Paper>
|
</Paper>
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -17,30 +17,26 @@ function DetailPosyandu() {
|
|||||||
</Box>
|
</Box>
|
||||||
<Paper w={{ base: "100%", md: "50%" }} bg={colors['white-1']} p={'md'}>
|
<Paper w={{ base: "100%", md: "50%" }} bg={colors['white-1']} p={'md'}>
|
||||||
<Stack>
|
<Stack>
|
||||||
<Text fz={"xl"} fw={"bold"}>Detail Potensi</Text>
|
<Text fz={"xl"} fw={"bold"}>Detail Posyandu</Text>
|
||||||
|
|
||||||
<Paper bg={colors['BG-trans']} p={'md'}>
|
<Paper bg={colors['BG-trans']} p={'md'}>
|
||||||
<Stack gap={"xs"}>
|
<Stack gap={"xs"}>
|
||||||
<Box>
|
<Box>
|
||||||
<Text fz={"lg"} fw={"bold"}>Judul</Text>
|
<Text fz={"lg"} fw={"bold"}>Nama Posyandu</Text>
|
||||||
<Text fz={"lg"}>Test Judul</Text>
|
<Text fz={"lg"}>Test Judul</Text>
|
||||||
</Box>
|
</Box>
|
||||||
<Box>
|
<Box>
|
||||||
<Text fz={"lg"} fw={"bold"}>Kategori</Text>
|
<Text fz={"lg"} fw={"bold"}>Nomor Posyandu</Text>
|
||||||
<Text fz={"lg"}>Test Kategori</Text>
|
<Text fz={"lg"}>089647038426</Text>
|
||||||
</Box>
|
</Box>
|
||||||
<Box>
|
<Box>
|
||||||
<Text fz={"lg"} fw={"bold"}>Deskripsi</Text>
|
<Text fz={"lg"} fw={"bold"}>Deskripsi Posyandu</Text>
|
||||||
<Text fz={"lg"}>Test Deskripsi</Text>
|
<Text fz={"lg"}>Test Kategori</Text>
|
||||||
</Box>
|
</Box>
|
||||||
<Box>
|
<Box>
|
||||||
<Text fz={"lg"} fw={"bold"}>Gambar</Text>
|
<Text fz={"lg"} fw={"bold"}>Gambar</Text>
|
||||||
<Image src={"/"} alt="gambar" />
|
<Image src={"/"} alt="gambar" />
|
||||||
</Box>
|
</Box>
|
||||||
<Box>
|
|
||||||
<Text fz={"lg"} fw={"bold"}>Konten</Text>
|
|
||||||
<Text fz={"lg"} >Test Konten</Text>
|
|
||||||
</Box>
|
|
||||||
<Box>
|
<Box>
|
||||||
<Flex gap={"xs"}>
|
<Flex gap={"xs"}>
|
||||||
<Button color="red">
|
<Button color="red">
|
||||||
@@ -61,7 +57,7 @@ function DetailPosyandu() {
|
|||||||
opened={modalHapus}
|
opened={modalHapus}
|
||||||
onClose={() => setModalHapus(false)}
|
onClose={() => setModalHapus(false)}
|
||||||
onConfirm={handleHapus}
|
onConfirm={handleHapus}
|
||||||
text="Apakah anda yakin ingin menghapus potensi ini?"
|
text="Apakah anda yakin ingin menghapus penanganan darurat ini?"
|
||||||
/> */}
|
/> */}
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -26,6 +26,10 @@ function EditPosyandu() {
|
|||||||
<TextInput
|
<TextInput
|
||||||
label={<Text fw={"bold"} fz={"sm"}>Nama Posyandu</Text>}
|
label={<Text fw={"bold"} fz={"sm"}>Nama Posyandu</Text>}
|
||||||
placeholder='Masukkan nama posyandu'
|
placeholder='Masukkan nama posyandu'
|
||||||
|
/>
|
||||||
|
<TextInput
|
||||||
|
label={<Text fw={"bold"} fz={"sm"}>Nomor Posyandu</Text>}
|
||||||
|
placeholder='Masukkan nomor posyandu'
|
||||||
/>
|
/>
|
||||||
<Box>
|
<Box>
|
||||||
<Text fw={"bold"} fz={"sm"}>Deskripsi Posyandu</Text>
|
<Text fw={"bold"} fz={"sm"}>Deskripsi Posyandu</Text>
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ const fileStorageCreate = async (context: Context) => {
|
|||||||
body: "UPLOAD_DIR is not defined",
|
body: "UPLOAD_DIR is not defined",
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
const pathName = "desa/ppid/profile-ppid";
|
const pathName = "allFile";
|
||||||
const rootPath = path.join(UPLOAD_DIR, pathName);
|
const rootPath = path.join(UPLOAD_DIR, pathName);
|
||||||
await fs.mkdir(rootPath, { recursive: true });
|
await fs.mkdir(rootPath, { recursive: true });
|
||||||
|
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ import Prevention from "./data_kesehatan_warga/artikel_kesehatan/prevention";
|
|||||||
import FirstAid from "./data_kesehatan_warga/artikel_kesehatan/first_aid";
|
import FirstAid from "./data_kesehatan_warga/artikel_kesehatan/first_aid";
|
||||||
import MythVsFact from "./data_kesehatan_warga/artikel_kesehatan/myth_vs_fact";
|
import MythVsFact from "./data_kesehatan_warga/artikel_kesehatan/myth_vs_fact";
|
||||||
import DoctorSign from "./data_kesehatan_warga/artikel_kesehatan/doctor_sign";
|
import DoctorSign from "./data_kesehatan_warga/artikel_kesehatan/doctor_sign";
|
||||||
|
import Posyandu from "./posyandu";
|
||||||
|
|
||||||
|
|
||||||
const Kesehatan = new Elysia({
|
const Kesehatan = new Elysia({
|
||||||
@@ -45,4 +46,5 @@ const Kesehatan = new Elysia({
|
|||||||
.use(FirstAid)
|
.use(FirstAid)
|
||||||
.use(MythVsFact)
|
.use(MythVsFact)
|
||||||
.use(DoctorSign)
|
.use(DoctorSign)
|
||||||
|
.use(Posyandu)
|
||||||
export default Kesehatan;
|
export default Kesehatan;
|
||||||
|
|||||||
31
src/app/api/[[...slugs]]/_lib/kesehatan/posyandu/create.ts
Normal file
31
src/app/api/[[...slugs]]/_lib/kesehatan/posyandu/create.ts
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
import prisma from "@/lib/prisma";
|
||||||
|
import { Prisma } from "@prisma/client";
|
||||||
|
import { Context } from "elysia";
|
||||||
|
|
||||||
|
type FormCreate = Prisma.PosyanduGetPayload<{
|
||||||
|
select: {
|
||||||
|
name: true;
|
||||||
|
nomor: true;
|
||||||
|
deskripsi: true;
|
||||||
|
imageId: true;
|
||||||
|
};
|
||||||
|
}>;
|
||||||
|
export default async function posyanduCreate(context: Context) {
|
||||||
|
const body = context.body as FormCreate;
|
||||||
|
|
||||||
|
await prisma.posyandu.create({
|
||||||
|
data: {
|
||||||
|
name: body.name,
|
||||||
|
nomor: body.nomor,
|
||||||
|
deskripsi: body.deskripsi,
|
||||||
|
imageId: body.imageId,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
message: "Success create posyandu",
|
||||||
|
data: {
|
||||||
|
...body,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
52
src/app/api/[[...slugs]]/_lib/kesehatan/posyandu/del.ts
Normal file
52
src/app/api/[[...slugs]]/_lib/kesehatan/posyandu/del.ts
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
import prisma from "@/lib/prisma";
|
||||||
|
import fs from "fs/promises";
|
||||||
|
import { Context } from "elysia";
|
||||||
|
import path from "path";
|
||||||
|
|
||||||
|
const posyanduDelete = async (context: Context) => {
|
||||||
|
const id = context.params?.id as string;
|
||||||
|
|
||||||
|
if (!id) {
|
||||||
|
return {
|
||||||
|
status: 400,
|
||||||
|
body: "ID tidak diberikan",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const posyandu = await prisma.posyandu.findUnique({
|
||||||
|
where: { id },
|
||||||
|
include: { image: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!posyandu) {
|
||||||
|
return {
|
||||||
|
status: 404,
|
||||||
|
body: "Posyandu tidak ditemukan",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hapus file gambar dari filesystem jika ada
|
||||||
|
if (posyandu.image) {
|
||||||
|
try {
|
||||||
|
const filePath = path.join(posyandu.image.path, posyandu.image.name);
|
||||||
|
await fs.unlink(filePath);
|
||||||
|
await prisma.fileStorage.delete({
|
||||||
|
where: { id: posyandu.image.id },
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Gagal hapus file image:", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hapus berita dari DB
|
||||||
|
await prisma.posyandu.delete({
|
||||||
|
where: { id },
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
message: "Posyandu dan file terkait berhasil dihapus",
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export default posyanduDelete;
|
||||||
@@ -0,0 +1,49 @@
|
|||||||
|
import prisma from "@/lib/prisma";
|
||||||
|
|
||||||
|
export default async function findPosyanduById(request: Request) {
|
||||||
|
const url = new URL(request.url);
|
||||||
|
const pathSegments = url.pathname.split('/');
|
||||||
|
const id = pathSegments[pathSegments.length - 1];
|
||||||
|
|
||||||
|
if (!id) {
|
||||||
|
return Response.json({
|
||||||
|
success: false,
|
||||||
|
message: "ID tidak boleh kosong",
|
||||||
|
}, { status: 400 })
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (typeof id !== 'string') {
|
||||||
|
return Response.json({
|
||||||
|
success: false,
|
||||||
|
message: "ID tidak valid",
|
||||||
|
}, { status: 400 })
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await prisma.posyandu.findUnique({
|
||||||
|
where: {id},
|
||||||
|
include: {
|
||||||
|
image: true
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!data) {
|
||||||
|
return Response.json({
|
||||||
|
success: false,
|
||||||
|
message: "Data tidak ditemukan",
|
||||||
|
}, { status: 404 })
|
||||||
|
}
|
||||||
|
|
||||||
|
return Response.json({
|
||||||
|
success: true,
|
||||||
|
message: "Success fetch posyandu by ID",
|
||||||
|
data,
|
||||||
|
}, { status: 200 })
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Find by ID error:", error);
|
||||||
|
return Response.json({
|
||||||
|
success: false,
|
||||||
|
message: "Gagal mengambil posyandu: " + (error instanceof Error ? error.message : 'Unknown error'),
|
||||||
|
}, { status: 500 })
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
import prisma from "@/lib/prisma";
|
||||||
|
|
||||||
|
export default async function posyanduFindMany() {
|
||||||
|
try {
|
||||||
|
const data = await prisma.posyandu.findMany({
|
||||||
|
where: {
|
||||||
|
isActive: true,
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
image: true,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
message: "Success fetch posyandu",
|
||||||
|
data,
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Find many error:", error);
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "Failed fetch posyandu",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
42
src/app/api/[[...slugs]]/_lib/kesehatan/posyandu/index.ts
Normal file
42
src/app/api/[[...slugs]]/_lib/kesehatan/posyandu/index.ts
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
import Elysia, { t } from "elysia";
|
||||||
|
import posyanduCreate from "./create";
|
||||||
|
import posyanduDelete from "./del";
|
||||||
|
import findPosyanduById from "./find-by-id";
|
||||||
|
import posyanduUpdate from "./updt";
|
||||||
|
import posyanduFindMany from "./find-many";
|
||||||
|
|
||||||
|
const Posyandu = new Elysia({
|
||||||
|
prefix: "/posyandu",
|
||||||
|
tags: ["Kesehatan/Posyandu"]
|
||||||
|
})
|
||||||
|
.post("/create", posyanduCreate, {
|
||||||
|
body: t.Object({
|
||||||
|
name: t.String(),
|
||||||
|
nomor: t.String(),
|
||||||
|
deskripsi: t.String(),
|
||||||
|
imageId: t.String(),
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.get("/find-many", posyanduFindMany)
|
||||||
|
.delete("/del/:id", posyanduDelete)
|
||||||
|
.get("/:id", async (context) => {
|
||||||
|
const response = await findPosyanduById(new Request(context.request));
|
||||||
|
return response;
|
||||||
|
})
|
||||||
|
.put(
|
||||||
|
"/:id",
|
||||||
|
async (context) => {
|
||||||
|
const response = await posyanduUpdate(context);
|
||||||
|
return response;
|
||||||
|
},
|
||||||
|
{
|
||||||
|
body: t.Object({
|
||||||
|
name: t.String(),
|
||||||
|
nomor: t.String(),
|
||||||
|
deskripsi: t.String(),
|
||||||
|
imageId: t.String(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
export default Posyandu;
|
||||||
105
src/app/api/[[...slugs]]/_lib/kesehatan/posyandu/updt.ts
Normal file
105
src/app/api/[[...slugs]]/_lib/kesehatan/posyandu/updt.ts
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
import prisma from "@/lib/prisma";
|
||||||
|
import { Prisma } from "@prisma/client";
|
||||||
|
import { Context } from "elysia";
|
||||||
|
import path from "path";
|
||||||
|
import fs from "fs/promises";
|
||||||
|
|
||||||
|
type FormUpdate = Prisma.PosyanduGetPayload<{
|
||||||
|
select: {
|
||||||
|
id: true;
|
||||||
|
name: true;
|
||||||
|
nomor: true;
|
||||||
|
deskripsi: true;
|
||||||
|
imageId: true;
|
||||||
|
}
|
||||||
|
}>
|
||||||
|
|
||||||
|
export default async function posyanduUpdate(context: Context) {
|
||||||
|
try {
|
||||||
|
const id = context.params?.id as string;
|
||||||
|
const body = (await context.body) as Omit<FormUpdate, "id">;
|
||||||
|
|
||||||
|
const {
|
||||||
|
name,
|
||||||
|
nomor,
|
||||||
|
deskripsi,
|
||||||
|
imageId,
|
||||||
|
} = body;
|
||||||
|
|
||||||
|
if(!id) {
|
||||||
|
return new Response(JSON.stringify({
|
||||||
|
success: false,
|
||||||
|
message: "ID tidak boleh kosong",
|
||||||
|
}), {
|
||||||
|
status: 400,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const existing = await prisma.posyandu.findUnique({
|
||||||
|
where: { id },
|
||||||
|
include: {
|
||||||
|
image: true,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!existing) {
|
||||||
|
return new Response(JSON.stringify({
|
||||||
|
success: false,
|
||||||
|
message: "Posyandu tidak ditemukan",
|
||||||
|
}), {
|
||||||
|
status: 404,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if (existing.imageId && existing.imageId !== imageId) {
|
||||||
|
const oldImage = existing.image;
|
||||||
|
if (oldImage) {
|
||||||
|
try {
|
||||||
|
const filePath = path.join(oldImage.path, oldImage.name);
|
||||||
|
await fs.unlink(filePath);
|
||||||
|
await prisma.fileStorage.delete({
|
||||||
|
where: { id: oldImage.id },
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
console.error("Gagal hapus gambar lama:", err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const updated = await prisma.posyandu.update({
|
||||||
|
where: { id },
|
||||||
|
data: {
|
||||||
|
name,
|
||||||
|
nomor,
|
||||||
|
deskripsi,
|
||||||
|
imageId,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
return new Response(JSON.stringify({
|
||||||
|
success: true,
|
||||||
|
message: "Posyandu berhasil diupdate",
|
||||||
|
data: updated,
|
||||||
|
}), {
|
||||||
|
status: 200,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
}
|
||||||
|
})
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error updating posyandu:", error);
|
||||||
|
return new Response(
|
||||||
|
JSON.stringify({
|
||||||
|
success: false,
|
||||||
|
message: "Terjadi kesalahan saat mengupdate posyandu",
|
||||||
|
}),
|
||||||
|
{ status: 500, headers: { 'Content-Type': 'application/json' } }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user