Compare commits

...

52 Commits

Author SHA1 Message Date
5fd5c15394 upd: api monitoring detail desa 2026-04-07 17:25:14 +08:00
cb565ba0bd upd: api monitoring menu desa 2026-04-07 14:52:46 +08:00
940fa5a5b7 Merge pull request 'upd: api monitoring' (#29) from amalia/06-apr-26 into join
Reviewed-on: #29
2026-04-06 17:35:18 +08:00
0b9f07e543 upd: api monitoring 2026-04-06 17:23:32 +08:00
8440374424 Merge pull request 'upd: url otp' (#28) from amalia/27-mar-26 into join
Reviewed-on: #28
2026-03-27 14:09:48 +08:00
eaa1a74290 upd: url otp 2026-03-27 14:07:35 +08:00
1326338335 Merge pull request 'upd: api noc' (#27) from amalia/25-mar-26 into join
Reviewed-on: #27
2026-03-25 17:05:36 +08:00
d1f553ee32 upd: api noc 2026-03-25 17:02:26 +08:00
b14ae8e5ff Merge pull request 'upd: api version' (#26) from amalia/16-mar-26 into join
Reviewed-on: #26
2026-03-16 16:13:58 +08:00
270875a95c upd: api version 2026-03-16 10:39:23 +08:00
09bd75d5e5 Merge pull request 'upd: api noc' (#25) from amalia/12-mar-26 into join
Reviewed-on: #25
2026-03-12 16:11:03 +08:00
339b1e25cc upd: api noc 2026-03-12 16:08:42 +08:00
d9c6f486a9 Merge pull request 'upd: api noc' (#24) from amalia/11-mar-26 into join
Reviewed-on: #24
2026-03-11 16:41:40 +08:00
1a20697f4c upd: api noc 2026-03-11 16:40:42 +08:00
3927a6b756 Merge pull request 'amalia/09-mar-26' (#23) from amalia/09-mar-26 into join
Reviewed-on: #23
2026-03-10 16:45:28 +08:00
079395654d update version dan data seeder 2026-03-09 11:34:13 +08:00
93e7f33f7c update version 2026-03-09 10:36:31 +08:00
aba7a4c8fc update workflow 2026-03-09 10:35:37 +08:00
f55b171987 Merge pull request 'amalia/06-mar-26' (#22) from amalia/06-mar-26 into join
Reviewed-on: #22
2026-03-06 16:37:01 +08:00
bipproduction
d401ebb208 fix: add custom Pages Router _error page for 404/500 prerendering
Override default Next.js _error page that imports <Html> from
next/document, which fails during Docker build prerendering.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 11:35:48 +08:00
bipproduction
5230a31942 fix: add custom 404 and global error pages for Docker build
Create not-found.tsx and global-error.tsx to prevent Next.js from
falling back to legacy Pages Router _error pages during static
generation, which fail with useContext null in Docker.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 11:28:37 +08:00
bipproduction
5e7eb20c26 fix: force dynamic rendering to skip static prerendering
Add `export const dynamic = 'force-dynamic'` to root layout so all
pages are rendered at request time instead of build time. Fixes
useContext null error during Docker build prerendering.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 11:19:36 +08:00
bipproduction
b7063d3658 fix: update Dockerfile bun version and remove --ignore-scripts
Upgrade bun 1.3.1 to 1.3.6 and remove --ignore-scripts flag to fix
prerender error during Docker build (null useContext dispatcher).

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 11:12:24 +08:00
4abaa97cc0 upd bun 2026-03-06 10:59:14 +08:00
069174cba1 update env example dummy 2026-03-06 10:42:45 +08:00
a04e0186a2 update env example 2026-03-06 10:33:22 +08:00
2af22b4bc7 build 2026-03-06 10:21:40 +08:00
0f90302f11 Merge pull request 'upd: unregistered token log logot' (#20) from amalia/04-mar-26 into join
Reviewed-on: #20
2026-03-04 16:38:22 +08:00
1b1a6b1b51 upd: unregistered token log logot 2026-03-04 16:33:47 +08:00
3a116ce212 Merge pull request 'upd: token, login dan version' (#19) from amalia/03-mar-26 into join
Reviewed-on: #19
2026-03-03 16:50:26 +08:00
60e88f5c9b upd: token, login dan version 2026-03-03 16:41:39 +08:00
2cd931dcfd Merge pull request 'upd: next config:' (#18) from amalia/25-feb-26 into join
Reviewed-on: #18
2026-02-25 12:44:21 +08:00
64fbc486f0 upd: next config: 2026-02-25 12:43:06 +08:00
02c9decbd8 Merge pull request 'upd: seeder dan version' (#17) from amalia/24-feb-26 into join
Reviewed-on: #17
2026-02-24 15:43:34 +08:00
c13340d254 upd: seeder dan version 2026-02-24 15:42:40 +08:00
757595e6af Merge pull request 'upd: seeder' (#16) from amalia/24-feb-26 into join
Reviewed-on: #16
2026-02-24 15:33:21 +08:00
5b3b39c19d upd: seeder 2026-02-24 15:32:36 +08:00
6b14427a2e Merge pull request 'upd: fix error dan seeder setting' (#15) from amalia/24-feb-26 into join
Reviewed-on: #15
2026-02-24 15:08:54 +08:00
4d73e4c875 upd: fix error dan seeder setting 2026-02-24 15:07:27 +08:00
519adeb376 Merge pull request 'upd: api mobile revisi' (#14) from amalia/23-feb-26 into join
Reviewed-on: #14
2026-02-24 13:38:27 +08:00
0ed01d287f upd: api mobile revisi 2026-02-23 14:37:26 +08:00
e62909b070 Merge pull request 'amalia/05-feb-26' (#12) from amalia/05-feb-26 into join
Reviewed-on: http://wibugit.wibudev.com/wibu/sistem-desa-mandiri/pulls/12
2026-02-05 17:27:41 +08:00
30611802f4 upd: seeder data dummy 2026-02-05 17:26:03 +08:00
854921935a data dummy seeder 2026-02-05 16:25:07 +08:00
191e567e12 Merge pull request 'upd: update seeder data desa dummy' (#11) from amalia/05-feb-26 into join
Reviewed-on: http://wibugit.wibudev.com/wibu/sistem-desa-mandiri/pulls/11
2026-02-05 14:05:57 +08:00
474ced6a38 upd: update seeder data desa dummy
Deskripsi:
- untuk presentasi
- untuk testing

No Issues
2026-02-05 14:04:49 +08:00
2b746b77e6 Merge pull request 'amalia/04-feb-26' (#10) from amalia/04-feb-26 into join
Reviewed-on: http://wibugit.wibudev.com/wibu/sistem-desa-mandiri/pulls/10
2026-02-04 17:33:52 +08:00
352469ce32 upd: seeder
Deskripsi:
- tambah data dummy desa untuk testing dan presentasi

No Issues
2026-02-04 17:31:18 +08:00
44b400cfb8 upd: panduan penggunaan by QWEN 2026-02-04 13:56:22 +08:00
e6b4adc8c2 Merge pull request 'upd: api tahun' (#9) from amalia/03-feb-26 into join
Reviewed-on: http://wibugit.wibudev.com/wibu/sistem-desa-mandiri/pulls/9
2026-02-03 12:25:34 +08:00
f5e36f5ac7 upd: api tahun
Deskripsi:
- update api mobile filter tahun pada fitur divisi tugas

No Issues
2026-02-03 12:24:23 +08:00
fa9b883c2e Merge pull request 'upd: api project' (#8) from amalia/02-feb-26 into join
Reviewed-on: http://wibugit.wibudev.com/wibu/sistem-desa-mandiri/pulls/8
2026-02-02 17:44:01 +08:00
57 changed files with 4670 additions and 101 deletions

52
.env.example Normal file
View File

@@ -0,0 +1,52 @@
# ===========================================
# SISTEM DESA MANDIRI - ENVIRONMENT VARIABLES
# ===========================================
# Copy this file to .env and fill in the appropriate values
# ===========================================
# DATABASE CONFIGURATION
# ===========================================
# PostgreSQL, MySQL, or SQLite connection string
# Example (PostgreSQL): postgresql://user:password@localhost:5432/dbname
# Example (MySQL): mysql://user:password@localhost:3306/dbname
# Example (SQLite): file:./dev.db
DATABASE_URL="your-database-url-here"
# ===========================================
# FIREBASE ADMIN SDK (For FCM Push Notifications)
# ===========================================
# Google Cloud project ID
GOOGLE_PROJECT_ID="your-google-project-id"
# Google service account client email
GOOGLE_CLIENT_EMAIL="your-service-account-email@your-project.iam.gserviceaccount.com"
# Google service account private key (include the full key with newlines)
GOOGLE_PRIVATE_KEY="-----BEGIN PRIVATE KEY-----\nYOUR_PRIVATE_KEY_HERE\n-----END PRIVATE KEY-----"
# Google service account private key ID (optional but recommended)
GOOGLE_PRIVATE_KEY_ID="your-private-key-id"
# ===========================================
# WEB PUSH NOTIFICATIONS (VAPID Keys)
# ===========================================
# VAPID public key (exposed to client-side, must start with NEXT_PUBLIC_)
NEXT_PUBLIC_VAPID_PUBLIC_KEY="BJlglqrIZCbPCZyUs8UIzEP1Wi18hzvGaC3-KPLkQuoCV_EOKdyGJNbu7fs5jYaO571ipVAMko8YiwIMa1VjQEg"
# VAPID private key (keep secret, server-side only)
VAPID_PRIVATE_KEY="UHDY8M3-0beVIA2kt2zL3ZeMStJ0j6zVkVd2Cfqpgrc"
# ===========================================
# FILE STORAGE / WEBSOCKET API
# ===========================================
# API key for file operations (upload, delete, copy, view directory)
WS_APIKEY="your-websocket-api-key"
# ===========================================
# APPLICATION SETTINGS
# ===========================================
# Next.js node environment (development, production, test)
NODE_ENV="development"
# Application URL (optional, for reference)
NEXT_PUBLIC_APP_URL="http://localhost:3000"

76
.github/workflows/publish.yml vendored Normal file
View File

@@ -0,0 +1,76 @@
name: Publish Docker to GHCR
on:
workflow_dispatch:
inputs:
stack_env:
description: "stack env"
required: true
type: choice
default: "dev"
options:
- dev
- prod
- stg
tag:
description: "Image tag (e.g. 1.0.0)"
required: true
default: "1.0.0"
env:
REGISTRY: ghcr.io
IMAGE_NAME: ${{ github.repository }}
jobs:
publish:
name: Build & Push to GHCR ${{ github.repository }}:${{ github.event.inputs.stack_env }}-${{ github.event.inputs.tag }}
runs-on: ubuntu-latest
permissions:
contents: read
packages: write
steps:
- name: Free disk space
run: |
sudo rm -rf /usr/share/dotnet
sudo rm -rf /usr/local/lib/android
sudo rm -rf /opt/ghc
sudo rm -rf /opt/hostedtoolcache/CodeQL
sudo docker image prune --all --force
df -h
- name: Checkout repository
uses: actions/checkout@v4
- name: Set up QEMU
uses: docker/setup-qemu-action@v3
- name: Set up Docker Buildx
uses: docker/setup-buildx-action@v3
- name: Log in to GitHub Container Registry
uses: docker/login-action@v3
with:
registry: ${{ env.REGISTRY }}
username: ${{ github.actor }}
password: ${{ secrets.GITHUB_TOKEN }}
- name: Generate image metadata
id: meta
uses: docker/metadata-action@v5
with:
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
tags: |
type=raw,value=${{ github.event.inputs.stack_env }}-${{ github.event.inputs.tag }}
type=raw,value=${{ github.event.inputs.stack_env }}-latest
- name: Build and push Docker image
uses: docker/build-push-action@v6
with:
context: .
file: ./Dockerfile
push: true
platforms: linux/amd64
tags: ${{ steps.meta.outputs.tags }}
labels: ${{ steps.meta.outputs.labels }}
no-cache: true

37
.github/workflows/re-pull.yml vendored Normal file
View File

@@ -0,0 +1,37 @@
name: Re-Pull Docker
on:
workflow_dispatch:
inputs:
stack_name:
description: "stack name"
required: true
type: string
stack_env:
description: "stack env"
required: true
type: choice
default: "dev"
options:
- dev
- stg
- prod
jobs:
publish:
name: Re-Pull Docker ${{ github.event.inputs.stack_name }}
runs-on: ubuntu-latest
environment: ${{ vars.PORTAINER_ENV || 'portainer' }}
permissions:
contents: read
packages: write
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Deploy ke Portainer
run: bash ./.github/workflows/script/re-pull.sh
env:
PORTAINER_USERNAME: ${{ secrets.PORTAINER_USERNAME }}
PORTAINER_PASSWORD: ${{ secrets.PORTAINER_PASSWORD }}
PORTAINER_URL: ${{ secrets.PORTAINER_URL }}
STACK_NAME: ${{ github.event.inputs.stack_name }}-${{ github.event.inputs.stack_env }}

93
.github/workflows/script/re-pull.sh vendored Normal file
View File

@@ -0,0 +1,93 @@
#!/bin/bash
: "${PORTAINER_URL:?PORTAINER_URL tidak di-set}"
: "${PORTAINER_USERNAME:?PORTAINER_USERNAME tidak di-set}"
: "${PORTAINER_PASSWORD:?PORTAINER_PASSWORD tidak di-set}"
: "${STACK_NAME:?STACK_NAME tidak di-set}"
echo "🔐 Autentikasi ke Portainer..."
TOKEN=$(curl -s -X POST https://${PORTAINER_URL}/api/auth \
-H "Content-Type: application/json" \
-d "{\"username\": \"${PORTAINER_USERNAME}\", \"password\": \"${PORTAINER_PASSWORD}\"}" \
| jq -r .jwt)
if [ -z "$TOKEN" ] || [ "$TOKEN" = "null" ]; then
echo "❌ Autentikasi gagal! Cek PORTAINER_URL, USERNAME, dan PASSWORD."
exit 1
fi
echo "🔍 Mencari stack: $STACK_NAME..."
STACK=$(curl -s -X GET https://${PORTAINER_URL}/api/stacks \
-H "Authorization: Bearer ${TOKEN}" \
| jq ".[] | select(.Name == \"$STACK_NAME\")")
if [ -z "$STACK" ]; then
echo "❌ Stack '$STACK_NAME' tidak ditemukan di Portainer!"
echo " Pastikan nama stack sudah benar."
exit 1
fi
STACK_ID=$(echo "$STACK" | jq -r .Id)
ENDPOINT_ID=$(echo "$STACK" | jq -r .EndpointId)
ENV=$(echo "$STACK" | jq '.Env // []')
echo "📄 Mengambil compose file..."
STACK_FILE=$(curl -s -X GET "https://${PORTAINER_URL}/api/stacks/${STACK_ID}/file" \
-H "Authorization: Bearer ${TOKEN}" \
| jq -r .StackFileContent)
PAYLOAD=$(jq -n \
--arg content "$STACK_FILE" \
--argjson env "$ENV" \
'{stackFileContent: $content, env: $env, pullImage: true}')
echo "🚀 Redeploying $STACK_NAME (pull latest image)..."
HTTP_STATUS=$(curl -s -o /tmp/portainer_response.json -w "%{http_code}" \
-X PUT "https://${PORTAINER_URL}/api/stacks/${STACK_ID}?endpointId=${ENDPOINT_ID}" \
-H "Authorization: Bearer ${TOKEN}" \
-H "Content-Type: application/json" \
-d "$PAYLOAD")
if [ "$HTTP_STATUS" != "200" ]; then
echo "❌ Redeploy gagal! HTTP Status: $HTTP_STATUS"
cat /tmp/portainer_response.json | jq .
exit 1
fi
echo "⏳ Menunggu container running..."
MAX_RETRY=15
COUNT=0
while [ $COUNT -lt $MAX_RETRY ]; do
sleep 5
COUNT=$((COUNT + 1))
CONTAINERS=$(curl -s -X GET \
"https://${PORTAINER_URL}/api/endpoints/${ENDPOINT_ID}/docker/containers/json?all=true&filters=%7B%22label%22%3A%5B%22com.docker.compose.project%3D${STACK_NAME}%22%5D%7D" \
-H "Authorization: Bearer ${TOKEN}")
TOTAL=$(echo "$CONTAINERS" | jq 'length')
RUNNING=$(echo "$CONTAINERS" | jq '[.[] | select(.State == "running")] | length')
FAILED=$(echo "$CONTAINERS" | jq '[.[] | select(.State == "exited" and (.Status | test("Exited \\(0\\)") | not))] | length')
echo "🔄 [${COUNT}/${MAX_RETRY}] Running: ${RUNNING} | Failed: ${FAILED} | Total: ${TOTAL}"
echo "$CONTAINERS" | jq -r '.[] | " → \(.Names[0]) | \(.State) | \(.Status)"'
if [ "$FAILED" -gt "0" ]; then
echo ""
echo "❌ Ada container yang crash!"
echo "$CONTAINERS" | jq -r '.[] | select(.State == "exited" and (.Status | test("Exited \\(0\\)") | not)) | " → \(.Names[0]) | \(.Status)"'
exit 1
fi
if [ "$RUNNING" -gt "0" ]; then
echo ""
echo "✅ Stack $STACK_NAME berhasil di-redeploy dan running!"
exit 0
fi
done
echo ""
echo "❌ Timeout! Stack tidak kunjung running setelah $((MAX_RETRY * 5)) detik."
exit 1

83
Dockerfile Normal file
View File

@@ -0,0 +1,83 @@
# ==============================
# Stage 1: Builder (Bun)
# ==============================
FROM oven/bun:1.3.6-debian AS builder
WORKDIR /app
RUN apt-get update && apt-get install -y --no-install-recommends \
libc6 \
git \
openssl \
ca-certificates \
&& rm -rf /var/lib/apt/lists/*
COPY package.json bun.lockb* ./
COPY prisma ./prisma
ENV ONNXRUNTIME_NODE_INSTALL_CUDA=0
ENV SHARP_IGNORE_GLOBAL_LIBVIPS=1
ENV NEXT_TELEMETRY_DISABLED=1
RUN bun install
COPY . .
# Gunakan .env jika ada, fallback ke .env.example.
# Untuk build dengan .env custom, hapus .env dari .dockerignore
# atau berikan via: docker build --secret id=env,src=.env (BuildKit)
RUN if [ -f .env ]; then \
echo "INFO: Menggunakan .env"; \
elif [ -f .env.example ]; then \
cp .env.example .env; \
echo "WARNING: .env tidak ditemukan, menggunakan .env.example (isi dengan nilai yang benar)"; \
else \
echo "WARNING: Tidak ada .env atau .env.example"; \
fi
# Generate prisma client
RUN ./node_modules/.bin/prisma generate
# Build Next.js
RUN bun run build
# ==============================
# Stage 2: Runner (Bun)
# ==============================
FROM oven/bun:1.3.6-debian AS runner
WORKDIR /app
ENV NODE_ENV=production
ENV NEXT_TELEMETRY_DISABLED=1
RUN apt-get update && apt-get install -y --no-install-recommends \
openssl \
ca-certificates \
&& rm -rf /var/lib/apt/lists/*
RUN groupadd --system --gid 1001 nodejs \
&& useradd --system --uid 1001 --gid nodejs nextjs
COPY --from=builder /app/public ./public
COPY --from=builder /app/.next ./.next
COPY --from=builder /app/node_modules ./node_modules
COPY --from=builder /app/package.json ./package.json
COPY --from=builder /app/tsconfig.json ./tsconfig.json
COPY --from=builder /app/prisma ./prisma
COPY --from=builder /app/src ./src
# Env vars runtime dikelola oleh Portainer (stack env / container env).
# Tidak perlu copy .env ke runner — image tetap bersih tanpa secrets.
RUN chown -R nextjs:nodejs /app
USER nextjs
EXPOSE 3000
ENV PORT=3000
ENV HOSTNAME="0.0.0.0"
CMD ["bun", "run", "start"]

255
PANDUAN PENGGUNAAN.md Normal file
View File

@@ -0,0 +1,255 @@
# Panduan Penggunaan Sistem Desa Mandiri
## Daftar Isi
1. [Gambaran Umum](#gambaran-umum)
2. [Peran Pengguna dan Hak Akses](#peran-pengguna-dan-hak-akses)
3. [Fitur-Fitur Utama dan Aksesnya](#fitur-fitur-utama-dan-aksesnya)
4. [Cara Menggunakan Aplikasi](#cara-menggunakan-aplikasi)
5. [Tips dan Trik](#tips-dan-trik)
## Gambaran Umum
Sistem Desa Mandiri adalah aplikasi web yang dirancang untuk membantu pengelolaan administrasi dan informasi di tingkat desa. Aplikasi ini dibangun dengan teknologi Next.js dan menyediakan berbagai fitur untuk mendukung kegiatan desa, mulai dari pengumuman, diskusi, manajemen proyek, hingga administrasi kependudukan.
## Peran Pengguna dan Hak Akses
Aplikasi ini memiliki beberapa peran pengguna dengan hak akses berbeda:
### 1. Super Admin
- **Hak Akses**: Memiliki akses penuh ke semua fitur aplikasi
- **Fungsi**: Mengelola seluruh sistem, termasuk pembuatan akun admin, pengaturan desa, dan manajemen sistem secara keseluruhan
- **Dapat Mengakses**: Semua fitur dalam aplikasi
### 2. Admin Desa
- **Hak Akses**: Memiliki akses ke fitur-fitur yang berkaitan dengan desa tertentu
- **Fungsi**: Mengelola data dan informasi dalam lingkup desa tertentu
- **Dapat Mengakses**: Semua fitur terkait desa yang dikelola, termasuk pengumuman, proyek, divisi, dan pengguna
### 3. Ketua Divisi
- **Hak Akses**: Memiliki akses administratif dalam divisi tertentu
- **Fungsi**: Mengelola anggota, proyek, dan kegiatan dalam divisi
- **Dapat Mengakses**: Fitur-fitur terkait divisi yang dipimpin, termasuk manajemen anggota, proyek, diskusi, dan dokumentasi
### 4. Anggota Divisi
- **Hak Akses**: Dapat mengakses dan berpartisipasi dalam kegiatan divisi
- **Fungsi**: Menjalankan tugas dan berkontribusi dalam kegiatan divisi
- **Dapat Mengakses**: Kegiatan dan informasi dalam divisi yang diikuti
### 5. Warga/Perangkat Desa
- **Hak Akses**: Akses dasar ke fitur-fitur umum
- **Fungsi**: Melihat informasi, berpartisipasi dalam diskusi umum
- **Dapat Mengakses**: Pengumuman, diskusi umum, kalender kegiatan umum
## Fitur-Fitur Utama dan Aksesnya
### 1. Manajemen Pengguna
- **Deskripsi**: Fitur untuk mendaftarkan dan mengelola data anggota desa serta mengatur hak akses berdasarkan peran
- **Dapat Diakses Oleh**: Super Admin, Admin Desa
- **Fungsi**:
- Registrasi pengguna baru
- Pengelolaan data pengguna
- Penetapan peran pengguna
- Pengelolaan grup dan posisi dalam desa
### 2. Pengumuman
- **Deskripsi**: Fitur untuk membuat dan menyebarkan pengumuman penting kepada warga
- **Dapat Diakses Oleh**: Super Admin, Admin Desa, Ketua Divisi (untuk divisi masing-masing)
- **Fungsi**:
- Membuat pengumuman baru
- Menargetkan pengumuman ke grup atau divisi tertentu
- Melampirkan file dalam pengumuman
- Mengedit atau menghapus pengumuman
### 3. Diskusi Umum
- **Deskripsi**: Forum diskusi umum untuk seluruh warga desa
- **Dapat Diakses Oleh**: Seluruh pengguna terdaftar
- **Fungsi**:
- Membuat topik diskusi baru
- Memberikan komentar dalam diskusi
- Melihat riwayat diskusi
- Melampirkan file dalam diskusi
### 4. Diskusi Divisi
- **Deskripsi**: Forum diskusi internal dalam divisi-divisi dalam desa
- **Dapat Diakses Oleh**: Anggota divisi yang bersangkutan
- **Fungsi**:
- Membuat topik diskusi internal divisi
- Memberikan komentar dalam diskusi divisi
- Menambahkan anggota ke dalam diskusi
- Melampirkan dokumen terkait diskusi
### 5. Manajemen Proyek
- **Deskripsi**: Fitur untuk membuat dan mengelola proyek-proyek desa
- **Dapat Diakses Oleh**: Super Admin, Admin Desa, Ketua Divisi
- **Fungsi**:
- Membuat proyek baru
- Menetapkan anggota tim proyek
- Melacak kemajuan proyek dan tugas-tugasnya
- Melampirkan dokumen dan tautan terkait proyek
- Menambahkan laporan kemajuan proyek
- Menyelesaikan atau membatalkan proyek
### 6. Manajemen Tugas
- **Deskripsi**: Fitur untuk mengelola tugas-tugas dalam proyek atau divisi
- **Dapat Diakses Oleh**: Super Admin, Admin Desa, Ketua Divisi, Leader Proyek
- **Fungsi**:
- Membuat tugas baru
- Menetapkan anggota yang bertugas
- Melacak kemajuan tugas
- Menambahkan detail waktu pelaksanaan
- Melampirkan dokumen terkait tugas
### 7. Divisi
- **Deskripsi**: Fitur untuk membuat dan mengelola divisi-divisi dalam desa
- **Dapat Diakses Oleh**: Super Admin, Admin Desa
- **Fungsi**:
- Membuat divisi baru
- Mengelola anggota dalam divisi
- Menetapkan admin dan leader divisi
- Mengelola proyek yang dikelola oleh divisi
- Mengelola diskusi internal divisi
- Mengelola dokumentasi divisi
- Mengelola kalender kegiatan divisi
### 8. Dokumentasi
- **Deskripsi**: Fitur untuk penyimpanan dokumen terpusat dalam divisi
- **Dapat Diakses Oleh**: Admin Divisi, Anggota Divisi (tergantung izin)
- **Fungsi**:
- Upload dokumen ke dalam folder
- Membuat struktur folder
- Berbagi dokumen antar divisi
- Cut dan paste dokumen antar folder
- Melihat riwayat dokumen
### 9. Kalender
- **Deskripsi**: Fitur untuk mengelola jadwal kegiatan desa dan divisi
- **Dapat Diakses Oleh**: Super Admin, Admin Desa, Ketua Divisi
- **Fungsi**:
- Membuat jadwal kegiatan baru
- Mengatur pengingat kegiatan
- Menetapkan peserta kegiatan
- Mengelola kegiatan berulang
- Melihat riwayat kegiatan
### 10. Tema Warna
- **Deskripsi**: Fitur untuk mengelola tampilan warna aplikasi berdasarkan desa
- **Dapat Diakses Oleh**: Super Admin, Admin Desa
- **Fungsi**:
- Mengatur warna utama aplikasi
- Mengatur warna latar belakang
- Mengatur warna elemen-elemen tampilan
### 11. Banner
- **Deskripsi**: Fitur untuk mengelola banner tampilan utama aplikasi
- **Dapat Diakses Oleh**: Super Admin, Admin Desa
- **Fungsi**:
- Upload banner baru
- Mengatur tampilan banner
- Menghapus banner lama
### 12. Notifikasi
- **Deskripsi**: Fitur untuk mengelola dan menerima notifikasi dalam aplikasi
- **Dapat Diakses Oleh**: Seluruh pengguna
- **Fungsi**:
- Menerima notifikasi real-time
- Melihat riwayat notifikasi
- Mengelola pengaturan notifikasi
## Cara Menggunakan Aplikasi
### 1. Login ke Sistem
- Buka browser dan kunjungi alamat aplikasi
- Masukkan NIK dan password yang telah didaftarkan
- Klik tombol "Login"
- Sistem akan mengarahkan ke dashboard sesuai dengan peran pengguna
### 2. Dashboard
- Setelah login, Anda akan diarahkan ke halaman dashboard
- Dashboard menampilkan ringkasan aktivitas dan informasi penting sesuai dengan hak akses Anda
- Gunakan menu navigasi di sisi kiri untuk mengakses fitur-fitur lain
### 3. Melihat dan Membuat Pengumuman
- **Melihat Pengumuman**:
- Klik menu "Pengumuman" di sidebar
- Pilih pengumuman yang ingin dibaca
- Anda juga dapat mengunduh file terlampir jika ada
- **Membuat Pengumuman (Untuk Pengguna Berwenang)**:
- Klik menu "Pengumuman" di sidebar
- Klik tombol "Buat Pengumuman Baru"
- Isi judul, deskripsi, dan pilih grup/divisi yang akan menerima
- Lampirkan file jika diperlukan
- Klik "Simpan" untuk menerbitkan pengumuman
### 4. Bergabung dalam Diskusi
- **Diskusi Umum**:
- Klik menu "Diskusi Umum" di sidebar
- Pilih forum diskusi yang tersedia
- Klik pada topik diskusi untuk membacanya
- Tulis komentar Anda dan klik "Kirim"
- **Diskusi Divisi**:
- Klik menu "Divisi" di sidebar
- Pilih divisi yang Anda ikuti
- Klik pada tab "Diskusi"
- Ikuti proses diskusi seperti pada diskusi umum
### 5. Mengelola Proyek
- Klik menu "Proyek" di sidebar
- Untuk membuat proyek baru, klik "Tambah Proyek"
- Isi informasi proyek seperti judul, deskripsi, tanggal mulai, dll.
- Tambahkan anggota tim proyek
- Buat tugas-tugas dalam proyek dan tetapkan ke anggota
- Pantau kemajuan proyek secara real-time
### 6. Mengelola Divisi
- Klik menu "Divisi" di sidebar
- Untuk membuat divisi baru, klik "Tambah Divisi"
- Isi informasi divisi seperti nama, deskripsi, dll.
- Tambahkan anggota ke dalam divisi
- Sebagai ketua divisi, Anda dapat menambahkan anggota
- Tetapkan admin dan leader divisi
- Kelola proyek, diskusi, dan dokumentasi dalam divisi
### 7. Mengelola Dokumen
- Klik menu "Divisi" di sidebar
- Pilih divisi yang Anda kelola atau ikuti
- Klik pada tab "Dokumen"
- Buat folder untuk mengorganisir dokumen
- Upload dokumen dengan klik tombol "Upload"
- Bagikan dokumen dengan divisi lain jika diperlukan
### 8. Menggunakan Kalender
- Klik menu "Divisi" di sidebar
- Pilih divisi yang Anda kelola atau ikuti
- Klik pada tab "Kalender"
- Lihat jadwal kegiatan yang telah direncanakan
- Klik "Tambah Kegiatan" untuk membuat jadwal baru
- Atur tanggal, waktu, dan pengingat untuk kegiatan
### 9. Mengelola Profil
- Klik foto profil Anda di pojok kanan atas
- Pilih "Profil" untuk melihat atau mengedit informasi pribadi
- Ganti foto profil, password, atau informasi kontak
## Tips dan Trik
1. **Gunakan Fitur Pencarian**: Gunakan fitur pencarian untuk menemukan pengumuman, diskusi, atau dokumen secara cepat.
2. **Atur Notifikasi**: Sesuaikan pengaturan notifikasi agar hanya menerima informasi yang relevan dengan peran Anda.
3. **Gunakan Filter**: Gunakan filter untuk menampilkan data yang spesifik sesuai kebutuhan (misalnya proyek aktif, pengumuman terbaru, dll.).
4. **Organisasi Dokumen**: Buat folder yang terstruktur untuk mengorganisasi dokumen agar mudah dicari kembali.
5. **Update Informasi**: Pastikan informasi pribadi Anda selalu diperbarui agar komunikasi berjalan efektif.
6. **Gunakan Mobile Version**: Aplikasi ini responsif dan dapat digunakan di perangkat mobile untuk kemudahan akses.
7. **Ikuti Aturan Diskusi**: Hormati sesama pengguna saat berdiskusi dan gunakan bahasa yang sopan.
8. **Gunakan Kalender**: Manfaatkan fitur kalender untuk tidak ketinggalan kegiatan penting di desa.
9. **Laporan Masalah**: Jika menemui masalah teknis, laporkan segera kepada admin untuk ditindaklanjuti.
10. **Pelajari Fitur Lainnya**: Luangkan waktu untuk menjelajahi semua fitur yang tersedia agar dapat memanfaatkan aplikasi secara maksimal.

View File

@@ -3,6 +3,12 @@ const nextConfig = {
devIndicators: {
buildActivityPosition: 'bottom-right',
},
typescript: {
ignoreBuildErrors: true, // ini yang fix TypeScript error
},
eslint: {
ignoreDuringBuilds: true,
},
};
export default nextConfig;

View File

@@ -0,0 +1,879 @@
-- CreateTable
CREATE TABLE "AdminRole" (
"id" TEXT NOT NULL,
"name" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "AdminRole_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "Admin" (
"id" TEXT NOT NULL,
"idAdminRole" TEXT NOT NULL,
"name" TEXT NOT NULL,
"phone" TEXT NOT NULL,
"email" TEXT,
"gender" TEXT NOT NULL DEFAULT 'M',
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "Admin_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "UserRole" (
"id" TEXT NOT NULL,
"name" TEXT NOT NULL,
"desc" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "UserRole_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "Village" (
"id" TEXT NOT NULL,
"idTheme" TEXT,
"name" TEXT NOT NULL,
"desc" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "Village_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "Group" (
"id" TEXT NOT NULL,
"idVillage" TEXT NOT NULL,
"name" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "Group_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "Position" (
"id" TEXT NOT NULL,
"idGroup" TEXT NOT NULL,
"name" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "Position_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "User" (
"id" TEXT NOT NULL,
"idUserRole" TEXT NOT NULL,
"idVillage" TEXT NOT NULL,
"idGroup" TEXT NOT NULL,
"idPosition" TEXT,
"nik" TEXT NOT NULL,
"name" TEXT NOT NULL,
"phone" TEXT NOT NULL,
"email" TEXT,
"gender" TEXT NOT NULL DEFAULT 'M',
"img" TEXT,
"isFirstLogin" BOOLEAN NOT NULL DEFAULT true,
"isWithoutOTP" BOOLEAN NOT NULL DEFAULT false,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "User_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "TokenDeviceUser" (
"id" TEXT NOT NULL,
"idUser" TEXT NOT NULL,
"token" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "TokenDeviceUser_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "UserLog" (
"id" TEXT NOT NULL,
"idUser" TEXT NOT NULL,
"action" TEXT NOT NULL,
"desc" TEXT NOT NULL,
"idContent" TEXT NOT NULL,
"tbContent" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "UserLog_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "Announcement" (
"id" TEXT NOT NULL,
"idVillage" TEXT NOT NULL,
"title" TEXT NOT NULL,
"desc" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdBy" TEXT NOT NULL,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "Announcement_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "AnnouncementMember" (
"id" TEXT NOT NULL,
"idAnnouncement" TEXT NOT NULL,
"idGroup" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "AnnouncementMember_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "AnnouncementFile" (
"id" TEXT NOT NULL,
"idAnnouncement" TEXT NOT NULL,
"name" TEXT NOT NULL,
"extension" TEXT NOT NULL,
"idStorage" TEXT,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "AnnouncementFile_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "Project" (
"id" TEXT NOT NULL,
"idVillage" TEXT NOT NULL,
"idGroup" TEXT NOT NULL,
"title" TEXT NOT NULL,
"status" INTEGER NOT NULL DEFAULT 0,
"desc" TEXT,
"reason" TEXT,
"report" TEXT,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdBy" TEXT NOT NULL,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "Project_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "ProjectMember" (
"id" TEXT NOT NULL,
"idProject" TEXT NOT NULL,
"idUser" TEXT NOT NULL,
"isLeader" BOOLEAN NOT NULL DEFAULT false,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "ProjectMember_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "ProjectFile" (
"id" TEXT NOT NULL,
"idProject" TEXT NOT NULL,
"name" TEXT NOT NULL,
"extension" TEXT NOT NULL,
"idStorage" TEXT,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "ProjectFile_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "ProjectLink" (
"id" TEXT NOT NULL,
"idProject" TEXT NOT NULL,
"link" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "ProjectLink_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "ProjectTask" (
"id" TEXT NOT NULL,
"idProject" TEXT NOT NULL,
"title" TEXT NOT NULL,
"desc" TEXT,
"status" INTEGER NOT NULL DEFAULT 0,
"notifikasi" BOOLEAN NOT NULL DEFAULT false,
"dateStart" DATE NOT NULL,
"dateEnd" DATE NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "ProjectTask_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "ProjectTaskDetail" (
"id" TEXT NOT NULL,
"idTask" TEXT NOT NULL,
"date" DATE NOT NULL,
"timeStart" TIME,
"timeEnd" TIME,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "ProjectTaskDetail_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "Division" (
"id" TEXT NOT NULL,
"idVillage" TEXT NOT NULL,
"idGroup" TEXT NOT NULL,
"name" TEXT NOT NULL,
"desc" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdBy" TEXT NOT NULL,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "Division_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionMember" (
"id" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"idUser" TEXT NOT NULL,
"isAdmin" BOOLEAN NOT NULL DEFAULT false,
"isLeader" BOOLEAN NOT NULL DEFAULT false,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionMember_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionProject" (
"id" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"title" TEXT NOT NULL,
"desc" TEXT,
"reason" TEXT,
"report" TEXT,
"status" INTEGER NOT NULL DEFAULT 0,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionProject_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionProjectLink" (
"id" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"idProject" TEXT NOT NULL,
"link" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionProjectLink_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionProjectTask" (
"id" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"idProject" TEXT NOT NULL,
"title" TEXT NOT NULL,
"desc" TEXT,
"status" INTEGER NOT NULL DEFAULT 0,
"notifikasi" BOOLEAN NOT NULL DEFAULT false,
"dateStart" DATE NOT NULL,
"dateEnd" DATE NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionProjectTask_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionProjectTaskDetail" (
"id" TEXT NOT NULL,
"idTask" TEXT NOT NULL,
"date" DATE NOT NULL,
"timeStart" TIME,
"timeEnd" TIME,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionProjectTaskDetail_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionProjectMember" (
"id" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"idProject" TEXT NOT NULL,
"idUser" TEXT NOT NULL,
"isLeader" BOOLEAN NOT NULL DEFAULT false,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionProjectMember_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionProjectFile" (
"id" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"idProject" TEXT NOT NULL,
"idFile" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdBy" TEXT NOT NULL,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionProjectFile_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionDisscussion" (
"id" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"title" TEXT,
"desc" TEXT NOT NULL,
"status" INTEGER NOT NULL DEFAULT 1,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdBy" TEXT NOT NULL,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionDisscussion_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionDisscussionComment" (
"id" TEXT NOT NULL,
"idDisscussion" TEXT NOT NULL,
"comment" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdBy" TEXT NOT NULL,
"isEdited" BOOLEAN NOT NULL DEFAULT false,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionDisscussionComment_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionDiscussionFile" (
"id" TEXT NOT NULL,
"idDiscussion" TEXT NOT NULL,
"name" TEXT NOT NULL,
"extension" TEXT NOT NULL,
"idStorage" TEXT,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionDiscussionFile_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionDocumentFolderFile" (
"id" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"idStorage" TEXT,
"category" TEXT NOT NULL DEFAULT 'FOLDER',
"name" TEXT NOT NULL,
"extension" TEXT NOT NULL,
"path" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdBy" TEXT NOT NULL,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionDocumentFolderFile_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionDocumentShare" (
"id" TEXT NOT NULL,
"idDocument" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionDocumentShare_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionCalendar" (
"id" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"title" TEXT NOT NULL,
"desc" TEXT NOT NULL,
"linkMeet" TEXT,
"dateStart" DATE NOT NULL,
"dateEnd" DATE,
"timeStart" TIME NOT NULL,
"timeEnd" TIME NOT NULL,
"repeatEventTyper" TEXT NOT NULL,
"repeatValue" INTEGER NOT NULL DEFAULT 1,
"reminderInterval" TEXT,
"status" INTEGER NOT NULL DEFAULT 0,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdBy" TEXT NOT NULL,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
CONSTRAINT "DivisionCalendar_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionCalendarReminder" (
"id" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"idCalendar" TEXT NOT NULL,
"dateStart" DATE NOT NULL,
"dateEnd" DATE,
"timeStart" TIME NOT NULL,
"timeEnd" TIME NOT NULL,
"status" INTEGER NOT NULL DEFAULT 0,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionCalendarReminder_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DivisionCalendarMember" (
"id" TEXT NOT NULL,
"idCalendar" TEXT NOT NULL,
"idUser" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DivisionCalendarMember_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "ContainerImage" (
"id" TEXT NOT NULL,
"category" TEXT NOT NULL,
"idCategory" TEXT NOT NULL,
"extension" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "ContainerImage_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "ContainerFileDivision" (
"id" TEXT NOT NULL,
"idDivision" TEXT NOT NULL,
"idStorage" TEXT,
"name" TEXT NOT NULL,
"extension" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "ContainerFileDivision_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "ColorTheme" (
"id" TEXT NOT NULL,
"idVillage" TEXT,
"name" TEXT NOT NULL,
"utama" TEXT NOT NULL,
"bgUtama" TEXT NOT NULL,
"bgIcon" TEXT NOT NULL,
"bgFiturHome" TEXT NOT NULL,
"bgFiturDivision" TEXT NOT NULL,
"bgTotalKegiatan" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "ColorTheme_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "BannerImage" (
"id" TEXT NOT NULL,
"idVillage" TEXT,
"title" TEXT NOT NULL,
"extension" TEXT NOT NULL,
"image" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "BannerImage_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "Notifications" (
"id" TEXT NOT NULL,
"idUserTo" TEXT NOT NULL,
"idUserFrom" TEXT NOT NULL,
"category" TEXT NOT NULL,
"idContent" TEXT NOT NULL,
"title" TEXT NOT NULL,
"desc" TEXT NOT NULL,
"isRead" BOOLEAN NOT NULL DEFAULT false,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "Notifications_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "Subscribe" (
"id" TEXT NOT NULL,
"idUser" TEXT NOT NULL,
"subscription" TEXT NOT NULL,
"createdAt" TIMESTAMP(3) DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3),
CONSTRAINT "Subscribe_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "Discussion" (
"id" TEXT NOT NULL,
"idVillage" TEXT NOT NULL,
"idGroup" TEXT NOT NULL,
"title" TEXT,
"desc" TEXT NOT NULL,
"status" INTEGER NOT NULL DEFAULT 1,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdBy" TEXT NOT NULL,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "Discussion_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DiscussionMember" (
"id" TEXT NOT NULL,
"idDiscussion" TEXT NOT NULL,
"idUser" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DiscussionMember_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DiscussionComment" (
"id" TEXT NOT NULL,
"idDiscussion" TEXT NOT NULL,
"idUser" TEXT NOT NULL,
"comment" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"isEdited" BOOLEAN NOT NULL DEFAULT false,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DiscussionComment_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "DiscussionFile" (
"id" TEXT NOT NULL,
"idDiscussion" TEXT NOT NULL,
"name" TEXT NOT NULL,
"extension" TEXT NOT NULL,
"idStorage" TEXT,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "DiscussionFile_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "Setting" (
"id" TEXT NOT NULL,
"name" TEXT NOT NULL,
"value" TEXT NOT NULL,
"isActive" BOOLEAN NOT NULL DEFAULT true,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "Setting_pkey" PRIMARY KEY ("id")
);
-- CreateIndex
CREATE UNIQUE INDEX "Admin_phone_key" ON "Admin"("phone");
-- CreateIndex
CREATE UNIQUE INDEX "Admin_email_key" ON "Admin"("email");
-- CreateIndex
CREATE UNIQUE INDEX "User_nik_key" ON "User"("nik");
-- CreateIndex
CREATE UNIQUE INDEX "User_phone_key" ON "User"("phone");
-- CreateIndex
CREATE UNIQUE INDEX "User_email_key" ON "User"("email");
-- CreateIndex
CREATE UNIQUE INDEX "Subscribe_idUser_key" ON "Subscribe"("idUser");
-- AddForeignKey
ALTER TABLE "Admin" ADD CONSTRAINT "Admin_idAdminRole_fkey" FOREIGN KEY ("idAdminRole") REFERENCES "AdminRole"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Group" ADD CONSTRAINT "Group_idVillage_fkey" FOREIGN KEY ("idVillage") REFERENCES "Village"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Position" ADD CONSTRAINT "Position_idGroup_fkey" FOREIGN KEY ("idGroup") REFERENCES "Group"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "User" ADD CONSTRAINT "User_idUserRole_fkey" FOREIGN KEY ("idUserRole") REFERENCES "UserRole"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "User" ADD CONSTRAINT "User_idVillage_fkey" FOREIGN KEY ("idVillage") REFERENCES "Village"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "User" ADD CONSTRAINT "User_idGroup_fkey" FOREIGN KEY ("idGroup") REFERENCES "Group"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "User" ADD CONSTRAINT "User_idPosition_fkey" FOREIGN KEY ("idPosition") REFERENCES "Position"("id") ON DELETE SET NULL ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "TokenDeviceUser" ADD CONSTRAINT "TokenDeviceUser_idUser_fkey" FOREIGN KEY ("idUser") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "UserLog" ADD CONSTRAINT "UserLog_idUser_fkey" FOREIGN KEY ("idUser") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Announcement" ADD CONSTRAINT "Announcement_idVillage_fkey" FOREIGN KEY ("idVillage") REFERENCES "Village"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Announcement" ADD CONSTRAINT "Announcement_createdBy_fkey" FOREIGN KEY ("createdBy") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "AnnouncementMember" ADD CONSTRAINT "AnnouncementMember_idAnnouncement_fkey" FOREIGN KEY ("idAnnouncement") REFERENCES "Announcement"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "AnnouncementMember" ADD CONSTRAINT "AnnouncementMember_idGroup_fkey" FOREIGN KEY ("idGroup") REFERENCES "Group"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "AnnouncementMember" ADD CONSTRAINT "AnnouncementMember_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "AnnouncementFile" ADD CONSTRAINT "AnnouncementFile_idAnnouncement_fkey" FOREIGN KEY ("idAnnouncement") REFERENCES "Announcement"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Project" ADD CONSTRAINT "Project_idVillage_fkey" FOREIGN KEY ("idVillage") REFERENCES "Village"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Project" ADD CONSTRAINT "Project_idGroup_fkey" FOREIGN KEY ("idGroup") REFERENCES "Group"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Project" ADD CONSTRAINT "Project_createdBy_fkey" FOREIGN KEY ("createdBy") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "ProjectMember" ADD CONSTRAINT "ProjectMember_idProject_fkey" FOREIGN KEY ("idProject") REFERENCES "Project"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "ProjectMember" ADD CONSTRAINT "ProjectMember_idUser_fkey" FOREIGN KEY ("idUser") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "ProjectFile" ADD CONSTRAINT "ProjectFile_idProject_fkey" FOREIGN KEY ("idProject") REFERENCES "Project"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "ProjectLink" ADD CONSTRAINT "ProjectLink_idProject_fkey" FOREIGN KEY ("idProject") REFERENCES "Project"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "ProjectTask" ADD CONSTRAINT "ProjectTask_idProject_fkey" FOREIGN KEY ("idProject") REFERENCES "Project"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "ProjectTaskDetail" ADD CONSTRAINT "ProjectTaskDetail_idTask_fkey" FOREIGN KEY ("idTask") REFERENCES "ProjectTask"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Division" ADD CONSTRAINT "Division_idVillage_fkey" FOREIGN KEY ("idVillage") REFERENCES "Village"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Division" ADD CONSTRAINT "Division_idGroup_fkey" FOREIGN KEY ("idGroup") REFERENCES "Group"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Division" ADD CONSTRAINT "Division_createdBy_fkey" FOREIGN KEY ("createdBy") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionMember" ADD CONSTRAINT "DivisionMember_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionMember" ADD CONSTRAINT "DivisionMember_idUser_fkey" FOREIGN KEY ("idUser") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProject" ADD CONSTRAINT "DivisionProject_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProjectLink" ADD CONSTRAINT "DivisionProjectLink_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProjectLink" ADD CONSTRAINT "DivisionProjectLink_idProject_fkey" FOREIGN KEY ("idProject") REFERENCES "DivisionProject"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProjectTask" ADD CONSTRAINT "DivisionProjectTask_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProjectTask" ADD CONSTRAINT "DivisionProjectTask_idProject_fkey" FOREIGN KEY ("idProject") REFERENCES "DivisionProject"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProjectTaskDetail" ADD CONSTRAINT "DivisionProjectTaskDetail_idTask_fkey" FOREIGN KEY ("idTask") REFERENCES "DivisionProjectTask"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProjectMember" ADD CONSTRAINT "DivisionProjectMember_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProjectMember" ADD CONSTRAINT "DivisionProjectMember_idProject_fkey" FOREIGN KEY ("idProject") REFERENCES "DivisionProject"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProjectMember" ADD CONSTRAINT "DivisionProjectMember_idUser_fkey" FOREIGN KEY ("idUser") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProjectFile" ADD CONSTRAINT "DivisionProjectFile_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProjectFile" ADD CONSTRAINT "DivisionProjectFile_idProject_fkey" FOREIGN KEY ("idProject") REFERENCES "DivisionProject"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProjectFile" ADD CONSTRAINT "DivisionProjectFile_idFile_fkey" FOREIGN KEY ("idFile") REFERENCES "ContainerFileDivision"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionProjectFile" ADD CONSTRAINT "DivisionProjectFile_createdBy_fkey" FOREIGN KEY ("createdBy") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionDisscussion" ADD CONSTRAINT "DivisionDisscussion_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionDisscussion" ADD CONSTRAINT "DivisionDisscussion_createdBy_fkey" FOREIGN KEY ("createdBy") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionDisscussionComment" ADD CONSTRAINT "DivisionDisscussionComment_idDisscussion_fkey" FOREIGN KEY ("idDisscussion") REFERENCES "DivisionDisscussion"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionDisscussionComment" ADD CONSTRAINT "DivisionDisscussionComment_createdBy_fkey" FOREIGN KEY ("createdBy") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionDiscussionFile" ADD CONSTRAINT "DivisionDiscussionFile_idDiscussion_fkey" FOREIGN KEY ("idDiscussion") REFERENCES "DivisionDisscussion"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionDocumentFolderFile" ADD CONSTRAINT "DivisionDocumentFolderFile_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionDocumentFolderFile" ADD CONSTRAINT "DivisionDocumentFolderFile_createdBy_fkey" FOREIGN KEY ("createdBy") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionDocumentShare" ADD CONSTRAINT "DivisionDocumentShare_idDocument_fkey" FOREIGN KEY ("idDocument") REFERENCES "DivisionDocumentFolderFile"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionDocumentShare" ADD CONSTRAINT "DivisionDocumentShare_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionCalendar" ADD CONSTRAINT "DivisionCalendar_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionCalendar" ADD CONSTRAINT "DivisionCalendar_createdBy_fkey" FOREIGN KEY ("createdBy") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionCalendarReminder" ADD CONSTRAINT "DivisionCalendarReminder_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionCalendarReminder" ADD CONSTRAINT "DivisionCalendarReminder_idCalendar_fkey" FOREIGN KEY ("idCalendar") REFERENCES "DivisionCalendar"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionCalendarMember" ADD CONSTRAINT "DivisionCalendarMember_idCalendar_fkey" FOREIGN KEY ("idCalendar") REFERENCES "DivisionCalendar"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DivisionCalendarMember" ADD CONSTRAINT "DivisionCalendarMember_idUser_fkey" FOREIGN KEY ("idUser") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "ContainerFileDivision" ADD CONSTRAINT "ContainerFileDivision_idDivision_fkey" FOREIGN KEY ("idDivision") REFERENCES "Division"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "ColorTheme" ADD CONSTRAINT "ColorTheme_idVillage_fkey" FOREIGN KEY ("idVillage") REFERENCES "Village"("id") ON DELETE SET NULL ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "BannerImage" ADD CONSTRAINT "BannerImage_idVillage_fkey" FOREIGN KEY ("idVillage") REFERENCES "Village"("id") ON DELETE SET NULL ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Notifications" ADD CONSTRAINT "UserToUserMap" FOREIGN KEY ("idUserTo") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Notifications" ADD CONSTRAINT "UserFromUserMap" FOREIGN KEY ("idUserFrom") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Subscribe" ADD CONSTRAINT "Subscribe_idUser_fkey" FOREIGN KEY ("idUser") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Discussion" ADD CONSTRAINT "Discussion_idVillage_fkey" FOREIGN KEY ("idVillage") REFERENCES "Village"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Discussion" ADD CONSTRAINT "Discussion_idGroup_fkey" FOREIGN KEY ("idGroup") REFERENCES "Group"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "Discussion" ADD CONSTRAINT "Discussion_createdBy_fkey" FOREIGN KEY ("createdBy") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DiscussionMember" ADD CONSTRAINT "DiscussionMember_idDiscussion_fkey" FOREIGN KEY ("idDiscussion") REFERENCES "Discussion"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DiscussionMember" ADD CONSTRAINT "DiscussionMember_idUser_fkey" FOREIGN KEY ("idUser") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DiscussionComment" ADD CONSTRAINT "DiscussionComment_idDiscussion_fkey" FOREIGN KEY ("idDiscussion") REFERENCES "Discussion"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DiscussionComment" ADD CONSTRAINT "DiscussionComment_idUser_fkey" FOREIGN KEY ("idUser") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "DiscussionFile" ADD CONSTRAINT "DiscussionFile_idDiscussion_fkey" FOREIGN KEY ("idDiscussion") REFERENCES "Discussion"("id") ON DELETE RESTRICT ON UPDATE CASCADE;

View File

@@ -0,0 +1,2 @@
-- AlterTable
ALTER TABLE "Village" ADD COLUMN "isDummy" BOOLEAN NOT NULL DEFAULT false;

View File

@@ -0,0 +1,3 @@
# Please do not edit this file manually
# It should be added in your version-control system (i.e. Git)
provider = "postgresql"

View File

@@ -51,6 +51,7 @@ model Village {
name String
desc String @db.Text
isActive Boolean @default(true)
isDummy Boolean @default(false)
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
Group Group[]
@@ -658,3 +659,12 @@ model DiscussionFile {
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
}
model Setting{
id String @id @default(cuid())
name String
value String
isActive Boolean @default(true)
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
}

View File

@@ -1,7 +1,11 @@
import { seederAdmin, seederAdminRole, seederDesa, seederGroup, seederPosition, seederTheme, seederUser, seederUserRole } from '@/module/seeder';
import { seederAdmin, seederAdminRole, seederAnnouncement, seederAnnouncementMember, seederDesa, seederDiscussion, seederDiscussionMember, seederDivision, seederDivisionMember, seederGroup, seederPosition, seederProject, seederProjectMember, seederProjectTask, seederSetting, seederTheme, seederUser, seederUserRole } from '@/module/seeder';
import { PrismaClient } from '@prisma/client';
const prisma = new PrismaClient()
// DATA YG DI SEEDER MERUPAKAN DATA REAL(DARMASABA) & DATA DUMMY (MANDALA)
// DATA JSON GABUNGAN (REAL & DUMMY) ADALAH adminRole, admin, theme, desa, group, position, user, userRole, user, dan setting
// Selain table yg disebutkan, data lainnya merupakan data dummy
async function main() {
// ADMIN ROLE
for (let data of seederAdminRole) {
@@ -144,7 +148,7 @@ async function main() {
})
}
// USER
// USER
for (let data of seederUser) {
await prisma.user.upsert({
where: {
@@ -155,10 +159,10 @@ async function main() {
idGroup: data.idGroup,
idPosition: data.idPosition,
idUserRole: data.idUserRole,
nik: data.nik,
// nik: data.nik,
name: data.name,
// phone: data.phone,
email: data.email,
// email: data.email,
gender: data.gender
},
create: {
@@ -176,6 +180,228 @@ async function main() {
})
}
// DISCUSSION
for (let data of seederDiscussion) {
await prisma.discussion.upsert({
where: {
id: data.id
},
update: {
idVillage: data.idVillage,
idGroup: data.idGroup,
title: data.title,
desc: data.desc,
status: data.status,
createdBy: data.createdBy
},
create: {
id: data.id,
idVillage: data.idVillage,
idGroup: data.idGroup,
title: data.title,
desc: data.desc,
status: data.status,
createdBy: data.createdBy
},
})
}
// DISSCUSSION MEMBER
for (let data of seederDiscussionMember) {
await prisma.discussionMember.upsert({
where: {
id: data.id
},
update: {
idDiscussion: data.idDiscussion,
idUser: data.idUser
},
create: {
id: data.id,
idDiscussion: data.idDiscussion,
idUser: data.idUser
},
})
}
// PROJECT
for (let data of seederProject) {
await prisma.project.upsert({
where: {
id: data.id
},
update: {
idVillage: data.idVillage,
idGroup: data.idGroup,
title: data.title,
desc: data.desc,
status: data.status,
createdBy: data.createdBy
},
create: {
id: data.id,
idVillage: data.idVillage,
idGroup: data.idGroup,
title: data.title,
desc: data.desc,
status: data.status,
createdBy: data.createdBy
},
})
}
// PROJECT MEMBER
for (let data of seederProjectMember) {
await prisma.projectMember.upsert({
where: {
id: data.id
},
update: {
idProject: data.idProject,
idUser: data.idUser,
isLeader: data.isLeader
},
create: {
id: data.id,
idProject: data.idProject,
idUser: data.idUser,
isLeader: data.isLeader
},
})
}
// PROJECT TASK
for (let data of seederProjectTask) {
await prisma.projectTask.upsert({
where: {
id: data.id
},
update: {
idProject: data.idProject,
title: data.title,
desc: data.desc,
status: data.status,
dateStart: new Date(data.dateStart),
dateEnd: new Date(data.dateEnd)
},
create: {
id: data.id,
idProject: data.idProject,
title: data.title,
desc: data.desc,
status: data.status,
dateStart: new Date(data.dateStart),
dateEnd: new Date(data.dateEnd)
},
})
}
// DIVISION
for (let data of seederDivision) {
await prisma.division.upsert({
where: {
id: data.id
},
update: {
name: data.name,
desc: data.desc,
createdBy: data.createdBy
},
create: {
id: data.id,
idVillage: data.idVillage,
idGroup: data.idGroup,
name: data.name,
desc: data.desc,
createdBy: data.createdBy,
isActive: true
}
})
}
// DIVISION MEMBER
for (let data of seederDivisionMember) {
await prisma.divisionMember.upsert({
where: {
id: data.id
},
update: {
idUser: data.idUser,
isAdmin: data.isAdmin,
isLeader: data.isLeader
},
create: {
id: data.id,
idDivision: data.idDivision,
idUser: data.idUser,
isAdmin: data.isAdmin,
isLeader: data.isLeader,
isActive: true
}
})
}
// ANNOUNCEMENT
for (let data of seederAnnouncement) {
await prisma.announcement.upsert({
where: {
id: data.id
},
update: {
title: data.title,
desc: data.desc,
createdBy: data.createdBy
},
create: {
id: data.id,
idVillage: data.idVillage,
title: data.title,
desc: data.desc,
createdBy: data.createdBy,
isActive: true
}
})
}
// ANNOUNCEMENT MEMBER
for (let data of seederAnnouncementMember) {
await prisma.announcementMember.upsert({
where: {
id: data.id
},
update: {
idAnnouncement: data.idAnnouncement,
idGroup: data.idGroup,
idDivision: data.idDivision
},
create: {
id: data.id,
idAnnouncement: data.idAnnouncement,
idGroup: data.idGroup,
idDivision: data.idDivision,
isActive: true
}
})
}
// SETTING
for (let data of seederSetting) {
await prisma.setting.upsert({
where: {
id: data.id
},
update: {
name: data.name,
},
create: {
id: data.id,
name: data.name,
value: data.value
}
})
}
}
main().then(async () => {

View File

@@ -0,0 +1,59 @@
import { prisma } from "@/module/_global";
import { ILogin } from "@/types";
import { NextRequest } from "next/server";
export async function POST(req: NextRequest) {
try {
const { phone }: ILogin = await req.json();
const user = await prisma.user.findUnique({
where: { phone, isActive: true },
select: { id: true, phone: true, isWithoutOTP: true },
});
if (!user) {
return Response.json({
success: false,
message: "Nomor telepon tidak terdaftar",
});
}
// Generate OTP
const code = Math.floor(1000 + Math.random() * 9000);
const message = `Desa+\nMasukkan kode ini ${code} pada web app Desa+ anda. Jangan berikan pada siapapun.`;
// Send WhatsApp
try {
const resWa = await fetch(`${process.env.URL_OTP}/api/wa/send-text`, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
Authorization: `Bearer ${process.env.WA_SERVER_TOKEN}`,
},
body: JSON.stringify({
number: user.phone,
text: message,
}),
});
if (!resWa.ok) {
console.error("WhatsApp API Error:", resWa.status);
}
} catch (error) {
console.error("WhatsApp Fetch Error:", error);
}
return Response.json({
success: true,
message: "Sukses",
phone: user.phone,
isWithoutOTP: user.isWithoutOTP,
id: user.id,
otp: code, // Return OTP for client-side verification (as per existing logic)
});
} catch (error) {
console.error(error);
return Response.json({ message: "Internal Server Error (error: 500)", success: false });
}
}

View File

@@ -167,12 +167,19 @@ export async function DELETE(request: Request, context: { params: { id: string }
// EDIT PENGUMUMAN
export async function PUT(request: Request, context: { params: { id: string } }) {
try {
const body = await request.formData()
const dataBody = body.get("data")
const cekFile = body.has("file0")
const contentType = request.headers.get("content-type");
let title, desc, groups, user, oldFile: any[] = [], cekFile, body: FormData | undefined
if (contentType?.includes("multipart/form-data")) {
body = await request.formData()
const dataBody = body.get("data")
cekFile = body.has("file0");
({ title, desc, groups, user, oldFile } = JSON.parse(dataBody as string))
} else {
({ title, desc, groups, user } = await request.json());
}
// const { title, desc, groups, user } = (await request.json());
const { title, desc, groups, user, oldFile } = JSON.parse(dataBody as string)
const { id } = context.params;
const userMobile = await funGetUserById({ id: String(user) })
@@ -245,7 +252,7 @@ export async function PUT(request: Request, context: { params: { id: string } })
}
}
if (cekFile) {
if (cekFile && body) {
body.delete("data")
for (var pair of body.entries()) {
if (String(pair[0]).substring(0, 4) == "file") {

View File

@@ -113,12 +113,19 @@ export async function GET(request: Request) {
// CREATE PENGUMUMAN
export async function POST(request: Request) {
try {
const body = await request.formData()
const dataBody = body.get("data")
const cekFile = body.has("file0")
const contentType = request.headers.get("content-type");
let title, desc, groups, user, cekFile, body: FormData | undefined
if (contentType?.includes("multipart/form-data")) {
body = await request.formData()
const dataBody = body.get("data")
cekFile = body.has("file0");
({ title, desc, groups, user } = JSON.parse(dataBody as string))
} else {
({ title, desc, groups, user } = await request.json());
}
// const { title, desc, groups, user } = (await request.json());
const { title, desc, groups, user } = JSON.parse(dataBody as string)
const userMobile = await funGetUserById({ id: String(user) })
if (userMobile.id == "null" || userMobile.id == undefined || userMobile.id == "") {
@@ -144,7 +151,6 @@ export async function POST(request: Request) {
let memberDivision = []
for (var i = 0, l = groups.length; i < l; i++) {
2
var obj = groups[i].Division;
for (let index = 0; index < obj.length; index++) {
const element = obj[index];
@@ -158,7 +164,7 @@ export async function POST(request: Request) {
}
if (cekFile) {
if (cekFile && body) {
body.delete("data")
for (var pair of body.entries()) {
if (String(pair[0]).substring(0, 4) == "file") {
@@ -247,7 +253,7 @@ export async function POST(request: Request) {
where: {
isActive: true,
idUserRole: "supadmin",
idVillage: user.idVillage
idVillage: String(villaId)
},
select: {
id: true,

View File

@@ -0,0 +1,32 @@
import { prisma } from "@/module/_global";
import { funGetUserById } from "@/module/auth";
import { NextResponse } from "next/server";
export async function POST(request: Request) {
try {
const { token, user } = (await request.json());
const userMobile = await funGetUserById({ id: 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 cek = await prisma.tokenDeviceUser.count({
where: {
idUser: userMobile.id,
token
}
})
if (cek > 0) {
return NextResponse.json({ success: true, message: "Token terdaftar", data: true }, { status: 200 });
} else {
return NextResponse.json({ success: false, message: "Token tidak terdaftar", data: false }, { status: 200 })
}
} catch (error) {
console.error(error);
return NextResponse.json({ success: false, message: "Gagal mengecek token, coba lagi nanti (error: 500)", reason: (error as Error).message, }, { status: 500 });
}
};

View File

@@ -5,7 +5,7 @@ import { NextResponse } from "next/server";
export async function POST(request: Request) {
try {
const { token, user } = (await request.json());
const { token, user, category } = (await request.json());
const userMobile = await funGetUserById({ id: user })
if (userMobile.id == "null" || userMobile.id == undefined || userMobile.id == "") {
@@ -19,8 +19,10 @@ export async function POST(request: Request) {
}
})
// create log user
const log = await createLogUserMobile({ act: 'LOGIN', desc: 'User login', table: 'user', data: '', user: userMobile.id })
if (category != "register") {
// create log user
const log = await createLogUserMobile({ act: 'LOGIN', desc: 'User login', table: 'user', data: '', user: userMobile.id })
}
if (cek == 0 && token != "" && token != undefined && token != null) {
const data = await prisma.tokenDeviceUser.create({
@@ -43,7 +45,7 @@ export async function POST(request: Request) {
export async function PUT(request: Request) {
try {
const { token, user } = (await request.json());
const { token, user, category } = (await request.json());
const userMobile = await funGetUserById({ id: user })
if (userMobile.id == "null" || userMobile.id == undefined || userMobile.id == "") {
@@ -60,8 +62,10 @@ export async function PUT(request: Request) {
}
// create log user
const log = await createLogUserMobile({ act: 'LOGOUT', desc: 'User logout', table: 'user', data: '', user: userMobile.id })
if (category != "unregister") {
// create log user
const log = await createLogUserMobile({ act: 'LOGOUT', desc: 'User logout', table: 'user', data: '', user: userMobile.id })
}
return NextResponse.json({ success: true, message: "Berhasil menghapus token", }, { status: 200 });
} catch (error) {

View File

@@ -242,10 +242,10 @@ export async function DELETE(request: Request, context: { params: { id: string }
// create log user
if (active) {
const log = await createLogUserMobile({ act: 'DELETE', desc: 'User mengaktifkan data diskusi umum', table: 'disscussion', data: id, user: userMobile.id })
return NextResponse.json({ success: true, message: "Berhasil mengaktifkan diskusi umum", user: user.id }, { status: 200 });
return NextResponse.json({ success: true, message: "Berhasil mengaktifkan diskusi umum" }, { status: 200 });
} else {
const log = await createLogUserMobile({ act: 'DELETE', desc: 'User mengarsipkan data diskusi umum', table: 'disscussion', data: id, user: userMobile.id })
return NextResponse.json({ success: true, message: "Berhasil mengarsipkan diskusi umum", user: user.id }, { status: 200 });
return NextResponse.json({ success: true, message: "Berhasil mengarsipkan diskusi umum" }, { status: 200 });
}
@@ -260,12 +260,19 @@ export async function DELETE(request: Request, context: { params: { id: string }
export async function PUT(request: Request, context: { params: { id: string } }) {
try {
const { id } = context.params
const body = await request.formData()
const dataBody = body.get("data")
const cekFile = body.has("file0")
const contentType = request.headers.get("content-type");
let title, desc, user, oldFile: any[] = [], cekFile, body: FormData | undefined
if (contentType?.includes("multipart/form-data")) {
body = await request.formData()
const dataBody = body.get("data")
cekFile = body.has("file0");
({ title, desc, user, oldFile } = JSON.parse(dataBody as string))
} else {
({ title, desc, user } = await request.json());
}
// const { title, desc, user } = (await request.json());
const { title, desc, user, oldFile } = JSON.parse(dataBody as string)
const userMobile = await funGetUserById({ id: String(user) })
@@ -306,7 +313,7 @@ export async function PUT(request: Request, context: { params: { id: string } })
}
}
if (cekFile) {
if (cekFile && body) {
body.delete("data")
for (var pair of body.entries()) {
if (String(pair[0]).substring(0, 4) == "file") {

View File

@@ -109,16 +109,20 @@ export async function GET(request: Request) {
// CREATE DISCUSSION GENERALE
// CREATE DISCUSSION GENERAL
export async function POST(request: Request) {
try {
const contentType = request.headers.get("content-type");
let idGroup, user, title, desc, member, cekFile, body: FormData | undefined
if (contentType?.includes("multipart/form-data")) {
body = await request.formData()
const dataBody = body.get("data")
cekFile = body.has("file0");
({ idGroup, user, title, desc, member } = JSON.parse(dataBody as string))
} else {
({ idGroup, user, title, desc, member } = await request.json());
}
const body = await request.formData()
const dataBody = body.get("data")
const cekFile = body.has("file0")
// const { idGroup, user, title, desc, member } = await request.json();
const { idGroup, user, title, desc, member } = JSON.parse(dataBody as string)
const userMobile = await funGetUserById({ id: user })
@@ -153,7 +157,7 @@ export async function POST(request: Request) {
})
if (cekFile) {
if (cekFile && body) {
body.delete("data")
for (var pair of body.entries()) {
if (String(pair[0]).substring(0, 4) == "file") {
@@ -213,7 +217,7 @@ export async function POST(request: Request) {
where: {
isActive: true,
idUserRole: "supadmin",
idVillage: user.idVillage
idVillage: String(userMobile.idVillage)
},
select: {
id: true,

View File

@@ -227,12 +227,18 @@ export async function PUT(request: Request, context: { params: { id: string } })
export async function POST(request: Request, context: { params: { id: string } }) {
try {
const { id } = context.params
const body = await request.formData()
const dataBody = body.get("data")
const cekFile = body.has("file0")
const contentType = request.headers.get("content-type");
// const { title, desc, user } = (await request.json())
const { title, desc, user, oldFile } = JSON.parse(dataBody as string)
let title, desc, user, oldFile: any[] = [], cekFile, body: FormData | undefined
if (contentType?.includes("multipart/form-data")) {
body = await request.formData()
const dataBody = body.get("data")
cekFile = body.has("file0");
({ title, desc, user, oldFile } = JSON.parse(dataBody as string))
} else {
({ title, desc, user } = await request.json());
}
const userMobile = await funGetUserById({ id: String(user) })
@@ -273,7 +279,7 @@ export async function POST(request: Request, context: { params: { id: string } }
}
}
if (cekFile) {
if (cekFile && body) {
body.delete("data")
for (var pair of body.entries()) {
if (String(pair[0]).substring(0, 4) == "file") {

View File

@@ -102,12 +102,17 @@ export async function GET(request: Request) {
// CREATE DISCUSSION
export async function POST(request: Request) {
try {
const body = await request.formData()
const dataBody = body.get("data")
const cekFile = body.has("file0")
const contentType = request.headers.get("content-type");
// const { idDivision, desc, user } = (await request.json());
const { idDivision, desc, user } = JSON.parse(String(dataBody));
let idDivision, desc, user, cekFile, body: FormData | undefined
if (contentType?.includes("multipart/form-data")) {
body = await request.formData()
const dataBody = body.get("data")
cekFile = body.has("file0");
({ idDivision, desc, user } = JSON.parse(String(dataBody)));
} else {
({ idDivision, desc, user } = await request.json());
}
const userMobile = await funGetUserById({ id: String(user) })
@@ -143,7 +148,7 @@ export async function POST(request: Request) {
});
if (cekFile) {
if (cekFile && body) {
body.delete("data")
for (var pair of body.entries()) {
if (String(pair[0]).substring(0, 4) == "file") {

View File

@@ -18,10 +18,6 @@ export async function GET(request: Request) {
const dataSkip = Number(page) * 10 - 10;
const tahun = searchParams.get("year");
const tahunFilter = tahun ? tahun : new Date().getFullYear().toString();
const startTahun = new Date(`${tahunFilter}-01-01T00:00:00.000Z`);
const endTahun = new Date(`${parseInt(tahunFilter) + 1}-01-01T00:00:00.000Z`);
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 });
@@ -38,6 +34,15 @@ export async function GET(request: Request) {
return NextResponse.json({ success: false, message: "Gagal mendapatkan divisi, data tidak ditemukan", }, { status: 200 });
}
let tahunFilter = String(tahun)
if (tahunFilter == "null" || tahunFilter == undefined || tahunFilter == "" || tahunFilter == "undefined") {
tahunFilter = new Date().getFullYear().toString();
}
const startTahun = new Date(`${tahunFilter}-01-01T00:00:00.000Z`);
const endTahun = new Date(`${parseInt(tahunFilter) + 1}-01-01T00:00:00.000Z`);
const data = await prisma.divisionProject.findMany({
skip: dataSkip,
take: 10,
@@ -104,7 +109,7 @@ export async function GET(request: Request) {
}
})
return NextResponse.json({ success: true, message: "Berhasil mendapatkan divisi", data: formatData, total: totalData }, { status: 200 });
return NextResponse.json({ success: true, message: "Berhasil mendapatkan divisi", data: formatData, tahun: tahunFilter, total: totalData }, { status: 200 });
} catch (error) {
console.error(error);

View File

@@ -35,7 +35,13 @@ export async function GET(request: Request) {
// (opsional) urutkan dari terbaru ke lama
uniqueYears.sort((a, b) => b - a);
const formattedData = uniqueYears.map(year => ({
id: String(year),
name: String(year)
}));
return NextResponse.json({ success: true, message: "Success", data: uniqueYears }, { status: 200 });
return NextResponse.json({ success: true, message: "Success", data: formattedData }, { status: 200 });
}

View File

@@ -217,7 +217,7 @@ export async function PUT(request: Request, context: { params: { id: string } })
const resize = await sharp(imageBuffer).resize(300).toBuffer();
// Convert buffer ke Blob
const blob = new Blob([resize], { type: file.type });
const blob = new Blob([resize as any], { type: file.type });
// Convert Blob ke File
const resizedFile = new File([blob], fileName, {

View File

@@ -133,7 +133,7 @@ export async function PUT(request: Request) {
const resize = await sharp(imageBuffer).resize(300).toBuffer();
// Convert buffer ke Blob
const blob = new Blob([resize], { type: file.type });
const blob = new Blob([resize as any], { type: file.type });
// Convert Blob ke File
const resizedFile = new File([blob], fileName, {

View File

@@ -210,7 +210,7 @@ export async function POST(request: Request) {
const resize = await sharp(imageBuffer).resize(300).toBuffer();
// Convert buffer ke Blob
const blob = new Blob([resize], { type: file.type });
const blob = new Blob([resize as any], { type: file.type });
// Convert Blob ke File
const resizedFile = new File([blob], fileName, {

View File

@@ -0,0 +1,25 @@
import { prisma } from "@/module/_global";
import { NextResponse } from "next/server";
export async function GET(request: Request) {
try {
const data = await prisma.setting.findMany({
where: {
isActive: true,
id: {
contains: "mobile_"
}
},
select: {
id: true,
name: true,
value: true
}
})
return NextResponse.json({ success: true, data }, { status: 200 });
} catch (error) {
console.error(error);
return NextResponse.json({ success: false, reason: (error as Error).message, }, { status: 500 });
}
}

View File

@@ -0,0 +1,628 @@
import formatDateTime from "@/lib/formatDateTime";
import { prisma } from "@/module/_global";
import cors from "@elysiajs/cors";
import { swagger } from "@elysiajs/swagger";
import Elysia, { t } from "elysia";
import _ from "lodash";
import moment from "moment";
import "moment/locale/id";
// Gabungkan semua ke dalam satu instance server yang dipasang di /api/monitoring
const MonitoringServer = new Elysia({ prefix: "/api/monitoring" })
.use(cors({
origin: "*",
methods: ["GET", "POST", "OPTIONS"],
}))
.use(swagger({
path: "/docs", // Karena prefix instance adalah /api/monitoring, maka ini akan diakses di /api/monitoring/docs
documentation: {
info: {
title: "Des Plus - Monitoring API",
version: "1.0.0",
description: "API Khusus untuk kebutuhan Dashboard Monitoring",
}
}
}))
.get(
"/grid-overview",
async ({ query, set }) => {
try {
const version = await prisma.setting.findMany({
select: {
id: true,
name: true,
value: true
}
});
const result_version = Object.fromEntries(version.map(item => [item.id, item.value]));
const activity_today = await prisma.userLog.count({
where: {
createdAt: {
gte: moment().subtract(1, 'days').toDate(),
lte: moment().toDate(),
}
}
})
const activity_yesterday = await prisma.userLog.count({
where: {
createdAt: {
gte: moment().subtract(2, 'days').toDate(),
lte: moment().subtract(1, 'days').toDate(),
}
}
})
const activity_increase = (activity_today - activity_yesterday);
const percentage_increase = (activity_increase / activity_yesterday) * 100
const total_village = await prisma.village.findMany({
where: {
isDummy: false
}
})
const total_village_active = total_village.filter((item) => item.isActive).length
const total_village_inactive = total_village.filter((item) => !item.isActive).length
return {
success: true,
message: "Berhasil mendapatkan data",
data: {
version: result_version,
activity: {
today: activity_today,
increase: _.isNaN(percentage_increase) ? 0 : percentage_increase.toFixed(2),
},
village: {
active: total_village_active,
inactive: total_village_inactive,
},
},
};
} catch (error) {
console.error("[overview] grid-overview error:", error);
set.status = 500;
return {
success: false,
message: "Terjadi kesalahan pada server",
data: null,
};
}
},
{
detail: {
summary: "Grid Overview",
description: "Menu Overview - Mendapatkan daftar versi aplikasi.",
tags: ["overview"],
},
}
)
.get(
"/daily-activity",
async ({ query, set }) => {
try {
// const data = await prisma.userLog.findMany({
// where: {
// User: {
// Village: {
// isDummy: false
// }
// },
// createdAt: {
// gte: moment().subtract(7, 'days').toDate(),
// lte: moment().toDate(),
// }
// },
// select: {
// createdAt: true,
// }
// })
const data = await prisma.$queryRaw`
SELECT
DATE(ul."createdAt") AS tanggal,
COUNT(*) AS total
FROM "UserLog" ul
JOIN "User" u ON ul."idUser" = u."id"
JOIN "Village" v ON u."idVillage" = v."id"
WHERE v."isDummy" = false
AND ul."createdAt" >= NOW() - INTERVAL '7 days'
GROUP BY tanggal
ORDER BY tanggal;` as any[];
const result = [];
// ubah data ke map biar gampang lookup
const map = data.reduce((acc: any, item: any) => {
const key = moment(item.tanggal).format('YYYY-MM-DD');
acc[key] = Number(item.total);
return acc;
}, {});
// generate 7 hari terakhir
for (let i = 6; i >= 0; i--) {
const date = moment().subtract(i, 'days');
const key = date.format('YYYY-MM-DD');
result.push({
date: date.format('DD MMM'),
logs: map[key] || 0
});
}
return {
success: true,
message: "Berhasil mendapatkan data",
data: result,
};
} catch (error) {
console.error("[overview] daily-activity error:", error);
set.status = 500;
return {
success: false,
message: "Terjadi kesalahan pada server",
data: null,
};
}
},
{
detail: {
summary: "Daily Activity",
description: "Menu Overview - Mendapatkan data grafik aktivitas harian semua desa.",
tags: ["overview"],
},
}
)
.get(
"/comparison-activity",
async ({ query, set }) => {
try {
const data = await prisma.$queryRaw`
SELECT
v."name",
COUNT(ul."id") AS total_logs
FROM "UserLog" ul
JOIN "User" u ON ul."idUser" = u."id"
JOIN "Village" v ON u."idVillage" = v."id"
WHERE v."isDummy" = false
AND ul."createdAt" >= NOW() - INTERVAL '7 days'
GROUP BY v."id", v."name"
ORDER BY total_logs DESC;
` as any[];
const result = data.map(item => ({
village: item.name,
activity: Number(item.total_logs)
}));
return {
success: true,
message: "Berhasil mendapatkan data",
data: result,
};
} catch (error) {
console.error("[overview] comparison-activity error:", error);
set.status = 500;
return {
success: false,
message: "Terjadi kesalahan pada server",
data: null,
};
}
},
{
detail: {
summary: "Comparison Activity",
description: "Menu Overview - Mendapatkan data grafik perbandingan aktivitas desa selama 7 hari terakhir.",
tags: ["overview"],
},
}
)
.get(
"/get-villages",
async ({ query, set }) => {
const { search, page } = query;
const pageNum = Number(page ?? 1);
try {
const data = await prisma.village.findMany({
where: {
isDummy: false,
...(search && { name: { contains: search, mode: 'insensitive' } })
},
select: {
id: true,
name: true,
isActive: true,
createdAt: true,
User: {
where: {
idUserRole: "supadmin"
},
select: {
name: true,
},
take: 1,
},
},
skip: (pageNum - 1) * 10,
take: 10,
})
const result = data.map((village) => ({
id: village.id,
name: village.name,
isActive: village.isActive,
createdAt: formatDateTime(village.createdAt),
perbekel: village.User[0]?.name || null,
}));
return {
success: true,
message: "Berhasil mendapatkan data",
data: result,
};
} catch (error) {
console.error("[villages] get-villages error:", error);
set.status = 500;
return {
success: false,
message: "Terjadi kesalahan pada server",
data: null,
};
}
},
{
query: t.Object({
search: t.Optional(t.String({ description: "Kata kunci pencarian nama desa" })),
page: t.Optional(t.String({ description: "Halaman data (default: 1)" })),
}),
detail: {
summary: "Get Villages",
description: "Menu Villages - Mendapatkan semua data desa.",
tags: ["villages"],
},
}
)
.get(
"/info-villages",
async ({ query, set }) => {
const { id } = query;
try {
const data = await prisma.village.findUnique({
where: {
id: id,
},
select: {
id: true,
name: true,
isActive: true,
createdAt: true,
User: {
where: {
idUserRole: "supadmin"
},
select: {
name: true,
},
take: 1,
},
},
})
if (!data) {
set.status = 404;
return {
success: false,
message: "Desa tidak ditemukan",
data: null,
};
}
const result = data ? {
id: data?.id,
name: data?.name,
isActive: data?.isActive,
createdAt: data?.createdAt ? formatDateTime(data.createdAt) : null,
perbekel: data?.User[0]?.name || null,
} : null;
return {
success: true,
message: "Berhasil mendapatkan data",
data: result,
};
} catch (error) {
console.error("[detail-villages] info-villages error:", error);
set.status = 500;
return {
success: false,
message: "Terjadi kesalahan pada server",
data: null,
};
}
},
{
query: t.Object({
id: t.Optional(t.String({ description: "ID desa" })),
}),
detail: {
summary: "Info Villages",
description: "Menu Detail Villages - Mendapatkan info data desa untuk header dan kolom Informasi Sistem.",
tags: ["detail-villages"],
},
}
)
.get(
"/grid-villages",
async ({ query, set }) => {
const { id } = query;
try {
const village = await prisma.village.findUnique({
where: { id: id }
});
if (!village) {
set.status = 404;
return {
success: false,
message: "Desa tidak ditemukan",
data: null,
};
}
const dataUser = await prisma.user.findMany({
where: {
idVillage: id,
NOT: {
idUserRole: "developer"
}
}
})
const dataGroup = await prisma.group.findMany({
where: {
idVillage: id,
}
})
const dataDivision = await prisma.division.findMany({
where: {
idVillage: id,
}
})
const dataProject = await prisma.project.findMany({
where: {
idVillage: id
}
})
const result = {
user: {
active: dataUser.filter((user) => user.isActive).length,
nonActive: dataUser.filter((user) => !user.isActive).length,
},
group: {
active: dataGroup.filter((group) => group.isActive).length,
nonActive: dataGroup.filter((group) => !group.isActive).length,
},
division: {
active: dataDivision.filter((division) => division.isActive).length,
nonActive: dataDivision.filter((division) => !division.isActive).length,
},
project: {
active: dataProject.filter((project) => project.isActive).length,
nonActive: dataProject.filter((project) => !project.isActive).length,
}
};
return {
success: true,
message: "Berhasil mendapatkan data",
data: result,
};
} catch (error) {
console.error("[detail-villages] grid-villages error:", error);
set.status = 500;
return {
success: false,
message: "Terjadi kesalahan pada server",
data: null,
};
}
},
{
query: t.Object({
id: t.Optional(t.String({ description: "ID desa" })),
}),
detail: {
summary: "Grid Villages",
description: "Menu Grid Villages - Mendapatkan info data desa untuk 4 grid untuk halaman detail desa.",
tags: ["detail-villages"],
},
}
)
.get("/log-villages", async ({ query, set }) => {
const { id, time } = query;
try {
const village = await prisma.village.findUnique({
where: { id },
});
if (!village) {
set.status = 404;
return {
success: false,
message: "Desa tidak ditemukan",
data: null,
};
}
const now = new Date();
let startDate: Date;
if (time === "daily") {
startDate = new Date();
startDate.setDate(now.getDate() - 13); // 14 hari
} else if (time === "monthly") {
startDate = new Date(now.getFullYear(), 0, 1); // awal tahun
} else if (time === "yearly") {
startDate = new Date(now.getFullYear() - 4, 0, 1); // 5 tahun terakhir (opsional)
} else {
startDate = new Date(0);
}
const dataLog = await prisma.userLog.findMany({
where: {
createdAt: {
gte: startDate,
},
User: {
idVillage: id,
},
},
select: {
createdAt: true,
},
});
// =========================
// 🔥 GROUPING
// =========================
const map: Record<string, number> = {};
dataLog.forEach((log) => {
const date = new Date(log.createdAt);
let label = "";
if (time === "daily") {
label = date.toLocaleDateString("id-ID", {
day: "2-digit",
month: "short",
});
} else if (time === "monthly") {
label = date.toLocaleDateString("id-ID", {
month: "short",
});
} else if (time === "yearly") {
label = date.getFullYear().toString();
}
map[label] = (map[label] || 0) + 1;
});
// =========================
// 🔥 FORMAT FINAL
// =========================
let result: any[] = [];
if (time === "daily") {
for (let i = 13; i >= 0; i--) {
const d = new Date();
d.setDate(d.getDate() - i);
const label = d.toLocaleDateString("id-ID", {
day: "2-digit",
month: "short",
});
result.push({
label,
aktivitas: map[label] || 0,
});
}
} else if (time === "monthly") {
const year = now.getFullYear();
for (let m = 0; m <= 11; m++) {
const d = new Date(year, m, 1);
const label = d.toLocaleDateString("id-ID", {
month: "short",
});
result.push({
label,
aktivitas: map[label] || 0,
});
}
} else if (time === "yearly") {
const years = Object.keys(map).map(Number);
if (years.length === 0) {
const currentYear = new Date().getFullYear();
result = [
{ label: currentYear.toString(), aktivitas: 0 }
];
} else {
const minYear = Math.min(...years);
const maxYear = Math.max(...years);
result = [];
for (let y = minYear; y <= maxYear; y++) {
const label = y.toString();
result.push({
label,
aktivitas: map[label] || 0,
});
}
}
}
return {
success: true,
message: "Berhasil mendapatkan data",
data: result,
};
} catch (error) {
console.error("[log-villages] error:", error);
set.status = 500;
return {
success: false,
message: "Terjadi kesalahan pada server",
data: null,
};
}
},
{
query: t.Object({
id: t.String({ description: "ID desa" }),
time: t.Enum(
{
daily: "daily",
monthly: "monthly",
yearly: "yearly",
},
{
description: "Rentang waktu (daily = 14 hari, monthly = 1 tahun, yearly = per tahun)",
}
),
}),
detail: {
summary: "Log Villages",
description:
"Mendapatkan data log aktivitas desa berdasarkan rentang waktu (harian, bulanan, tahunan)",
tags: ["detail-villages"],
},
}
);
;
export const GET = MonitoringServer.handle;
export const POST = MonitoringServer.handle;

View File

@@ -0,0 +1,687 @@
import { prisma } from "@/module/_global";
import cors from "@elysiajs/cors";
import { swagger } from "@elysiajs/swagger";
import Elysia, { t } from "elysia";
import _ from "lodash";
import moment from "moment";
import "moment/locale/id";
// Gabungkan semua ke dalam satu instance server yang dipasang di /api/noc
const NocServer = new Elysia({ prefix: "/api/noc" })
.use(cors({
origin: "*",
methods: ["GET", "POST", "OPTIONS"],
}))
.use(swagger({
path: "/docs", // Karena prefix instance adalah /api/noc, maka ini akan diakses di /api/noc/docs
documentation: {
info: {
title: "Sistem Desa Mandiri - NOC API",
version: "1.0.0",
description: "API Khusus untuk kebutuhan NOC (Network Operation Center) dan Monitoring Desa",
},
tags: [
{ name: "NOC", description: "Endpoint khusus monitoring" }
]
}
}))
// ── GET /api/noc/active-divisions ──────────────────────────────────────────
.get(
"/active-divisions",
async ({ query, set }) => {
const { idDesa, limit } = query;
if (!idDesa) {
set.status = 400;
return {
success: false,
message: "Parameter idDesa wajib diisi",
data: null,
};
}
const maxResults = Number(limit ?? 5);
try {
// Cek apakah desa ada
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,
};
}
// Ambil semua divisi milik desa ini
const divisions = await prisma.division.findMany({
where: {
idVillage: idDesa,
isActive: true,
},
select: {
id: true,
name: true,
idGroup: true,
Group: {
select: {
name: true,
},
},
_count: {
select: {
DivisionProject: true,
},
},
},
});
// Hitung total kegiatan per divisi & urutkan descending, ambil top sesuai limit
const ranked = divisions
.map((d: any) => ({
id: d.id,
division: d.name,
group: d.Group.name,
totalKegiatan: d._count.DivisionProject
}))
.sort((a: any, b: any) => b.totalKegiatan - a.totalKegiatan)
.slice(0, maxResults);
return {
success: true,
message: "Berhasil mendapatkan divisi teraktif",
data: {
idDesa: village.id,
namaDesa: village.name,
divisi: ranked,
},
};
} catch (error) {
console.error("[NOC] active-divisions 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: "Jumlah maksimal data (default: 5)" })),
}),
detail: {
summary: "Divisi Teraktif",
description: "Menu Beranda - Mendapatkan daftar divisi teraktif berdasarkan jumlah proyek pada desa tertentu.",
tags: ["NOC"],
},
}
)
// ── GET /api/noc/latest-projects ──────────────────────────────────────────
.get(
"/latest-projects",
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 ?? 5), 50);
try {
// Cek apakah desa ada
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,
};
}
// Ambil proyek umum terbaru dari desa ini
const projects = await prisma.project.findMany({
where: {
idVillage: idDesa,
isActive: true,
},
select: {
id: true,
title: true,
status: true,
desc: true,
updatedAt: true,
Group: {
select: {
name: true,
},
},
User: {
select: {
name: true,
},
},
},
orderBy: {
updatedAt: "desc",
},
take: maxResults,
});
const mapped = projects.map((p: any) => ({
id: p.id,
title: p.title,
status: p.status,
desc: p.desc,
group: p.Group.name,
createdBy: p.User.name,
updatedAt: p.updatedAt,
}));
return {
success: true,
message: "Berhasil mendapatkan proyek terbaru",
data: {
idDesa: village.id,
namaDesa: village.name,
total: mapped.length,
projects: mapped,
},
};
} catch (error) {
console.error("[NOC] latest-projects 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: "Jumlah maksimal proyek (default: 5, maks: 50)" })
),
}),
detail: {
summary: "Latest Projects General",
description: "Menu kinerja divisi - Mendapatkan daftar proyek umum terbaru dari berbagai grup pada desa tertentu.",
tags: ["NOC"],
},
}
)
// ── GET /api/noc/upcoming-events ───────────────────────────────────────────
.get(
"/upcoming-events",
async ({ query, set }) => {
const { idDesa, limit, filter } = query;
if (!idDesa) {
set.status = 400;
return {
success: false,
message: "Parameter idDesa wajib diisi",
data: null,
};
}
const maxResults = Math.min(Number(limit ?? 10), 50);
const today = moment().startOf("day").toDate();
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 events = await prisma.divisionCalendarReminder.findMany({
where: {
isActive: true,
dateStart: {
gte: today,
},
Division: {
idVillage: idDesa,
isActive: true,
},
DivisionCalendar: {
isActive: true,
},
},
select: {
id: true,
idCalendar: true,
dateStart: true,
dateEnd: true,
timeStart: true,
timeEnd: true,
status: true,
Division: {
select: {
id: true,
name: true,
},
},
DivisionCalendar: {
select: {
title: true,
desc: true,
linkMeet: true,
repeatEventTyper: true,
User: {
select: {
name: true,
},
},
},
},
},
orderBy: [
{ dateStart: "asc" },
{ timeStart: "asc" },
],
take: maxResults,
});
const todayMoment = moment().startOf("day");
const mapper = (e: any) => ({
id: e.id,
idCalendar: e.idCalendar,
title: e.DivisionCalendar.title,
desc: e.DivisionCalendar.desc,
linkMeet: e.DivisionCalendar.linkMeet ?? null,
repeatEventTyper: e.DivisionCalendar.repeatEventTyper,
dateStart: moment(e.dateStart).format("YYYY-MM-DD"),
dateEnd: e.dateEnd
? moment(e.dateEnd).format("YYYY-MM-DD")
: null,
timeStart: moment.utc(e.timeStart).format("HH:mm"),
timeEnd: moment.utc(e.timeEnd).format("HH:mm"),
status: e.status,
createdBy: e.DivisionCalendar.User.name,
divisi: {
id: e.Division.id,
name: e.Division.name,
},
});
const todayEvents = events.filter((e: any) => moment(e.dateStart).isSame(todayMoment, 'day')).map(mapper);
const upcomingEvents = events.filter((e: any) => 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 {
success: true,
message: "Berhasil mendapatkan events",
data: data,
};
} catch (error) {
console.error("[NOC] upcoming-events 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: "Jumlah maksimal event (default: 10, maks: 50)" })
),
filter: t.Optional(
t.String({ description: "Filter event: 'today' atau 'upcoming'" })
),
}),
detail: {
summary: "Events (Today & Upcoming)",
description: "Menu beranda dan kinerja divisi - 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: "Menu kinerja divisi - 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: any, { _count }: any) => 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: "Menu kinerja divisi - 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: "Menu kinerja divisi - Mendapatkan latest discussion pada desa tertentu.",
tags: ["NOC"],
},
}
);
export const GET = NocServer.handle;
export const POST = NocServer.handle;

View File

@@ -215,7 +215,7 @@ export async function PUT(request: Request, context: { params: { id: string } })
const resize = await sharp(imageBuffer).resize(300).toBuffer();
// Convert buffer ke Blob
const blob = new Blob([resize], { type: file.type });
const blob = new Blob([resize as any], { type: file.type });
// Convert Blob ke File
const resizedFile = new File([blob], fileName, {

View File

@@ -133,7 +133,7 @@ export async function PUT(request: Request) {
const resize = await sharp(imageBuffer).resize(300).toBuffer();
// Convert buffer ke Blob
const blob = new Blob([resize], { type: file.type });
const blob = new Blob([resize as any], { type: file.type });
// Convert Blob ke File
const resizedFile = new File([blob], fileName, {

View File

@@ -207,7 +207,7 @@ export async function POST(request: Request) {
const resize = await sharp(imageBuffer).resize(300).toBuffer();
// Convert buffer ke Blob
const blob = new Blob([resize], { type: file.type });
const blob = new Blob([resize as any], { type: file.type });
// Convert Blob ke File
const resizedFile = new File([blob], fileName, {

View File

@@ -2,7 +2,7 @@ import { NextResponse } from "next/server";
export async function GET(request: Request) {
try {
return NextResponse.json({ success: true, version: "2.1.0", tahap: "beta", update: "-api mobile; -login tanpa otp (mobile app); -tambah laporan pada project dan tugas divisi; -tambah upload link pada project dan tugas divisi; -tambah detail tanggal dan jam pada project dan tugas divisi; -api jenna ai; -privacy policy" }, { status: 200 });
return NextResponse.json({ success: true, version: "2.1.9", tahap: "beta", update: "-api untuk dashboard monitoring" }, { status: 200 });
} catch (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 });

41
src/app/global-error.tsx Normal file
View File

@@ -0,0 +1,41 @@
"use client";
export default function GlobalError({
reset,
}: {
error: Error & { digest?: string };
reset: () => void;
}) {
return (
<html lang="en">
<body
style={{
backgroundColor: "#252A2F",
display: "flex",
flexDirection: "column",
alignItems: "center",
justifyContent: "center",
minHeight: "100vh",
gap: 16,
fontFamily: "Lato, sans-serif",
color: "white",
}}
>
<h2>Terjadi Kesalahan</h2>
<button
onClick={() => reset()}
style={{
padding: "8px 16px",
borderRadius: 4,
border: "1px solid #ccc",
background: "transparent",
color: "white",
cursor: "pointer",
}}
>
Coba Lagi
</button>
</body>
</html>
);
}

View File

@@ -9,6 +9,8 @@ import '@mantine/notifications/styles.css';
import { Lato } from "next/font/google";
import { Toaster } from 'react-hot-toast';
export const dynamic = 'force-dynamic';
export const metadata = {
title: "SISTEM DESA MANDIRI",
description: "I have followed setup instructions carefully",

24
src/app/not-found.tsx Normal file
View File

@@ -0,0 +1,24 @@
import { Box, Text, Button } from "@mantine/core";
import Link from "next/link";
export default function NotFound() {
return (
<Box
style={{
display: "flex",
flexDirection: "column",
alignItems: "center",
justifyContent: "center",
minHeight: "100vh",
gap: 16,
}}
>
<Text size="xl" fw={700} c="white">
404 - Halaman Tidak Ditemukan
</Text>
<Button component={Link} href="/" variant="light">
Kembali ke Beranda
</Button>
</Box>
);
}

11
src/lib/formatDateTime.ts Normal file
View File

@@ -0,0 +1,11 @@
function formatDateTime(date: Date) {
return new Intl.DateTimeFormat('id-ID', {
hour: '2-digit',
minute: '2-digit',
day: '2-digit',
month: 'short',
year: 'numeric',
}).format(date);
}
export default formatDateTime

View File

@@ -5,7 +5,6 @@ import { useFocusTrap } from "@mantine/hooks";
import { useState } from "react";
import toast from "react-hot-toast";
import ViewVerification from "../../varification/view/view_verification";
function ViewLogin() {
const focusTrapRef = useFocusTrap()
const textInfo = "Kami akan mengirimkan kode verifikasi melalui WhatsApp untuk mengonfirmasi nomor Anda.";
@@ -34,23 +33,24 @@ function ViewLogin() {
})
const cekLogin = await cek.json()
if (cekLogin.success) {
const code = Math.floor(1000 + Math.random() * 9000)
try {
const res = await fetch(`https://wa.wibudev.com/code?nom=${cekLogin.phone}&text=*DARMASABA*%0A%0A
JANGAN BERIKAN KODE RAHASIA ini kepada siapa pun TERMASUK PIHAK DARMASABA. Masukkan otentikasi: *${encodeURIComponent(code)}*`).then(
async (res) => {
if (res.status == 200) {
setValPhone(cekLogin.phone)
setOTP(code)
setUser(cekLogin.id)
setVerif(true)
toast.success('Kode verifikasi telah dikirim')
} else {
console.error(res.status)
toast.error('Internal Server Error')
}
}
)
const res = await fetch('/api/auth/otp', {
method: 'POST',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify({ phone: isPhone })
})
const data = await res.json()
if (data.success) {
setValPhone(data.phone)
setOTP(data.otp)
setUser(data.id)
setVerif(true)
toast.success('Kode verifikasi telah dikirim')
} else {
toast.error(data.message || 'Gagal mengirim kode verifikasi')
}
} catch (error) {
console.error(error)
toast.error('Internal Server Error')

View File

@@ -15,19 +15,20 @@ export default function ViewVerification({ phone, otp, user }: IVerification) {
async function onResend() {
try {
const code = Math.floor(1000 + Math.random() * 9000)
const res = await fetch(`https://wa.wibudev.com/code?nom=${phone}&text=*DARMASABA*%0A%0A
JANGAN BERIKAN KODE RAHASIA ini kepada siapa pun TERMASUK PIHAK DARMASABA. Masukkan otentikasi: *${encodeURIComponent(code)}*`)
.then(
async (res) => {
if (res.status == 200) {
toast.success('Kode verifikasi telah dikirim')
setOTP(code)
} else {
toast.error('Internal Server Error')
}
}
);
const res = await fetch('/api/auth/otp', {
method: 'POST',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify({ phone })
})
const data = await res.json()
if (data.success) {
toast.success('Kode verifikasi telah dikirim')
setOTP(data.otp)
} else {
toast.error(data.message || 'Gagal mengirim ulang kode')
}
} catch (error) {
console.error(error)
toast.error('Internal Server Error')

View File

@@ -0,0 +1,922 @@
import { prisma } from "@/module/_global";
import {
seederAdmin,
seederAdminRole,
seederDesa,
seederGroup,
seederPosition,
seederTheme,
seederUser,
seederUserRole
} from '@/module/seeder';
async function seedCompleteVillageData() {
console.log("Starting complete village data seeding process...");
// Define comprehensive dummy data variables outside transaction scope
// GROUP - Comprehensive dummy data
const seederGroupComprehensive = [
{
"id": "group_rt01",
"idVillage": "desaDummy",
"name": "Dinas"
},
{
"id": "group_rt02",
"idVillage": "desaDummy",
"name": "Adat"
},
{
"id": "group_karang_taruna",
"idVillage": "desaDummy",
"name": "Karang Taruna"
},
{
"id": "group_bumdes",
"idVillage": "desaDummy",
"name": "PKK"
}
];
// POSITION - Comprehensive dummy data
const seederPositionComprehensive = [
{
"id": "pos_ketua_rt01",
"idGroup": "group_rt01",
"name": "Perbekel"
},
{
"id": "pos_sekretaris_rt01",
"idGroup": "group_rt01",
"name": "Sekretaris"
},
{
"id": "pos_bendahara_rt01",
"idGroup": "group_rt01",
"name": "Bendahara"
},
{
"id": "pos_staff_rt01",
"idGroup": "group_rt01",
"name": "Staff"
},
{
"id": "pos_staff_rt02",
"idGroup": "group_rt02",
"name": "Staff"
},
{
"id": "pos_ketua_karang_taruna",
"idGroup": "group_karang_taruna",
"name": "Ketua Karang Taruna"
},
{
"id": "pos_ketua_bumdes",
"idGroup": "group_bumdes",
"name": "Ketua PKK"
}
];
// USER - Comprehensive dummy data
const seederUserComprehensive = [
{
"id": "user_kades",
"idUserRole": "supadmin",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"idPosition": "pos_ketua_rt01",
"nik": "3201010101010001",
"name": "Kepala Desa",
"phone": "081234567890",
"email": "kades@desamandiri.test",
"gender": "M"
},
{
"id": "user_sekdes",
"idUserRole": "admin",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"idPosition": "pos_sekretaris_rt01",
"nik": "3201010101010002",
"name": "Sekretaris Desa",
"phone": "081234567891",
"email": "sekdes@desamandiri.test",
"gender": "M"
},
{
"id": "user_ketua_bumdes",
"idUserRole": "admin",
"idVillage": "desaDummy",
"idGroup": "group_bumdes",
"idPosition": "pos_ketua_bumdes",
"nik": "3201010101010003",
"name": "Ketua BUMDES",
"phone": "081234567892",
"email": "ketuabumdes@desamandiri.test",
"gender": "M"
},
{
"id": "user_ketua_karang_taruna",
"idUserRole": "user",
"idVillage": "desaDummy",
"idGroup": "group_karang_taruna",
"idPosition": "pos_ketua_karang_taruna",
"nik": "3201010101010004",
"name": "Ketua Karang Taruna",
"phone": "081234567893",
"email": "ketuakt@desamandiri.test",
"gender": "M"
},
{
"id": "user_warga1",
"idUserRole": "user",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"idPosition": "pos_staff_rt01",
"nik": "3201010101010005",
"name": "Warga Satu",
"phone": "081234567894",
"email": "wargasatu@desamandiri.test",
"gender": "F"
},
{
"id": "user_warga2",
"idUserRole": "user",
"idVillage": "desaDummy",
"idGroup": "group_rt02",
"idPosition": "pos_staff_rt02",
"nik": "3201010101010006",
"name": "Warga Dua",
"phone": "081234567895",
"email": "wargadua@desamandiri.test",
"gender": "M"
}
];
// ANNOUNCEMENTS - Comprehensive dummy data
const seederAnnouncementComprehensive = [
{
"id": "ann_pembangunan_jalan",
"idVillage": "desaDummy",
"title": "Pembangunan Jalan Desa Tahap 2",
"desc": "Pada bulan ini akan dilakukan pembangunan jalan desa tahap 2 yang mencakup wilayah RT 01 dan RT 02. Mohon kerjasama warga untuk menyesuaikan aktivitas selama masa pembangunan.",
"createdBy": "user_kades"
},
{
"id": "ann_posyandu",
"idVillage": "desaDummy",
"title": "Posyandu Bulanan",
"desc": "Posyandu bulan akan diselenggarakan pada tanggal 21 setiap bulannya di Balai Desa. Warga diharapkan aktif membawa balita untuk pemeriksaan kesehatan.",
"createdBy": "user_sekdes"
},
{
"id": "ann_rapat_warga",
"idVillage": "desaDummy",
"title": "Rapat Warga Bulanan",
"desc": "Rapat warga bulanan akan diselenggarakan pada hari Sabtu, 25 Februari 2026 pukul 09.00 WIB di Balai Desa. Hadirilah tepat waktu.",
"createdBy": "user_kades"
}
];
// ANNOUNCEMENT MEMBERS - Comprehensive dummy data
const seederAnnouncementMemberComprehensive = [
{
"id": "ann_mem_pembangunan_jalan_rt01",
"idAnnouncement": "ann_pembangunan_jalan",
"idGroup": "group_rt01",
"idDivision": null
},
{
"id": "ann_mem_pembangunan_jalan_rt02",
"idAnnouncement": "ann_pembangunan_jalan",
"idGroup": "group_rt02",
"idDivision": null
},
{
"id": "ann_mem_posyandu_all",
"idAnnouncement": "ann_posyandu",
"idGroup": null,
"idDivision": null
},
{
"id": "ann_mem_rapat_warga_all",
"idAnnouncement": "ann_rapat_warga",
"idGroup": null,
"idDivision": null
}
];
// DIVISIONS - Comprehensive dummy data
const seederDivisionComprehensive = [
{
"id": "div_bumdes",
"idVillage": "desaDummy",
"idGroup": "group_bumdes",
"name": "BUMDES Desa Mandiri",
"desc": "Badan Usaha Milik Desa yang bertujuan untuk meningkatkan kesejahteraan masyarakat desa melalui berbagai usaha produktif.",
"createdBy": "user_ketua_bumdes"
},
{
"id": "div_karang_taruna",
"idVillage": "desaDummy",
"idGroup": "group_karang_taruna",
"name": "Karang Taruna Desa Mandiri",
"desc": "Organisasi pemuda desa yang berfokus pada pengembangan potensi pemuda dan kegiatan sosial kemasyarakatan.",
"createdBy": "user_ketua_karang_taruna"
},
{
"id": "div_linmas",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"name": "Linmas Desa Mandiri",
"desc": "Perlindungan Masyarakat yang bertugas menjaga ketertiban dan keamanan di wilayah desa.",
"createdBy": "user_kades"
}
];
// DIVISION MEMBERS - Comprehensive dummy data
const seederDivisionMemberComprehensive = [
{
"id": "div_mem_bumdes_ketua",
"idDivision": "div_bumdes",
"idUser": "user_ketua_bumdes",
"isAdmin": true,
"isLeader": true
},
{
"id": "div_mem_bumdes_anggota1",
"idDivision": "div_bumdes",
"idUser": "user_warga1",
"isAdmin": false,
"isLeader": false
},
{
"id": "div_mem_karang_taruna_ketua",
"idDivision": "div_karang_taruna",
"idUser": "user_ketua_karang_taruna",
"isAdmin": true,
"isLeader": true
},
{
"id": "div_mem_karang_taruna_anggota1",
"idDivision": "div_karang_taruna",
"idUser": "user_warga2",
"isAdmin": false,
"isLeader": false
},
{
"id": "div_mem_linmas_kades",
"idDivision": "div_linmas",
"idUser": "user_kades",
"isAdmin": true,
"isLeader": true
}
];
// PROJECTS - Comprehensive dummy data
const seederProjectComprehensive = [
{
"id": "proj_pembangunan_jalan",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"title": "Pembangunan Jalan Desa Tahap 2",
"desc": "Pembangunan jalan desa tahap 2 yang mencakup wilayah RT 01 dan RT 02",
"status": 1,
"createdBy": "user_kades"
},
{
"id": "proj_penghijauan",
"idVillage": "desaDummy",
"idGroup": "group_karang_taruna",
"title": "Program Penghijauan Desa",
"desc": "Penanaman pohon di sepanjang jalan desa dan area publik",
"status": 0,
"createdBy": "user_ketua_karang_taruna"
},
{
"id": "proj_pembukuan_bumdes",
"idVillage": "desaDummy",
"idGroup": "group_bumdes",
"title": "Sistem Pembukuan Digital BUMDES",
"desc": "Pembuatan sistem pembukuan digital untuk BUMDES Desa Mandiri",
"status": 0,
"createdBy": "user_ketua_bumdes"
}
];
// PROJECT MEMBERS - Comprehensive dummy data
const seederProjectMemberComprehensive = [
{
"id": "proj_mem_pembangunan_jalan_kades",
"idProject": "proj_pembangunan_jalan",
"idUser": "user_kades",
"isLeader": true
},
{
"id": "proj_mem_pembangunan_jalan_sekdes",
"idProject": "proj_pembangunan_jalan",
"idUser": "user_sekdes",
"isLeader": false
},
{
"id": "proj_mem_penghijauan_ketua_kt",
"idProject": "proj_penghijauan",
"idUser": "user_ketua_karang_taruna",
"isLeader": true
},
{
"id": "proj_mem_penghijauan_warga1",
"idProject": "proj_penghijauan",
"idUser": "user_warga1",
"isLeader": false
},
{
"id": "proj_mem_pembukuan_bumdes_ketua",
"idProject": "proj_pembukuan_bumdes",
"idUser": "user_ketua_bumdes",
"isLeader": true
}
];
// PROJECT TASKS - Comprehensive dummy data
const seederProjectTaskComprehensive = [
{
"id": "task_survey_lokasi",
"idProject": "proj_pembangunan_jalan",
"title": "Survey Lokasi",
"desc": "Melakukan survey lokasi untuk menentukan titik pembangunan jalan",
"status": 1,
"dateStart": "2026-01-15T00:00:00.000Z",
"dateEnd": "2026-01-20T00:00:00.000Z"
},
{
"id": "task_pengadaan_material",
"idProject": "proj_pembangunan_jalan",
"title": "Pengadaan Material",
"desc": "Mengadakan material pembangunan seperti pasir, batu, dan semen",
"status": 0,
"dateStart": "2026-02-01T00:00:00.000Z",
"dateEnd": "2026-02-10T00:00:00.000Z"
},
{
"id": "task_pelaksanaan_pembangunan",
"idProject": "proj_pembangunan_jalan",
"title": "Pelaksanaan Pembangunan",
"desc": "Melaksanakan pembangunan jalan sesuai dengan desain yang telah ditentukan",
"status": 0,
"dateStart": "2026-02-15T00:00:00.000Z",
"dateEnd": "2026-03-15T00:00:00.000Z"
},
{
"id": "task_penanaman_pohon",
"idProject": "proj_penghijauan",
"title": "Penanaman Pohon",
"desc": "Menanam pohon di sepanjang jalan desa dan area publik",
"status": 0,
"dateStart": "2026-03-01T00:00:00.000Z",
"dateEnd": "2026-03-15T00:00:00.000Z"
}
];
// DISCUSSIONS - Comprehensive dummy data
const seederDiscussionComprehensive = [
{
"id": "disc_kegiatan_desa",
"idVillage": "desaDummy",
"idGroup": null,
"title": "Pembahasan Kegiatan Desa Mendatang",
"desc": "Diskusi untuk merencanakan kegiatan desa yang akan datang dan menyerap aspirasi warga",
"status": 1,
"createdBy": "user_kades"
},
{
"id": "disc_pengelolaan_sampah",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"title": "Pengelolaan Sampah di RT 01",
"desc": "Diskusi internal RT 01 mengenai pengelolaan sampah rumah tangga dan lingkungan",
"status": 1,
"createdBy": "user_kades"
},
{
"id": "disc_program_karang_taruna",
"idVillage": "desaDummy",
"idGroup": "group_karang_taruna",
"title": "Program Kerja Karang Taruna",
"desc": "Merancang program kerja Karang Taruna untuk tahun ini",
"status": 1,
"createdBy": "user_ketua_karang_taruna"
}
];
// DISCUSSION MEMBERS - Comprehensive dummy data
const seederDiscussionMemberComprehensive = [
{
"id": "disc_mem_kegiatan_desa_kades",
"idDiscussion": "disc_kegiatan_desa",
"idUser": "user_kades"
},
{
"id": "disc_mem_kegiatan_desa_sekdes",
"idDiscussion": "disc_kegiatan_desa",
"idUser": "user_sekdes"
},
{
"id": "disc_mem_kegiatan_desa_warga1",
"idDiscussion": "disc_kegiatan_desa",
"idUser": "user_warga1"
},
{
"id": "disc_mem_pengelolaan_sampah_kades",
"idDiscussion": "disc_pengelolaan_sampah",
"idUser": "user_kades"
},
{
"id": "disc_mem_pengelolaan_sampah_warga1",
"idDiscussion": "disc_pengelolaan_sampah",
"idUser": "user_warga1"
},
{
"id": "disc_mem_program_kt_ketua",
"idDiscussion": "disc_program_karang_taruna",
"idUser": "user_ketua_karang_taruna"
},
{
"id": "disc_mem_program_kt_warga2",
"idDiscussion": "disc_program_karang_taruna",
"idUser": "user_warga2"
}
];
try {
// Start transaction to ensure data consistency
await prisma.$transaction(async (tx) => {
// ADMIN ROLE
for (let data of seederAdminRole) {
await tx.adminRole.upsert({
where: {
id: data.id
},
update: {
name: data.name
},
create: {
id: data.id,
name: data.name,
},
})
}
// ADMIN
for (let data of seederAdmin) {
await tx.admin.upsert({
where: {
id: data.id
},
update: {
name: data.name,
idAdminRole: data.idAdminRole,
phone: data.phone,
email: data.email,
gender: data.gender
},
create: {
id: data.id,
idAdminRole: data.idAdminRole,
phone: data.phone,
email: data.email,
gender: data.gender,
name: data.name
},
})
}
// THEME
for (let data of seederTheme) {
await tx.colorTheme.upsert({
where: {
id: data.id
},
update: {
name: data.name,
utama: data.utama,
bgUtama: data.bgUtama,
bgIcon: data.bgIcon,
bgFiturHome: data.bgFiturHome,
bgFiturDivision: data.bgFiturDivisi,
bgTotalKegiatan: data.bgTotalKegiatan
},
create: {
id: data.id,
name: data.name,
utama: data.utama,
bgUtama: data.bgUtama,
bgIcon: data.bgIcon,
bgFiturHome: data.bgFiturHome,
bgFiturDivision: data.bgFiturDivisi,
bgTotalKegiatan: data.bgTotalKegiatan
}
})
}
// DESA - Original data
for (let data of seederDesa) {
await tx.village.upsert({
where: {
id: data.id
},
update: {
name: data.name,
desc: data.desc,
idTheme: "theme1"
},
create: {
id: data.id,
name: data.name,
desc: data.desc,
idTheme: "theme1"
}
})
}
// GROUP - Original data
for (let data of seederGroup) {
await tx.group.upsert({
where: {
id: data.id
},
update: {
name: data.name,
idVillage: data.idVillage
},
create: {
id: data.id,
name: data.name,
idVillage: data.idVillage
}
})
}
for (let data of seederGroupComprehensive) {
await tx.group.upsert({
where: {
id: data.id
},
update: {
name: data.name,
idVillage: data.idVillage
},
create: {
id: data.id,
name: data.name,
idVillage: data.idVillage
}
})
}
// POSITION - Original data
for (let data of seederPosition) {
await tx.position.upsert({
where: {
id: data.id
},
update: {
name: data.name,
idGroup: data.idGroup
},
create: {
id: data.id,
name: data.name,
idGroup: data.idGroup
}
})
}
for (let data of seederPositionComprehensive) {
await tx.position.upsert({
where: {
id: data.id
},
update: {
name: data.name,
idGroup: data.idGroup
},
create: {
id: data.id,
name: data.name,
idGroup: data.idGroup
}
})
}
// USER ROLE
for (let data of seederUserRole) {
await tx.userRole.upsert({
where: {
id: data.id
},
update: {
name: data.name
},
create: {
id: data.id,
name: data.name,
desc: data.desc
},
})
}
// USER - Original data
for (let data of seederUser) {
await tx.user.upsert({
where: {
id: data.id
},
update: {
idVillage: data.idVillage,
idGroup: data.idGroup,
idPosition: data.idPosition,
idUserRole: data.idUserRole,
nik: data.nik,
name: data.name,
phone: data.phone,
email: data.email,
gender: data.gender
},
create: {
id: data.id,
idVillage: data.idVillage,
idGroup: data.idGroup,
idPosition: data.idPosition,
idUserRole: data.idUserRole,
nik: data.nik,
name: data.name,
phone: data.phone,
email: data.email,
gender: data.gender
},
})
}
for (let data of seederUserComprehensive) {
await tx.user.upsert({
where: {
id: data.id
},
update: {
idVillage: data.idVillage,
idGroup: data.idGroup,
idPosition: data.idPosition,
idUserRole: data.idUserRole,
nik: data.nik,
name: data.name,
phone: data.phone,
email: data.email,
gender: data.gender
},
create: {
id: data.id,
idVillage: data.idVillage,
idGroup: data.idGroup,
idPosition: data.idPosition,
idUserRole: data.idUserRole,
nik: data.nik,
name: data.name,
phone: data.phone,
email: data.email,
gender: data.gender
},
})
}
for (let data of seederAnnouncementComprehensive) {
await tx.announcement.upsert({
where: {
id: data.id
},
update: {
title: data.title,
desc: data.desc,
createdBy: data.createdBy
},
create: {
id: data.id,
idVillage: data.idVillage,
title: data.title,
desc: data.desc,
createdBy: data.createdBy,
isActive: true
}
})
}
for (let data of seederAnnouncementMemberComprehensive) {
await tx.announcementMember.upsert({
where: {
id: data.id
},
update: {
idAnnouncement: data.idAnnouncement,
idGroup: data.idGroup!,
idDivision: data.idDivision!
},
create: {
id: data.id,
idAnnouncement: data.idAnnouncement,
idGroup: data.idGroup!,
idDivision: data.idDivision!,
isActive: true
}
})
}
for (let data of seederDivisionComprehensive) {
await tx.division.upsert({
where: {
id: data.id
},
update: {
name: data.name,
desc: data.desc,
createdBy: data.createdBy
},
create: {
id: data.id,
idVillage: data.idVillage,
idGroup: data.idGroup,
name: data.name,
desc: data.desc,
createdBy: data.createdBy,
isActive: true
}
})
}
for (let data of seederDivisionMemberComprehensive) {
await tx.divisionMember.upsert({
where: {
id: data.id
},
update: {
idUser: data.idUser,
isAdmin: data.isAdmin,
isLeader: data.isLeader
},
create: {
id: data.id,
idDivision: data.idDivision,
idUser: data.idUser,
isAdmin: data.isAdmin,
isLeader: data.isLeader,
isActive: true
}
})
}
for (let data of seederProjectComprehensive) {
await tx.project.upsert({
where: {
id: data.id
},
update: {
title: data.title,
desc: data.desc,
status: data.status,
createdBy: data.createdBy
},
create: {
id: data.id,
idVillage: data.idVillage,
idGroup: data.idGroup,
title: data.title,
desc: data.desc,
status: data.status,
createdBy: data.createdBy,
isActive: true
}
})
}
for (let data of seederProjectMemberComprehensive) {
await tx.projectMember.upsert({
where: {
id: data.id
},
update: {
idUser: data.idUser,
isLeader: data.isLeader
},
create: {
id: data.id,
idProject: data.idProject,
idUser: data.idUser,
isLeader: data.isLeader,
isActive: true
}
})
}
for (let data of seederProjectTaskComprehensive) {
await tx.projectTask.upsert({
where: {
id: data.id
},
update: {
title: data.title,
desc: data.desc,
status: data.status,
dateStart: new Date(data.dateStart),
dateEnd: new Date(data.dateEnd)
},
create: {
id: data.id,
idProject: data.idProject,
title: data.title,
desc: data.desc,
status: data.status,
dateStart: new Date(data.dateStart),
dateEnd: new Date(data.dateEnd),
isActive: true
}
})
}
for (let data of seederDiscussionComprehensive) {
await tx.discussion.upsert({
where: {
id: data.id
},
update: {
title: data.title,
desc: data.desc,
status: data.status,
createdBy: data.createdBy
},
create: {
id: data.id,
idVillage: data.idVillage,
idGroup: data.idGroup!,
title: data.title,
desc: data.desc,
status: data.status,
createdBy: data.createdBy,
isActive: true
}
})
}
for (let data of seederDiscussionMemberComprehensive) {
await tx.discussionMember.upsert({
where: {
id: data.id
},
update: {
idUser: data.idUser
},
create: {
id: data.id,
idDiscussion: data.idDiscussion,
idUser: data.idUser,
isActive: true
}
})
}
});
console.log("\n✅ Complete village data seeding completed successfully!");
console.log(`📊 Total admin roles processed: ${seederAdminRole.length}`);
console.log(`📊 Total admins processed: ${seederAdmin.length}`);
console.log(`📊 Total themes processed: ${seederTheme.length}`);
console.log(`📊 Total villages processed: ${seederDesa.length}`);
console.log(`📊 Total groups processed: ${[...seederGroup, ...seederGroupComprehensive].length}`);
console.log(`📊 Total positions processed: ${[...seederPosition, ...seederPositionComprehensive].length}`);
console.log(`📊 Total user roles processed: ${seederUserRole.length}`);
console.log(`📊 Total users processed: ${[...seederUser, ...seederUserComprehensive].length}`);
console.log(`📊 Total announcements processed: ${seederAnnouncementComprehensive.length}`);
console.log(`📊 Total announcement members processed: ${seederAnnouncementMemberComprehensive.length}`);
console.log(`📊 Total divisions processed: ${seederDivisionComprehensive.length}`);
console.log(`📊 Total division members processed: ${seederDivisionMemberComprehensive.length}`);
console.log(`📊 Total projects processed: ${seederProjectComprehensive.length}`);
console.log(`📊 Total project members processed: ${seederProjectMemberComprehensive.length}`);
console.log(`📊 Total project tasks processed: ${seederProjectTaskComprehensive.length}`);
console.log(`📊 Total discussions processed: ${seederDiscussionComprehensive.length}`);
console.log(`📊 Total discussion members processed: ${seederDiscussionMemberComprehensive.length}`);
} catch (error) {
console.error("\n❌ Error during seeding:", error);
throw new Error("Seeding process failed");
} finally {
await prisma.$disconnect();
}
}
// Execute seeding if called directly from command line
if (require.main === module) {
seedCompleteVillageData()
.catch((e) => {
console.error(e);
process.exit(1);
});
}
export default seedCompleteVillageData;

View File

@@ -8,7 +8,7 @@
"gender": "F"
},
{
"id": "devMalik",
"id": "devLukman",
"idAdminRole": "dev",
"name": "Malik",
"phone": "6289697338821",

View File

@@ -0,0 +1,16 @@
[
{
"id": "ann_pembangunan_jalan",
"idVillage": "desaDummy",
"title": "Pembangunan Jalan Desa Tahap 2",
"desc": "Pada bulan ini akan dilakukan pembangunan jalan desa tahap 2 yang mencakup wilayah RT 01 dan RT 02. Mohon kerjasama warga untuk menyesuaikan aktivitas selama masa pembangunan.",
"createdBy": "user_kades"
},
{
"id": "ann_rapat_warga",
"idVillage": "desaDummy",
"title": "Rapat Bulanan",
"desc": "Rapat bulanan akan diselenggarakan pada hari Sabtu, 25 Februari 2026 pukul 09.00 WIB di Balai Desa. Hadirilah tepat waktu.",
"createdBy": "user_kades"
}
]

View File

@@ -0,0 +1,20 @@
[
{
"id": "ann_mem_pembangunan_jalan_rt01",
"idAnnouncement": "ann_pembangunan_jalan",
"idGroup": "group_rt01",
"idDivision": "div_pelayanan"
},
{
"id": "ann_mem_pembangunan_jalan_rt02",
"idAnnouncement": "ann_pembangunan_jalan",
"idGroup": "group_rt01",
"idDivision": "div_umum"
},
{
"id": "ann_mem_rapat_warga_all",
"idAnnouncement": "ann_rapat_warga",
"idGroup": "group_rt01",
"idDivision": "div_umum"
}
]

View File

@@ -3,5 +3,10 @@
"id": "desa1",
"name": "Darmasaba",
"desc": "-"
},
{
"id": "desaDummy",
"name": "Mandala",
"desc": "Desa Dummy untuk testing"
}
]

View File

@@ -0,0 +1,20 @@
[
{
"id": "disc_kegiatan_desa",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"title": "Pembahasan Kegiatan Desa Mendatang",
"desc": "Diskusi untuk merencanakan kegiatan desa yang akan datang dan menyerap aspirasi warga",
"status": 1,
"createdBy": "user_kades"
},
{
"id": "disc_pengelolaan_sampah",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"title": "Pengelolaan Sampah di RT 01",
"desc": "Diskusi internal RT 01 mengenai pengelolaan sampah rumah tangga dan lingkungan",
"status": 1,
"createdBy": "user_kades"
}
]

View File

@@ -0,0 +1,27 @@
[
{
"id": "disc_mem_kegiatan_desa_kades",
"idDiscussion": "disc_kegiatan_desa",
"idUser": "user_sekdes"
},
{
"id": "disc_mem_kegiatan_desa_sekdes",
"idDiscussion": "disc_kegiatan_desa",
"idUser": "user_warga1"
},
{
"id": "disc_mem_kegiatan_desa_warga1",
"idDiscussion": "disc_kegiatan_desa",
"idUser": "user_warga2"
},
{
"id": "disc_mem_pengelolaan_sampah_kades",
"idDiscussion": "disc_pengelolaan_sampah",
"idUser": "user_warga1"
},
{
"id": "disc_mem_pengelolaan_sampah_warga1",
"idDiscussion": "disc_pengelolaan_sampah",
"idUser": "user_warga2"
}
]

View File

@@ -0,0 +1,18 @@
[
{
"id": "div_pelayanan",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"name": "Seksi Pelayanan",
"desc": "Bertanggung jawab atas pelayanan administrasi umum, perizinan, dan kebutuhan dokumen masyarakat desa.",
"createdBy": "user_kades"
},
{
"id": "div_umum",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"name": "Urusan Tata Usaha dan Umum",
"desc": "Menangani administrasi perkantoran, arsip desa, inventaris aset, dan operasional kantor desa.",
"createdBy": "user_kades"
}
]

View File

@@ -0,0 +1,30 @@
[
{
"id": "div_mem_linmas_warga1",
"idDivision": "div_pelayanan",
"idUser": "user_warga1",
"isAdmin": false,
"isLeader": false
},
{
"id": "div_mem_linmas_warga2",
"idDivision": "div_pelayanan",
"idUser": "user_warga2",
"isAdmin": false,
"isLeader": false
},
{
"id": "div_mem_umum_warga3",
"idDivision": "div_umum",
"idUser": "user_warga3",
"isAdmin": false,
"isLeader": false
},
{
"id": "div_mem_umum_warga4",
"idDivision": "div_umum",
"idUser": "user_warga4",
"isAdmin": false,
"isLeader": false
}
]

View File

@@ -3,5 +3,25 @@
"id": "group1",
"idVillage": "desa1",
"name": "Dinas"
},
{
"id": "group_rt01",
"idVillage": "desaDummy",
"name": "Dinas"
},
{
"id": "group_rt02",
"idVillage": "desaDummy",
"name": "Adat"
},
{
"id": "group_karang_taruna",
"idVillage": "desaDummy",
"name": "Karang Taruna"
},
{
"id": "group_bumdes",
"idVillage": "desaDummy",
"name": "PKK"
}
]

View File

@@ -3,5 +3,25 @@
"id": "position1",
"idGroup": "group1",
"name": "Perbekel"
},
{
"id": "pos_ketua_rt01",
"idGroup": "group_rt01",
"name": "Perbekel"
},
{
"id": "pos_sekretaris_rt01",
"idGroup": "group_rt01",
"name": "Sekretaris"
},
{
"id": "pos_bendahara_rt01",
"idGroup": "group_rt01",
"name": "Bendahara"
},
{
"id": "pos_staff_rt01",
"idGroup": "group_rt01",
"name": "Staff"
}
]

View File

@@ -0,0 +1,11 @@
[
{
"id": "proj_pembangunan_jalan",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"title": "Pembangunan Jalan Desa Tahap 2",
"desc": "Pembangunan jalan desa tahap 2 yang mencakup wilayah RT 01 dan RT 02",
"status": 1,
"createdBy": "user_kades"
}
]

View File

@@ -0,0 +1,14 @@
[
{
"id": "proj_mem_pembangunan_jalan_kades",
"idProject": "proj_pembangunan_jalan",
"idUser": "user_warga1",
"isLeader": true
},
{
"id": "proj_mem_pembangunan_jalan_sekdes",
"idProject": "proj_pembangunan_jalan",
"idUser": "user_warga2",
"isLeader": false
}
]

View File

@@ -0,0 +1,29 @@
[
{
"id": "task_survey_lokasi",
"idProject": "proj_pembangunan_jalan",
"title": "Survey Lokasi",
"desc": "Melakukan survey lokasi untuk menentukan titik pembangunan jalan",
"status": 1,
"dateStart": "2026-01-15T00:00:00.000Z",
"dateEnd": "2026-01-20T00:00:00.000Z"
},
{
"id": "task_pengadaan_material",
"idProject": "proj_pembangunan_jalan",
"title": "Pengadaan Material",
"desc": "Mengadakan material pembangunan seperti pasir, batu, dan semen",
"status": 0,
"dateStart": "2026-02-01T00:00:00.000Z",
"dateEnd": "2026-02-10T00:00:00.000Z"
},
{
"id": "task_pelaksanaan_pembangunan",
"idProject": "proj_pembangunan_jalan",
"title": "Pelaksanaan Pembangunan",
"desc": "Melaksanakan pembangunan jalan sesuai dengan desain yang telah ditentukan",
"status": 0,
"dateStart": "2026-02-15T00:00:00.000Z",
"dateEnd": "2026-03-15T00:00:00.000Z"
}
]

View File

@@ -0,0 +1,22 @@
[
{
"id": "mobile_latest_version",
"name": "latest version",
"value": "2.0.5"
},
{
"id": "mobile_minimum_version",
"name": "minimum version",
"value": "2.0.5"
},
{
"id": "mobile_maintenance",
"name": "maintenance",
"value": "false"
},
{
"id": "mobile_message_update",
"name": "message update",
"value": "Kami telah meningkatkan performa aplikasi"
}
]

View File

@@ -10,5 +10,77 @@
"phone": "628980185458",
"email": "amalia_dev@bip.com",
"gender": "F"
},
{
"id": "user_kades",
"idUserRole": "supadmin",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"idPosition": "pos_ketua_rt01",
"nik": "3201010101010001",
"name": "Juli Ningrum",
"phone": "6281234567890",
"email": "juliningrum@gmail.com",
"gender": "F"
},
{
"id": "user_sekdes",
"idUserRole": "admin",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"idPosition": "pos_sekretaris_rt01",
"nik": "3201010101010002",
"name": "Salwa Kusmawati",
"phone": "6281234567891",
"email": "salwakusmawati@gmail.com",
"gender": "F"
},
{
"id": "user_warga1",
"idUserRole": "user",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"idPosition": "pos_staff_rt01",
"nik": "3201010101010005",
"name": "Bakidin Wibowo",
"phone": "6281234567894",
"email": "bakidinwibowo@gmail.com",
"gender": "M"
},
{
"id": "user_warga2",
"idUserRole": "user",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"idPosition": "pos_staff_rt01",
"nik": "3201010101010006",
"name": "Jais Kurniawan",
"phone": "6281234567895",
"email": "jaiskurniawan@gmail.com",
"gender": "M"
},
{
"id": "user_warga3",
"idUserRole": "user",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"idPosition": "pos_staff_rt01",
"nik": "3201010101010007",
"name": "Safira Oktaviani S.I.Kom",
"phone": "6281234567896",
"email": "safiraoktaviani@gmail.com",
"gender": "F"
},
{
"id": "user_warga4",
"idUserRole": "user",
"idVillage": "desaDummy",
"idGroup": "group_rt01",
"idPosition": "pos_staff_rt01",
"nik": "3201010101010008",
"name": "Agus Setiawan",
"phone": "6281234567897",
"email": "agussetiawannn@gmail.com",
"gender": "M"
}
]

View File

@@ -1,10 +1,21 @@
import seederAdminRole from "./data/admin_role.json";
import seederAdmin from "./data/admin.json";
import seederUserRole from "./data/user_role.json";
import seederUser from "./data/user.json";
import seederAdminRole from "./data/admin_role.json";
import seederAnnouncement from "./data/announcement.json";
import seederAnnouncementMember from "./data/announcement_member.json";
import seederDesa from "./data/desa.json";
import seederDiscussion from "./data/discussion.json";
import seederDiscussionMember from "./data/discussion_member.json";
import seederDivision from "./data/division.json";
import seederDivisionMember from "./data/division_member.json";
import seederGroup from "./data/group.json";
import seederPosition from "./data/position.json";
import seederProject from "./data/project.json";
import seederProjectMember from "./data/project_member.json";
import seederProjectTask from "./data/project_task.json";
import seederSetting from "./data/setting.json";
import seederTheme from "./data/theme.json";
import seederUser from "./data/user.json";
import seederUserRole from "./data/user_role.json";
export { seederAdmin, seederAdminRole, seederAnnouncement, seederAnnouncementMember, seederDesa, seederDiscussion, seederDiscussionMember, seederDivision, seederDivisionMember, seederGroup, seederPosition, seederProject, seederProjectMember, seederProjectTask, seederSetting, seederTheme, seederUser, seederUserRole };
export { seederAdminRole, seederAdmin, seederDesa, seederGroup, seederPosition, seederUserRole, seederUser, seederTheme }

30
src/pages/_error.tsx Normal file
View File

@@ -0,0 +1,30 @@
import { NextPageContext } from "next";
function ErrorPage({ statusCode }: { statusCode?: number }) {
return (
<div
style={{
display: "flex",
alignItems: "center",
justifyContent: "center",
minHeight: "100vh",
backgroundColor: "#252A2F",
color: "white",
fontFamily: "Lato, sans-serif",
}}
>
<h1 style={{ fontSize: 24, fontWeight: 700 }}>
{statusCode === 404
? "404 - Halaman Tidak Ditemukan"
: "Terjadi Kesalahan"}
</h1>
</div>
);
}
ErrorPage.getInitialProps = ({ res, err }: NextPageContext) => {
const statusCode = res ? res.statusCode : err ? err.statusCode : 404;
return { statusCode };
};
export default ErrorPage;