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:
Timo Knuth
2026-02-18 19:03:37 +01:00
parent fc68285cf1
commit fca42db4d2
116 changed files with 9329 additions and 6479 deletions

View 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")
}

View 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()
})