Python implementation of XCTrack's task format
Project description
pyxctsk
A Python implementation of XCTrack's task format for paragliding and hang gliding competitions. This library enables parsing, manipulation, and visualization of XCTrack task files used to define competition routes.
pyxctsk provides a comprehensive toolkit for working with the XCTrack task specification, including reading/writing task files, generating QR codes for task sharing, and performing distance calculations on competition routes.
The library implements the full XCTrack Competition Interfaces specification, ensuring compatibility with the XCTrack mobile app and other tools in the paragliding competition ecosystem.
Technical Highlights
- Immutable Data Model: Core domain objects use Python's dataclasses with strict validation to ensure task integrity (task.py)
- Advanced Distance Calculation: Implements sophisticated route optimization algorithms to accurately calculate task distances with iterative refinement to avoid look-ahead bias (distance.py, route_optimization.py)
- Efficient QR Code Representation: Implements XCTrack's compact QR code format with polyline compression for efficient task sharing via small QR codes that work well in direct sunlight (qrcode_task.py)
- Flexible Parsing Pipeline: Single entry point that intelligently detects and parses multiple input formats (JSON, URL, QR code image) (parser.py)
- Type Safety: Comprehensive type hints throughout the codebase with strict mypy enforcement
Dependencies and Libraries
Core
- click: Command-line interface framework for the CLI tools
- geopy: Geographic calculations for distance and point manipulation
- polyline: Polyline encoding/decoding for compact coordinate representation
- pyproj: Projection calculations for accurate distance measurements
- scipy: Scientific computing library used for route optimization algorithms
Optional
- Pillow: Image processing for QR code generation and parsing
- qrcode: QR code generation
- zxing-cpp: QR code parsing from images (ships binary wheels; no system library needed)
Development
- pytest: Testing framework
- ruff: Linting and formatting
- mypy: Static type checking
- lefthook: Git hook manager
Project Structure
.
├── src/
│ └── pyxctsk/ # Core package implementation
│ ├── __init__.py # Package exports
│ ├── task.py # Core data models
│ ├── parser.py # Input format parser
│ ├── distance.py # Distance calculation interface
│ ├── qrcode_task.py # QR code format implementation
│ └── ...
├── tests/ # Test suite
│ ├── test_basic.py
│ ├── test_distance.py
│ └── ...
├── scripts/ # Utility scripts
├── pyproject.toml # Project configuration
└── README.md
src/pyxctsk/: Core library implementation with immutable data models and parsing logictests/: Comprehensive test suite with basic tests and specialized distance calculation testsscripts/: Utility scripts for automation and testing
Installation
Core Library Only
pip install pyxctsk
Development Installation
This project uses uv for dependency management.
git clone https://github.com/simonsteiner/pyxctsk.git
cd pyxctsk
# Create the virtual environment and install the project (editable) with the
# dev dependency group plus the web and analysis extras.
uv sync --all-extras
# Run tests
uv run pytest
# Run single test with parameter
# -s: disables output capturing, allowing print statements and other outputs to be shown in the terminal.
# -vv: increases verbosity, providing more detailed test results.
uv run pytest -s tests/test_qrcode.py -vv
# (Optional) To check QR code dependencies, run:
uv run python scripts/check_qr_deps.py
Code Quality & Formatting
The project uses lefthook to run ruff (lint + format), mypy, and cspell on commit:
# Install the git hooks (one-time)
uv run lefthook install
# Run the pre-commit hooks against staged files
uv run lefthook run pre-commit
# Or run the tools directly
uv run ruff check --fix src/ tests/ scripts/
uv run ruff format src/ tests/ scripts/
uv run mypy --config-file mypy.ini src/
Usage Examples
from pyxctsk import parse_task, Task, TaskType, Turnpoint, Waypoint
# Parse from file, URL, or QR code
task = parse_task('task.xctsk') # From .xctsk file
task = parse_task('XCTSK:{...}') # From XCTrack URL
task = parse_task('qr_code.png') # From QR code image
# Access task data
print(f"Task type: {task.task_type}")
print(f"Number of turnpoints: {len(task.turnpoints)}")
# Create a new task
task = Task(
task_type=TaskType.CLASSIC,
version=1,
turnpoints=[
Turnpoint(
radius=1000,
waypoint=Waypoint(
name="TP01", lat=46.5, lon=8.0, alt_smoothed=1000
)
)
]
)
# Save as JSON
with open('task.xctsk', 'w') as f:
f.write(task.to_json())
# Generate QR code
from pyxctsk.qrcode_image import generate_qrcode_image
qr_task = task.to_qr_code_task()
qr_image = generate_qrcode_image(qr_task.to_string())
qr_image.save('task_qr.png')
Command Line Interface
The pyxctsk CLI enables conversion and inspection of XCTrack task files in multiple formats, with strict error handling and clear messaging.
Parameter Options:
--format [json|kml|png|qrcode-json]
Output format.json: Standard JSON representationkml: KML for mapping toolspng: QR code image (PNG)qrcode-json: Compact QR string (XCTSK: URL)
Default:json
--output, -o <file>
Output file (default: stdout). Forpng, writes a PNG image; for others, writes text.<input_file>
Input file (optional). If omitted, reads from stdin. Accepts.xctskfiles or QR code images.
Examples:
# Convert task to different formats
pyxctsk convert task.xctsk --format json # JSON output
pyxctsk convert task.xctsk --format kml -o task.kml # KML output
pyxctsk convert task.xctsk --format png -o qr.png # QR code image
pyxctsk convert task.xctsk --format qrcode-json # XCTSK: URL string
# Parse from different inputs
pyxctsk convert qr_code.png --format json # From QR image
cat task.xctsk | pyxctsk convert --format kml # From stdin
Supported formats:
- Input:
.xctsk(XCTrack task files), QR code image (PNG) - Output: JSON, KML, QR code (PNG or XCTSK: URL string)
See the CLI startup message (pyxctsk --help or running the CLI with no arguments) for a quick summary of options and supported formats.
Requirements
- Python 3.11+
- Optional dependencies can be installed with extras:
pip install pyxctsk[web]for web interface componentspip install pyxctsk[analysis]for analysis tools
- Development tooling lives in the
devdependency group and is installed automatically byuv sync(oruv sync --all-extrasto include the extras).
License
This project is licensed under the MIT License - see the LICENSE file for details.
Original Implementation
This Python library is based on the Go implementation by Tom Payne: https://github.com/twpayne/go-xctrack
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file pyxctsk-0.4.1.tar.gz.
File metadata
- Download URL: pyxctsk-0.4.1.tar.gz
- Upload date:
- Size: 67.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.25 {"installer":{"name":"uv","version":"0.11.25","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
2dc4c50287b7c045e56d004239ac4215882c26fd543ef63b2a11c5b9b7f59b2f
|
|
| MD5 |
1580325495607d0778dc5dea5f7ef146
|
|
| BLAKE2b-256 |
74680c7c6e9e2cc169323d348a4c9c1b9700d3cb4396f459b04ec5ab7084ceb2
|
File details
Details for the file pyxctsk-0.4.1-py3-none-any.whl.
File metadata
- Download URL: pyxctsk-0.4.1-py3-none-any.whl
- Upload date:
- Size: 50.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.25 {"installer":{"name":"uv","version":"0.11.25","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6cb583d54319c4102764d416efc3304be5a363f5b77391787a2d07ec7cf3f2f8
|
|
| MD5 |
9b30f6cf6ee55ba781f06272a41a2774
|
|
| BLAKE2b-256 |
4dce6038f3be0f111a2babf12691d44b43378577e03b33068c5514fefce398c1
|