From 45885b8bac05fb85623d1919fecc69a40aeb350d Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 18 Nov 2024 16:08:55 +0000 Subject: [PATCH 01/14] Bump serde_json from 1.0.132 to 1.0.133 (#1321) Bumps [serde_json](https://github.com/serde-rs/json) from 1.0.132 to 1.0.133. - [Release notes](https://github.com/serde-rs/json/releases) - [Commits](https://github.com/serde-rs/json/compare/v1.0.132...v1.0.133) --- updated-dependencies: - dependency-name: serde_json dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- Cargo.lock | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 368420462..053eba660 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -596,9 +596,9 @@ dependencies = [ [[package]] name = "serde_json" -version = "1.0.132" +version = "1.0.133" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d726bfaff4b320266d395898905d0eba0345aae23b54aee3a737e260fd46db03" +checksum = "c7fceb2473b9166b2294ef05efcb65a3db80803f0b03ef86a5fc88a2b85ee377" dependencies = [ "itoa", "memchr", From ff611f22bf488d1b6dd06edb21ef46afc7c1b65b Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 18 Nov 2024 16:55:25 +0000 Subject: [PATCH 02/14] Bump quick-xml from 0.37.0 to 0.37.1 (#1320) Bumps [quick-xml](https://github.com/tafia/quick-xml) from 0.37.0 to 0.37.1. - [Release notes](https://github.com/tafia/quick-xml/releases) - [Changelog](https://github.com/tafia/quick-xml/blob/master/Changelog.md) - [Commits](https://github.com/tafia/quick-xml/compare/v0.37.0...v0.37.1) --- updated-dependencies: - dependency-name: quick-xml dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- Cargo.lock | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 053eba660..45fd8248a 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -425,9 +425,9 @@ dependencies = [ [[package]] name = "quick-xml" -version = "0.37.0" +version = "0.37.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ffbfb3ddf5364c9cfcd65549a1e7b801d0e8d1b14c1a1590a6408aa93cfbfa84" +checksum = "f22f29bdff3987b4d8632ef95fd6424ec7e4e0a57e2f4fc63e489e75357f6a03" dependencies = [ "memchr", ] From 0017d000e6bf82aed0da095410de6f4a4596fa4d Mon Sep 17 00:00:00 2001 From: Fabio Mazza Date: Tue, 19 Nov 2024 23:38:57 +0100 Subject: [PATCH 03/14] Read gzipped graphml files (#1315) * Use flate2 crate to read gzipped graphml files * fix typo * run rustfmt * apply suggestion from clippy * add test for gzipped graphml * write separate function * add changelog * reformat * Revert "write separate function" This reverts commit 2dba2529004f6cb7424eb27cbae319c12d6e4bf6. * run with compression argument * update contribution * lint python * add stub * try avoid error in test in Windows * use Option for compression variable * correct text signature --------- Co-authored-by: Ivan Carvalho <8753214+IvanIsCoding@users.noreply.github.com> --- Cargo.lock | 35 ++++++++++++ Cargo.toml | 1 + ...t-description-string-564c7e376b8e7304.yaml | 5 ++ rustworkx/rustworkx.pyi | 6 +- src/graphml.rs | 47 +++++++++++++--- tests/test_graphml.py | 55 ++++++++++++++++++- 6 files changed, 139 insertions(+), 10 deletions(-) create mode 100644 releasenotes/notes/short-description-string-564c7e376b8e7304.yaml diff --git a/Cargo.lock b/Cargo.lock index 45fd8248a..457b25b0b 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2,6 +2,12 @@ # It is not intended for manual editing. version = 3 +[[package]] +name = "adler2" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "512761e0bb2578dd7380c6baaa0f4ce03e84f95e960231d1dec8bf4d7d6e2627" + [[package]] name = "ahash" version = "0.8.11" @@ -39,6 +45,15 @@ version = "1.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" +[[package]] +name = "crc32fast" +version = "1.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a97769d94ddab943e4510d138150169a2758b5ef3eb191a9ee688de3e23ef7b3" +dependencies = [ + "cfg-if", +] + [[package]] name = "crossbeam-deque" version = "0.8.5" @@ -82,6 +97,16 @@ version = "0.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0ce7134b9999ecaf8bcd65542e436736ef32ddca1b3e06094cb6ec5755203b80" +[[package]] +name = "flate2" +version = "1.0.35" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c936bfdafb507ebbf50b8074c54fa31c5be9a1e7e5f467dd659697041407d07c" +dependencies = [ + "crc32fast", + "miniz_oxide", +] + [[package]] name = "getrandom" version = "0.2.15" @@ -194,6 +219,15 @@ dependencies = [ "autocfg", ] +[[package]] +name = "miniz_oxide" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e2d80299ef12ff69b16a84bb182e3b9df68b5a91574d3d4fa6e41b65deec4df1" +dependencies = [ + "adler2", +] + [[package]] name = "ndarray" version = "0.16.1" @@ -529,6 +563,7 @@ version = "0.16.0" dependencies = [ "ahash", "fixedbitset", + "flate2", "hashbrown 0.14.5", "indexmap", "ndarray", diff --git a/Cargo.toml b/Cargo.toml index 3bffc9f37..a07c76d63 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -60,6 +60,7 @@ serde = { version = "1.0", features = ["derive"] } serde_json = "1.0" smallvec = { version = "1.0", features = ["union"] } rustworkx-core = { path = "rustworkx-core", version = "=0.16.0" } +flate2 = "1.0.35" [dependencies.pyo3] version = "0.22.6" diff --git a/releasenotes/notes/short-description-string-564c7e376b8e7304.yaml b/releasenotes/notes/short-description-string-564c7e376b8e7304.yaml new file mode 100644 index 000000000..5ddc62d18 --- /dev/null +++ b/releasenotes/notes/short-description-string-564c7e376b8e7304.yaml @@ -0,0 +1,5 @@ +--- +features: + - | + Added the ability to read GraphML files that are compressed using gzip, with function :func:`~rustworkx.read_graphml`. + The extensions `.graphmlz` and `.gz` are automatically recognised, but the gzip decompression can be forced with the "compression" optional argument. diff --git a/rustworkx/rustworkx.pyi b/rustworkx/rustworkx.pyi index 69dcac8dd..beebd5041 100644 --- a/rustworkx/rustworkx.pyi +++ b/rustworkx/rustworkx.pyi @@ -646,7 +646,11 @@ def directed_random_bipartite_graph( # Read Write -def read_graphml(path: str, /) -> list[PyGraph | PyDiGraph]: ... +def read_graphml( + path: str, + /, + compression: str | None = ..., +) -> list[PyGraph | PyDiGraph]: ... def digraph_node_link_json( graph: PyDiGraph[_S, _T], /, diff --git a/src/graphml.rs b/src/graphml.rs index 6211b25ce..89c71b79d 100644 --- a/src/graphml.rs +++ b/src/graphml.rs @@ -13,11 +13,15 @@ #![allow(clippy::borrow_as_ptr)] use std::convert::From; +use std::ffi::OsStr; +use std::fs::File; +use std::io::{BufRead, BufReader}; use std::iter::FromIterator; use std::num::{ParseFloatError, ParseIntError}; use std::path::Path; use std::str::ParseBoolError; +use flate2::bufread::GzDecoder; use hashbrown::HashMap; use indexmap::IndexMap; @@ -524,19 +528,27 @@ impl GraphML { Ok(()) } + /// Open file compressed with gzip, using the GzDecoder + /// Returns a quick_xml Reader instance + fn open_file_gzip>( + path: P, + ) -> Result>>>, quick_xml::Error> { + let file = File::open(path)?; + let reader = BufReader::new(file); + let gzip_reader = BufReader::new(GzDecoder::new(reader)); + Ok(Reader::from_reader(gzip_reader)) + } - /// Parse a file written in GraphML format. + /// Parse a file written in GraphML format from a BufReader /// /// The implementation is based on a state machine in order to /// accept only valid GraphML syntax (e.g a `` element should /// be nested inside a `` element) where the internal state changes /// after handling each quick_xml event. - fn from_file>(path: P) -> Result { + fn read_graph_from_reader(mut reader: Reader) -> Result { let mut graphml = GraphML::default(); let mut buf = Vec::new(); - let mut reader = Reader::from_file(path)?; - let mut state = State::Start; let mut domain_of_last_key = Domain::Node; let mut last_data_key = String::new(); @@ -677,6 +689,23 @@ impl GraphML { Ok(graphml) } + + /// Read a graph from a file in the GraphML format + /// If the the file extension is "graphmlz" or "gz", decompress it on the fly + fn from_file>(path: P, compression: &str) -> Result { + let extension = path.as_ref().extension().unwrap_or(OsStr::new("")); + + let graph: Result = + if extension.eq("graphmlz") || extension.eq("gz") || compression.eq("gzip") { + let reader = Self::open_file_gzip(path)?; + Self::read_graph_from_reader(reader) + } else { + let reader = Reader::from_file(path)?; + Self::read_graph_from_reader(reader) + }; + + graph + } } /// Read a list of graphs from a file in GraphML format. @@ -703,9 +732,13 @@ impl GraphML { /// :rtype: list[Union[PyGraph, PyDiGraph]] /// :raises RuntimeError: when an error is encountered while parsing the GraphML file. #[pyfunction] -#[pyo3(text_signature = "(path, /)")] -pub fn read_graphml(py: Python, path: &str) -> PyResult> { - let graphml = GraphML::from_file(path)?; +#[pyo3(signature=(path, compression=None),text_signature = "(path, /, compression=None)")] +pub fn read_graphml( + py: Python, + path: &str, + compression: Option, +) -> PyResult> { + let graphml = GraphML::from_file(path, &compression.unwrap_or_default())?; let mut out = Vec::new(); for graph in graphml.graphs { diff --git a/tests/test_graphml.py b/tests/test_graphml.py index fee85da4a..517a79d26 100644 --- a/tests/test_graphml.py +++ b/tests/test_graphml.py @@ -12,6 +12,8 @@ import unittest import tempfile +import gzip + import numpy import rustworkx @@ -55,8 +57,8 @@ def assertGraphMLRaises(self, graph_xml): with self.assertRaises(Exception): rustworkx.read_graphml(fd.name) - def test_simple(self): - graph_xml = self.HEADER.format( + def graphml_xml_example(self): + return self.HEADER.format( """ yellow @@ -80,6 +82,8 @@ def test_simple(self): """ ) + def test_simple(self): + graph_xml = self.graphml_xml_example() with tempfile.NamedTemporaryFile("wt") as fd: fd.write(graph_xml) fd.flush() @@ -96,6 +100,53 @@ def test_simple(self): ] self.assertGraphEqual(graph, nodes, edges, directed=False) + def test_gzipped(self): + graph_xml = self.graphml_xml_example() + + ## Test reading a graphmlz + with tempfile.NamedTemporaryFile("w+b") as fd: + fd.flush() + newname = fd.name + ".gz" + with gzip.open(newname, "wt") as wf: + wf.write(graph_xml) + + graphml = rustworkx.read_graphml(newname) + graph = graphml[0] + nodes = [ + {"id": "n0", "color": "blue"}, + {"id": "n1", "color": "yellow"}, + {"id": "n2", "color": "green"}, + ] + edges = [ + ("n0", "n1", {"fidelity": 0.98}), + ("n0", "n2", {"fidelity": 0.95}), + ] + self.assertGraphEqual(graph, nodes, edges, directed=False) + + def test_gzipped_force(self): + graph_xml = self.graphml_xml_example() + + ## Test reading a graphmlz + with tempfile.NamedTemporaryFile("w+b") as fd: + # close the file + fd.flush() + newname = fd.name + ".ext" + with gzip.open(newname, "wt") as wf: + wf.write(graph_xml) + + graphml = rustworkx.read_graphml(newname, compression="gzip") + graph = graphml[0] + nodes = [ + {"id": "n0", "color": "blue"}, + {"id": "n1", "color": "yellow"}, + {"id": "n2", "color": "green"}, + ] + edges = [ + ("n0", "n1", {"fidelity": 0.98}), + ("n0", "n2", {"fidelity": 0.95}), + ] + self.assertGraphEqual(graph, nodes, edges, directed=False) + def test_multiple_graphs_in_single_file(self): graph_xml = self.HEADER.format( """ From 537f67f48e08b28bfdbe23cfcd6281831f911d89 Mon Sep 17 00:00:00 2001 From: Etienne Wodey <44871469+airwoodix@users.noreply.github.com> Date: Wed, 20 Nov 2024 23:33:29 +0100 Subject: [PATCH 04/14] Add immediate_dominators function (#1323) --- .../api/algorithm_functions/dominance.rst | 9 ++ docs/source/api/algorithm_functions/index.rst | 1 + ...immediate-dominators-0a713b22657cd19a.yaml | 6 + rustworkx/__init__.pyi | 1 + rustworkx/rustworkx.pyi | 4 + src/dominance.rs | 60 ++++++++ src/lib.rs | 3 + tests/digraph/test_dominance.py | 139 ++++++++++++++++++ 8 files changed, 223 insertions(+) create mode 100644 docs/source/api/algorithm_functions/dominance.rst create mode 100644 releasenotes/notes/digraph-immediate-dominators-0a713b22657cd19a.yaml create mode 100644 src/dominance.rs create mode 100644 tests/digraph/test_dominance.py diff --git a/docs/source/api/algorithm_functions/dominance.rst b/docs/source/api/algorithm_functions/dominance.rst new file mode 100644 index 000000000..d711e89a7 --- /dev/null +++ b/docs/source/api/algorithm_functions/dominance.rst @@ -0,0 +1,9 @@ +.. _dominance: + +Dominance +========= + +.. autosummary:: + :toctree: ../../apiref + + rustworkx.immediate_dominators diff --git a/docs/source/api/algorithm_functions/index.rst b/docs/source/api/algorithm_functions/index.rst index 1241cae34..0bbd84b28 100644 --- a/docs/source/api/algorithm_functions/index.rst +++ b/docs/source/api/algorithm_functions/index.rst @@ -10,6 +10,7 @@ Algorithm Functions coloring connectivity_and_cycles dag_algorithms + dominance graph_operations isomorphism link_analysis diff --git a/releasenotes/notes/digraph-immediate-dominators-0a713b22657cd19a.yaml b/releasenotes/notes/digraph-immediate-dominators-0a713b22657cd19a.yaml new file mode 100644 index 000000000..dc89cc6cc --- /dev/null +++ b/releasenotes/notes/digraph-immediate-dominators-0a713b22657cd19a.yaml @@ -0,0 +1,6 @@ +--- +features: + - | + Add :func:`rustworkx.immediate_dominators` function for computing + immediate dominators of all nodes in a directed graph. + This function mirrors the ``networkx.immediate_dominators`` function. diff --git a/rustworkx/__init__.pyi b/rustworkx/__init__.pyi index 2152b4f60..7a7af8be4 100644 --- a/rustworkx/__init__.pyi +++ b/rustworkx/__init__.pyi @@ -242,6 +242,7 @@ from .rustworkx import steiner_tree as steiner_tree from .rustworkx import metric_closure as metric_closure from .rustworkx import digraph_union as digraph_union from .rustworkx import graph_union as graph_union +from .rustworkx import immediate_dominators as immediate_dominators from .rustworkx import NodeIndices as NodeIndices from .rustworkx import PathLengthMapping as PathLengthMapping from .rustworkx import PathMapping as PathMapping diff --git a/rustworkx/rustworkx.pyi b/rustworkx/rustworkx.pyi index beebd5041..1655a759e 100644 --- a/rustworkx/rustworkx.pyi +++ b/rustworkx/rustworkx.pyi @@ -1052,6 +1052,10 @@ def graph_union( merge_edges: bool = ..., ) -> PyGraph[_S, _T]: ... +# Dominance + +def immediate_dominators(graph: PyDiGraph[_S, _T], start_node: int, /) -> dict[int, int]: ... + # Iterators _T_co = TypeVar("_T_co", covariant=True) diff --git a/src/dominance.rs b/src/dominance.rs new file mode 100644 index 000000000..2dd31e0f1 --- /dev/null +++ b/src/dominance.rs @@ -0,0 +1,60 @@ +// Licensed under the Apache License, Version 2.0 (the "License"); you may +// not use this file except in compliance with the License. You may obtain +// a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +// License for the specific language governing permissions and limitations +// under the License. + +use super::{digraph, InvalidNode, NullGraph}; +use rustworkx_core::dictmap::DictMap; + +use petgraph::algo::dominators; +use petgraph::graph::NodeIndex; + +use pyo3::prelude::*; + +/// Determine the immediate dominators of all nodes in a directed graph. +/// +/// The dominance computation uses the algorithm published in 2006 by +/// Cooper, Harvey, and Kennedy (https://hdl.handle.net/1911/96345). +/// The time complexity is quadratic in the number of vertices. +/// +/// :param PyDiGraph graph: directed graph +/// :param int start_node: the start node for the dominance computation +/// +/// :returns: a mapping of node indices to their immediate dominators +/// :rtype: dict[int, int] +/// +/// :raises NullGraph: the passed graph is empty +/// :raises InvalidNode: the start node is not in the graph +#[pyfunction] +#[pyo3(text_signature = "(graph, start_node, /)")] +pub fn immediate_dominators( + graph: &digraph::PyDiGraph, + start_node: usize, +) -> PyResult> { + if graph.graph.node_count() == 0 { + return Err(NullGraph::new_err("Invalid operation on a NullGraph")); + } + + let start_node_index = NodeIndex::new(start_node); + + if !graph.graph.contains_node(start_node_index) { + return Err(InvalidNode::new_err("Start node is not in the graph")); + } + + let dom = dominators::simple_fast(&graph.graph, start_node_index); + + // Include the root node to match networkx.immediate_dominators + let root_dom = [(start_node, start_node)]; + let others_dom = graph.graph.node_indices().filter_map(|index| { + dom.immediate_dominator(index) + .map(|res| (index.index(), res.index())) + }); + Ok(root_dom.into_iter().chain(others_dom).collect()) +} diff --git a/src/lib.rs b/src/lib.rs index 4d83e4158..45a9629d6 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -17,6 +17,7 @@ mod coloring; mod connectivity; mod dag_algo; mod digraph; +mod dominance; mod dot_utils; mod generators; mod graph; @@ -47,6 +48,7 @@ use centrality::*; use coloring::*; use connectivity::*; use dag_algo::*; +use dominance::*; use graphml::*; use isomorphism::*; use json::*; @@ -464,6 +466,7 @@ fn rustworkx(py: Python<'_>, m: &Bound) -> PyResult<()> { m.add_wrapped(wrap_pyfunction!(graph_vf2_mapping))?; m.add_wrapped(wrap_pyfunction!(digraph_union))?; m.add_wrapped(wrap_pyfunction!(graph_union))?; + m.add_wrapped(wrap_pyfunction!(immediate_dominators))?; m.add_wrapped(wrap_pyfunction!(digraph_maximum_bisimulation))?; m.add_wrapped(wrap_pyfunction!(digraph_cartesian_product))?; m.add_wrapped(wrap_pyfunction!(graph_cartesian_product))?; diff --git a/tests/digraph/test_dominance.py b/tests/digraph/test_dominance.py new file mode 100644 index 000000000..ed37a5496 --- /dev/null +++ b/tests/digraph/test_dominance.py @@ -0,0 +1,139 @@ +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +import unittest + +import rustworkx as rx +import networkx as nx + + +class TestImmediateDominators(unittest.TestCase): + """Test `rustworkx.immediate_dominators`. + + Test cases adapted from `networkx`: + https://github.com/networkx/networkx/blob/9c5ca54b7e5310a21568bb2e0104f8c87bf74ff7/networkx/algorithms/tests/test_dominance.py + (Copyright 2004-2024 NetworkX Developers, 3-clause BSD License) + """ + + def test_empty(self): + """ + Edge case: empty graph. + """ + graph = rx.PyDiGraph() + + with self.assertRaises(rx.NullGraph): + rx.immediate_dominators(graph, 0) + + def test_start_node_not_in_graph(self): + """ + Edge case: start_node is not in the graph. + """ + graph = rx.PyDiGraph() + graph.add_node(0) + + self.assertEqual(list(graph.node_indices()), [0]) + + with self.assertRaises(rx.InvalidNode): + rx.immediate_dominators(graph, 1) + + def test_singleton(self): + """ + Edge cases: single node, optionally cyclic. + """ + graph = rx.PyDiGraph() + graph.add_node(0) + self.assertDictEqual(rx.immediate_dominators(graph, 0), {0: 0}) + graph.add_edge(0, 0, None) + self.assertDictEqual(rx.immediate_dominators(graph, 0), {0: 0}) + + nx_graph = nx.DiGraph() + nx_graph.add_edges_from(graph.edge_list()) + self.assertDictEqual(nx.immediate_dominators(nx_graph, 0), {0: 0}) + + def test_irreducible1(self): + """ + Graph taken from figure 2 of "A simple, fast dominance algorithm." (2006). + https://hdl.handle.net/1911/96345 + """ + edges = [(1, 2), (2, 1), (3, 2), (4, 1), (5, 3), (5, 4)] + graph = rx.PyDiGraph() + graph.add_node(0) + graph.extend_from_edge_list(edges) + + result = rx.immediate_dominators(graph, 5) + self.assertDictEqual(result, {i: 5 for i in range(1, 6)}) + + nx_graph = nx.DiGraph() + nx_graph.add_edges_from(graph.edge_list()) + self.assertDictEqual(nx.immediate_dominators(nx_graph, 5), result) + + def test_irreducible2(self): + """ + Graph taken from figure 4 of "A simple, fast dominance algorithm." (2006). + https://hdl.handle.net/1911/96345 + """ + edges = [(1, 2), (2, 1), (2, 3), (3, 2), (4, 2), (4, 3), (5, 1), (6, 4), (6, 5)] + graph = rx.PyDiGraph() + graph.add_node(0) + graph.extend_from_edge_list(edges) + + result = rx.immediate_dominators(graph, 6) + self.assertDictEqual(result, {i: 6 for i in range(1, 7)}) + + nx_graph = nx.DiGraph() + nx_graph.add_edges_from(graph.edge_list()) + self.assertDictEqual(nx.immediate_dominators(nx_graph, 6), result) + + def test_domrel_png(self): + """ + Graph taken from https://commons.wikipedia.org/wiki/File:Domrel.png + """ + edges = [(1, 2), (2, 3), (2, 4), (2, 6), (3, 5), (4, 5), (5, 2)] + graph = rx.PyDiGraph() + graph.add_node(0) + graph.extend_from_edge_list(edges) + + result = rx.immediate_dominators(graph, 1) + self.assertDictEqual(result, {1: 1, 2: 1, 3: 2, 4: 2, 5: 2, 6: 2}) + + nx_graph = nx.DiGraph() + nx_graph.add_edges_from(graph.edge_list()) + self.assertDictEqual(nx.immediate_dominators(nx_graph, 1), result) + + # Test postdominance. + graph.reverse() + result = rx.immediate_dominators(graph, 6) + self.assertDictEqual(result, {1: 2, 2: 6, 3: 5, 4: 5, 5: 2, 6: 6}) + + self.assertDictEqual(nx.immediate_dominators(nx_graph.reverse(copy=False), 6), result) + + def test_boost_example(self): + """ + Graph taken from Figure 1 of + http://www.boost.org/doc/libs/1_56_0/libs/graph/doc/lengauer_tarjan_dominator.htm + """ + edges = [(0, 1), (1, 2), (1, 3), (2, 7), (3, 4), (4, 5), (4, 6), (5, 7), (6, 4)] + graph = rx.PyDiGraph() + graph.extend_from_edge_list(edges) + result = rx.immediate_dominators(graph, 0) + self.assertDictEqual(result, {0: 0, 1: 0, 2: 1, 3: 1, 4: 3, 5: 4, 6: 4, 7: 1}) + + nx_graph = nx.DiGraph() + nx_graph.add_edges_from(graph.edge_list()) + self.assertDictEqual(nx.immediate_dominators(nx_graph, 0), result) + + # Test postdominance. + graph.reverse() + result = rx.immediate_dominators(graph, 7) + self.assertDictEqual(result, {0: 1, 1: 7, 2: 7, 3: 4, 4: 5, 5: 7, 6: 4, 7: 7}) + + self.assertDictEqual(nx.immediate_dominators(nx_graph.reverse(copy=False), 7), result) From eaee0b577f25ce1e773f2fd7df8fb0e779981b1c Mon Sep 17 00:00:00 2001 From: "mergify[bot]" <37929162+mergify[bot]@users.noreply.github.com> Date: Sat, 23 Nov 2024 17:46:49 +0000 Subject: [PATCH 05/14] ci(mergify): upgrade configuration to current format (#1326) * ci(mergify): upgrade configuration to current format * Use mergify only for backporting --------- Co-authored-by: Mergify <37929162+mergify[bot]@users.noreply.github.com> Co-authored-by: Ivan Carvalho --- .github/.mergify.yml | 8 ++++++++ .mergify.yml | 33 --------------------------------- 2 files changed, 8 insertions(+), 33 deletions(-) create mode 100644 .github/.mergify.yml delete mode 100644 .mergify.yml diff --git a/.github/.mergify.yml b/.github/.mergify.yml new file mode 100644 index 000000000..4a9ee1ef6 --- /dev/null +++ b/.github/.mergify.yml @@ -0,0 +1,8 @@ +pull_request_rules: + - name: backport + conditions: + - label=stable-backport-potential + actions: + backport: + branches: + - stable/0.15 diff --git a/.mergify.yml b/.mergify.yml deleted file mode 100644 index 3438adc73..000000000 --- a/.mergify.yml +++ /dev/null @@ -1,33 +0,0 @@ -queue_rules: - - name: automerge - conditions: - - check-success=python3.9-x64 windows-latest - - check-success=python3.9-x64 ubuntu-latest - - check-success=python3.9-x64 macOS-latest - - check-success=python3.10-x64 windows-latest - - check-success=python3.10-x64 ubuntu-latest - - check-success=python3.10-x64 macOS-latest - - check-success=python3.11-x64 windows-latest - - check-success=python3.11-x64 ubuntu-latest - - check-success=python3.11-x64 macOS-latest - - check-success=Build, rustfmt, and python lint - - check-success=Coverage - - check-success=Build Docs - -pull_request_rules: - - name: automatic merge on CI success and review - conditions: - - "#approved-reviews-by>=1" - - label=automerge - - label!=on hold - actions: - queue: - name: automerge - method: squash - - name: backport - conditions: - - label=stable-backport-potential - actions: - backport: - branches: - - stable/0.14 From 37bee6f326aede01926f8b528a6ef536e07a4961 Mon Sep 17 00:00:00 2001 From: Etienne Wodey <44871469+airwoodix@users.noreply.github.com> Date: Sat, 23 Nov 2024 23:39:23 +0100 Subject: [PATCH 06/14] Add dominance_frontiers function (#1329) Co-authored-by: Ivan Carvalho <8753214+IvanIsCoding@users.noreply.github.com> --- .../api/algorithm_functions/dominance.rst | 1 + .../dominance-frontiers-6e3dcd59e9201b24.yaml | 6 + rustworkx/__init__.pyi | 1 + rustworkx/rustworkx.pyi | 1 + src/dominance.rs | 52 +++++ src/lib.rs | 1 + tests/digraph/test_dominance.py | 206 ++++++++++++++++++ 7 files changed, 268 insertions(+) create mode 100644 releasenotes/notes/dominance-frontiers-6e3dcd59e9201b24.yaml diff --git a/docs/source/api/algorithm_functions/dominance.rst b/docs/source/api/algorithm_functions/dominance.rst index d711e89a7..368ea2641 100644 --- a/docs/source/api/algorithm_functions/dominance.rst +++ b/docs/source/api/algorithm_functions/dominance.rst @@ -7,3 +7,4 @@ Dominance :toctree: ../../apiref rustworkx.immediate_dominators + rustworkx.dominance_frontiers diff --git a/releasenotes/notes/dominance-frontiers-6e3dcd59e9201b24.yaml b/releasenotes/notes/dominance-frontiers-6e3dcd59e9201b24.yaml new file mode 100644 index 000000000..947537bd3 --- /dev/null +++ b/releasenotes/notes/dominance-frontiers-6e3dcd59e9201b24.yaml @@ -0,0 +1,6 @@ +--- +features: + - | + Add the :func:`rustworkx.dominance_frontiers` function to compute + the dominance frontiers of all nodes in a directed graph. + This function mirrors the ``networkx.dominance_frontiers`` function. diff --git a/rustworkx/__init__.pyi b/rustworkx/__init__.pyi index 7a7af8be4..5952e177e 100644 --- a/rustworkx/__init__.pyi +++ b/rustworkx/__init__.pyi @@ -243,6 +243,7 @@ from .rustworkx import metric_closure as metric_closure from .rustworkx import digraph_union as digraph_union from .rustworkx import graph_union as graph_union from .rustworkx import immediate_dominators as immediate_dominators +from .rustworkx import dominance_frontiers as dominance_frontiers from .rustworkx import NodeIndices as NodeIndices from .rustworkx import PathLengthMapping as PathLengthMapping from .rustworkx import PathMapping as PathMapping diff --git a/rustworkx/rustworkx.pyi b/rustworkx/rustworkx.pyi index 1655a759e..4a18edd61 100644 --- a/rustworkx/rustworkx.pyi +++ b/rustworkx/rustworkx.pyi @@ -1055,6 +1055,7 @@ def graph_union( # Dominance def immediate_dominators(graph: PyDiGraph[_S, _T], start_node: int, /) -> dict[int, int]: ... +def dominance_frontiers(graph: PyDiGraph[_S, _T], start_node: int, /) -> dict[int, set[int]]: ... # Iterators diff --git a/src/dominance.rs b/src/dominance.rs index 2dd31e0f1..3f1dcc5a4 100644 --- a/src/dominance.rs +++ b/src/dominance.rs @@ -13,6 +13,8 @@ use super::{digraph, InvalidNode, NullGraph}; use rustworkx_core::dictmap::DictMap; +use hashbrown::HashSet; + use petgraph::algo::dominators; use petgraph::graph::NodeIndex; @@ -58,3 +60,53 @@ pub fn immediate_dominators( }); Ok(root_dom.into_iter().chain(others_dom).collect()) } + +/// Compute the dominance frontiers of all nodes in a directed graph. +/// +/// The dominance and dominance frontiers computations use the +/// algorithms published in 2006 by Cooper, Harvey, and Kennedy +/// (https://hdl.handle.net/1911/96345). +/// +/// :param PyDiGraph graph: directed graph +/// :param int start_node: the start node for the dominance computation +/// +/// :returns: a mapping of node indices to their dominance frontiers +/// :rtype: dict[int, set[int]] +/// +/// :raises NullGraph: the passed graph is empty +/// :raises InvalidNode: the start node is not in the graph +#[pyfunction] +#[pyo3(text_signature = "(graph, start_node, /)")] +pub fn dominance_frontiers( + graph: &digraph::PyDiGraph, + start_node: usize, +) -> PyResult>> { + let idom = immediate_dominators(graph, start_node)?; + + let mut df: DictMap<_, _> = idom + .iter() + .map(|(&node, _)| (node, HashSet::default())) + .collect(); + + for (&node, &node_idom) in &idom { + let preds = graph.predecessor_indices(node); + if preds.nodes.len() >= 2 { + for mut runner in preds.nodes { + while runner != node_idom { + df.entry(runner) + .and_modify(|e| { + e.insert(node); + }) + .or_insert([node].into_iter().collect()); + if let Some(&runner_idom) = idom.get(&runner) { + runner = runner_idom; + } else { + break; + } + } + } + } + } + + Ok(df) +} diff --git a/src/lib.rs b/src/lib.rs index 45a9629d6..16c421453 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -467,6 +467,7 @@ fn rustworkx(py: Python<'_>, m: &Bound) -> PyResult<()> { m.add_wrapped(wrap_pyfunction!(digraph_union))?; m.add_wrapped(wrap_pyfunction!(graph_union))?; m.add_wrapped(wrap_pyfunction!(immediate_dominators))?; + m.add_wrapped(wrap_pyfunction!(dominance_frontiers))?; m.add_wrapped(wrap_pyfunction!(digraph_maximum_bisimulation))?; m.add_wrapped(wrap_pyfunction!(digraph_cartesian_product))?; m.add_wrapped(wrap_pyfunction!(graph_cartesian_product))?; diff --git a/tests/digraph/test_dominance.py b/tests/digraph/test_dominance.py index ed37a5496..33e1ed6a4 100644 --- a/tests/digraph/test_dominance.py +++ b/tests/digraph/test_dominance.py @@ -137,3 +137,209 @@ def test_boost_example(self): self.assertDictEqual(result, {0: 1, 1: 7, 2: 7, 3: 4, 4: 5, 5: 7, 6: 4, 7: 7}) self.assertDictEqual(nx.immediate_dominators(nx_graph.reverse(copy=False), 7), result) + + +class TestDominanceFrontiers(unittest.TestCase): + """ + Test `rustworkx.dominance_frontiers`. + + Test cases adapted from `networkx`: + https://github.com/networkx/networkx/blob/9c5ca54b7e5310a21568bb2e0104f8c87bf74ff7/networkx/algorithms/tests/test_dominance.py + (Copyright 2004-2024 NetworkX Developers, 3-clause BSD License) + """ + + def test_empty(self): + """ + Edge case: empty graph. + """ + graph = rx.PyDiGraph() + + with self.assertRaises(rx.NullGraph): + rx.dominance_frontiers(graph, 0) + + def test_start_node_not_in_graph(self): + """ + Edge case: start_node is not in the graph. + """ + graph = rx.PyDiGraph() + graph.add_node(0) + + self.assertEqual(list(graph.node_indices()), [0]) + + with self.assertRaises(rx.InvalidNode): + rx.dominance_frontiers(graph, 1) + + def test_singleton(self): + """ + Edge cases: single node, optionally cyclic. + """ + graph = rx.PyDiGraph() + graph.add_node(0) + self.assertDictEqual(rx.dominance_frontiers(graph, 0), {0: set()}) + + graph.add_edge(0, 0, None) + self.assertDictEqual(rx.dominance_frontiers(graph, 0), {0: set()}) + + def test_irreducible1(self): + """ + Graph taken from figure 2 of "A simple, fast dominance algorithm." (2006). + https://hdl.handle.net/1911/96345 + """ + edges = [(1, 2), (2, 1), (3, 2), (4, 1), (5, 3), (5, 4)] + graph = rx.PyDiGraph() + graph.add_node(0) + graph.extend_from_edge_list(edges) + + result = rx.dominance_frontiers(graph, 5) + self.assertDictEqual(result, {1: {2}, 2: {1}, 3: {2}, 4: {1}, 5: set()}) + + nx_graph = nx.DiGraph() + nx_graph.add_edges_from(graph.edge_list()) + self.assertDictEqual(nx.dominance_frontiers(nx_graph, 5), result) + + def test_irreducible2(self): + """ + Graph taken from figure 4 of "A simple, fast dominance algorithm." (2006). + https://hdl.handle.net/1911/96345 + """ + edges = [(1, 2), (2, 1), (2, 3), (3, 2), (4, 2), (4, 3), (5, 1), (6, 4), (6, 5)] + graph = rx.PyDiGraph() + graph.add_node(0) + graph.extend_from_edge_list(edges) + + result = rx.dominance_frontiers(graph, 6) + + self.assertDictEqual( + result, + { + 1: {2}, + 2: {1, 3}, + 3: {2}, + 4: {2, 3}, + 5: {1}, + 6: set(), + }, + ) + + nx_graph = nx.DiGraph() + nx_graph.add_edges_from(graph.edge_list()) + self.assertDictEqual(nx.dominance_frontiers(nx_graph, 6), result) + + def test_domrel_png(self): + """ + Graph taken from https://commons.wikipedia.org/wiki/File:Domrel.png + """ + edges = [(1, 2), (2, 3), (2, 4), (2, 6), (3, 5), (4, 5), (5, 2)] + graph = rx.PyDiGraph() + graph.add_node(0) + graph.extend_from_edge_list(edges) + + result = rx.dominance_frontiers(graph, 1) + + self.assertDictEqual( + result, + { + 1: set(), + 2: {2}, + 3: {5}, + 4: {5}, + 5: {2}, + 6: set(), + }, + ) + + nx_graph = nx.DiGraph() + nx_graph.add_edges_from(graph.edge_list()) + self.assertDictEqual(nx.dominance_frontiers(nx_graph, 1), result) + + # Test postdominance. + graph.reverse() + result = rx.dominance_frontiers(graph, 6) + self.assertDictEqual( + result, + { + 1: set(), + 2: {2}, + 3: {2}, + 4: {2}, + 5: {2}, + 6: set(), + }, + ) + + self.assertDictEqual(nx.dominance_frontiers(nx_graph.reverse(copy=False), 6), result) + + def test_boost_example(self): + """ + Graph taken from Figure 1 of + http://www.boost.org/doc/libs/1_56_0/libs/graph/doc/lengauer_tarjan_dominator.htm + """ + edges = [(0, 1), (1, 2), (1, 3), (2, 7), (3, 4), (4, 5), (4, 6), (5, 7), (6, 4)] + graph = rx.PyDiGraph() + graph.extend_from_edge_list(edges) + + nx_graph = nx.DiGraph() + nx_graph.add_edges_from(graph.edge_list()) + + result = rx.dominance_frontiers(graph, 0) + self.assertDictEqual( + result, + { + 0: set(), + 1: set(), + 2: {7}, + 3: {7}, + 4: {4, 7}, + 5: {7}, + 6: {4}, + 7: set(), + }, + ) + + self.assertDictEqual(nx.dominance_frontiers(nx_graph, 0), result) + + # Test postdominance + graph.reverse() + result = rx.dominance_frontiers(graph, 7) + self.assertDictEqual( + result, + { + 0: set(), + 1: set(), + 2: {1}, + 3: {1}, + 4: {1, 4}, + 5: {1}, + 6: {4}, + 7: set(), + }, + ) + + self.assertDictEqual(nx.dominance_frontiers(nx_graph.reverse(copy=False), 7), result) + + def test_missing_immediate_doms(self): + """ + Test that the `dominance_frontiers` function doesn't regress on + https://github.com/networkx/networkx/issues/2070 + """ + edges = [(0, 1), (1, 2), (2, 3), (3, 4), (5, 3)] + graph = rx.PyDiGraph() + graph.extend_from_edge_list(edges) + + idom = rx.immediate_dominators(graph, 0) + self.assertNotIn(5, idom) + + # In networkx#2070, the call would fail because node 5 + # has no immediate dominators + result = rx.dominance_frontiers(graph, 0) + self.assertDictEqual( + result, + { + 0: set(), + 1: set(), + 2: set(), + 3: set(), + 4: set(), + 5: {3}, + }, + ) From b66662f7747642fadd1993686437a1fb2a7beb64 Mon Sep 17 00:00:00 2001 From: Etienne Wodey <44871469+airwoodix@users.noreply.github.com> Date: Sat, 30 Nov 2024 18:46:46 +0100 Subject: [PATCH 07/14] Fix clippy for Rust 1.83 (#1332) * clippy: fix empty_line_after_doc_comment * clippy: fix empty_line_after_outer_attr * clippy: autofix needless_return and needless_lifetimes * reno: add fragment for #1332 * Revert "reno: add fragment for #1332" This reverts commit 9d1b0c98dcf22206d05eef5e071fd7b19d96c905. --------- Co-authored-by: Ivan Carvalho --- rustworkx-core/src/generators/star_graph.rs | 2 +- rustworkx-core/src/token_swapper.rs | 1 - src/dag_algo/mod.rs | 6 +++--- src/digraph.rs | 4 ++-- src/graph.rs | 2 +- src/iterators.rs | 12 ++++++------ src/lib.rs | 2 +- 7 files changed, 14 insertions(+), 15 deletions(-) diff --git a/rustworkx-core/src/generators/star_graph.rs b/rustworkx-core/src/generators/star_graph.rs index 4fea65d27..a09565c93 100644 --- a/rustworkx-core/src/generators/star_graph.rs +++ b/rustworkx-core/src/generators/star_graph.rs @@ -34,7 +34,7 @@ use super::InvalidInputError; /// * `bidirectional` - Whether edges are added bidirectionally. If set to /// `true` then for any edge `(u, v)` an edge `(v, u)` will also be added. /// If the graph is undirected this will result in a parallel edge. - +/// /// # Example /// ```rust /// use rustworkx_core::petgraph; diff --git a/rustworkx-core/src/token_swapper.rs b/rustworkx-core/src/token_swapper.rs index a982d35e7..39695a58d 100644 --- a/rustworkx-core/src/token_swapper.rs +++ b/rustworkx-core/src/token_swapper.rs @@ -437,7 +437,6 @@ where /// assert_eq!(3, output.len()); /// /// ``` - pub fn token_swapper( graph: G, mapping: HashMap, diff --git a/src/dag_algo/mod.rs b/src/dag_algo/mod.rs index 769582ce1..0adfff12a 100644 --- a/src/dag_algo/mod.rs +++ b/src/dag_algo/mod.rs @@ -524,7 +524,7 @@ pub fn collect_runs( // This is where a filter function error will be returned, otherwise Result is stripped away let py_run: Vec = run_result? .iter() - .map(|node| return graph.graph.node_weight(*node).into_py(py)) + .map(|node| graph.graph.node_weight(*node).into_py(py)) .collect(); result.push(py_run) @@ -667,7 +667,7 @@ pub fn transitive_reduction( ); } } - return Ok(( + Ok(( digraph::PyDiGraph { graph: tr, node_removed: false, @@ -680,5 +680,5 @@ pub fn transitive_reduction( .iter() .map(|(k, v)| (k.index(), v.index())) .collect::>(), - )); + )) } diff --git a/src/digraph.rs b/src/digraph.rs index 07aa2be14..47b70b93a 100644 --- a/src/digraph.rs +++ b/src/digraph.rs @@ -200,7 +200,7 @@ impl GraphBase for PyDiGraph { type EdgeId = EdgeIndex; } -impl<'a> NodesRemoved for &'a PyDiGraph { +impl NodesRemoved for &PyDiGraph { fn nodes_removed(&self) -> bool { self.node_removed } @@ -886,7 +886,7 @@ impl PyDiGraph { /// /// :param int node_a: The index for the first node /// :param int node_b: The index for the second node - + /// /// :returns: A list with all the data objects for the edges between nodes /// :rtype: list /// :raises NoEdgeBetweenNodes: When there is no edge between nodes diff --git a/src/graph.rs b/src/graph.rs index f74933928..c3e627e5c 100644 --- a/src/graph.rs +++ b/src/graph.rs @@ -161,7 +161,7 @@ impl GraphBase for PyGraph { type EdgeId = EdgeIndex; } -impl<'a> NodesRemoved for &'a PyGraph { +impl NodesRemoved for &PyGraph { fn nodes_removed(&self) -> bool { self.node_removed } diff --git a/src/iterators.rs b/src/iterators.rs index 390aa3925..cabfbe1c3 100644 --- a/src/iterators.rs +++ b/src/iterators.rs @@ -282,7 +282,7 @@ where } } -impl<'py, T> PyEq> for T +impl PyEq> for T where for<'p> T: PyEq + Clone + FromPyObject<'p>, { @@ -1031,7 +1031,7 @@ impl PyHash for EdgeList { } } -impl<'py> PyEq> for EdgeList { +impl PyEq> for EdgeList { #[inline] fn eq(&self, other: &Bound, py: Python) -> PyResult { PyEq::eq(&self.edges, other, py) @@ -1119,7 +1119,7 @@ impl PyHash for IndexPartitionBlock { } } -impl<'py> PyEq> for IndexPartitionBlock { +impl PyEq> for IndexPartitionBlock { #[inline] fn eq(&self, other: &Bound, py: Python) -> PyResult { PyEq::eq(&self.block, other, py) @@ -1522,7 +1522,7 @@ impl PyHash for PathMapping { } } -impl<'py> PyEq> for PathMapping { +impl PyEq> for PathMapping { #[inline] fn eq(&self, other: &Bound, py: Python) -> PyResult { PyEq::eq(&self.paths, other, py) @@ -1686,7 +1686,7 @@ impl PyHash for MultiplePathMapping { } } -impl<'py> PyEq> for MultiplePathMapping { +impl PyEq> for MultiplePathMapping { #[inline] fn eq(&self, other: &Bound, py: Python) -> PyResult { PyEq::eq(&self.paths, other, py) @@ -1750,7 +1750,7 @@ impl PyHash for PathLengthMapping { } } -impl<'py> PyEq> for PathLengthMapping { +impl PyEq> for PathLengthMapping { #[inline] fn eq(&self, other: &Bound, py: Python) -> PyResult { PyEq::eq(&self.path_lengths, other, py) diff --git a/src/lib.rs b/src/lib.rs index 16c421453..4ee4189a7 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -203,7 +203,7 @@ pub trait NodesRemoved { fn nodes_removed(&self) -> bool; } -impl<'a, Ty> NodesRemoved for &'a StablePyGraph +impl NodesRemoved for &StablePyGraph where Ty: EdgeType, { From 5d28053f31240182d697a6841d595d6e5fcad8f1 Mon Sep 17 00:00:00 2001 From: Etienne Wodey <44871469+airwoodix@users.noreply.github.com> Date: Sat, 30 Nov 2024 19:45:45 +0100 Subject: [PATCH 08/14] rustworkx-core: fix docs build warnings (#1333) * core/steiner_tree: fix rustdoc warnings * core/generators/dorogovtsev_goltsev_mendes: fix rustdoc warnings * core/centrality: fix rustdoc warnings * workflows/main: error on warnings during rustworkx-core doc build * Pin Rust to 1.82 for clippy * Revert: Pin Rust to 1.82 to work around #1331 This reverts commit 1977f669fa2bd78ad7baaf5c1bf6f303e992daa6. --------- Co-authored-by: Ivan Carvalho <8753214+IvanIsCoding@users.noreply.github.com> --- .github/workflows/main.yml | 2 + rustworkx-core/src/centrality.rs | 11 +++-- .../dorogovtsev_goltsev_mendes_graph.rs | 17 +++---- rustworkx-core/src/steiner_tree.rs | 44 +++++++++++-------- 4 files changed, 44 insertions(+), 30 deletions(-) diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index 5579fb21a..f635c608e 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -46,6 +46,8 @@ jobs: run: pushd rustworkx-core && cargo test && popd - name: rustworkx-core Docs run: pushd rustworkx-core && cargo doc && popd + env: + RUSTDOCFLAGS: '-D warnings' - uses: actions/upload-artifact@v4 with: name: rustworkx_core_docs diff --git a/rustworkx-core/src/centrality.rs b/rustworkx-core/src/centrality.rs index 47dfd2136..a90876e4d 100644 --- a/rustworkx-core/src/centrality.rs +++ b/rustworkx-core/src/centrality.rs @@ -1073,18 +1073,21 @@ mod test_katz_centrality { /// In the case of a graphs with more than one connected component there is /// an alternative improved formula that calculates the closeness centrality /// as "a ratio of the fraction of actors in the group who are reachable, to -/// the average distance" [^WF]. You can enable this by setting `wf_improved` to `true`. +/// the average distance".[^WF] +/// You can enable this by setting `wf_improved` to `true`. /// -/// [^WF] Wasserman, S., & Faust, K. (1994). Social Network Analysis: +/// [^WF]: Wasserman, S., & Faust, K. (1994). Social Network Analysis: /// Methods and Applications (Structural Analysis in the Social Sciences). -/// Cambridge: Cambridge University Press. doi:10.1017/CBO9780511815478 +/// Cambridge: Cambridge University Press. +/// /// -/// Arguments: +/// # Arguments /// /// * `graph` - The graph object to run the algorithm on /// * `wf_improved` - If `true`, scale by the fraction of nodes reachable. /// /// # Example +/// /// ```rust /// use rustworkx_core::petgraph; /// use rustworkx_core::centrality::closeness_centrality; diff --git a/rustworkx-core/src/generators/dorogovtsev_goltsev_mendes_graph.rs b/rustworkx-core/src/generators/dorogovtsev_goltsev_mendes_graph.rs index 99954f498..2bd10991b 100644 --- a/rustworkx-core/src/generators/dorogovtsev_goltsev_mendes_graph.rs +++ b/rustworkx-core/src/generators/dorogovtsev_goltsev_mendes_graph.rs @@ -16,18 +16,19 @@ use super::InvalidInputError; /// Generate a Dorogovtsev-Goltsev-Mendes graph /// -/// Generate a graph following the recursive procedure in [1]. +/// Generate a graph following the recursive procedure by Dorogovtsev, Goltsev, +/// and Mendes[^DGM2002]. /// Starting from the two-node, one-edge graph, iterating `n` times generates -/// a graph with `(3**n + 3) // 2` nodes and `3**n` edges. +/// a graph with `(3^n + 3) // 2` nodes and `3^n` edges. /// +/// # Arguments /// -/// Arguments: -/// -/// * `n` - The number of iterations to perform. n=0 returns the two-node, one-edge graph. +/// * `n` - The number of iterations to perform. `n = 0` returns the two-node, one-edge graph. /// * `default_node_weight` - A callable that will return the weight to use for newly created nodes. /// * `default_edge_weight` - A callable that will return the weight object to use for newly created edges. /// /// # Example +/// /// ```rust /// use rustworkx_core::petgraph; /// use rustworkx_core::generators::dorogovtsev_goltsev_mendes_graph; @@ -43,10 +44,10 @@ use super::InvalidInputError; /// ); /// ``` /// -/// .. [1] S. N. Dorogovtsev, A. V. Goltsev and J. F. F. Mendes +/// [^DGM2002]: S. N. Dorogovtsev, A. V. Goltsev and J. F. F. Mendes /// “Pseudofractal scale-free web” -/// Physical Review E 65, 066122, 2002 -/// https://arxiv.org/abs/cond-mat/0112143 +/// Physical Review E 65, 066122 (2002) +/// /// pub fn dorogovtsev_goltsev_mendes_graph( n: usize, diff --git a/rustworkx-core/src/steiner_tree.rs b/rustworkx-core/src/steiner_tree.rs index f9bdcc60b..80fab566f 100644 --- a/rustworkx-core/src/steiner_tree.rs +++ b/rustworkx-core/src/steiner_tree.rs @@ -437,6 +437,9 @@ where Ok(out_edges) } +/// Solution to a minimum Steiner tree problem. +/// +/// This `struct` is created by the [steiner_tree] function. pub struct SteinerTreeResult { pub used_node_indices: HashSet, pub used_edge_endpoints: HashSet<(usize, usize)>, @@ -454,21 +457,25 @@ pub struct SteinerTreeResult { /// complete graph in which each edge is weighted by the shortest path distance /// between nodes in ``graph``. /// -/// This algorithm [1]_ produces a tree whose weight is within a -/// :math:`(2 - (2 / t))` factor of the weight of the optimal Steiner tree -/// where :math:`t` is the number of terminal nodes. The algorithm implemented -/// here is due to [2]_ . It avoids computing all pairs shortest paths but rather -/// reduces the problem to a single source shortest path and a minimum spanning tree -/// problem. +/// This algorithm by Kou, Markowsky, and Berman[^KouMarkowskyBerman1981] +/// produces a tree whose weight is within a `(2 - (2 / t))` factor of +/// the weight of the optimal Steiner tree where `t` is the number of +/// terminal nodes. +/// The algorithm implemented here is due to Mehlhorn[^Mehlhorn1987]. It avoids +/// computing all pairs shortest paths but rather reduces the problem to a +/// single source shortest path and a minimum spanning tree problem. /// -/// Arguments: -/// `graph`: The input graph to compute the steiner tree of -/// `terminal_nodes`: The terminal nodes of the steiner tree -/// `weight_fn`: A callable weight function that will be passed an edge reference -/// for each edge in the graph and it is expected to return a `Result` -/// which if it doesn't error represents the weight of that edge. +/// # Arguments +/// +/// - `graph` - The input graph to compute the Steiner tree of +/// - `terminal_nodes` - The terminal nodes of the Steiner tree +/// - `weight_fn` - A callable weight function that will be passed an edge reference +/// for each edge in the graph and it is expected to return a [`Result`] +/// which if it doesn't error represents the weight of that edge. +/// +/// # Returns /// -/// Returns a custom struct that contains a set of nodes and edges and `None` +/// A custom struct that contains a set of nodes and edges and `None` /// if the graph is disconnected relative to the terminal nodes. /// /// # Example @@ -509,13 +516,14 @@ pub struct SteinerTreeResult { /// let tree = steiner_tree(&input_graph, &terminal_nodes, weight_fn).unwrap().unwrap(); /// ``` /// -/// .. [1] Kou, Markowsky & Berman, +/// [^KouMarkowskyBerman1981]: Kou, Markowsky & Berman, /// "A fast algorithm for Steiner trees" -/// Acta Informatica 15, 141–145 (1981). -/// https://link.springer.com/article/10.1007/BF00288961 -/// .. [2] Kurt Mehlhorn, +/// Acta Informatica 15, 141–145 (1981) +/// +/// [^Mehlhorn1987]: Kurt Mehlhorn, /// "A faster approximation algorithm for the Steiner problem in graphs" -/// https://doi.org/10.1016/0020-0190(88)90066-X +/// Information Processing Letters 27(3), 125-128 (1987) +/// pub fn steiner_tree( graph: G, terminal_nodes: &[G::NodeId], From 680789b5147198b02f426fef036d0b2269e98044 Mon Sep 17 00:00:00 2001 From: Etienne Wodey <44871469+airwoodix@users.noreply.github.com> Date: Sat, 30 Nov 2024 20:24:43 +0100 Subject: [PATCH 09/14] Fix typos detected by the typos tool (#1330) * typos: apply automatic corrections * typos: apply manual corrections * typos: run as linter * typos: apply automatic corrections in releasenotes/ * typos: also check releasenotes content * Add release notes fragment * workflows/main: add spell check * Fix docs build * typos: upgrade to 1.28 * workflows/main: pin rust to 1.82 * workflows/main: remove spell check * Use Rust stable again * Typos: dedicated nox command * Move typos configuration to pyproject.toml --------- Co-authored-by: Ivan Carvalho Co-authored-by: Ivan Carvalho <8753214+IvanIsCoding@users.noreply.github.com> --- .github/workflows/main.yml | 2 +- CONTRIBUTING.md | 4 +-- README.md | 2 +- docs/source/benchmarks.rst | 2 +- docs/source/install.rst | 4 +-- docs/source/release_notes.rst | 4 +-- .../tutorial/betweenness_centrality.rst | 4 +-- docs/source/tutorial/dags.rst | 4 +-- noxfile.py | 8 ++++++ pyproject.toml | 5 ++++ .../edge-index-methods-427f7301c720f565.yaml | 2 +- .../0.11/fix-dispatch-3596ef110cc68338.yaml | 2 +- .../0.11/prepare-0.11-af688e532712c830.yaml | 4 +-- ...ix-sequence-protocol-e95246e864cc850a.yaml | 2 +- .../0.13/prepare-0.13.0-5e579fb3ab1e3b60.yaml | 2 +- ...apping-token_swapper-55d5b045b0b55345.yaml | 2 +- .../platform-updates-e9b296144e633c95.yaml | 2 +- .../0.14/s390x-tier-4-1701a0f044759cd1.yaml | 2 +- ...transitive-reduction-6db2b80351c15887.yaml | 2 +- .../custom-iterators-dce8557a8f87e8c0.yaml | 2 +- ...l-draw-digraph-plots-aecf86738ab9b0db.yaml | 2 +- ...hical-topo-sort-core-e85fba409d612600.yaml | 2 +- ...maximum-bisimulation-942a9d0dc9b46ee4.yaml | 2 +- .../0.15/swap-nox-tox-dea2bb14c400641c.yaml | 2 +- .../add-nx-converter-1feffc8d5aa13365.yaml | 2 +- .../notes/fix-typos-8f68ff3d0680b924.yaml | 5 ++++ rustworkx-core/src/bipartite_coloring.rs | 2 +- rustworkx-core/src/coloring.rs | 2 +- .../src/connectivity/all_simple_paths.rs | 8 +++--- .../src/connectivity/biconnected.rs | 2 +- rustworkx-core/src/connectivity/chain.rs | 2 +- .../src/connectivity/cycle_basis.rs | 4 +-- rustworkx-core/src/dag_algo.rs | 4 +-- .../src/generators/complete_graph.rs | 2 +- rustworkx-core/src/generators/cycle_graph.rs | 2 +- rustworkx-core/src/generators/karate_club.rs | 4 +-- rustworkx-core/src/generators/path_graph.rs | 2 +- rustworkx-core/src/generators/star_graph.rs | 2 +- rustworkx-core/src/max_weight_matching.rs | 22 ++++++++-------- rustworkx-core/src/planar/lr_planar.rs | 2 +- rustworkx-core/src/steiner_tree.rs | 2 +- rustworkx/__init__.py | 16 ++++++------ rustworkx/visualization/matplotlib.py | 4 +-- src/bisimulation.rs | 8 +++--- src/connectivity/mod.rs | 2 +- src/dag_algo/mod.rs | 2 +- src/digraph.rs | 26 +++++++++---------- src/generators.rs | 4 +-- src/graph.rs | 6 ++--- src/isomorphism/mod.rs | 8 +++--- src/isomorphism/vf2.rs | 12 ++++----- src/iterators.rs | 6 ++--- src/layout/mod.rs | 2 +- src/layout/spring.rs | 2 +- src/shortest_path/all_pairs_dijkstra.rs | 4 +-- src/shortest_path/mod.rs | 4 +-- src/steiner_tree.rs | 2 +- src/transitivity.rs | 12 ++++----- src/traversal/mod.rs | 16 ++++++------ tests/digraph/test_bellman_ford.py | 14 +++++----- tests/digraph/test_bisimulation.py | 2 +- tests/digraph/test_dijkstra.py | 6 ++--- tests/digraph/test_hits.py | 2 +- tests/digraph/test_k_shortest_path.py | 4 +-- tests/digraph/test_nodes.py | 2 +- tests/digraph/test_pagerank.py | 2 +- tests/digraph/test_pred_succ.py | 2 +- tests/graph/test_bellman_ford.py | 14 +++++----- tests/graph/test_dijkstra.py | 6 ++--- tests/graph/test_k_shortest_path.py | 4 +-- tests/graph/test_max_weight_matching.py | 2 +- tests/test_custom_return_types.py | 2 +- tests/test_token_swapper.py | 2 +- tox.ini | 12 ++++----- 74 files changed, 184 insertions(+), 166 deletions(-) create mode 100644 releasenotes/notes/fix-typos-8f68ff3d0680b924.yaml diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index f635c608e..92ee8ec96 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -29,7 +29,7 @@ jobs: - run: pip install -U ruff==0.6.8 black~=24.8 - uses: dtolnay/rust-toolchain@stable with: - components: rustfmt + components: rustfmt, clippy - name: Test Build run: cargo build - name: Rust Format diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 5ae80b1d9..fdfb4b575 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -284,7 +284,7 @@ cargo doc --open ### Type Annotations If you have added new methods, functions, or classes, and/or changed any -signatures, type anotations for Python are required to be included in a pull +signatures, type annotations for Python are required to be included in a pull request. Type annotations are added using type [stub files](https://typing.readthedocs.io/en/latest/source/stubs.html) which provide type annotations to python tooling which use type annotations. The stub @@ -523,7 +523,7 @@ primary exception to this is adding support for new python versions. If a new python version is released backporting that feature change with that new support is an acceptable backport. -In rustworkx at least until the 1.0 release we only maintaing a single stable +In rustworkx at least until the 1.0 release we only maintain a single stable branch at a time for the most recent minor version release. #### Backporting procedure diff --git a/README.md b/README.md index 0c4575e9c..7e2292637 100644 --- a/README.md +++ b/README.md @@ -77,7 +77,7 @@ it just as it would if there was a prebuilt binary available. > [!NOTE] > To build from source you will need to ensure you have pip >=19.0.0 installed, which supports PEP-517, or that you have manually installed -`setuptools-rust` prior to running `pip install rustworkx`. If you recieve an +`setuptools-rust` prior to running `pip install rustworkx`. If you receive an error about `setuptools-rust` not being found you should upgrade pip with `pip install -U pip` or manually install `setuptools-rust` with `pip install setuptools-rust` and try again. diff --git a/docs/source/benchmarks.rst b/docs/source/benchmarks.rst index 959573ae3..ce2e4ceea 100644 --- a/docs/source/benchmarks.rst +++ b/docs/source/benchmarks.rst @@ -2,7 +2,7 @@ Rustworkx Comparison Benchmarks With Other Libraries **************************************************** -rustworkx is competitive against other popular graph libraries for Python. We compared rustworkx to the igraph, graph-tools and NetworkIt libraries `in a benchmark consisting of four tasks available on Github for reproducibility `__. We report the results from a machine with an Intel(R) i9-9900K CPU at 3.60GHz with eight cores, 16 theads, and 32GB of RAM avaialble. +rustworkx is competitive against other popular graph libraries for Python. We compared rustworkx to the igraph, graph-tools and NetworkIt libraries `in a benchmark consisting of four tasks available on Github for reproducibility `__. We report the results from a machine with an Intel(R) i9-9900K CPU at 3.60GHz with eight cores, 16 threads, and 32GB of RAM available. Graph Creation ============== diff --git a/docs/source/install.rst b/docs/source/install.rst index 942614b2c..92b85e7a8 100644 --- a/docs/source/install.rst +++ b/docs/source/install.rst @@ -44,12 +44,12 @@ just as it would if there was a prebuilt binary available. To build from source you will need to ensure you have pip >=19.0.0 installed, which supports PEP-517, or that you have manually installed - setuptools-rust prior to running pip install rustworkx. If you recieve an + setuptools-rust prior to running pip install rustworkx. If you receive an error about ``setuptools-rust`` not being found you should upgrade pip with ``pip install -U pip`` or manually install ``setuptools-rust`` with: ``pip install setuptools-rust`` and try again. -.. _platform-suppport: +.. _platform-support: Platform Support ================ diff --git a/docs/source/release_notes.rst b/docs/source/release_notes.rst index f8d414bc7..79ba7847d 100644 --- a/docs/source/release_notes.rst +++ b/docs/source/release_notes.rst @@ -266,7 +266,7 @@ New Features - A new method, :meth:`~rustworkx.PyDiGraph.remove_node_retain_edges`, has been added to the :class:`~rustworkx.PyDiGraph` class. This method can be used to - remove a node and add edges from its predecesors to its successors. + remove a node and add edges from its predecessors to its successors. - Two new methods, :meth:`~rustworkx.PyGraph.edge_list` and :meth:`~rustworkx.PyGraph.weighted_edge_list`, for getting a list of tuples with the edge source and target (with or without edge weights) have been @@ -285,7 +285,7 @@ New Features edge list file and will read that file and generate a new object from the contents. - Two new methods, :meth:`~rustworkx.PyGraph.extend_from_edge_list` and - :meth:`~rustworkx.PyGraoh.extend_from_weighted_edge_list` has been added + :meth:`~rustworkx.PyGraph.extend_from_weighted_edge_list` has been added to :class:`~rustworkx.PyGraph` and :class:`~rustworkx.PyDiGraph` (:meth:`~rustworkx.PyDiGraph.extend_from_edge_list` and :meth:`~rustworkx.PyDiGraph.extend_from_weighted_edge_list`). This method diff --git a/docs/source/tutorial/betweenness_centrality.rst b/docs/source/tutorial/betweenness_centrality.rst index 38114101f..2ccc51102 100644 --- a/docs/source/tutorial/betweenness_centrality.rst +++ b/docs/source/tutorial/betweenness_centrality.rst @@ -37,8 +37,8 @@ To start we need to generate a graph: mpl_draw(graph) -Calculate the Betweeness Centrality ------------------------------------ +Calculate the Betweenness Centrality +------------------------------------ The :func:`~rustworkx.betweenness_centrality` function can be used to calculate the betweenness centrality for each node in the graph. diff --git a/docs/source/tutorial/dags.rst b/docs/source/tutorial/dags.rst index cc8780e7f..70fd44361 100644 --- a/docs/source/tutorial/dags.rst +++ b/docs/source/tutorial/dags.rst @@ -113,7 +113,7 @@ jobs. For example: Above we define a DAG with 6 jobs and dependency relationship between these jobs. Now if we run the :func:`~rustworkx.topological_sort` function on the graph it will return a linear order to execute the jobs that will respect -the dependency releationship. +the dependency relationship. .. jupyter-execute:: @@ -151,7 +151,7 @@ computation. A quantum circuit is represented graphically like: The specifics of this circuit aren't important here beyond the fact that we have 2 qubits, ``q_0`` and ``q_1``, 2 classical bits, ``c_0`` and ``c_1``, -and a series of operations on those qubits with a depedency ordering. The last +and a series of operations on those qubits with a dependency ordering. The last operation on each qubit is a measurement on ``q_0`` that is stored in ``c_0`` and ``q_1`` that is stored in ``c_1``. diff --git a/noxfile.py b/noxfile.py index d7d5229cb..c0f08ff96 100644 --- a/noxfile.py +++ b/noxfile.py @@ -15,6 +15,7 @@ "black~=24.8", "ruff~=0.6", "setuptools-rust", + "typos~=1.28", ] stubs_deps = [ @@ -45,6 +46,7 @@ def test_with_version(session): @nox.session(python=["3"]) def lint(session): black(session) + typos(session) session.install(*lint_deps) session.run("ruff", "check", "rustworkx", "retworkx", "setup.py") session.run("cargo", "fmt", "--all", "--", "--check", external=True) @@ -69,6 +71,12 @@ def black(session): session.install(*[d for d in lint_deps if "black" in d]) session.run("black", "rustworkx", "tests", "retworkx", *session.posargs) +@nox.session(python=["3"]) +def typos(session): + session.install(*[d for d in lint_deps if "typos" in d]) + session.run("typos", "--exclude", "releasenotes") + session.run("typos", "--no-check-filenames", "releasenotes") + @nox.session(python=["3"]) def stubs(session): install_rustworkx(session) diff --git a/pyproject.toml b/pyproject.toml index 2a752ec1b..759440db3 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -23,6 +23,11 @@ extend-exclude = ["doc"] "rustworkx/__init__.py" = ["F405", "F403"] "*.pyi" = ["F403", "F405", "PYI001", "PYI002"] +[tool.typos.default] +extend-ignore-words-re = [ + "[Ss]toer", +] + [tool.cibuildwheel] manylinux-x86_64-image = "manylinux2014" manylinux-i686-image = "manylinux2014" diff --git a/releasenotes/notes/0.11/edge-index-methods-427f7301c720f565.yaml b/releasenotes/notes/0.11/edge-index-methods-427f7301c720f565.yaml index 3c55e2c45..ff41fbeae 100644 --- a/releasenotes/notes/0.11/edge-index-methods-427f7301c720f565.yaml +++ b/releasenotes/notes/0.11/edge-index-methods-427f7301c720f565.yaml @@ -8,7 +8,7 @@ features: Added a new method, :meth:`~rustworkx.PyGraph.incident_edge_index_map`, to the :class:`~rustworkx.PyGraph` and :class:`~rustworkx.PyDiGraph` class. This method returns a mapping of edge indices for edges incident to a provided node - to the endoint and weight tuple for that edge index. For example: + to the endpoint and weight tuple for that edge index. For example: .. jupyter-execute:: diff --git a/releasenotes/notes/0.11/fix-dispatch-3596ef110cc68338.yaml b/releasenotes/notes/0.11/fix-dispatch-3596ef110cc68338.yaml index 0ebbeb750..1127ba938 100644 --- a/releasenotes/notes/0.11/fix-dispatch-3596ef110cc68338.yaml +++ b/releasenotes/notes/0.11/fix-dispatch-3596ef110cc68338.yaml @@ -8,4 +8,4 @@ fixes: - | Fixed an oversight in the :func:`~rustworkx.union` function where user-defined values for the ``merge_nodes`` and ``merge_edges`` arguments - were being ingored. + were being ignored. diff --git a/releasenotes/notes/0.11/prepare-0.11-af688e532712c830.yaml b/releasenotes/notes/0.11/prepare-0.11-af688e532712c830.yaml index 3d07b3068..b009abef9 100644 --- a/releasenotes/notes/0.11/prepare-0.11-af688e532712c830.yaml +++ b/releasenotes/notes/0.11/prepare-0.11-af688e532712c830.yaml @@ -56,7 +56,7 @@ features: g = nx.Graph() g.add_nodes_from([ ("A", {"color": "turquoise", "size": "extra large"}), - ("B", {"color": "fuschia", "size": "tiny"}), + ("B", {"color": "fuchsia", "size": "tiny"}), ]) g.add_edge("A", "B") rx_graph = rx.networkx_converter(g, keep_attributes=True) @@ -65,7 +65,7 @@ features: will output:: - [{'color': 'turquoise', 'size': 'extra large', '__networkx_node__': 'A'}, {'color': 'fuschia', 'size': 'tiny', '__networkx_node__': 'B'}] + [{'color': 'turquoise', 'size': 'extra large', '__networkx_node__': 'A'}, {'color': 'fuchsia', 'size': 'tiny', '__networkx_node__': 'B'}] WeightedEdgeList[(0, 1, {})] fixes: diff --git a/releasenotes/notes/0.13/fix-sequence-protocol-e95246e864cc850a.yaml b/releasenotes/notes/0.13/fix-sequence-protocol-e95246e864cc850a.yaml index 4bfb6fe37..19934be6b 100644 --- a/releasenotes/notes/0.13/fix-sequence-protocol-e95246e864cc850a.yaml +++ b/releasenotes/notes/0.13/fix-sequence-protocol-e95246e864cc850a.yaml @@ -4,7 +4,7 @@ fixes: Fixed an issue with the custom sequence return types, :class:`~.BFSSuccessors`, :class:`~.NodeIndices`, :class:`~.EdgeList`, :class:`~.WeightedEdgeList`, :class:`~.EdgeIndices`, and :class:`~.Chains` - where they previosuly were missing certain attributes that prevented them + where they previously were missing certain attributes that prevented them being used as a sequence for certain built-in functions such as ``reversed()``. Fixed `#696 `__. diff --git a/releasenotes/notes/0.13/prepare-0.13.0-5e579fb3ab1e3b60.yaml b/releasenotes/notes/0.13/prepare-0.13.0-5e579fb3ab1e3b60.yaml index 3104afa19..9a1f21917 100644 --- a/releasenotes/notes/0.13/prepare-0.13.0-5e579fb3ab1e3b60.yaml +++ b/releasenotes/notes/0.13/prepare-0.13.0-5e579fb3ab1e3b60.yaml @@ -13,5 +13,5 @@ prelude: | This is also the final rustworkx release that supports running with Python 3.7. Starting in the 0.14.0 release Python >= 3.8 will be required to use - rustworkx. This release also increased the minimum suported Rust version for + rustworkx. This release also increased the minimum supported Rust version for compiling rustworkx and rustworkx-core from source to 1.56.1. diff --git a/releasenotes/notes/0.14/handle-invalid-mapping-token_swapper-55d5b045b0b55345.yaml b/releasenotes/notes/0.14/handle-invalid-mapping-token_swapper-55d5b045b0b55345.yaml index 5cc97023e..c88a4475e 100644 --- a/releasenotes/notes/0.14/handle-invalid-mapping-token_swapper-55d5b045b0b55345.yaml +++ b/releasenotes/notes/0.14/handle-invalid-mapping-token_swapper-55d5b045b0b55345.yaml @@ -31,4 +31,4 @@ upgrade: token_swapper(&g, mapping, Some(10), Some(4), Some(50)); will now return ``Err(MapNotPossible)`` instead of panicking. If you were using this - funciton before you'll need to handle the result type. + function before you'll need to handle the result type. diff --git a/releasenotes/notes/0.14/platform-updates-e9b296144e633c95.yaml b/releasenotes/notes/0.14/platform-updates-e9b296144e633c95.yaml index adbf3bd27..8e0df2bbd 100644 --- a/releasenotes/notes/0.14/platform-updates-e9b296144e633c95.yaml +++ b/releasenotes/notes/0.14/platform-updates-e9b296144e633c95.yaml @@ -6,7 +6,7 @@ features: upgrade: - | Support for the Linux ppc64le pllatform has changed from tier 3 to tier 4 - (as documented in :ref:`platform-suppport`). This is a result of no longer + (as documented in :ref:`platform-support`). This is a result of no longer being able to run tests during the pre-compiled wheel publishing jobs due to constraints in the available CI infrastructure. There hopefully shouldn't be any meaningful impact resulting from this change, but as there diff --git a/releasenotes/notes/0.14/s390x-tier-4-1701a0f044759cd1.yaml b/releasenotes/notes/0.14/s390x-tier-4-1701a0f044759cd1.yaml index b4f3b3f9a..5794720de 100644 --- a/releasenotes/notes/0.14/s390x-tier-4-1701a0f044759cd1.yaml +++ b/releasenotes/notes/0.14/s390x-tier-4-1701a0f044759cd1.yaml @@ -2,7 +2,7 @@ upgrade: - | Support for the Linux s390x platform has changed from tier 3 to tier 4 (as - documented in :ref:`platform-suppport`). This is a result of no longer being + documented in :ref:`platform-support`). This is a result of no longer being able to run tests during the pre-compiled wheel publishing jobs due to constraints in the available CI infrastructure. There hopefully shouldn't be any meaningful impact resulting from this change, but as there are no longer tests being diff --git a/releasenotes/notes/0.14/transitive-reduction-6db2b80351c15887.yaml b/releasenotes/notes/0.14/transitive-reduction-6db2b80351c15887.yaml index 03ce860c6..a6cced9e4 100644 --- a/releasenotes/notes/0.14/transitive-reduction-6db2b80351c15887.yaml +++ b/releasenotes/notes/0.14/transitive-reduction-6db2b80351c15887.yaml @@ -1,7 +1,7 @@ --- features: - | - Added a new function, :func:`~.transitive_reduction` which returns the transtive reduction + Added a new function, :func:`~.transitive_reduction` which returns the transitive reduction of a given :class:`~rustworkx.PyDiGraph` and a dictionary with the mapping of indices from the given graph to the returned graph. The given graph must be a Directed Acyclic Graph (DAG). For example: diff --git a/releasenotes/notes/0.15/custom-iterators-dce8557a8f87e8c0.yaml b/releasenotes/notes/0.15/custom-iterators-dce8557a8f87e8c0.yaml index dd19a50e9..43b0ff813 100644 --- a/releasenotes/notes/0.15/custom-iterators-dce8557a8f87e8c0.yaml +++ b/releasenotes/notes/0.15/custom-iterators-dce8557a8f87e8c0.yaml @@ -7,4 +7,4 @@ features: of approximately 40% for iterating through the custom iterables. These types are not directly nameable or constructable from Python space, and other than the - performance improvement, the behavior should largely not be noticable from Python space. + performance improvement, the behavior should largely not be noticeable from Python space. diff --git a/releasenotes/notes/0.15/fix-mpl-draw-digraph-plots-aecf86738ab9b0db.yaml b/releasenotes/notes/0.15/fix-mpl-draw-digraph-plots-aecf86738ab9b0db.yaml index 377d74e67..2ffd392d9 100644 --- a/releasenotes/notes/0.15/fix-mpl-draw-digraph-plots-aecf86738ab9b0db.yaml +++ b/releasenotes/notes/0.15/fix-mpl-draw-digraph-plots-aecf86738ab9b0db.yaml @@ -4,7 +4,7 @@ fixes: Fixed the plots of multigraphs using :func:`.mpl_draw`. Previously, parallel edges of multigraphs were plotted on top of each other, with overlapping arrows and labels. The radius of parallel edges of the multigraph was fixed to be `0.25` for - `connectionstyle` supporting this argument in :func:`.draw_edges`. The edge lables + `connectionstyle` supporting this argument in :func:`.draw_edges`. The edge labels were offset to `0.25` in :func:`.draw_edge_labels` to align with their respective edges. This fix can be tested using the following code: diff --git a/releasenotes/notes/0.15/lexicographical-topo-sort-core-e85fba409d612600.yaml b/releasenotes/notes/0.15/lexicographical-topo-sort-core-e85fba409d612600.yaml index 7aa6d5f6a..fc8049afd 100644 --- a/releasenotes/notes/0.15/lexicographical-topo-sort-core-e85fba409d612600.yaml +++ b/releasenotes/notes/0.15/lexicographical-topo-sort-core-e85fba409d612600.yaml @@ -2,6 +2,6 @@ features: - | Added a new function ``lexicographical_topological_sort`` to the - ``rustworkx_core::dag_algo`` module. That is a gneric Rust implementation + ``rustworkx_core::dag_algo`` module. That is a generic Rust implementation for the core rust library that provides the :func:`.lexicographical_topological_sort` function to Rust users. diff --git a/releasenotes/notes/0.15/maximum-bisimulation-942a9d0dc9b46ee4.yaml b/releasenotes/notes/0.15/maximum-bisimulation-942a9d0dc9b46ee4.yaml index bcb9e2049..cb3b3bfdf 100644 --- a/releasenotes/notes/0.15/maximum-bisimulation-942a9d0dc9b46ee4.yaml +++ b/releasenotes/notes/0.15/maximum-bisimulation-942a9d0dc9b46ee4.yaml @@ -4,7 +4,7 @@ features: compute the maximum bisimulation or relational coarsest partition of a graph. This function is based on the algorithm described in the publication "Three partition refinement algorithms" by Paige and Tarjan. This function - recieves a graph and returns a + receives a graph and returns a :class:`~rustworkx.RelationalCoarsestPartition`. - | Added a new class :class:`~rustworkx.RelationalCoarsestPartition` to output diff --git a/releasenotes/notes/0.15/swap-nox-tox-dea2bb14c400641c.yaml b/releasenotes/notes/0.15/swap-nox-tox-dea2bb14c400641c.yaml index fd94ad28d..d3bca1466 100644 --- a/releasenotes/notes/0.15/swap-nox-tox-dea2bb14c400641c.yaml +++ b/releasenotes/notes/0.15/swap-nox-tox-dea2bb14c400641c.yaml @@ -1,7 +1,7 @@ --- other: - | - For developement of rustworkx the automated testing environment + For development of rustworkx the automated testing environment tooling used has switched from Tox to instead `Nox `__. This is has no impact for end users and is only relevant if you contribute code to rustworkx. diff --git a/releasenotes/notes/0.9.0/add-nx-converter-1feffc8d5aa13365.yaml b/releasenotes/notes/0.9.0/add-nx-converter-1feffc8d5aa13365.yaml index 2d8fb96a2..2046bf24f 100644 --- a/releasenotes/notes/0.9.0/add-nx-converter-1feffc8d5aa13365.yaml +++ b/releasenotes/notes/0.9.0/add-nx-converter-1feffc8d5aa13365.yaml @@ -4,7 +4,7 @@ features: A new function :func:`rustworkx.networkx_converter` has been added. This function takes in a networkx ``Graph`` object and will generate an equivalent :class:`~rustworkx.PyGraph` or :class:`~rustworkx.PyDiGraph` - object. While this function is provided as a convience for users of + object. While this function is provided as a convenience for users of both rustworkx and networkx, networkx will **not** be added as a dependency of rustworkx (which precludes a rustworkx->networkx converter, see :ref:`networkx_converter` for example code on how to build this yourself). diff --git a/releasenotes/notes/fix-typos-8f68ff3d0680b924.yaml b/releasenotes/notes/fix-typos-8f68ff3d0680b924.yaml new file mode 100644 index 000000000..21f6b9712 --- /dev/null +++ b/releasenotes/notes/fix-typos-8f68ff3d0680b924.yaml @@ -0,0 +1,5 @@ +--- +fixes: + - | + Fix typos detected by `typos `_. + Add spell checker invocations to the Nox ``lint`` session. diff --git a/rustworkx-core/src/bipartite_coloring.rs b/rustworkx-core/src/bipartite_coloring.rs index a4bdbecf4..a8519f357 100644 --- a/rustworkx-core/src/bipartite_coloring.rs +++ b/rustworkx-core/src/bipartite_coloring.rs @@ -538,7 +538,7 @@ where } // Reconstruct coloring of the original graph by iterating over the edges, finding the - // correponding edge (endpoints) in the multigraph, and selecting the last (not yet + // corresponding edge (endpoints) in the multigraph, and selecting the last (not yet // assigned) color of that edge let mut edge_coloring: DictMap = DictMap::with_capacity(graph.edge_count()); for edge in graph.edge_references() { diff --git a/rustworkx-core/src/coloring.rs b/rustworkx-core/src/coloring.rs index 730119edc..d764ffcf5 100644 --- a/rustworkx-core/src/coloring.rs +++ b/rustworkx-core/src/coloring.rs @@ -401,7 +401,7 @@ where /// Arguments: /// /// * `graph` - The graph object to run the algorithm on -/// * `preset_color_fn` - A callback function that will recieve the node identifier +/// * `preset_color_fn` - A callback function that will receive the node identifier /// for each node in the graph and is expected to return an `Option` /// (wrapped in a `Result`) that is `None` if the node has no preset and /// the usize represents the preset color. diff --git a/rustworkx-core/src/connectivity/all_simple_paths.rs b/rustworkx-core/src/connectivity/all_simple_paths.rs index c3f08c48f..bbe39acf7 100644 --- a/rustworkx-core/src/connectivity/all_simple_paths.rs +++ b/rustworkx-core/src/connectivity/all_simple_paths.rs @@ -80,8 +80,8 @@ where // list of visited nodes let mut visited: IndexSet = IndexSet::from_iter(Some(from)); - // list of childs of currently exploring path nodes, - // last elem is list of childs of last visited node + // list of children of currently exploring path nodes, + // last elem is list of children of last visited node let mut stack = vec![graph.neighbors_directed(from, Outgoing)]; let mut output: DictMap>> = DictMap::with_capacity(to.len()); @@ -174,8 +174,8 @@ where { // list of visited nodes let mut visited: IndexSet = IndexSet::from_iter(Some(from)); - // list of childs of currently exploring path nodes, - // last elem is list of childs of last visited node + // list of children of currently exploring path nodes, + // last elem is list of children of last visited node let mut stack = vec![graph.neighbors_directed(from, Outgoing)]; let mut output_path: Option> = None; diff --git a/rustworkx-core/src/connectivity/biconnected.rs b/rustworkx-core/src/connectivity/biconnected.rs index b0a1b0f6a..edf0d492d 100644 --- a/rustworkx-core/src/connectivity/biconnected.rs +++ b/rustworkx-core/src/connectivity/biconnected.rs @@ -336,7 +336,7 @@ mod tests { #[test] fn test_biconnected_components1() { - // exmaple from https://web.archive.org/web/20121229123447/http://www.ibluemojo.com/school/articul_algorithm.html + // example from https://web.archive.org/web/20121229123447/http://www.ibluemojo.com/school/articul_algorithm.html let graph = UnGraph::<(), ()>::from_edges([ (0, 1), (0, 5), diff --git a/rustworkx-core/src/connectivity/chain.rs b/rustworkx-core/src/connectivity/chain.rs index 5f8c52ca5..c2190b0ec 100644 --- a/rustworkx-core/src/connectivity/chain.rs +++ b/rustworkx-core/src/connectivity/chain.rs @@ -61,7 +61,7 @@ where /// The graph should be undirected. If `source` is specified only the chain /// decomposition for the connected component containing this node will be returned. /// This node indicates the root of the depth-first search tree. If it's not -/// specified, a source will be chosen arbitrarly and repeated until all components +/// specified, a source will be chosen arbitrarily and repeated until all components /// of the graph are searched. /// /// Returns a list of list of edges where each inner list is a chain. diff --git a/rustworkx-core/src/connectivity/cycle_basis.rs b/rustworkx-core/src/connectivity/cycle_basis.rs index f638aa0ef..4f8e72b4e 100644 --- a/rustworkx-core/src/connectivity/cycle_basis.rs +++ b/rustworkx-core/src/connectivity/cycle_basis.rs @@ -92,10 +92,10 @@ where cycles.push(cycle); // A cycle was found: } else if !used.get(&z).unwrap().contains(&neighbor) { - let pn = used.get(&neighbor).unwrap(); + let prev_n = used.get(&neighbor).unwrap(); let mut cycle: Vec = vec![neighbor, z]; let mut p = pred.get(&z).unwrap(); - while !pn.contains(p) { + while !prev_n.contains(p) { cycle.push(*p); p = pred.get(p).unwrap(); } diff --git a/rustworkx-core/src/dag_algo.rs b/rustworkx-core/src/dag_algo.rs index 7af62d620..34c7e19f2 100644 --- a/rustworkx-core/src/dag_algo.rs +++ b/rustworkx-core/src/dag_algo.rs @@ -33,7 +33,7 @@ use num_traits::{Num, Zero}; use crate::err::LayersError; /// Return a pair of [`petgraph::Direction`] values corresponding to the "forwards" and "backwards" -/// direction of graph traversal, based on whether the graph is being traved forwards (following +/// direction of graph traversal, based on whether the graph is being traversed forwards (following /// the edges) or backward (reversing along edges). The order of returns is (forwards, backwards). #[inline(always)] pub fn traversal_directions(reverse: bool) -> (petgraph::Direction, petgraph::Direction) { @@ -719,7 +719,7 @@ where Some(runs) } -/// Auxiliary struct to make the output of [`collect_runs`] iteratable +/// Auxiliary struct to make the output of [`collect_runs`] iterable /// /// If the filtering function passed to [`collect_runs`] returns an error, it is propagated /// through `next` as `Err`. In this case the run in which the error occurred will be skipped diff --git a/rustworkx-core/src/generators/complete_graph.rs b/rustworkx-core/src/generators/complete_graph.rs index c07f4ee3d..df8b36d7e 100644 --- a/rustworkx-core/src/generators/complete_graph.rs +++ b/rustworkx-core/src/generators/complete_graph.rs @@ -22,7 +22,7 @@ use super::InvalidInputError; /// /// * `num_nodes` - The number of nodes to create a complete graph for. Either this or /// `weights` must be specified. If both this and `weights` are specified, `weights` -/// will take priorty and this argument will be ignored +/// will take priority and this argument will be ignored /// * `weights` - A `Vec` of node weight objects. /// * `default_node_weight` - A callable that will return the weight to use /// for newly created nodes. This is ignored if `weights` is specified. diff --git a/rustworkx-core/src/generators/cycle_graph.rs b/rustworkx-core/src/generators/cycle_graph.rs index 0cc603d8b..6462e0515 100644 --- a/rustworkx-core/src/generators/cycle_graph.rs +++ b/rustworkx-core/src/generators/cycle_graph.rs @@ -22,7 +22,7 @@ use super::InvalidInputError; /// /// * `num_nodes` - The number of nodes to create a cycle graph for. Either this or /// `weights` must be specified. If both this and `weights` are specified, `weights` -/// will take priorty and this argument will be ignored. +/// will take priority and this argument will be ignored. /// * `weights` - A `Vec` of node weight objects. /// * `default_node_weight` - A callable that will return the weight to use /// for newly created nodes. This is ignored if `weights` is specified. diff --git a/rustworkx-core/src/generators/karate_club.rs b/rustworkx-core/src/generators/karate_club.rs index d0e0ac495..63698665c 100644 --- a/rustworkx-core/src/generators/karate_club.rs +++ b/rustworkx-core/src/generators/karate_club.rs @@ -23,9 +23,9 @@ use petgraph::visit::{Data, NodeIndexable}; /// /// * `default_node_weight` - A callable that will receive a boolean, indicating /// if a node is part of Mr Hi's faction (True) or the Officer's faction (false). -/// It shoudl return the node weight according to the desired type. +/// It should return the node weight according to the desired type. /// * `default_edge_weight` - A callable that will receive the integer representing -/// the strenght of the relation between two nodes. It should return the edge +/// the strength of the relation between two nodes. It should return the edge /// weight according to the desired type. /// pub fn karate_club_graph(mut default_node_weight: F, mut default_edge_weight: H) -> G diff --git a/rustworkx-core/src/generators/path_graph.rs b/rustworkx-core/src/generators/path_graph.rs index 7aff79d7b..373a7d62a 100644 --- a/rustworkx-core/src/generators/path_graph.rs +++ b/rustworkx-core/src/generators/path_graph.rs @@ -22,7 +22,7 @@ use super::InvalidInputError; /// /// * `num_nodes` - The number of nodes to create a path graph for. Either this or /// `weights` must be specified. If both this and `weights` are specified, `weights` -/// will take priorty and this argument will be ignored +/// will take priority and this argument will be ignored /// * `weights` - A `Vec` of node weight objects. /// * `default_node_weight` - A callable that will return the weight to use /// for newly created nodes. This is ignored if `weights` is specified. diff --git a/rustworkx-core/src/generators/star_graph.rs b/rustworkx-core/src/generators/star_graph.rs index a09565c93..d9630dfdc 100644 --- a/rustworkx-core/src/generators/star_graph.rs +++ b/rustworkx-core/src/generators/star_graph.rs @@ -22,7 +22,7 @@ use super::InvalidInputError; /// /// * `num_nodes` - The number of nodes to create a star graph for. Either this or /// `weights` must be specified. If both this and `weights` are specified, weights -/// will take priorty and this argument will be ignored +/// will take priority and this argument will be ignored /// * `weights` - A `Vec` of node weight objects. /// * `default_node_weight` - A callable that will return the weight to use /// for newly created nodes. This is ignored if `weights` is specified. diff --git a/rustworkx-core/src/max_weight_matching.rs b/rustworkx-core/src/max_weight_matching.rs index 193b1ff87..d7400d32b 100644 --- a/rustworkx-core/src/max_weight_matching.rs +++ b/rustworkx-core/src/max_weight_matching.rs @@ -11,7 +11,7 @@ // under the License. // Needed to pass shared state between functions -// closures don't work because of recurssion +// closures don't work because of recursion #![allow(clippy::too_many_arguments)] // Allow single character names to match naming convention from // paper @@ -87,7 +87,7 @@ fn assign_label( best_edge[w] = None; best_edge[b] = None; if t == 1 { - // b became an S-vertex/blossom; add it(s verticies) to the queue + // b became an S-vertex/blossom; add it(s vertices) to the queue queue.append(&mut blossom_leaves(b, num_nodes, blossom_children)?); } else if t == 2 { // b became a T-vertex/blossom; assign label S to its mate. @@ -142,7 +142,7 @@ fn scan_blossom( assert!(labels[blossom] == Some(1)); path.push(blossom); labels[blossom] = Some(5); - // Trace one step bacl. + // Trace one step back. assert!(label_ends[blossom] == mate.get(&blossom_base[blossom].unwrap()).copied()); if label_ends[blossom].is_none() { // The base of blossom is single; stop tracing this path @@ -160,7 +160,7 @@ fn scan_blossom( mem::swap(&mut v, &mut w); } } - // Remvoe breadcrumbs. + // Remove breadcrumbs. for blossom in path { labels[blossom] = Some(1); } @@ -371,7 +371,7 @@ fn expand_blossom( // base. assert!(label_ends[blossom].is_some()); let entry_child = in_blossoms[endpoints[label_ends[blossom].unwrap() ^ 1]]; - // Decied in which direction we will go around the blossom. + // Decide in which direction we will go around the blossom. let i = blossom_children[blossom] .iter() .position(|x| *x == entry_child) @@ -809,7 +809,7 @@ fn verify_optimum( /// Based on networkx implementation /// /// -/// With reference to the standalone protoype implementation from: +/// With reference to the standalone prototype implementation from: /// /// /// @@ -884,9 +884,9 @@ where if num_edges == 0 { return Ok(HashSet::new()); } - // Node indicies in the PyGraph may not be contiguous however the + // Node indices in the PyGraph may not be contiguous however the // algorithm operates on contiguous indices 0..num_nodes. node_map maps - // the PyGraph's NodeIndex to the contingous usize used inside the + // the PyGraph's NodeIndex to the contiguous usize used inside the // algorithm let node_map: HashMap = graph .node_identifiers() @@ -1018,7 +1018,7 @@ where best_edge = vec![None; 2 * num_nodes]; blossom_best_edges.splice(num_nodes.., (0..num_nodes).map(|_| Vec::new())); // Loss of labeling means that we can not be sure that currently - // allowable edges remain allowable througout this stage. + // allowable edges remain allowable throughout this stage. allowed_edge = vec![false; num_edges]; // Make queue empty queue.clear(); @@ -1248,7 +1248,7 @@ where if delta_type == -1 { // No further improvement possible; max-cardinality optimum // reached. Do a final delta update to make the optimum - // verifyable + // verifiable assert!(max_cardinality); delta_type = 1; delta = Some(max(0, *dual_var[..num_nodes].iter().min().unwrap())); @@ -1275,7 +1275,7 @@ where } } } - // Take action at the point where minimum delta occured. + // Take action at the point where minimum delta occurred. if delta_type == 1 { // No further improvement possible; optimum reached break; diff --git a/rustworkx-core/src/planar/lr_planar.rs b/rustworkx-core/src/planar/lr_planar.rs index bfe77fe5c..7db398a5c 100644 --- a/rustworkx-core/src/planar/lr_planar.rs +++ b/rustworkx-core/src/planar/lr_planar.rs @@ -211,7 +211,7 @@ where graph: G, /// roots of the DFS forest. roots: Vec, - /// distnace from root. + /// distance from root. height: HashMap, /// parent edge. eparent: HashMap>, diff --git a/rustworkx-core/src/steiner_tree.rs b/rustworkx-core/src/steiner_tree.rs index 80fab566f..3d6afb297 100644 --- a/rustworkx-core/src/steiner_tree.rs +++ b/rustworkx-core/src/steiner_tree.rs @@ -449,7 +449,7 @@ pub struct SteinerTreeResult { /// /// The minimum tree of ``graph`` with regard to a set of ``terminal_nodes`` /// is a tree within ``graph`` that spans those nodes and has a minimum size -/// (measured as the sum of edge weights) amoung all such trees. +/// (measured as the sum of edge weights) among all such trees. /// /// The minimum steiner tree can be approximated by computing the minimum /// spanning tree of the subgraph of the metric closure of ``graph`` induced diff --git a/rustworkx/__init__.py b/rustworkx/__init__.py index a470c71f0..a00dd5a4a 100644 --- a/rustworkx/__init__.py +++ b/rustworkx/__init__.py @@ -24,7 +24,7 @@ class PyDAG(PyDiGraph): """A class for creating direct acyclic graphs. PyDAG is just an alias of the PyDiGraph class and behaves identically to - the :class:`~rustworkx.PyDiGraph` class and can be used interchangably + the :class:`~rustworkx.PyDiGraph` class and can be used interchangeably with ``PyDiGraph``. It currently exists solely as a backwards compatibility alias for users of rustworkx from prior to the 0.4.0 release when there was no PyDiGraph class. @@ -641,7 +641,7 @@ def dfs_edges(graph, source=None): :param int source: An optional node index to use as the starting node for the depth-first search. The edge list will only return edges in the components reachable from this index. If this is not specified - then a source will be chosen arbitrarly and repeated until all + then a source will be chosen arbitrarily and repeated until all components of the graph are searched. :returns: A list of edges as a tuple of the form ``(source, target)`` in @@ -1003,7 +1003,7 @@ def bipartite_layout( :param graph: The graph to generate the layout for. Can either be a :class:`~rustworkx.PyGraph` or :class:`~rustworkx.PyDiGraph` :param set first_nodes: The set of node indices on the left (or top if - horitontal is true) + horizontal is true) :param bool horizontal: An optional bool specifying the orientation of the layout :param float scale: An optional scaling factor to scale positions @@ -1344,7 +1344,7 @@ def vf2_mapping( """ Return an iterator over all vf2 mappings between two graphs. - This funcion will run the vf2 algorithm used from + This function will run the vf2 algorithm used from :func:`~rustworkx.is_isomorphic` and :func:`~rustworkx.is_subgraph_isomorphic` but instead of returning a boolean it will return an iterator over all possible mapping of node ids found from ``first`` to ``second``. If the graphs are not @@ -1381,7 +1381,7 @@ def vf2_mapping( algorithm visits while searching for a solution. If it exceeds this limit, the algorithm will stop. Default: ``None``. - :returns: An iterator over dicitonaries of node indices from ``first`` to node + :returns: An iterator over dictionaries of node indices from ``first`` to node indices in ``second`` representing the mapping found. :rtype: Iterable[NodeMap] """ @@ -1555,7 +1555,7 @@ def tree_edge(self, edge): or a :class:`~rustworkx.PyDiGraph` :param List[int] source: An optional list of node indices to use as the starting nodes for the breadth-first search. If this is not specified then a source - will be chosen arbitrarly and repeated until all components of the + will be chosen arbitrarily and repeated until all components of the graph are searched. :param visitor: A visitor object that is invoked at the event points inside the algorithm. This should be a subclass of :class:`~rustworkx.visit.BFSVisitor`. @@ -1625,7 +1625,7 @@ def tree_edge(self, edge): :param PyGraph graph: The graph to be used. :param List[int] source: An optional list of node indices to use as the starting nodes for the depth-first search. If this is not specified then a source - will be chosen arbitrarly and repeated until all components of the + will be chosen arbitrarily and repeated until all components of the graph are searched. :param visitor: A visitor object that is invoked at the event points inside the algorithm. This should be a subclass of :class:`~rustworkx.visit.DFSVisitor`. @@ -1678,7 +1678,7 @@ def dijkstra_search(graph, source, weight_fn, visitor): or a :class:`~rustworkx.PyDiGraph`. :param List[int] source: An optional list of node indices to use as the starting nodes for the dijkstra search. If this is not specified then a source - will be chosen arbitrarly and repeated until all components of the + will be chosen arbitrarily and repeated until all components of the graph are searched. :param weight_fn: An optional weight function for an edge. It will accept a single argument, the edge's weight object and will return a float which diff --git a/rustworkx/visualization/matplotlib.py b/rustworkx/visualization/matplotlib.py index bd06c243b..458eafe25 100644 --- a/rustworkx/visualization/matplotlib.py +++ b/rustworkx/visualization/matplotlib.py @@ -581,7 +581,7 @@ def draw_edges( Label for legend min_source_margin : int (default=0) - The minimum margin (gap) at the begining of the edge at the source. + The minimum margin (gap) at the beginning of the edge at the source. min_target_margin : int (default=0) The minimum margin (gap) at the end of the edge at the target. @@ -978,7 +978,7 @@ def draw_edge_labels( ax : Matplotlib Axes object, optional Draw the graph in the specified Matplotlib axes. - rotate : bool (deafult=True) + rotate : bool (default=True) Rotate edge labels to lie parallel to edges clip_on : bool (default=True) diff --git a/src/bisimulation.rs b/src/bisimulation.rs index 4a0525f38..96883e96c 100644 --- a/src/bisimulation.rs +++ b/src/bisimulation.rs @@ -309,11 +309,11 @@ fn maximum_bisimulation(graph: &StablePyGraph) -> Option> { let mut counterimage = build_counterimage(graph, smaller_component); let counterimage_group = group_by_counterimage(counterimage.clone(), &node_to_block_vec); - let ((new_fine_blocks, removeable_fine_blocks), coarse_block_that_are_now_compound) = + let ((new_fine_blocks, removable_fine_blocks), coarse_block_that_are_now_compound) = split_blocks_with_grouped_counterimage(counterimage_group, &mut node_to_block_vec); all_fine_blocks.extend(new_fine_blocks); - all_fine_blocks.retain(|x| !removeable_fine_blocks.iter().any(|y| Rc::ptr_eq(x, y))); + all_fine_blocks.retain(|x| !removable_fine_blocks.iter().any(|y| Rc::ptr_eq(x, y))); queue.extend(coarse_block_that_are_now_compound); // counterimage = E^{-1}(B) - E^{-1}(S-B) @@ -327,11 +327,11 @@ fn maximum_bisimulation(graph: &StablePyGraph) -> Option> { } let counterimage_group = group_by_counterimage(counterimage, &node_to_block_vec); - let ((new_fine_blocks, removeable_fine_blocks), coarse_block_that_are_now_compound) = + let ((new_fine_blocks, removable_fine_blocks), coarse_block_that_are_now_compound) = split_blocks_with_grouped_counterimage(counterimage_group, &mut node_to_block_vec); all_fine_blocks.extend(new_fine_blocks); - all_fine_blocks.retain(|x| !removeable_fine_blocks.iter().any(|y| Rc::ptr_eq(x, y))); + all_fine_blocks.retain(|x| !removable_fine_blocks.iter().any(|y| Rc::ptr_eq(x, y))); queue.extend(coarse_block_that_are_now_compound); } diff --git a/src/connectivity/mod.rs b/src/connectivity/mod.rs index 353a57fe2..ecd2858be 100644 --- a/src/connectivity/mod.rs +++ b/src/connectivity/mod.rs @@ -1073,7 +1073,7 @@ pub fn biconnected_components(graph: &graph::PyGraph) -> BiconnectedComponents { /// only the chain decomposition for the connected component containing /// this node will be returned. This node indicates the root of the depth-first /// search tree. If this is not specified then a source will be chosen -/// arbitrarly and repeated until all components of the graph are searched. +/// arbitrarily and repeated until all components of the graph are searched. /// :returns: A list of list of edges where each inner list is a chain. /// :rtype: list of EdgeList /// diff --git a/src/dag_algo/mod.rs b/src/dag_algo/mod.rs index 0adfff12a..da5de0cb5 100644 --- a/src/dag_algo/mod.rs +++ b/src/dag_algo/mod.rs @@ -81,7 +81,7 @@ where } /// Return a pair of [`petgraph::Direction`] values corresponding to the "forwards" and "backwards" -/// direction of graph traversal, based on whether the graph is being traved forwards (following +/// direction of graph traversal, based on whether the graph is being traversed forwards (following /// the edges) or backward (reversing along edges). The order of returns is (forwards, backwards). #[inline(always)] pub fn traversal_directions(reverse: bool) -> (petgraph::Direction, petgraph::Direction) { diff --git a/src/digraph.rs b/src/digraph.rs index 47b70b93a..e70d111b6 100644 --- a/src/digraph.rs +++ b/src/digraph.rs @@ -637,15 +637,15 @@ impl PyDiGraph { let children = self .graph .neighbors_directed(index, petgraph::Direction::Outgoing); - let mut succesors: Vec<&PyObject> = Vec::new(); + let mut successors: Vec<&PyObject> = Vec::new(); let mut used_indices: HashSet = HashSet::new(); for succ in children { if !used_indices.contains(&succ) { - succesors.push(self.graph.node_weight(succ).unwrap()); + successors.push(self.graph.node_weight(succ).unwrap()); used_indices.insert(succ); } } - succesors + successors } /// Return a list of all the node predecessor data. @@ -692,7 +692,7 @@ impl PyDiGraph { filter_fn: PyObject, ) -> PyResult> { let index = NodeIndex::new(node); - let mut succesors: Vec<&PyObject> = Vec::new(); + let mut successors: Vec<&PyObject> = Vec::new(); let mut used_indices: HashSet = HashSet::new(); let filter_edge = |edge: &PyObject| -> PyResult { @@ -710,11 +710,11 @@ impl PyDiGraph { let edge_weight = edge.weight(); if filter_edge(edge_weight)? { used_indices.insert(succ); - succesors.push(self.graph.node_weight(succ).unwrap()); + successors.push(self.graph.node_weight(succ).unwrap()); } } } - Ok(succesors) + Ok(successors) } /// Return a filtered list of predecessor data such that each @@ -1004,7 +1004,7 @@ impl PyDiGraph { /// :meth:`remove_node_retain_edges_by_id`. /// /// :param int node: The index of the node to remove. If the index is not - /// present in the graph it will be ingored and this function willl have + /// present in the graph it will be ignored and this function will have /// no effect. /// :param bool use_outgoing: If set to true the weight/data from the /// edge outgoing from ``node`` will be used in the retained edge @@ -1754,12 +1754,12 @@ impl PyDiGraph { /// Get the successor indices of a node. /// - /// This will return a list of the node indicies for the succesors of + /// This will return a list of the node indices for the successors of /// a node /// /// :param int node: The index of the node to get the successors of /// - /// :returns: A list of the neighbor node indicies + /// :returns: A list of the neighbor node indices /// :rtype: NodeIndices #[pyo3(text_signature = "(self, node, /)")] pub fn successor_indices(&self, node: usize) -> NodeIndices { @@ -1774,12 +1774,12 @@ impl PyDiGraph { /// Get the predecessor indices of a node. /// - /// This will return a list of the node indicies for the predecessors of + /// This will return a list of the node indices for the predecessors of /// a node /// /// :param int node: The index of the node to get the predecessors of /// - /// :returns: A list of the neighbor node indicies + /// :returns: A list of the neighbor node indices /// :rtype: NodeIndices #[pyo3(text_signature = "(self, node, /)")] pub fn predecessor_indices(&self, node: usize) -> NodeIndices { @@ -2162,8 +2162,8 @@ impl PyDiGraph { /// Read an edge list file and create a new PyDiGraph object from the /// contents /// - /// The expected format for the edge list file is a line seperated list - /// of deliminated node ids. If there are more than 3 elements on + /// The expected format for the edge list file is a line separated list + /// of delimited node ids. If there are more than 3 elements on /// a line the 3rd on will be treated as a string weight for the edge /// /// :param str path: The path of the file to open diff --git a/src/generators.rs b/src/generators.rs index 82b75b8a4..3db2765d3 100644 --- a/src/generators.rs +++ b/src/generators.rs @@ -989,7 +989,7 @@ pub fn binomial_tree_graph( /// /// :returns: A directed binomial tree with 2^n vertices and 2^n - 1 edges. /// :rtype: PyDiGraph -/// :raises IndexError: If the lenght of ``weights`` is greater that 2^n +/// :raises IndexError: If the length of ``weights`` is greater that 2^n /// :raises OverflowError: If the input order exceeds the maximum value for the /// current platform. /// @@ -1059,7 +1059,7 @@ pub fn directed_binomial_tree_graph( /// /// :returns: A r-ary tree. /// :rtype: PyGraph -/// :raises IndexError: If the lenght of ``weights`` is greater that n +/// :raises IndexError: If the length of ``weights`` is greater that n /// /// .. jupyter-execute:: /// diff --git a/src/graph.rs b/src/graph.rs index c3e627e5c..21f375b93 100644 --- a/src/graph.rs +++ b/src/graph.rs @@ -1124,7 +1124,7 @@ impl PyGraph { /// /// :param int node: The index of the node to get the neighbors of /// - /// :returns: A list of the neighbor node indicies + /// :returns: A list of the neighbor node indices /// :rtype: NodeIndices #[pyo3(text_signature = "(self, node, /)")] pub fn neighbors(&self, node: usize) -> NodeIndices { @@ -1284,8 +1284,8 @@ impl PyGraph { /// Read an edge list file and create a new PyGraph object from the /// contents /// - /// The expected format for the edge list file is a line seperated list - /// of deliminated node ids. If there are more than 3 elements on + /// The expected format for the edge list file is a line separated list + /// of delimited node ids. If there are more than 3 elements on /// a line the 3rd on will be treated as a string weight for the edge /// /// :param str path: The path of the file to open diff --git a/src/isomorphism/mod.rs b/src/isomorphism/mod.rs index 6e48f1f10..43d4100b2 100644 --- a/src/isomorphism/mod.rs +++ b/src/isomorphism/mod.rs @@ -295,7 +295,7 @@ pub fn graph_is_subgraph_isomorphic( /// Return an iterator over all vf2 mappings between two :class:`~rustworkx.PyDiGraph` objects /// -/// This funcion will run the vf2 algorithm used from +/// This function will run the vf2 algorithm used from /// :func:`~rustworkx.is_isomorphic` and :func:`~rustworkx.is_subgraph_isomorphic` /// but instead of returning a boolean it will return an iterator over all possible /// mapping of node ids found from ``first`` to ``second``. If the graphs are not @@ -333,7 +333,7 @@ pub fn graph_is_subgraph_isomorphic( /// visits while searching for a solution. If it exceeds this limit, the algorithm /// will stop. /// -/// :returns: An iterator over dicitonaries of node indices from ``first`` to node +/// :returns: An iterator over dictionaries of node indices from ``first`` to node /// indices in ``second`` representing the mapping found. /// :rtype: Iterable[NodeMap] #[pyfunction] @@ -374,7 +374,7 @@ pub fn digraph_vf2_mapping( /// Return an iterator over all vf2 mappings between two :class:`~rustworkx.PyGraph` objects /// -/// This funcion will run the vf2 algorithm used from +/// This function will run the vf2 algorithm used from /// :func:`~rustworkx.is_isomorphic` and :func:`~rustworkx.is_subgraph_isomorphic` /// but instead of returning a boolean it will return an iterator over all possible /// mapping of node ids found from ``first`` to ``second``. If the graphs are not @@ -411,7 +411,7 @@ pub fn digraph_vf2_mapping( /// visits while searching for a solution. If it exceeds this limit, the algorithm /// will stop. Default: ``None``. /// -/// :returns: An iterator over dicitonaries of node indices from ``first`` to node +/// :returns: An iterator over dictionaries of node indices from ``first`` to node /// indices in ``second`` representing the mapping found. /// :rtype: Iterable[NodeMap] #[pyfunction] diff --git a/src/isomorphism/vf2.rs b/src/isomorphism/vf2.rs index f3e77c7c1..af8104d50 100644 --- a/src/isomorphism/vf2.rs +++ b/src/isomorphism/vf2.rs @@ -168,7 +168,7 @@ where fn sort(&self, graph: &StablePyGraph) -> Vec { let n = graph.node_bound(); - let dout: Vec = (0..n) + let d_out: Vec = (0..n) .map(|idx| { graph .neighbors_directed(graph.from_index(idx), Outgoing) @@ -176,9 +176,9 @@ where }) .collect(); - let mut din: Vec = vec![0; n]; + let mut d_in: Vec = vec![0; n]; if graph.is_directed() { - din = (0..n) + d_in = (0..n) .map(|idx| { graph .neighbors_directed(graph.from_index(idx), Incoming) @@ -202,9 +202,9 @@ where .max_by_key(|&(_, &node)| { ( conn_in[node], - dout[node], + d_out[node], conn_out[node], - din[node], + d_in[node], Reverse(node), ) }) @@ -256,7 +256,7 @@ where }; let mut sorted_nodes: Vec = graph.node_indices().map(|node| node.index()).collect(); - sorted_nodes.par_sort_by_key(|&node| (dout[node], din[node], Reverse(node))); + sorted_nodes.par_sort_by_key(|&node| (d_out[node], d_in[node], Reverse(node))); sorted_nodes.reverse(); for node in sorted_nodes { diff --git a/src/iterators.rs b/src/iterators.rs index cabfbe1c3..34acb8569 100644 --- a/src/iterators.rs +++ b/src/iterators.rs @@ -34,7 +34,7 @@ // don't store any python object, just use `impl PyGCProtocol for MyReadOnlyType {}`. // // Types `T, K, V` above should implement `PyHash`, `PyEq`, `PyDisplay` traits. -// These are arleady implemented for many primitive rust types and `PyObject`. +// These are already implemented for many primitive rust types and `PyObject`. #![allow(clippy::float_cmp, clippy::upper_case_acronyms)] @@ -1088,7 +1088,7 @@ custom_vec_iter_impl!( The class is a read-only sequence of integers instances. - This class is a container class for the results of the digraph_maximum_bisimulation funtion. + This class is a container class for the results of the digraph_maximum_bisimulation function. It implements the Python sequence protocol. So you can treat the return as a read-only sequence/list that is integer indexed. If you want to use it as an iterator you @@ -1144,7 +1144,7 @@ custom_vec_iter_impl!( The class is a read-only sequence of :class:`.NodeIndices` instances. - This class is a container class for the results of the digraph_maximum_bisimulation funtion. + This class is a container class for the results of the digraph_maximum_bisimulation function. It implements the Python sequence protocol. So you can treat the return as a read-only sequence/list that is integer indexed. If you want to use it as an iterator you diff --git a/src/layout/mod.rs b/src/layout/mod.rs index 802207b1c..786642bfb 100644 --- a/src/layout/mod.rs +++ b/src/layout/mod.rs @@ -237,7 +237,7 @@ pub fn digraph_random_layout( /// /// :param PyGraph graph: The graph to generate the layout for /// :param set first_nodes: The set of node indices on the left (or top if -/// horitontal is true) +/// horizontal is true) /// :param bool horizontal: An optional bool specifying the orientation of the /// layout /// :param float scale: An optional scaling factor to scale positions diff --git a/src/layout/spring.rs b/src/layout/spring.rs index 620952499..8aa96223f 100644 --- a/src/layout/spring.rs +++ b/src/layout/spring.rs @@ -183,7 +183,7 @@ pub fn rescale(pos: &mut [Point], scale: Nt, indices: Vec) { mu[0] /= n as Nt; mu[1] /= n as Nt; - // substract mean and find max coordinate for all axes + // subtract mean and find max coordinate for all axes let mut lim = f64::NEG_INFINITY; for n in indices { let [px, py] = pos.get_mut(n).unwrap(); diff --git a/src/shortest_path/all_pairs_dijkstra.rs b/src/shortest_path/all_pairs_dijkstra.rs index a6daff611..bcfe94485 100644 --- a/src/shortest_path/all_pairs_dijkstra.rs +++ b/src/shortest_path/all_pairs_dijkstra.rs @@ -76,10 +76,10 @@ pub fn all_pairs_dijkstra_path_lengths( let out_map: DictMap = node_indices .into_par_iter() .map(|x| { - let path_lenghts: PyResult>> = + let path_lengths: PyResult>> = dijkstra(graph, x, None, |e| edge_cost(e.id()), None); let out_map = PathLengthMapping { - path_lengths: path_lenghts + path_lengths: path_lengths .unwrap() .into_iter() .enumerate() diff --git a/src/shortest_path/mod.rs b/src/shortest_path/mod.rs index 1b9970327..c062945d2 100644 --- a/src/shortest_path/mod.rs +++ b/src/shortest_path/mod.rs @@ -1361,7 +1361,7 @@ pub fn graph_num_shortest_paths_unweighted( /// output distance matrix. /// :param float null_value: An optional float that will treated as a null /// value. This element will be the default in the matrix and represents -/// the absense of a path in the graph. By default this is ``0.0``. +/// the absence of a path in the graph. By default this is ``0.0``. /// /// :returns: The distance matrix /// :rtype: numpy.ndarray @@ -1403,7 +1403,7 @@ pub fn digraph_distance_matrix( /// be tuned /// :param float null_value: An optional float that will treated as a null /// value. This element will be the default in the matrix and represents -/// the absense of a path in the graph. By default this is ``0.0``. +/// the absence of a path in the graph. By default this is ``0.0``. /// /// :returns: The distance matrix /// :rtype: numpy.ndarray diff --git a/src/steiner_tree.rs b/src/steiner_tree.rs index 57819b992..a854dcc1d 100644 --- a/src/steiner_tree.rs +++ b/src/steiner_tree.rs @@ -79,7 +79,7 @@ pub fn metric_closure( /// /// The minimum tree of ``graph`` with regard to a set of ``terminal_nodes`` /// is a tree within ``graph`` that spans those nodes and has a minimum size -/// (measured as the sum of edge weights) amoung all such trees. +/// (measured as the sum of edge weights) among all such trees. /// /// The minimum steiner tree can be approximated by computing the minimum /// spanning tree of the subgraph of the metric closure of ``graph`` induced diff --git a/src/transitivity.rs b/src/transitivity.rs index 6bcb25f94..565342eef 100644 --- a/src/transitivity.rs +++ b/src/transitivity.rs @@ -127,14 +127,14 @@ fn _digraph_triangles(graph: &digraph::PyDiGraph, node: usize) -> (usize, usize) .sum::(); } - let din: usize = in_neighbors.len(); - let dout: usize = out_neighbors.len(); + let d_in: usize = in_neighbors.len(); + let d_out: usize = out_neighbors.len(); - let dtot = dout + din; - let dbil: usize = out_neighbors.intersection(&in_neighbors).count(); - let triples: usize = match dtot { + let d_tot = d_out + d_in; + let d_bil: usize = out_neighbors.intersection(&in_neighbors).count(); + let triples: usize = match d_tot { 0 => 0, - _ => dtot * (dtot - 1) - 2 * dbil, + _ => d_tot * (d_tot - 1) - 2 * d_bil, }; (triangles / 2, triples) diff --git a/src/traversal/mod.rs b/src/traversal/mod.rs index 2d8531b48..17444431d 100644 --- a/src/traversal/mod.rs +++ b/src/traversal/mod.rs @@ -64,7 +64,7 @@ use crate::iterators::EdgeList; /// :param int source: An optional node index to use as the starting node /// for the depth-first search. The edge list will only return edges in /// the components reachable from this index. If this is not specified -/// then a source will be chosen arbitrarly and repeated until all +/// then a source will be chosen arbitrarily and repeated until all /// components of the graph are searched. /// /// :returns: A list of edges as a tuple of the form ``(source, target)`` in @@ -109,7 +109,7 @@ pub fn digraph_dfs_edges(graph: &digraph::PyDiGraph, source: Option) -> E /// :param int source: An optional node index to use as the starting node /// for the depth-first search. The edge list will only return edges in /// the components reachable from this index. If this is not specified -/// then a source will be chosen arbitrarly and repeated until all +/// then a source will be chosen arbitrarily and repeated until all /// components of the graph are searched. /// /// :returns: A list of edges as a tuple of the form ``(source, target)`` in @@ -310,7 +310,7 @@ pub fn descendants(graph: &digraph::PyDiGraph, node: usize) -> HashSet { /// :param PyDiGraph graph: The graph to be used. /// :param List[int] source: An optional list of node indices to use as the starting nodes /// for the breadth-first search. If this is not specified then a source -/// will be chosen arbitrarly and repeated until all components of the +/// will be chosen arbitrarily and repeated until all components of the /// graph are searched. /// :param visitor: A visitor object that is invoked at the event points inside the /// algorithm. This should be a subclass of :class:`~rustworkx.visit.BFSVisitor`. @@ -403,7 +403,7 @@ pub fn digraph_bfs_search( /// :param PyGraph graph: The graph to be used. /// :param List[int] source: An optional list of node indices to use as the starting nodes /// for the breadth-first search. If this is not specified then a source -/// will be chosen arbitrarly and repeated until all components of the +/// will be chosen arbitrarily and repeated until all components of the /// graph are searched. /// :param visitor: A visitor object that is invoked at the event points inside the /// algorithm. This should be a subclass of :class:`~rustworkx.visit.BFSVisitor`. @@ -494,7 +494,7 @@ pub fn graph_bfs_search( /// :param PyDiGraph graph: The graph to be used. /// :param List[int] source: An optional list of node indices to use as the starting nodes /// for the depth-first search. If this is not specified then a source -/// will be chosen arbitrarly and repeated until all components of the +/// will be chosen arbitrarily and repeated until all components of the /// graph are searched. /// :param visitor: A visitor object that is invoked at the event points inside the /// algorithm. This should be a subclass of :class:`~rustworkx.visit.DFSVisitor`. @@ -585,7 +585,7 @@ pub fn digraph_dfs_search( /// :param PyGraph graph: The graph to be used. /// :param List[int] source: An optional list of node indices to use as the starting nodes /// for the depth-first search. If this is not specified then a source -/// will be chosen arbitrarly and repeated until all components of the +/// will be chosen arbitrarily and repeated until all components of the /// graph are searched. /// :param visitor: A visitor object that is invoked at the event points inside the /// algorithm. This should be a subclass of :class:`~rustworkx.visit.DFSVisitor`. @@ -658,7 +658,7 @@ pub fn graph_dfs_search( /// :param PyDiGraph graph: The graph to be used. /// :param List[int] source: An optional list of node indices to use as the starting nodes /// for the dijkstra search. If this is not specified then a source -/// will be chosen arbitrarly and repeated until all components of the +/// will be chosen arbitrarily and repeated until all components of the /// graph are searched. /// :param weight_fn: An optional weight function for an edge. It will accept /// a single argument, the edge's weight object and will return a float which @@ -740,7 +740,7 @@ pub fn digraph_dijkstra_search( /// :param PyGraph graph: The graph to be used. /// :param List[int] source: An optional list of node indices to use as the starting nodes /// for the dijkstra search. If this is not specified then a source -/// will be chosen arbitrarly and repeated until all components of the +/// will be chosen arbitrarily and repeated until all components of the /// graph are searched. /// :param weight_fn: An optional weight function for an edge. It will accept /// a single argument, the edge's weight object and will return a float which diff --git a/tests/digraph/test_bellman_ford.py b/tests/digraph/test_bellman_ford.py index a502a6956..ee5b5731e 100644 --- a/tests/digraph/test_bellman_ford.py +++ b/tests/digraph/test_bellman_ford.py @@ -48,11 +48,11 @@ def test_bellman_ford_length_with_no_path(self): g = rustworkx.PyDiGraph() a = g.add_node("A") g.add_node("B") - path_lenghts = rustworkx.digraph_bellman_ford_shortest_path_lengths( + path_lengths = rustworkx.digraph_bellman_ford_shortest_path_lengths( g, a, edge_cost_fn=float ) expected = {} - self.assertEqual(expected, path_lenghts) + self.assertEqual(expected, path_lengths) def test_bellman_ford_path(self): paths = rustworkx.digraph_bellman_ford_shortest_paths(self.graph, self.a) @@ -136,13 +136,13 @@ def test_bellman_ford_length_with_no_path_and_goal(self): g = rustworkx.PyDiGraph() a = g.add_node("A") b = g.add_node("B") - path_lenghts = rustworkx.digraph_bellman_ford_shortest_path_lengths( + path_lengths = rustworkx.digraph_bellman_ford_shortest_path_lengths( g, a, edge_cost_fn=float, goal=b ) expected = rustworkx.digraph_dijkstra_shortest_path_lengths( g, a, edge_cost_fn=float, goal=b ) - self.assertEqual(expected, path_lenghts) + self.assertEqual(expected, path_lengths) def test_bellman_ford_with_no_path(self): g = rustworkx.PyDiGraph() @@ -212,7 +212,7 @@ def test_raises_negative_cycle_bellman_ford_paths(self): with self.assertRaises(rustworkx.NegativeCycle): rustworkx.bellman_ford_shortest_paths(graph, 0, weight_fn=float) - def test_raises_negative_cycle_bellman_ford_path_lenghts(self): + def test_raises_negative_cycle_bellman_ford_path_lengths(self): graph = rustworkx.PyDiGraph() graph.add_nodes_from(list(range(4))) graph.add_edges_from( @@ -428,7 +428,7 @@ def test_raises_negative_cycle_all_pairs_bellman_ford_paths(self): with self.assertRaises(rustworkx.NegativeCycle): rustworkx.all_pairs_bellman_ford_shortest_paths(graph, float) - def test_raises_negative_cycle_all_pairs_bellman_ford_path_lenghts(self): + def test_raises_negative_cycle_all_pairs_bellman_ford_path_lengths(self): graph = rustworkx.PyDiGraph() graph.add_nodes_from(list(range(4))) graph.add_edges_from( @@ -449,7 +449,7 @@ def test_raises_index_error_bellman_ford_paths(self): self.graph, len(self.graph.node_indices()) + 1, weight_fn=lambda x: float(x) ) - def test_raises_index_error_bellman_ford_path_lenghts(self): + def test_raises_index_error_bellman_ford_path_lengths(self): with self.assertRaises(IndexError): rustworkx.digraph_bellman_ford_shortest_path_lengths( self.graph, len(self.graph.node_indices()) + 1, edge_cost_fn=lambda x: float(x) diff --git a/tests/digraph/test_bisimulation.py b/tests/digraph/test_bisimulation.py index ddf4fb012..a55f28199 100644 --- a/tests/digraph/test_bisimulation.py +++ b/tests/digraph/test_bisimulation.py @@ -34,7 +34,7 @@ def test_empty_graph(self): res = rustworkx.digraph_maximum_bisimulation(graph) self.assertEqual(res, []) - def test_multigraph_compatability(self): + def test_multigraph_compatibility(self): graph = rustworkx.PyDiGraph() graph.add_nodes_from(range(5)) graph.add_edges_from_no_data([(0, 1), (1, 4), (1, 4), (1, 4), (1, 4), (2, 3), (3, 0)]) diff --git a/tests/digraph/test_dijkstra.py b/tests/digraph/test_dijkstra.py index 614503cbe..2a4320671 100644 --- a/tests/digraph/test_dijkstra.py +++ b/tests/digraph/test_dijkstra.py @@ -48,11 +48,11 @@ def test_dijkstra_length_with_no_path(self): g = rustworkx.PyDiGraph() a = g.add_node("A") b = g.add_node("B") - path_lenghts = rustworkx.digraph_dijkstra_shortest_path_lengths( + path_lengths = rustworkx.digraph_dijkstra_shortest_path_lengths( g, a, edge_cost_fn=float, goal=b ) expected = {} - self.assertEqual(expected, path_lenghts) + self.assertEqual(expected, path_lengths) def test_dijkstra_path(self): paths = rustworkx.digraph_dijkstra_shortest_paths(self.graph, self.a) @@ -318,7 +318,7 @@ def all_pairs_dijkstra_with_invalid_weights(self): graph, edge_cost_fn=lambda _: invalid_weight ) - def all_pairs_dijkstra_lenghts_with_invalid_weights(self): + def all_pairs_dijkstra_lengths_with_invalid_weights(self): graph = rustworkx.generators.directed_path_graph(2) for invalid_weight in [float("nan"), -1]: with self.subTest(invalid_weight=invalid_weight): diff --git a/tests/digraph/test_hits.py b/tests/digraph/test_hits.py index 6fb0ec8b6..a8d5bc9a2 100644 --- a/tests/digraph/test_hits.py +++ b/tests/digraph/test_hits.py @@ -46,7 +46,7 @@ # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # -# These tests are adapated from the networkx test cases: +# These tests are adapted from the networkx test cases: # https://github.com/networkx/networkx/blob/cea310f9066efc0d5ff76f63d33dbc3eefe61f6b/networkx/algorithms/link_analysis/tests/test_pagerank.py import unittest diff --git a/tests/digraph/test_k_shortest_path.py b/tests/digraph/test_k_shortest_path.py index cbc40695e..f3f19ddc5 100644 --- a/tests/digraph/test_k_shortest_path.py +++ b/tests/digraph/test_k_shortest_path.py @@ -89,8 +89,8 @@ def test_k_shortest_path_with_no_path(self): g = rustworkx.PyDiGraph() a = g.add_node("A") b = g.add_node("B") - path_lenghts = rustworkx.digraph_k_shortest_path_lengths( + path_lengths = rustworkx.digraph_k_shortest_path_lengths( g, start=a, k=1, edge_cost=float, goal=b ) expected = {} - self.assertEqual(expected, path_lenghts) + self.assertEqual(expected, path_lengths) diff --git a/tests/digraph/test_nodes.py b/tests/digraph/test_nodes.py index 773713b22..73876710b 100644 --- a/tests/digraph/test_nodes.py +++ b/tests/digraph/test_nodes.py @@ -211,7 +211,7 @@ def test_remove_nodes_retain_edges_by_id_parallel(self): for weight in weights: dag.add_edge(nodes[0], nodes[1], weight) dag.add_edge(nodes[1], nodes[2], weight) - # The middle node has three precessor edges and three successor edges, where each set has + # The middle node has three predecessor edges and three successor edges, where each set has # one edge each of three weights. Edges should be paired up in bijection during the removal. dag.remove_node_retain_edges_by_id(nodes[1]) self.assertEqual(set(dag.node_indices()), {nodes[0], nodes[2]}) diff --git a/tests/digraph/test_pagerank.py b/tests/digraph/test_pagerank.py index 3682611b9..eea2c91af 100644 --- a/tests/digraph/test_pagerank.py +++ b/tests/digraph/test_pagerank.py @@ -46,7 +46,7 @@ # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # -# These tests are adapated from the networkx test cases: +# These tests are adapted from the networkx test cases: # https://github.com/networkx/networkx/blob/cea310f9066efc0d5ff76f63d33dbc3eefe61f6b/networkx/algorithms/link_analysis/tests/test_pagerank.py import unittest diff --git a/tests/digraph/test_pred_succ.py b/tests/digraph/test_pred_succ.py index 10d0a62d3..4ebf5924e 100644 --- a/tests/digraph/test_pred_succ.py +++ b/tests/digraph/test_pred_succ.py @@ -274,7 +274,7 @@ def test_many_children(self): res, ) - def test_bfs_succesors(self): + def test_bfs_successors(self): dag = rustworkx.PyDAG() node_a = dag.add_node(0) node_b = dag.add_child(node_a, 1, {}) diff --git a/tests/graph/test_bellman_ford.py b/tests/graph/test_bellman_ford.py index d0dd3f106..1eb54d69a 100644 --- a/tests/graph/test_bellman_ford.py +++ b/tests/graph/test_bellman_ford.py @@ -81,19 +81,19 @@ def test_bellman_ford_length_with_no_path_and_goal(self): g = rustworkx.PyGraph() a = g.add_node("A") b = g.add_node("B") - path_lenghts = rustworkx.graph_bellman_ford_shortest_path_lengths( + path_lengths = rustworkx.graph_bellman_ford_shortest_path_lengths( g, a, edge_cost_fn=float, goal=b ) expected = rustworkx.graph_dijkstra_shortest_path_lengths(g, a, edge_cost_fn=float, goal=b) - self.assertEqual(expected, path_lenghts) + self.assertEqual(expected, path_lengths) def test_bellman_ford_length_with_no_path(self): g = rustworkx.PyGraph() a = g.add_node("A") g.add_node("B") - path_lenghts = rustworkx.graph_bellman_ford_shortest_path_lengths(g, a, edge_cost_fn=float) + path_lengths = rustworkx.graph_bellman_ford_shortest_path_lengths(g, a, edge_cost_fn=float) expected = {} - self.assertEqual(expected, path_lenghts) + self.assertEqual(expected, path_lengths) def test_bellman_ford_path_with_no_goal_set(self): path = rustworkx.graph_bellman_ford_shortest_paths(self.graph, self.a) @@ -175,7 +175,7 @@ def test_raises_negative_cycle_bellman_ford_paths(self): with self.assertRaises(rustworkx.NegativeCycle): rustworkx.bellman_ford_shortest_paths(graph, 0, weight_fn=float) - def test_raises_negative_cycle_bellman_ford_path_lenghts(self): + def test_raises_negative_cycle_bellman_ford_path_lengths(self): graph = rustworkx.PyGraph() graph.add_nodes_from(list(range(4))) graph.add_edges_from( @@ -292,7 +292,7 @@ def test_raises_negative_cycle_all_pairs_bellman_ford_paths(self): with self.assertRaises(rustworkx.NegativeCycle): rustworkx.all_pairs_bellman_ford_shortest_paths(graph, float) - def test_raises_negative_cycle_all_pairs_bellman_ford_path_lenghts(self): + def test_raises_negative_cycle_all_pairs_bellman_ford_path_lengths(self): graph = rustworkx.PyGraph() graph.add_nodes_from(list(range(4))) graph.add_edges_from( @@ -313,7 +313,7 @@ def test_raises_index_error_bellman_ford_paths(self): self.graph, len(self.graph.node_indices()) + 1, weight_fn=lambda x: float(x) ) - def test_raises_index_error_bellman_ford_path_lenghts(self): + def test_raises_index_error_bellman_ford_path_lengths(self): with self.assertRaises(IndexError): rustworkx.graph_bellman_ford_shortest_path_lengths( self.graph, len(self.graph.node_indices()) + 1, edge_cost_fn=lambda x: float(x) diff --git a/tests/graph/test_dijkstra.py b/tests/graph/test_dijkstra.py index 745502761..d6f7121ec 100644 --- a/tests/graph/test_dijkstra.py +++ b/tests/graph/test_dijkstra.py @@ -74,11 +74,11 @@ def test_dijkstra_length_with_no_path(self): g = rustworkx.PyGraph() a = g.add_node("A") b = g.add_node("B") - path_lenghts = rustworkx.graph_dijkstra_shortest_path_lengths( + path_lengths = rustworkx.graph_dijkstra_shortest_path_lengths( g, a, edge_cost_fn=float, goal=b ) expected = {} - self.assertEqual(expected, path_lenghts) + self.assertEqual(expected, path_lengths) def test_dijkstra_path_with_no_goal_set(self): path = rustworkx.graph_dijkstra_shortest_paths(self.graph, self.a) @@ -253,7 +253,7 @@ def all_pairs_dijkstra_with_invalid_weights(self): graph, edge_cost_fn=lambda _: invalid_weight ) - def all_pairs_dijkstra_lenghts_with_invalid_weights(self): + def all_pairs_dijkstra_lengths_with_invalid_weights(self): graph = rustworkx.generators.path_graph(2) for invalid_weight in [float("nan"), -1]: with self.subTest(invalid_weight=invalid_weight): diff --git a/tests/graph/test_k_shortest_path.py b/tests/graph/test_k_shortest_path.py index 6497de38a..8ff1a1a4a 100644 --- a/tests/graph/test_k_shortest_path.py +++ b/tests/graph/test_k_shortest_path.py @@ -68,8 +68,8 @@ def test_k_shortest_path_with_no_path(self): g = rustworkx.PyGraph() a = g.add_node("A") b = g.add_node("B") - path_lenghts = rustworkx.graph_k_shortest_path_lengths( + path_lengths = rustworkx.graph_k_shortest_path_lengths( g, start=a, k=1, edge_cost=float, goal=b ) expected = {} - self.assertEqual(expected, path_lenghts) + self.assertEqual(expected, path_lengths) diff --git a/tests/graph/test_max_weight_matching.py b/tests/graph/test_max_weight_matching.py index 06fbbde5e..7a2bb83d4 100644 --- a/tests/graph/test_max_weight_matching.py +++ b/tests/graph/test_max_weight_matching.py @@ -10,7 +10,7 @@ # License for the specific language governing permissions and limitations # under the License. -# These tests are adapated from the networkx test cases: +# These tests are adapted from the networkx test cases: # https://github.com/networkx/networkx/blob/3351206a3ce5b3a39bb2fc451e93ef545b96c95b/networkx/algorithms/tests/test_matching.py import random diff --git a/tests/test_custom_return_types.py b/tests/test_custom_return_types.py index 725cf73ed..4795e45f4 100644 --- a/tests/test_custom_return_types.py +++ b/tests/test_custom_return_types.py @@ -1200,7 +1200,7 @@ def test_pickle(self): def test_str(self): res = rustworkx.all_pairs_dijkstra_path_lengths(self.dag, lambda _: 3.14) # Since all_pairs_dijkstra_path_lengths() is parallel the order of the - # output is non-determinisitic + # output is non-deterministic valid_values = [ "AllPairsPathLengthMapping{1: PathLengthMapping{}, " "0: PathLengthMapping{1: 3.14}}", "AllPairsPathLengthMapping{" diff --git a/tests/test_token_swapper.py b/tests/test_token_swapper.py index 2fa9b22de..17232498c 100644 --- a/tests/test_token_swapper.py +++ b/tests/test_token_swapper.py @@ -57,7 +57,7 @@ def test_small(self) -> None: self.assertEqual({i: i for i in range(8)}, permutation) def test_bug1(self) -> None: - """Tests for a bug that occured in happy swap chains of length >2.""" + """Tests for a bug that occurred in happy swap chains of length >2.""" graph = rx.PyGraph() graph.extend_from_edge_list( [(0, 1), (0, 2), (0, 3), (0, 4), (1, 2), (1, 3), (1, 4), (2, 3), (2, 4), (3, 4), (3, 6)] diff --git a/tox.ini b/tox.ini index 712139c01..618baf392 100644 --- a/tox.ini +++ b/tox.ini @@ -27,7 +27,7 @@ passenv = changedir = {toxinidir}/tests commands = stestr run {posargs} - python -c "print('\nrustworkx no longer supports tox. Please run the equivalent comand with nox:\n\n\tnox -e test\n')" + python -c "print('\nrustworkx no longer supports tox. Please run the equivalent command with nox:\n\n\tnox -e test\n')" [testenv:lint] basepython = python3 @@ -41,7 +41,7 @@ commands = ruff check ../rustworkx ../retworkx . ../setup.py cargo fmt --all -- --check python {toxinidir}/tools/find_stray_release_notes.py - python -c "print('\nrustworkx no longer supports tox. Please run the equivalent comand with nox:\n\n\tnox -e lint\n')" + python -c "print('\nrustworkx no longer supports tox. Please run the equivalent command with nox:\n\n\tnox -e lint\n')" [testenv:docs] @@ -61,7 +61,7 @@ commands = python -m ipykernel install --user jupyter kernelspec list sphinx-build -W -d {toxinidir}/docs/build/.doctrees -b html source build/html {posargs} - python -c "print('\nrustworkx no longer supports tox. Please run the equivalent comand with nox:\n\n\tnox -e docs\n')" + python -c "print('\nrustworkx no longer supports tox. Please run the equivalent command with nox:\n\n\tnox -e docs\n')" [testenv:docs-clean] skip_install = true @@ -69,7 +69,7 @@ deps = allowlist_externals = rm commands = rm -rf {toxinidir}/docs/build {toxinidir}/docs/source/apiref - python -c "print('\nrustworkx no longer supports tox. Please run the equivalent comand with nox:\n\n\tnox -e docs_clean\n')" + python -c "print('\nrustworkx no longer supports tox. Please run the equivalent command with nox:\n\n\tnox -e docs_clean\n')" [testenv:black] basepython = python3 @@ -78,7 +78,7 @@ deps = black~=24.8 commands = black {posargs} '../rustworkx' '../tests' '../retworkx' - python -c "print('\nrustworkx no longer supports tox. Please run the equivalent comand with nox:\n\tnox -e black\n')" + python -c "print('\nrustworkx no longer supports tox. Please run the equivalent command with nox:\n\tnox -e black\n')" [testenv:stubs] basepython = python3 @@ -90,4 +90,4 @@ extras = graphviz commands = python -m mypy.stubtest --concise rustworkx - python -c "print('\nrustworkx no longer supports tox. Please run the equivalent comand with nox:\n\n\tnox -e stubs\n')" + python -c "print('\nrustworkx no longer supports tox. Please run the equivalent command with nox:\n\n\tnox -e stubs\n')" From 73288c3629306dd9cf8d5d86fc8145f52ac19b81 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 2 Dec 2024 10:32:09 -0500 Subject: [PATCH 10/14] Bump indexmap from 2.6.0 to 2.7.0 (#1334) Bumps [indexmap](https://github.com/indexmap-rs/indexmap) from 2.6.0 to 2.7.0. - [Changelog](https://github.com/indexmap-rs/indexmap/blob/master/RELEASES.md) - [Commits](https://github.com/indexmap-rs/indexmap/compare/2.6.0...2.7.0) --- updated-dependencies: - dependency-name: indexmap dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- Cargo.lock | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 457b25b0b..5cfc8a377 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -149,9 +149,9 @@ checksum = "d231dfb89cfffdbc30e7fc41579ed6066ad03abda9e567ccafae602b97ec5024" [[package]] name = "indexmap" -version = "2.6.0" +version = "2.7.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "707907fe3c25f5424cce2cb7e1cbcafee6bdbe735ca90ef77c29e84591e5b9da" +checksum = "62f822373a4fe84d4bb149bf54e584a7f4abec90e072ed49cda0edea5b95471f" dependencies = [ "equivalent", "hashbrown 0.15.0", From 5dbc246de95cc46584de3af7a33ce28af0ac83b5 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 11 Dec 2024 12:00:45 -0500 Subject: [PATCH 11/14] Bump serde from 1.0.215 to 1.0.216 (#1338) Bumps [serde](https://github.com/serde-rs/serde) from 1.0.215 to 1.0.216. - [Release notes](https://github.com/serde-rs/serde/releases) - [Commits](https://github.com/serde-rs/serde/compare/v1.0.215...v1.0.216) --- updated-dependencies: - dependency-name: serde dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- Cargo.lock | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 5cfc8a377..1d30e4250 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -611,18 +611,18 @@ checksum = "f3cb5ba0dc43242ce17de99c180e96db90b235b8a9fdc9543c96d2209116bd9f" [[package]] name = "serde" -version = "1.0.215" +version = "1.0.216" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6513c1ad0b11a9376da888e3e0baa0077f1aed55c17f50e7b2397136129fb88f" +checksum = "0b9781016e935a97e8beecf0c933758c97a5520d32930e460142b4cd80c6338e" dependencies = [ "serde_derive", ] [[package]] name = "serde_derive" -version = "1.0.215" +version = "1.0.216" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ad1e866f866923f252f05c889987993144fb74e722403468a4ebd70c3cd756c0" +checksum = "46f859dbbf73865c6627ed570e78961cd3ac92407a2d117204c49232485da55e" dependencies = [ "proc-macro2", "quote", From 25077aa8517361d6873b68fe5ffb144177625339 Mon Sep 17 00:00:00 2001 From: Lars Esser <80355619+larsesser@users.noreply.github.com> Date: Sat, 14 Dec 2024 18:19:49 +0100 Subject: [PATCH 12/14] graphviz.py: add missing comma in IMAGE_TYPES (#1339) this prevented the usage of "vmlz" and "vrml" image type --- rustworkx/visualization/graphviz.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/rustworkx/visualization/graphviz.py b/rustworkx/visualization/graphviz.py index 6d614e47d..1fae8d99f 100644 --- a/rustworkx/visualization/graphviz.py +++ b/rustworkx/visualization/graphviz.py @@ -64,7 +64,8 @@ "svg", "svgz", "vml", - "vmlz" "vrml", + "vmlz", + "vrml", "vtx", "wbmp", "xdor", From d21fe54b97d6c293b289d55dc9212f9ee95ad94c Mon Sep 17 00:00:00 2001 From: Ivan Carvalho <8753214+IvanIsCoding@users.noreply.github.com> Date: Mon, 16 Dec 2024 19:53:43 -0500 Subject: [PATCH 13/14] Add missing release notes for #1292 and #1306 (#1336) * Add release notes for #1292 * Add release notes for degree centrality * Add centrality entries to the docs * Update releasenotes/notes/accept-generators-31f080871015233c.yaml Co-authored-by: Matthew Treinish * Update releasenotes/notes/accept-generators-31f080871015233c.yaml --------- Co-authored-by: Matthew Treinish --- .../api/algorithm_functions/centrality.rst | 3 ++ .../accept-generators-31f080871015233c.yaml | 11 +++++++ .../degree-centrality-e7ddf61a9a8fbafc.yaml | 33 +++++++++++++++++++ 3 files changed, 47 insertions(+) create mode 100644 releasenotes/notes/accept-generators-31f080871015233c.yaml create mode 100644 releasenotes/notes/degree-centrality-e7ddf61a9a8fbafc.yaml diff --git a/docs/source/api/algorithm_functions/centrality.rst b/docs/source/api/algorithm_functions/centrality.rst index 78b54514e..dfbaaa598 100644 --- a/docs/source/api/algorithm_functions/centrality.rst +++ b/docs/source/api/algorithm_functions/centrality.rst @@ -7,7 +7,10 @@ Centrality :toctree: ../../apiref rustworkx.betweenness_centrality + rustworkx.degree_centrality rustworkx.edge_betweenness_centrality rustworkx.eigenvector_centrality rustworkx.katz_centrality rustworkx.closeness_centrality + rustworkx.in_degree_centrality + rustworkx.out_degree_centrality \ No newline at end of file diff --git a/releasenotes/notes/accept-generators-31f080871015233c.yaml b/releasenotes/notes/accept-generators-31f080871015233c.yaml new file mode 100644 index 000000000..de35fb3ca --- /dev/null +++ b/releasenotes/notes/accept-generators-31f080871015233c.yaml @@ -0,0 +1,11 @@ +--- +features: + - | + The following methods now support sequences and generators as inputs, in addition to + the existing support for lists: + * :meth:`~rustworkx.PyGraph.add_nodes_from` and :meth:`~rustworkx.PyDiGraph.add_nodes_from` + * :meth:`~rustworkx.PyGraph.add_edges_from` and :meth:`~rustworkx.PyDiGraph.add_edges_from` + * :meth:`~rustworkx.PyGraph.add_edges_from_no_data` and :meth:`~rustworkx.PyDiGraph.add_edges_from_no_data` + * :meth:`~rustworkx.PyGraph.extend_from_edge_list` and :meth:`~rustworkx.PyDiGraph.extend_from_edge_list` + * :meth:`~rustworkx.PyGraph.extend_from_weighted_edge_list` and :meth:`~rustworkx.PyDiGraph.extend_from_weighted_edge_list` + diff --git a/releasenotes/notes/degree-centrality-e7ddf61a9a8fbafc.yaml b/releasenotes/notes/degree-centrality-e7ddf61a9a8fbafc.yaml new file mode 100644 index 000000000..391c664fb --- /dev/null +++ b/releasenotes/notes/degree-centrality-e7ddf61a9a8fbafc.yaml @@ -0,0 +1,33 @@ +--- +features: + - | + Added a new function, :func:`~rustworkx.degree_centrality` which is used to + compute the degree centrality for all nodes in a given graph. For + example: + + .. jupyter-execute:: + + import rustworkx as rx + from rustworkx.visualization import mpl_draw + + graph = rx.generators.hexagonal_lattice_graph(4, 4) + centrality = rx.degree_centrality(graph) + + # Generate a color list + colors = [] + for node in graph.node_indices(): + centrality_score = centrality[node] + graph[node] = centrality_score + colors.append(centrality_score) + mpl_draw( + graph, + with_labels=True, + node_color=colors, + node_size=650, + labels=lambda x: "{0:.2f}".format(x) + ) + + - | + Added two new functions, :func:`~rustworkx.in_degree_centrality` and + :func:`~rustworkx.out_degree_centrality` to calculate two special + types of degree centrality for directed graphs. \ No newline at end of file From 6bc3933121d04e4d4ae3484625cebebfe9de7fd5 Mon Sep 17 00:00:00 2001 From: Ivan Carvalho <8753214+IvanIsCoding@users.noreply.github.com> Date: Mon, 16 Dec 2024 19:54:09 -0500 Subject: [PATCH 14/14] Fix find_node_by_weight type annotations (#1324) --- .../fix-find-node-by-weight-stub-94e971291e1e6c96.yaml | 7 +++++++ rustworkx/rustworkx.pyi | 4 ++-- 2 files changed, 9 insertions(+), 2 deletions(-) create mode 100644 releasenotes/notes/fix-find-node-by-weight-stub-94e971291e1e6c96.yaml diff --git a/releasenotes/notes/fix-find-node-by-weight-stub-94e971291e1e6c96.yaml b/releasenotes/notes/fix-find-node-by-weight-stub-94e971291e1e6c96.yaml new file mode 100644 index 000000000..cc99652c3 --- /dev/null +++ b/releasenotes/notes/fix-find-node-by-weight-stub-94e971291e1e6c96.yaml @@ -0,0 +1,7 @@ +--- +fixes: + - | + Fixed a bug in the type hints for :meth:`~rustworkx.PyGraph.find_node_by_weight` + and :meth:`~rustworkx.PyDiGraph.find_node_by_weight`. + Refer to `issue 1243 `__ for + more information. diff --git a/rustworkx/rustworkx.pyi b/rustworkx/rustworkx.pyi index 4a18edd61..5d4219706 100644 --- a/rustworkx/rustworkx.pyi +++ b/rustworkx/rustworkx.pyi @@ -1224,7 +1224,7 @@ class PyGraph(Generic[_S, _T]): def filter_nodes(self, filter_function: Callable[[_S], bool]) -> NodeIndices: ... def find_node_by_weight( self, - obj: Callable[[_S], bool], + obj: _S, /, ) -> int | None: ... @staticmethod @@ -1378,7 +1378,7 @@ class PyDiGraph(Generic[_S, _T]): def find_adjacent_node_by_edge(self, node: int, predicate: Callable[[_T], bool], /) -> _S: ... def find_node_by_weight( self, - obj: Callable[[_S], bool], + obj: _S, /, ) -> int | None: ... def find_predecessors_by_edge(