hoffentlich

This commit is contained in:
knuthtimo-lab 2026-02-28 18:45:45 +01:00
parent 02bb2ed994
commit 35c23164bf
8 changed files with 44 additions and 25 deletions

View File

@ -1,9 +1,12 @@
# ============================================= # =============================================
# Stage 1: Dependencies # Stage 1: Dependencies
# ============================================= # =============================================
FROM node:20-alpine AS deps FROM node:20-slim AS deps
RUN corepack enable && corepack prepare pnpm@9.12.0 --activate RUN corepack enable && corepack prepare pnpm@9.12.0 --activate
# Install OpenSSL for Prisma
RUN apt-get update && apt-get install -y openssl ca-certificates && rm -rf /var/lib/apt/lists/*
WORKDIR /app WORKDIR /app
# Copy workspace config files # Copy workspace config files
@ -17,9 +20,12 @@ RUN pnpm install --frozen-lockfile
# ============================================= # =============================================
# Stage 2: Build # Stage 2: Build
# ============================================= # =============================================
FROM node:20-alpine AS builder FROM node:20-slim AS builder
RUN corepack enable && corepack prepare pnpm@9.12.0 --activate RUN corepack enable && corepack prepare pnpm@9.12.0 --activate
# Install OpenSSL for Prisma
RUN apt-get update && apt-get install -y openssl ca-certificates && rm -rf /var/lib/apt/lists/*
WORKDIR /app WORKDIR /app
COPY --from=deps /app/node_modules ./node_modules COPY --from=deps /app/node_modules ./node_modules
@ -28,7 +34,7 @@ COPY --from=deps /app/packages/shared/node_modules ./packages/shared/node_module
COPY . . COPY . .
# Generate Prisma client # Generate Prisma client for Alpine Linux
RUN pnpm --filter @innungsapp/shared prisma:generate RUN pnpm --filter @innungsapp/shared prisma:generate
# Build the admin app # Build the admin app
@ -39,9 +45,12 @@ RUN pnpm --filter @innungsapp/admin build
# ============================================= # =============================================
# Stage 3: Production Runner # Stage 3: Production Runner
# ============================================= # =============================================
FROM node:20-alpine AS runner FROM node:20-slim AS runner
RUN corepack enable && corepack prepare pnpm@9.12.0 --activate RUN corepack enable && corepack prepare pnpm@9.12.0 --activate
# Install OpenSSL for Prisma
RUN apt-get update && apt-get install -y openssl ca-certificates wget && rm -rf /var/lib/apt/lists/*
WORKDIR /app WORKDIR /app
ENV NODE_ENV=production ENV NODE_ENV=production
@ -51,17 +60,20 @@ ENV NEXT_TELEMETRY_DISABLED=1
RUN addgroup --system --gid 1001 nodejs && \ RUN addgroup --system --gid 1001 nodejs && \
adduser --system --uid 1001 nextjs adduser --system --uid 1001 nextjs
# Copy built output # Copy built output (standalone includes all necessary node_modules)
COPY --from=builder /app/apps/admin/.next/standalone ./ COPY --from=builder /app/apps/admin/.next/standalone ./
COPY --from=builder /app/apps/admin/.next/static ./apps/admin/.next/static COPY --from=builder /app/apps/admin/.next/static ./apps/admin/.next/static
COPY --from=builder /app/apps/admin/public ./apps/admin/public COPY --from=builder /app/apps/admin/public ./apps/admin/public
# Copy Prisma schema + migrations for runtime migrations # Copy Prisma schema + migrations for runtime migrations
COPY --from=builder /app/packages/shared/prisma ./packages/shared/prisma COPY --from=builder /app/packages/shared/prisma ./packages/shared/prisma
COPY --from=builder /app/node_modules/.pnpm ./node_modules/.pnpm
COPY --from=builder /app/node_modules/@prisma ./node_modules/@prisma # Copy Prisma Engine binaries directly to .next/server (where Next.js looks for them)
COPY --from=builder /app/node_modules/prisma ./node_modules/prisma COPY --from=builder /app/node_modules/.pnpm/@prisma+client@5.22.0_prisma@5.22.0/node_modules/.prisma/client/libquery_engine-debian-openssl-3.0.x.so.node /app/apps/admin/.next/server/
COPY --from=builder /app/node_modules/.bin/prisma ./node_modules/.bin/prisma COPY --from=builder /app/node_modules/.pnpm/@prisma+client@5.22.0_prisma@5.22.0/node_modules/.prisma/client/schema.prisma /app/apps/admin/.next/server/
# Install Prisma CLI globally for runtime migrations
RUN npm install -g prisma@5.22.0
# Create uploads directory # Create uploads directory
RUN mkdir -p /app/uploads && chown nextjs:nodejs /app/uploads RUN mkdir -p /app/uploads && chown nextjs:nodejs /app/uploads

View File

@ -64,13 +64,13 @@ export default async function MitgliederPage(props: {
} }
}) })
const adminUserIds = new Set(admins.map(a => a.userId)) const adminUserIds = new Set(admins.map((a: typeof admins[number]) => a.userId))
// Map userId → member record so admin entries show real member data // Map userId → member record so admin entries show real member data
const memberByUserId = new Map(members.filter(m => m.userId).map(m => [m.userId!, m])) const memberByUserId = new Map<string, typeof members[number]>(members.filter((m: typeof members[number]) => m.userId).map((m: typeof members[number]) => [m.userId!, m]))
const combinedList = [ const combinedList = [
// Include admins only if there's no status filter, or if filtering for 'aktiv' // Include admins only if there's no status filter, or if filtering for 'aktiv'
...(!statusFilter || statusFilter === 'aktiv' ? admins.map(a => { ...(!statusFilter || statusFilter === 'aktiv' ? admins.map((a: typeof admins[number]) => {
const m = memberByUserId.get(a.user.id) const m = memberByUserId.get(a.user.id)
return { return {
id: m ? m.id : `admin-${a.user.id}`, id: m ? m.id : `admin-${a.user.id}`,
@ -86,7 +86,7 @@ export default async function MitgliederPage(props: {
role: 'Administrator', role: 'Administrator',
} }
}) : []), }) : []),
...members.filter(m => !adminUserIds.has(m.userId ?? '')).map(m => ({ ...members.filter((m: typeof members[number]) => !adminUserIds.has(m.userId ?? '')).map((m: typeof members[number]) => ({
id: m.id, id: m.id,
name: m.name, name: m.name,
betrieb: m.betrieb, betrieb: m.betrieb,
@ -101,7 +101,7 @@ export default async function MitgliederPage(props: {
})) }))
] ]
combinedList.sort((a, b) => a.name.localeCompare(b.name)) combinedList.sort((a: typeof combinedList[number], b: typeof combinedList[number]) => a.name.localeCompare(b.name))
return ( return (
<div className="space-y-6"> <div className="space-y-6">

View File

@ -43,7 +43,7 @@ export default function NewsEditPage({ params }: { params: Promise<{ id: string
setBody(news.body) setBody(news.body)
setKategorie(news.kategorie) setKategorie(news.kategorie)
if (news.attachments) { if (news.attachments) {
setAttachments(news.attachments.map(a => ({ ...a, sizeBytes: a.sizeBytes ?? 0 }))) setAttachments(news.attachments.map((a: typeof news.attachments[number]) => ({ ...a, sizeBytes: a.sizeBytes ?? 0 })))
} }
} }
}, [news]) }, [news])

View File

@ -30,8 +30,8 @@ export default async function NewsPage() {
orderBy: [{ publishedAt: 'desc' }, { createdAt: 'desc' }], orderBy: [{ publishedAt: 'desc' }, { createdAt: 'desc' }],
}) })
const published = news.filter((n) => n.publishedAt) const published = news.filter((n: typeof news[number]) => n.publishedAt)
const drafts = news.filter((n) => !n.publishedAt) const drafts = news.filter((n: typeof news[number]) => !n.publishedAt)
return ( return (
<div className="space-y-6"> <div className="space-y-6">
@ -56,7 +56,7 @@ export default async function NewsPage() {
<div className="bg-white rounded-lg border overflow-hidden"> <div className="bg-white rounded-lg border overflow-hidden">
<table className="w-full data-table"> <table className="w-full data-table">
<tbody> <tbody>
{drafts.map((n) => ( {drafts.map((n: typeof drafts[number]) => (
<tr key={n.id}> <tr key={n.id}>
<td className="w-full"> <td className="w-full">
<p className="font-medium text-gray-900">{n.title}</p> <p className="font-medium text-gray-900">{n.title}</p>
@ -96,7 +96,7 @@ export default async function NewsPage() {
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
{published.map((n) => ( {published.map((n: typeof published[number]) => (
<tr key={n.id}> <tr key={n.id}>
<td className="font-medium text-gray-900">{n.title}</td> <td className="font-medium text-gray-900">{n.title}</td>
<td> <td>

View File

@ -68,7 +68,7 @@ export default async function DashboardPage() {
</Link> </Link>
</div> </div>
<div className="space-y-3"> <div className="space-y-3">
{recentNews.map((n) => ( {recentNews.map((n: typeof recentNews[number]) => (
<div key={n.id} className="flex items-start gap-3 py-2 border-b last:border-0"> <div key={n.id} className="flex items-start gap-3 py-2 border-b last:border-0">
<div className="flex-1 min-w-0"> <div className="flex-1 min-w-0">
<p className="font-medium text-sm text-gray-900 truncate">{n.title}</p> <p className="font-medium text-sm text-gray-900 truncate">{n.title}</p>
@ -99,7 +99,7 @@ export default async function DashboardPage() {
{nextTermine.length === 0 && ( {nextTermine.length === 0 && (
<p className="text-sm text-gray-500">Keine bevorstehenden Termine</p> <p className="text-sm text-gray-500">Keine bevorstehenden Termine</p>
)} )}
{nextTermine.map((t) => ( {nextTermine.map((t: typeof nextTermine[number]) => (
<div key={t.id} className="flex items-start gap-3 py-2 border-b last:border-0"> <div key={t.id} className="flex items-start gap-3 py-2 border-b last:border-0">
<div className="text-center min-w-[40px]"> <div className="text-center min-w-[40px]">
<p className="text-lg font-bold text-brand-500 leading-none"> <p className="text-lg font-bold text-brand-500 leading-none">

View File

@ -64,13 +64,13 @@ export default function StelleNeuPage() {
required required
value={form.memberId} value={form.memberId}
onChange={(e) => { onChange={(e) => {
const selected = members?.find((m) => m.id === e.target.value) const selected = members?.find((m: NonNullable<typeof members>[number]) => m.id === e.target.value)
setForm({ ...form, memberId: e.target.value, sparte: selected?.sparte ?? form.sparte }) setForm({ ...form, memberId: e.target.value, sparte: selected?.sparte ?? form.sparte })
}} }}
className={inputClass} className={inputClass}
> >
<option value="">Mitglied auswählen...</option> <option value="">Mitglied auswählen...</option>
{members?.map((m) => ( {members?.map((m: NonNullable<typeof members>[number]) => (
<option key={m.id} value={m.id}> <option key={m.id} value={m.id}>
{m.betrieb} {m.name} {m.betrieb} {m.name}
</option> </option>

View File

@ -9,10 +9,10 @@ MIGRATIONS_DIR="./packages/shared/prisma/migrations"
set -- "$MIGRATIONS_DIR"/*/migration.sql set -- "$MIGRATIONS_DIR"/*/migration.sql
if [ -f "$1" ]; then if [ -f "$1" ]; then
echo "Applying Prisma migrations..." echo "Applying Prisma migrations..."
node_modules/.bin/prisma migrate deploy --schema=./packages/shared/prisma/schema.prisma npx prisma migrate deploy --schema=./packages/shared/prisma/schema.prisma
else else
echo "No Prisma migrations found. Syncing schema with db push..." echo "No Prisma migrations found. Syncing schema with db push..."
node_modules/.bin/prisma db push --skip-generate --schema=./packages/shared/prisma/schema.prisma npx prisma db push --skip-generate --schema=./packages/shared/prisma/schema.prisma
fi fi
echo "Starting Next.js server..." echo "Starting Next.js server..."

View File

@ -4,6 +4,13 @@ const nextConfig: NextConfig = {
transpilePackages: ['@innungsapp/shared'], transpilePackages: ['@innungsapp/shared'],
output: process.env.DOCKER_BUILD ? 'standalone' : undefined, output: process.env.DOCKER_BUILD ? 'standalone' : undefined,
experimental: {}, experimental: {},
// Include Prisma binaries in standalone build
outputFileTracingIncludes: {
'/': [
'./node_modules/.pnpm/@prisma+client@5.22.0_prisma@5.22.0/node_modules/.prisma/**/*',
'./node_modules/.pnpm/@prisma+client@5.22.0_prisma@5.22.0/node_modules/@prisma/client/**/*',
],
},
webpack: (config, { dev }) => { webpack: (config, { dev }) => {
if (dev) { if (dev) {
// Avoid filesystem cache writes on very low-disk dev machines (ENOSPC). // Avoid filesystem cache writes on very low-disk dev machines (ENOSPC).