Skip to content

Commit

Permalink
Add tool to generate the Urquhart spanning graph
Browse files Browse the repository at this point in the history
It's technically a sub graph of the Delaunay triangulation graph, and a
super graph of the Euclidean minimal spanning tree
  • Loading branch information
Notgnoshi committed Nov 29, 2022
1 parent 197de9b commit bf1d7dc
Show file tree
Hide file tree
Showing 4 changed files with 128 additions and 3 deletions.
4 changes: 4 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,10 @@ path = "tools/transform/main.rs"
name = "triangulate"
path = "tools/triangulate/main.rs"

[[bin]]
name = "urquhart"
path = "tools/urquhart/main.rs"

[dependencies]
clap = {version="4.0", features=["derive"]}
delaunator = "1.0"
Expand Down
68 changes: 65 additions & 3 deletions generative/triangulation.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
use crate::wkio::write_wkt_geometries;
use clap::ValueEnum;
use geo::Point;
use petgraph::{Directed, Undirected};
use petgraph::{visit::EdgeRef, Directed, Undirected};

type NodeData = Point;
type EdgeWeight = ();
Expand Down Expand Up @@ -194,13 +196,73 @@ impl Triangulation {
}
}

#[derive(Debug, Clone, ValueEnum)]
pub enum GraphFormat {
Tgf,
Wkt,
}

pub fn write_graph<Direction, W>(
writer: W,
graph: petgraph::Graph<NodeData, EdgeWeight, Direction, NodeIndex>,
format: &GraphFormat,
) where
W: std::io::Write,
Direction: petgraph::EdgeType,
{
match format {
GraphFormat::Tgf => write_graph_tgf(writer, graph),
GraphFormat::Wkt => write_graph_wkt(writer, graph),
}
}

fn write_graph_tgf<Direction, W>(
mut writer: W,
graph: petgraph::Graph<NodeData, EdgeWeight, Direction, NodeIndex>,
) where
W: std::io::Write,
Direction: petgraph::EdgeType,
{
// let (nodes, edges) = graph.into_nodes_edges();
for idx in graph.node_indices() {
let coord = graph
.node_weight(idx)
.expect("Got index to nonexistent node.");
let index = idx.index();
writeln!(writer, "{}\tPOINT({} {})", index, coord.x(), coord.y())
.expect("Failed to write node label");
}
writeln!(writer, "#").expect("Failed to write node/edge separator");
for edge in graph.edge_references() {
writeln!(
writer,
"{}\t {}",
edge.source().index(),
edge.target().index()
)
.expect("Failed to write edge");
}
}

fn write_graph_wkt<Direction, W>(
writer: W,
graph: petgraph::Graph<NodeData, EdgeWeight, Direction, NodeIndex>,
) where
W: std::io::Write,
Direction: petgraph::EdgeType,
{
let edges = graph
.edge_references()
.map(|e| geo::Line::new(graph[e.source()], graph[e.target()]))
.map(geo::Geometry::Line);
write_wkt_geometries(writer, edges);
}

#[cfg(test)]
mod tests {
use super::*;
use crate::flatten::flatten_geometries_into_points_ref;
use crate::wkio::read_wkt_geometries;
#[cfg(feature = "test-io")]
use crate::wkio::write_wkt_geometries;
use delaunator::EMPTY;

#[test]
Expand Down
31 changes: 31 additions & 0 deletions tools/urquhart/cmdline.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
use clap::Parser;
use generative::triangulation::GraphFormat;
use generative::wkio::GeometryFormat;
use std::path::PathBuf;

/// Generate the Urquhart graph of the given geometries
///
/// Approximates the point cloud's relative neighborhood.
#[derive(Debug, Parser)]
#[clap(name = "urquhart", verbatim_doc_comment)]
pub struct CmdlineOptions {
/// Increase logging verbosity. Defaults to ERROR level.
#[clap(short, long, action = clap::ArgAction::Count)]
pub verbosity: u8,

/// Output file to write result to. Defaults to stdout.
#[clap(short, long)]
pub output: Option<PathBuf>,

/// Output geometry format.
#[clap(short = 'O', long, default_value = "wkt")]
pub output_format: GraphFormat,

/// Input file to read input from. Defaults to stdin.
#[clap(short, long)]
pub input: Option<PathBuf>,

/// Input geometry format.
#[clap(short = 'I', long, default_value_t = GeometryFormat::Wkt)]
pub input_format: GeometryFormat,
}
28 changes: 28 additions & 0 deletions tools/urquhart/main.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
mod cmdline;

use clap::Parser;
use generative::flatten::flatten_geometries_into_points;
use generative::stdio::{get_input_reader, get_output_writer};
use generative::triangulation::{triangulate, write_graph};
use generative::wkio::read_geometries;
use stderrlog::ColorChoice;

fn main() {
let args = cmdline::CmdlineOptions::parse();

stderrlog::new()
.verbosity(args.verbosity as usize + 1) // Default to WARN level.
.color(ColorChoice::Auto)
.init()
.expect("Failed to initialize stderrlog");

let reader = get_input_reader(&args.input).unwrap();
let geometries = read_geometries(reader, &args.input_format); // lazily loaded

let points = flatten_geometries_into_points(geometries);
let triangulation = triangulate(points);
let urquhart = triangulation.urquhart();

let writer = get_output_writer(&args.output).unwrap();
write_graph(writer, urquhart, &args.output_format);
}

0 comments on commit bf1d7dc

Please sign in to comment.