Skip to main content

No project description provided

Project description

routx-python

GitHub | Documentation | Issue Tracker | PyPI

Python bindings for routx - library for simple routing over OpenStreetMap data.

Routx converts OSM data into a standard weighted directed graph representation, and runs A* to find shortest paths between nodes. Interpretation of OSM data is customizable via profiles. Routx supports one-way streets, access tags (on ways only) and turn restrictions.

Usage

pip install routx in a virtual environment.

Precompiled wheels are available for most popular platforms (aarch64, x86-64 × GNU Linux, musl Linux, MacOS and Windows). On anything else, cargo (please don't curl | sh and install from your system's package manager), ninja and a C/C++ compiler toolchain are required to properly compile the library.

import routx

g = routx.Graph()
g.add_from_osm_file("path/to/monaco.pbf", routx.OsmProfile.CAR)

start_node = g.find_nearest_node(43.7384, 7.4246)
end_node = g.find_nearest_node(43.7478, 7.4323)
route = g.find_route(start_node.id, end_node.id)

for node_id in route:
    node = g[node_id]
    print(node.lat, node.lon)

Reference

Unless noted otherwise, __init__ methods behave as expected for standard NamedTuple/dataclass/IntEnum objects.

routx.DEFAULT_STEP_LIMIT

DEFAULT_STEP_LIMIT: Final[int] = 1000000

Recommended A* step limit for routx.Graph.find_route.

routx.Node

class Node(NamedTuple):
    id: int
    osm_id: int
    lat: float
    lon: float

An element of a Graph. A named tuple.

Due to turn restriction processing, one OpenStreetMap node may be represented by multiple nodes in the graph. If that is the case, a "canonical" node (not bound by any turn restriction) will have id == osm_id.

Nodes with id == 0 are used by the underlying library to signify the absence of a node, are considered false-y and must not be used by consumers.

Methods:

  • __bool__(self) -> bool - True if id is not zero.

Read-only properties:

  • is_canonical: bool - True if id == osm_id.

routx.Edge

class Edge(NamedTuple):
    to: int
    cost: float

Outgoing (one-way) connection from a Node. A named tuple.

cost must be greater than the crow-flies distance between the two nodes.

routx.Graph

class Graph(MutableMapping[int, Node]):
    pass  # no attributes, constructor accepts no arguments

OpenStreetMap-based network representation as a set of nodes and edges between them.

Node access is implemented through the standard MutableMapping interface from ids (integers) to nodes.

Note that overwriting existing nodes preserves all outgoing and incoming edges. Thus updating a node position might result in violation of the Edge invariant and break route finding. It is discouraged to update nodes, and it is the caller's responsibility not to break this invariant.

Methods (other than the implemented for and inherited from MutableMapping[int, Node]):

  • find_nearest_node(lat: float, lon: float) -> Node - find the closest canonical (id == osm_id) Node to the given position.

    This function requires computing distance to every Node in the graph and is not suitable for large graphs or for multiple searches. Use KDTree for faster NN finding.

    If the graph is empty, raises KeyError.

  • get_edges(self, from_: int) -> list[Edge] - gets all outgoing edges from a node with a given id.

  • get_edge(self, from_: int, to: int) -> float - gets the cost of traversing an edge between nodes with provided ids. Returns positive infinity when the provided edge does not exist.

  • set_edge(self, from_: int, to: int, cost: float) -> bool - creates or updates an Edge from one node to another.

    The cost must not be smaller than the crow-flies distance between nodes, as this would violate the A* invariant and break route finding. It is the caller's responsibility to uphold this invariant.

    Returns True if an existing edge was overwritten, False otherwise.

    Note that given an Edge object, this method may be called with graph.set_edge(from_, *edge).

  • delete_edge(self, from_: int, to: int, /, missing_ok: bool = True) -> None - ensures an Edge from one node to another does not exist. If no such edge exists and missing_ok is set to False, raises KeyError.

  •   find_route(
          self,
          from_: int,
          to: int,
          /,
          without_turn_around: bool = True,
          step_limit: int = DEFAULT_STEP_LIMIT,
      ) -> array[int]
    

    Finds the cheapest way between two nodes using the A* algorithm. Returns an array of type q containing node IDs of such route. The array may be empty if no route exists.

    from_id must identify a specific node in the graph, and to_id must identify a specific canonical (id == osm_id) node; otherwise KeyError is raised.

    without_turn_around defaults to True and prevents the algorithm from circumventing turn restrictions by suppressing unrealistic turn-around instructions (A-B-A). This introduces an extra dimension to the search space, so if the graph doesn't contain any turn restrictions, this parameter should be set to False.

    step_limit limits how many nodes can be expanded during search before raising StepLimitExceeded. Concluding that no route exists requires expanding all nodes accessible from the start, which is usually very time consuming, especially on large datasets.

  • simplify_route(self, route: Iterable[int], epsilon: float = 1e-5) -> array[int] - Simplifies a route (sequence of nodes) using the Ramer-Douglas-Peucker algorithm.

    If line is an instance of an array of type q (int64), then it is modified in-place, and a slice containing the simplified route is returned. Other elements are left undefined. Note that find_route returns such an array.

    Epsilon represents the maximum distance (in decimal degrees, as the implementation assumes flat, Euclidean geometry) for a point's distance to a line segment to be considered insignificant and therefore removed.

  •   add_from_osm_file(
          self,
          filename: str | bytes | PathLike[str] | PathLike[bytes],
          profile: OsmProfile | OsmCustomProfile,
          /,
          format: OsmFormat = OsmFormat.UNKNOWN,
          bbox: tuple[float, float, float, float] = (0.0, 0.0, 0.0, 0.0),
      ) -> None
    

    Parses OSM data from the provided file and adds them to this graph.

    profile describes how the OSM data should be interpreted.

    format describes the file format of the input OSM data. Defaults to auto-detection.

    bbox filters features by a specific bounding box, in order: left (min lon), bottom (min lat), right (max lon), top (max lat). Ignored if all values are zero (default).

  •   add_from_osm_memory(
          self,
          mv: memoryview,
          profile: OsmProfile | OsmCustomProfile,
          /,
          format: OsmFormat = OsmFormat.UNKNOWN,
          bbox: tuple[float, float, float, float] = (0.0, 0.0, 0.0, 0.0),
      ) -> None
    

    Parses OSM data from the provided contents of a memory file. The buffer must be contiguous and also mutable (for reasons only known to ctypes, because the underlying library takes a const pointer).

    profile describes how the OSM data should be interpreted.

    format describes the file format of the input OSM data. Defaults to auto-detection.

    bbox filters features by a specific bounding box, in order: left (min lon), bottom (min lat), right (max lon), top (max lat). Ignored if all values are zero (default).

  •   def read_from_file(
          self,
          filename: str | bytes | PathLike[str] | PathLike[bytes],
          format: GraphFormat,
      ) -> None
    

    Reads data from a serialized graph file, and adds it to the provided graph.

  • read_from_memory(self, mv: memoryview, format: GraphFormat) -> None - Reads data from a serialized graph buffer, and adds it to the provided graph.

    The buffer must be contiguous and also mutable (for reasons only known to ctypes, because the underlying library takes a const pointer).

  •   write_to_file(
          self,
          filename: str | bytes | PathLike[str] | PathLike[bytes],
          format: GraphFormat,
      ) -> None
    

    Writes graph data to a file, as per the selected wire format.

  • write_to_memory(self, format: GraphFormat) -> bytearray - Writes graph data to an in-memory buffer, as per the selected wire format.

routx.KDTree

class KDTree:
    pass # no attributes, private constructor

Methods:

  • find_nearest_node(self, lat: float, lon: float) -> Node - finds the closest node to the provided position and returns its id. Raises KeyError when the k-d tree contains no nodes.

Class Methods:

  • KDTree.build(graph: Graph) -> Self - builds a k-d tree with all canonical (id == osm_id) nodes contained in the provided graph.

routx.GraphFormat

class GraphFormat(IntEnum):
    BINARY = 1

Wire format of a full/serialized Graph. An IntEnum.

  • BINARY - Custom, routx binary file format.

routx.OsmProfile

class OsmProfile(IntEnum):
    CAR = 1
    BUS = 2
    BICYCLE = 3
    FOOT = 4
    RAILWAY = 5
    TRAM = 6
    SUBWAY = 7

Predefined OSM conversion profiles. An IntEnum.

CAR Penalties:
Tag Penalty
highway=motorway 1.0
highway=motorway_link 1.0
highway=trunk 2.0
highway=trunk_link 2.0
highway=primary 5.0
highway=primary_link 5.0
highway=secondary 6.5
highway=secondary_link 6.5
highway=tertiary 10.0
highway=tertiary_link 10.0
highway=unclassified 10.0
highway=minor 10.0
highway=residential 15.0
highway=living_street 20.0
highway=track 20.0
highway=service 20.0

Access tags: access, vehicle, motor_vehicle, motorcar.

Allows motorroads and considers turn restrictions.

BUS Penalties:
Tag Penalty
highway=motorway 1.0
highway=motorway_link 1.0
highway=trunk 1.0
highway=trunk_link 1.0
highway=primary 1.1
highway=primary_link 1.1
highway=secondary 1.15
highway=secondary_link 1.15
highway=tertiary 1.15
highway=tertiary_link 1.15
highway=unclassified 1.5
highway=minor 1.5
highway=residential 2.5
highway=living_street 2.5
highway=track 5.0
highway=service 5.0

Access tags: access, vehicle, motor_vehicle, psv, bus, routing:ztm.

Allows motorroads and considers turn restrictions.

BICYCLE Penalties:
Tag Penalty
highway=trunk 50.0
highway=trunk_link 50.0
highway=primary 10.0
highway=primary_link 10.0
highway=secondary 3.0
highway=secondary_link 3.0
highway=tertiary 2.5
highway=tertiary_link 2.5
highway=unclassified 2.5
highway=minor 2.5
highway=cycleway 1.0
highway=residential 1.0
highway=living_street 1.5
highway=track 2.0
highway=service 2.0
highway=bridleway 3.0
highway=footway 3.0
highway=steps 5.0
highway=path 2.0

Access tags: access, vehicle, bicycle.

Disallows motorroads and considers turn restrictions.

FOOT Penalties:
Tag Penalty
highway=trunk 4.0
highway=trunk_link 4.0
highway=primary 2.0
highway=primary_link 2.0
highway=secondary 1.3
highway=secondary_link 1.3
highway=tertiary 1.2
highway=tertiary_link 1.2
highway=unclassified 1.2
highway=minor 1.2
highway=residential 1.2
highway=living_street 1.2
highway=track 1.2
highway=service 1.2
highway=bridleway 1.2
highway=footway 1.05
highway=path 1.05
highway=steps 1.15
highway=pedestrian 1.0
highway=platform 1.1
railway=platform 1.1
public_transport=platform 1.1

Access tags: access, foot.

Disallows motorroads.

One-way is only considered when explicitly tagged with oneway:foot or on highway=footway, highway=path, highway=steps, highway/public_transport/railway=platform.

Turn restrictions are only considered when explicitly tagged with restriction:foot.

RAILWAY Penalties:
Tag Penalty
railway=rail 1.0
railway=light_rail 1.0
railway=subway 1.0
railway=narrow_gauge 1.0

Access tags: access, train.

Allows motorroads and considers turn restrictions.

TRAM Penalties:
Tag Penalty
railway=tram 1.0
railway=light_rail 1.0

Access tags: access, tram.

Allows motorroads and considers turn restrictions.

SUBWAY Penalties:
Tag Penalty
railway=subway 1.0

Access tags: access, subway.

Allows motorroads and considers turn restrictions.

routx.OsmFormat

class OsmFormat(IntEnum):
    UNKNOWN = 0
    XML = 1
    XML_GZ = 2
    XML_BZ2 = 3
    PBF = 4

Format of the input OSM file. An IntEnum.

Values:

  • UNKNOWN - unknown format - guess based on content.
  • XML - force uncompressed OSM XML.
  • XML_GZ - force OSM XML with gzip compression.
  • XML_BZ2 - force OSM XML with bzip2 compression.
  • PBF - force OSM PBF.

routx.OsmPenalty

class OsmPenalty(NamedTuple):
    key: str
    value: str
    penalty: float

Numeric multiplier for OSM ways with specific keys and values. A named tuple.

Attributes:

  • key: str - key of an OSM way for which this penalty applies, used for value comparison (e.g. "highway" or "railway").
  • value: str - value under key of an OSM way for which this penalty applies.E.g. "motorway", "residential", or "rail".
  • penalty: float - multiplier of the length, to express preference for a specific way. Must be not less than one and be finite.

routx.OsmCustomProfile

@dataclass
class OsmCustomProfile:
    name: str
    penalties: list[OsmPenalty]
    access: list[str]
    disallow_motorroad: bool = False
    disable_restrictions: bool = False

Describes how to convert OSM data into a Graph. A dataclass.

If possible, usage of pre-defined OsmProfiles should be preferred. Using custom profile involves reallocation of all arrays and strings two times to match ABIs (first from Python to C, then from C to Rust). This is only a constant cost incurred on call to Graph.add_from_file or Graph.add_from_memory.

Attributes:

  • name: str - human readable name of the routing profile, customary the most specific access tag.

    This value is not used for actual OSM data interpretation, except when set to "foot", which adds the following logic:

    • oneway tags are ignored - only oneway:foot tags are considered, except on:
      • highway=footway,
      • highway=path,
      • highway=steps,
      • highway=platform
      • public_transport=platform,
      • railway=platform;
    • only restriction:foot turn restrictions are considered.
  • penalties: list[OsmPenalty] - tags of OSM ways which can be used for routing.

    A way is matched against all OsmPenalty objects in order, and once an exact key and value match is found; the way is used for routing, and each connection between two nodes gets a resulting cost equal to the distance between nodes multiplied the penalty.

    All penalties must be normal and not less than zero.

    For example, if there are two penalties:

    1. highway=motorway, penalty=1
    2. highway=trunk, penalty=1.5

    This will result in:

    • a highway=motorway stretch of 100 meters will be used for routing with a cost of 100.
    • a highway=trunk motorway of 100 meters will be used for routing with a cost of 150.
    • a highway=motorway_link or highway=primary won't be used for routing, as they do not any penalty.
  • access: list[str] - list of OSM access tags (in order from least to most specific) to consider when checking for road prohibitions.

    This list is used mainly to follow the access tags, but also to follow mode-specific one-way and turn restrictions.

  • disallow_motorroad: bool - force no routing over motorroad=yes ways.

  • disable_restrictions: bool - force ignoring of turn restrictions.

routx.OsmLoadingError

class OsmLoadingError(ValueError):
    pass  # attributes and constructor the same as for ValueError

Raised when the underlying library has failed to load OSM data data. See logs for details.

routx.SerializationError

class SerializationError(ValueError):
    pass

Raised when the underlying library has failed to read/write graph data. See logs for details.

routx.StepLimitExceeded

class StepLimitExceeded(ValueError):
    pass  # attributes and constructor the same as for ValueError

Raised when Graph.find_route exceeded its step limit.

routx.earth_distance

def earth_distance(
    lat1: float,
    lon1: float,
    lat2: float,
    lon2: float,
) -> float:

Calculates the great-circle distance between two positions using the haversine formula.

Returns the result in kilometers.

routx.simplify_line

simplify_line(line: Iterable[float], epsilon: float = 1e-5) -> array[float]

Simplifies a line (an iterable of point coordinates) using the Ramer-Douglas-Peucker algorithm.

Points must be encoded as [x0 y0 x1 y1 x2 y2 ...]. Any odd trailing elements are ignored.

If line is an instance of an array of type f (float32), then it is modified in-place, and a slice containing the simplified line is returned. Other elements are left undefined.

Epsilon represents the maximum distance (in decimal degrees, as the implementation assumes flat, Euclidean geometry) for a point's distance to a line segment to be considered insignificant and therefore removed.

License

routx and routx-python are made available under the MIT license.

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

routx-1.1.1.tar.gz (26.0 kB view details)

Uploaded Source

Built Distributions

If you're not sure about the file name format, learn more about wheel file names.

routx-1.1.1-py3-none-win_arm64.whl (324.6 kB view details)

Uploaded Python 3Windows ARM64

routx-1.1.1-py3-none-win_amd64.whl (350.7 kB view details)

Uploaded Python 3Windows x86-64

routx-1.1.1-py3-none-musllinux_1_1_x86_64.whl (486.3 kB view details)

Uploaded Python 3musllinux: musl 1.1+ x86-64

routx-1.1.1-py3-none-musllinux_1_1_aarch64.whl (453.1 kB view details)

Uploaded Python 3musllinux: musl 1.1+ ARM64

routx-1.1.1-py3-none-manylinux_2_17_x86_64.whl (487.3 kB view details)

Uploaded Python 3manylinux: glibc 2.17+ x86-64

routx-1.1.1-py3-none-manylinux_2_17_aarch64.whl (454.5 kB view details)

Uploaded Python 3manylinux: glibc 2.17+ ARM64

routx-1.1.1-py3-none-macosx_11_0_x86_64.whl (1.8 MB view details)

Uploaded Python 3macOS 11.0+ x86-64

routx-1.1.1-py3-none-macosx_11_0_arm64.whl (1.8 MB view details)

Uploaded Python 3macOS 11.0+ ARM64

File details

Details for the file routx-1.1.1.tar.gz.

File metadata

  • Download URL: routx-1.1.1.tar.gz
  • Upload date:
  • Size: 26.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.6

File hashes

Hashes for routx-1.1.1.tar.gz
Algorithm Hash digest
SHA256 5ff80bc32809ca34d8d49e3c4b03261603b14feeba43040ecf6b63232d4a5346
MD5 6066b4c3a4757239967b681ca6b6a0a6
BLAKE2b-256 d19516996d2c8641da92c7403a7889589030860e3da6d81b9339237558347091

See more details on using hashes here.

File details

Details for the file routx-1.1.1-py3-none-win_arm64.whl.

File metadata

  • Download URL: routx-1.1.1-py3-none-win_arm64.whl
  • Upload date:
  • Size: 324.6 kB
  • Tags: Python 3, Windows ARM64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.6

File hashes

Hashes for routx-1.1.1-py3-none-win_arm64.whl
Algorithm Hash digest
SHA256 958557dcaca3a92bd2e0a9119efc9c5db808a791933bd4df482b81765bc1179c
MD5 83bb835302703e8c45b59e2278ada20f
BLAKE2b-256 e9e5aa05070e45bc3a1b5f4371c80dad4419173bf2761e28002cb0ddf6c2667c

See more details on using hashes here.

File details

Details for the file routx-1.1.1-py3-none-win_amd64.whl.

File metadata

  • Download URL: routx-1.1.1-py3-none-win_amd64.whl
  • Upload date:
  • Size: 350.7 kB
  • Tags: Python 3, Windows x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.6

File hashes

Hashes for routx-1.1.1-py3-none-win_amd64.whl
Algorithm Hash digest
SHA256 e8beaf4511c85d06be4ac068ab5fdbb968708af62d21830c0e58d7dd5209a163
MD5 cbc0dc55cf03967cb358c53e055c5da9
BLAKE2b-256 e598f8aeed330d7c60f6fe1ee58c8290787319724bd497aca4ec7ecbb5be5e15

See more details on using hashes here.

File details

Details for the file routx-1.1.1-py3-none-musllinux_1_1_x86_64.whl.

File metadata

  • Download URL: routx-1.1.1-py3-none-musllinux_1_1_x86_64.whl
  • Upload date:
  • Size: 486.3 kB
  • Tags: Python 3, musllinux: musl 1.1+ x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.6

File hashes

Hashes for routx-1.1.1-py3-none-musllinux_1_1_x86_64.whl
Algorithm Hash digest
SHA256 1d0905fe22d326cdba4105d73f59f085104e28d489c075b222f3814de166cf1a
MD5 148e21b2f75fb26beac35bcde63441b1
BLAKE2b-256 c72b789cdf40bebd44a421cc16ed1b2e04894f8eeb5c317d56a910e0c2516e16

See more details on using hashes here.

File details

Details for the file routx-1.1.1-py3-none-musllinux_1_1_aarch64.whl.

File metadata

File hashes

Hashes for routx-1.1.1-py3-none-musllinux_1_1_aarch64.whl
Algorithm Hash digest
SHA256 c358f4f7c24fd0a62d5ff205bc1335ce8fc29962622cfcc822ca7908b4dcceb8
MD5 480983c7108079e3e12b6b588292ff35
BLAKE2b-256 a547c8a7ce7ecdc7b150de7b12e495bcb85000ea6335a6e8fc7428f1700048ee

See more details on using hashes here.

File details

Details for the file routx-1.1.1-py3-none-manylinux_2_17_x86_64.whl.

File metadata

File hashes

Hashes for routx-1.1.1-py3-none-manylinux_2_17_x86_64.whl
Algorithm Hash digest
SHA256 c6774648e2ce89d62ba1d63f66f1d81902ba2663cf50924147931b630e0e962f
MD5 0dd16226ad3a986ae8a299ecd34ff43c
BLAKE2b-256 9dec24e2a883b003fc4b09b77fc5d159bbf44340ca5f4f4502fe5a775aa18a36

See more details on using hashes here.

File details

Details for the file routx-1.1.1-py3-none-manylinux_2_17_aarch64.whl.

File metadata

File hashes

Hashes for routx-1.1.1-py3-none-manylinux_2_17_aarch64.whl
Algorithm Hash digest
SHA256 49fa37763ef81bf486622961d61983dddbeac97a470fbd922df34c2e585228d9
MD5 5738218090906c5031961962ee2d8204
BLAKE2b-256 820b99adba094c5cd56f672702fb87058347d5dd3711c298254e1f7572f40949

See more details on using hashes here.

File details

Details for the file routx-1.1.1-py3-none-macosx_11_0_x86_64.whl.

File metadata

File hashes

Hashes for routx-1.1.1-py3-none-macosx_11_0_x86_64.whl
Algorithm Hash digest
SHA256 919ceb8774d775b7a09458c6dea0b294c97e628fd016e24ef150512831243614
MD5 e41c92852e218e9f6c4fd8bd83c592b2
BLAKE2b-256 61bfb0b2bc6e20dc088866e9fc13bfb8a37e2118551fcbb61c4d1f3aabd7b5c4

See more details on using hashes here.

File details

Details for the file routx-1.1.1-py3-none-macosx_11_0_arm64.whl.

File metadata

  • Download URL: routx-1.1.1-py3-none-macosx_11_0_arm64.whl
  • Upload date:
  • Size: 1.8 MB
  • Tags: Python 3, macOS 11.0+ ARM64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.6

File hashes

Hashes for routx-1.1.1-py3-none-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 66a65fbbcc11c9eecb1d76f4ee9a6bd22481b39ccb715efe324db489b879f9c7
MD5 b35ad216cc337980a5f9afd933a7048c
BLAKE2b-256 4b681ac94a27dd106c786c9da9d4003526d28b59bb4690317651b916a663bdd7

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page