TP-Lib: Train Positioning Library
Status: โ Production Ready - All 66 Tasks Complete
Train positioning library for processing GNSS data by projecting positions onto railway track netelements (track axis centerlines). Developed for Infrabel infrastructure management.
Features
- ๐ High Performance: R-tree spatial indexing for O(log n) nearest-track search
- ๐ Accurate Projection: Haversine distance and geodesic calculations with geo-rs
- ๐ CRS Aware: Explicit coordinate reference system handling (EPSG codes)
- โฐ Timezone Support: RFC3339 timestamps with explicit timezone offsets
- ๐ Multiple Formats: CSV and GeoJSON input/output
- ๐งช Well Tested: 84 comprehensive tests (all passing) - unit, integration, contract, CLI, and doctests
- โก Production Ready: Full CLI interface with validation and error handling
Project Structure
tp-lib/ # Rust workspace root
โโโ tp-core/ # Core Rust library
โ โโโ src/
โ โ โโโ models/ # Data models (GnssPosition, Netelement, ProjectedPosition)
โ โ โโโ projection/ # Projection algorithms (geom, spatial indexing)
โ โ โโโ io/ # Input/output (CSV, GeoJSON, Arrow)
โ โ โโโ crs/ # Coordinate reference system transformations
โ โ โโโ temporal/ # Timezone handling utilities
โ โ โโโ errors.rs # Error types
โ โโโ tests/
โ โ โโโ unit/ # Unit tests
โ โ โโโ integration/ # Integration tests
โ โโโ benches/ # Performance benchmarks
โโโ tp-cli/ # Command-line interface
โโโ tp-py/ # Python bindings (PyO3)
Quick Start
Prerequisites
- Rust 1.91.1+ (install from rustup.rs)
- Python 3.12+ (for Python bindings)
Build from Source
# Clone repository
# Build all crates
# Run tests
# Run benchmarks
Docker Usage
Production Deployment
Use Docker to run the CLI without installing Rust:
# Build production image
# Run with mounted data directory
# Or use docker-compose
Running Tests in Docker
Run the complete test suite including CRS transformation tests:
# Using docker-compose (recommended)
# Or build and run test image directly
# Run specific tests
# Run only CRS transformation tests
# Interactive shell for debugging
Why Docker for tests?
- Complete test coverage: Runs all tests including CRS transformation tests
- Consistent environment: Same Rust version across all machines
- No local setup needed: No need to install Rust toolchain locally
- CI/CD ready: Use
Dockerfile.testin GitHub Actions or other CI systems
Usage Examples
# CLI usage - CSV input/output
# GeoJSON output with custom warning threshold
# Custom CSV column names
Library Usage
use ;
use ;
Development Status
โ Phase 1 Complete: Setup (T001-T015)
- Workspace structure with tp-core, tp-cli, tp-py crates
- Cargo.toml configuration for workspace and dependencies
- Git repository initialization with .gitignore
- Directory structure (models, projection, io, crs, temporal)
- Error types (ProjectionError enum with thiserror)
โ Phase 2 Complete: Foundational (T016-T025)
- Data models (GnssPosition, Netelement, ProjectedPosition)
- Basic validation (latitude/longitude ranges, timezone presence)
- Module structure and public API exports
- Unit tests for all models
- Test fixtures and integration test framework
โ Phase 3 Complete: User Story 1 MVP (T026-T049)
- Geometric Projection (T026-T028): ClosestPoint algorithm, measure calculation, 8 unit tests
- Spatial Indexing (T029-T031): R-tree implementation, O(log n) nearest-neighbor, 3 unit tests
- Input Parsing (T032-T035): CSV/GeoJSON readers with Polars/geojson crates, 3 integration tests
- Main Pipeline (T036-T040): RailwayNetwork struct, project_gnss() function, 1 end-to-end test
- Output Writers (T041-T042): CSV/GeoJSON serialization, 2 integration tests
- CLI Interface (T043-T047): clap argument parsing, validation, exit codes, help documentation
- Integration Tests (T048): Full pipeline test with 3 GNSS positions ร 2 netelements
- Configuration (T049): ProjectionConfig with warning threshold and CRS transform flag
Result: Fully functional CLI and library with 28 passing tests
โ Phase 4 Complete: Polish & Cross-Cutting (T050-T066)
- Documentation (T050-T053): Rustdoc comments, README files
- Performance Benchmarks (T054-T056): Criterion benchmarks, naive vs optimized
- Python Bindings (T057-T060): PyO3 wrappers, error conversion, pytest tests
- Additional Testing (T061-T064): Contract tests, GNSS validation, CRS transform tests, CLI integration tests
- Structured Logging (T065-T066): Tracing instrumentation, subscriber configuration
Implementation Notes
Performance
- Target: < 10 seconds for 1000 GNSS positions ร 50 netelements (SC-001)
- Memory: Handles 10,000+ positions without exhaustion (SC-006)
- Accuracy: 95% of positions within 2m projection distance (GPS quality dependent)
- R-tree Complexity: O(log n) nearest-neighbor search
Input Data Requirements
GNSS CSV:
latitude,longitude,timestamp,altitude,hdop
50.8503,4.3517,2025-12-09T14:30:00+01:00,100.0,2.0
- RFC3339 timestamps with timezone (+HH:MM format required)
- CRS must be specified via
--gnss-crsflag - Column names configurable with
--lat-col,--lon-col,--time-col
Railway Network GeoJSON:
- LineString geometries (track centerlines)
- Unique
idproperty per netelement crsproperty with EPSG code
Troubleshooting
"Large projection distance" warnings:**
Indicates GNSS position is far from nearest track (> threshold). Possible causes:
- GPS inaccuracy or poor signal quality
- Train on parallel track not in network
- Missing netelement in railway network
- CRS mismatch between GNSS and network
- Track geometry outdated or incorrect
Adjust threshold with --warning-threshold flag or investigate data quality.
"No Python 3.x interpreter found" build error:
Building with default features requires Python for PyO3 bindings. Disable with:
Or install Python 3.12+ and ensure it's in your PATH.
Known Issues
- Windows Build Dependencies: Requires MSVC toolchain or mingw-w64 for some native dependencies
- Python Bindings: Requires Python 3.12+ installed (excluded from tp-py crate builds by default)
CRS Transformations
TP-Lib uses proj4rs, a pure Rust implementation of PROJ.4, for coordinate reference system transformations. This eliminates system dependencies and enables cross-platform compatibility.
Key Features:
- Pure Rust: No external C libraries required (libproj, sqlite3, etc.)
- Zero system dependencies: Works on Windows, Linux, macOS without installation
- EPSG support: Uses
crs-definitionscrate for EPSG code lookup - WASM compatible: Can be used in browser environments
- Always enabled: CRS transformations are available by default
Supported Transformations:
TP-Lib has been tested with Belgian railway coordinate systems:
- EPSG:4326 (WGS84) โ EPSG:31370 (Belgian Lambert 72)
- EPSG:4326 (WGS84) โ EPSG:3812 (Belgian Lambert 2008)
- Any EPSG codes supported by crs-definitions
Usage:
use CrsTransformer;
use Point;
// Create transformer (EPSG codes or PROJ strings)
let transformer = new?;
// Transform point (automatic degree/radian conversion)
let wgs84_point = new;
let lambert_point = transformer.transform?;
Technical Details:
- proj4rs automatically handles radian/degree conversions for geographic CRS
- EPSG codes are resolved to PROJ strings using the crs-definitions crate
- Custom PROJ strings can be used directly instead of EPSG codes
- Transformation accuracy matches PROJ for standard 2D transformations
Limitations:
- proj4rs implements PROJ.4 API (2D transformations only)
- No 3D/4D or orthometric transformations
- Grid shift support is experimental
- For complex geodetic requirements, consider using PROJ directly
Documentation
API Documentation
Online: https://matdata-eu.github.io/tp-lib/
The documentation is automatically built and deployed on every push to main. It includes:
- tp-core: Core library API with examples
- tp-cli: Command-line interface documentation
- tp-py: Python bindings API reference
Build locally:
# Generate documentation for all workspace crates
# Open in browser (on Windows)
# Open in browser (on Linux/macOS)
Specification Documents
CI/CD & Workflows
This project uses automated workflows for continuous integration and deployment:
- ๐ Continuous Integration: Automated testing, linting, and security checks on every push
- ๐ฆ crates.io Publishing: Automatic release to Rust package registry
- ๐ PyPI Publishing: Automatic release to Python package index
- ๐ Documentation Deployment: Auto-deployed to GitHub Pages
See CI/CD Workflows Documentation for details on:
- Build and test automation
- Release process and version management
- Security and license validation
- Publishing to crates.io and PyPI
- Documentation deployment
Constitution Compliance
This project follows the TP-Lib Constitution v1.1.0 principles:
- โ I. Library-First: Single unified library with quality external dependencies
- โ II. CLI Mandatory: Command-line interface for all functionality
- โ III. High Performance: Apache Arrow, R-tree spatial indexing
- โ IV. TDD: Test-driven development with FIRST TEST validation
- โ V. Full Coverage: Comprehensive test suite (unit, integration, property-based)
- โ VI. Timezone Awareness: DateTime for all timestamps
- โ VII. CRS Explicit: All coordinates include CRS specification
- โ VIII. Error Handling: Typed errors with thiserror, fail-fast validation
- โ IX. Data Provenance: Preserve original GNSS data, audit logging
- โ X. Integration Flexibility: Rust API + CLI + Python bindings
Contributing
This project follows strict TDD workflow:
- Write test first (RED)
- Implement minimum code to pass (GREEN)
- Refactor while keeping tests green
See CONTRIBUTING.md for guidelines.
License
Apache License 2.0 - See LICENSE for details
Contact
TP-Lib Contributors - GitHub Issues