ziongh / H3.net

Port of Uber's H3 to .NET 5

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

H3 Logo

H3.net: A port of Uber's Hexagonal Hierarchical Geospatial Indexing System to C#

This is a port of Uber's H3 library to C#, with additional functionality to support NetTopologySuite geometries. It is based on release 3.7.2 of the library.

H3 is a geospatial indexing system using a hexagonal grid that can be (approximately) subdivided into finer and finer hexagonal grids, combining the benefits of a hexagonal grid with S2's hierarchical subdivisions.

Upstream documentation is available at https://h3geo.org/. Documentation for this port of the library is unfortunately currently limited to documentation in the source and examples via unit tests.

Work in Progress!

This is a work in progress and likely contains the odd silly bug or poorly performing/confusing implementation choice (PR's are welcome!). I have tried to make the library work and feel more "C# like" using operators, extension methods, enumerables vs. arrays and so on, but some things may be implemented more or less as straight ports of the upstream library. Other than the obvious difference in some method names and whatnot, the biggest difference vs. the upstream library is the reliance on NTS for geometries (e.g. cell boundary polygons) and algorithms such as polyfill instead of in-library classes such as the GeoBoundary and vertex graph stuff.

While the majority of the core H3 API should be here in one form or another, there's still gaps particularly in terms of documentation and tests to validate behaviour vs. upstream. My focus so far has been on getting the algorithm side of things (such as k-rings, polyfill and so on) working and tested as I needed those features for the project(s) I am working on, however given that the majority of those methods depend on the basics of the library working, test coverage is "ok". I also make use of the PostgreSQL bindings in my work, so I tend to validate results with that as well.

PRs to improve code, tests and documentation are definitely welcome, although please keep in mind I am quite busy these days and may be a bit slow to respond. Sorry in advance!

Installing

Available on nuget.org as pocketken.H3.

PM> Install-Package pocketken.H3 -Version 3.7.2

See CHANGES.md for a list of changes between releases.

Some Mostly-Pointless Benchmarks

There is an extremely basic set of benchmarks using BenchmarkDotNet that I have begun to use in order to track performance and perform optimizations as things progress. You can check the code out to run the benchmarks locally if you want, e.g.:

$ dotnet run --configuration Release --project .\test\H3.Benchmarks\H3.Benchmarks.csproj --filter *Uncompact* --join

All numbers here are from my primary Windows development VM:

BenchmarkDotNet=v0.12.1, OS=Windows 10.0.19042
AMD Ryzen 9 3900X, 1 CPU, 12 logical and 6 physical cores
.NET Core SDK=5.0.201
  [Host]        : .NET Core 5.0.4 (CoreCLR 5.0.421.11614, CoreFX 5.0.421.11614), X64 RyuJIT
  .NET Core 5.0 : .NET Core 5.0.4 (CoreCLR 5.0.421.11614, CoreFX 5.0.421.11614), X64 RyuJIT

Job=.NET Core 5.0  Runtime=.NET Core 5.0

While there are some comparisons here against H3Lib, I still need to work on getting some benchmarks for the other H3 package which wraps the native library; if anyone is interested in assisting PRs are welcome!

Hierarchy Ops

GetParentForResolution

Using 89283080dcbffff (Uber's SF Test index @ resolution 9) to get parent at resolution 0 (a silly microbenchmark):

Method Mean Error StdDev Gen 0 Gen 1 Gen 2 Allocated
pocketken.H3.GetParentForResolution 4.458 ns 0.0474 ns 0.0443 ns 0.0029 - - 24 B
H3Lib.ToParent 20.817 ns 0.1196 ns 0.1118 ns - - - -

GetChildrenForResolution

Using 89283080dcbffff (Uber's SF Test index @ resolution 9) to get all children at resolution 15.

Method Mean Error StdDev Gen 0 Gen 1 Gen 2 Allocated
pocketken.H3.GetChildrenForResolution 9.128 ms 0.1809 ms 0.2415 ms 796.8750 781.2500 484.3750 4.69 MB
H3Lib.ToChildren 9.671 ms 0.1904 ms 0.2266 ms 3453.1250 1671.8750 984.3750 23.55 MB

Lines

Line from 8e283080dc80007 to 8e48e1d7038d527 (DistanceTo of 554,625 cells).

Method Mean Error StdDev Gen 0 Gen 1 Gen 2 Allocated
pocketken.H3.LineTo 972.3 ms 8.73 ms 7.29 ms 45000.0000 11000.0000 1000.0000 355.44 MB
H3Lib.LineTo 4,422.1 ms 10.23 ms 9.57 ms 1057000.0000 3000.0000 1000.0000 8449.31 MB

Rings

hex is a hexagon index (8f48e1d7038d520).

Method Mean Error StdDev Gen 0 Gen 1 Gen 2 Allocated
'pocketken.H3.GetKRing(hex, k = 50)' 460.1 us 1.99 us 1.86 us 73.7305 36.6211 - 607.75 KB
'pocketken.H3.GetKRingFast(hex, k = 50)' 471.4 us 2.48 us 2.32 us 66.4063 33.2031 - 547.92 KB
'pocketken.H3.GetKRingSlow(hex, k = 50)' 4,852.8 us 38.70 us 36.20 us 179.6875 85.9375 85.9375 1634.09 KB
'H3Lib.KRingDistances(hex, k = 50)' 381.6 us 0.87 us 0.68 us 99.6094 99.6094 99.6094 486.59 KB

pent is a pentagon index (8e0800000000007) which forces the use of the iterative (recursive in the case of H3Lib) method of generating the ring due to the fast method's inability to handle pentagons.

Method Mean Error StdDev Gen 0 Gen 1 Gen 2 Allocated
'pocketken.H3.GetKRing(pent, k = 50)' 4.088 ms 0.0263 ms 0.0246 ms 179.6875 85.9375 85.9375 1.6 MB
'pocketken.H3.GetKRingSlow(pent, k = 50)' 3.985 ms 0.0138 ms 0.0115 ms 179.6875 85.9375 85.9375 1.6 MB
'H3Lib.KRingDistances(pent, k = 50)' 59,216.102 ms 960.4448 ms 898.4006 ms 7692000.0000 6112000.0000 5064000.0000 71358.66 MB

Sets

  • Compact: Result of compacting all cells at resolution 5.
  • Uncompact: Result of uncompacting all base cells to resolution of 5.
Method Mean Error StdDev Gen 0 Gen 1 Gen 2 Allocated
pocketken.H3.Compact 330.3 ms 5.50 ms 5.14 ms 11000.0000 3000.0000 - 243.51 MB
H3Lib.Compact 381.0 ms 5.83 ms 5.45 ms 9000.0000 4000.0000 2000.0000 305.24 MB
pocketken.H3.Uncompact 136.6 ms 0.36 ms 0.32 ms 6500.0000 3500.0000 1000.0000 78.18 MB
H3Lib.Uncompact 203.3 ms 4.05 ms 4.16 ms 43000.0000 7333.3333 666.6667 493.02 MB

About

Port of Uber's H3 to .NET 5

License:Apache License 2.0


Languages

Language:C# 100.0%