AshNeo4j.Geo (AshNeo4j v0.8.0)

Copy Markdown View Source

Geodesic geometry primitives for the in-memory side of AshNeo4j's spatial predicates.

The single source of truth for distance math. The in-memory st_* functions (st_distance, st_dwithin, st_closest_point) use these so they agree with the values Neo4j returns on the pushdown path.

Matching Neo4j's distance model

Neo4j's point.distance/2 for WGS-84 geographic points is a spherical haversine on the WGS-84 equatorial radius (the semi-major axis, 6 378 137 m) — not the mean Earth radius (6 371 000 m) a naive haversine would reach for. Using the mean radius disagrees with Neo4j by ~0.11 % (≈800 m over a 700 km span), which means the same st_distance query would return different answers depending on whether it pushed down to Cypher or evaluated in Elixir.

We deliberately use the same radius Neo4j does, so the two execution paths agree to sub-metre over continental distances. Neo4j's model is spherical, not ellipsoidal — true ellipsoidal distance (Vincenty / Karney) would differ by a further ~0.1–0.5 %, but since we push down to Neo4j, Neo4j's model is the reference we match.

Summary

Functions

The point on segment ab closest to p, as a {lng, lat} pair. Clamps to the segment's endpoints. See point_segment_meters/3 for the projection model.

Geodesic (great-circle haversine) distance in metres between two WGS-84 {lng, lat} coordinate pairs. Matches Neo4j's point.distance/2 to sub-metre over continental distances.

Minimum point_segment_meters/3 from p to any segment formed by consecutive vertices of coords (a list of {lng, lat} pairs). Used for point-to-LineString and point-to-polygon-ring-edge distance. A single-vertex list degenerates to the distance to that vertex; an empty list returns :infinity (a sentinel that orders above any real distance under min/2, so callers can fold it away).

Geodesic distance in metres from point p to the nearest point on the segment ab (each a {lng, lat} pair) — the true closest-point-on-segment distance, not closest-vertex.

Functions

closest_point_on_segment(arg, a, b)

@spec closest_point_on_segment(
  {number(), number()},
  {number(), number()},
  {number(), number()}
) ::
  {number(), number()}

The point on segment ab closest to p, as a {lng, lat} pair. Clamps to the segment's endpoints. See point_segment_meters/3 for the projection model.

haversine_meters(arg1, arg2)

@spec haversine_meters({number(), number()}, {number(), number()}) :: float()

Geodesic (great-circle haversine) distance in metres between two WGS-84 {lng, lat} coordinate pairs. Matches Neo4j's point.distance/2 to sub-metre over continental distances.

min_segment_meters(p, coords)

@spec min_segment_meters(
  {number(), number()},
  [{number(), number()}]
) :: float() | :infinity

Minimum point_segment_meters/3 from p to any segment formed by consecutive vertices of coords (a list of {lng, lat} pairs). Used for point-to-LineString and point-to-polygon-ring-edge distance. A single-vertex list degenerates to the distance to that vertex; an empty list returns :infinity (a sentinel that orders above any real distance under min/2, so callers can fold it away).

point_segment_meters(p, a, b)

@spec point_segment_meters(
  {number(), number()},
  {number(), number()},
  {number(), number()}
) :: float()

Geodesic distance in metres from point p to the nearest point on the segment ab (each a {lng, lat} pair) — the true closest-point-on-segment distance, not closest-vertex.

The closest point is found by projecting p onto the segment in a local equirectangular frame (longitude scaled by cos(lat) so the projection isn't distorted by meridian convergence), then the distance to that point is measured with haversine_meters/2. Accurate for the short segments of fibre paths and admin-boundary edges; a degenerate segment (a == b) falls back to the distance to a.