Files
runners-calendar/frontend/src/api/races.ts
Vaka.pro 42ee36d0a2
Some checks failed
CI / build-and-test (pull_request) Has been cancelled
feat: русский UI, версии в футере, даты и устойчивость загрузки API
- API: дата старта всегда YYYY-MM-DD; фронт: parseRaceDate без двойного T00:00:00
- GET /health с version из package.json; Vite define __FRONTEND_VERSION__
- Футер с версиями клиента/сервера (BEM), сетка app-shell на три ряда
- AbortController для карточки старта; ретраи GET при 502–504 и понятные ошибки шлюза
- Русские подписи навигации/страниц, lang=ru, без английских фраз в интерфейсе
2026-04-08 00:40:03 +03:00

118 lines
3.3 KiB
TypeScript

import { ApiError } from "./errors";
import { requestJson } from "./http";
import type { CreateRacePayload, Race, RacesQuery, UpdateRacePayload } from "./types";
function isString(value: unknown): value is string {
return typeof value === "string";
}
function isNullableString(value: unknown): value is string | null {
return value === null || typeof value === "string";
}
function normalizeRace(input: unknown): Race {
const race = input as Partial<Race>;
const isValid =
isString(race?.id) &&
isString(race?.date) &&
isString(race?.title) &&
typeof race?.distanceKm === "number" &&
(race?.status === null ||
race?.status === "planned" ||
race?.status === "registered" ||
race?.status === "completed") &&
isNullableString(race?.officialUrl) &&
isNullableString(race?.startTime) &&
isNullableString(race?.clusterSchedule) &&
isNullableString(race?.bibPickup) &&
isNullableString(race?.bibNumber) &&
isNullableString(race?.finishTime) &&
isNullableString(race?.finishPlace) &&
isNullableString(race?.notes) &&
isString(race?.createdAt) &&
(race?.updatedAt === null || isString(race?.updatedAt));
if (!isValid) {
throw new ApiError({
code: "unknown_error",
status: null,
message: "Некорректный формат данных от API.",
});
}
return {
id: race.id,
date: race.date,
title: race.title,
distanceKm: race.distanceKm,
status: race.status,
officialUrl: race.officialUrl,
startTime: race.startTime,
clusterSchedule: race.clusterSchedule,
bibPickup: race.bibPickup,
bibNumber: race.bibNumber,
finishTime: race.finishTime,
finishPlace: race.finishPlace,
notes: race.notes,
createdAt: race.createdAt,
updatedAt: race.updatedAt,
};
}
function buildRacesQuery(query?: RacesQuery): string {
if (!query) {
return "";
}
const params = new URLSearchParams();
if (typeof query.year === "number") {
params.set("year", String(query.year));
}
if (typeof query.month === "number") {
params.set("month", String(query.month));
}
const serialized = params.toString();
return serialized ? `?${serialized}` : "";
}
export async function getRaces(query?: RacesQuery, init?: RequestInit): Promise<Race[]> {
const response = await requestJson<unknown[]>(`/races${buildRacesQuery(query)}`, init);
if (!Array.isArray(response)) {
throw new ApiError({
code: "unknown_error",
status: null,
message: "Некорректный формат списка забегов от API.",
});
}
return response.map(normalizeRace);
}
export async function getRaceById(id: string, init?: RequestInit): Promise<Race> {
return normalizeRace(await requestJson<unknown>(`/races/${id}`, init));
}
export async function createRace(payload: CreateRacePayload): Promise<Race> {
return normalizeRace(
await requestJson<unknown>("/races", {
method: "POST",
body: JSON.stringify(payload),
}),
);
}
export async function updateRace(id: string, payload: UpdateRacePayload): Promise<Race> {
return normalizeRace(
await requestJson<unknown>(`/races/${id}`, {
method: "PATCH",
body: JSON.stringify(payload),
}),
);
}
export async function deleteRace(id: string): Promise<void> {
await requestJson<void>(`/races/${id}`, { method: "DELETE" });
}