This commit is contained in:
2026-03-04 14:13:16 +01:00
parent b7d826e29c
commit 56ea3348d6
41 changed files with 846 additions and 162 deletions

View File

@@ -14,6 +14,7 @@
"prisma:push": "prisma db push",
"prisma:studio": "prisma studio",
"prisma:seed": "tsx prisma/seed.ts",
"prisma:seed-superadmin": "tsx prisma/seed-superadmin.ts",
"prisma:seed-admin": "tsx prisma/seed-admin-password.ts",
"prisma:seed-demo-members": "tsx prisma/seed-demo-members.ts"
},

View File

@@ -0,0 +1,350 @@
-- CreateTable
CREATE TABLE "user" (
"id" TEXT NOT NULL,
"name" TEXT NOT NULL,
"email" TEXT NOT NULL,
"email_verified" BOOLEAN NOT NULL DEFAULT false,
"image" TEXT,
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updated_at" TIMESTAMP(3) NOT NULL,
"role" TEXT,
"banned" BOOLEAN DEFAULT false,
"ban_reason" TEXT,
"ban_expires" TIMESTAMP(3),
"must_change_password" BOOLEAN DEFAULT false,
CONSTRAINT "user_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "session" (
"id" TEXT NOT NULL,
"expires_at" TIMESTAMP(3) NOT NULL,
"token" TEXT NOT NULL,
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updated_at" TIMESTAMP(3) NOT NULL,
"ip_address" TEXT,
"user_agent" TEXT,
"user_id" TEXT NOT NULL,
CONSTRAINT "session_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "account" (
"id" TEXT NOT NULL,
"account_id" TEXT NOT NULL,
"provider_id" TEXT NOT NULL,
"user_id" TEXT NOT NULL,
"access_token" TEXT,
"refresh_token" TEXT,
"id_token" TEXT,
"access_token_expires_at" TIMESTAMP(3),
"refresh_token_expires_at" TIMESTAMP(3),
"scope" TEXT,
"password" TEXT,
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updated_at" TIMESTAMP(3) NOT NULL,
CONSTRAINT "account_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "verification" (
"id" TEXT NOT NULL,
"identifier" TEXT NOT NULL,
"value" TEXT NOT NULL,
"expires_at" TIMESTAMP(3) NOT NULL,
"created_at" TIMESTAMP(3) DEFAULT CURRENT_TIMESTAMP,
"updated_at" TIMESTAMP(3),
CONSTRAINT "verification_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "organizations" (
"id" TEXT NOT NULL,
"name" TEXT NOT NULL,
"slug" TEXT NOT NULL,
"plan" TEXT NOT NULL DEFAULT 'pilot',
"logo_url" TEXT,
"primary_color" TEXT NOT NULL DEFAULT '#E63946',
"secondary_color" TEXT,
"contact_email" TEXT,
"avv_accepted" BOOLEAN NOT NULL DEFAULT false,
"avv_accepted_at" TIMESTAMP(3),
"landing_page_title" TEXT,
"landing_page_text" TEXT,
"landing_page_section_title" TEXT,
"landing_page_button_text" TEXT,
"landing_page_hero_image" TEXT,
"landing_page_hero_overlay_opacity" INTEGER DEFAULT 50,
"landing_page_features" JSONB,
"landing_page_footer" JSONB,
"app_store_url" TEXT,
"play_store_url" TEXT,
"ai_enabled" BOOLEAN NOT NULL DEFAULT false,
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
CONSTRAINT "organizations_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "members" (
"id" TEXT NOT NULL,
"org_id" TEXT NOT NULL,
"user_id" TEXT,
"name" TEXT NOT NULL,
"betrieb" TEXT NOT NULL,
"sparte" TEXT NOT NULL,
"ort" TEXT NOT NULL,
"telefon" TEXT,
"email" TEXT NOT NULL,
"status" TEXT NOT NULL DEFAULT 'aktiv',
"ist_ausbildungsbetrieb" BOOLEAN NOT NULL DEFAULT false,
"seit" INTEGER,
"avatar_url" TEXT,
"push_token" TEXT,
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updated_at" TIMESTAMP(3) NOT NULL,
CONSTRAINT "members_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "user_roles" (
"id" TEXT NOT NULL,
"org_id" TEXT NOT NULL,
"user_id" TEXT NOT NULL,
"role" TEXT NOT NULL,
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
CONSTRAINT "user_roles_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "news" (
"id" TEXT NOT NULL,
"org_id" TEXT NOT NULL,
"author_id" TEXT,
"title" TEXT NOT NULL,
"body" TEXT NOT NULL,
"kategorie" TEXT NOT NULL,
"published_at" TIMESTAMP(3),
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
CONSTRAINT "news_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "news_reads" (
"id" TEXT NOT NULL,
"news_id" TEXT NOT NULL,
"user_id" TEXT NOT NULL,
"read_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
CONSTRAINT "news_reads_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "news_attachments" (
"id" TEXT NOT NULL,
"news_id" TEXT NOT NULL,
"name" TEXT NOT NULL,
"storage_path" TEXT NOT NULL,
"mime_type" TEXT,
"size_bytes" INTEGER,
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
CONSTRAINT "news_attachments_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "stellen" (
"id" TEXT NOT NULL,
"org_id" TEXT NOT NULL,
"member_id" TEXT NOT NULL,
"sparte" TEXT NOT NULL,
"stellen_anz" INTEGER NOT NULL DEFAULT 1,
"verguetung" TEXT,
"lehrjahr" TEXT,
"beschreibung" TEXT,
"kontakt_email" TEXT NOT NULL,
"kontakt_name" TEXT,
"aktiv" BOOLEAN NOT NULL DEFAULT true,
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updated_at" TIMESTAMP(3) NOT NULL,
CONSTRAINT "stellen_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "termine" (
"id" TEXT NOT NULL,
"org_id" TEXT NOT NULL,
"titel" TEXT NOT NULL,
"datum" TIMESTAMP(3) NOT NULL,
"uhrzeit" TEXT,
"ende_datum" TIMESTAMP(3),
"ende_uhrzeit" TEXT,
"ort" TEXT,
"adresse" TEXT,
"typ" TEXT NOT NULL,
"beschreibung" TEXT,
"max_teilnehmer" INTEGER,
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
CONSTRAINT "termine_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "termin_anmeldungen" (
"id" TEXT NOT NULL,
"termin_id" TEXT NOT NULL,
"member_id" TEXT NOT NULL,
"angemeldet_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
CONSTRAINT "termin_anmeldungen_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "conversations" (
"id" TEXT NOT NULL,
"org_id" TEXT NOT NULL,
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updated_at" TIMESTAMP(3) NOT NULL,
CONSTRAINT "conversations_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "conversation_members" (
"id" TEXT NOT NULL,
"conversation_id" TEXT NOT NULL,
"member_id" TEXT NOT NULL,
"last_read_at" TIMESTAMP(3),
CONSTRAINT "conversation_members_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "messages" (
"id" TEXT NOT NULL,
"conversation_id" TEXT NOT NULL,
"sender_id" TEXT NOT NULL,
"body" TEXT NOT NULL,
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
CONSTRAINT "messages_pkey" PRIMARY KEY ("id")
);
-- CreateIndex
CREATE UNIQUE INDEX "user_email_key" ON "user"("email");
-- CreateIndex
CREATE UNIQUE INDEX "session_token_key" ON "session"("token");
-- CreateIndex
CREATE UNIQUE INDEX "organizations_slug_key" ON "organizations"("slug");
-- CreateIndex
CREATE UNIQUE INDEX "members_user_id_key" ON "members"("user_id");
-- CreateIndex
CREATE INDEX "members_org_id_idx" ON "members"("org_id");
-- CreateIndex
CREATE INDEX "members_status_idx" ON "members"("status");
-- CreateIndex
CREATE UNIQUE INDEX "user_roles_org_id_user_id_key" ON "user_roles"("org_id", "user_id");
-- CreateIndex
CREATE INDEX "news_org_id_idx" ON "news"("org_id");
-- CreateIndex
CREATE INDEX "news_published_at_idx" ON "news"("published_at");
-- CreateIndex
CREATE UNIQUE INDEX "news_reads_news_id_user_id_key" ON "news_reads"("news_id", "user_id");
-- CreateIndex
CREATE INDEX "stellen_org_id_idx" ON "stellen"("org_id");
-- CreateIndex
CREATE INDEX "stellen_aktiv_idx" ON "stellen"("aktiv");
-- CreateIndex
CREATE INDEX "termine_org_id_idx" ON "termine"("org_id");
-- CreateIndex
CREATE INDEX "termine_datum_idx" ON "termine"("datum");
-- CreateIndex
CREATE UNIQUE INDEX "termin_anmeldungen_termin_id_member_id_key" ON "termin_anmeldungen"("termin_id", "member_id");
-- CreateIndex
CREATE INDEX "conversations_org_id_idx" ON "conversations"("org_id");
-- CreateIndex
CREATE UNIQUE INDEX "conversation_members_conversation_id_member_id_key" ON "conversation_members"("conversation_id", "member_id");
-- CreateIndex
CREATE INDEX "messages_conversation_id_idx" ON "messages"("conversation_id");
-- AddForeignKey
ALTER TABLE "session" ADD CONSTRAINT "session_user_id_fkey" FOREIGN KEY ("user_id") REFERENCES "user"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "account" ADD CONSTRAINT "account_user_id_fkey" FOREIGN KEY ("user_id") REFERENCES "user"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "members" ADD CONSTRAINT "members_org_id_fkey" FOREIGN KEY ("org_id") REFERENCES "organizations"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "members" ADD CONSTRAINT "members_user_id_fkey" FOREIGN KEY ("user_id") REFERENCES "user"("id") ON DELETE SET NULL ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "user_roles" ADD CONSTRAINT "user_roles_org_id_fkey" FOREIGN KEY ("org_id") REFERENCES "organizations"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "user_roles" ADD CONSTRAINT "user_roles_user_id_fkey" FOREIGN KEY ("user_id") REFERENCES "user"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "news" ADD CONSTRAINT "news_org_id_fkey" FOREIGN KEY ("org_id") REFERENCES "organizations"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "news" ADD CONSTRAINT "news_author_id_fkey" FOREIGN KEY ("author_id") REFERENCES "members"("id") ON DELETE SET NULL ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "news_reads" ADD CONSTRAINT "news_reads_news_id_fkey" FOREIGN KEY ("news_id") REFERENCES "news"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "news_attachments" ADD CONSTRAINT "news_attachments_news_id_fkey" FOREIGN KEY ("news_id") REFERENCES "news"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "stellen" ADD CONSTRAINT "stellen_org_id_fkey" FOREIGN KEY ("org_id") REFERENCES "organizations"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "stellen" ADD CONSTRAINT "stellen_member_id_fkey" FOREIGN KEY ("member_id") REFERENCES "members"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "termine" ADD CONSTRAINT "termine_org_id_fkey" FOREIGN KEY ("org_id") REFERENCES "organizations"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "termin_anmeldungen" ADD CONSTRAINT "termin_anmeldungen_termin_id_fkey" FOREIGN KEY ("termin_id") REFERENCES "termine"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "termin_anmeldungen" ADD CONSTRAINT "termin_anmeldungen_member_id_fkey" FOREIGN KEY ("member_id") REFERENCES "members"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "conversation_members" ADD CONSTRAINT "conversation_members_conversation_id_fkey" FOREIGN KEY ("conversation_id") REFERENCES "conversations"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "conversation_members" ADD CONSTRAINT "conversation_members_member_id_fkey" FOREIGN KEY ("member_id") REFERENCES "members"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "messages" ADD CONSTRAINT "messages_conversation_id_fkey" FOREIGN KEY ("conversation_id") REFERENCES "conversations"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "messages" ADD CONSTRAINT "messages_sender_id_fkey" FOREIGN KEY ("sender_id") REFERENCES "members"("id") ON DELETE CASCADE ON UPDATE CASCADE;

View File

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

View File

@@ -1,14 +1,12 @@
// InnungsApp — Prisma Schema
// Stack: SQLite + Prisma ORM + better-auth
// Note: SQLite has no native enum support — enum fields are stored as String.
// Valid values are enforced at the application layer (Zod).
// Stack: PostgreSQL + Prisma ORM + better-auth
generator client {
provider = "prisma-client-js"
}
datasource db {
provider = "sqlite"
provider = "postgresql"
url = env("DATABASE_URL")
}
@@ -108,8 +106,8 @@ model Organization {
landingPageButtonText String? @map("landing_page_button_text")
landingPageHeroImage String? @map("landing_page_hero_image")
landingPageHeroOverlayOpacity Int? @default(50) @map("landing_page_hero_overlay_opacity")
landingPageFeatures String? @map("landing_page_features")
landingPageFooter String? @map("landing_page_footer")
landingPageFeatures Json? @map("landing_page_features") @db.JsonB
landingPageFooter Json? @map("landing_page_footer") @db.JsonB
appStoreUrl String? @map("app_store_url")
playStoreUrl String? @map("play_store_url")
aiEnabled Boolean @default(false) @map("ai_enabled")

View File

@@ -16,32 +16,53 @@ async function hashPassword(password) {
return `${salt}:${key.toString('hex')}`
}
async function main() {
const email = 'superadmin@innungsapp.de'
const password = 'demo1234'
const userId = 'superadmin-user-id'
const accountId = 'superadmin-account-id'
function getEnv(name) {
return (process.env[name] || '').trim()
}
console.log('Seeding superadmin...')
async function main() {
const email = (getEnv('SUPERADMIN_EMAIL') || 'superadmin@innungsapp.de').toLowerCase()
const name = getEnv('SUPERADMIN_NAME') || 'Super Admin'
const userId = getEnv('SUPERADMIN_USER_ID') || 'superadmin-user-id'
const accountId = getEnv('SUPERADMIN_ACCOUNT_ID') || 'superadmin-account-id'
let password = getEnv('SUPERADMIN_PASSWORD')
if (!password) {
if (process.env.NODE_ENV === 'production') {
throw new Error('SUPERADMIN_PASSWORD must be set in production.')
}
password = 'demo1234'
console.warn('SUPERADMIN_PASSWORD not set. Using development fallback password.')
}
console.log(`Seeding superadmin user for ${email}...`)
const hash = await hashPassword(password)
const user = await prisma.user.upsert({
where: { email },
update: {
name: 'Super Admin',
name,
emailVerified: true,
role: 'admin',
},
create: {
id: userId,
name: 'Super Admin',
name,
email,
emailVerified: true,
role: 'admin',
},
})
await prisma.account.upsert({
where: { id: accountId },
update: { password: hash },
update: {
accountId: user.id,
providerId: 'credential',
userId: user.id,
password: hash,
},
create: {
id: accountId,
accountId: user.id,

View File

@@ -13,27 +13,55 @@ async function hashPassword(password: string): Promise<string> {
return `${salt}:${key.toString('hex')}`
}
function getEnv(name: string): string {
return (process.env[name] ?? '').trim()
}
async function main() {
console.log('Seeding superadmin...')
const hash = await hashPassword('demo1234')
console.log('Hash generated.')
const email = getEnv('SUPERADMIN_EMAIL').toLowerCase() || 'superadmin@innungsapp.de'
const name = getEnv('SUPERADMIN_NAME') || 'Super Admin'
const userId = getEnv('SUPERADMIN_USER_ID') || 'superadmin-user-id'
const accountId = getEnv('SUPERADMIN_ACCOUNT_ID') || 'superadmin-account-id'
let password = getEnv('SUPERADMIN_PASSWORD')
if (!password) {
if (process.env.NODE_ENV === 'production') {
throw new Error('SUPERADMIN_PASSWORD must be set in production.')
}
password = 'demo1234'
console.warn('SUPERADMIN_PASSWORD not set. Using development fallback password.')
}
console.log(`Seeding superadmin user for ${email}...`)
const hash = await hashPassword(password)
const superAdminUser = await prisma.user.upsert({
where: { email: 'superadmin@innungsapp.de' },
update: {},
create: {
id: 'superadmin-user-id',
name: 'Super Admin',
email: 'superadmin@innungsapp.de',
where: { email },
update: {
name,
emailVerified: true,
role: 'admin',
},
create: {
id: userId,
name,
email,
emailVerified: true,
role: 'admin',
},
})
await prisma.account.upsert({
where: { id: 'superadmin-account-id' },
update: { password: hash },
where: { id: accountId },
update: {
accountId: superAdminUser.id,
userId: superAdminUser.id,
providerId: 'credential',
password: hash,
},
create: {
id: 'superadmin-account-id',
id: accountId,
accountId: superAdminUser.id,
providerId: 'credential',
userId: superAdminUser.id,
@@ -41,7 +69,7 @@ async function main() {
},
})
console.log('Done! Login: superadmin@innungsapp.de / demo1234')
console.log(`Done. Login: ${email} / ${password}`)
}
main()

View File

@@ -1,2 +1,3 @@
export { prisma } from './lib/prisma'
export { Prisma } from '@prisma/client'
export * from './types/index'