Rebuild as InnungsApp project: replace stadtwerke analysis with full documentation
- PRD: vollständige Produktspezifikation (5 Module, Scope, Akzeptanzkriterien) - ARCHITECTURE: Tech Stack, Ordnerstruktur, Multi-Tenancy, Push, Kosten - DATABASE_SCHEMA: Vollständiges SQL-Schema mit RLS Policies und Views - USER_STORIES: 40+ Stories nach Rolle (Admin, Mitglied, Azubi, Obermeister) - PERSONAS: 5 detaillierte Nutzerprofile mit Alltag, Zitaten und Erwartungen - BUSINESS_MODEL: Preistabellen, Unit Economics, Revenue-Projektionen, Distribution - ROADMAP: 6 Phasen, Sprint-Planung, Meilensteine und KPIs - COMPETITIVE_ANALYSIS: Wettbewerbsmatrix, USPs, Preispositionierung - API_DESIGN: Supabase Query Patterns, Edge Functions, Realtime Subscriptions - ONBOARDING_FLOWS: 7 User Flows von Setup bis Fehlerfall - GTM_STRATEGY: 3-Phasen-Vertrieb, Outreach-Sequenz, Einwandbehandlung - AZUBI_MODULE: Video-Feed, 1-Click-Apply, Chat, Berichtsheft, Quiz - DSGVO_KONZEPT: Rechtsgrundlagen, TOMs, AVV, Minderjährige, Incident Response - FEATURES_BACKLOG: 72 Features nach MoSCoW + Technische Schulden Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
310
innungsapp/packages/shared/prisma/schema.prisma
Normal file
310
innungsapp/packages/shared/prisma/schema.prisma
Normal file
@@ -0,0 +1,310 @@
|
||||
// InnungsApp — Prisma Schema
|
||||
// Stack: PostgreSQL + Prisma ORM + better-auth
|
||||
|
||||
generator client {
|
||||
provider = "prisma-client-js"
|
||||
}
|
||||
|
||||
datasource db {
|
||||
provider = "postgresql"
|
||||
url = env("DATABASE_URL")
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// BETTER-AUTH TABLES
|
||||
// =============================================
|
||||
|
||||
model User {
|
||||
id String @id
|
||||
name String
|
||||
email String @unique
|
||||
emailVerified Boolean @default(false) @map("email_verified")
|
||||
image String?
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
updatedAt DateTime @updatedAt @map("updated_at")
|
||||
|
||||
// better-auth admin plugin fields
|
||||
role String?
|
||||
banned Boolean? @default(false)
|
||||
banReason String? @map("ban_reason")
|
||||
banExpires DateTime? @map("ban_expires")
|
||||
|
||||
// App relations
|
||||
sessions Session[]
|
||||
accounts Account[]
|
||||
member Member?
|
||||
userRoles UserRole[]
|
||||
|
||||
@@map("user")
|
||||
}
|
||||
|
||||
model Session {
|
||||
id String @id
|
||||
expiresAt DateTime @map("expires_at")
|
||||
token String @unique
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
updatedAt DateTime @updatedAt @map("updated_at")
|
||||
ipAddress String? @map("ip_address")
|
||||
userAgent String? @map("user_agent")
|
||||
userId String @map("user_id")
|
||||
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
||||
|
||||
@@map("session")
|
||||
}
|
||||
|
||||
model Account {
|
||||
id String @id
|
||||
accountId String @map("account_id")
|
||||
providerId String @map("provider_id")
|
||||
userId String @map("user_id")
|
||||
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
||||
accessToken String? @map("access_token")
|
||||
refreshToken String? @map("refresh_token")
|
||||
idToken String? @map("id_token")
|
||||
accessTokenExpiresAt DateTime? @map("access_token_expires_at")
|
||||
refreshTokenExpiresAt DateTime? @map("refresh_token_expires_at")
|
||||
scope String?
|
||||
password String?
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
updatedAt DateTime @updatedAt @map("updated_at")
|
||||
|
||||
@@map("account")
|
||||
}
|
||||
|
||||
model Verification {
|
||||
id String @id
|
||||
identifier String
|
||||
value String
|
||||
expiresAt DateTime @map("expires_at")
|
||||
createdAt DateTime? @default(now()) @map("created_at")
|
||||
updatedAt DateTime? @updatedAt @map("updated_at")
|
||||
|
||||
@@map("verification")
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// ORGANIZATIONS
|
||||
// =============================================
|
||||
|
||||
model Organization {
|
||||
id String @id @default(uuid())
|
||||
name String
|
||||
slug String @unique
|
||||
plan Plan @default(pilot)
|
||||
logoUrl String? @map("logo_url")
|
||||
primaryColor String @default("#E63946") @map("primary_color")
|
||||
contactEmail String? @map("contact_email")
|
||||
avvAccepted Boolean @default(false) @map("avv_accepted")
|
||||
avvAcceptedAt DateTime? @map("avv_accepted_at")
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
|
||||
members Member[]
|
||||
userRoles UserRole[]
|
||||
news News[]
|
||||
stellen Stelle[]
|
||||
termine Termin[]
|
||||
|
||||
@@map("organizations")
|
||||
}
|
||||
|
||||
enum Plan {
|
||||
pilot
|
||||
standard
|
||||
pro
|
||||
verband
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// MEMBERS
|
||||
// =============================================
|
||||
|
||||
model Member {
|
||||
id String @id @default(uuid())
|
||||
orgId String @map("org_id")
|
||||
userId String? @unique @map("user_id") // NULL until magic-link clicked
|
||||
name String
|
||||
betrieb String
|
||||
sparte String
|
||||
ort String
|
||||
telefon String?
|
||||
email String
|
||||
status MemberStatus @default(aktiv)
|
||||
istAusbildungsbetrieb Boolean @default(false) @map("ist_ausbildungsbetrieb")
|
||||
seit Int?
|
||||
avatarUrl String? @map("avatar_url")
|
||||
pushToken String? @map("push_token")
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
updatedAt DateTime @updatedAt @map("updated_at")
|
||||
|
||||
org Organization @relation(fields: [orgId], references: [id], onDelete: Cascade)
|
||||
user User? @relation(fields: [userId], references: [id], onDelete: SetNull)
|
||||
newsAuthored News[] @relation("NewsAuthor")
|
||||
stellen Stelle[]
|
||||
terminAnmeldungen TerminAnmeldung[]
|
||||
|
||||
@@index([orgId])
|
||||
@@index([status])
|
||||
@@map("members")
|
||||
}
|
||||
|
||||
enum MemberStatus {
|
||||
aktiv
|
||||
ruhend
|
||||
ausgetreten
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// USER ROLES (multi-tenancy)
|
||||
// =============================================
|
||||
|
||||
model UserRole {
|
||||
id String @id @default(uuid())
|
||||
orgId String @map("org_id")
|
||||
userId String @map("user_id")
|
||||
role OrgRole
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
|
||||
org Organization @relation(fields: [orgId], references: [id], onDelete: Cascade)
|
||||
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
||||
|
||||
@@unique([orgId, userId])
|
||||
@@map("user_roles")
|
||||
}
|
||||
|
||||
enum OrgRole {
|
||||
admin
|
||||
member
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// NEWS
|
||||
// =============================================
|
||||
|
||||
model News {
|
||||
id String @id @default(uuid())
|
||||
orgId String @map("org_id")
|
||||
authorId String? @map("author_id")
|
||||
title String
|
||||
body String // Markdown
|
||||
kategorie NewsKategorie
|
||||
publishedAt DateTime? @map("published_at") // NULL = Entwurf
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
|
||||
org Organization @relation(fields: [orgId], references: [id], onDelete: Cascade)
|
||||
author Member? @relation("NewsAuthor", fields: [authorId], references: [id], onDelete: SetNull)
|
||||
reads NewsRead[]
|
||||
attachments NewsAttachment[]
|
||||
|
||||
@@index([orgId])
|
||||
@@index([publishedAt])
|
||||
@@map("news")
|
||||
}
|
||||
|
||||
enum NewsKategorie {
|
||||
Wichtig
|
||||
Pruefung
|
||||
Foerderung
|
||||
Veranstaltung
|
||||
Allgemein
|
||||
}
|
||||
|
||||
model NewsRead {
|
||||
id String @id @default(uuid())
|
||||
newsId String @map("news_id")
|
||||
userId String @map("user_id")
|
||||
readAt DateTime @default(now()) @map("read_at")
|
||||
|
||||
news News @relation(fields: [newsId], references: [id], onDelete: Cascade)
|
||||
|
||||
@@unique([newsId, userId])
|
||||
@@map("news_reads")
|
||||
}
|
||||
|
||||
model NewsAttachment {
|
||||
id String @id @default(uuid())
|
||||
newsId String @map("news_id")
|
||||
name String
|
||||
storagePath String @map("storage_path")
|
||||
mimeType String? @map("mime_type")
|
||||
sizeBytes Int? @map("size_bytes")
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
|
||||
news News @relation(fields: [newsId], references: [id], onDelete: Cascade)
|
||||
|
||||
@@map("news_attachments")
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// STELLENANGEBOTE (Lehrlingsbörse)
|
||||
// =============================================
|
||||
|
||||
model Stelle {
|
||||
id String @id @default(uuid())
|
||||
orgId String @map("org_id")
|
||||
memberId String @map("member_id")
|
||||
sparte String
|
||||
stellenAnz Int @default(1) @map("stellen_anz")
|
||||
verguetung String? // "600-800 € / Monat"
|
||||
lehrjahr String? // "1. Lehrjahr" | "beliebig"
|
||||
beschreibung String?
|
||||
kontaktEmail String @map("kontakt_email")
|
||||
kontaktName String? @map("kontakt_name")
|
||||
aktiv Boolean @default(true)
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
updatedAt DateTime @updatedAt @map("updated_at")
|
||||
|
||||
org Organization @relation(fields: [orgId], references: [id], onDelete: Cascade)
|
||||
member Member @relation(fields: [memberId], references: [id], onDelete: Cascade)
|
||||
|
||||
@@index([orgId])
|
||||
@@index([aktiv])
|
||||
@@map("stellen")
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// TERMINE
|
||||
// =============================================
|
||||
|
||||
model Termin {
|
||||
id String @id @default(uuid())
|
||||
orgId String @map("org_id")
|
||||
titel String
|
||||
datum DateTime @db.Date
|
||||
uhrzeit String? // stored as "HH:MM"
|
||||
endeDatum DateTime? @map("ende_datum") @db.Date
|
||||
endeUhrzeit String? @map("ende_uhrzeit")
|
||||
ort String?
|
||||
adresse String?
|
||||
typ TerminTyp
|
||||
beschreibung String?
|
||||
maxTeilnehmer Int? @map("max_teilnehmer") // NULL = unbegrenzt
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
|
||||
org Organization @relation(fields: [orgId], references: [id], onDelete: Cascade)
|
||||
anmeldungen TerminAnmeldung[]
|
||||
|
||||
@@index([orgId])
|
||||
@@index([datum])
|
||||
@@map("termine")
|
||||
}
|
||||
|
||||
enum TerminTyp {
|
||||
Pruefung
|
||||
Versammlung
|
||||
Kurs
|
||||
Event
|
||||
Sonstiges
|
||||
}
|
||||
|
||||
model TerminAnmeldung {
|
||||
id String @id @default(uuid())
|
||||
terminId String @map("termin_id")
|
||||
memberId String @map("member_id")
|
||||
angemeldetAt DateTime @default(now()) @map("angemeldet_at")
|
||||
|
||||
termin Termin @relation(fields: [terminId], references: [id], onDelete: Cascade)
|
||||
member Member @relation(fields: [memberId], references: [id], onDelete: Cascade)
|
||||
|
||||
@@unique([terminId, memberId])
|
||||
@@map("termin_anmeldungen")
|
||||
}
|
||||
209
innungsapp/packages/shared/prisma/seed.ts
Normal file
209
innungsapp/packages/shared/prisma/seed.ts
Normal file
@@ -0,0 +1,209 @@
|
||||
import { PrismaClient } from '@prisma/client'
|
||||
|
||||
const prisma = new PrismaClient()
|
||||
|
||||
async function main() {
|
||||
console.log('Seeding database...')
|
||||
|
||||
// Create demo organization
|
||||
const org = await prisma.organization.upsert({
|
||||
where: { slug: 'innung-elektro-stuttgart' },
|
||||
update: {},
|
||||
create: {
|
||||
name: 'Innung Elektrotechnik Stuttgart',
|
||||
slug: 'innung-elektro-stuttgart',
|
||||
plan: 'pilot',
|
||||
primaryColor: '#E63946',
|
||||
contactEmail: 'kontakt@innung-elektro-stuttgart.de',
|
||||
},
|
||||
})
|
||||
|
||||
console.log(`Created organization: ${org.name}`)
|
||||
|
||||
// Create demo admin user (better-auth manages sessions, we just pre-create the role)
|
||||
// In production: use the invite flow via the admin dashboard
|
||||
const adminUser = await prisma.user.upsert({
|
||||
where: { email: 'admin@demo.de' },
|
||||
update: {},
|
||||
create: {
|
||||
id: 'demo-admin-user-id',
|
||||
name: 'Demo Admin',
|
||||
email: 'admin@demo.de',
|
||||
emailVerified: true,
|
||||
},
|
||||
})
|
||||
|
||||
await prisma.userRole.upsert({
|
||||
where: { orgId_userId: { orgId: org.id, userId: adminUser.id } },
|
||||
update: {},
|
||||
create: {
|
||||
orgId: org.id,
|
||||
userId: adminUser.id,
|
||||
role: 'admin',
|
||||
},
|
||||
})
|
||||
|
||||
// Create demo admin member record
|
||||
const adminMember = await prisma.member.upsert({
|
||||
where: { userId: adminUser.id },
|
||||
update: {},
|
||||
create: {
|
||||
orgId: org.id,
|
||||
userId: adminUser.id,
|
||||
name: 'Demo Admin',
|
||||
betrieb: 'Innungsgeschäftsstelle',
|
||||
sparte: 'Elektrotechnik',
|
||||
ort: 'Stuttgart',
|
||||
email: 'admin@demo.de',
|
||||
status: 'aktiv',
|
||||
},
|
||||
})
|
||||
|
||||
// Create demo members
|
||||
const demoMembers = [
|
||||
{
|
||||
name: 'Klaus Müller',
|
||||
betrieb: 'Elektro Müller GmbH',
|
||||
sparte: 'Elektrotechnik',
|
||||
ort: 'Stuttgart',
|
||||
telefon: '+49 711 123456',
|
||||
email: 'mueller@elektro-mueller.de',
|
||||
istAusbildungsbetrieb: true,
|
||||
seit: 2015,
|
||||
},
|
||||
{
|
||||
name: 'Maria Schmidt',
|
||||
betrieb: 'Schmidt Elektrik',
|
||||
sparte: 'Elektrotechnik',
|
||||
ort: 'Ludwigsburg',
|
||||
telefon: '+49 7141 987654',
|
||||
email: 'schmidt@schmidt-elektrik.de',
|
||||
istAusbildungsbetrieb: false,
|
||||
seit: 2018,
|
||||
},
|
||||
{
|
||||
name: 'Thomas Weber',
|
||||
betrieb: 'Weber & Söhne Elektro',
|
||||
sparte: 'Informationstechnik',
|
||||
ort: 'Esslingen',
|
||||
telefon: '+49 711 555123',
|
||||
email: 'weber@weber-elektro.de',
|
||||
istAusbildungsbetrieb: true,
|
||||
seit: 2012,
|
||||
},
|
||||
]
|
||||
|
||||
const createdMembers = []
|
||||
for (const m of demoMembers) {
|
||||
const member = await prisma.member.create({
|
||||
data: { orgId: org.id, status: 'aktiv', ...m },
|
||||
})
|
||||
createdMembers.push(member)
|
||||
}
|
||||
console.log(`Created ${createdMembers.length} demo members`)
|
||||
|
||||
// Create demo news
|
||||
const news1 = await prisma.news.create({
|
||||
data: {
|
||||
orgId: org.id,
|
||||
authorId: adminMember.id,
|
||||
title: 'Wichtige Änderungen bei der Gesellenprüfung 2025',
|
||||
body: `## Änderungen ab Herbst 2025
|
||||
|
||||
Die Prüfungsordnung wurde angepasst. Folgende Änderungen sind zu beachten:
|
||||
|
||||
- **Prüfungsteil A** (Gesellenstück) wird auf 2 Tage ausgeweitet
|
||||
- Neue digitale Komponenten in Prüfungsteil B
|
||||
- Anmeldeschluss ist der **15. September 2025**
|
||||
|
||||
Weitere Details entnehmen Sie dem beigefügten Rundschreiben.`,
|
||||
kategorie: 'Pruefung',
|
||||
publishedAt: new Date(),
|
||||
},
|
||||
})
|
||||
|
||||
await prisma.news.create({
|
||||
data: {
|
||||
orgId: org.id,
|
||||
authorId: adminMember.id,
|
||||
title: 'Förderung für Ausbildungsbetriebe: Jetzt beantragen!',
|
||||
body: `## Neue Fördergelder verfügbar
|
||||
|
||||
Das Land Baden-Württemberg stellt für das Jahr 2025 zusätzliche Fördermittel für Ausbildungsbetriebe bereit.
|
||||
|
||||
**Förderhöhe:** Bis zu 3.000 € pro Auszubildenden
|
||||
|
||||
**Voraussetzungen:**
|
||||
- Mitglied in einer anerkannten Innung
|
||||
- Erstauszubildende oder benachteiligte Jugendliche
|
||||
|
||||
Anträge bis **31. März 2025** einreichen.`,
|
||||
kategorie: 'Foerderung',
|
||||
publishedAt: new Date(Date.now() - 7 * 24 * 60 * 60 * 1000),
|
||||
},
|
||||
})
|
||||
|
||||
console.log('Created demo news articles')
|
||||
|
||||
// Create demo Termin
|
||||
await prisma.termin.create({
|
||||
data: {
|
||||
orgId: org.id,
|
||||
titel: 'Herbst-Gesellenprüfung 2025',
|
||||
datum: new Date('2025-11-15'),
|
||||
uhrzeit: '08:00',
|
||||
endeDatum: new Date('2025-11-16'),
|
||||
endeUhrzeit: '17:00',
|
||||
ort: 'Berufsschule Stuttgart-Mitte',
|
||||
adresse: 'Neckarstraße 22, 70190 Stuttgart',
|
||||
typ: 'Pruefung',
|
||||
beschreibung: 'Praktische und theoretische Gesellenprüfung für Elektrotechniker.',
|
||||
maxTeilnehmer: 30,
|
||||
},
|
||||
})
|
||||
|
||||
await prisma.termin.create({
|
||||
data: {
|
||||
orgId: org.id,
|
||||
titel: 'Innungsversammlung Winter 2025',
|
||||
datum: new Date('2025-12-03'),
|
||||
uhrzeit: '19:00',
|
||||
endeUhrzeit: '21:00',
|
||||
ort: 'Gasthof Zum Lamm',
|
||||
adresse: 'Marktplatz 5, 70173 Stuttgart',
|
||||
typ: 'Versammlung',
|
||||
beschreibung: 'Jährliche Winterversammlung mit Jahresrückblick und Vorstandswahlen.',
|
||||
},
|
||||
})
|
||||
|
||||
console.log('Created demo Termine')
|
||||
|
||||
// Create demo Stelle
|
||||
await prisma.stelle.create({
|
||||
data: {
|
||||
orgId: org.id,
|
||||
memberId: createdMembers[0].id,
|
||||
sparte: 'Elektrotechnik',
|
||||
stellenAnz: 2,
|
||||
verguetung: '620-750 € / Monat',
|
||||
lehrjahr: 'beliebig',
|
||||
beschreibung:
|
||||
'Wir suchen motivierte Auszubildende für unser erfolgreiches Elektroinstallationsunternehmen. Moderner Fuhrpark, faire Vergütung, Übernahmechancen.',
|
||||
kontaktEmail: 'mueller@elektro-mueller.de',
|
||||
kontaktName: 'Klaus Müller',
|
||||
aktiv: true,
|
||||
},
|
||||
})
|
||||
|
||||
console.log('Created demo Stelle')
|
||||
console.log('Seeding complete!')
|
||||
}
|
||||
|
||||
main()
|
||||
.catch((e) => {
|
||||
console.error(e)
|
||||
process.exit(1)
|
||||
})
|
||||
.finally(async () => {
|
||||
await prisma.$disconnect()
|
||||
})
|
||||
Reference in New Issue
Block a user