log
This commit is contained in:
Binary file not shown.
@@ -1,350 +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;
|
||||
-- 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;
|
||||
|
||||
@@ -1,3 +1,3 @@
|
||||
# Please do not edit this file manually
|
||||
# It should be added in your version-control system (i.e. Git)
|
||||
# Please do not edit this file manually
|
||||
# It should be added in your version-control system (i.e. Git)
|
||||
provider = "postgresql"
|
||||
@@ -1,337 +1,337 @@
|
||||
// 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")
|
||||
|
||||
// Password management
|
||||
mustChangePassword Boolean? @default(false) @map("must_change_password")
|
||||
|
||||
// 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 String @default("pilot") // pilot | standard | pro | verband
|
||||
logoUrl String? @map("logo_url")
|
||||
primaryColor String @default("#E63946") @map("primary_color")
|
||||
secondaryColor String? @map("secondary_color")
|
||||
contactEmail String? @map("contact_email")
|
||||
avvAccepted Boolean @default(false) @map("avv_accepted")
|
||||
avvAcceptedAt DateTime? @map("avv_accepted_at")
|
||||
landingPageTitle String? @map("landing_page_title")
|
||||
landingPageText String? @map("landing_page_text")
|
||||
landingPageSectionTitle String? @map("landing_page_section_title")
|
||||
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 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")
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
|
||||
members Member[]
|
||||
userRoles UserRole[]
|
||||
news News[]
|
||||
stellen Stelle[]
|
||||
termine Termin[]
|
||||
|
||||
@@map("organizations")
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// 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 String @default("aktiv") // aktiv | ruhend | ausgetreten
|
||||
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[]
|
||||
sentMessages Message[]
|
||||
conversationMembers ConversationMember[]
|
||||
|
||||
@@index([orgId])
|
||||
@@index([status])
|
||||
@@map("members")
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// USER ROLES (multi-tenancy)
|
||||
// =============================================
|
||||
|
||||
model UserRole {
|
||||
id String @id @default(uuid())
|
||||
orgId String @map("org_id")
|
||||
userId String @map("user_id")
|
||||
role String // admin | member
|
||||
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")
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// NEWS
|
||||
// =============================================
|
||||
|
||||
model News {
|
||||
id String @id @default(uuid())
|
||||
orgId String @map("org_id")
|
||||
authorId String? @map("author_id")
|
||||
title String
|
||||
body String // Markdown
|
||||
kategorie String // Wichtig | Pruefung | Foerderung | Veranstaltung | Allgemein
|
||||
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")
|
||||
}
|
||||
|
||||
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
|
||||
uhrzeit String? // stored as "HH:MM"
|
||||
endeDatum DateTime? @map("ende_datum")
|
||||
endeUhrzeit String? @map("ende_uhrzeit")
|
||||
ort String?
|
||||
adresse String?
|
||||
typ String // Pruefung | Versammlung | Kurs | Event | Sonstiges
|
||||
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")
|
||||
}
|
||||
|
||||
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")
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// DIREKTNACHRICHTEN (Chat)
|
||||
// =============================================
|
||||
|
||||
model Conversation {
|
||||
id String @id @default(uuid())
|
||||
orgId String @map("org_id")
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
updatedAt DateTime @updatedAt @map("updated_at")
|
||||
|
||||
members ConversationMember[]
|
||||
messages Message[]
|
||||
|
||||
@@index([orgId])
|
||||
@@map("conversations")
|
||||
}
|
||||
|
||||
model ConversationMember {
|
||||
id String @id @default(uuid())
|
||||
conversationId String @map("conversation_id")
|
||||
memberId String @map("member_id")
|
||||
lastReadAt DateTime? @map("last_read_at")
|
||||
|
||||
conversation Conversation @relation(fields: [conversationId], references: [id], onDelete: Cascade)
|
||||
member Member @relation(fields: [memberId], references: [id], onDelete: Cascade)
|
||||
|
||||
@@unique([conversationId, memberId])
|
||||
@@map("conversation_members")
|
||||
}
|
||||
|
||||
model Message {
|
||||
id String @id @default(uuid())
|
||||
conversationId String @map("conversation_id")
|
||||
senderId String @map("sender_id")
|
||||
body String
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
|
||||
conversation Conversation @relation(fields: [conversationId], references: [id], onDelete: Cascade)
|
||||
sender Member @relation(fields: [senderId], references: [id], onDelete: Cascade)
|
||||
|
||||
@@index([conversationId])
|
||||
@@map("messages")
|
||||
}
|
||||
// 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")
|
||||
|
||||
// Password management
|
||||
mustChangePassword Boolean? @default(false) @map("must_change_password")
|
||||
|
||||
// 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 String @default("pilot") // pilot | standard | pro | verband
|
||||
logoUrl String? @map("logo_url")
|
||||
primaryColor String @default("#E63946") @map("primary_color")
|
||||
secondaryColor String? @map("secondary_color")
|
||||
contactEmail String? @map("contact_email")
|
||||
avvAccepted Boolean @default(false) @map("avv_accepted")
|
||||
avvAcceptedAt DateTime? @map("avv_accepted_at")
|
||||
landingPageTitle String? @map("landing_page_title")
|
||||
landingPageText String? @map("landing_page_text")
|
||||
landingPageSectionTitle String? @map("landing_page_section_title")
|
||||
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 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")
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
|
||||
members Member[]
|
||||
userRoles UserRole[]
|
||||
news News[]
|
||||
stellen Stelle[]
|
||||
termine Termin[]
|
||||
|
||||
@@map("organizations")
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// 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 String @default("aktiv") // aktiv | ruhend | ausgetreten
|
||||
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[]
|
||||
sentMessages Message[]
|
||||
conversationMembers ConversationMember[]
|
||||
|
||||
@@index([orgId])
|
||||
@@index([status])
|
||||
@@map("members")
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// USER ROLES (multi-tenancy)
|
||||
// =============================================
|
||||
|
||||
model UserRole {
|
||||
id String @id @default(uuid())
|
||||
orgId String @map("org_id")
|
||||
userId String @map("user_id")
|
||||
role String // admin | member
|
||||
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")
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// NEWS
|
||||
// =============================================
|
||||
|
||||
model News {
|
||||
id String @id @default(uuid())
|
||||
orgId String @map("org_id")
|
||||
authorId String? @map("author_id")
|
||||
title String
|
||||
body String // Markdown
|
||||
kategorie String // Wichtig | Pruefung | Foerderung | Veranstaltung | Allgemein
|
||||
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")
|
||||
}
|
||||
|
||||
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
|
||||
uhrzeit String? // stored as "HH:MM"
|
||||
endeDatum DateTime? @map("ende_datum")
|
||||
endeUhrzeit String? @map("ende_uhrzeit")
|
||||
ort String?
|
||||
adresse String?
|
||||
typ String // Pruefung | Versammlung | Kurs | Event | Sonstiges
|
||||
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")
|
||||
}
|
||||
|
||||
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")
|
||||
}
|
||||
|
||||
// =============================================
|
||||
// DIREKTNACHRICHTEN (Chat)
|
||||
// =============================================
|
||||
|
||||
model Conversation {
|
||||
id String @id @default(uuid())
|
||||
orgId String @map("org_id")
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
updatedAt DateTime @updatedAt @map("updated_at")
|
||||
|
||||
members ConversationMember[]
|
||||
messages Message[]
|
||||
|
||||
@@index([orgId])
|
||||
@@map("conversations")
|
||||
}
|
||||
|
||||
model ConversationMember {
|
||||
id String @id @default(uuid())
|
||||
conversationId String @map("conversation_id")
|
||||
memberId String @map("member_id")
|
||||
lastReadAt DateTime? @map("last_read_at")
|
||||
|
||||
conversation Conversation @relation(fields: [conversationId], references: [id], onDelete: Cascade)
|
||||
member Member @relation(fields: [memberId], references: [id], onDelete: Cascade)
|
||||
|
||||
@@unique([conversationId, memberId])
|
||||
@@map("conversation_members")
|
||||
}
|
||||
|
||||
model Message {
|
||||
id String @id @default(uuid())
|
||||
conversationId String @map("conversation_id")
|
||||
senderId String @map("sender_id")
|
||||
body String
|
||||
createdAt DateTime @default(now()) @map("created_at")
|
||||
|
||||
conversation Conversation @relation(fields: [conversationId], references: [id], onDelete: Cascade)
|
||||
sender Member @relation(fields: [senderId], references: [id], onDelete: Cascade)
|
||||
|
||||
@@index([conversationId])
|
||||
@@map("messages")
|
||||
}
|
||||
|
||||
@@ -1,85 +1,85 @@
|
||||
const { PrismaClient } = require('@prisma/client')
|
||||
const { randomBytes, scrypt } = require('crypto')
|
||||
const { promisify } = require('util')
|
||||
|
||||
const prisma = new PrismaClient()
|
||||
const scryptAsync = promisify(scrypt)
|
||||
|
||||
async function hashPassword(password) {
|
||||
const salt = randomBytes(16).toString('hex')
|
||||
const key = await scryptAsync(password.normalize('NFKC'), salt, 64, {
|
||||
N: 16384,
|
||||
r: 16,
|
||||
p: 1,
|
||||
maxmem: 128 * 16384 * 16 * 2,
|
||||
})
|
||||
return `${salt}:${key.toString('hex')}`
|
||||
}
|
||||
|
||||
function getEnv(name) {
|
||||
return (process.env[name] || '').trim()
|
||||
}
|
||||
|
||||
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,
|
||||
emailVerified: true,
|
||||
role: 'admin',
|
||||
},
|
||||
create: {
|
||||
id: userId,
|
||||
name,
|
||||
email,
|
||||
emailVerified: true,
|
||||
role: 'admin',
|
||||
},
|
||||
})
|
||||
|
||||
await prisma.account.upsert({
|
||||
where: { id: accountId },
|
||||
update: {
|
||||
accountId: user.id,
|
||||
providerId: 'credential',
|
||||
userId: user.id,
|
||||
password: hash,
|
||||
},
|
||||
create: {
|
||||
id: accountId,
|
||||
accountId: user.id,
|
||||
providerId: 'credential',
|
||||
userId: user.id,
|
||||
password: hash,
|
||||
},
|
||||
})
|
||||
|
||||
console.log(`Done. Login: ${email} / ${password}`)
|
||||
}
|
||||
|
||||
main()
|
||||
.catch((error) => {
|
||||
console.error(error)
|
||||
process.exit(1)
|
||||
})
|
||||
.finally(async () => {
|
||||
await prisma.$disconnect()
|
||||
})
|
||||
const { PrismaClient } = require('@prisma/client')
|
||||
const { randomBytes, scrypt } = require('crypto')
|
||||
const { promisify } = require('util')
|
||||
|
||||
const prisma = new PrismaClient()
|
||||
const scryptAsync = promisify(scrypt)
|
||||
|
||||
async function hashPassword(password) {
|
||||
const salt = randomBytes(16).toString('hex')
|
||||
const key = await scryptAsync(password.normalize('NFKC'), salt, 64, {
|
||||
N: 16384,
|
||||
r: 16,
|
||||
p: 1,
|
||||
maxmem: 128 * 16384 * 16 * 2,
|
||||
})
|
||||
return `${salt}:${key.toString('hex')}`
|
||||
}
|
||||
|
||||
function getEnv(name) {
|
||||
return (process.env[name] || '').trim()
|
||||
}
|
||||
|
||||
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,
|
||||
emailVerified: true,
|
||||
role: 'admin',
|
||||
},
|
||||
create: {
|
||||
id: userId,
|
||||
name,
|
||||
email,
|
||||
emailVerified: true,
|
||||
role: 'admin',
|
||||
},
|
||||
})
|
||||
|
||||
await prisma.account.upsert({
|
||||
where: { id: accountId },
|
||||
update: {
|
||||
accountId: user.id,
|
||||
providerId: 'credential',
|
||||
userId: user.id,
|
||||
password: hash,
|
||||
},
|
||||
create: {
|
||||
id: accountId,
|
||||
accountId: user.id,
|
||||
providerId: 'credential',
|
||||
userId: user.id,
|
||||
password: hash,
|
||||
},
|
||||
})
|
||||
|
||||
console.log(`Done. Login: ${email} / ${password}`)
|
||||
}
|
||||
|
||||
main()
|
||||
.catch((error) => {
|
||||
console.error(error)
|
||||
process.exit(1)
|
||||
})
|
||||
.finally(async () => {
|
||||
await prisma.$disconnect()
|
||||
})
|
||||
|
||||
@@ -1,82 +1,82 @@
|
||||
import { PrismaClient } from '@prisma/client'
|
||||
import { scrypt, randomBytes } from 'crypto'
|
||||
import { promisify } from 'util'
|
||||
|
||||
const scryptAsync = promisify(scrypt)
|
||||
const prisma = new PrismaClient()
|
||||
|
||||
async function hashPassword(password: string): Promise<string> {
|
||||
const salt = randomBytes(16).toString('hex')
|
||||
const key = await scryptAsync(password.normalize('NFKC'), salt, 64, {
|
||||
N: 16384, r: 16, p: 1, maxmem: 128 * 16384 * 16 * 2,
|
||||
}) as Buffer
|
||||
return `${salt}:${key.toString('hex')}`
|
||||
}
|
||||
|
||||
function getEnv(name: string): string {
|
||||
return (process.env[name] ?? '').trim()
|
||||
}
|
||||
|
||||
async function main() {
|
||||
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 },
|
||||
update: {
|
||||
name,
|
||||
emailVerified: true,
|
||||
role: 'admin',
|
||||
},
|
||||
create: {
|
||||
id: userId,
|
||||
name,
|
||||
email,
|
||||
emailVerified: true,
|
||||
role: 'admin',
|
||||
},
|
||||
})
|
||||
|
||||
await prisma.account.upsert({
|
||||
where: { id: accountId },
|
||||
update: {
|
||||
accountId: superAdminUser.id,
|
||||
userId: superAdminUser.id,
|
||||
providerId: 'credential',
|
||||
password: hash,
|
||||
},
|
||||
create: {
|
||||
id: accountId,
|
||||
accountId: superAdminUser.id,
|
||||
providerId: 'credential',
|
||||
userId: superAdminUser.id,
|
||||
password: hash,
|
||||
},
|
||||
})
|
||||
|
||||
console.log(`Done. Login: ${email} / ${password}`)
|
||||
}
|
||||
|
||||
main()
|
||||
.catch((e) => {
|
||||
console.error(e)
|
||||
process.exit(1)
|
||||
})
|
||||
.finally(async () => {
|
||||
await prisma.$disconnect()
|
||||
})
|
||||
import { PrismaClient } from '@prisma/client'
|
||||
import { scrypt, randomBytes } from 'crypto'
|
||||
import { promisify } from 'util'
|
||||
|
||||
const scryptAsync = promisify(scrypt)
|
||||
const prisma = new PrismaClient()
|
||||
|
||||
async function hashPassword(password: string): Promise<string> {
|
||||
const salt = randomBytes(16).toString('hex')
|
||||
const key = await scryptAsync(password.normalize('NFKC'), salt, 64, {
|
||||
N: 16384, r: 16, p: 1, maxmem: 128 * 16384 * 16 * 2,
|
||||
}) as Buffer
|
||||
return `${salt}:${key.toString('hex')}`
|
||||
}
|
||||
|
||||
function getEnv(name: string): string {
|
||||
return (process.env[name] ?? '').trim()
|
||||
}
|
||||
|
||||
async function main() {
|
||||
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 },
|
||||
update: {
|
||||
name,
|
||||
emailVerified: true,
|
||||
role: 'admin',
|
||||
},
|
||||
create: {
|
||||
id: userId,
|
||||
name,
|
||||
email,
|
||||
emailVerified: true,
|
||||
role: 'admin',
|
||||
},
|
||||
})
|
||||
|
||||
await prisma.account.upsert({
|
||||
where: { id: accountId },
|
||||
update: {
|
||||
accountId: superAdminUser.id,
|
||||
userId: superAdminUser.id,
|
||||
providerId: 'credential',
|
||||
password: hash,
|
||||
},
|
||||
create: {
|
||||
id: accountId,
|
||||
accountId: superAdminUser.id,
|
||||
providerId: 'credential',
|
||||
userId: superAdminUser.id,
|
||||
password: hash,
|
||||
},
|
||||
})
|
||||
|
||||
console.log(`Done. Login: ${email} / ${password}`)
|
||||
}
|
||||
|
||||
main()
|
||||
.catch((e) => {
|
||||
console.error(e)
|
||||
process.exit(1)
|
||||
})
|
||||
.finally(async () => {
|
||||
await prisma.$disconnect()
|
||||
})
|
||||
|
||||
Reference in New Issue
Block a user