feat: dockerize full stack (web + api + postgres)

- Multi-stage Dockerfiles for API (NestJS, prisma migrate on start) and web (Vite + nginx reverse proxy)
- docker-compose.yml orchestrating db/api/web with healthcheck and persistent volumes
- nginx proxies /api and /avatars to API; web built with relative API URL
- scripts/docker-up.sh: ExFAT-safe wrapper that strips macOS AppleDouble (._*) sidecars before build
- Conditionally register GoogleStrategy only when GOOGLE_CLIENT_ID is set
- Fix unused-variable TS errors blocking production build
This commit is contained in:
Dwindi Ramadhana
2026-06-17 22:35:58 +07:00
parent 6a6e74562c
commit d85b813701
12 changed files with 236 additions and 16 deletions

10
apps/web/.dockerignore Executable file
View File

@@ -0,0 +1,10 @@
node_modules
dist
.git
.gitignore
.env
.env.local
*.log
.DS_Store
._*
.tsbuildinfo

18
apps/web/Dockerfile Executable file
View File

@@ -0,0 +1,18 @@
# syntax=docker/dockerfile:1
# ---- Build the Vite/React app ----
FROM node:20-alpine AS builder
WORKDIR /app
ARG VITE_API_URL=
ENV VITE_API_URL=$VITE_API_URL
COPY package*.json ./
RUN npm ci
COPY . .
RUN npm run build
# ---- Serve static build with nginx + reverse proxy to API ----
FROM nginx:alpine AS runner
COPY --from=builder /app/dist /usr/share/nginx/html
COPY nginx.conf /etc/nginx/conf.d/default.conf
EXPOSE 80
CMD ["nginx", "-g", "daemon off;"]

30
apps/web/nginx.conf Executable file
View File

@@ -0,0 +1,30 @@
server {
listen 80;
server_name _;
root /usr/share/nginx/html;
index index.html;
# Client upload body size (avatars, etc.)
client_max_body_size 10m;
# Proxy API calls to the api service (keeps /api prefix)
location /api/ {
proxy_pass http://api:3001;
proxy_http_version 1.1;
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto $scheme;
}
# Proxy static uploads (avatars served by the API from /public)
location /avatars/ {
proxy_pass http://api:3001;
proxy_set_header Host $host;
}
# SPA fallback
location / {
try_files $uri $uri/ /index.html;
}
}

View File

@@ -50,7 +50,7 @@ interface AddMoneyDialogProps {
onSuccess: () => void;
}
export function AddMoneyDialog({ open, onOpenChange, goalId, goalCurrency, onSuccess }: AddMoneyDialogProps) {
export function AddMoneyDialog({ open, onOpenChange, goalId, onSuccess }: AddMoneyDialogProps) {
const { t } = useLanguage();
const [loading, setLoading] = useState(false);
const [wallets, setWallets] = useState<Wallet[]>([]);

View File

@@ -69,19 +69,6 @@ export function WalletCard({ wallet, balance, onEdit, onDelete, onClick }: Walle
loadRates();
}, []);
// Format large numbers with suffix
const formatCompactNumber = (num: number): string => {
const absNum = Math.abs(num);
if (absNum >= 1_000_000_000) {
return `${(num / 1_000_000_000).toFixed(1)} ${t.numberFormat.billion}`;
} else if (absNum >= 1_000_000) {
return `${(num / 1_000_000).toFixed(1)} ${t.numberFormat.million}`;
} else if (absNum >= 1_000) {
return `${(num / 1_000).toFixed(1)} ${t.numberFormat.thousand}`;
}
return num.toLocaleString('id-ID');
};
// Fetch goal allocations for this wallet
useEffect(() => {
const fetchAllocations = async () => {
@@ -128,7 +115,7 @@ export function WalletCard({ wallet, balance, onEdit, onDelete, onClick }: Walle
}
}, [wallet.id, balance.reservedBalance, balance.totalBalance, exchangeRates]);
const formatBalance = (amount: number, isReserved = false) => {
const formatBalance = (amount: number, _isReserved = false) => {
if (wallet.kind === 'asset' && wallet.unit) {
// For assets: amount is already in units, convert to IDR for secondary display
const idrValue = amount * (balance.pricePerUnit || 0);