Backend:
- Миграция gps_tracks_001_init.sql: таблицы tracks + pipeline_runs
- Пакет src/api/gps_tracks/: models, db (WAL+upsert с dedup), dedup
(bbox+length+date bucket-hash), mvt (LRU-кэш 1024 тайла), endpoint
(GET /api/gps-tracks, GET /api/gps-tracks/tiles/{z}/{x}/{y}.mvt,
GET /api/gps-tracks/health, POST /api/gps-tracks/cache/clear), config
- Парсеры: osm (split_bbox, haversine, defusedxml XXE-защита),
enduro_russia + ttrails — заглушки (ADR-010/011 proposed, блокированы)
- Licensing guard: pipeline проверяет status ADR-файла до запуска источника
- scripts/gps_collect.py: CLI с --region/--source/--dry-run/--gc
Frontend:
- src/web/gps_tracks.js: двухрежимный слой (MVT z≤11, GeoJSON z≥12),
debounced fetch + AbortController, фильтры активности/источника,
цветовая палитра by-source/by-activity, halo на спутнике, popup трека,
restorePublicTracksState(), localStorage persistence
- index.html: чекбокс «Публичные треки» в terrain-popup, #sheet-gps-filters
- app.css: .terrain-link-btn, .gps-filter-grid, .track-popup
- app.js: вызов restorePublicTracksState() в rebuildMapOverlays(),
applyGpsHaloVisibility() в applyBaseLayer()
Конфиги:
- config/gps_sources.yaml: osm (enabled), enduro_russia/ttrails (disabled)
- config/gps_regions.yaml: ЦФО+Чувашия (enabled), Кавказ (disabled)
Docker:
- gps-collector service с profiles: [batch]
Тесты: 48 новых тестов (unit + integration), 125/125 pass
Refs: ET-008
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
53 lines
1.3 KiB
Python
53 lines
1.3 KiB
Python
"""Pydantic-модели и константы для публичных GPS-треков (ET-008)."""
|
|
from pydantic import BaseModel
|
|
from typing import Optional, List
|
|
|
|
ACTIVITY_TYPES = [
|
|
"enduro", "moto", "offroad", "bicycle", "hike", "ski", "other"
|
|
]
|
|
|
|
|
|
class TrackRecord(BaseModel):
|
|
"""Трек из БД, готовый к отдаче через API."""
|
|
|
|
id: int
|
|
dedup_key: str
|
|
name: Optional[str] = None
|
|
description: Optional[str] = None
|
|
activity_type: Optional[str] = "other"
|
|
user: Optional[str] = None
|
|
created_at: Optional[str] = None
|
|
length_m: float
|
|
points_count: int
|
|
min_lon: float
|
|
min_lat: float
|
|
max_lon: float
|
|
max_lat: float
|
|
sources: List[str]
|
|
external_urls: List[str]
|
|
tags: List[str]
|
|
inserted_at: str
|
|
updated_at: str
|
|
|
|
|
|
class TrackInsert(BaseModel):
|
|
"""Трек для вставки в БД (из парсера)."""
|
|
|
|
external_id: str
|
|
source_id: str
|
|
external_url: Optional[str] = None
|
|
name: Optional[str] = None
|
|
description: Optional[str] = None
|
|
activity_type: str = "other"
|
|
user: Optional[str] = None
|
|
created_at: Optional[str] = None
|
|
length_m: float
|
|
points_count: int
|
|
geom_wkb: bytes # WKB bytes
|
|
min_lon: float
|
|
min_lat: float
|
|
max_lon: float
|
|
max_lat: float
|
|
tags: List[str] = []
|
|
source_priority: int = 999
|