A Python toolkit for cross-modal accessibility analysis on transport networks — routing, distance/time computation, utility-based travel costs, and gravity- and logsum-based accessibility metrics on networkx graphs (routed via scipy.sparse.csgraph).
The name is Latin/Italian for open — the condition that accessibility, at root, measures.
Pre-1.0, alpha. Published alongside a toolkit paper (in submission). APIs may change without notice until v1.0.
pip install aperta # algorithms only
pip install 'aperta[osm]' # + OSM ingestion (osmnx)
pip install 'aperta[examples]' # + everything needed to run the example notebooksRequires Python ≥ 3.11.
The osm_helpers and topography modules import their backing libraries
(osmnx, rasterio, requests) lazily — install the matching [osm] /
[topo] extra if you use them, otherwise an ImportError surfaces at first use.
For development:
git clone git@github.com:mmiotti/aperta.git
cd aperta
pip install -e ".[osm,topo,h3]"
python -m unittest discover -s tests -t .If you plan to edit the example notebooks under
examples/, run the jupytext + nbstripout setup once after cloning. Not needed if you're only using the library or modifying Python source.
Aperta is organized around a six-phase workflow. Phases 4 and 5's calibration sub-step are optional; the rest is the minimum end-to-end pipeline.
- Load and prepare data — networks (one per mode), land use, topography, optional ground-truth data (traffic counters, travel-survey times).
- Map data to units — aggregate source data into the
cells → zoneshierarchy; snap geo units to network nodes. Snapping is two complementary functions:insert_projected_nodesoptionally enriches the graph by inserting virtual nodes onto road segments where points would otherwise have no graph node within snap distance (with optional filtering, e.g. main roads only);snap_to_network_nodesthen does the actual point-to-node match (optionally two-tier with a priority node set for "prefer main-road nodes" semantics). - Build sparse OD pairs — the tiered OD structure with per-cell origins at near range and zone-aggregated destinations at far range, keeping per-origin compute bounded independently of network extent.
- (Optional) Estimate traffic flows — sampled betweenness centrality (essentially an network-based "2.5-step" travel demand model); optionally calibrate against observed traffic counter data.
- Estimate travel costs — shortest paths on the routing graph. Three optional features: (a) trip overheads for parking search, unlocking a bicycle, etc (usually estimated through correlation with urban characteristics such as density); (b) utility-based generalized costs and (c) edge-weight calibration against observed travel times.
- Calculate accessibilities — cumulative-opportunity, gravity, nearest-k, logsum (and cross-modal aggregation across per-mode results).
See the API reference for which module covers each phase and for the specific functions.
Runnable examples, in increasing depth:
- examples/minimal/accessibility.ipynb — what aperta does in ~50 lines using only OpenStreetMap. Cambridge MA, ~10 s.
- examples/walkthrough/accessibility.ipynb — guided tour of every primitive; walking + cycling, cross-modal logsum, path-first per-edge feature aggregation. Central Paris, ~1 min end-to-end.
- examples/extended/ — production-scale Bern + 40 km: prep pipeline, calibration against observed travel times, traffic-flow estimation, accessibility analysis. ~30 min.
The toy-world end-to-end test in tests/test_workflow.py doubles as the smallest possible walk-through (~150 lines, runs in a second).
The three-line core of an accessibility analysis: build the tiered OD pairs, route shortest paths, count opportunities within a travel-time budget.
from aperta import accessibility, od_pairs, routing
pairs = od_pairs.get_pairs(cells, r_cells=2000.0, node_column='node_id')
times = routing.tiered_path_costs(pairs, graph, weight='walk_time_s')
acc = accessibility.cumulative_opportunities(
times, {'supermarkets': weights}, {},
[accessibility.Bin('15min', 0, 15 * 60)],
)A complete, runnable version (OSM ingestion, plotting): examples/minimal/accessibility.ipynb.
See the API reference for module-by-module documentation.
What aperta is:
- Path-first. Routing returns the realized route alongside the cost as a single primitive, so per-edge attributes (gradient, exposure, surface, perceived safety) aggregate along the path natively — the architectural prerequisite for utility-based and route-aware accessibility.
- Cross-modal. Mode and network are orthogonal: one network per mode, with
min/logsumaggregation across modes as a first-class operation. Generalizes to any axis of network variation — time-of-day, congestion regime, infrastructure scenario. - Multi-scale. A tiered cell / zone OD structure bounds per-origin computation independently of network extent — country-scale reach without country-scale destination counts.
- Live-graph routing. Dijkstra on the graph directly, no precomputed index. Slower per query than contraction-hierarchy tools, but edge-weight changes are immediate — what makes iterative calibration and scenario comparison practical.
What aperta is not:
- No filesystem assumptions. Algorithm functions take plain
networkxgraphs,pandas/geopandasframes, andnumpyarrays — no file I/O. - No DAG engine, no global state. No caching, no dependency tracking, no orchestration. Every function takes its inputs explicitly. For DAG features, layer DVC or Snakemake on top.
Aperta deliberately doesn't try to do everything in-house. Two interoperability patterns are worth flagging:
- Public transit via R5. Aperta has no native public-transit support right now (no GTFS reader, no RAPTOR-style time-dependent routing). Anything that can be expressed as a
networkxgraph with appropriate edge weights — including simplified transit-as-graph models — will route in aperta like any other network. For full GTFS-based transit routing (calendars, transfers, frequency-based services), the pragmatic pattern is to compute the transit OD cost matrix with R5 (via r5py), align its origins/destinations to the same cell layer aperta uses, and feed the resulting per-mode cost ODM intood_pairs.aggregate_across_modesalongside the walk / cycle / car ODMs computed by aperta. The cross-modal aggregation proceeds identically whether each per-mode ODM came from aperta's router or elsewhere. - Faster cost-only routing via Pandana/pandarm. Aperta's live-graph routing is the right trade-off for path-first, iterative, and scenario-comparative workloads, but for one-shot cost-only accessibility on a large fixed network, contraction-hierarchy backends like Pandana (and its recent modernized fork pandarm) route faster per query. The calibrated edge weights produced by
calibration.calibrate_edge_weightsare plain per-edge attributes on thenetworkxgraph and transfer cleanly to a Pandana/pandarm network built from the same OSM extract — i.e., you can calibrate edge weights in aperta and then route with them in Pandana/pandarm.
Ultimate speed for the full accessibility stack was not aperta's goal. Nonetheless, aperta typically runs within 1–5× of Pandana on equivalent cumulative-opportunity workloads. When the area of interest (for which to calculate accessibilities) is substantially smaller than the buffer zone (destinations to consider), or when aiming to recalculate accessibilities for a select subset of locations after a graph topology or edge weight change, aperta can even be faster than Pandana. See the benchmark for the full setup and numbers, or run examples/extended/benchmark.py to reproduce.
Aperta was developed at the Chair of Ecological Systems Design at ETH Zurich in the context of the BlueCity project and LUMOS.
If you use aperta in a publication, please cite the archived release:
@software{miotti_aperta_2026,
author = {Miotti, Marco},
title = {{Aperta: Path-first, cross-modal accessibility analysis in Python}},
year = 2026,
publisher = {Zenodo},
doi = {10.5281/zenodo.20473787},
url = {https://doi.org/10.5281/zenodo.20473787}
}MIT. See LICENSE.
