Perbaikan UI & API Menu Ekonomi Pasar Desa
This commit is contained in:
@@ -0,0 +1,41 @@
|
|||||||
|
/*
|
||||||
|
Warnings:
|
||||||
|
|
||||||
|
- You are about to drop the `_ProdukToKategori` table. If the table is not empty, all the data it contains will be lost.
|
||||||
|
|
||||||
|
*/
|
||||||
|
-- DropForeignKey
|
||||||
|
ALTER TABLE "_ProdukToKategori" DROP CONSTRAINT "_ProdukToKategori_A_fkey";
|
||||||
|
|
||||||
|
-- DropForeignKey
|
||||||
|
ALTER TABLE "_ProdukToKategori" DROP CONSTRAINT "_ProdukToKategori_B_fkey";
|
||||||
|
|
||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "KategoriProduk" ADD COLUMN "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
ADD COLUMN "isActive" BOOLEAN NOT NULL DEFAULT true;
|
||||||
|
|
||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "PasarDesa" ADD COLUMN "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
ADD COLUMN "isActive" BOOLEAN NOT NULL DEFAULT true;
|
||||||
|
|
||||||
|
-- DropTable
|
||||||
|
DROP TABLE "_ProdukToKategori";
|
||||||
|
|
||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "KategoriToPasar" (
|
||||||
|
"id" TEXT NOT NULL,
|
||||||
|
"kategoriId" TEXT NOT NULL,
|
||||||
|
"pasarDesaId" TEXT NOT NULL,
|
||||||
|
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"updatedAt" TIMESTAMP(3) NOT NULL,
|
||||||
|
"deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"isActive" BOOLEAN NOT NULL DEFAULT true,
|
||||||
|
|
||||||
|
CONSTRAINT "KategoriToPasar_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "KategoriToPasar" ADD CONSTRAINT "KategoriToPasar_kategoriId_fkey" FOREIGN KEY ("kategoriId") REFERENCES "KategoriProduk"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "KategoriToPasar" ADD CONSTRAINT "KategoriToPasar_pasarDesaId_fkey" FOREIGN KEY ("pasarDesaId") REFERENCES "PasarDesa"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
|
||||||
@@ -1050,28 +1050,43 @@ model MenuTipsKeamanan {
|
|||||||
// ========================================= MENU EKONOMI ========================================= //
|
// ========================================= MENU EKONOMI ========================================= //
|
||||||
// ========================================= PASAR DESA ========================================= //
|
// ========================================= PASAR DESA ========================================= //
|
||||||
model PasarDesa {
|
model PasarDesa {
|
||||||
id String @id @default(uuid())
|
id String @id @default(uuid())
|
||||||
nama String
|
nama String
|
||||||
image FileStorage? @relation(fields: [imageId], references: [id])
|
image FileStorage? @relation(fields: [imageId], references: [id])
|
||||||
imageId String?
|
imageId String?
|
||||||
harga Int
|
harga Int
|
||||||
rating Float
|
rating Float
|
||||||
alamatUsaha String
|
alamatUsaha String
|
||||||
kategori KategoriProduk[] @relation("ProdukToKategori")
|
createdAt DateTime @default(now())
|
||||||
createdAt DateTime @default(now())
|
updatedAt DateTime @updatedAt
|
||||||
updatedAt DateTime @updatedAt
|
deletedAt DateTime @default(now())
|
||||||
deletedAt DateTime @default(now())
|
isActive Boolean @default(true)
|
||||||
isActive Boolean @default(true)
|
kategoriProduk KategoriProduk @relation(fields: [kategoriProdukId], references: [id])
|
||||||
|
kategoriProdukId String
|
||||||
|
KategoriToPasar KategoriToPasar[]
|
||||||
}
|
}
|
||||||
|
|
||||||
model KategoriProduk {
|
model KategoriProduk {
|
||||||
id String @id @default(uuid())
|
id String @id @default(uuid())
|
||||||
nama String
|
nama String
|
||||||
produk PasarDesa[] @relation("ProdukToKategori")
|
createdAt DateTime @default(now())
|
||||||
createdAt DateTime @default(now())
|
updatedAt DateTime @updatedAt
|
||||||
updatedAt DateTime @updatedAt
|
deletedAt DateTime @default(now())
|
||||||
deletedAt DateTime @default(now())
|
isActive Boolean @default(true)
|
||||||
isActive Boolean @default(true)
|
KategoriToPasar KategoriToPasar[]
|
||||||
|
PasarDesa PasarDesa[]
|
||||||
|
}
|
||||||
|
|
||||||
|
model KategoriToPasar {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
kategori KategoriProduk @relation(fields: [kategoriId], references: [id])
|
||||||
|
kategoriId String
|
||||||
|
pasarDesa PasarDesa @relation(fields: [pasarDesaId], references: [id])
|
||||||
|
pasarDesaId String
|
||||||
|
createdAt DateTime @default(now())
|
||||||
|
updatedAt DateTime @updatedAt
|
||||||
|
deletedAt DateTime @default(now())
|
||||||
|
isActive Boolean @default(true)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ========================================= LOWONGAN KERJA LOKAL ========================================= //
|
// ========================================= LOWONGAN KERJA LOKAL ========================================= //
|
||||||
|
|||||||
@@ -53,11 +53,18 @@ const pasarDesa = proxy({
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
findMany: {
|
findMany: {
|
||||||
data: null as
|
data: null as Array<
|
||||||
| Prisma.PasarDesaGetPayload<{
|
Prisma.PasarDesaGetPayload<{
|
||||||
include: { kategori: true; image: true };
|
include: {
|
||||||
}>[]
|
image: true;
|
||||||
| null,
|
KategoriToPasar: {
|
||||||
|
include: {
|
||||||
|
kategori: true;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}>
|
||||||
|
> | null,
|
||||||
async load() {
|
async load() {
|
||||||
const res = await ApiFetch.api.ekonomi.pasardesa["find-many"].get();
|
const res = await ApiFetch.api.ekonomi.pasardesa["find-many"].get();
|
||||||
if (res.status === 200) {
|
if (res.status === 200) {
|
||||||
@@ -67,7 +74,14 @@ const pasarDesa = proxy({
|
|||||||
},
|
},
|
||||||
findUnique: {
|
findUnique: {
|
||||||
data: null as Prisma.PasarDesaGetPayload<{
|
data: null as Prisma.PasarDesaGetPayload<{
|
||||||
include: { kategori: true; image: true };
|
include: {
|
||||||
|
image: true;
|
||||||
|
KategoriToPasar: {
|
||||||
|
include: {
|
||||||
|
kategori: true;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
}> | null,
|
}> | null,
|
||||||
async load(id: string) {
|
async load(id: string) {
|
||||||
try {
|
try {
|
||||||
|
|||||||
@@ -43,7 +43,8 @@ function EditPasarDesa() {
|
|||||||
alamatUsaha: data.alamatUsaha || "",
|
alamatUsaha: data.alamatUsaha || "",
|
||||||
imageId: data.imageId || "",
|
imageId: data.imageId || "",
|
||||||
rating: data.rating || 0,
|
rating: data.rating || 0,
|
||||||
kategoriId: data.kategori?.map((k: any) => k.nama) || [],
|
// Use the IDs from KategoriToPasar relationship
|
||||||
|
kategoriId: data.KategoriToPasar?.map((k: any) => k.kategoriId) || [],
|
||||||
});
|
});
|
||||||
// Tampilkan preview gambar
|
// Tampilkan preview gambar
|
||||||
if (data.image?.link) {
|
if (data.image?.link) {
|
||||||
@@ -198,17 +199,19 @@ function EditPasarDesa() {
|
|||||||
/>
|
/>
|
||||||
<MultiSelect
|
<MultiSelect
|
||||||
value={formData.kategoriId}
|
value={formData.kategoriId}
|
||||||
onChange={(val) => {
|
onChange={(val) => setFormData({ ...formData, kategoriId: val })}
|
||||||
setFormData({ ...formData, kategoriId: val });
|
|
||||||
}}
|
|
||||||
label={<Text fw={"bold"} fz={"sm"}>Kategori Produk</Text>}
|
label={<Text fw={"bold"} fz={"sm"}>Kategori Produk</Text>}
|
||||||
placeholder='Pilih kategori produk'
|
placeholder='Pilih kategori produk'
|
||||||
data={
|
data={
|
||||||
pasarState.kategoriProduk.findMany.data?.map((v) => ({
|
pasarState.kategoriProduk.findMany.data?.map((v) => ({
|
||||||
value: v.id,
|
value: v.id, // Make sure this is using the ID
|
||||||
label: v.nama
|
label: v.nama
|
||||||
})) || []
|
})) || []
|
||||||
}
|
}
|
||||||
|
clearable
|
||||||
|
searchable
|
||||||
|
required
|
||||||
|
error={!formData.kategoriId.length ? "Pilih minimal satu kategori" : undefined}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Group>
|
<Group>
|
||||||
|
|||||||
@@ -17,7 +17,6 @@ function DetailPasarDesa() {
|
|||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
useShallowEffect(() => {
|
useShallowEffect(() => {
|
||||||
statePasar.kategoriProduk.findUnique.load(params?.id as string)
|
|
||||||
statePasar.pasarDesa.findUnique.load(params?.id as string)
|
statePasar.pasarDesa.findUnique.load(params?.id as string)
|
||||||
}, [])
|
}, [])
|
||||||
|
|
||||||
@@ -73,12 +72,17 @@ function DetailPasarDesa() {
|
|||||||
<Box>
|
<Box>
|
||||||
<Text fz={"lg"} fw={"bold"}>Kategori</Text>
|
<Text fz={"lg"} fw={"bold"}>Kategori</Text>
|
||||||
<Stack gap={4}>
|
<Stack gap={4}>
|
||||||
{statePasar.pasarDesa.findUnique.data?.kategori?.length > 0 ? (
|
{statePasar.pasarDesa.findUnique.data?.KategoriToPasar &&
|
||||||
statePasar.pasarDesa.findUnique.data.kategori.map((kat) => (
|
statePasar.pasarDesa.findUnique.data.KategoriToPasar.length > 0 ? (
|
||||||
<Text fz={"lg"} key={kat.id}>• {kat.nama}</Text>
|
statePasar.pasarDesa.findUnique.data.KategoriToPasar.map((kategori) => (
|
||||||
|
<Text fz={"lg"} key={kategori.id}>
|
||||||
|
• {kategori.kategori.nama}
|
||||||
|
</Text>
|
||||||
))
|
))
|
||||||
) : (
|
) : (
|
||||||
<Text fz={"lg"} c="dimmed">Tidak ada kategori</Text>
|
<Text fz={"lg"} c="dimmed">
|
||||||
|
Tidak ada kategori
|
||||||
|
</Text>
|
||||||
)}
|
)}
|
||||||
</Stack>
|
</Stack>
|
||||||
</Box>
|
</Box>
|
||||||
|
|||||||
@@ -7,32 +7,61 @@ type FormCreate = {
|
|||||||
alamatUsaha: string;
|
alamatUsaha: string;
|
||||||
imageId: string;
|
imageId: string;
|
||||||
rating: number;
|
rating: number;
|
||||||
kategoriId: string[]; // Array of KategoriMakanan IDs
|
kategoriId: string[]; // Array of KategoriProduk IDs
|
||||||
};
|
};
|
||||||
|
|
||||||
export default async function pasarDesaCreate(context: Context) {
|
export default async function pasarDesaCreate(context: Context) {
|
||||||
const body = context.body as FormCreate;
|
const body = context.body as FormCreate;
|
||||||
|
|
||||||
// First, create the PasarDesa record
|
if (!body.kategoriId || body.kategoriId.length === 0) {
|
||||||
const pasarDesa = await prisma.pasarDesa.create({
|
throw new Error("At least one kategoriId is required");
|
||||||
data: {
|
}
|
||||||
nama: body.nama,
|
|
||||||
harga: Number(body.harga),
|
try {
|
||||||
alamatUsaha: body.alamatUsaha,
|
// Start a transaction to ensure data consistency
|
||||||
imageId: body.imageId,
|
const result = await prisma.$transaction(async (prisma) => {
|
||||||
rating: Number(body.rating),
|
// 1. Create PasarDesa with the first kategoriId as the main category
|
||||||
kategori: {
|
const pasarDesa = await prisma.pasarDesa.create({
|
||||||
connect: body.kategoriId.map((id) => ({ id })),
|
data: {
|
||||||
},
|
nama: body.nama,
|
||||||
},
|
harga: Number(body.harga),
|
||||||
include: {
|
alamatUsaha: body.alamatUsaha,
|
||||||
image: true,
|
imageId: body.imageId,
|
||||||
kategori: true,
|
rating: Number(body.rating),
|
||||||
},
|
kategoriProdukId: body.kategoriId[0], // Use the first category as the main one
|
||||||
});
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// 2. Create category relationships in KategoriToPasar for all categories
|
||||||
|
await prisma.kategoriToPasar.createMany({
|
||||||
|
data: body.kategoriId.map((kategoriId) => ({
|
||||||
|
pasarDesaId: pasarDesa.id,
|
||||||
|
kategoriId: kategoriId, // Note: The field is 'kategoriId' in the schema, not 'kategoriProdukId'
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
|
||||||
|
// 3. Get the complete data with relationships
|
||||||
|
return await prisma.pasarDesa.findUnique({
|
||||||
|
where: { id: pasarDesa.id },
|
||||||
|
include: {
|
||||||
|
image: true,
|
||||||
|
kategoriProduk: true,
|
||||||
|
KategoriToPasar: {
|
||||||
|
include: {
|
||||||
|
kategori: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
success: true,
|
success: true,
|
||||||
message: "Success create pasar desa",
|
message: "Success create pasar desa",
|
||||||
data: pasarDesa,
|
data: result,
|
||||||
};
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error creating PasarDesa:", error);
|
||||||
|
throw new Error("Failed to create PasarDesa: " + (error as Error).message);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,54 +1,27 @@
|
|||||||
import prisma from "@/lib/prisma";
|
import prisma from "@/lib/prisma";
|
||||||
import { Context } from "elysia";
|
import { Context } from "elysia";
|
||||||
import fs from "fs/promises";
|
|
||||||
import path from "path";
|
|
||||||
|
|
||||||
const pasarDesaDelete = async (context: Context) => {
|
export default async function pasarDesaDelete(context: Context) {
|
||||||
const id = context.params?.id as string;
|
const { params } = context;
|
||||||
|
const id = params?.id as string;
|
||||||
|
|
||||||
if (!id) {
|
if (!id) {
|
||||||
return {
|
throw new Error("ID tidak ditemukan dalam parameter");
|
||||||
status: 400,
|
|
||||||
body: "ID tidak diberikan",
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const pasarDesa = await prisma.pasarDesa.findUnique({
|
// 1. Hapus relasi dari pivot
|
||||||
where: { id },
|
await prisma.kategoriToPasar.deleteMany({
|
||||||
include: {
|
where: { pasarDesaId: id },
|
||||||
image: true,
|
|
||||||
kategori: true,
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!pasarDesa) {
|
// 2. Hapus pasar desa utama
|
||||||
return {
|
const deleted = await prisma.pasarDesa.delete({
|
||||||
status: 404,
|
|
||||||
body: "Pasar desa tidak ditemukan",
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (pasarDesa.image) {
|
|
||||||
try {
|
|
||||||
const filePath = path.join(pasarDesa.image.path, pasarDesa.image.name);
|
|
||||||
await fs.unlink(filePath);
|
|
||||||
await prisma.fileStorage.delete({
|
|
||||||
where: { id: pasarDesa.image.id },
|
|
||||||
});
|
|
||||||
} catch (err) {
|
|
||||||
console.error("Gagal hapus file image:", err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
await prisma.pasarDesa.delete({
|
|
||||||
where: { id },
|
where: { id },
|
||||||
});
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
status: 200,
|
|
||||||
success: true,
|
success: true,
|
||||||
message: "Pasar desa berhasil dihapus",
|
message: "Berhasil menghapus pasar desa",
|
||||||
|
data: deleted,
|
||||||
};
|
};
|
||||||
};
|
}
|
||||||
export default pasarDesaDelete;
|
|
||||||
|
|||||||
@@ -1,25 +1,26 @@
|
|||||||
import prisma from "@/lib/prisma";
|
import prisma from "@/lib/prisma";
|
||||||
|
|
||||||
export default async function pasarDesaFindMany() {
|
export default async function pasarDesaFindMany() {
|
||||||
try {
|
const data = await prisma.pasarDesa.findMany({
|
||||||
const data = await prisma.pasarDesa.findMany({
|
where: {
|
||||||
where: { isActive: true },
|
isActive: true, // Opsional filter
|
||||||
include: {
|
},
|
||||||
image: true,
|
orderBy: {
|
||||||
kategori: true,
|
createdAt: "desc",
|
||||||
},
|
},
|
||||||
});
|
include: {
|
||||||
|
image: true,
|
||||||
|
KategoriToPasar: {
|
||||||
|
include: {
|
||||||
|
kategori: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
success: true,
|
success: true,
|
||||||
message: "Success fetch pasar desa",
|
message: "Berhasil mengambil semua data pasar desa",
|
||||||
data,
|
data,
|
||||||
};
|
};
|
||||||
} catch (e) {
|
}
|
||||||
console.error("Find many error:", e);
|
|
||||||
return {
|
|
||||||
success: false,
|
|
||||||
message: "Failed fetch pasar desa",
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,54 +1,33 @@
|
|||||||
import prisma from "@/lib/prisma";
|
import prisma from "@/lib/prisma";
|
||||||
|
import { Context } from "elysia";
|
||||||
|
|
||||||
export default async function pasarDesaFindUnique(request: Request){
|
export default async function pasarDesaFindUnique(context: Context) {
|
||||||
const url = new URL(request.url);
|
const { params } = context;
|
||||||
const pathSegments = url.pathname.split('/');
|
const id = params?.id as string;
|
||||||
const id = pathSegments[pathSegments.length - 1];
|
|
||||||
|
|
||||||
if(!id){
|
if (!id) {
|
||||||
return Response.json({
|
throw new Error("ID tidak ditemukan dalam parameter");
|
||||||
success: false,
|
}
|
||||||
message: "ID tidak boleh kosong",
|
|
||||||
}, { status: 400 });
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
const data = await prisma.pasarDesa.findUnique({
|
||||||
if (typeof id !== 'string') {
|
where: { id },
|
||||||
return Response.json({
|
include: {
|
||||||
success: false,
|
image: true,
|
||||||
message: "ID tidak valid",
|
KategoriToPasar: {
|
||||||
}, { status: 400 });
|
include: {
|
||||||
}
|
kategori: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
const data = await prisma.pasarDesa.findUnique({
|
if (!data) {
|
||||||
where: { id },
|
throw new Error("Pasar desa tidak ditemukan");
|
||||||
include: {
|
}
|
||||||
image: true,
|
|
||||||
kategori: true,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!data) {
|
return {
|
||||||
return Response.json({
|
success: true,
|
||||||
success: false,
|
message: "Data pasar desa ditemukan",
|
||||||
message: "Pasar desa tidak ditemukan",
|
data,
|
||||||
}, { status: 404 });
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
return Response.json({
|
|
||||||
success: true,
|
|
||||||
message: "Success fetch pasar desa by ID",
|
|
||||||
data,
|
|
||||||
}, {
|
|
||||||
status: 200,
|
|
||||||
});
|
|
||||||
} catch (e) {
|
|
||||||
console.error("Find by ID error:", e);
|
|
||||||
return Response.json({
|
|
||||||
success: false,
|
|
||||||
message: "Gagal mengambil pasar desa: " + (e instanceof Error ? e.message : 'Unknown error'),
|
|
||||||
}, {
|
|
||||||
status: 500,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -9,28 +9,26 @@ const PasarDesa = new Elysia({
|
|||||||
prefix: "/pasardesa",
|
prefix: "/pasardesa",
|
||||||
tags: ["Ekonomi/Pasar Desa"],
|
tags: ["Ekonomi/Pasar Desa"],
|
||||||
})
|
})
|
||||||
|
// GET all
|
||||||
.get("/find-many", pasarDesaFindMany)
|
.get("/find-many", pasarDesaFindMany)
|
||||||
.get("/:id", async (context) => {
|
|
||||||
const response = await pasarDesaFindUnique(new Request(context.request));
|
// GET by ID
|
||||||
return response;
|
.get(
|
||||||
})
|
|
||||||
.post("/create", pasarDesaCreate, {
|
|
||||||
body: t.Object({
|
|
||||||
nama: t.String(),
|
|
||||||
harga: t.Number(),
|
|
||||||
alamatUsaha: t.String(),
|
|
||||||
imageId: t.String(),
|
|
||||||
rating: t.Number(),
|
|
||||||
kategoriId:t.Array(t.String()),
|
|
||||||
}),
|
|
||||||
})
|
|
||||||
.delete("/del/:id", pasarDesaDelete)
|
|
||||||
.put(
|
|
||||||
"/:id",
|
"/:id",
|
||||||
async (context) => {
|
async (context) => {
|
||||||
const response = await pasarDesaUpdate(context);
|
return await pasarDesaFindUnique(context);
|
||||||
return response;
|
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
params: t.Object({
|
||||||
|
id: t.String(),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// POST create
|
||||||
|
.post(
|
||||||
|
"/create",
|
||||||
|
pasarDesaCreate,
|
||||||
{
|
{
|
||||||
body: t.Object({
|
body: t.Object({
|
||||||
nama: t.String(),
|
nama: t.String(),
|
||||||
@@ -38,7 +36,49 @@ const PasarDesa = new Elysia({
|
|||||||
alamatUsaha: t.String(),
|
alamatUsaha: t.String(),
|
||||||
imageId: t.String(),
|
imageId: t.String(),
|
||||||
rating: t.Number(),
|
rating: t.Number(),
|
||||||
kategoriId:t.Array(t.String()),
|
kategoriId: t.Array(t.String()),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// DELETE
|
||||||
|
.delete(
|
||||||
|
"/del/:id",
|
||||||
|
pasarDesaDelete,
|
||||||
|
{
|
||||||
|
params: t.Object({
|
||||||
|
id: t.String(),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// PUT update
|
||||||
|
.put(
|
||||||
|
"/:id",
|
||||||
|
async (context) => {
|
||||||
|
const body = context.body;
|
||||||
|
const id = context.params.id;
|
||||||
|
|
||||||
|
// Gabungkan id ke body
|
||||||
|
return await pasarDesaUpdate({
|
||||||
|
...context,
|
||||||
|
body: {
|
||||||
|
...body,
|
||||||
|
id,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
},
|
||||||
|
{
|
||||||
|
params: t.Object({
|
||||||
|
id: t.String(),
|
||||||
|
}),
|
||||||
|
body: t.Object({
|
||||||
|
nama: t.String(),
|
||||||
|
harga: t.Number(),
|
||||||
|
alamatUsaha: t.String(),
|
||||||
|
imageId: t.String(),
|
||||||
|
rating: t.Number(),
|
||||||
|
kategoriId: t.Array(t.String()),
|
||||||
}),
|
}),
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,98 +1,68 @@
|
|||||||
import prisma from "@/lib/prisma";
|
import prisma from "@/lib/prisma";
|
||||||
import { Context } from "elysia";
|
import { Context } from "elysia";
|
||||||
import fs from "fs/promises";
|
|
||||||
import path from "path";
|
|
||||||
|
|
||||||
type FormUpdate = {
|
type FormUpdate = {
|
||||||
nama: string;
|
id: string;
|
||||||
harga: number;
|
nama: string;
|
||||||
alamatUsaha: string;
|
harga: number;
|
||||||
imageId: string;
|
alamatUsaha: string;
|
||||||
rating: number;
|
imageId: string;
|
||||||
kategoriId: string[]; // Array of KategoriMakanan IDs
|
rating: number;
|
||||||
|
kategoriId: string[]; // Array of KategoriProduk IDs
|
||||||
|
};
|
||||||
|
|
||||||
|
export default async function pasarDesaUpdate(context: Context) {
|
||||||
|
const body = context.body as FormUpdate;
|
||||||
|
|
||||||
|
if (!body.id) {
|
||||||
|
throw new Error("ID pasar desa tidak boleh kosong");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!body.kategoriId || body.kategoriId.length === 0) {
|
||||||
|
throw new Error("Minimal 1 kategori harus dipilih");
|
||||||
|
}
|
||||||
|
|
||||||
|
// 1. Update data utama pasar desa
|
||||||
|
await prisma.pasarDesa.update({
|
||||||
|
where: { id: body.id },
|
||||||
|
data: {
|
||||||
|
nama: body.nama,
|
||||||
|
harga: Number(body.harga),
|
||||||
|
alamatUsaha: body.alamatUsaha,
|
||||||
|
imageId: body.imageId,
|
||||||
|
rating: Number(body.rating),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// 2. Hapus semua relasi kategori lama
|
||||||
|
await prisma.kategoriToPasar.deleteMany({
|
||||||
|
where: { pasarDesaId: body.id },
|
||||||
|
});
|
||||||
|
|
||||||
|
// 3. Tambah relasi kategori yang baru
|
||||||
|
await prisma.kategoriToPasar.createMany({
|
||||||
|
data: body.kategoriId.map((kategoriProdukId) => ({
|
||||||
|
pasarDesaId: body.id,
|
||||||
|
kategoriId: kategoriProdukId,
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
|
||||||
|
// 4. Ambil data lengkap setelah update
|
||||||
|
const updated = await prisma.pasarDesa.findUnique({
|
||||||
|
where: { id: body.id },
|
||||||
|
include: {
|
||||||
|
image: true,
|
||||||
|
KategoriToPasar: {
|
||||||
|
include: {
|
||||||
|
kategori: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
message: "Success update pasar desa",
|
||||||
|
data: updated,
|
||||||
};
|
};
|
||||||
|
}
|
||||||
export default async function pasarDesaUpdate(context: Context){
|
|
||||||
try {
|
|
||||||
const id = context.params?.id;
|
|
||||||
const body = context.body as FormUpdate;
|
|
||||||
|
|
||||||
const { nama, harga, alamatUsaha, imageId, rating, kategoriId } = body;
|
|
||||||
|
|
||||||
if (!id) {
|
|
||||||
return Response.json({
|
|
||||||
success: false,
|
|
||||||
message: "ID tidak boleh kosong",
|
|
||||||
}, { status: 400 });
|
|
||||||
}
|
|
||||||
|
|
||||||
const existing = await prisma.pasarDesa.findUnique({
|
|
||||||
where: { id },
|
|
||||||
include: {
|
|
||||||
image: true,
|
|
||||||
kategori: true,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
if (!existing) {
|
|
||||||
return Response.json({
|
|
||||||
success: false,
|
|
||||||
message: "Pasar desa tidak ditemukan",
|
|
||||||
}, { status: 404 });
|
|
||||||
}
|
|
||||||
|
|
||||||
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);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// First, update the main PasarDesa record
|
|
||||||
await prisma.pasarDesa.update({
|
|
||||||
where: { id },
|
|
||||||
data: {
|
|
||||||
nama,
|
|
||||||
harga,
|
|
||||||
alamatUsaha,
|
|
||||||
imageId,
|
|
||||||
rating,
|
|
||||||
kategori: {
|
|
||||||
connect: kategoriId.map((id) => ({ id })),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
// Fetch the updated record with all relations
|
|
||||||
const updated = await prisma.pasarDesa.findUnique({
|
|
||||||
where: { id },
|
|
||||||
include: {
|
|
||||||
image: true,
|
|
||||||
kategori: true,
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return Response.json({
|
|
||||||
success: true,
|
|
||||||
message: "Success update pasar desa",
|
|
||||||
data: updated,
|
|
||||||
}, {
|
|
||||||
status: 200,
|
|
||||||
});
|
|
||||||
} catch (e) {
|
|
||||||
console.error("Update error:", e);
|
|
||||||
return Response.json({
|
|
||||||
success: false,
|
|
||||||
message: "Gagal mengupdate pasar desa: " + (e instanceof Error ? e.message : 'Unknown error'),
|
|
||||||
}, {
|
|
||||||
status: 500,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
Reference in New Issue
Block a user