Files
web-inspector/backend/app/models/site_schemas.py
jungwoo choi bffce65aca feat: 접근성 검사 표준 선택 기능 — WCAG/KWCAG 버전별 선택 지원
3가지 검사 모드(한 페이지, 사이트 크롤링, 목록 업로드) 모두에서 접근성 표준을
선택할 수 있도록 추가. WCAG 2.0 A/AA, 2.1 AA, 2.2 AA와 KWCAG 2.1, 2.2를
지원하며, KWCAG 선택 시 axe-core 결과를 KWCAG 검사항목으로 자동 매핑.

- KWCAG 2.2 (33항목) / 2.1 (24항목) ↔ WCAG 매핑 테이블 (kwcag_mapping.py)
- AccessibilityChecker에 표준 파싱 및 KWCAG 변환 로직 추가
- 전체 API 파이프라인에 accessibility_standard 파라미터 전파
- 프론트엔드 3개 폼에 공용 표준 선택 드롭다운 추가

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-14 08:36:14 +09:00

124 lines
3.3 KiB
Python

"""
Pydantic models for site-wide inspection request/response validation.
"""
from pydantic import BaseModel, Field, HttpUrl
from typing import Optional
from datetime import datetime
from enum import Enum
# Default accessibility standard
DEFAULT_ACCESSIBILITY_STANDARD = "wcag_2.1_aa"
# --- Enums ---
class SiteInspectionStatus(str, Enum):
CRAWLING = "crawling"
INSPECTING = "inspecting"
COMPLETED = "completed"
ERROR = "error"
class PageStatus(str, Enum):
PENDING = "pending"
INSPECTING = "inspecting"
COMPLETED = "completed"
ERROR = "error"
# --- Request ---
class StartSiteInspectionRequest(BaseModel):
url: HttpUrl
max_pages: int = Field(default=20, ge=0, le=500, description="최대 크롤링 페이지 수 (0=무제한)")
max_depth: int = Field(default=2, ge=1, le=3, description="최대 크롤링 깊이")
concurrency: int = Field(default=4, ge=1, le=8, description="동시 검사 수")
accessibility_standard: str = Field(
default=DEFAULT_ACCESSIBILITY_STANDARD,
description="접근성 검사 표준 (wcag_2.0_a, wcag_2.0_aa, wcag_2.1_aa, wcag_2.2_aa, kwcag_2.1, kwcag_2.2)",
)
class InspectPageRequest(BaseModel):
url: HttpUrl
# --- Core Data Models ---
class DiscoveredPage(BaseModel):
"""크롤링으로 발견된 개별 페이지."""
url: str
depth: int = 0
parent_url: Optional[str] = None
inspection_id: Optional[str] = None
status: PageStatus = PageStatus.PENDING
title: Optional[str] = None
overall_score: Optional[int] = None
grade: Optional[str] = None
class AggregateScores(BaseModel):
"""사이트 전체 집계 점수."""
overall_score: int = Field(ge=0, le=100, default=0)
grade: str = "F"
html_css: int = Field(ge=0, le=100, default=0)
accessibility: int = Field(ge=0, le=100, default=0)
seo: int = Field(ge=0, le=100, default=0)
performance_security: int = Field(ge=0, le=100, default=0)
total_issues: int = 0
pages_inspected: int = 0
pages_total: int = 0
class SiteInspectionConfig(BaseModel):
"""사이트 검사 설정."""
max_pages: int = 20
max_depth: int = 2
concurrency: int = 4
accessibility_standard: str = DEFAULT_ACCESSIBILITY_STANDARD
# --- Response Models ---
class StartSiteInspectionResponse(BaseModel):
site_inspection_id: str
status: str = "crawling"
root_url: str
stream_url: str
class SiteInspectionResult(BaseModel):
"""사이트 검사 전체 결과."""
site_inspection_id: str
root_url: str
domain: str
status: SiteInspectionStatus
created_at: datetime
completed_at: Optional[datetime] = None
config: SiteInspectionConfig
discovered_pages: list[DiscoveredPage] = []
aggregate_scores: Optional[AggregateScores] = None
class SiteInspectionListItem(BaseModel):
"""사이트 검사 목록 항목 (요약)."""
site_inspection_id: str
root_url: str
domain: str
status: SiteInspectionStatus
created_at: datetime
pages_total: int = 0
pages_inspected: int = 0
overall_score: Optional[int] = None
grade: Optional[str] = None
class SiteInspectionPaginatedResponse(BaseModel):
"""사이트 검사 목록 페이지네이션 응답."""
items: list[SiteInspectionListItem]
total: int
page: int
limit: int
total_pages: int