mirror of
https://gitlab.com/mbugroup/lti-api.git
synced 2026-05-20 13:31:56 +00:00
Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| b4da37731c |
@@ -3,7 +3,7 @@ root = "."
|
|||||||
tmp_dir = "tmp"
|
tmp_dir = "tmp"
|
||||||
|
|
||||||
[build]
|
[build]
|
||||||
cmd = "go build -buildvcs=false -o ./tmp/main ./cmd/api"
|
cmd = "go build -o ./tmp/main ./cmd/api"
|
||||||
bin = "tmp/main"
|
bin = "tmp/main"
|
||||||
full_bin = "APP_ENV=dev ./tmp/main"
|
full_bin = "APP_ENV=dev ./tmp/main"
|
||||||
include_ext = ["go", "tpl", "tmpl", "html"]
|
include_ext = ["go", "tpl", "tmpl", "html"]
|
||||||
|
|||||||
+1
-4
@@ -9,13 +9,11 @@ main
|
|||||||
bin/
|
bin/
|
||||||
*.exe
|
*.exe
|
||||||
*.out
|
*.out
|
||||||
.air.toml
|
|
||||||
Makefile
|
Makefile
|
||||||
docker-compose.local.yml
|
docker-compose.local.yml
|
||||||
docker-compose.yaml
|
docker-compose.yaml
|
||||||
Dockerfile
|
|
||||||
Dockerfile.local
|
Dockerfile.local
|
||||||
.gitlab-ci.yml
|
|
||||||
# Go build cache
|
# Go build cache
|
||||||
.gocache/
|
.gocache/
|
||||||
vendor
|
vendor
|
||||||
@@ -29,4 +27,3 @@ coverage/
|
|||||||
.vscode/
|
.vscode/
|
||||||
.idea/
|
.idea/
|
||||||
*.swp
|
*.swp
|
||||||
.DS_Store
|
|
||||||
|
|||||||
+11
-29
@@ -1,38 +1,20 @@
|
|||||||
# =========================
|
FROM golang:1.23-alpine
|
||||||
# Builder stage
|
|
||||||
# =========================
|
|
||||||
FROM golang:1.23-alpine AS builder
|
|
||||||
|
|
||||||
RUN apk add --no-cache git ca-certificates tzdata
|
# Install dependensi dasar
|
||||||
WORKDIR /app
|
RUN apk add --no-cache git curl bash build-base
|
||||||
|
|
||||||
|
# Install Air (pakai repo baru air-verse)
|
||||||
|
RUN go install github.com/air-verse/air@v1.52.3
|
||||||
|
|
||||||
|
WORKDIR /lti-api
|
||||||
|
|
||||||
|
# Cache dependencies
|
||||||
COPY go.mod go.sum ./
|
COPY go.mod go.sum ./
|
||||||
RUN go mod download
|
RUN go mod download
|
||||||
|
|
||||||
|
# Copy source code
|
||||||
COPY . .
|
COPY . .
|
||||||
|
|
||||||
# Build API binary
|
|
||||||
RUN CGO_ENABLED=0 GOOS=linux GOARCH=amd64 \
|
|
||||||
go build -trimpath -ldflags="-s -w" -o lti-api ./cmd/api
|
|
||||||
|
|
||||||
# Build SEED binary (pastikan cmd/seed ada)
|
|
||||||
RUN CGO_ENABLED=0 GOOS=linux GOARCH=amd64 \
|
|
||||||
go build -trimpath -ldflags="-s -w" -o lti-seed ./cmd/seed
|
|
||||||
|
|
||||||
# =========================
|
|
||||||
# Runtime stage
|
|
||||||
# =========================
|
|
||||||
FROM alpine:3.20
|
|
||||||
|
|
||||||
RUN apk add --no-cache ca-certificates tzdata curl bash postgresql-client \
|
|
||||||
&& adduser -D -H -u 10001 appuser
|
|
||||||
|
|
||||||
WORKDIR /app
|
|
||||||
|
|
||||||
COPY --from=builder /app/lti-api /app/lti-api
|
|
||||||
COPY --from=builder /app/lti-seed /app/lti-seed
|
|
||||||
|
|
||||||
USER appuser
|
|
||||||
EXPOSE 8081
|
EXPOSE 8081
|
||||||
|
|
||||||
CMD ["/app/lti-api"]
|
CMD ["air", "-c", ".air.toml"]
|
||||||
|
|||||||
@@ -108,6 +108,6 @@ internal/
|
|||||||
|
|
||||||
IT Development PT Mitra Berlian Unggas Group
|
IT Development PT Mitra Berlian Unggas Group
|
||||||
|
|
||||||
## 📃 Licensee
|
## 📃 License
|
||||||
|
|
||||||
> This project is private. All rights reserved.
|
This project is private. All rights reserved.
|
||||||
|
|||||||
@@ -0,0 +1,77 @@
|
|||||||
|
services:
|
||||||
|
postgresdb:
|
||||||
|
image: postgres:alpine
|
||||||
|
restart: always
|
||||||
|
ports:
|
||||||
|
- "${DB_PORT_HOST:-5542}:5432"
|
||||||
|
environment:
|
||||||
|
POSTGRES_USER: ${DB_USER:-postgres}
|
||||||
|
POSTGRES_PASSWORD: ${DB_PASSWORD:-postgres}
|
||||||
|
POSTGRES_DB: ${DB_NAME:-db_lti_erp}
|
||||||
|
volumes:
|
||||||
|
- dbdata:/var/lib/postgresql/data
|
||||||
|
- ./internal/database/init:/docker-entrypoint-initdb.d
|
||||||
|
networks: [go-network]
|
||||||
|
healthcheck:
|
||||||
|
test:
|
||||||
|
[
|
||||||
|
"CMD-SHELL",
|
||||||
|
"pg_isready -U ${DB_USER:-postgres} -d ${DB_NAME:-db_lti_erp}",
|
||||||
|
]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 5s
|
||||||
|
retries: 5
|
||||||
|
redis:
|
||||||
|
image: redis:7-alpine
|
||||||
|
restart: unless-stopped
|
||||||
|
ports:
|
||||||
|
- "${REDIS_PORT_HOST:-6381}:6379"
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD-SHELL", "redis-cli ping | grep PONG"]
|
||||||
|
interval: 5s
|
||||||
|
timeout: 3s
|
||||||
|
retries: 10
|
||||||
|
networks: [go-network]
|
||||||
|
|
||||||
|
app:
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
dockerfile: Dockerfile.local
|
||||||
|
image: cosmtrek/air:v1.52.3
|
||||||
|
working_dir: /lti-api
|
||||||
|
volumes:
|
||||||
|
- .:/lti-api
|
||||||
|
- ./internal/config/jwtRS256.key:/run/keys/jwtRS256.key
|
||||||
|
- ./internal/config/jwtRS256.key.pub:/run/keys/jwtRS256.key.pub
|
||||||
|
command: air -c .air.toml
|
||||||
|
env_file:
|
||||||
|
- .env
|
||||||
|
environment:
|
||||||
|
DB_HOST: postgresdb
|
||||||
|
DB_PORT: 5432
|
||||||
|
DB_USER: ${DB_USER:-postgres}
|
||||||
|
DB_PASSWORD: ${DB_PASSWORD:-postgres}
|
||||||
|
DB_NAME: ${DB_NAME:-db_lti_erp}
|
||||||
|
REDIS_URL: ${REDIS_URL:-redis://redis:6379/0}
|
||||||
|
ports:
|
||||||
|
- "${APP_PORT:-8081}:8081"
|
||||||
|
depends_on:
|
||||||
|
postgresdb:
|
||||||
|
condition: service_healthy
|
||||||
|
networks: [go-network]
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD-SHELL", "wget -qO- http://localhost:8081/healthz || exit 1"]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 3s
|
||||||
|
retries: 10
|
||||||
|
start_period: 10s
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
dbdata:
|
||||||
|
go-mod-cache:
|
||||||
|
go-build-cache:
|
||||||
|
|
||||||
|
networks:
|
||||||
|
go-network:
|
||||||
|
name: lti-api_go-network
|
||||||
|
driver: bridge
|
||||||
@@ -0,0 +1,98 @@
|
|||||||
|
services:
|
||||||
|
dev-api-lti:
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
dockerfile: Dockerfile
|
||||||
|
container_name: dev-api-lti
|
||||||
|
working_dir: /lti-api
|
||||||
|
command: ["/bin/sh", "scripts/entrypoint.sh"]
|
||||||
|
ports:
|
||||||
|
- "8081:8081"
|
||||||
|
env_file:
|
||||||
|
- .env
|
||||||
|
environment:
|
||||||
|
# override agar koneksi ke container internal
|
||||||
|
DB_HOST: dev-postgres-lti
|
||||||
|
DB_PORT: 5432
|
||||||
|
REDIS_URL: redis://dev-redis-lti:6379/0
|
||||||
|
volumes:
|
||||||
|
- .:/lti-api
|
||||||
|
- ./.air.toml:/lti-api/.air.toml:ro
|
||||||
|
- ./internal/config/jwtRS256.key:/run/keys/jwtRS256.key
|
||||||
|
- ./internal/config/jwtRS256.key.pub:/run/keys/jwtRS256.key.pub
|
||||||
|
depends_on:
|
||||||
|
- dev-postgres-lti
|
||||||
|
- dev-redis-lti
|
||||||
|
networks:
|
||||||
|
- lti-network
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD-SHELL", "wget -qO- http://localhost:8081/healthz || exit 1"]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 3s
|
||||||
|
retries: 10
|
||||||
|
start_period: 10s
|
||||||
|
deploy:
|
||||||
|
resources:
|
||||||
|
limits:
|
||||||
|
cpus: "2.0"
|
||||||
|
memory: 2G
|
||||||
|
reservations:
|
||||||
|
cpus: "1.0"
|
||||||
|
memory: 512M
|
||||||
|
|
||||||
|
dev-postgres-lti:
|
||||||
|
image: postgres:15-alpine
|
||||||
|
container_name: dev-postgres-lti
|
||||||
|
restart: always
|
||||||
|
env_file:
|
||||||
|
- credential/.env.db
|
||||||
|
ports:
|
||||||
|
- "5433:5432"
|
||||||
|
volumes:
|
||||||
|
- dev-postgres-lti-data:/var/lib/postgresql/data
|
||||||
|
- ./credential:/docker-entrypoint-initdb.d:ro
|
||||||
|
networks:
|
||||||
|
- lti-network
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD-SHELL", "pg_isready -U ${DB_USER:-postgres} -d ${DB_NAME:-db_lti_erp}"]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 5s
|
||||||
|
retries: 5
|
||||||
|
start_period: 5s
|
||||||
|
deploy:
|
||||||
|
resources:
|
||||||
|
limits:
|
||||||
|
cpus: "1.0"
|
||||||
|
memory: 2G
|
||||||
|
reservations:
|
||||||
|
cpus: "0.5"
|
||||||
|
memory: 512M
|
||||||
|
|
||||||
|
dev-redis-lti:
|
||||||
|
image: redis:7-alpine
|
||||||
|
container_name: dev-redis-lti
|
||||||
|
restart: always
|
||||||
|
ports:
|
||||||
|
- "6380:6379"
|
||||||
|
networks:
|
||||||
|
- lti-network
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD", "redis-cli", "ping"]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 3s
|
||||||
|
retries: 10
|
||||||
|
deploy:
|
||||||
|
resources:
|
||||||
|
limits:
|
||||||
|
cpus: "0.5"
|
||||||
|
memory: 512M
|
||||||
|
reservations:
|
||||||
|
cpus: "0.2"
|
||||||
|
memory: 256M
|
||||||
|
|
||||||
|
networks:
|
||||||
|
lti-network:
|
||||||
|
driver: bridge
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
dev-postgres-lti-data:
|
||||||
@@ -9,18 +9,15 @@ require (
|
|||||||
github.com/aws/aws-sdk-go-v2/credentials v1.19.2
|
github.com/aws/aws-sdk-go-v2/credentials v1.19.2
|
||||||
github.com/aws/aws-sdk-go-v2/service/s3 v1.92.1
|
github.com/aws/aws-sdk-go-v2/service/s3 v1.92.1
|
||||||
github.com/bytedance/sonic v1.12.1
|
github.com/bytedance/sonic v1.12.1
|
||||||
github.com/glebarez/sqlite v1.11.0
|
|
||||||
github.com/go-playground/validator/v10 v10.27.0
|
github.com/go-playground/validator/v10 v10.27.0
|
||||||
github.com/gofiber/contrib/jwt v1.0.10
|
github.com/gofiber/contrib/jwt v1.0.10
|
||||||
github.com/gofiber/fiber/v2 v2.52.5
|
github.com/gofiber/fiber/v2 v2.52.5
|
||||||
github.com/golang-jwt/jwt/v5 v5.2.1
|
github.com/golang-jwt/jwt/v5 v5.2.1
|
||||||
github.com/google/uuid v1.6.0
|
github.com/google/uuid v1.6.0
|
||||||
github.com/jackc/pgconn v1.14.1
|
github.com/jackc/pgconn v1.14.1
|
||||||
github.com/jackc/pgx/v5 v5.5.5
|
|
||||||
github.com/redis/go-redis/v9 v9.14.0
|
github.com/redis/go-redis/v9 v9.14.0
|
||||||
github.com/sirupsen/logrus v1.9.3
|
github.com/sirupsen/logrus v1.9.3
|
||||||
github.com/spf13/viper v1.19.0
|
github.com/spf13/viper v1.19.0
|
||||||
github.com/xuri/excelize/v2 v2.9.0
|
|
||||||
golang.org/x/crypto v0.33.0
|
golang.org/x/crypto v0.33.0
|
||||||
gorm.io/driver/postgres v1.5.9
|
gorm.io/driver/postgres v1.5.9
|
||||||
gorm.io/gorm v1.25.11
|
gorm.io/gorm v1.25.11
|
||||||
@@ -48,10 +45,8 @@ require (
|
|||||||
github.com/cloudwego/base64x v0.1.4 // indirect
|
github.com/cloudwego/base64x v0.1.4 // indirect
|
||||||
github.com/cloudwego/iasm v0.2.0 // indirect
|
github.com/cloudwego/iasm v0.2.0 // indirect
|
||||||
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect
|
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect
|
||||||
github.com/dustin/go-humanize v1.0.1 // indirect
|
|
||||||
github.com/fsnotify/fsnotify v1.7.0 // indirect
|
github.com/fsnotify/fsnotify v1.7.0 // indirect
|
||||||
github.com/gabriel-vasile/mimetype v1.4.8 // indirect
|
github.com/gabriel-vasile/mimetype v1.4.8 // indirect
|
||||||
github.com/glebarez/go-sqlite v1.21.2 // indirect
|
|
||||||
github.com/go-playground/locales v0.14.1 // indirect
|
github.com/go-playground/locales v0.14.1 // indirect
|
||||||
github.com/go-playground/universal-translator v0.18.1 // indirect
|
github.com/go-playground/universal-translator v0.18.1 // indirect
|
||||||
github.com/google/go-cmp v0.6.0 // indirect
|
github.com/google/go-cmp v0.6.0 // indirect
|
||||||
@@ -61,6 +56,7 @@ require (
|
|||||||
github.com/jackc/pgpassfile v1.0.0 // indirect
|
github.com/jackc/pgpassfile v1.0.0 // indirect
|
||||||
github.com/jackc/pgproto3/v2 v2.3.2 // indirect
|
github.com/jackc/pgproto3/v2 v2.3.2 // indirect
|
||||||
github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect
|
github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect
|
||||||
|
github.com/jackc/pgx/v5 v5.5.5 // indirect
|
||||||
github.com/jackc/puddle/v2 v2.2.1 // indirect
|
github.com/jackc/puddle/v2 v2.2.1 // indirect
|
||||||
github.com/jinzhu/inflection v1.0.0 // indirect
|
github.com/jinzhu/inflection v1.0.0 // indirect
|
||||||
github.com/jinzhu/now v1.1.5 // indirect
|
github.com/jinzhu/now v1.1.5 // indirect
|
||||||
@@ -72,12 +68,8 @@ require (
|
|||||||
github.com/mattn/go-isatty v0.0.20 // indirect
|
github.com/mattn/go-isatty v0.0.20 // indirect
|
||||||
github.com/mattn/go-runewidth v0.0.16 // indirect
|
github.com/mattn/go-runewidth v0.0.16 // indirect
|
||||||
github.com/mitchellh/mapstructure v1.5.0 // indirect
|
github.com/mitchellh/mapstructure v1.5.0 // indirect
|
||||||
github.com/mohae/deepcopy v0.0.0-20170929034955-c48cc78d4826 // indirect
|
|
||||||
github.com/pelletier/go-toml/v2 v2.2.2 // indirect
|
github.com/pelletier/go-toml/v2 v2.2.2 // indirect
|
||||||
github.com/philhofer/fwd v1.1.2 // indirect
|
github.com/philhofer/fwd v1.1.2 // indirect
|
||||||
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec // indirect
|
|
||||||
github.com/richardlehane/mscfb v1.0.4 // indirect
|
|
||||||
github.com/richardlehane/msoleps v1.0.4 // indirect
|
|
||||||
github.com/rivo/uniseg v0.4.7 // indirect
|
github.com/rivo/uniseg v0.4.7 // indirect
|
||||||
github.com/rogpeppe/go-internal v1.11.0 // indirect
|
github.com/rogpeppe/go-internal v1.11.0 // indirect
|
||||||
github.com/sagikazarmark/locafero v0.4.0 // indirect
|
github.com/sagikazarmark/locafero v0.4.0 // indirect
|
||||||
@@ -86,15 +78,12 @@ require (
|
|||||||
github.com/spf13/afero v1.11.0 // indirect
|
github.com/spf13/afero v1.11.0 // indirect
|
||||||
github.com/spf13/cast v1.6.0 // indirect
|
github.com/spf13/cast v1.6.0 // indirect
|
||||||
github.com/spf13/pflag v1.0.5 // indirect
|
github.com/spf13/pflag v1.0.5 // indirect
|
||||||
github.com/stretchr/testify v1.11.1 // indirect
|
|
||||||
github.com/subosito/gotenv v1.6.0 // indirect
|
github.com/subosito/gotenv v1.6.0 // indirect
|
||||||
github.com/tinylib/msgp v1.1.8 // indirect
|
github.com/tinylib/msgp v1.1.8 // indirect
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
||||||
github.com/valyala/bytebufferpool v1.0.0 // indirect
|
github.com/valyala/bytebufferpool v1.0.0 // indirect
|
||||||
github.com/valyala/fasthttp v1.55.0 // indirect
|
github.com/valyala/fasthttp v1.55.0 // indirect
|
||||||
github.com/valyala/tcplisten v1.0.0 // indirect
|
github.com/valyala/tcplisten v1.0.0 // indirect
|
||||||
github.com/xuri/efp v0.0.0-20240408161823-9ad904a10d6d // indirect
|
|
||||||
github.com/xuri/nfp v0.0.0-20240318013403-ab9948c2c4a7 // indirect
|
|
||||||
go.uber.org/atomic v1.9.0 // indirect
|
go.uber.org/atomic v1.9.0 // indirect
|
||||||
go.uber.org/multierr v1.9.0 // indirect
|
go.uber.org/multierr v1.9.0 // indirect
|
||||||
golang.org/x/arch v0.0.0-20210923205945-b76863e36670 // indirect
|
golang.org/x/arch v0.0.0-20210923205945-b76863e36670 // indirect
|
||||||
@@ -105,8 +94,4 @@ require (
|
|||||||
golang.org/x/text v0.22.0 // indirect
|
golang.org/x/text v0.22.0 // indirect
|
||||||
gopkg.in/ini.v1 v1.67.0 // indirect
|
gopkg.in/ini.v1 v1.67.0 // indirect
|
||||||
gopkg.in/yaml.v3 v3.0.1 // indirect
|
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||||
modernc.org/libc v1.22.5 // indirect
|
|
||||||
modernc.org/mathutil v1.5.0 // indirect
|
|
||||||
modernc.org/memory v1.5.0 // indirect
|
|
||||||
modernc.org/sqlite v1.23.1 // indirect
|
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -65,18 +65,12 @@ github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1
|
|||||||
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78=
|
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78=
|
||||||
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc=
|
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc=
|
||||||
github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY=
|
|
||||||
github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto=
|
|
||||||
github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8=
|
github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8=
|
||||||
github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0=
|
github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0=
|
||||||
github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA=
|
github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA=
|
||||||
github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM=
|
github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM=
|
||||||
github.com/gabriel-vasile/mimetype v1.4.8 h1:FfZ3gj38NjllZIeJAmMhr+qKL8Wu+nOoI3GqacKw1NM=
|
github.com/gabriel-vasile/mimetype v1.4.8 h1:FfZ3gj38NjllZIeJAmMhr+qKL8Wu+nOoI3GqacKw1NM=
|
||||||
github.com/gabriel-vasile/mimetype v1.4.8/go.mod h1:ByKUIKGjh1ODkGM1asKUbQZOLGrPjydw3hYPU2YU9t8=
|
github.com/gabriel-vasile/mimetype v1.4.8/go.mod h1:ByKUIKGjh1ODkGM1asKUbQZOLGrPjydw3hYPU2YU9t8=
|
||||||
github.com/glebarez/go-sqlite v1.21.2 h1:3a6LFC4sKahUunAmynQKLZceZCOzUthkRkEAl9gAXWo=
|
|
||||||
github.com/glebarez/go-sqlite v1.21.2/go.mod h1:sfxdZyhQjTM2Wry3gVYWaW072Ri1WMdWJi0k6+3382k=
|
|
||||||
github.com/glebarez/sqlite v1.11.0 h1:wSG0irqzP6VurnMEpFGer5Li19RpIRi2qvQz++w0GMw=
|
|
||||||
github.com/glebarez/sqlite v1.11.0/go.mod h1:h8/o8j5wiAsqSPoWELDUdJXhjAhsVliSn7bWZjOhrgQ=
|
|
||||||
github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s=
|
github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s=
|
||||||
github.com/go-playground/assert/v2 v2.2.0/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
|
github.com/go-playground/assert/v2 v2.2.0/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
|
||||||
github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA=
|
github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA=
|
||||||
@@ -94,8 +88,6 @@ github.com/golang-jwt/jwt/v5 v5.2.1 h1:OuVbFODueb089Lh128TAcimifWaLhJwVflnrgM17w
|
|||||||
github.com/golang-jwt/jwt/v5 v5.2.1/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk=
|
github.com/golang-jwt/jwt/v5 v5.2.1/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk=
|
||||||
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
|
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
|
||||||
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
||||||
github.com/google/pprof v0.0.0-20221118152302-e6195bd50e26 h1:Xim43kblpZXfIBQsbuBVKCudVG457BR2GZFIz3uw3hQ=
|
|
||||||
github.com/google/pprof v0.0.0-20221118152302-e6195bd50e26/go.mod h1:dDKJzRmX4S37WGHujM7tX//fmj1uioxKzKxz3lo4HJo=
|
|
||||||
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
|
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
|
||||||
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||||
github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4=
|
github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4=
|
||||||
@@ -182,8 +174,6 @@ github.com/mattn/go-runewidth v0.0.16 h1:E5ScNMtiwvlvB5paMFdw9p4kSQzbXFikJ5SQO6T
|
|||||||
github.com/mattn/go-runewidth v0.0.16/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w=
|
github.com/mattn/go-runewidth v0.0.16/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w=
|
||||||
github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyuac5Z2hdY=
|
github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyuac5Z2hdY=
|
||||||
github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo=
|
github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo=
|
||||||
github.com/mohae/deepcopy v0.0.0-20170929034955-c48cc78d4826 h1:RWengNIwukTxcDr9M+97sNutRR1RKhG96O6jWumTTnw=
|
|
||||||
github.com/mohae/deepcopy v0.0.0-20170929034955-c48cc78d4826/go.mod h1:TaXosZuwdSHYgviHp1DAtfrULt5eUgsSMsZf+YrPgl8=
|
|
||||||
github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM=
|
github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM=
|
||||||
github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs=
|
github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs=
|
||||||
github.com/philhofer/fwd v1.1.2 h1:bnDivRJ1EWPjUIRXV5KfORO897HTbpFAQddBdE8t7Gw=
|
github.com/philhofer/fwd v1.1.2 h1:bnDivRJ1EWPjUIRXV5KfORO897HTbpFAQddBdE8t7Gw=
|
||||||
@@ -194,14 +184,6 @@ github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRI
|
|||||||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
github.com/redis/go-redis/v9 v9.14.0 h1:u4tNCjXOyzfgeLN+vAZaW1xUooqWDqVEsZN0U01jfAE=
|
github.com/redis/go-redis/v9 v9.14.0 h1:u4tNCjXOyzfgeLN+vAZaW1xUooqWDqVEsZN0U01jfAE=
|
||||||
github.com/redis/go-redis/v9 v9.14.0/go.mod h1:huWgSWd8mW6+m0VPhJjSSQ+d6Nh1VICQ6Q5lHuCH/Iw=
|
github.com/redis/go-redis/v9 v9.14.0/go.mod h1:huWgSWd8mW6+m0VPhJjSSQ+d6Nh1VICQ6Q5lHuCH/Iw=
|
||||||
github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo=
|
|
||||||
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE=
|
|
||||||
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo=
|
|
||||||
github.com/richardlehane/mscfb v1.0.4 h1:WULscsljNPConisD5hR0+OyZjwK46Pfyr6mPu5ZawpM=
|
|
||||||
github.com/richardlehane/mscfb v1.0.4/go.mod h1:YzVpcZg9czvAuhk9T+a3avCpcFPMUWm7gK3DypaEsUk=
|
|
||||||
github.com/richardlehane/msoleps v1.0.1/go.mod h1:BWev5JBpU9Ko2WAgmZEuiz4/u3ZYTKbjLycmwiWUfWg=
|
|
||||||
github.com/richardlehane/msoleps v1.0.4 h1:WuESlvhX3gH2IHcd8UqyCuFY5yiq/GR/yqaSM/9/g00=
|
|
||||||
github.com/richardlehane/msoleps v1.0.4/go.mod h1:BWev5JBpU9Ko2WAgmZEuiz4/u3ZYTKbjLycmwiWUfWg=
|
|
||||||
github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc=
|
github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc=
|
||||||
github.com/rivo/uniseg v0.4.7 h1:WUdvkW8uEhrYfLC4ZzdpI2ztxP1I582+49Oc5Mq64VQ=
|
github.com/rivo/uniseg v0.4.7 h1:WUdvkW8uEhrYfLC4ZzdpI2ztxP1I582+49Oc5Mq64VQ=
|
||||||
github.com/rivo/uniseg v0.4.7/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88=
|
github.com/rivo/uniseg v0.4.7/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88=
|
||||||
@@ -245,9 +227,8 @@ github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/
|
|||||||
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
||||||
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
||||||
github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
|
github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
|
||||||
|
github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg=
|
||||||
github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
||||||
github.com/stretchr/testify v1.11.1 h1:7s2iGBzp5EwR7/aIZr8ao5+dra3wiQyKjjFuvgVKu7U=
|
|
||||||
github.com/stretchr/testify v1.11.1/go.mod h1:wZwfW3scLgRK+23gO65QZefKpKQRnfz6sD981Nm4B6U=
|
|
||||||
github.com/subosito/gotenv v1.6.0 h1:9NlTDc1FTs4qu0DDq7AEtTPNw6SVm7uBMsUCUjABIf8=
|
github.com/subosito/gotenv v1.6.0 h1:9NlTDc1FTs4qu0DDq7AEtTPNw6SVm7uBMsUCUjABIf8=
|
||||||
github.com/subosito/gotenv v1.6.0/go.mod h1:Dk4QP5c2W3ibzajGcXpNraDfq2IrhjMIvMSWPKKo0FU=
|
github.com/subosito/gotenv v1.6.0/go.mod h1:Dk4QP5c2W3ibzajGcXpNraDfq2IrhjMIvMSWPKKo0FU=
|
||||||
github.com/tinylib/msgp v1.1.8 h1:FCXC1xanKO4I8plpHGH2P7koL/RzZs12l/+r7vakfm0=
|
github.com/tinylib/msgp v1.1.8 h1:FCXC1xanKO4I8plpHGH2P7koL/RzZs12l/+r7vakfm0=
|
||||||
@@ -260,12 +241,6 @@ github.com/valyala/fasthttp v1.55.0 h1:Zkefzgt6a7+bVKHnu/YaYSOPfNYNisSVBo/unVCf8
|
|||||||
github.com/valyala/fasthttp v1.55.0/go.mod h1:NkY9JtkrpPKmgwV3HTaS2HWaJss9RSIsRVfcxxoHiOM=
|
github.com/valyala/fasthttp v1.55.0/go.mod h1:NkY9JtkrpPKmgwV3HTaS2HWaJss9RSIsRVfcxxoHiOM=
|
||||||
github.com/valyala/tcplisten v1.0.0 h1:rBHj/Xf+E1tRGZyWIWwJDiRY0zc1Js+CV5DqwacVSA8=
|
github.com/valyala/tcplisten v1.0.0 h1:rBHj/Xf+E1tRGZyWIWwJDiRY0zc1Js+CV5DqwacVSA8=
|
||||||
github.com/valyala/tcplisten v1.0.0/go.mod h1:T0xQ8SeCZGxckz9qRXTfG43PvQ/mcWh7FwZEA7Ioqkc=
|
github.com/valyala/tcplisten v1.0.0/go.mod h1:T0xQ8SeCZGxckz9qRXTfG43PvQ/mcWh7FwZEA7Ioqkc=
|
||||||
github.com/xuri/efp v0.0.0-20240408161823-9ad904a10d6d h1:llb0neMWDQe87IzJLS4Ci7psK/lVsjIS2otl+1WyRyY=
|
|
||||||
github.com/xuri/efp v0.0.0-20240408161823-9ad904a10d6d/go.mod h1:ybY/Jr0T0GTCnYjKqmdwxyxn2BQf2RcQIIvex5QldPI=
|
|
||||||
github.com/xuri/excelize/v2 v2.9.0 h1:1tgOaEq92IOEumR1/JfYS/eR0KHOCsRv/rYXXh6YJQE=
|
|
||||||
github.com/xuri/excelize/v2 v2.9.0/go.mod h1:uqey4QBZ9gdMeWApPLdhm9x+9o2lq4iVmjiLfBS5hdE=
|
|
||||||
github.com/xuri/nfp v0.0.0-20240318013403-ab9948c2c4a7 h1:hPVCafDV85blFTabnqKgNhDCkJX25eik94Si9cTER4A=
|
|
||||||
github.com/xuri/nfp v0.0.0-20240318013403-ab9948c2c4a7/go.mod h1:WwHg+CVyzlv/TX9xqBFXEZAuxOPxn2k1GNHwG41IIUQ=
|
|
||||||
github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
|
github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
|
||||||
github.com/zenazn/goji v0.9.0/go.mod h1:7S9M489iMyHBNxwZnk9/EHS098H4/F6TATF2mIxtB1Q=
|
github.com/zenazn/goji v0.9.0/go.mod h1:7S9M489iMyHBNxwZnk9/EHS098H4/F6TATF2mIxtB1Q=
|
||||||
go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
|
go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
|
||||||
@@ -292,8 +267,6 @@ golang.org/x/crypto v0.33.0 h1:IOBPskki6Lysi0lo9qQvbxiQ+FvsCC/YWOecCHAixus=
|
|||||||
golang.org/x/crypto v0.33.0/go.mod h1:bVdXmD7IV/4GdElGPozy6U7lWdRXA4qyRVGJV57uQ5M=
|
golang.org/x/crypto v0.33.0/go.mod h1:bVdXmD7IV/4GdElGPozy6U7lWdRXA4qyRVGJV57uQ5M=
|
||||||
golang.org/x/exp v0.0.0-20230905200255-921286631fa9 h1:GoHiUyI/Tp2nVkLI2mCxVkOjsbSXD66ic0XW0js0R9g=
|
golang.org/x/exp v0.0.0-20230905200255-921286631fa9 h1:GoHiUyI/Tp2nVkLI2mCxVkOjsbSXD66ic0XW0js0R9g=
|
||||||
golang.org/x/exp v0.0.0-20230905200255-921286631fa9/go.mod h1:S2oDrQGGwySpoQPVqRShND87VCbxmc6bL1Yd2oYrm6k=
|
golang.org/x/exp v0.0.0-20230905200255-921286631fa9/go.mod h1:S2oDrQGGwySpoQPVqRShND87VCbxmc6bL1Yd2oYrm6k=
|
||||||
golang.org/x/image v0.18.0 h1:jGzIakQa/ZXI1I0Fxvaa9W7yP25TqT6cHIHn+6CqvSQ=
|
|
||||||
golang.org/x/image v0.18.0/go.mod h1:4yyo5vMFQjVjUcVk4jEQcU9MGy/rulF5WvUILseCM2E=
|
|
||||||
golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
|
golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
|
||||||
golang.org/x/mod v0.7.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
|
golang.org/x/mod v0.7.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
|
||||||
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
||||||
@@ -371,12 +344,4 @@ gorm.io/driver/postgres v1.5.9 h1:DkegyItji119OlcaLjqN11kHoUgZ/j13E0jkJZgD6A8=
|
|||||||
gorm.io/driver/postgres v1.5.9/go.mod h1:DX3GReXH+3FPWGrrgffdvCk3DQ1dwDPdmbenSkweRGI=
|
gorm.io/driver/postgres v1.5.9/go.mod h1:DX3GReXH+3FPWGrrgffdvCk3DQ1dwDPdmbenSkweRGI=
|
||||||
gorm.io/gorm v1.25.11 h1:/Wfyg1B/je1hnDx3sMkX+gAlxrlZpn6X0BXRlwXlvHg=
|
gorm.io/gorm v1.25.11 h1:/Wfyg1B/je1hnDx3sMkX+gAlxrlZpn6X0BXRlwXlvHg=
|
||||||
gorm.io/gorm v1.25.11/go.mod h1:xh7N7RHfYlNc5EmcI/El95gXusucDrQnHXe0+CgWcLQ=
|
gorm.io/gorm v1.25.11/go.mod h1:xh7N7RHfYlNc5EmcI/El95gXusucDrQnHXe0+CgWcLQ=
|
||||||
modernc.org/libc v1.22.5 h1:91BNch/e5B0uPbJFgqbxXuOnxBQjlS//icfQEGmvyjE=
|
|
||||||
modernc.org/libc v1.22.5/go.mod h1:jj+Z7dTNX8fBScMVNRAYZ/jF91K8fdT2hYMThc3YjBY=
|
|
||||||
modernc.org/mathutil v1.5.0 h1:rV0Ko/6SfM+8G+yKiyI830l3Wuz1zRutdslNoQ0kfiQ=
|
|
||||||
modernc.org/mathutil v1.5.0/go.mod h1:mZW8CKdRPY1v87qxC/wUdX5O1qDzXMP5TH3wjfpga6E=
|
|
||||||
modernc.org/memory v1.5.0 h1:N+/8c5rE6EqugZwHii4IFsaJ7MUhoWX07J5tC/iI5Ds=
|
|
||||||
modernc.org/memory v1.5.0/go.mod h1:PkUhL0Mugw21sHPeskwZW4D6VscE/GQJOnIpCnW6pSU=
|
|
||||||
modernc.org/sqlite v1.23.1 h1:nrSBg4aRQQwq59JpvGEQ15tNxoO5pX/kUjcRNwSAGQM=
|
|
||||||
modernc.org/sqlite v1.23.1/go.mod h1:OrDj17Mggn6MhE+iPbBNf7RGKODDE9NFT0f3EwDzJqk=
|
|
||||||
nullprogram.com/x/optparse v1.0.0/go.mod h1:KdyPE+Igbe0jQUrVfMqDMeJQIJZEuyV7pjYmp6pbG50=
|
nullprogram.com/x/optparse v1.0.0/go.mod h1:KdyPE+Igbe0jQUrVfMqDMeJQIJZEuyV7pjYmp6pbG50=
|
||||||
|
|||||||
@@ -0,0 +1,44 @@
|
|||||||
|
package capabilities
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
recordings "gitlab.com/mbugroup/lti-api.git/internal/modules/production/recordings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// FromPermissions returns a filtered map of capabilities that the frontend can use
|
||||||
|
// to toggle features. Only permissions recognized by the application are exposed.
|
||||||
|
func FromPermissions(perms []string) map[string]bool {
|
||||||
|
if len(perms) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
out := make(map[string]bool)
|
||||||
|
for _, perm := range perms {
|
||||||
|
if key, ok := normalizeAndAllow(perm); ok {
|
||||||
|
out[key] = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if len(out) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
func normalizeAndAllow(perm string) (string, bool) {
|
||||||
|
perm = strings.ToLower(strings.TrimSpace(perm))
|
||||||
|
if perm == "" {
|
||||||
|
return "", false
|
||||||
|
}
|
||||||
|
if _, ok := allowed[perm]; !ok {
|
||||||
|
return "", false
|
||||||
|
}
|
||||||
|
return perm, true
|
||||||
|
}
|
||||||
|
|
||||||
|
var allowed = map[string]struct{}{
|
||||||
|
recordings.PermissionRecordingRead: {},
|
||||||
|
recordings.PermissionRecordingCreate: {},
|
||||||
|
recordings.PermissionRecordingUpdate: {},
|
||||||
|
recordings.PermissionRecordingDelete: {},
|
||||||
|
}
|
||||||
@@ -84,9 +84,8 @@ func (r *approvalRepositoryImpl) LatestByTargets(
|
|||||||
result := make(map[uint]entity.Approval, len(approvableIDs))
|
result := make(map[uint]entity.Approval, len(approvableIDs))
|
||||||
|
|
||||||
q := r.DB().WithContext(ctx).
|
q := r.DB().WithContext(ctx).
|
||||||
Select("DISTINCT ON (approvable_id) *").
|
|
||||||
Where("approvable_type = ? AND approvable_id IN ?", workflow, approvableIDs).
|
Where("approvable_type = ? AND approvable_id IN ?", workflow, approvableIDs).
|
||||||
Order("approvable_id, action_at DESC")
|
Order("action_at DESC")
|
||||||
|
|
||||||
if modifier != nil {
|
if modifier != nil {
|
||||||
q = modifier(q)
|
q = modifier(q)
|
||||||
|
|||||||
@@ -187,11 +187,10 @@ func (r *BaseRepositoryImpl[T]) PatchOne(
|
|||||||
updates map[string]any,
|
updates map[string]any,
|
||||||
modifier func(*gorm.DB) *gorm.DB,
|
modifier func(*gorm.DB) *gorm.DB,
|
||||||
) error {
|
) error {
|
||||||
q := r.db.WithContext(ctx)
|
q := r.db.WithContext(ctx).Model(new(T)).Where("id = ?", id)
|
||||||
if modifier != nil {
|
if modifier != nil {
|
||||||
q = modifier(q)
|
q = modifier(q)
|
||||||
}
|
}
|
||||||
q = q.Model(new(T)).Where("id = ?", id)
|
|
||||||
|
|
||||||
result := q.Updates(updates)
|
result := q.Updates(updates)
|
||||||
if result.Error != nil {
|
if result.Error != nil {
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ package repository
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
@@ -10,59 +9,45 @@ import (
|
|||||||
|
|
||||||
// Exists reports whether a record with the given ID exists for type T.
|
// Exists reports whether a record with the given ID exists for type T.
|
||||||
func Exists[T any](ctx context.Context, db *gorm.DB, id uint) (bool, error) {
|
func Exists[T any](ctx context.Context, db *gorm.DB, id uint) (bool, error) {
|
||||||
var marker int
|
var count int64
|
||||||
err := db.WithContext(ctx).
|
if err := db.WithContext(ctx).
|
||||||
Model(new(T)).
|
Model(new(T)).
|
||||||
Select("1").
|
|
||||||
Where("id = ?", id).
|
Where("id = ?", id).
|
||||||
Limit(1).
|
Count(&count).Error; err != nil {
|
||||||
Take(&marker).Error
|
|
||||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
return true, nil
|
return count > 0, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func ExistsByName[T any](ctx context.Context, db *gorm.DB, name string, excludeID *uint) (bool, error) {
|
func ExistsByName[T any](ctx context.Context, db *gorm.DB, name string, excludeID *uint) (bool, error) {
|
||||||
|
var count int64
|
||||||
q := db.WithContext(ctx).
|
q := db.WithContext(ctx).
|
||||||
Model(new(T)).
|
Model(new(T)).
|
||||||
Select("1").
|
|
||||||
Where("name = ?", name).
|
Where("name = ?", name).
|
||||||
Where("deleted_at IS NULL")
|
Where("deleted_at IS NULL")
|
||||||
if excludeID != nil {
|
if excludeID != nil {
|
||||||
q = q.Where("id <> ?", *excludeID)
|
q = q.Where("id <> ?", *excludeID)
|
||||||
}
|
}
|
||||||
var marker int
|
if err := q.Count(&count).Error; err != nil {
|
||||||
if err := q.Limit(1).Take(&marker).Error; err != nil {
|
|
||||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
return true, nil
|
return count > 0, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func ExistsByField[T any](ctx context.Context, db *gorm.DB, field string, value any, excludeID *uint) (bool, error) {
|
func ExistsByField[T any](ctx context.Context, db *gorm.DB, field string, value any, excludeID *uint) (bool, error) {
|
||||||
if field == "" {
|
if field == "" {
|
||||||
return false, fmt.Errorf("field is required")
|
return false, fmt.Errorf("field is required")
|
||||||
}
|
}
|
||||||
|
var count int64
|
||||||
q := db.WithContext(ctx).
|
q := db.WithContext(ctx).
|
||||||
Model(new(T)).
|
Model(new(T)).
|
||||||
Select("1").
|
|
||||||
Where(fmt.Sprintf("%s = ?", field), value).
|
Where(fmt.Sprintf("%s = ?", field), value).
|
||||||
Where("deleted_at IS NULL")
|
Where("deleted_at IS NULL")
|
||||||
if excludeID != nil {
|
if excludeID != nil {
|
||||||
q = q.Where("id <> ?", *excludeID)
|
q = q.Where("id <> ?", *excludeID)
|
||||||
}
|
}
|
||||||
var marker int
|
if err := q.Count(&count).Error; err != nil {
|
||||||
if err := q.Limit(1).Take(&marker).Error; err != nil {
|
|
||||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
return true, nil
|
return count > 0, nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -63,14 +63,13 @@ func (r *StockAllocationRepositoryImpl) ReleaseByUsable(
|
|||||||
updates["note"] = *note
|
updates["note"] = *note
|
||||||
}
|
}
|
||||||
|
|
||||||
baseDB := r.DB()
|
q := r.DB().WithContext(ctx).
|
||||||
if modifier != nil {
|
|
||||||
baseDB = modifier(baseDB)
|
|
||||||
}
|
|
||||||
|
|
||||||
q := baseDB.WithContext(ctx).
|
|
||||||
Model(&entity.StockAllocation{}).
|
Model(&entity.StockAllocation{}).
|
||||||
Where("usable_type = ? AND usable_id = ? AND status = ?", usableType, usableID, entity.StockAllocationStatusActive)
|
Where("usable_type = ? AND usable_id = ? AND status = ?", usableType, usableID, entity.StockAllocationStatusActive)
|
||||||
|
|
||||||
|
if modifier != nil {
|
||||||
|
q = modifier(q)
|
||||||
|
}
|
||||||
|
|
||||||
return q.Updates(updates).Error
|
return q.Updates(updates).Error
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,120 +0,0 @@
|
|||||||
package service
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
|
|
||||||
productWarehouseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/product-warehouses/repositories"
|
|
||||||
warehouseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/master/warehouses/repositories"
|
|
||||||
projectFlockKandangRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/production/project_flocks/repositories"
|
|
||||||
|
|
||||||
"github.com/gofiber/fiber/v2"
|
|
||||||
"gorm.io/gorm"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Dipakai untuk semua module yang butuh cek:
|
|
||||||
// "PW ini → warehouse → kandang → project_flock_kandang sudah closing atau belum"
|
|
||||||
func EnsureProjectFlockNotClosedForProductWarehouses(
|
|
||||||
ctx context.Context,
|
|
||||||
db *gorm.DB,
|
|
||||||
productWarehouseIDs []uint,
|
|
||||||
) error {
|
|
||||||
if len(productWarehouseIDs) == 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
pwRepo := productWarehouseRepo.NewProductWarehouseRepository(db)
|
|
||||||
wRepo := warehouseRepo.NewWarehouseRepository(db)
|
|
||||||
pfkRepo := projectFlockKandangRepo.NewProjectFlockKandangRepository(db)
|
|
||||||
|
|
||||||
seenPW := make(map[uint]struct{})
|
|
||||||
seenKandang := make(map[uint]struct{})
|
|
||||||
|
|
||||||
for _, pwID := range productWarehouseIDs {
|
|
||||||
if pwID == 0 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if _, ok := seenPW[pwID]; ok {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
seenPW[pwID] = struct{}{}
|
|
||||||
|
|
||||||
pw, err := pwRepo.GetByID(ctx, pwID, nil)
|
|
||||||
if err != nil {
|
|
||||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
||||||
return fiber.NewError(fiber.StatusBadRequest,
|
|
||||||
fmt.Sprintf("Product warehouse %d tidak ditemukan", pwID))
|
|
||||||
}
|
|
||||||
return fiber.NewError(fiber.StatusInternalServerError, "Failed to validate product warehouse")
|
|
||||||
}
|
|
||||||
|
|
||||||
wh, err := wRepo.GetByID(ctx, uint(pw.WarehouseId), nil)
|
|
||||||
if err != nil {
|
|
||||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
||||||
return fiber.NewError(fiber.StatusBadRequest,
|
|
||||||
fmt.Sprintf("Warehouse %d tidak ditemukan", pw.WarehouseId))
|
|
||||||
}
|
|
||||||
return fiber.NewError(fiber.StatusInternalServerError, "Failed to validate warehouse")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Warehouse tanpa kandang → bukan kandang produksi → skip
|
|
||||||
if wh.KandangId == nil || *wh.KandangId == 0 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
kandangID := uint(*wh.KandangId)
|
|
||||||
if _, ok := seenKandang[kandangID]; ok {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
seenKandang[kandangID] = struct{}{}
|
|
||||||
|
|
||||||
pfk, err := pfkRepo.GetActiveByKandangID(ctx, kandangID)
|
|
||||||
if err != nil {
|
|
||||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
||||||
// nggak ada project aktif untuk kandang ini → aman
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
return fiber.NewError(fiber.StatusInternalServerError, "Failed to validate project flock")
|
|
||||||
}
|
|
||||||
// INTI RULE: kalau aktif tapi sudah punya ClosedAt → anggap "project sudah closing"
|
|
||||||
if pfk != nil && pfk.ClosedAt != nil {
|
|
||||||
return fiber.NewError(fiber.StatusBadRequest, "Project sudah closing")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func EnsureProjectFlockNotClosedByProjectFlockKandangID(
|
|
||||||
ctx context.Context,
|
|
||||||
db *gorm.DB,
|
|
||||||
pfkIDs []uint,
|
|
||||||
) error {
|
|
||||||
pfkRepo := projectFlockKandangRepo.NewProjectFlockKandangRepository(db)
|
|
||||||
|
|
||||||
seen := make(map[uint]struct{})
|
|
||||||
for _, id := range pfkIDs {
|
|
||||||
if id == 0 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if _, ok := seen[id]; ok {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
seen[id] = struct{}{}
|
|
||||||
|
|
||||||
pfk, err := pfkRepo.GetByID(ctx, id)
|
|
||||||
if err != nil {
|
|
||||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
||||||
return fiber.NewError(fiber.StatusBadRequest,
|
|
||||||
fmt.Sprintf("Project flock kandang %d tidak ditemukan", id))
|
|
||||||
}
|
|
||||||
return fiber.NewError(fiber.StatusInternalServerError, "Failed to validate project flock")
|
|
||||||
}
|
|
||||||
|
|
||||||
if pfk.ClosedAt != nil {
|
|
||||||
return fiber.NewError(fiber.StatusBadRequest, "Project sudah closing")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
@@ -6,10 +6,8 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"mime"
|
"mime"
|
||||||
"mime/multipart"
|
"mime/multipart"
|
||||||
"net/url"
|
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
|
||||||
|
|
||||||
commonRepo "gitlab.com/mbugroup/lti-api.git/internal/common/repository"
|
commonRepo "gitlab.com/mbugroup/lti-api.git/internal/common/repository"
|
||||||
"gitlab.com/mbugroup/lti-api.git/internal/config"
|
"gitlab.com/mbugroup/lti-api.git/internal/config"
|
||||||
@@ -31,7 +29,6 @@ type DocumentService interface {
|
|||||||
DeleteDocuments(ctx context.Context, ids []uint, removeFromStorage bool) error
|
DeleteDocuments(ctx context.Context, ids []uint, removeFromStorage bool) error
|
||||||
DeleteByTarget(ctx context.Context, documentableType string, documentableID uint64, removeFromStorage bool) error
|
DeleteByTarget(ctx context.Context, documentableType string, documentableID uint64, removeFromStorage bool) error
|
||||||
PublicURL(document entity.Document) string
|
PublicURL(document entity.Document) string
|
||||||
PresignURL(ctx context.Context, document entity.Document, expires time.Duration) (string, error)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type DocumentUploadRequest struct {
|
type DocumentUploadRequest struct {
|
||||||
@@ -296,66 +293,6 @@ func (s *documentService) PublicURL(document entity.Document) string {
|
|||||||
return s.storage.URL(document.Path)
|
return s.storage.URL(document.Path)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *documentService) PresignURL(ctx context.Context, document entity.Document, expires time.Duration) (string, error) {
|
|
||||||
if s.storage == nil {
|
|
||||||
return "", errors.New("document storage not configured")
|
|
||||||
}
|
|
||||||
if strings.TrimSpace(document.Path) == "" {
|
|
||||||
return "", errors.New("document path is required")
|
|
||||||
}
|
|
||||||
return s.storage.PresignURL(ctx, document.Path, expires)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ResolveDocumentURL normalizes a stored path or URL into a presigned URL.
|
|
||||||
func ResolveDocumentURL(
|
|
||||||
ctx context.Context,
|
|
||||||
svc DocumentService,
|
|
||||||
rawPath string,
|
|
||||||
expires time.Duration,
|
|
||||||
) (string, error) {
|
|
||||||
if svc == nil {
|
|
||||||
return "", nil
|
|
||||||
}
|
|
||||||
|
|
||||||
rawPath = strings.TrimSpace(rawPath)
|
|
||||||
if rawPath == "" {
|
|
||||||
return "", nil
|
|
||||||
}
|
|
||||||
|
|
||||||
key := rawPath
|
|
||||||
lower := strings.ToLower(rawPath)
|
|
||||||
if strings.HasPrefix(lower, "http://") || strings.HasPrefix(lower, "https://") {
|
|
||||||
key = extractS3KeyFromURL(rawPath)
|
|
||||||
if key == "" {
|
|
||||||
return "", nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return svc.PresignURL(ctx, entity.Document{Path: key}, expires)
|
|
||||||
}
|
|
||||||
|
|
||||||
func extractS3KeyFromURL(raw string) string {
|
|
||||||
parsed, err := url.Parse(strings.TrimSpace(raw))
|
|
||||||
if err != nil {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
path := strings.TrimPrefix(parsed.Path, "/")
|
|
||||||
if path == "" {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
host := strings.ToLower(strings.TrimSpace(parsed.Host))
|
|
||||||
if strings.HasPrefix(host, "s3.") || strings.HasPrefix(host, "s3-") {
|
|
||||||
parts := strings.SplitN(path, "/", 2)
|
|
||||||
if len(parts) == 2 {
|
|
||||||
return parts[1]
|
|
||||||
}
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
return path
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *documentService) generateObjectKey(ext string) (string, error) {
|
func (s *documentService) generateObjectKey(ext string) (string, error) {
|
||||||
normalizedExt := strings.TrimSpace(ext)
|
normalizedExt := strings.TrimSpace(ext)
|
||||||
if normalizedExt != "" && !strings.HasPrefix(normalizedExt, ".") {
|
if normalizedExt != "" && !strings.HasPrefix(normalizedExt, ".") {
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/aws/aws-sdk-go-v2/aws"
|
"github.com/aws/aws-sdk-go-v2/aws"
|
||||||
awsconfig "github.com/aws/aws-sdk-go-v2/config"
|
awsconfig "github.com/aws/aws-sdk-go-v2/config"
|
||||||
@@ -18,7 +17,6 @@ type DocumentStorage interface {
|
|||||||
Upload(ctx context.Context, key string, body io.Reader, size int64, contentType string) (DocumentStorageUploadResult, error)
|
Upload(ctx context.Context, key string, body io.Reader, size int64, contentType string) (DocumentStorageUploadResult, error)
|
||||||
Delete(ctx context.Context, key string) error
|
Delete(ctx context.Context, key string) error
|
||||||
URL(key string) string
|
URL(key string) string
|
||||||
PresignURL(ctx context.Context, key string, expires time.Duration) (string, error)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type DocumentStorageUploadResult struct {
|
type DocumentStorageUploadResult struct {
|
||||||
@@ -38,10 +36,9 @@ type S3DocumentStorageConfig struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type s3DocumentStorage struct {
|
type s3DocumentStorage struct {
|
||||||
client *s3.Client
|
client *s3.Client
|
||||||
presignClient *s3.PresignClient
|
bucket string
|
||||||
bucket string
|
base string
|
||||||
base string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewS3DocumentStorage(ctx context.Context, cfg S3DocumentStorageConfig) (DocumentStorage, error) {
|
func NewS3DocumentStorage(ctx context.Context, cfg S3DocumentStorageConfig) (DocumentStorage, error) {
|
||||||
@@ -89,7 +86,6 @@ func NewS3DocumentStorage(ctx context.Context, cfg S3DocumentStorageConfig) (Doc
|
|||||||
client := s3.NewFromConfig(awsCfg, func(o *s3.Options) {
|
client := s3.NewFromConfig(awsCfg, func(o *s3.Options) {
|
||||||
o.UsePathStyle = cfg.ForcePathStyle
|
o.UsePathStyle = cfg.ForcePathStyle
|
||||||
})
|
})
|
||||||
presignClient := s3.NewPresignClient(client)
|
|
||||||
|
|
||||||
baseURL := strings.TrimSuffix(strings.TrimSpace(cfg.BaseURL), "/")
|
baseURL := strings.TrimSuffix(strings.TrimSpace(cfg.BaseURL), "/")
|
||||||
if baseURL == "" {
|
if baseURL == "" {
|
||||||
@@ -101,10 +97,9 @@ func NewS3DocumentStorage(ctx context.Context, cfg S3DocumentStorageConfig) (Doc
|
|||||||
}
|
}
|
||||||
|
|
||||||
return &s3DocumentStorage{
|
return &s3DocumentStorage{
|
||||||
client: client,
|
client: client,
|
||||||
presignClient: presignClient,
|
bucket: bucket,
|
||||||
bucket: bucket,
|
base: baseURL,
|
||||||
base: baseURL,
|
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -163,23 +158,3 @@ func (s *s3DocumentStorage) URL(key string) string {
|
|||||||
}
|
}
|
||||||
return fmt.Sprintf("%s/%s", s.base, key)
|
return fmt.Sprintf("%s/%s", s.base, key)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *s3DocumentStorage) PresignURL(ctx context.Context, key string, expires time.Duration) (string, error) {
|
|
||||||
key = strings.TrimPrefix(strings.TrimSpace(key), "/")
|
|
||||||
if key == "" {
|
|
||||||
return "", errors.New("storage key is required")
|
|
||||||
}
|
|
||||||
if expires <= 0 {
|
|
||||||
expires = 15 * time.Minute
|
|
||||||
}
|
|
||||||
|
|
||||||
out, err := s.presignClient.PresignGetObject(ctx, &s3.GetObjectInput{
|
|
||||||
Bucket: aws.String(s.bucket),
|
|
||||||
Key: aws.String(key),
|
|
||||||
}, s3.WithPresignExpires(expires))
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
|
|
||||||
return out.URL, nil
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -192,6 +192,7 @@ func (s *fifoService) Consume(ctx context.Context, req StockConsumeRequest) (*St
|
|||||||
if req.Quantity < 0 {
|
if req.Quantity < 0 {
|
||||||
return nil, errors.New("quantity must be zero or greater")
|
return nil, errors.New("quantity must be zero or greater")
|
||||||
}
|
}
|
||||||
|
|
||||||
cfg, ok := fifo.Usable(req.UsableKey)
|
cfg, ok := fifo.Usable(req.UsableKey)
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, fmt.Errorf("usable %q is not registered", req.UsableKey)
|
return nil, fmt.Errorf("usable %q is not registered", req.UsableKey)
|
||||||
@@ -219,6 +220,7 @@ func (s *fifoService) Consume(ctx context.Context, req StockConsumeRequest) (*St
|
|||||||
currentPending := ctxRow.PendingQty
|
currentPending := ctxRow.PendingQty
|
||||||
currentTotal := currentUsage + currentPending
|
currentTotal := currentUsage + currentPending
|
||||||
delta := req.Quantity - currentTotal
|
delta := req.Quantity - currentTotal
|
||||||
|
|
||||||
var (
|
var (
|
||||||
usageDelta float64
|
usageDelta float64
|
||||||
pendingDelta float64
|
pendingDelta float64
|
||||||
@@ -228,13 +230,7 @@ func (s *fifoService) Consume(ctx context.Context, req StockConsumeRequest) (*St
|
|||||||
|
|
||||||
switch {
|
switch {
|
||||||
case delta > 0:
|
case delta > 0:
|
||||||
|
allocationRes, err := s.allocateFromStock(ctx, tx, productWarehouseID, req.UsableKey, req.UsableID, delta)
|
||||||
var excludedStockables []fifo.StockableKey
|
|
||||||
if cfg.ExcludedStockables != nil {
|
|
||||||
excludedStockables = cfg.ExcludedStockables
|
|
||||||
}
|
|
||||||
|
|
||||||
allocationRes, err := s.allocateFromStock(ctx, tx, productWarehouseID, req.UsableKey, req.UsableID, delta, excludedStockables)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -289,6 +285,7 @@ func (s *fifoService) Consume(ctx context.Context, req StockConsumeRequest) (*St
|
|||||||
result.ReleasedQuantity = releasedAmount
|
result.ReleasedQuantity = releasedAmount
|
||||||
result.UsageQuantity = currentUsage + usageDelta
|
result.UsageQuantity = currentUsage + usageDelta
|
||||||
result.PendingQuantity = currentPending + pendingDelta
|
result.PendingQuantity = currentPending + pendingDelta
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -302,6 +299,7 @@ func (s *fifoService) ReleaseUsage(ctx context.Context, req StockReleaseRequest)
|
|||||||
if req.UsableID == 0 || strings.TrimSpace(req.UsableKey.String()) == "" {
|
if req.UsableID == 0 || strings.TrimSpace(req.UsableKey.String()) == "" {
|
||||||
return errors.New("usable key and id are required")
|
return errors.New("usable key and id are required")
|
||||||
}
|
}
|
||||||
|
|
||||||
return s.withTransaction(ctx, req.Tx, func(tx *gorm.DB) error {
|
return s.withTransaction(ctx, req.Tx, func(tx *gorm.DB) error {
|
||||||
cfg, ok := fifo.Usable(req.UsableKey)
|
cfg, ok := fifo.Usable(req.UsableKey)
|
||||||
if !ok {
|
if !ok {
|
||||||
@@ -312,6 +310,7 @@ func (s *fifoService) ReleaseUsage(ctx context.Context, req StockReleaseRequest)
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
var usageDelta, pendingDelta float64
|
var usageDelta, pendingDelta float64
|
||||||
if ctxRow.UsageQty > 0 {
|
if ctxRow.UsageQty > 0 {
|
||||||
if _, err := s.releaseUsagePortion(ctx, tx, req.UsableKey, req.UsableID, ctxRow.UsageQty); err != nil {
|
if _, err := s.releaseUsagePortion(ctx, tx, req.UsableKey, req.UsableID, ctxRow.UsageQty); err != nil {
|
||||||
@@ -416,9 +415,8 @@ func (s *fifoService) allocateFromStock(
|
|||||||
usableKey fifo.UsableKey,
|
usableKey fifo.UsableKey,
|
||||||
usableID uint,
|
usableID uint,
|
||||||
requestQty float64,
|
requestQty float64,
|
||||||
excludedStockables []fifo.StockableKey,
|
|
||||||
) (*allocationOutcome, error) {
|
) (*allocationOutcome, error) {
|
||||||
lots, err := s.fetchStockLots(ctx, tx, productWarehouseID, excludedStockables)
|
lots, err := s.fetchStockLots(ctx, tx, productWarehouseID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -499,43 +497,20 @@ func (s *fifoService) allocateFromStock(
|
|||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *fifoService) fetchStockLots(ctx context.Context, tx *gorm.DB, productWarehouseID uint, excludedStockables []fifo.StockableKey) ([]stockLot, error) {
|
func (s *fifoService) fetchStockLots(ctx context.Context, tx *gorm.DB, productWarehouseID uint) ([]stockLot, error) {
|
||||||
configs := fifo.Stockables()
|
configs := fifo.Stockables()
|
||||||
if len(configs) == 0 {
|
if len(configs) == 0 {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create exclusion set for faster lookup
|
|
||||||
excludedSet := make(map[fifo.StockableKey]bool)
|
|
||||||
for _, key := range excludedStockables {
|
|
||||||
excludedSet[key] = true
|
|
||||||
}
|
|
||||||
|
|
||||||
var lots []stockLot
|
var lots []stockLot
|
||||||
for key, cfg := range configs {
|
for key, cfg := range configs {
|
||||||
// Skip excluded stockables
|
selectStmt := fmt.Sprintf(
|
||||||
if excludedSet[key] {
|
"%s AS id, %s AS available_qty, %s AS created_at",
|
||||||
continue
|
cfg.Columns.ID,
|
||||||
}
|
fmt.Sprintf("%s - COALESCE(%s,0)", cfg.Columns.TotalQuantity, cfg.Columns.TotalUsedQuantity),
|
||||||
|
cfg.Columns.CreatedAt,
|
||||||
usesNumericTime := cfg.Columns.CreatedAt == cfg.Columns.ID
|
)
|
||||||
|
|
||||||
var selectStmt string
|
|
||||||
if usesNumericTime {
|
|
||||||
|
|
||||||
selectStmt = fmt.Sprintf(
|
|
||||||
"%s AS id, %s AS available_qty, '1970-01-01 00:00:00 UTC'::timestamp AS created_at",
|
|
||||||
cfg.Columns.ID,
|
|
||||||
fmt.Sprintf("%s - COALESCE(%s,0)", cfg.Columns.TotalQuantity, cfg.Columns.TotalUsedQuantity),
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
selectStmt = fmt.Sprintf(
|
|
||||||
"%s AS id, %s AS available_qty, %s AS created_at",
|
|
||||||
cfg.Columns.ID,
|
|
||||||
fmt.Sprintf("%s - COALESCE(%s,0)", cfg.Columns.TotalQuantity, cfg.Columns.TotalUsedQuantity),
|
|
||||||
cfg.Columns.CreatedAt,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
var rows []struct {
|
var rows []struct {
|
||||||
ID uint
|
ID uint
|
||||||
@@ -633,13 +608,7 @@ func (s *fifoService) resolvePendingForWarehouse(ctx context.Context, tx *gorm.D
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get excluded stockables from candidate usable config
|
outcome, err := s.allocateFromStock(ctx, tx, productWarehouseID, candidate.UsableKey, candidate.UsableID, candidate.Pending)
|
||||||
var excludedStockables []fifo.StockableKey
|
|
||||||
if candidate.Config.ExcludedStockables != nil {
|
|
||||||
excludedStockables = candidate.Config.ExcludedStockables
|
|
||||||
}
|
|
||||||
|
|
||||||
outcome, err := s.allocateFromStock(ctx, tx, productWarehouseID, candidate.UsableKey, candidate.UsableID, candidate.Pending, excludedStockables)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -733,7 +702,7 @@ func (s *fifoService) releaseUsagePortion(
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if err := s.allocations.PatchOne(ctx, allocation.Id, map[string]any{
|
if err := s.allocations.PatchOne(ctx, allocation.Id, map[string]any{
|
||||||
"qty": allocation.Qty - releaseAmt,
|
"quantity": allocation.Qty - releaseAmt,
|
||||||
}, func(db *gorm.DB) *gorm.DB {
|
}, func(db *gorm.DB) *gorm.DB {
|
||||||
return s.txOrDB(tx, db)
|
return s.txOrDB(tx, db)
|
||||||
}); err != nil {
|
}); err != nil {
|
||||||
|
|||||||
@@ -54,7 +54,6 @@ var (
|
|||||||
SSOAuthorizeURL string
|
SSOAuthorizeURL string
|
||||||
SSOTokenURL string
|
SSOTokenURL string
|
||||||
SSOGetMeURL string
|
SSOGetMeURL string
|
||||||
SSOPortalURL string
|
|
||||||
SSOClients map[string]SSOClientConfig
|
SSOClients map[string]SSOClientConfig
|
||||||
SSOAccessCookieName string
|
SSOAccessCookieName string
|
||||||
SSORefreshCookieName string
|
SSORefreshCookieName string
|
||||||
@@ -132,7 +131,6 @@ func init() {
|
|||||||
SSOAuthorizeURL = viper.GetString("SSO_AUTHORIZE_URL")
|
SSOAuthorizeURL = viper.GetString("SSO_AUTHORIZE_URL")
|
||||||
SSOTokenURL = viper.GetString("SSO_TOKEN_URL")
|
SSOTokenURL = viper.GetString("SSO_TOKEN_URL")
|
||||||
SSOGetMeURL = viper.GetString("SSO_GETME_URL")
|
SSOGetMeURL = viper.GetString("SSO_GETME_URL")
|
||||||
SSOPortalURL = strings.TrimSpace(viper.GetString("SSO_PORTAL_URL"))
|
|
||||||
SSOAccessCookieName = defaultString(viper.GetString("SSO_ACCESS_COOKIE_NAME"), "sso_access")
|
SSOAccessCookieName = defaultString(viper.GetString("SSO_ACCESS_COOKIE_NAME"), "sso_access")
|
||||||
SSORefreshCookieName = defaultString(viper.GetString("SSO_REFRESH_COOKIE_NAME"), "sso_refresh")
|
SSORefreshCookieName = defaultString(viper.GetString("SSO_REFRESH_COOKIE_NAME"), "sso_refresh")
|
||||||
SSOCookieDomain = viper.GetString("SSO_COOKIE_DOMAIN")
|
SSOCookieDomain = viper.GetString("SSO_COOKIE_DOMAIN")
|
||||||
|
|||||||
@@ -13,7 +13,6 @@ func FiberConfig() fiber.Config {
|
|||||||
CaseSensitive: true,
|
CaseSensitive: true,
|
||||||
ServerHeader: "Fiber",
|
ServerHeader: "Fiber",
|
||||||
AppName: "Fiber API",
|
AppName: "Fiber API",
|
||||||
BodyLimit: 8 * 1024 * 1024,
|
|
||||||
ErrorHandler: utils.ErrorHandler,
|
ErrorHandler: utils.ErrorHandler,
|
||||||
JSONEncoder: sonic.Marshal,
|
JSONEncoder: sonic.Marshal,
|
||||||
JSONDecoder: sonic.Unmarshal,
|
JSONDecoder: sonic.Unmarshal,
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ ADD CONSTRAINT fk_project_chickins_kandang FOREIGN KEY (project_flock_kandang_id
|
|||||||
|
|
||||||
-- Relasi ke product_warehouses
|
-- Relasi ke product_warehouses
|
||||||
ALTER TABLE project_chickins
|
ALTER TABLE project_chickins
|
||||||
ADD CONSTRAINT fk_project_chickins_warehouse FOREIGN KEY (product_warehouse_id) REFERENCES product_warehouses (id) ON DELETE CASCADE ON UPDATE CASCADE;
|
ADD CONSTRAINT fk_project_chickins_warehouse FOREIGN KEY (product_warehouse_id) REFERENCES product_warehouses (id) ON DELETE RESTRICT ON UPDATE CASCADE;
|
||||||
|
|
||||||
-- Relasi ke users
|
-- Relasi ke users
|
||||||
ALTER TABLE project_chickins
|
ALTER TABLE project_chickins
|
||||||
|
|||||||
@@ -1,2 +1 @@
|
|||||||
DROP SEQUENCE IF EXISTS expenses_ref_seq;
|
DROP TABLE IF EXISTS expenses;
|
||||||
DROP TABLE IF EXISTS expenses;
|
|
||||||
-3
@@ -1,3 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS idx_project_flock_kandangs_closed_at;
|
|
||||||
ALTER TABLE project_flock_kandangs
|
|
||||||
DROP COLUMN IF EXISTS closed_at;
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
ALTER TABLE project_flock_kandangs
|
|
||||||
ADD COLUMN IF NOT EXISTS closed_at TIMESTAMPTZ;
|
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_project_flock_kandangs_closed_at
|
|
||||||
ON project_flock_kandangs (closed_at);
|
|
||||||
+1
-1
@@ -20,7 +20,7 @@ ALTER TABLE product_warehouses
|
|||||||
|
|
||||||
-- Restore audit/soft-delete columns
|
-- Restore audit/soft-delete columns
|
||||||
ALTER TABLE product_warehouses
|
ALTER TABLE product_warehouses
|
||||||
ADD COLUMN IF NOT EXISTS created_by BIGINT REFERENCES users (id),
|
ADD COLUMN IF NOT EXISTS created_by BIGINT NOT NULL REFERENCES users (id),
|
||||||
ADD COLUMN IF NOT EXISTS created_at TIMESTAMPTZ DEFAULT NOW(),
|
ADD COLUMN IF NOT EXISTS created_at TIMESTAMPTZ DEFAULT NOW(),
|
||||||
ADD COLUMN IF NOT EXISTS updated_at TIMESTAMPTZ DEFAULT NOW(),
|
ADD COLUMN IF NOT EXISTS updated_at TIMESTAMPTZ DEFAULT NOW(),
|
||||||
ADD COLUMN IF NOT EXISTS deleted_at TIMESTAMPTZ;
|
ADD COLUMN IF NOT EXISTS deleted_at TIMESTAMPTZ;
|
||||||
|
|||||||
-33
@@ -1,33 +0,0 @@
|
|||||||
BEGIN;
|
|
||||||
|
|
||||||
-- Remove grading details from recording_eggs
|
|
||||||
ALTER TABLE recording_eggs
|
|
||||||
DROP CONSTRAINT IF EXISTS chk_recording_eggs_qty;
|
|
||||||
|
|
||||||
ALTER TABLE recording_eggs
|
|
||||||
DROP COLUMN IF EXISTS weight;
|
|
||||||
|
|
||||||
ALTER TABLE recording_eggs
|
|
||||||
ADD CONSTRAINT chk_recording_eggs_qty CHECK (qty >= 0);
|
|
||||||
|
|
||||||
-- Restore grading_eggs table for rollback scenarios
|
|
||||||
CREATE TABLE grading_eggs (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
recording_egg_id BIGINT NOT NULL,
|
|
||||||
qty NUMERIC(15,3) NOT NULL,
|
|
||||||
grade VARCHAR,
|
|
||||||
created_by BIGINT,
|
|
||||||
created_at TIMESTAMPTZ DEFAULT NOW(),
|
|
||||||
updated_at TIMESTAMPTZ DEFAULT NOW(),
|
|
||||||
|
|
||||||
CONSTRAINT fk_grading_eggs_recording_egg
|
|
||||||
FOREIGN KEY (recording_egg_id) REFERENCES recording_eggs(id) ON DELETE CASCADE,
|
|
||||||
CONSTRAINT fk_grading_eggs_created_by
|
|
||||||
FOREIGN KEY (created_by) REFERENCES users(id),
|
|
||||||
CONSTRAINT chk_grading_eggs_qty CHECK (qty >= 0)
|
|
||||||
);
|
|
||||||
|
|
||||||
CREATE INDEX idx_grading_eggs_recording_egg
|
|
||||||
ON grading_eggs (recording_egg_id);
|
|
||||||
|
|
||||||
COMMIT;
|
|
||||||
-18
@@ -1,18 +0,0 @@
|
|||||||
BEGIN;
|
|
||||||
|
|
||||||
-- Remove separate grading table and move grading details into recording_eggs
|
|
||||||
DROP INDEX IF EXISTS idx_grading_eggs_recording_egg;
|
|
||||||
DROP TABLE IF EXISTS grading_eggs;
|
|
||||||
|
|
||||||
ALTER TABLE recording_eggs
|
|
||||||
ADD COLUMN IF NOT EXISTS weight NUMERIC(10,3);
|
|
||||||
|
|
||||||
ALTER TABLE recording_eggs
|
|
||||||
DROP CONSTRAINT IF EXISTS chk_recording_eggs_qty;
|
|
||||||
|
|
||||||
ALTER TABLE recording_eggs
|
|
||||||
ADD CONSTRAINT chk_recording_eggs_qty CHECK (
|
|
||||||
qty >= 0 AND (weight IS NULL OR weight >= 0)
|
|
||||||
);
|
|
||||||
|
|
||||||
COMMIT;
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint WHERE conname = 'fk_purchase_items_expense_nonstock'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE purchase_items
|
|
||||||
DROP CONSTRAINT fk_purchase_items_expense_nonstock;
|
|
||||||
END IF;
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint WHERE conname = 'fk_purchase_items_project_flock_kandang'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE purchase_items
|
|
||||||
DROP CONSTRAINT fk_purchase_items_project_flock_kandang;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
DROP INDEX IF EXISTS idx_purchase_items_expense_nonstock_id;
|
|
||||||
DROP INDEX IF EXISTS idx_purchase_items_project_flock_kandang_id;
|
|
||||||
|
|
||||||
ALTER TABLE purchase_items
|
|
||||||
DROP COLUMN IF EXISTS expense_nonstock_id,
|
|
||||||
DROP COLUMN IF EXISTS project_flock_kandang_id,
|
|
||||||
ALTER COLUMN vehicle_number DROP NOT NULL,
|
|
||||||
ALTER COLUMN vehicle_number TYPE VARCHAR USING vehicle_number;
|
|
||||||
|
|
||||||
ALTER TABLE purchases
|
|
||||||
ALTER COLUMN pr_number TYPE VARCHAR USING pr_number,
|
|
||||||
ALTER COLUMN po_number TYPE VARCHAR USING po_number,
|
|
||||||
ALTER COLUMN created_at DROP DEFAULT,
|
|
||||||
ALTER COLUMN updated_at DROP DEFAULT;
|
|
||||||
|
|
||||||
ALTER TABLE purchases
|
|
||||||
ADD COLUMN credit_term INT NOT NULL DEFAULT 0,
|
|
||||||
ADD COLUMN grand_total NUMERIC(15, 3) NOT NULL DEFAULT 0;
|
|
||||||
|
|
||||||
ALTER TABLE purchases
|
|
||||||
ALTER COLUMN credit_term DROP DEFAULT,
|
|
||||||
ALTER COLUMN grand_total DROP DEFAULT;
|
|
||||||
@@ -1,57 +0,0 @@
|
|||||||
-- Adjust purchases table to new purchasing schema
|
|
||||||
ALTER TABLE purchases
|
|
||||||
ALTER COLUMN pr_number TYPE VARCHAR(50) USING LEFT(pr_number, 50),
|
|
||||||
ALTER COLUMN po_number TYPE VARCHAR(50) USING LEFT(po_number, 50),
|
|
||||||
ALTER COLUMN created_at SET DEFAULT now(),
|
|
||||||
ALTER COLUMN updated_at SET DEFAULT now();
|
|
||||||
|
|
||||||
ALTER TABLE purchases
|
|
||||||
DROP COLUMN IF EXISTS credit_term,
|
|
||||||
DROP COLUMN IF EXISTS grand_total;
|
|
||||||
|
|
||||||
-- Bring purchase_items in line with new requirements
|
|
||||||
ALTER TABLE purchase_items
|
|
||||||
ADD COLUMN IF NOT EXISTS expense_nonstock_id BIGINT,
|
|
||||||
ADD COLUMN IF NOT EXISTS project_flock_kandang_id BIGINT;
|
|
||||||
|
|
||||||
UPDATE purchase_items
|
|
||||||
SET vehicle_number = ''
|
|
||||||
WHERE vehicle_number IS NULL;
|
|
||||||
|
|
||||||
ALTER TABLE purchase_items
|
|
||||||
ALTER COLUMN vehicle_number TYPE VARCHAR(10) USING LEFT(vehicle_number, 10),
|
|
||||||
ALTER COLUMN vehicle_number SET NOT NULL;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'expense_nonstocks') THEN
|
|
||||||
IF NOT EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint WHERE conname = 'fk_purchase_items_expense_nonstock'
|
|
||||||
) THEN
|
|
||||||
EXECUTE
|
|
||||||
'ALTER TABLE purchase_items
|
|
||||||
ADD CONSTRAINT fk_purchase_items_expense_nonstock
|
|
||||||
FOREIGN KEY (expense_nonstock_id)
|
|
||||||
REFERENCES expense_nonstocks(id)
|
|
||||||
ON DELETE SET NULL ON UPDATE CASCADE';
|
|
||||||
END IF;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'project_flock_kandangs') THEN
|
|
||||||
IF NOT EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint WHERE conname = 'fk_purchase_items_project_flock_kandang'
|
|
||||||
) THEN
|
|
||||||
EXECUTE
|
|
||||||
'ALTER TABLE purchase_items
|
|
||||||
ADD CONSTRAINT fk_purchase_items_project_flock_kandang
|
|
||||||
FOREIGN KEY (project_flock_kandang_id)
|
|
||||||
REFERENCES project_flock_kandangs(id)
|
|
||||||
ON DELETE SET NULL ON UPDATE CASCADE';
|
|
||||||
END IF;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_purchase_items_expense_nonstock_id
|
|
||||||
ON purchase_items (expense_nonstock_id);
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_purchase_items_project_flock_kandang_id
|
|
||||||
ON purchase_items (project_flock_kandang_id);
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
-- Drop function and sequence for sales order numbers
|
|
||||||
DROP SEQUENCE IF EXISTS so_number_seq;
|
|
||||||
DROP FUNCTION IF EXISTS generate_so_number();
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
-- Create sequence for sales order numbers
|
|
||||||
CREATE SEQUENCE so_number_seq START WITH 1 INCREMENT BY 1;
|
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION generate_so_number()
|
|
||||||
RETURNS VARCHAR AS $$
|
|
||||||
DECLARE
|
|
||||||
next_val INTEGER;
|
|
||||||
BEGIN
|
|
||||||
next_val := nextval('so_number_seq');
|
|
||||||
RETURN 'SO-' || LPAD(next_val::TEXT, 5, '0');
|
|
||||||
END;
|
|
||||||
$$ LANGUAGE plpgsql;
|
|
||||||
-2
@@ -1,2 +0,0 @@
|
|||||||
ALTER TABLE purchases
|
|
||||||
DROP COLUMN IF EXISTS credit_term;
|
|
||||||
-5
@@ -1,5 +0,0 @@
|
|||||||
ALTER TABLE purchases
|
|
||||||
ADD COLUMN IF NOT EXISTS credit_term INT NOT NULL DEFAULT 0;
|
|
||||||
|
|
||||||
ALTER TABLE purchases
|
|
||||||
ALTER COLUMN credit_term DROP DEFAULT;
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS idx_payments_bank_id;
|
|
||||||
DROP INDEX IF EXISTS payments_party_polymorphic;
|
|
||||||
DROP TABLE IF EXISTS payments;
|
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
CREATE TABLE IF NOT EXISTS payments (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
payment_code VARCHAR(50) NOT NULL,
|
|
||||||
reference_number VARCHAR(100) NULL,
|
|
||||||
transaction_type VARCHAR(50),
|
|
||||||
party_type VARCHAR(50) NOT NULL,
|
|
||||||
party_id BIGINT NOT NULL,
|
|
||||||
payment_date TIMESTAMPTZ NOT NULL,
|
|
||||||
payment_method VARCHAR(20) NOT NULL,
|
|
||||||
bank_id BIGINT NULL REFERENCES banks(id) ON DELETE RESTRICT ON UPDATE CASCADE,
|
|
||||||
direction VARCHAR(5) NOT NULL,
|
|
||||||
nominal NUMERIC(15, 3) NOT NULL,
|
|
||||||
notes TEXT NOT NULL,
|
|
||||||
created_at TIMESTAMPTZ NOT NULL DEFAULT NOW(),
|
|
||||||
updated_at TIMESTAMPTZ NOT NULL DEFAULT NOW(),
|
|
||||||
deleted_at TIMESTAMPTZ NULL,
|
|
||||||
created_by BIGINT REFERENCES users (id) ON DELETE SET NULL ON UPDATE CASCADE
|
|
||||||
);
|
|
||||||
|
|
||||||
-- Indexes
|
|
||||||
CREATE INDEX payments_party_polymorphic ON payments (party_type, party_id);
|
|
||||||
CREATE INDEX idx_payments_bank_id ON payments (bank_id);
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
DO $$
|
|
||||||
DECLARE
|
|
||||||
r record;
|
|
||||||
trigger_name text;
|
|
||||||
BEGIN
|
|
||||||
FOR r IN
|
|
||||||
SELECT table_schema, table_name
|
|
||||||
FROM information_schema.columns
|
|
||||||
WHERE column_name = 'deleted_at'
|
|
||||||
AND table_schema = 'public'
|
|
||||||
GROUP BY table_schema, table_name
|
|
||||||
LOOP
|
|
||||||
trigger_name := format('trg_soft_delete_fk_%s', r.table_name);
|
|
||||||
EXECUTE format('DROP TRIGGER IF EXISTS %I ON %I.%I', trigger_name, r.table_schema, r.table_name);
|
|
||||||
END LOOP;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
DROP FUNCTION IF EXISTS soft_delete_handle_fk();
|
|
||||||
@@ -1,126 +0,0 @@
|
|||||||
CREATE OR REPLACE FUNCTION soft_delete_handle_fk() RETURNS TRIGGER AS $$
|
|
||||||
DECLARE
|
|
||||||
fk record;
|
|
||||||
child_column text;
|
|
||||||
parent_column text;
|
|
||||||
parent_value text;
|
|
||||||
child_has_deleted_at boolean;
|
|
||||||
ref_exists boolean;
|
|
||||||
sql text;
|
|
||||||
BEGIN
|
|
||||||
IF OLD.deleted_at IS NULL AND NEW.deleted_at IS NOT NULL THEN
|
|
||||||
FOR fk IN
|
|
||||||
SELECT conrelid::regclass AS child_table,
|
|
||||||
conkey AS child_cols,
|
|
||||||
confkey AS parent_cols,
|
|
||||||
confdeltype
|
|
||||||
FROM pg_constraint
|
|
||||||
WHERE contype = 'f'
|
|
||||||
AND confrelid = TG_RELID
|
|
||||||
LOOP
|
|
||||||
IF array_length(fk.child_cols, 1) IS DISTINCT FROM 1
|
|
||||||
OR array_length(fk.parent_cols, 1) IS DISTINCT FROM 1 THEN
|
|
||||||
RAISE NOTICE 'soft_delete_handle_fk skipped composite fk on %', fk.child_table;
|
|
||||||
CONTINUE;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
SELECT attname INTO child_column
|
|
||||||
FROM pg_attribute
|
|
||||||
WHERE attrelid = fk.child_table
|
|
||||||
AND attnum = fk.child_cols[1]
|
|
||||||
AND NOT attisdropped;
|
|
||||||
|
|
||||||
SELECT attname INTO parent_column
|
|
||||||
FROM pg_attribute
|
|
||||||
WHERE attrelid = TG_RELID
|
|
||||||
AND attnum = fk.parent_cols[1]
|
|
||||||
AND NOT attisdropped;
|
|
||||||
|
|
||||||
EXECUTE format('SELECT ($1).%I', parent_column)
|
|
||||||
INTO parent_value
|
|
||||||
USING OLD;
|
|
||||||
|
|
||||||
SELECT EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_attribute
|
|
||||||
WHERE attrelid = fk.child_table
|
|
||||||
AND attname = 'deleted_at'
|
|
||||||
AND NOT attisdropped
|
|
||||||
) INTO child_has_deleted_at;
|
|
||||||
|
|
||||||
IF fk.confdeltype IN ('r', 'a') THEN
|
|
||||||
sql := format(
|
|
||||||
'SELECT EXISTS (SELECT 1 FROM %s WHERE %I = $1 %s)',
|
|
||||||
fk.child_table,
|
|
||||||
child_column,
|
|
||||||
CASE WHEN child_has_deleted_at THEN 'AND deleted_at IS NULL' ELSE '' END
|
|
||||||
);
|
|
||||||
EXECUTE sql INTO ref_exists USING parent_value;
|
|
||||||
IF ref_exists THEN
|
|
||||||
RAISE EXCEPTION 'Cannot soft delete %, still referenced by %',
|
|
||||||
TG_TABLE_NAME, fk.child_table;
|
|
||||||
END IF;
|
|
||||||
ELSIF fk.confdeltype = 'n' THEN
|
|
||||||
sql := format(
|
|
||||||
'UPDATE %s SET %I = NULL WHERE %I = $1 %s',
|
|
||||||
fk.child_table,
|
|
||||||
child_column,
|
|
||||||
child_column,
|
|
||||||
CASE WHEN child_has_deleted_at THEN 'AND deleted_at IS NULL' ELSE '' END
|
|
||||||
);
|
|
||||||
EXECUTE sql USING parent_value;
|
|
||||||
ELSIF fk.confdeltype = 'c' THEN
|
|
||||||
IF child_has_deleted_at THEN
|
|
||||||
sql := format(
|
|
||||||
'UPDATE %s SET deleted_at = NOW() WHERE %I = $1 AND deleted_at IS NULL',
|
|
||||||
fk.child_table,
|
|
||||||
child_column
|
|
||||||
);
|
|
||||||
EXECUTE sql USING parent_value;
|
|
||||||
ELSE
|
|
||||||
sql := format(
|
|
||||||
'DELETE FROM %s WHERE %I = $1',
|
|
||||||
fk.child_table,
|
|
||||||
child_column
|
|
||||||
);
|
|
||||||
EXECUTE sql USING parent_value;
|
|
||||||
END IF;
|
|
||||||
ELSIF fk.confdeltype = 'd' THEN
|
|
||||||
sql := format(
|
|
||||||
'UPDATE %s SET %I = DEFAULT WHERE %I = $1 %s',
|
|
||||||
fk.child_table,
|
|
||||||
child_column,
|
|
||||||
child_column,
|
|
||||||
CASE WHEN child_has_deleted_at THEN 'AND deleted_at IS NULL' ELSE '' END
|
|
||||||
);
|
|
||||||
EXECUTE sql USING parent_value;
|
|
||||||
END IF;
|
|
||||||
END LOOP;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
RETURN NEW;
|
|
||||||
END;
|
|
||||||
$$ LANGUAGE plpgsql;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
DECLARE
|
|
||||||
r record;
|
|
||||||
trigger_name text;
|
|
||||||
BEGIN
|
|
||||||
FOR r IN
|
|
||||||
SELECT table_schema, table_name
|
|
||||||
FROM information_schema.columns
|
|
||||||
WHERE column_name = 'deleted_at'
|
|
||||||
AND table_schema = 'public'
|
|
||||||
GROUP BY table_schema, table_name
|
|
||||||
LOOP
|
|
||||||
trigger_name := format('trg_soft_delete_fk_%s', r.table_name);
|
|
||||||
EXECUTE format('DROP TRIGGER IF EXISTS %I ON %I.%I', trigger_name, r.table_schema, r.table_name);
|
|
||||||
EXECUTE format(
|
|
||||||
'CREATE TRIGGER %I BEFORE UPDATE OF deleted_at ON %I.%I FOR EACH ROW EXECUTE FUNCTION soft_delete_handle_fk()',
|
|
||||||
trigger_name,
|
|
||||||
r.table_schema,
|
|
||||||
r.table_name
|
|
||||||
);
|
|
||||||
END LOOP;
|
|
||||||
END $$;
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP SEQUENCE IF EXISTS payments_code_seq;
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
CREATE SEQUENCE IF NOT EXISTS payments_code_seq START WITH 1 INCREMENT BY 1;
|
|
||||||
-3
@@ -1,3 +0,0 @@
|
|||||||
-- Rollback: restore document columns to expenses table
|
|
||||||
ALTER TABLE expenses ADD COLUMN IF NOT EXISTS document_path JSON;
|
|
||||||
ALTER TABLE expenses ADD COLUMN IF NOT EXISTS realization_document_path JSON;
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
-- Delete document columns from expenses table since we now use Document service with polymorphic relations
|
|
||||||
ALTER TABLE expenses DROP COLUMN IF EXISTS document_path;
|
|
||||||
ALTER TABLE expenses DROP COLUMN IF EXISTS realization_document_path;
|
|
||||||
-28
@@ -1,28 +0,0 @@
|
|||||||
-- ============================================
|
|
||||||
-- Rollback: Remove FIFO fields and restore qty column
|
|
||||||
-- ============================================
|
|
||||||
|
|
||||||
-- STEP 1: Drop indexes
|
|
||||||
DROP INDEX IF EXISTS idx_marketing_delivery_products_fifo_lookup;
|
|
||||||
DROP INDEX IF EXISTS idx_marketing_delivery_products_pending_qty;
|
|
||||||
DROP INDEX IF EXISTS idx_marketing_delivery_products_usage_qty;
|
|
||||||
DROP INDEX IF EXISTS idx_marketing_delivery_products_created_at;
|
|
||||||
|
|
||||||
-- STEP 2: Drop constraints
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
DROP CONSTRAINT IF EXISTS chk_marketing_delivery_products_fifo_nonneg;
|
|
||||||
|
|
||||||
-- STEP 3: Restore qty column from usage_qty data
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
ADD COLUMN IF NOT EXISTS qty NUMERIC(15, 3) DEFAULT 0 NOT NULL;
|
|
||||||
|
|
||||||
-- Migrate data back from usage_qty to qty
|
|
||||||
UPDATE marketing_delivery_products
|
|
||||||
SET qty = usage_qty
|
|
||||||
WHERE qty = 0;
|
|
||||||
|
|
||||||
-- STEP 4: Drop FIFO columns
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
DROP COLUMN IF EXISTS usage_qty,
|
|
||||||
DROP COLUMN IF EXISTS pending_qty,
|
|
||||||
DROP COLUMN IF EXISTS created_at;
|
|
||||||
-58
@@ -1,58 +0,0 @@
|
|||||||
-- ============================================
|
|
||||||
-- Add FIFO fields to marketing_delivery_products
|
|
||||||
-- This migration adds fields needed for FIFO stock management
|
|
||||||
-- and removes the old qty field in favor of FIFO-based allocation
|
|
||||||
-- ============================================
|
|
||||||
|
|
||||||
-- STEP 0: Drop orphan indexes from previous migration
|
|
||||||
DROP INDEX IF EXISTS idx_marketing_delivery_products_deleted_at;
|
|
||||||
|
|
||||||
-- STEP 1: Add created_at column (required for FIFO ordering)
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
ADD COLUMN IF NOT EXISTS created_at TIMESTAMPTZ DEFAULT NOW();
|
|
||||||
|
|
||||||
-- STEP 2: Add FIFO tracking fields
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
ADD COLUMN IF NOT EXISTS usage_qty NUMERIC(15, 3) DEFAULT 0,
|
|
||||||
ADD COLUMN IF NOT EXISTS pending_qty NUMERIC(15, 3) DEFAULT 0;
|
|
||||||
|
|
||||||
-- STEP 3: Migrate data from old qty to usage_qty for existing records
|
|
||||||
-- This preserves existing quantity data as allocated quantity
|
|
||||||
UPDATE marketing_delivery_products
|
|
||||||
SET
|
|
||||||
usage_qty = COALESCE(qty, 0),
|
|
||||||
pending_qty = 0
|
|
||||||
WHERE usage_qty = 0;
|
|
||||||
|
|
||||||
-- STEP 4: Drop the old qty column (replaced by usage_qty + pending_qty)
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
DROP COLUMN IF EXISTS qty;
|
|
||||||
|
|
||||||
-- STEP 5: Make FIFO fields NOT NULL
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
ALTER COLUMN usage_qty SET NOT NULL,
|
|
||||||
ALTER COLUMN pending_qty SET NOT NULL,
|
|
||||||
ALTER COLUMN created_at SET NOT NULL;
|
|
||||||
|
|
||||||
-- STEP 6: Add constraints to ensure non-negative values
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
ADD CONSTRAINT chk_marketing_delivery_products_fifo_nonneg CHECK (
|
|
||||||
usage_qty >= 0 AND
|
|
||||||
pending_qty >= 0
|
|
||||||
);
|
|
||||||
|
|
||||||
-- STEP 7: Create indexes for FIFO operations
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_marketing_delivery_products_created_at
|
|
||||||
ON marketing_delivery_products(created_at DESC);
|
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_marketing_delivery_products_usage_qty
|
|
||||||
ON marketing_delivery_products(usage_qty)
|
|
||||||
WHERE usage_qty > 0;
|
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_marketing_delivery_products_pending_qty
|
|
||||||
ON marketing_delivery_products(pending_qty)
|
|
||||||
WHERE pending_qty > 0;
|
|
||||||
|
|
||||||
-- Composite index for FIFO lookups
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_marketing_delivery_products_fifo_lookup
|
|
||||||
ON marketing_delivery_products(marketing_product_id, created_at DESC);
|
|
||||||
-7
@@ -1,7 +0,0 @@
|
|||||||
-- Remove foreign key constraint
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
DROP CONSTRAINT IF EXISTS fk_marketing_delivery_products_product_warehouse;
|
|
||||||
|
|
||||||
-- Drop product_warehouse_id column
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
DROP COLUMN IF EXISTS product_warehouse_id;
|
|
||||||
-19
@@ -1,19 +0,0 @@
|
|||||||
-- Add product_warehouse_id column to marketing_delivery_products
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
ADD COLUMN IF NOT EXISTS product_warehouse_id INT NOT NULL DEFAULT 0;
|
|
||||||
|
|
||||||
-- Fill product_warehouse_id from marketing_products
|
|
||||||
UPDATE marketing_delivery_products mdp
|
|
||||||
SET product_warehouse_id = mp.product_warehouse_id
|
|
||||||
FROM marketing_products mp
|
|
||||||
WHERE mdp.marketing_product_id = mp.id
|
|
||||||
AND mdp.product_warehouse_id = 0;
|
|
||||||
|
|
||||||
-- Set NOT NULL constraint
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
ALTER COLUMN product_warehouse_id SET NOT NULL;
|
|
||||||
|
|
||||||
-- Add foreign key constraint
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
ADD CONSTRAINT fk_marketing_delivery_products_product_warehouse
|
|
||||||
FOREIGN KEY (product_warehouse_id) REFERENCES product_warehouses(id);
|
|
||||||
-10
@@ -1,10 +0,0 @@
|
|||||||
-- Drop indexes
|
|
||||||
DROP INDEX IF EXISTS idx_standard_growth_details_standard_week;
|
|
||||||
DROP INDEX IF EXISTS idx_production_standard_details_standard_week;
|
|
||||||
DROP INDEX IF EXISTS idx_production_standards_project_category;
|
|
||||||
DROP INDEX IF EXISTS idx_production_standards_deleted_at;
|
|
||||||
|
|
||||||
-- Drop tables (in reverse order due to foreign keys)
|
|
||||||
DROP TABLE IF EXISTS standard_growth_details;
|
|
||||||
DROP TABLE IF EXISTS production_standard_details;
|
|
||||||
DROP TABLE IF EXISTS production_standards;
|
|
||||||
@@ -1,96 +0,0 @@
|
|||||||
-- Create production_standards table
|
|
||||||
CREATE TABLE IF NOT EXISTS production_standards (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
name VARCHAR(100) UNIQUE NOT NULL,
|
|
||||||
project_category VARCHAR(20) NOT NULL CHECK (project_category IN ('GROWING', 'LAYING')),
|
|
||||||
created_at TIMESTAMPTZ DEFAULT NOW(),
|
|
||||||
updated_at TIMESTAMPTZ DEFAULT NOW(),
|
|
||||||
deleted_at TIMESTAMPTZ,
|
|
||||||
created_by BIGINT
|
|
||||||
);
|
|
||||||
|
|
||||||
-- Create index for deleted_at (soft delete)
|
|
||||||
CREATE INDEX idx_production_standards_deleted_at ON production_standards(deleted_at);
|
|
||||||
|
|
||||||
-- Tambahkan Foreign Key ke users
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'users') THEN
|
|
||||||
ALTER TABLE production_standards
|
|
||||||
ADD CONSTRAINT fk_production_standards_created_by
|
|
||||||
FOREIGN KEY (created_by) REFERENCES users(id) ON DELETE SET NULL ON UPDATE CASCADE;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Index
|
|
||||||
CREATE INDEX idx_production_standards_created_by ON production_standards(created_by);
|
|
||||||
|
|
||||||
-- Create production_standard_details table
|
|
||||||
CREATE TABLE IF NOT EXISTS production_standard_details (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
production_standard_id BIGINT NOT NULL,
|
|
||||||
week INT NOT NULL,
|
|
||||||
target_hen_day_production NUMERIC(15, 3),
|
|
||||||
target_hen_house_production NUMERIC(15, 3),
|
|
||||||
target_egg_weight NUMERIC(15, 3),
|
|
||||||
target_egg_mass NUMERIC(15, 3),
|
|
||||||
created_at TIMESTAMPTZ DEFAULT NOW(),
|
|
||||||
updated_at TIMESTAMPTZ DEFAULT NOW()
|
|
||||||
);
|
|
||||||
|
|
||||||
-- Tambahkan Foreign Key ke production_standards
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'production_standards') THEN
|
|
||||||
ALTER TABLE production_standard_details
|
|
||||||
ADD CONSTRAINT fk_production_standard_details_standard
|
|
||||||
FOREIGN KEY (production_standard_id) REFERENCES production_standards(id) ON DELETE CASCADE;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Create unique constraint for standard_id + week
|
|
||||||
CREATE UNIQUE INDEX idx_production_standard_details_standard_week
|
|
||||||
ON production_standard_details(production_standard_id, week);
|
|
||||||
|
|
||||||
-- Create standard_growth_details table
|
|
||||||
CREATE TABLE IF NOT EXISTS standard_growth_details (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
production_standard_id BIGINT NOT NULL,
|
|
||||||
target_mean_bw NUMERIC(15, 3),
|
|
||||||
max_depletion NUMERIC(15, 3),
|
|
||||||
min_uniformity NUMERIC(15, 3) NOT NULL,
|
|
||||||
week INT NOT NULL,
|
|
||||||
feed_intake NUMERIC(15, 3),
|
|
||||||
created_at TIMESTAMPTZ DEFAULT NOW(),
|
|
||||||
created_by BIGINT
|
|
||||||
);
|
|
||||||
|
|
||||||
-- Tambahkan Foreign Key ke production_standards
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'production_standards') THEN
|
|
||||||
ALTER TABLE standard_growth_details
|
|
||||||
ADD CONSTRAINT fk_standard_growth_details_standard
|
|
||||||
FOREIGN KEY (production_standard_id) REFERENCES production_standards(id) ON DELETE CASCADE;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Tambahkan Foreign Key ke users
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'users') THEN
|
|
||||||
ALTER TABLE standard_growth_details
|
|
||||||
ADD CONSTRAINT fk_standard_growth_details_created_by
|
|
||||||
FOREIGN KEY (created_by) REFERENCES users(id) ON DELETE SET NULL ON UPDATE CASCADE;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Create unique constraint for standard_id + week
|
|
||||||
CREATE UNIQUE INDEX idx_standard_growth_details_standard_week
|
|
||||||
ON standard_growth_details(production_standard_id, week);
|
|
||||||
|
|
||||||
-- Index
|
|
||||||
CREATE INDEX idx_standard_growth_details_created_by ON standard_growth_details(created_by);
|
|
||||||
|
|
||||||
-- Create index for project_category
|
|
||||||
CREATE INDEX idx_production_standards_project_category ON production_standards(project_category);
|
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
-- Rollback: Update expense and expense_nonstocks tables
|
|
||||||
|
|
||||||
-- Drop indexes
|
|
||||||
DROP INDEX IF EXISTS idx_expenses_project_flock_id;
|
|
||||||
DROP INDEX IF EXISTS idx_expenses_location_id;
|
|
||||||
|
|
||||||
-- Drop Foreign Key constraint
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_expenses_location_id'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE expenses
|
|
||||||
DROP CONSTRAINT fk_expenses_location_id;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Drop columns from expenses table
|
|
||||||
ALTER TABLE expenses
|
|
||||||
DROP COLUMN IF EXISTS project_flock_id;
|
|
||||||
|
|
||||||
ALTER TABLE expenses
|
|
||||||
DROP COLUMN IF EXISTS location_id;
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
-- Migration: Update expense and expense_nonstocks tables
|
|
||||||
|
|
||||||
-- Add location_id column to expenses table
|
|
||||||
ALTER TABLE expenses
|
|
||||||
ADD COLUMN IF NOT EXISTS location_id BIGINT NOT NULL DEFAULT 1;
|
|
||||||
|
|
||||||
-- Add project_flock_id column to expenses table (JSON type)
|
|
||||||
ALTER TABLE expenses
|
|
||||||
ADD COLUMN IF NOT EXISTS project_flock_id JSON NULL;
|
|
||||||
|
|
||||||
-- Add Foreign Key constraint to locations table
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'locations') THEN
|
|
||||||
ALTER TABLE expenses
|
|
||||||
ADD CONSTRAINT fk_expenses_location_id
|
|
||||||
FOREIGN KEY (location_id) REFERENCES locations(id) ON DELETE RESTRICT ON UPDATE CASCADE;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Create index for location_id
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_expenses_location_id ON expenses (location_id);
|
|
||||||
|
|
||||||
-- Create index for project_flock_id
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_expenses_project_flock_id ON expenses ((project_flock_id::text));
|
|
||||||
|
|
||||||
-- Ensure kandang_id is nullable in expense_nonstocks table
|
|
||||||
ALTER TABLE expense_nonstocks
|
|
||||||
ALTER COLUMN kandang_id DROP NOT NULL;
|
|
||||||
-6
@@ -1,6 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS idx_project_flock_kandang_uniformity_deleted_at;
|
|
||||||
DROP INDEX IF EXISTS idx_project_flock_kandang_uniformity_created_by;
|
|
||||||
DROP INDEX IF EXISTS idx_project_flock_kandang_uniformity_project_flock_kandang_week;
|
|
||||||
DROP INDEX IF EXISTS idx_project_flock_kandang_uniformity_project_flock_kandang_id;
|
|
||||||
|
|
||||||
DROP TABLE IF EXISTS project_flock_kandang_uniformity;
|
|
||||||
-58
@@ -1,58 +0,0 @@
|
|||||||
CREATE TABLE IF NOT EXISTS project_flock_kandang_uniformity (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
uniformity NUMERIC(15, 3),
|
|
||||||
week INT NOT NULL,
|
|
||||||
cv NUMERIC(15, 3),
|
|
||||||
chick_qty_of_weight NUMERIC(15, 3),
|
|
||||||
mean_up NUMERIC(15, 3),
|
|
||||||
mean_down NUMERIC(15, 3),
|
|
||||||
project_flock_kandang_id BIGINT NOT NULL,
|
|
||||||
uniform_qty NUMERIC(15, 3),
|
|
||||||
not_uniform_qty NUMERIC(15, 3),
|
|
||||||
uniform_date TIMESTAMPTZ,
|
|
||||||
created_at TIMESTAMPTZ DEFAULT NOW(),
|
|
||||||
updated_at TIMESTAMPTZ DEFAULT NOW(),
|
|
||||||
deleted_at TIMESTAMPTZ,
|
|
||||||
created_by BIGINT NOT NULL
|
|
||||||
);
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'project_flock_kandangs') THEN
|
|
||||||
IF NOT EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint WHERE conname = 'fk_project_flock_kandang_uniformity_project_flock_kandang'
|
|
||||||
) THEN
|
|
||||||
EXECUTE
|
|
||||||
'ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
ADD CONSTRAINT fk_project_flock_kandang_uniformity_project_flock_kandang
|
|
||||||
FOREIGN KEY (project_flock_kandang_id)
|
|
||||||
REFERENCES project_flock_kandangs(id)
|
|
||||||
ON DELETE RESTRICT ON UPDATE CASCADE';
|
|
||||||
END IF;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'users') THEN
|
|
||||||
IF NOT EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint WHERE conname = 'fk_project_flock_kandang_uniformity_created_by'
|
|
||||||
) THEN
|
|
||||||
EXECUTE
|
|
||||||
'ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
ADD CONSTRAINT fk_project_flock_kandang_uniformity_created_by
|
|
||||||
FOREIGN KEY (created_by)
|
|
||||||
REFERENCES users(id)
|
|
||||||
ON DELETE RESTRICT ON UPDATE CASCADE';
|
|
||||||
END IF;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_project_flock_kandang_uniformity_project_flock_kandang_id
|
|
||||||
ON project_flock_kandang_uniformity (project_flock_kandang_id);
|
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_project_flock_kandang_uniformity_project_flock_kandang_week
|
|
||||||
ON project_flock_kandang_uniformity (project_flock_kandang_id, week);
|
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_project_flock_kandang_uniformity_created_by
|
|
||||||
ON project_flock_kandang_uniformity (created_by);
|
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_project_flock_kandang_uniformity_deleted_at
|
|
||||||
ON project_flock_kandang_uniformity (deleted_at);
|
|
||||||
-42
@@ -1,42 +0,0 @@
|
|||||||
-- ===============================================================
|
|
||||||
-- ROLLBACK: Remove FIFO fields from STOCK_TRANSFER_DETAILS
|
|
||||||
-- ===============================================================
|
|
||||||
|
|
||||||
-- Drop indexes
|
|
||||||
DROP INDEX IF EXISTS idx_stock_transfer_details_dest_pw;
|
|
||||||
DROP INDEX IF EXISTS idx_stock_transfer_details_source_pw;
|
|
||||||
|
|
||||||
-- Drop foreign keys
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_stock_transfer_details_source_pw'
|
|
||||||
) THEN
|
|
||||||
EXECUTE 'ALTER TABLE stock_transfer_details
|
|
||||||
DROP CONSTRAINT fk_stock_transfer_details_source_pw';
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_stock_transfer_details_dest_pw'
|
|
||||||
) THEN
|
|
||||||
EXECUTE 'ALTER TABLE stock_transfer_details
|
|
||||||
DROP CONSTRAINT fk_stock_transfer_details_dest_pw';
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Drop FIFO columns
|
|
||||||
ALTER TABLE stock_transfer_details
|
|
||||||
DROP COLUMN IF EXISTS total_used,
|
|
||||||
DROP COLUMN IF EXISTS total_qty,
|
|
||||||
DROP COLUMN IF EXISTS pending_qty,
|
|
||||||
DROP COLUMN IF EXISTS usage_qty,
|
|
||||||
DROP COLUMN IF EXISTS dest_product_warehouse_id,
|
|
||||||
DROP COLUMN IF EXISTS source_product_warehouse_id;
|
|
||||||
|
|
||||||
-- Restore original columns (in case rollback)
|
|
||||||
ALTER TABLE stock_transfer_details
|
|
||||||
ADD COLUMN IF NOT EXISTS quantity NUMERIC(15, 3) NOT NULL DEFAULT 0,
|
|
||||||
ADD COLUMN IF NOT EXISTS before_quantity NUMERIC(15, 3),
|
|
||||||
ADD COLUMN IF NOT EXISTS after_quantity NUMERIC(15, 3);
|
|
||||||
-83
@@ -1,83 +0,0 @@
|
|||||||
-- ===============================================================
|
|
||||||
-- ADD FIFO FIELDS TO STOCK_TRANSFER_DETAILS
|
|
||||||
-- Enable transfer module to work with FIFO stock system
|
|
||||||
--
|
|
||||||
-- Notes:
|
|
||||||
-- - Field 'quantity' will be removed (replaced by usage_qty + pending_qty)
|
|
||||||
-- - Fields 'before_quantity' & 'after_quantity' will be removed (unused legacy)
|
|
||||||
-- - New FIFO fields track actual allocation instead of requested quantity
|
|
||||||
-- ===============================================================
|
|
||||||
|
|
||||||
-- Add FIFO tracking fields
|
|
||||||
ALTER TABLE stock_transfer_details
|
|
||||||
ADD COLUMN IF NOT EXISTS source_product_warehouse_id BIGINT,
|
|
||||||
ADD COLUMN IF NOT EXISTS dest_product_warehouse_id BIGINT,
|
|
||||||
ADD COLUMN IF NOT EXISTS usage_qty NUMERIC(15, 3) DEFAULT 0,
|
|
||||||
ADD COLUMN IF NOT EXISTS pending_qty NUMERIC(15, 3) DEFAULT 0,
|
|
||||||
ADD COLUMN IF NOT EXISTS total_qty NUMERIC(15, 3) DEFAULT 0,
|
|
||||||
ADD COLUMN IF NOT EXISTS total_used NUMERIC(15, 3) DEFAULT 0;
|
|
||||||
|
|
||||||
-- Remove obsolete columns (quantity replaced by FIFO fields, legacy fields never used)
|
|
||||||
ALTER TABLE stock_transfer_details
|
|
||||||
DROP COLUMN IF EXISTS quantity,
|
|
||||||
DROP COLUMN IF EXISTS before_quantity,
|
|
||||||
DROP COLUMN IF EXISTS after_quantity;
|
|
||||||
|
|
||||||
-- Add foreign keys for product warehouse references
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'product_warehouses') THEN
|
|
||||||
-- Source warehouse foreign key
|
|
||||||
IF NOT EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_stock_transfer_details_source_pw'
|
|
||||||
) THEN
|
|
||||||
EXECUTE
|
|
||||||
'ALTER TABLE stock_transfer_details
|
|
||||||
ADD CONSTRAINT fk_stock_transfer_details_source_pw
|
|
||||||
FOREIGN KEY (source_product_warehouse_id)
|
|
||||||
REFERENCES product_warehouses(id)
|
|
||||||
ON DELETE SET NULL ON UPDATE CASCADE';
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
-- Destination warehouse foreign key
|
|
||||||
IF NOT EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_stock_transfer_details_dest_pw'
|
|
||||||
) THEN
|
|
||||||
EXECUTE
|
|
||||||
'ALTER TABLE stock_transfer_details
|
|
||||||
ADD CONSTRAINT fk_stock_transfer_details_dest_pw
|
|
||||||
FOREIGN KEY (dest_product_warehouse_id)
|
|
||||||
REFERENCES product_warehouses(id)
|
|
||||||
ON DELETE SET NULL ON UPDATE CASCADE';
|
|
||||||
END IF;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Add indexes for FIFO operations
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_stock_transfer_details_source_pw
|
|
||||||
ON stock_transfer_details (source_product_warehouse_id);
|
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_stock_transfer_details_dest_pw
|
|
||||||
ON stock_transfer_details (dest_product_warehouse_id);
|
|
||||||
|
|
||||||
-- Add comments for documentation
|
|
||||||
COMMENT ON COLUMN stock_transfer_details.source_product_warehouse_id IS
|
|
||||||
'Source product warehouse ID - referensi warehouse asal (FIFO usable)';
|
|
||||||
|
|
||||||
COMMENT ON COLUMN stock_transfer_details.dest_product_warehouse_id IS
|
|
||||||
'Destination product warehouse ID - referensi warehouse tujuan (FIFO stockable)';
|
|
||||||
|
|
||||||
COMMENT ON COLUMN stock_transfer_details.usage_qty IS
|
|
||||||
'Actual quantity successfully taken from source warehouse (FIFO usable tracking) - replaces quantity field';
|
|
||||||
|
|
||||||
COMMENT ON COLUMN stock_transfer_details.pending_qty IS
|
|
||||||
'Quantity waiting for stock availability (FIFO usable tracking)';
|
|
||||||
|
|
||||||
COMMENT ON COLUMN stock_transfer_details.total_qty IS
|
|
||||||
'Total lot quantity available at destination warehouse (FIFO stockable tracking)';
|
|
||||||
|
|
||||||
COMMENT ON COLUMN stock_transfer_details.total_used IS
|
|
||||||
'Quantity already consumed from this lot at destination warehouse (FIFO stockable tracking)';
|
|
||||||
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
-- Rollback: Drop adjustment_stocks table
|
|
||||||
|
|
||||||
BEGIN;
|
|
||||||
|
|
||||||
DROP INDEX IF EXISTS idx_adjustment_stocks_product_warehouse;
|
|
||||||
DROP INDEX IF EXISTS idx_adjustment_stocks_stock_log;
|
|
||||||
|
|
||||||
ALTER TABLE adjustment_stocks
|
|
||||||
DROP CONSTRAINT IF EXISTS fk_adjustment_stocks_product_warehouse;
|
|
||||||
|
|
||||||
ALTER TABLE adjustment_stocks
|
|
||||||
DROP CONSTRAINT IF EXISTS fk_adjustment_stocks_stock_log;
|
|
||||||
|
|
||||||
DROP TABLE IF EXISTS adjustment_stocks;
|
|
||||||
|
|
||||||
COMMIT;
|
|
||||||
@@ -1,40 +0,0 @@
|
|||||||
-- Migration: Create adjustment_stocks table for FIFO tracking
|
|
||||||
-- This table tracks FIFO allocation for stock adjustments (both increase and decrease)
|
|
||||||
|
|
||||||
BEGIN;
|
|
||||||
|
|
||||||
CREATE TABLE IF NOT EXISTS adjustment_stocks (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
stock_log_id BIGINT NOT NULL,
|
|
||||||
product_warehouse_id BIGINT NOT NULL,
|
|
||||||
|
|
||||||
-- FIFO fields for Adjustment INCREASE (Stockable)
|
|
||||||
-- Tracks stock added to warehouse via adjustment
|
|
||||||
total_qty NUMERIC(15, 3) DEFAULT 0,
|
|
||||||
total_used NUMERIC(15, 3) DEFAULT 0,
|
|
||||||
|
|
||||||
-- FIFO fields for Adjustment DECREASE (Usable)
|
|
||||||
-- Tracks stock consumed from warehouse via adjustment
|
|
||||||
usage_qty NUMERIC(15, 3) DEFAULT 0,
|
|
||||||
pending_qty NUMERIC(15, 3) DEFAULT 0,
|
|
||||||
|
|
||||||
created_at TIMESTAMPTZ DEFAULT now(),
|
|
||||||
updated_at TIMESTAMPTZ DEFAULT now()
|
|
||||||
);
|
|
||||||
|
|
||||||
-- Foreign keys
|
|
||||||
ALTER TABLE adjustment_stocks
|
|
||||||
ADD CONSTRAINT fk_adjustment_stocks_stock_log
|
|
||||||
FOREIGN KEY (stock_log_id) REFERENCES stock_logs(id)
|
|
||||||
ON DELETE CASCADE ON UPDATE CASCADE;
|
|
||||||
|
|
||||||
ALTER TABLE adjustment_stocks
|
|
||||||
ADD CONSTRAINT fk_adjustment_stocks_product_warehouse
|
|
||||||
FOREIGN KEY (product_warehouse_id) REFERENCES product_warehouses(id)
|
|
||||||
ON DELETE CASCADE ON UPDATE CASCADE;
|
|
||||||
|
|
||||||
-- Indexes
|
|
||||||
CREATE INDEX idx_adjustment_stocks_stock_log ON adjustment_stocks(stock_log_id);
|
|
||||||
CREATE INDEX idx_adjustment_stocks_product_warehouse ON adjustment_stocks(product_warehouse_id);
|
|
||||||
|
|
||||||
COMMIT;
|
|
||||||
-54
@@ -1,54 +0,0 @@
|
|||||||
BEGIN;
|
|
||||||
|
|
||||||
CREATE TABLE IF NOT EXISTS recording_bws (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
recording_id BIGINT NOT NULL,
|
|
||||||
avg_weight NUMERIC(8,2) NOT NULL,
|
|
||||||
qty NUMERIC(15,3) NOT NULL DEFAULT 1,
|
|
||||||
total_weight NUMERIC(10,3) NOT NULL,
|
|
||||||
created_at TIMESTAMPTZ DEFAULT NOW(),
|
|
||||||
updated_at TIMESTAMPTZ DEFAULT NOW(),
|
|
||||||
|
|
||||||
CONSTRAINT fk_recording_bws_recording
|
|
||||||
FOREIGN KEY (recording_id) REFERENCES recordings(id) ON DELETE CASCADE,
|
|
||||||
|
|
||||||
CONSTRAINT chk_recording_bws_nonneg
|
|
||||||
CHECK (avg_weight >= 0 AND qty >= 0 AND total_weight >= 0)
|
|
||||||
);
|
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_recording_bws_recording
|
|
||||||
ON recording_bws (recording_id);
|
|
||||||
|
|
||||||
ALTER TABLE recordings
|
|
||||||
DROP CONSTRAINT IF EXISTS chk_recordings_nonnegatives_v3;
|
|
||||||
|
|
||||||
ALTER TABLE recordings
|
|
||||||
DROP COLUMN IF EXISTS hand_day,
|
|
||||||
DROP COLUMN IF EXISTS hand_house,
|
|
||||||
DROP COLUMN IF EXISTS feed_intake,
|
|
||||||
DROP COLUMN IF EXISTS egg_mesh,
|
|
||||||
DROP COLUMN IF EXISTS egg_weight;
|
|
||||||
|
|
||||||
ALTER TABLE recordings
|
|
||||||
ADD CONSTRAINT chk_recordings_nonnegatives_v2 CHECK (
|
|
||||||
(total_depletion_qty IS NULL OR total_depletion_qty >= 0) AND
|
|
||||||
(cum_depletion_rate IS NULL OR cum_depletion_rate >= 0) AND
|
|
||||||
(daily_gain IS NULL OR daily_gain >= 0) AND
|
|
||||||
(avg_daily_gain IS NULL OR avg_daily_gain >= 0) AND
|
|
||||||
(cum_intake IS NULL OR cum_intake >= 0) AND
|
|
||||||
(fcr_value IS NULL OR fcr_value >= 0) AND
|
|
||||||
(total_chick_qty IS NULL OR total_chick_qty >= 0)
|
|
||||||
);
|
|
||||||
|
|
||||||
ALTER TABLE recording_eggs
|
|
||||||
DROP CONSTRAINT IF EXISTS chk_recording_eggs_qty;
|
|
||||||
|
|
||||||
ALTER TABLE recording_eggs
|
|
||||||
ALTER COLUMN weight TYPE NUMERIC(10,3) USING weight::NUMERIC(10,3);
|
|
||||||
|
|
||||||
ALTER TABLE recording_eggs
|
|
||||||
ADD CONSTRAINT chk_recording_eggs_qty CHECK (
|
|
||||||
qty >= 0 AND (weight IS NULL OR weight >= 0)
|
|
||||||
);
|
|
||||||
|
|
||||||
COMMIT;
|
|
||||||
-44
@@ -1,44 +0,0 @@
|
|||||||
BEGIN;
|
|
||||||
|
|
||||||
ALTER TABLE recordings
|
|
||||||
DROP CONSTRAINT IF EXISTS chk_recordings_nonnegatives_v2;
|
|
||||||
|
|
||||||
ALTER TABLE recordings
|
|
||||||
ADD COLUMN IF NOT EXISTS hand_day NUMERIC(15,3),
|
|
||||||
ADD COLUMN IF NOT EXISTS hand_house NUMERIC(15,3),
|
|
||||||
ADD COLUMN IF NOT EXISTS feed_intake NUMERIC(15,3),
|
|
||||||
ADD COLUMN IF NOT EXISTS egg_mesh NUMERIC(15,3),
|
|
||||||
ADD COLUMN IF NOT EXISTS egg_weight NUMERIC(15,3);
|
|
||||||
|
|
||||||
ALTER TABLE recordings
|
|
||||||
ADD CONSTRAINT chk_recordings_nonnegatives_v3 CHECK (
|
|
||||||
(total_depletion_qty IS NULL OR total_depletion_qty >= 0) AND
|
|
||||||
(cum_depletion_rate IS NULL OR cum_depletion_rate >= 0) AND
|
|
||||||
(daily_gain IS NULL OR daily_gain >= 0) AND
|
|
||||||
(avg_daily_gain IS NULL OR avg_daily_gain >= 0) AND
|
|
||||||
(cum_intake IS NULL OR cum_intake >= 0) AND
|
|
||||||
(fcr_value IS NULL OR fcr_value >= 0) AND
|
|
||||||
(total_chick_qty IS NULL OR total_chick_qty >= 0) AND
|
|
||||||
(hand_day IS NULL OR hand_day >= 0) AND
|
|
||||||
(hand_house IS NULL OR hand_house >= 0) AND
|
|
||||||
(feed_intake IS NULL OR feed_intake >= 0) AND
|
|
||||||
(egg_mesh IS NULL OR egg_mesh >= 0) AND
|
|
||||||
(egg_weight IS NULL OR egg_weight >= 0)
|
|
||||||
);
|
|
||||||
|
|
||||||
ALTER TABLE recording_eggs
|
|
||||||
ALTER COLUMN weight TYPE NUMERIC(15,3) USING weight::NUMERIC(15,3);
|
|
||||||
|
|
||||||
ALTER TABLE recording_eggs
|
|
||||||
DROP CONSTRAINT IF EXISTS chk_recording_eggs_qty;
|
|
||||||
|
|
||||||
ALTER TABLE recording_eggs
|
|
||||||
ADD CONSTRAINT chk_recording_eggs_qty CHECK (
|
|
||||||
qty >= 0 AND
|
|
||||||
(weight IS NULL OR weight >= 0)
|
|
||||||
);
|
|
||||||
|
|
||||||
DROP INDEX IF EXISTS idx_recording_bws_recording;
|
|
||||||
DROP TABLE IF EXISTS recording_bws;
|
|
||||||
|
|
||||||
COMMIT;
|
|
||||||
-7
@@ -1,7 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS idx_project_flocks_production_standard_id;
|
|
||||||
|
|
||||||
ALTER TABLE project_flocks
|
|
||||||
DROP CONSTRAINT IF EXISTS fk_project_flocks_production_standard_id;
|
|
||||||
|
|
||||||
ALTER TABLE project_flocks
|
|
||||||
DROP COLUMN IF EXISTS production_standard_id;
|
|
||||||
-15
@@ -1,15 +0,0 @@
|
|||||||
-- Add production_standard_id to project_flocks
|
|
||||||
ALTER TABLE project_flocks
|
|
||||||
ADD COLUMN IF NOT EXISTS production_standard_id BIGINT;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'production_standards') THEN
|
|
||||||
ALTER TABLE project_flocks
|
|
||||||
ADD CONSTRAINT fk_project_flocks_production_standard_id
|
|
||||||
FOREIGN KEY (production_standard_id) REFERENCES production_standards (id) ON DELETE RESTRICT ON UPDATE CASCADE;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_project_flocks_production_standard_id
|
|
||||||
ON project_flocks (production_standard_id);
|
|
||||||
-3
@@ -1,3 +0,0 @@
|
|||||||
-- Remove standard_fcr column from production_standard_details table
|
|
||||||
ALTER TABLE production_standard_details
|
|
||||||
DROP COLUMN IF EXISTS standard_fcr;
|
|
||||||
-3
@@ -1,3 +0,0 @@
|
|||||||
-- Add standard_fcr column to production_standard_details table
|
|
||||||
ALTER TABLE production_standard_details
|
|
||||||
ADD COLUMN standard_fcr NUMERIC(15, 3);
|
|
||||||
-20
@@ -1,20 +0,0 @@
|
|||||||
-- Drop CASCADE constraint
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_project_chickins_kandang'
|
|
||||||
AND conrelid = 'project_chickins'::regclass
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE project_chickins
|
|
||||||
DROP CONSTRAINT fk_project_chickins_kandang;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Recreate foreign key constraint with RESTRICT (original behavior)
|
|
||||||
ALTER TABLE project_chickins
|
|
||||||
ADD CONSTRAINT fk_project_chickins_kandang
|
|
||||||
FOREIGN KEY (project_flock_kandang_id)
|
|
||||||
REFERENCES project_flock_kandangs(id)
|
|
||||||
ON DELETE RESTRICT ON UPDATE CASCADE;
|
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
-- Drop existing foreign key constraint with RESTRICT
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_project_chickins_kandang'
|
|
||||||
AND conrelid = 'project_chickins'::regclass
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE project_chickins
|
|
||||||
DROP CONSTRAINT fk_project_chickins_kandang;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Add new foreign key constraint with CASCADE delete
|
|
||||||
ALTER TABLE project_chickins
|
|
||||||
ADD CONSTRAINT fk_project_chickins_kandang
|
|
||||||
FOREIGN KEY (project_flock_kandang_id)
|
|
||||||
REFERENCES project_flock_kandangs(id)
|
|
||||||
ON DELETE CASCADE ON UPDATE CASCADE;
|
|
||||||
@@ -1,126 +0,0 @@
|
|||||||
CREATE OR REPLACE FUNCTION soft_delete_handle_fk() RETURNS TRIGGER AS $$
|
|
||||||
DECLARE
|
|
||||||
fk record;
|
|
||||||
child_column text;
|
|
||||||
parent_column text;
|
|
||||||
parent_value text;
|
|
||||||
child_has_deleted_at boolean;
|
|
||||||
ref_exists boolean;
|
|
||||||
sql text;
|
|
||||||
BEGIN
|
|
||||||
IF OLD.deleted_at IS NULL AND NEW.deleted_at IS NOT NULL THEN
|
|
||||||
FOR fk IN
|
|
||||||
SELECT conrelid::regclass AS child_table,
|
|
||||||
conkey AS child_cols,
|
|
||||||
confkey AS parent_cols,
|
|
||||||
confdeltype
|
|
||||||
FROM pg_constraint
|
|
||||||
WHERE contype = 'f'
|
|
||||||
AND confrelid = TG_RELID
|
|
||||||
LOOP
|
|
||||||
IF array_length(fk.child_cols, 1) IS DISTINCT FROM 1
|
|
||||||
OR array_length(fk.parent_cols, 1) IS DISTINCT FROM 1 THEN
|
|
||||||
RAISE NOTICE 'soft_delete_handle_fk skipped composite fk on %', fk.child_table;
|
|
||||||
CONTINUE;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
SELECT attname INTO child_column
|
|
||||||
FROM pg_attribute
|
|
||||||
WHERE attrelid = fk.child_table
|
|
||||||
AND attnum = fk.child_cols[1]
|
|
||||||
AND NOT attisdropped;
|
|
||||||
|
|
||||||
SELECT attname INTO parent_column
|
|
||||||
FROM pg_attribute
|
|
||||||
WHERE attrelid = TG_RELID
|
|
||||||
AND attnum = fk.parent_cols[1]
|
|
||||||
AND NOT attisdropped;
|
|
||||||
|
|
||||||
EXECUTE format('SELECT ($1).%I', parent_column)
|
|
||||||
INTO parent_value
|
|
||||||
USING OLD;
|
|
||||||
|
|
||||||
SELECT EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_attribute
|
|
||||||
WHERE attrelid = fk.child_table
|
|
||||||
AND attname = 'deleted_at'
|
|
||||||
AND NOT attisdropped
|
|
||||||
) INTO child_has_deleted_at;
|
|
||||||
|
|
||||||
IF fk.confdeltype IN ('r', 'a') THEN
|
|
||||||
sql := format(
|
|
||||||
'SELECT EXISTS (SELECT 1 FROM %s WHERE %I = $1 %s)',
|
|
||||||
fk.child_table,
|
|
||||||
child_column,
|
|
||||||
CASE WHEN child_has_deleted_at THEN 'AND deleted_at IS NULL' ELSE '' END
|
|
||||||
);
|
|
||||||
EXECUTE sql INTO ref_exists USING parent_value;
|
|
||||||
IF ref_exists THEN
|
|
||||||
RAISE EXCEPTION 'Cannot soft delete %, still referenced by %',
|
|
||||||
TG_TABLE_NAME, fk.child_table;
|
|
||||||
END IF;
|
|
||||||
ELSIF fk.confdeltype = 'n' THEN
|
|
||||||
sql := format(
|
|
||||||
'UPDATE %s SET %I = NULL WHERE %I = $1 %s',
|
|
||||||
fk.child_table,
|
|
||||||
child_column,
|
|
||||||
child_column,
|
|
||||||
CASE WHEN child_has_deleted_at THEN 'AND deleted_at IS NULL' ELSE '' END
|
|
||||||
);
|
|
||||||
EXECUTE sql USING parent_value;
|
|
||||||
ELSIF fk.confdeltype = 'c' THEN
|
|
||||||
IF child_has_deleted_at THEN
|
|
||||||
sql := format(
|
|
||||||
'UPDATE %s SET deleted_at = NOW() WHERE %I = $1 AND deleted_at IS NULL',
|
|
||||||
fk.child_table,
|
|
||||||
child_column
|
|
||||||
);
|
|
||||||
EXECUTE sql USING parent_value;
|
|
||||||
ELSE
|
|
||||||
sql := format(
|
|
||||||
'DELETE FROM %s WHERE %I = $1',
|
|
||||||
fk.child_table,
|
|
||||||
child_column
|
|
||||||
);
|
|
||||||
EXECUTE sql USING parent_value;
|
|
||||||
END IF;
|
|
||||||
ELSIF fk.confdeltype = 'd' THEN
|
|
||||||
sql := format(
|
|
||||||
'UPDATE %s SET %I = DEFAULT WHERE %I = $1 %s',
|
|
||||||
fk.child_table,
|
|
||||||
child_column,
|
|
||||||
child_column,
|
|
||||||
CASE WHEN child_has_deleted_at THEN 'AND deleted_at IS NULL' ELSE '' END
|
|
||||||
);
|
|
||||||
EXECUTE sql USING parent_value;
|
|
||||||
END IF;
|
|
||||||
END LOOP;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
RETURN NEW;
|
|
||||||
END;
|
|
||||||
$$ LANGUAGE plpgsql;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
DECLARE
|
|
||||||
r record;
|
|
||||||
trigger_name text;
|
|
||||||
BEGIN
|
|
||||||
FOR r IN
|
|
||||||
SELECT table_schema, table_name
|
|
||||||
FROM information_schema.columns
|
|
||||||
WHERE column_name = 'deleted_at'
|
|
||||||
AND table_schema = 'public'
|
|
||||||
GROUP BY table_schema, table_name
|
|
||||||
LOOP
|
|
||||||
trigger_name := format('trg_soft_delete_fk_%s', r.table_name);
|
|
||||||
EXECUTE format('DROP TRIGGER IF EXISTS %I ON %I.%I', trigger_name, r.table_schema, r.table_name);
|
|
||||||
EXECUTE format(
|
|
||||||
'CREATE TRIGGER %I BEFORE UPDATE OF deleted_at ON %I.%I FOR EACH ROW EXECUTE FUNCTION soft_delete_handle_fk()',
|
|
||||||
trigger_name,
|
|
||||||
r.table_schema,
|
|
||||||
r.table_name
|
|
||||||
);
|
|
||||||
END LOOP;
|
|
||||||
END $$;
|
|
||||||
@@ -1,142 +0,0 @@
|
|||||||
CREATE OR REPLACE FUNCTION soft_delete_handle_fk() RETURNS TRIGGER AS $$
|
|
||||||
DECLARE
|
|
||||||
fk record;
|
|
||||||
child_column text;
|
|
||||||
parent_column text;
|
|
||||||
parent_value text;
|
|
||||||
child_has_deleted_at boolean;
|
|
||||||
ref_exists boolean;
|
|
||||||
sql text;
|
|
||||||
child_type text;
|
|
||||||
BEGIN
|
|
||||||
IF OLD.deleted_at IS NULL AND NEW.deleted_at IS NOT NULL THEN
|
|
||||||
FOR fk IN
|
|
||||||
SELECT conrelid::regclass AS child_table,
|
|
||||||
conkey AS child_cols,
|
|
||||||
confkey AS parent_cols,
|
|
||||||
confdeltype
|
|
||||||
FROM pg_constraint
|
|
||||||
WHERE contype = 'f'
|
|
||||||
AND confrelid = TG_RELID
|
|
||||||
LOOP
|
|
||||||
IF array_length(fk.child_cols, 1) IS DISTINCT FROM 1
|
|
||||||
OR array_length(fk.parent_cols, 1) IS DISTINCT FROM 1 THEN
|
|
||||||
RAISE NOTICE 'soft_delete_handle_fk skipped composite fk on %', fk.child_table;
|
|
||||||
CONTINUE;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
SELECT attname INTO child_column
|
|
||||||
FROM pg_attribute
|
|
||||||
WHERE attrelid = fk.child_table
|
|
||||||
AND attnum = fk.child_cols[1]
|
|
||||||
AND NOT attisdropped;
|
|
||||||
|
|
||||||
SELECT attname INTO parent_column
|
|
||||||
FROM pg_attribute
|
|
||||||
WHERE attrelid = TG_RELID
|
|
||||||
AND attnum = fk.parent_cols[1]
|
|
||||||
AND NOT attisdropped;
|
|
||||||
|
|
||||||
SELECT format_type(atttypid, atttypmod) INTO child_type
|
|
||||||
FROM pg_attribute
|
|
||||||
WHERE attrelid = fk.child_table
|
|
||||||
AND attname = child_column
|
|
||||||
AND NOT attisdropped;
|
|
||||||
|
|
||||||
IF child_type IS NULL THEN
|
|
||||||
child_type := 'text';
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
EXECUTE format('SELECT ($1).%I', parent_column)
|
|
||||||
INTO parent_value
|
|
||||||
USING OLD;
|
|
||||||
|
|
||||||
SELECT EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_attribute
|
|
||||||
WHERE attrelid = fk.child_table
|
|
||||||
AND attname = 'deleted_at'
|
|
||||||
AND NOT attisdropped
|
|
||||||
) INTO child_has_deleted_at;
|
|
||||||
|
|
||||||
IF fk.confdeltype IN ('r', 'a') THEN
|
|
||||||
sql := format(
|
|
||||||
'SELECT EXISTS (SELECT 1 FROM %s WHERE %I = $1::%s %s)',
|
|
||||||
fk.child_table,
|
|
||||||
child_column,
|
|
||||||
child_type,
|
|
||||||
CASE WHEN child_has_deleted_at THEN 'AND deleted_at IS NULL' ELSE '' END
|
|
||||||
);
|
|
||||||
EXECUTE sql INTO ref_exists USING parent_value;
|
|
||||||
IF ref_exists THEN
|
|
||||||
RAISE EXCEPTION 'Cannot soft delete %, still referenced by %',
|
|
||||||
TG_TABLE_NAME, fk.child_table;
|
|
||||||
END IF;
|
|
||||||
ELSIF fk.confdeltype = 'n' THEN
|
|
||||||
sql := format(
|
|
||||||
'UPDATE %s SET %I = NULL WHERE %I = $1::%s %s',
|
|
||||||
fk.child_table,
|
|
||||||
child_column,
|
|
||||||
child_column,
|
|
||||||
child_type,
|
|
||||||
CASE WHEN child_has_deleted_at THEN 'AND deleted_at IS NULL' ELSE '' END
|
|
||||||
);
|
|
||||||
EXECUTE sql USING parent_value;
|
|
||||||
ELSIF fk.confdeltype = 'c' THEN
|
|
||||||
IF child_has_deleted_at THEN
|
|
||||||
sql := format(
|
|
||||||
'UPDATE %s SET deleted_at = NOW() WHERE %I = $1::%s AND deleted_at IS NULL',
|
|
||||||
fk.child_table,
|
|
||||||
child_column,
|
|
||||||
child_type
|
|
||||||
);
|
|
||||||
EXECUTE sql USING parent_value;
|
|
||||||
ELSE
|
|
||||||
sql := format(
|
|
||||||
'DELETE FROM %s WHERE %I = $1::%s',
|
|
||||||
fk.child_table,
|
|
||||||
child_column,
|
|
||||||
child_type
|
|
||||||
);
|
|
||||||
EXECUTE sql USING parent_value;
|
|
||||||
END IF;
|
|
||||||
ELSIF fk.confdeltype = 'd' THEN
|
|
||||||
sql := format(
|
|
||||||
'UPDATE %s SET %I = DEFAULT WHERE %I = $1::%s %s',
|
|
||||||
fk.child_table,
|
|
||||||
child_column,
|
|
||||||
child_column,
|
|
||||||
child_type,
|
|
||||||
CASE WHEN child_has_deleted_at THEN 'AND deleted_at IS NULL' ELSE '' END
|
|
||||||
);
|
|
||||||
EXECUTE sql USING parent_value;
|
|
||||||
END IF;
|
|
||||||
END LOOP;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
RETURN NEW;
|
|
||||||
END;
|
|
||||||
$$ LANGUAGE plpgsql;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
DECLARE
|
|
||||||
r record;
|
|
||||||
trigger_name text;
|
|
||||||
BEGIN
|
|
||||||
FOR r IN
|
|
||||||
SELECT table_schema, table_name
|
|
||||||
FROM information_schema.columns
|
|
||||||
WHERE column_name = 'deleted_at'
|
|
||||||
AND table_schema = 'public'
|
|
||||||
GROUP BY table_schema, table_name
|
|
||||||
LOOP
|
|
||||||
trigger_name := format('trg_soft_delete_fk_%s', r.table_name);
|
|
||||||
EXECUTE format('DROP TRIGGER IF EXISTS %I ON %I.%I', trigger_name, r.table_schema, r.table_name);
|
|
||||||
EXECUTE format(
|
|
||||||
'CREATE TRIGGER %I BEFORE UPDATE OF deleted_at ON %I.%I FOR EACH ROW EXECUTE FUNCTION soft_delete_handle_fk()',
|
|
||||||
trigger_name,
|
|
||||||
r.table_schema,
|
|
||||||
r.table_name
|
|
||||||
);
|
|
||||||
END LOOP;
|
|
||||||
END $$;
|
|
||||||
-86
@@ -1,86 +0,0 @@
|
|||||||
BEGIN;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_project_flock_kandang_uniformity_project_flock_kandang'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
DROP CONSTRAINT fk_project_flock_kandang_uniformity_project_flock_kandang;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
ADD CONSTRAINT fk_project_flock_kandang_uniformity_project_flock_kandang
|
|
||||||
FOREIGN KEY (project_flock_kandang_id)
|
|
||||||
REFERENCES project_flock_kandangs (id)
|
|
||||||
ON DELETE RESTRICT ON UPDATE CASCADE;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_tables
|
|
||||||
WHERE tablename = 'project_budgets'
|
|
||||||
) THEN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_project_budgets_project_flock_id'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE project_budgets
|
|
||||||
DROP CONSTRAINT fk_project_budgets_project_flock_id;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
ALTER TABLE project_budgets
|
|
||||||
ADD CONSTRAINT fk_project_budgets_project_flock_id
|
|
||||||
FOREIGN KEY (project_flock_id)
|
|
||||||
REFERENCES project_flocks(id);
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_tables
|
|
||||||
WHERE tablename = 'project_flock_kandang_uniformity'
|
|
||||||
) THEN
|
|
||||||
IF NOT EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM information_schema.columns
|
|
||||||
WHERE table_schema = 'public'
|
|
||||||
AND table_name = 'project_flock_kandang_uniformity'
|
|
||||||
AND column_name = 'created_at'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
ADD COLUMN created_at TIMESTAMPTZ DEFAULT NOW();
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
IF NOT EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM information_schema.columns
|
|
||||||
WHERE table_schema = 'public'
|
|
||||||
AND table_name = 'project_flock_kandang_uniformity'
|
|
||||||
AND column_name = 'updated_at'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
ADD COLUMN updated_at TIMESTAMPTZ DEFAULT NOW();
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
IF NOT EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM information_schema.columns
|
|
||||||
WHERE table_schema = 'public'
|
|
||||||
AND table_name = 'project_flock_kandang_uniformity'
|
|
||||||
AND column_name = 'deleted_at'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
ADD COLUMN deleted_at TIMESTAMPTZ;
|
|
||||||
END IF;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
COMMIT;
|
|
||||||
-90
@@ -1,90 +0,0 @@
|
|||||||
BEGIN;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_project_flock_kandang_uniformity_project_flock_kandang'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
DROP CONSTRAINT fk_project_flock_kandang_uniformity_project_flock_kandang;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
ADD CONSTRAINT fk_project_flock_kandang_uniformity_project_flock_kandang
|
|
||||||
FOREIGN KEY (project_flock_kandang_id)
|
|
||||||
REFERENCES project_flock_kandangs (id)
|
|
||||||
ON DELETE CASCADE ON UPDATE CASCADE;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_tables
|
|
||||||
WHERE tablename = 'project_budgets'
|
|
||||||
) THEN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_project_budgets_project_flock_id'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE project_budgets
|
|
||||||
DROP CONSTRAINT fk_project_budgets_project_flock_id;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
ALTER TABLE project_budgets
|
|
||||||
ADD CONSTRAINT fk_project_budgets_project_flock_id
|
|
||||||
FOREIGN KEY (project_flock_id)
|
|
||||||
REFERENCES project_flocks(id)
|
|
||||||
ON DELETE CASCADE ON UPDATE CASCADE;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_trigger
|
|
||||||
WHERE tgname = 'trg_soft_delete_fk_project_flock_kandang_uniformity'
|
|
||||||
) THEN
|
|
||||||
DROP TRIGGER trg_soft_delete_fk_project_flock_kandang_uniformity
|
|
||||||
ON project_flock_kandang_uniformity;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM information_schema.columns
|
|
||||||
WHERE table_schema = 'public'
|
|
||||||
AND table_name = 'project_flock_kandang_uniformity'
|
|
||||||
AND column_name = 'created_at'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
DROP COLUMN created_at;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM information_schema.columns
|
|
||||||
WHERE table_schema = 'public'
|
|
||||||
AND table_name = 'project_flock_kandang_uniformity'
|
|
||||||
AND column_name = 'updated_at'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
DROP COLUMN updated_at;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM information_schema.columns
|
|
||||||
WHERE table_schema = 'public'
|
|
||||||
AND table_name = 'project_flock_kandang_uniformity'
|
|
||||||
AND column_name = 'deleted_at'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
DROP COLUMN deleted_at;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
COMMIT;
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
-- Drop tables in correct order (child tables before parent tables)
|
|
||||||
DROP TABLE IF EXISTS daily_checklist_activity_task_assignments; -- Child table with FK to daily_checklist_activity_tasks
|
|
||||||
DROP TABLE IF EXISTS daily_checklist_activity_task_assignees;
|
|
||||||
DROP TABLE IF EXISTS daily_checklist_activity_tasks;
|
|
||||||
DROP TABLE IF EXISTS daily_checklist_tasks;
|
|
||||||
DROP TABLE IF EXISTS daily_checklist_phases;
|
|
||||||
DROP TABLE IF EXISTS daily_checklists;
|
|
||||||
DROP TABLE IF EXISTS checklists;
|
|
||||||
DROP TABLE IF EXISTS phase_activities;
|
|
||||||
DROP TABLE IF EXISTS phases;
|
|
||||||
DROP TABLE IF EXISTS employee_kandangs;
|
|
||||||
DROP TABLE IF EXISTS employees;
|
|
||||||
|
|
||||||
DROP TYPE IF EXISTS category_code;
|
|
||||||
@@ -1,194 +0,0 @@
|
|||||||
CREATE TYPE category_code AS ENUM (
|
|
||||||
'pullet_open',
|
|
||||||
'pullet_close',
|
|
||||||
'produksi_open',
|
|
||||||
'produksi_close'
|
|
||||||
);
|
|
||||||
|
|
||||||
-- MASTER TABLES
|
|
||||||
|
|
||||||
CREATE TABLE employees (
|
|
||||||
id bigint GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
|
||||||
name varchar NOT NULL,
|
|
||||||
is_active boolean NOT NULL DEFAULT true,
|
|
||||||
created_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
updated_at timestamptz NOT NULL DEFAULT now()
|
|
||||||
);
|
|
||||||
|
|
||||||
CREATE TABLE employee_kandangs (
|
|
||||||
id bigint GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
|
||||||
employee_id bigint NOT NULL,
|
|
||||||
kandang_id bigint NOT NULL,
|
|
||||||
created_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
updated_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
|
|
||||||
CONSTRAINT fk_employee_kandangs_employee
|
|
||||||
FOREIGN KEY (employee_id) REFERENCES employees(id)
|
|
||||||
ON DELETE CASCADE,
|
|
||||||
|
|
||||||
CONSTRAINT fk_employee_kandangs_kandang
|
|
||||||
FOREIGN KEY (kandang_id) REFERENCES kandangs(id)
|
|
||||||
ON DELETE CASCADE,
|
|
||||||
|
|
||||||
CONSTRAINT uq_employee_kandangs UNIQUE (employee_id, kandang_id)
|
|
||||||
);
|
|
||||||
|
|
||||||
-- PHASE & CHECKLIST
|
|
||||||
|
|
||||||
CREATE TABLE phases (
|
|
||||||
id bigint GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
|
||||||
name varchar NOT NULL,
|
|
||||||
is_active boolean NOT NULL DEFAULT true,
|
|
||||||
category category_code NOT NULL,
|
|
||||||
created_at timestamptz NOT NULL DEFAULT now()
|
|
||||||
);
|
|
||||||
|
|
||||||
CREATE TABLE phase_activities (
|
|
||||||
id bigint GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
|
||||||
phase_id bigint NOT NULL,
|
|
||||||
name varchar NOT NULL,
|
|
||||||
description text,
|
|
||||||
time_type text,
|
|
||||||
created_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
updated_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
|
|
||||||
CONSTRAINT fk_phase_activities_phase
|
|
||||||
FOREIGN KEY (phase_id) REFERENCES phases(id)
|
|
||||||
ON DELETE CASCADE
|
|
||||||
);
|
|
||||||
|
|
||||||
CREATE TABLE checklists (
|
|
||||||
id bigint GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
|
||||||
name varchar NOT NULL,
|
|
||||||
description text,
|
|
||||||
phase_id bigint,
|
|
||||||
created_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
updated_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
deleted_at timestamptz,
|
|
||||||
|
|
||||||
CONSTRAINT fk_checklists_phase
|
|
||||||
FOREIGN KEY (phase_id) REFERENCES phases(id)
|
|
||||||
ON DELETE SET NULL
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
-- DAILY CHECKLISTS
|
|
||||||
CREATE TABLE daily_checklists (
|
|
||||||
id bigint GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
|
||||||
kandang_id bigint NOT NULL,
|
|
||||||
checklist_id bigint NOT NULL,
|
|
||||||
date date NOT NULL,
|
|
||||||
name varchar,
|
|
||||||
status varchar,
|
|
||||||
category category_code NOT NULL,
|
|
||||||
total_score integer,
|
|
||||||
document_path varchar,
|
|
||||||
reject_reason text,
|
|
||||||
created_by bigint,
|
|
||||||
created_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
updated_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
|
|
||||||
CONSTRAINT fk_daily_checklists_kandang
|
|
||||||
FOREIGN KEY (kandang_id) REFERENCES kandangs(id)
|
|
||||||
ON DELETE CASCADE,
|
|
||||||
|
|
||||||
CONSTRAINT fk_daily_checklists_checklist
|
|
||||||
FOREIGN KEY (checklist_id) REFERENCES checklists(id)
|
|
||||||
ON DELETE RESTRICT,
|
|
||||||
|
|
||||||
CONSTRAINT fk_daily_checklists_created_by
|
|
||||||
FOREIGN KEY (created_by) REFERENCES users(id)
|
|
||||||
ON DELETE SET NULL
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
--RELASI CHECKLIST ⇄ PHASE
|
|
||||||
|
|
||||||
CREATE TABLE daily_checklist_phases (
|
|
||||||
id bigint GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
|
||||||
checklist_id bigint NOT NULL,
|
|
||||||
phase_id bigint NOT NULL,
|
|
||||||
created_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
|
|
||||||
CONSTRAINT fk_dcp_checklist
|
|
||||||
FOREIGN KEY (checklist_id) REFERENCES checklists(id)
|
|
||||||
ON DELETE CASCADE,
|
|
||||||
|
|
||||||
CONSTRAINT fk_dcp_phase
|
|
||||||
FOREIGN KEY (phase_id) REFERENCES phases(id)
|
|
||||||
ON DELETE CASCADE,
|
|
||||||
|
|
||||||
CONSTRAINT uq_daily_checklist_phases UNIQUE (checklist_id, phase_id)
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
--ACTIVITY TASKS & ASSIGNMENT
|
|
||||||
|
|
||||||
|
|
||||||
CREATE TABLE daily_checklist_activity_tasks (
|
|
||||||
id bigint GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
|
||||||
checklist_id bigint NOT NULL,
|
|
||||||
phase_id bigint NOT NULL,
|
|
||||||
phase_activity_id bigint NOT NULL,
|
|
||||||
time_type text,
|
|
||||||
notes text,
|
|
||||||
created_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
updated_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
|
|
||||||
CONSTRAINT fk_dcat_checklist
|
|
||||||
FOREIGN KEY (checklist_id) REFERENCES checklists(id)
|
|
||||||
ON DELETE CASCADE,
|
|
||||||
|
|
||||||
CONSTRAINT fk_dcat_phase
|
|
||||||
FOREIGN KEY (phase_id) REFERENCES phases(id)
|
|
||||||
ON DELETE CASCADE,
|
|
||||||
|
|
||||||
CONSTRAINT fk_dcat_phase_activity
|
|
||||||
FOREIGN KEY (phase_activity_id) REFERENCES phase_activities(id)
|
|
||||||
ON DELETE CASCADE
|
|
||||||
);
|
|
||||||
|
|
||||||
CREATE TABLE daily_checklist_activity_task_assignments (
|
|
||||||
id bigint GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
|
||||||
task_id bigint NOT NULL,
|
|
||||||
employee_id bigint NOT NULL,
|
|
||||||
checked boolean NOT NULL DEFAULT false,
|
|
||||||
note text,
|
|
||||||
created_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
updated_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
|
|
||||||
CONSTRAINT fk_assignment_task
|
|
||||||
FOREIGN KEY (task_id) REFERENCES daily_checklist_activity_tasks(id)
|
|
||||||
ON DELETE CASCADE,
|
|
||||||
|
|
||||||
CONSTRAINT fk_assignment_employee
|
|
||||||
FOREIGN KEY (employee_id) REFERENCES employees(id)
|
|
||||||
ON DELETE CASCADE
|
|
||||||
);
|
|
||||||
|
|
||||||
--DAILY CHECKLIST TASK RESULT
|
|
||||||
CREATE TABLE daily_checklist_tasks (
|
|
||||||
id bigint GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
|
||||||
daily_checklist_id bigint NOT NULL,
|
|
||||||
checklist_id bigint NOT NULL,
|
|
||||||
checklist_item_id bigint,
|
|
||||||
is_completed boolean NOT NULL DEFAULT false,
|
|
||||||
score_value integer,
|
|
||||||
notes text,
|
|
||||||
photo_proof varchar,
|
|
||||||
status varchar,
|
|
||||||
created_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
updated_at timestamptz NOT NULL DEFAULT now(),
|
|
||||||
|
|
||||||
CONSTRAINT fk_dct_daily
|
|
||||||
FOREIGN KEY (daily_checklist_id) REFERENCES daily_checklists(id)
|
|
||||||
ON DELETE CASCADE,
|
|
||||||
|
|
||||||
CONSTRAINT fk_dct_checklist
|
|
||||||
FOREIGN KEY (checklist_id) REFERENCES checklists(id)
|
|
||||||
ON DELETE CASCADE,
|
|
||||||
|
|
||||||
CONSTRAINT fk_dct_checklist_item
|
|
||||||
FOREIGN KEY (checklist_item_id) REFERENCES phase_activities(id)
|
|
||||||
ON DELETE SET NULL
|
|
||||||
);
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
BEGIN;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_recordings_project_flock_kandang'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE recordings
|
|
||||||
DROP CONSTRAINT fk_recordings_project_flock_kandang;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
ALTER TABLE recordings
|
|
||||||
ADD CONSTRAINT fk_recordings_project_flock_kandang
|
|
||||||
FOREIGN KEY (project_flock_kandangs_id)
|
|
||||||
REFERENCES project_flock_kandangs (id)
|
|
||||||
ON DELETE RESTRICT ON UPDATE CASCADE;
|
|
||||||
|
|
||||||
COMMIT;
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
BEGIN;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_recordings_project_flock_kandang'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE recordings
|
|
||||||
DROP CONSTRAINT fk_recordings_project_flock_kandang;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
ALTER TABLE recordings
|
|
||||||
ADD CONSTRAINT fk_recordings_project_flock_kandang
|
|
||||||
FOREIGN KEY (project_flock_kandangs_id)
|
|
||||||
REFERENCES project_flock_kandangs (id)
|
|
||||||
ON DELETE CASCADE ON UPDATE CASCADE;
|
|
||||||
|
|
||||||
COMMIT;
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
ALTER TABLE daily_checklists
|
|
||||||
DROP CONSTRAINT IF EXISTS daily_checklists_date_kandang_category_key;
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
ALTER TABLE daily_checklists
|
|
||||||
ADD CONSTRAINT daily_checklists_date_kandang_category_key
|
|
||||||
UNIQUE (date, kandang_id, category);
|
|
||||||
-2
@@ -1,2 +0,0 @@
|
|||||||
ALTER TABLE daily_checklists
|
|
||||||
ALTER COLUMN checklist_id SET NOT NULL;
|
|
||||||
-2
@@ -1,2 +0,0 @@
|
|||||||
ALTER TABLE daily_checklists
|
|
||||||
ALTER COLUMN checklist_id DROP NOT NULL;
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
ALTER TABLE daily_checklist_phases
|
|
||||||
DROP CONSTRAINT IF EXISTS fk_dcp_daily_checklist,
|
|
||||||
ADD CONSTRAINT fk_dcp_checklist
|
|
||||||
FOREIGN KEY (checklist_id) REFERENCES checklists(id) ON DELETE CASCADE;
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
ALTER TABLE daily_checklist_phases
|
|
||||||
DROP CONSTRAINT IF EXISTS fk_dcp_checklist,
|
|
||||||
ADD CONSTRAINT fk_dcp_daily_checklist
|
|
||||||
FOREIGN KEY (checklist_id) REFERENCES daily_checklists(id) ON DELETE CASCADE;
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
-- Revert back to NO ACTION (RESTRICT behavior)
|
|
||||||
ALTER TABLE expense_nonstocks DROP CONSTRAINT IF EXISTS fk_expense_nonstocks_expense_id;
|
|
||||||
|
|
||||||
ALTER TABLE expense_nonstocks
|
|
||||||
ADD CONSTRAINT fk_expense_nonstocks_expense_id
|
|
||||||
FOREIGN KEY (expense_id) REFERENCES expenses(id)
|
|
||||||
ON DELETE NO ACTION;
|
|
||||||
|
|
||||||
-- Revert expense_realizations FK
|
|
||||||
ALTER TABLE expense_realizations DROP CONSTRAINT IF EXISTS fk_expense_realizations_nonstock_id;
|
|
||||||
|
|
||||||
ALTER TABLE expense_realizations
|
|
||||||
ADD CONSTRAINT fk_expense_realizations_nonstock_id
|
|
||||||
FOREIGN KEY (expense_nonstock_id) REFERENCES expense_nonstocks(id)
|
|
||||||
ON DELETE NO ACTION;
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
-- Drop existing FK constraints
|
|
||||||
ALTER TABLE expense_nonstocks DROP CONSTRAINT IF EXISTS fk_expense_nonstocks_expense_id;
|
|
||||||
|
|
||||||
-- Recreate with ON DELETE CASCADE
|
|
||||||
ALTER TABLE expense_nonstocks
|
|
||||||
ADD CONSTRAINT fk_expense_nonstocks_expense_id
|
|
||||||
FOREIGN KEY (expense_id) REFERENCES expenses(id)
|
|
||||||
ON DELETE CASCADE;
|
|
||||||
|
|
||||||
-- Drop and recreate expense_realizations FK
|
|
||||||
ALTER TABLE expense_realizations DROP CONSTRAINT IF EXISTS fk_expense_realizations_nonstock_id;
|
|
||||||
|
|
||||||
ALTER TABLE expense_realizations
|
|
||||||
ADD CONSTRAINT fk_expense_realizations_nonstock_id
|
|
||||||
FOREIGN KEY (expense_nonstock_id) REFERENCES expense_nonstocks(id)
|
|
||||||
ON DELETE CASCADE;
|
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
-- Revert back to NO ACTION (for rollback safety)
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
ALTER TABLE marketing_products DROP CONSTRAINT IF EXISTS fk_marketing_products_marketing_id;
|
|
||||||
|
|
||||||
ALTER TABLE marketing_products
|
|
||||||
ADD CONSTRAINT fk_marketing_products_marketing_id
|
|
||||||
FOREIGN KEY (marketing_id) REFERENCES marketings(id)
|
|
||||||
ON DELETE NO ACTION;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
ALTER TABLE marketing_delivery_products DROP CONSTRAINT IF EXISTS fk_marketing_delivery_products_marketing_product_id;
|
|
||||||
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
ADD CONSTRAINT fk_marketing_delivery_products_marketing_product_id
|
|
||||||
FOREIGN KEY (marketing_product_id) REFERENCES marketing_products(id)
|
|
||||||
ON DELETE NO ACTION;
|
|
||||||
END $$;
|
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
-- Ensure marketing_products FK is CASCADE (it should already be, but let's make sure)
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
-- Drop existing FK if exists
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_marketing_products_marketing_id'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE marketing_products DROP CONSTRAINT fk_marketing_products_marketing_id;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
-- Recreate with ON DELETE CASCADE
|
|
||||||
ALTER TABLE marketing_products
|
|
||||||
ADD CONSTRAINT fk_marketing_products_marketing_id
|
|
||||||
FOREIGN KEY (marketing_id) REFERENCES marketings(id)
|
|
||||||
ON DELETE CASCADE;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Ensure marketing_delivery_products FK is CASCADE
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
-- Drop existing FK if exists
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1 FROM pg_constraint
|
|
||||||
WHERE conname = 'fk_marketing_delivery_products_marketing_product_id'
|
|
||||||
) THEN
|
|
||||||
ALTER TABLE marketing_delivery_products DROP CONSTRAINT fk_marketing_delivery_products_marketing_product_id;
|
|
||||||
END IF;
|
|
||||||
|
|
||||||
-- Recreate with ON DELETE CASCADE
|
|
||||||
ALTER TABLE marketing_delivery_products
|
|
||||||
ADD CONSTRAINT fk_marketing_delivery_products_marketing_product_id
|
|
||||||
FOREIGN KEY (marketing_product_id) REFERENCES marketing_products(id)
|
|
||||||
ON DELETE CASCADE;
|
|
||||||
END $$;
|
|
||||||
-9
@@ -1,9 +0,0 @@
|
|||||||
-- Drop foreign key and column
|
|
||||||
ALTER TABLE laying_transfers
|
|
||||||
DROP CONSTRAINT IF EXISTS fk_laying_transfers_product_warehouse_id;
|
|
||||||
|
|
||||||
ALTER TABLE laying_transfers
|
|
||||||
DROP COLUMN IF EXISTS product_warehouse_id;
|
|
||||||
|
|
||||||
-- Drop index
|
|
||||||
DROP INDEX IF EXISTS idx_laying_transfers_product_warehouse_id;
|
|
||||||
-19
@@ -1,19 +0,0 @@
|
|||||||
-- Add product_warehouse_id to laying_transfers for FIFO support
|
|
||||||
ALTER TABLE laying_transfers
|
|
||||||
ADD COLUMN product_warehouse_id BIGINT;
|
|
||||||
|
|
||||||
-- Add foreign key
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'product_warehouses') THEN
|
|
||||||
ALTER TABLE laying_transfers
|
|
||||||
ADD CONSTRAINT fk_laying_transfers_product_warehouse_id
|
|
||||||
FOREIGN KEY (product_warehouse_id)
|
|
||||||
REFERENCES product_warehouses(id)
|
|
||||||
ON DELETE SET NULL;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Add index
|
|
||||||
CREATE INDEX idx_laying_transfers_product_warehouse_id
|
|
||||||
ON laying_transfers(product_warehouse_id);
|
|
||||||
-4
@@ -1,4 +0,0 @@
|
|||||||
ALTER TABLE daily_checklist_activity_tasks
|
|
||||||
DROP CONSTRAINT IF EXISTS fk_dcat_daily_checklist,
|
|
||||||
ADD CONSTRAINT fk_dcat_checklist
|
|
||||||
FOREIGN KEY (checklist_id) REFERENCES checklists(id) ON DELETE CASCADE;
|
|
||||||
-4
@@ -1,4 +0,0 @@
|
|||||||
ALTER TABLE daily_checklist_activity_tasks
|
|
||||||
DROP CONSTRAINT IF EXISTS fk_dcat_checklist,
|
|
||||||
ADD CONSTRAINT fk_dcat_daily_checklist
|
|
||||||
FOREIGN KEY (checklist_id) REFERENCES daily_checklists(id) ON DELETE CASCADE;
|
|
||||||
-14
@@ -1,14 +0,0 @@
|
|||||||
-- Rollback: Remove STOCKABLE fields from laying_transfers
|
|
||||||
|
|
||||||
-- Drop index
|
|
||||||
DROP INDEX IF EXISTS idx_laying_transfers_dest_product_warehouse_id;
|
|
||||||
|
|
||||||
-- Drop foreign key constraint
|
|
||||||
ALTER TABLE laying_transfers
|
|
||||||
DROP CONSTRAINT IF EXISTS fk_laying_transfers_dest_product_warehouse_id;
|
|
||||||
|
|
||||||
-- Drop columns
|
|
||||||
ALTER TABLE laying_transfers
|
|
||||||
DROP COLUMN IF EXISTS dest_product_warehouse_id,
|
|
||||||
DROP COLUMN IF EXISTS total_qty,
|
|
||||||
DROP COLUMN IF EXISTS total_used;
|
|
||||||
-30
@@ -1,30 +0,0 @@
|
|||||||
-- Add STOCKABLE fields to laying_transfers for destination warehouse
|
|
||||||
-- This enables Transfer to Laying to work as DUAL ROLE (Stockable + Usable)
|
|
||||||
|
|
||||||
-- Add columns for STOCKABLE role (destination warehouse)
|
|
||||||
ALTER TABLE laying_transfers
|
|
||||||
ADD COLUMN dest_product_warehouse_id BIGINT,
|
|
||||||
ADD COLUMN total_qty NUMERIC(15, 3) DEFAULT 0,
|
|
||||||
ADD COLUMN total_used NUMERIC(15, 3) DEFAULT 0;
|
|
||||||
|
|
||||||
-- Add foreign key constraint
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (SELECT 1 FROM pg_tables WHERE tablename = 'product_warehouses') THEN
|
|
||||||
ALTER TABLE laying_transfers
|
|
||||||
ADD CONSTRAINT fk_laying_transfers_dest_product_warehouse_id
|
|
||||||
FOREIGN KEY (dest_product_warehouse_id)
|
|
||||||
REFERENCES product_warehouses(id)
|
|
||||||
ON DELETE SET NULL;
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
-- Add index for performance
|
|
||||||
CREATE INDEX idx_laying_transfers_dest_product_warehouse_id
|
|
||||||
ON laying_transfers(dest_product_warehouse_id);
|
|
||||||
|
|
||||||
-- Add comment for documentation
|
|
||||||
COMMENT ON COLUMN laying_transfers.product_warehouse_id IS 'Product warehouse at source (Growing flock) - for USABLE role';
|
|
||||||
COMMENT ON COLUMN laying_transfers.dest_product_warehouse_id IS 'Product warehouse at destination (Laying flock) - for STOCKABLE role';
|
|
||||||
COMMENT ON COLUMN laying_transfers.total_qty IS 'Total lot quantity introduced to destination warehouse - for STOCKABLE role';
|
|
||||||
COMMENT ON COLUMN laying_transfers.total_used IS 'Quantity already consumed from this lot at destination - for STOCKABLE role';
|
|
||||||
-2
@@ -1,2 +0,0 @@
|
|||||||
ALTER TABLE daily_checklist_activity_task_assignments
|
|
||||||
DROP CONSTRAINT IF EXISTS daily_checklist_activity_task_assignments_task_employee_key;
|
|
||||||
-3
@@ -1,3 +0,0 @@
|
|||||||
ALTER TABLE daily_checklist_activity_task_assignments
|
|
||||||
ADD CONSTRAINT daily_checklist_activity_task_assignments_task_employee_key
|
|
||||||
UNIQUE (task_id, employee_id);
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
ALTER TABLE phase_activities
|
|
||||||
DROP COLUMN IF EXISTS deleted_at;
|
|
||||||
|
|
||||||
ALTER TABLE phases
|
|
||||||
DROP COLUMN IF EXISTS deleted_at;
|
|
||||||
|
|
||||||
ALTER TABLE employees
|
|
||||||
DROP COLUMN IF EXISTS deleted_at;
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
ALTER TABLE employees
|
|
||||||
ADD COLUMN IF NOT EXISTS deleted_at TIMESTAMPTZ;
|
|
||||||
|
|
||||||
ALTER TABLE phases
|
|
||||||
ADD COLUMN IF NOT EXISTS deleted_at TIMESTAMPTZ;
|
|
||||||
|
|
||||||
ALTER TABLE phase_activities
|
|
||||||
ADD COLUMN IF NOT EXISTS deleted_at TIMESTAMPTZ;
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
-- Remove chart_data, uniform_date, and related indexes
|
|
||||||
DROP INDEX IF EXISTS idx_project_flock_kandang_uniformity_uniform_date;
|
|
||||||
DROP INDEX IF EXISTS idx_project_flock_kandang_uniformity_unique;
|
|
||||||
|
|
||||||
ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
DROP COLUMN IF EXISTS chart_data,
|
|
||||||
DROP COLUMN IF EXISTS uniform_date;
|
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
-- Add uniform_date (if missing), chart_data, and unique constraint for uniformity records
|
|
||||||
ALTER TABLE project_flock_kandang_uniformity
|
|
||||||
ADD COLUMN IF NOT EXISTS uniform_date TIMESTAMPTZ,
|
|
||||||
ADD COLUMN IF NOT EXISTS chart_data JSONB;
|
|
||||||
|
|
||||||
DO $$
|
|
||||||
BEGIN
|
|
||||||
IF EXISTS (
|
|
||||||
SELECT 1
|
|
||||||
FROM information_schema.columns
|
|
||||||
WHERE table_schema = 'public'
|
|
||||||
AND table_name = 'project_flock_kandang_uniformity'
|
|
||||||
AND column_name = 'deleted_at'
|
|
||||||
) THEN
|
|
||||||
CREATE UNIQUE INDEX IF NOT EXISTS idx_project_flock_kandang_uniformity_unique
|
|
||||||
ON project_flock_kandang_uniformity (project_flock_kandang_id, week, uniform_date)
|
|
||||||
WHERE deleted_at IS NULL;
|
|
||||||
ELSE
|
|
||||||
CREATE UNIQUE INDEX IF NOT EXISTS idx_project_flock_kandang_uniformity_unique
|
|
||||||
ON project_flock_kandang_uniformity (project_flock_kandang_id, week, uniform_date);
|
|
||||||
END IF;
|
|
||||||
END $$;
|
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_project_flock_kandang_uniformity_uniform_date
|
|
||||||
ON project_flock_kandang_uniformity (uniform_date);
|
|
||||||
-4
@@ -1,4 +0,0 @@
|
|||||||
-- Remove expense_nonstock_id from stock_transfer_details
|
|
||||||
ALTER TABLE stock_transfer_details DROP CONSTRAINT IF EXISTS fk_stock_transfer_details_expense_nonstock;
|
|
||||||
ALTER TABLE stock_transfer_details DROP COLUMN IF EXISTS expense_nonstock_id;
|
|
||||||
DROP INDEX IF EXISTS idx_stock_transfer_details_expense_nonstock_id;
|
|
||||||
-10
@@ -1,10 +0,0 @@
|
|||||||
-- Add expense_nonstock_id to stock_transfer_details
|
|
||||||
-- This allows tracking expedition/transport costs for stock transfers (same as purchase)
|
|
||||||
|
|
||||||
ALTER TABLE stock_transfer_details
|
|
||||||
ADD COLUMN expense_nonstock_id BIGINT,
|
|
||||||
ADD CONSTRAINT fk_stock_transfer_details_expense_nonstock
|
|
||||||
FOREIGN KEY (expense_nonstock_id) REFERENCES expense_nonstocks(id) ON DELETE SET NULL;
|
|
||||||
|
|
||||||
-- Create index for better query performance
|
|
||||||
CREATE INDEX idx_stock_transfer_details_expense_nonstock_id ON stock_transfer_details(expense_nonstock_id);
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP TABLE IF EXISTS config_checklists;
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
CREATE TABLE IF NOT EXISTS config_checklists (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
date DATE NOT NULL,
|
|
||||||
percentage_threshold_bad INTEGER NOT NULL,
|
|
||||||
percentage_threshold_enough INTEGER NOT NULL,
|
|
||||||
created_at TIMESTAMPTZ NOT NULL DEFAULT NOW(),
|
|
||||||
updated_at TIMESTAMPTZ NOT NULL DEFAULT NOW(),
|
|
||||||
deleted_at TIMESTAMPTZ
|
|
||||||
);
|
|
||||||
-6
@@ -1,6 +0,0 @@
|
|||||||
BEGIN;
|
|
||||||
|
|
||||||
ALTER TABLE payments
|
|
||||||
DROP COLUMN IF EXISTS party_account_number;
|
|
||||||
|
|
||||||
COMMIT;
|
|
||||||
-6
@@ -1,6 +0,0 @@
|
|||||||
BEGIN;
|
|
||||||
|
|
||||||
ALTER TABLE payments
|
|
||||||
ADD COLUMN IF NOT EXISTS party_account_number VARCHAR(50);
|
|
||||||
|
|
||||||
COMMIT;
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP TABLE IF EXISTS projects;
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user