mirror of
https://github.com/Sendouc/sendou.ink.git
synced 2026-03-21 18:04:39 -05:00
* Initial * Start implementing TournamentFormatSelector * Progress * Small progress * Add source * Progress * Skeleton * Progress * Rename progression * Can submit progression * Fix potential issue with caching errors * Settings * isFinals/isUnderground * Valid formats tests * New bracket check in progress * Perf optimization: simulate brackets only frontend * Admin check in fix * resolvesWinner logic * SAME_PLACEMENT_TO_MULTIPLE_BRACKETS * Structure work * Edit bracket while tournament in progress initial * Delayed check in to follow up bracket * Progress validation * NEGATIVE_PROGRESSION * test first sources = null * Different text when invitational * More checks * Validate changed are in preview * Rename * Translated errors * Disbale submti if bracket progression is bad * Adjust bracketIdx * changedBracketProgressionFormat * Progress * Fix E2E tests * Docs progress * Fix state change * Add docs
473 lines
12 KiB
TypeScript
473 lines
12 KiB
TypeScript
import type { ActionFunction } from "@remix-run/node";
|
|
import { z } from "zod";
|
|
import { requireUserId } from "~/features/auth/core/user.server";
|
|
import { userIsBanned } from "~/features/ban/core/banned.server";
|
|
import * as ShowcaseTournaments from "~/features/front-page/core/ShowcaseTournaments.server";
|
|
import * as Progression from "~/features/tournament-bracket/core/Progression";
|
|
import {
|
|
clearTournamentDataCache,
|
|
tournamentFromDB,
|
|
} from "~/features/tournament-bracket/core/Tournament.server";
|
|
import * as TournamentTeamRepository from "~/features/tournament/TournamentTeamRepository.server";
|
|
import invariant from "~/utils/invariant";
|
|
import { logger } from "~/utils/logger";
|
|
import {
|
|
badRequestIfFalsy,
|
|
parseRequestPayload,
|
|
validate,
|
|
} from "~/utils/remix.server";
|
|
import { assertUnreachable } from "~/utils/types";
|
|
import { USER } from "../../../constants";
|
|
import { _action, id } from "../../../utils/zod";
|
|
import { bracketProgressionSchema } from "../../calendar/actions/calendar.new.server";
|
|
import { bracketIdx } from "../../tournament-bracket/tournament-bracket-schemas.server";
|
|
import * as TournamentRepository from "../TournamentRepository.server";
|
|
import { changeTeamOwner } from "../queries/changeTeamOwner.server";
|
|
import { deleteTeam } from "../queries/deleteTeam.server";
|
|
import { joinTeam, leaveTeam } from "../queries/joinLeaveTeam.server";
|
|
import { teamName } from "../tournament-schemas.server";
|
|
import { tournamentIdFromParams } from "../tournament-utils";
|
|
import { inGameNameIfNeeded } from "../tournament-utils.server";
|
|
|
|
export const action: ActionFunction = async ({ request, params }) => {
|
|
const user = await requireUserId(request);
|
|
const data = await parseRequestPayload({
|
|
request,
|
|
schema: adminActionSchema,
|
|
});
|
|
|
|
const tournamentId = tournamentIdFromParams(params);
|
|
const tournament = await tournamentFromDB({ tournamentId, user });
|
|
|
|
const validateIsTournamentAdmin = () =>
|
|
validate(tournament.isAdmin(user), "Unauthorized", 401);
|
|
const validateIsTournamentOrganizer = () =>
|
|
validate(tournament.isOrganizer(user), "Unauthorized", 401);
|
|
|
|
switch (data._action) {
|
|
case "ADD_TEAM": {
|
|
validateIsTournamentOrganizer();
|
|
validate(
|
|
tournament.ctx.teams.every((t) => t.name !== data.teamName),
|
|
"Team name taken",
|
|
);
|
|
validate(
|
|
!tournament.teamMemberOfByUser({ id: data.userId }),
|
|
"User already on a team",
|
|
);
|
|
|
|
await TournamentTeamRepository.create({
|
|
ownerInGameName: await inGameNameIfNeeded({
|
|
tournament,
|
|
userId: data.userId,
|
|
}),
|
|
team: {
|
|
name: data.teamName,
|
|
noScreen: 0,
|
|
prefersNotToHost: 0,
|
|
teamId: null,
|
|
},
|
|
userId: data.userId,
|
|
tournamentId,
|
|
});
|
|
|
|
ShowcaseTournaments.addToParticipationInfoMap({
|
|
tournamentId,
|
|
type: "participant",
|
|
userId: data.userId,
|
|
});
|
|
|
|
break;
|
|
}
|
|
case "CHANGE_TEAM_OWNER": {
|
|
validateIsTournamentOrganizer();
|
|
const team = tournament.teamById(data.teamId);
|
|
validate(team, "Invalid team id");
|
|
const oldCaptain = team.members.find((m) => m.isOwner);
|
|
invariant(oldCaptain, "Team has no captain");
|
|
const newCaptain = team.members.find((m) => m.userId === data.memberId);
|
|
validate(newCaptain, "Invalid member id");
|
|
|
|
changeTeamOwner({
|
|
newCaptainId: data.memberId,
|
|
oldCaptainId: oldCaptain.userId,
|
|
tournamentTeamId: data.teamId,
|
|
});
|
|
|
|
break;
|
|
}
|
|
case "CHANGE_TEAM_NAME": {
|
|
validateIsTournamentOrganizer();
|
|
const team = tournament.teamById(data.teamId);
|
|
validate(team, "Invalid team id");
|
|
|
|
await TournamentRepository.updateTeamName({
|
|
tournamentTeamId: data.teamId,
|
|
name: data.teamName,
|
|
});
|
|
break;
|
|
}
|
|
case "CHECK_IN": {
|
|
validateIsTournamentOrganizer();
|
|
const team = tournament.teamById(data.teamId);
|
|
validate(team, "Invalid team id");
|
|
validate(
|
|
data.bracketIdx !== 0 ||
|
|
tournament.checkInConditionsFulfilledByTeamId(team.id),
|
|
"Can't check-in",
|
|
);
|
|
validate(
|
|
team.checkIns.length > 0 || data.bracketIdx === 0,
|
|
"Can't check-in to follow up bracket if not checked in for the event itself",
|
|
);
|
|
|
|
const bracket = tournament.bracketByIdx(data.bracketIdx);
|
|
invariant(bracket, "Invalid bracket idx");
|
|
validate(bracket.preview, "Bracket has been started");
|
|
|
|
await TournamentRepository.checkIn({
|
|
tournamentTeamId: data.teamId,
|
|
// no sources = regular check in
|
|
bracketIdx: !bracket.sources ? null : data.bracketIdx,
|
|
});
|
|
break;
|
|
}
|
|
case "CHECK_OUT": {
|
|
validateIsTournamentOrganizer();
|
|
const team = tournament.teamById(data.teamId);
|
|
validate(team, "Invalid team id");
|
|
validate(
|
|
data.bracketIdx !== 0 || !tournament.hasStarted,
|
|
"Tournament has started",
|
|
);
|
|
|
|
const bracket = tournament.bracketByIdx(data.bracketIdx);
|
|
invariant(bracket, "Invalid bracket idx");
|
|
validate(bracket.preview, "Bracket has been started");
|
|
|
|
await TournamentRepository.checkOut({
|
|
tournamentTeamId: data.teamId,
|
|
// no sources = regular check in
|
|
bracketIdx: !bracket.sources ? null : data.bracketIdx,
|
|
});
|
|
logger.info(
|
|
`Checked out: tournament team id: ${data.teamId} - user id: ${user.id} - tournament id: ${tournamentId} - bracket idx: ${data.bracketIdx}`,
|
|
);
|
|
break;
|
|
}
|
|
case "REMOVE_MEMBER": {
|
|
validateIsTournamentOrganizer();
|
|
const team = tournament.teamById(data.teamId);
|
|
validate(team, "Invalid team id");
|
|
validate(
|
|
team.checkIns.length === 0 || team.members.length > 4,
|
|
"Can't remove last member from checked in team",
|
|
);
|
|
validate(
|
|
!team.members.find((m) => m.userId === data.memberId)?.isOwner,
|
|
"Cannot remove team owner",
|
|
);
|
|
validate(
|
|
!tournament.hasStarted ||
|
|
!tournament
|
|
.participatedPlayersByTeamId(data.teamId)
|
|
.some((p) => p.userId === data.memberId),
|
|
"Cannot remove player that has participated in the tournament",
|
|
);
|
|
|
|
leaveTeam({
|
|
userId: data.memberId,
|
|
teamId: team.id,
|
|
});
|
|
|
|
ShowcaseTournaments.removeFromParticipationInfoMap({
|
|
tournamentId,
|
|
type: "participant",
|
|
userId: data.memberId,
|
|
});
|
|
|
|
break;
|
|
}
|
|
case "ADD_MEMBER": {
|
|
validateIsTournamentOrganizer();
|
|
const team = tournament.teamById(data.teamId);
|
|
validate(team, "Invalid team id");
|
|
|
|
const previousTeam = tournament.teamMemberOfByUser({ id: data.userId });
|
|
|
|
if (tournament.hasStarted) {
|
|
validate(
|
|
!previousTeam || previousTeam.checkIns.length === 0,
|
|
"User is already on a checked in team",
|
|
);
|
|
} else {
|
|
validate(!previousTeam, "User is already on a team");
|
|
}
|
|
|
|
validate(
|
|
!userIsBanned(data.userId),
|
|
"User trying to be added currently has an active ban from sendou.ink",
|
|
);
|
|
|
|
joinTeam({
|
|
userId: data.userId,
|
|
newTeamId: team.id,
|
|
previousTeamId: previousTeam?.id,
|
|
// this team is not checked in so we can simply delete it
|
|
whatToDoWithPreviousTeam: previousTeam ? "DELETE" : undefined,
|
|
tournamentId,
|
|
inGameName: await inGameNameIfNeeded({
|
|
tournament,
|
|
userId: data.userId,
|
|
}),
|
|
});
|
|
|
|
ShowcaseTournaments.addToParticipationInfoMap({
|
|
tournamentId,
|
|
type: "participant",
|
|
userId: data.userId,
|
|
});
|
|
|
|
break;
|
|
}
|
|
case "DELETE_TEAM": {
|
|
validateIsTournamentOrganizer();
|
|
const team = tournament.teamById(data.teamId);
|
|
validate(team, "Invalid team id");
|
|
validate(!tournament.hasStarted, "Tournament has started");
|
|
|
|
deleteTeam(team.id);
|
|
|
|
ShowcaseTournaments.clearParticipationInfoMap();
|
|
|
|
break;
|
|
}
|
|
case "ADD_STAFF": {
|
|
validateIsTournamentAdmin();
|
|
|
|
await TournamentRepository.addStaff({
|
|
role: data.role,
|
|
tournamentId: tournament.ctx.id,
|
|
userId: data.userId,
|
|
});
|
|
|
|
if (data.role === "ORGANIZER") {
|
|
ShowcaseTournaments.addToParticipationInfoMap({
|
|
tournamentId,
|
|
type: "organizer",
|
|
userId: data.userId,
|
|
});
|
|
}
|
|
|
|
break;
|
|
}
|
|
case "REMOVE_STAFF": {
|
|
validateIsTournamentAdmin();
|
|
|
|
await TournamentRepository.removeStaff({
|
|
tournamentId: tournament.ctx.id,
|
|
userId: data.userId,
|
|
});
|
|
|
|
ShowcaseTournaments.removeFromParticipationInfoMap({
|
|
tournamentId,
|
|
type: "organizer",
|
|
userId: data.userId,
|
|
});
|
|
|
|
break;
|
|
}
|
|
case "UPDATE_CAST_TWITCH_ACCOUNTS": {
|
|
validateIsTournamentOrganizer();
|
|
await TournamentRepository.updateCastTwitchAccounts({
|
|
tournamentId: tournament.ctx.id,
|
|
castTwitchAccounts: data.castTwitchAccounts,
|
|
});
|
|
break;
|
|
}
|
|
case "DROP_TEAM_OUT": {
|
|
validateIsTournamentOrganizer();
|
|
await TournamentRepository.dropTeamOut({
|
|
tournamentTeamId: data.teamId,
|
|
previewBracketIdxs: tournament.brackets.flatMap((b, idx) =>
|
|
b.preview ? idx : [],
|
|
),
|
|
});
|
|
break;
|
|
}
|
|
case "UNDO_DROP_TEAM_OUT": {
|
|
validateIsTournamentOrganizer();
|
|
|
|
await TournamentRepository.undoDropTeamOut(data.teamId);
|
|
break;
|
|
}
|
|
case "RESET_BRACKET": {
|
|
validateIsTournamentOrganizer();
|
|
validate(!tournament.ctx.isFinalized, "Tournament is finalized");
|
|
|
|
const bracketToResetIdx = tournament.brackets.findIndex(
|
|
(b) => b.id === data.stageId,
|
|
);
|
|
const bracketToReset = tournament.brackets[bracketToResetIdx];
|
|
validate(bracketToReset, "Invalid bracket id");
|
|
validate(!bracketToReset.preview, "Bracket has not started");
|
|
|
|
const inProgressBrackets = tournament.brackets.filter((b) => !b.preview);
|
|
validate(
|
|
inProgressBrackets.every(
|
|
(b) =>
|
|
!b.sources ||
|
|
b.sources.every((s) => s.bracketIdx !== bracketToResetIdx),
|
|
),
|
|
"Some bracket that sources teams from this bracket has started",
|
|
);
|
|
|
|
await TournamentRepository.resetBracket(data.stageId);
|
|
|
|
break;
|
|
}
|
|
case "UPDATE_IN_GAME_NAME": {
|
|
validateIsTournamentOrganizer();
|
|
|
|
const teamMemberOf = badRequestIfFalsy(
|
|
tournament.teamMemberOfByUser({ id: data.memberId }),
|
|
);
|
|
|
|
await TournamentTeamRepository.updateMemberInGameName({
|
|
userId: data.memberId,
|
|
inGameName: `${data.inGameNameText}#${data.inGameNameDiscriminator}`,
|
|
tournamentTeamId: teamMemberOf.id,
|
|
});
|
|
break;
|
|
}
|
|
case "DELETE_LOGO": {
|
|
validateIsTournamentOrganizer();
|
|
|
|
await TournamentTeamRepository.deleteLogo(data.teamId);
|
|
|
|
break;
|
|
}
|
|
case "UPDATE_TOURNAMENT_PROGRESSION": {
|
|
validateIsTournamentOrganizer();
|
|
validate(!tournament.ctx.isFinalized, "Tournament is finalized");
|
|
|
|
validate(
|
|
Progression.changedBracketProgression(
|
|
tournament.ctx.settings.bracketProgression,
|
|
data.bracketProgression,
|
|
).every(
|
|
(changedBracketIdx) =>
|
|
tournament.bracketByIdx(changedBracketIdx)?.preview,
|
|
),
|
|
"Can't change started brackets",
|
|
);
|
|
|
|
await TournamentRepository.updateProgression({
|
|
tournamentId: tournament.ctx.id,
|
|
bracketProgression: data.bracketProgression,
|
|
});
|
|
|
|
break;
|
|
}
|
|
default: {
|
|
assertUnreachable(data);
|
|
}
|
|
}
|
|
|
|
clearTournamentDataCache(tournamentId);
|
|
|
|
return null;
|
|
};
|
|
|
|
export const adminActionSchema = z.union([
|
|
z.object({
|
|
_action: _action("CHANGE_TEAM_OWNER"),
|
|
teamId: id,
|
|
memberId: id,
|
|
}),
|
|
z.object({
|
|
_action: _action("CHANGE_TEAM_NAME"),
|
|
teamId: id,
|
|
teamName,
|
|
}),
|
|
z.object({
|
|
_action: _action("CHECK_IN"),
|
|
teamId: id,
|
|
bracketIdx,
|
|
}),
|
|
z.object({
|
|
_action: _action("CHECK_OUT"),
|
|
teamId: id,
|
|
bracketIdx,
|
|
}),
|
|
z.object({
|
|
_action: _action("ADD_MEMBER"),
|
|
teamId: id,
|
|
userId: id,
|
|
}),
|
|
z.object({
|
|
_action: _action("REMOVE_MEMBER"),
|
|
teamId: id,
|
|
memberId: id,
|
|
}),
|
|
z.object({
|
|
_action: _action("DELETE_TEAM"),
|
|
teamId: id,
|
|
}),
|
|
z.object({
|
|
_action: _action("ADD_TEAM"),
|
|
userId: id,
|
|
teamName,
|
|
}),
|
|
z.object({
|
|
_action: _action("ADD_STAFF"),
|
|
userId: id,
|
|
role: z.enum(["ORGANIZER", "STREAMER"]),
|
|
}),
|
|
z.object({
|
|
_action: _action("REMOVE_STAFF"),
|
|
userId: id,
|
|
}),
|
|
z.object({
|
|
_action: _action("DROP_TEAM_OUT"),
|
|
teamId: id,
|
|
}),
|
|
z.object({
|
|
_action: _action("UNDO_DROP_TEAM_OUT"),
|
|
teamId: id,
|
|
}),
|
|
z.object({
|
|
_action: _action("DELETE_LOGO"),
|
|
teamId: id,
|
|
}),
|
|
z.object({
|
|
_action: _action("UPDATE_CAST_TWITCH_ACCOUNTS"),
|
|
castTwitchAccounts: z.preprocess(
|
|
(val) =>
|
|
typeof val === "string"
|
|
? val
|
|
.split(",")
|
|
.map((account) => account.trim())
|
|
.map((account) => account.toLowerCase())
|
|
: val,
|
|
z.array(z.string()),
|
|
),
|
|
}),
|
|
z.object({
|
|
_action: _action("RESET_BRACKET"),
|
|
stageId: id,
|
|
}),
|
|
z.object({
|
|
_action: _action("UPDATE_IN_GAME_NAME"),
|
|
inGameNameText: z.string().max(USER.IN_GAME_NAME_TEXT_MAX_LENGTH),
|
|
inGameNameDiscriminator: z
|
|
.string()
|
|
.refine((val) => /^[0-9a-z]{4,5}$/.test(val)),
|
|
memberId: id,
|
|
}),
|
|
z.object({
|
|
_action: _action("UPDATE_TOURNAMENT_PROGRESSION"),
|
|
bracketProgression: bracketProgressionSchema,
|
|
}),
|
|
]);
|