Rust D3 Geo

Rust 2021 Edition.

crates.io Documentation Download

This is a port d3-geo into RUST.

Here is a list of the currently supported projections.

Examples

Examples are provided to help developers convert their existing javascript to rust. To run the example please follow the "Running the examples" below.

Description
examples/globe/canvasrotating50m

This is a port into rust of this d3-geo example www.d3indepth.com/geographic/.

For perfomance reasons this example is best viewed as a static web site, rather the running the development build.

examples/globe/svg

SVG are useful when the semantic meaning of the data needs to be preserved. The example shows how to load/parse/display the globe as indivdual SVG PATH elements. It also includes code samples that generates SVG graticules.
examples/globe/dragandzoom

This is similar to the other globe applications. As an example it deliberatly mixes typescript methods with rust. The typescript is responsible for handling the mouse events and calculating the quaternion and finally calculating the appropiate change in rotation. In a typescript render loop calls to a rust function render the globe.
examples/projections

As a confidence building exercise, this demo shows a side by side comparison of the all the projections rendered by in both javascript and rust.

examples/ring
Sample code in both RUST and javascript that renders a complex multipolygon. ( Orthographic and Sterographic )

When to use the rust version of the library

The limits of the javascript library become obvious when developing interactive applications that process large datasets. For example the examples/globe applications operate on a 50m resolution map of the earth. On a desktop machine this is beyond the javascript version.

Current Status

Not all projections have been implemented and there are number of known bugs that I am activley working on. The majority of the library has been ported along with the associated tests. The aim is to eventaully release a "1.0" version. but at the moment is this alpha grade software.

Before the "1.0" release breaking changes trigger the minor version to increment. A summary of the change can be found in CHANGELOG.md


Here is a summary of things to-do before the crate is published.

Running the examples

See the details.


Requirements:


To view the application either create a devleopment build, or construct a static-web site as follows

Start And Run A Development Build

console git clone https://github.com/martinfrances107/rust_d3_geo.git cd rust_d3_geo/examples/ring/ npm install npm run start

The last command "npm run start" will automatically open your default browser at http:://localhost::8080

Building A Static Site

Much better performance can be acheived by bulding a static web site and viewing that directly.

console git clone https://github.com/martinfrances107/rust_d3_geo.git cd rust_d3_geo/examples/ring npm install npm run build npm run serve

This creates a static HTML site in the dist/ directory.

To view the site you cannot just point you browser at a location of the form file:://home/user/alice/dist/index.html

By security reasons, browsers prevent HTML pages with WASM files from being viewed this way. You must host the site first.


Benchmarking

See the details. In this project, we have two benchmarks, based on the ring and graticule examples ( see above. )

Also rustd3geo_voronoi uses this library, and that project contains a benchmark which contains an exact port of a benchmark in d3-geo-voronoi. Based on that benchmark rust is 31% faster, or permits a 37% increase in throughput.


Flamegraph

See the details.

profile_target is binary that outputs a HTML page containing a SVG image showing the globe with graticule markings.

A flamegraph can be created with the following

bash cd profile_target sudo CARGO_PROFILE_RELEASE_DEBUG=true cargo flamegraph

The complexity of rendering 240 countries/polygons provides a good view in memory allocation issues.


Future Multi thread support

See the details. On my todo list.


Architecture discussion

There is an aspect of the design that needs review. It related to the best way to implement a doubly-linked list which has cross links between nodes. A full discusion can be found here


Coding Standard

Unimplemented sections of the library

See the details. Support for a custom projection is not yet supported. For an example of this see the test labelled "projection.fitExtent(…) custom projection"

I am trying to get a program of mine to run faster, but I want this to eventually be a true library port. So feel free to add suggestions to my todo list.

A complete list of all ported projections can be found in invert-test.rs. Out of the 15 distinct projections listed only 7 have been ported so far.


Other To-do's

See the details.

Document API changes such as

Finally

todo.md contains a more detailed list