Merge remote-tracking branch 'origin/master'

This commit is contained in:
Андрей Сумин
2025-03-01 23:58:36 +03:00
9 changed files with 244 additions and 68 deletions
+3 -1
View File
@@ -2,4 +2,6 @@
sidebar_position: 1
---
# Начала!
# Начало!
Выбирай интересующий раздел слева и просвещайся!
+33 -29
View File
@@ -1,66 +1,68 @@
import {themes as prismThemes} from 'prism-react-renderer';
import type {Config} from '@docusaurus/types';
import type * as Preset from '@docusaurus/preset-classic';
import { themes as prismThemes } from "prism-react-renderer";
import type { Config } from "@docusaurus/types";
import type * as Preset from "@docusaurus/preset-classic";
// This runs in Node.js - Don't use client-side code here (browser APIs, JSX...)
const config: Config = {
title: 'DataRush',
tagline: 'Изучите основы анализа данных здесь!',
favicon: 'https://prod-team-15-2pc0i3lc.final.prodcontest.ru/dr.svg',
title: "DataRush",
tagline: "Изучите основы анализа данных здесь!",
favicon: "https://prod-team-15-2pc0i3lc.final.prodcontest.ru/dr.svg",
url: 'https://prod-team-15-2pc0i3lc.final.prodcontest.ru',
baseUrl: '/docs/',
url: "https://prod-team-15-2pc0i3lc.final.prodcontest.ru",
baseUrl: "/docs/",
organizationName: 'megazord',
projectName: 'megazord',
organizationName: "megazord",
projectName: "megazord",
onBrokenLinks: 'throw',
onBrokenMarkdownLinks: 'warn',
onBrokenLinks: "throw",
onBrokenMarkdownLinks: "warn",
i18n: {
defaultLocale: 'ru',
locales: ['ru'],
defaultLocale: "ru",
locales: ["ru"],
},
presets: [
[
'classic',
"classic",
{
docs: {},
docs: {
routeBasePath: "/",
},
theme: {
customCss: './src/css/custom.css',
customCss: "./src/css/custom.css",
},
} satisfies Preset.Options,
],
],
themeConfig: {
image: 'https://prod-team-15-2pc0i3lc.final.prodcontest.ru/dr.svg',
image: "https://prod-team-15-2pc0i3lc.final.prodcontest.ru/dr.svg",
navbar: {
title: 'DataRush',
title: "DataRush",
logo: {
alt: 'DataRush',
src: 'https://prod-team-15-2pc0i3lc.final.prodcontest.ru/dr.svg',
alt: "DataRush",
src: "https://prod-team-15-2pc0i3lc.final.prodcontest.ru/dr.svg",
},
items: [
{
type: 'docSidebar',
sidebarId: 'defaultSidebar',
position: 'left',
label: 'Документация',
type: "docSidebar",
sidebarId: "defaultSidebar",
position: "left",
label: "Документация",
},
],
},
footer: {
style: 'dark',
style: "dark",
links: [
{
title: 'Документация',
title: "Документация",
items: [
{
label: 'Начало',
to: '/docs/intro',
label: "Начало",
to: "/intro",
},
],
},
@@ -72,6 +74,8 @@ const config: Config = {
darkTheme: prismThemes.dracula,
},
} satisfies Preset.ThemeConfig,
staticDirectories: ["static"],
};
export default config;
+3
View File
@@ -19,7 +19,10 @@ class BearerAuth(HttpBearer):
except Exception:
raise AuthenticationError
try:
user = User.objects.get(id=data["id"])
except User.DoesNotExist:
raise AuthenticationError
return user
@staticmethod
+158
View File
@@ -1,5 +1,8 @@
import uuid
from datetime import timedelta, datetime, tzinfo
from dateutil.parser import isoparse
import pytz
from django.contrib.auth.hashers import make_password
from django.test import TestCase
@@ -101,3 +104,158 @@ class CompetitionEndpointTests(TestCase):
HTTP_AUTHORIZATION=header,
)
self.assertEqual(response.status_code, expected_status)
class CompetitionsEndpointTests(TestCase):
def setUp(self):
self.user = User.objects.create(
email="user@example.com",
password=make_password("password123"),
username="t1wk4"
)
resp = self.client.post(
"/api/v1/sign-in",
data={"email": self.user.email, "password": "password123"},
content_type="application/json",
).json()
token = resp["token"]
# Create test competitions
now = datetime.now(tz=pytz.utc)
self.competitions = []
for i in range(1, 6):
competition = Competition.objects.create(
title=f"Competition {i}",
description=f"Description {i}",
type=Competition.CompetitionType.SOLO,
participation_type=(
Competition.CompetitionParticipationType.EDU if i % 2 == 0
else Competition.CompetitionParticipationType.COMPETITIVE
),
start_date=(now + timedelta(days=i)).isoformat(),
end_date=(now + timedelta(days=i + 7)).isoformat(),
)
if i <= 2:
competition.participants.add(self.user)
self.competitions.append(competition)
self.valid_headers = {
"HTTP_AUTHORIZATION": f"Bearer {token}"
}
def get_url(self, params=None):
base_url = "/api/v1/competitions"
return f"{base_url}?{params}" if params else base_url
def test_get_participating_competitions(self):
"""Test filtering competitions where user is participating"""
response = self.client.get(
self.get_url("is_participating=true"),
**self.valid_headers
)
self.assertEqual(response.status_code, 200)
data = response.json()
self.assertEqual(len(data), 2)
self.assertEqual(
{item["id"] for item in data},
{str(self.competitions[0].id), str(self.competitions[1].id)}
)
def test_competition_type_values(self):
"""Test competition type choices are respected"""
response = self.client.get(
self.get_url("is_participating=true"),
**self.valid_headers
)
for item in response.json():
self.assertEqual(item["type"], "solo")
def test_participation_type_values(self):
"""Test participation type alternates between edu/competitive"""
response = self.client.get(
self.get_url("is_participating=false"),
**self.valid_headers
)
types = [item["participation_type"] for item in response.json()]
self.assertCountEqual(
types,
["competitive", "edu", "competitive"]
)
def test_datetime_formatting(self):
"""Test start/end date ISO formatting"""
response = self.client.get(
self.get_url("is_participating=true"),
**self.valid_headers
)
for item in response.json():
if item["start_date"]:
try:
isoparse(item["start_date"])
except ValueError:
self.fail("Invalid start_date format")
if item["end_date"]:
try:
isoparse(item["end_date"])
except ValueError:
self.fail("Invalid end_date format")
def test_competition_metadata(self):
"""Test competition metadata fields"""
response = self.client.get(
self.get_url("is_participating=true"),
**self.valid_headers
)
item = response.json()[0]
self.assertEqual(item["title"], "Competition 1")
self.assertEqual(item["description"], "Description 1")
self.assertEqual(item["type"], "solo")
self.assertEqual(item["participation_type"], "competitive")
def test_verbose_name_consistency(self):
"""Test model verbose names don't affect API schema"""
response = self.client.get(
self.get_url("is_participating=true"),
**self.valid_headers
)
item = response.json()[0]
self.assertNotIn("название", item) # Russian verbose name
self.assertIn("title", item) # Actual API field name
def test_null_dates_handling(self):
"""Test competitions with null dates"""
competition = Competition.objects.create(
title="No Dates Competition",
description="Test competition",
type=Competition.CompetitionType.SOLO,
participation_type=Competition.CompetitionParticipationType.EDU
)
response = self.client.get(
self.get_url("is_participating=false"),
**self.valid_headers
)
test_item = next(
item for item in response.json()
if item["id"] == str(competition.id)
)
self.assertIsNone(test_item["start_date"])
self.assertIsNone(test_item["end_date"])
def test_participation_status_filtering(self):
"""Test filtering by participation_type"""
response = self.client.get(
self.get_url("is_participating=false"),
**self.valid_headers
)
data = response.json()
self.assertEqual(len(data), 3)
+1
View File
@@ -60,6 +60,7 @@ class CompetitionTask(BaseModel):
return self.title
class Meta:
verbose_name = "задание"
verbose_name_plural = "задания"
+1
View File
@@ -23,6 +23,7 @@ dependencies = [
"psycopg2-binary>=2.9.10",
"pydantic>=2.10.5",
"pyjwt>=2.10.1",
"python-dateutil>=2.9.0.post0",
"python-gettext>=5.0",
"python-json-logger>=3.2.1",
"pytz>=2024.2",
@@ -1,10 +1,17 @@
const DataRush = ({ size = 52 }: { size?: number }) => {
const DataRush = ({
size = 52,
className,
}: {
size?: number;
className?: string;
}) => {
return (
<svg
height={size}
viewBox="0 0 149 52"
fill="none"
xmlns="http://www.w3.org/2000/svg"
className={className}
>
<rect width="149" height="52" fill="#333333" />
<path
+2 -2
View File
@@ -17,8 +17,8 @@ const LoginPage = () => {
}, []);
return (
<div className="flex h-screen flex-col items-center gap-10 px-4 py-10 sm:gap-18 sm:py-18">
<DataRush size={52} />
<div className="flex flex-col items-center gap-10 px-4 py-10 sm:gap-18 sm:py-18">
<DataRush size={52} className="min-h-[52px]" />
<div className="flex w-full max-w-96 flex-col items-center gap-7">
<h1 className="text-center text-4xl font-semibold">
Добро пожаловать!
@@ -6,7 +6,7 @@ const NavbarLayout = () => {
<>
<Header />
<div className="px-4 sm:px-6">
<div className="m-auto mt-6 w-full max-w-5xl">
<div className="m-auto my-6 w-full max-w-5xl">
<Outlet />
</div>
</div>