feat: add competition heats system backend

- Add 3 new database tables: divisions, competition_types, event_user_heats
- Add seed data for 6 divisions (NEW, NOV, INT, ADV, ALL, CHA) and 2 competition types (J&J, STR)
- Add API endpoints for divisions and competition types
- Add heats management endpoints in events route (POST/GET/DELETE)
- Implement unique constraint: cannot have same role in same division+competition type
- Add participant verification before allowing heats management
- Support heat numbers 1-9 with optional Leader/Follower role
This commit is contained in:
Radosław Gierwiało
2025-11-14 15:32:40 +01:00
parent 0e5dc34cbf
commit 02d3d7ac42
7 changed files with 584 additions and 8 deletions

View File

@@ -0,0 +1,66 @@
-- CreateTable
CREATE TABLE "divisions" (
"id" SERIAL NOT NULL,
"name" VARCHAR(50) NOT NULL,
"abbreviation" VARCHAR(3) NOT NULL,
"display_order" INTEGER NOT NULL,
CONSTRAINT "divisions_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "competition_types" (
"id" SERIAL NOT NULL,
"name" VARCHAR(50) NOT NULL,
"abbreviation" VARCHAR(3) NOT NULL,
CONSTRAINT "competition_types_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "event_user_heats" (
"id" SERIAL NOT NULL,
"user_id" INTEGER NOT NULL,
"event_id" INTEGER NOT NULL,
"division_id" INTEGER NOT NULL,
"competition_type_id" INTEGER NOT NULL,
"heat_number" INTEGER NOT NULL,
"role" VARCHAR(10),
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updated_at" TIMESTAMP(3) NOT NULL,
CONSTRAINT "event_user_heats_pkey" PRIMARY KEY ("id")
);
-- CreateIndex
CREATE UNIQUE INDEX "divisions_name_key" ON "divisions"("name");
-- CreateIndex
CREATE UNIQUE INDEX "divisions_abbreviation_key" ON "divisions"("abbreviation");
-- CreateIndex
CREATE UNIQUE INDEX "competition_types_name_key" ON "competition_types"("name");
-- CreateIndex
CREATE UNIQUE INDEX "competition_types_abbreviation_key" ON "competition_types"("abbreviation");
-- CreateIndex
CREATE INDEX "event_user_heats_user_id_event_id_idx" ON "event_user_heats"("user_id", "event_id");
-- CreateIndex
CREATE INDEX "event_user_heats_event_id_idx" ON "event_user_heats"("event_id");
-- CreateIndex
CREATE UNIQUE INDEX "event_user_heats_user_id_event_id_division_id_competition_t_key" ON "event_user_heats"("user_id", "event_id", "division_id", "competition_type_id", "role");
-- AddForeignKey
ALTER TABLE "event_user_heats" ADD CONSTRAINT "event_user_heats_user_id_fkey" FOREIGN KEY ("user_id") REFERENCES "users"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "event_user_heats" ADD CONSTRAINT "event_user_heats_event_id_fkey" FOREIGN KEY ("event_id") REFERENCES "events"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "event_user_heats" ADD CONSTRAINT "event_user_heats_division_id_fkey" FOREIGN KEY ("division_id") REFERENCES "divisions"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "event_user_heats" ADD CONSTRAINT "event_user_heats_competition_type_id_fkey" FOREIGN KEY ("competition_type_id") REFERENCES "competition_types"("id") ON DELETE RESTRICT ON UPDATE CASCADE;

View File

@@ -54,6 +54,7 @@ model User {
ratingsGiven Rating[] @relation("RaterRatings")
ratingsReceived Rating[] @relation("RatedRatings")
eventParticipants EventParticipant[]
heats EventUserHeat[]
@@map("users")
}
@@ -76,6 +77,7 @@ model Event {
matches Match[]
participants EventParticipant[]
checkinToken EventCheckinToken?
userHeats EventUserHeat[]
@@map("events")
}
@@ -187,3 +189,53 @@ model EventParticipant {
@@index([eventId])
@@map("event_participants")
}
// Competition divisions (Newcomer, Novice, Intermediate, etc.)
model Division {
id Int @id @default(autoincrement())
name String @unique @db.VarChar(50)
abbreviation String @unique @db.VarChar(3)
displayOrder Int @map("display_order")
// Relations
userHeats EventUserHeat[]
@@map("divisions")
}
// Competition types (Jack & Jill, Strictly, etc.)
model CompetitionType {
id Int @id @default(autoincrement())
name String @unique @db.VarChar(50)
abbreviation String @unique @db.VarChar(3)
// Relations
userHeats EventUserHeat[]
@@map("competition_types")
}
// User's declared heats for matchmaking
model EventUserHeat {
id Int @id @default(autoincrement())
userId Int @map("user_id")
eventId Int @map("event_id")
divisionId Int @map("division_id")
competitionTypeId Int @map("competition_type_id")
heatNumber Int @map("heat_number") // 1-9
role String? @db.VarChar(10) // 'Leader', 'Follower', or NULL
createdAt DateTime @default(now()) @map("created_at")
updatedAt DateTime @updatedAt @map("updated_at")
// Relations
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
event Event @relation(fields: [eventId], references: [id], onDelete: Cascade)
division Division @relation(fields: [divisionId], references: [id])
competitionType CompetitionType @relation(fields: [competitionTypeId], references: [id])
// Constraint: Cannot have same role in same division+competition type
@@unique([userId, eventId, divisionId, competitionTypeId, role])
@@index([userId, eventId])
@@index([eventId])
@@map("event_user_heats")
}

View File

@@ -5,10 +5,65 @@ const prisma = new PrismaClient();
async function main() {
console.log('🌱 Seeding database...');
// Create divisions
const divisions = await Promise.all([
prisma.division.upsert({
where: { name: 'Newcomer' },
update: {},
create: { name: 'Newcomer', abbreviation: 'NEW', displayOrder: 1 },
}),
prisma.division.upsert({
where: { name: 'Novice' },
update: {},
create: { name: 'Novice', abbreviation: 'NOV', displayOrder: 2 },
}),
prisma.division.upsert({
where: { name: 'Intermediate' },
update: {},
create: { name: 'Intermediate', abbreviation: 'INT', displayOrder: 3 },
}),
prisma.division.upsert({
where: { name: 'Advanced' },
update: {},
create: { name: 'Advanced', abbreviation: 'ADV', displayOrder: 4 },
}),
prisma.division.upsert({
where: { name: 'All-Star' },
update: {},
create: { name: 'All-Star', abbreviation: 'ALL', displayOrder: 5 },
}),
prisma.division.upsert({
where: { name: 'Champion' },
update: {},
create: { name: 'Champion', abbreviation: 'CHA', displayOrder: 6 },
}),
]);
console.log(`✅ Created ${divisions.length} divisions`);
// Create competition types
const competitionTypes = await Promise.all([
prisma.competitionType.upsert({
where: { name: 'Jack & Jill' },
update: {},
create: { name: 'Jack & Jill', abbreviation: 'J&J' },
}),
prisma.competitionType.upsert({
where: { name: 'Strictly' },
update: {},
create: { name: 'Strictly', abbreviation: 'STR' },
}),
]);
console.log(`✅ Created ${competitionTypes.length} competition types`);
// Create events
const events = await Promise.all([
prisma.event.create({
data: {
prisma.event.upsert({
where: { slug: 'warsaw-dance-festival-2025' },
update: {},
create: {
slug: 'warsaw-dance-festival-2025',
name: 'Warsaw Dance Festival 2025',
location: 'Warsaw, Poland',
startDate: new Date('2025-03-15'),
@@ -18,8 +73,11 @@ async function main() {
description: 'The biggest West Coast Swing event in Central Europe',
},
}),
prisma.event.create({
data: {
prisma.event.upsert({
where: { slug: 'swing-camp-barcelona-2025' },
update: {},
create: {
slug: 'swing-camp-barcelona-2025',
name: 'Swing Camp Barcelona 2025',
location: 'Barcelona, Spain',
startDate: new Date('2025-04-20'),
@@ -29,8 +87,11 @@ async function main() {
description: 'International swing dance camp with workshops and socials',
},
}),
prisma.event.create({
data: {
prisma.event.upsert({
where: { slug: 'blues-week-herrang-2025' },
update: {},
create: {
slug: 'blues-week-herrang-2025',
name: 'Blues Week Herräng 2025',
location: 'Herräng, Sweden',
startDate: new Date('2025-07-14'),
@@ -40,8 +101,11 @@ async function main() {
description: 'Week-long blues dance intensive in the heart of Sweden',
},
}),
prisma.event.create({
data: {
prisma.event.upsert({
where: { slug: 'krakow-swing-connection-2025' },
update: {},
create: {
slug: 'krakow-swing-connection-2025',
name: 'Krakow Swing Connection 2025',
location: 'Krakow, Poland',
startDate: new Date('2025-05-10'),
@@ -72,6 +136,8 @@ async function main() {
console.log('🎉 Seeding completed successfully!');
console.log('');
console.log('Created:');
console.log(` - ${divisions.length} divisions`);
console.log(` - ${competitionTypes.length} competition types`);
console.log(` - ${events.length} events`);
console.log(` - ${chatRooms.length} chat rooms`);
}