Skip to content

Commit

Permalink
chore: Split crate into multiple crates. (#56)
Browse files Browse the repository at this point in the history
 - pywr-core: Contains the low-level model and interfaces with the LP solvers.
 - pywr-schema: Contains the Serde declared JSON schema.
 - pywr-python: Contains the Python extension module and package.
 - pywr-cli: Contains the Rust CLI.

Add brief crate descriptions and updated instructions to the README.
  • Loading branch information
jetuk authored Sep 30, 2023
1 parent 33efe4e commit 9739076
Show file tree
Hide file tree
Showing 124 changed files with 866 additions and 755 deletions.
106 changes: 26 additions & 80 deletions Cargo.toml
Original file line number Diff line number Diff line change
@@ -1,92 +1,29 @@
[package]
name = "pywr"
version = "2.0.0-dev"
authors = ["James Tomlinson <[email protected]>"]
edition = "2021"
rust-version = "1.60"
description = "A generalised water resource allocation model."
readme = "README.md"
repository = "https://github.com/pywr/pywr-next/"
license = "MIT OR Apache-2.0"
license-file = "LICENSE"
keywords = ["water", "modelling"]
categories = ["science", "simulation"]

# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html

[dependencies]
libc = "0.2.97"
thiserror = "1.0.25"
ndarray = "0.15.3"
numpy = "0.19.0"
num = "0.4.0"
float-cmp = "0.9.0"
hdf5 = { version="0.8.1" }
csv = "1.1"
clp-sys = { path = "./clp-sys" }
ipm-ocl = { path = "./ipm-ocl", optional = true }
ipm-simd = { path = "./ipm-simd", optional = true }
wasmer = "4.0.0"
serde = { version = "1", features = ["derive"] }
serde_json = "1.0"
time = { version = "0.3", features = ["serde", "serde-well-known", "serde-human-readable", "macros"] }
svgbobdoc = { version = "0.3.0", features = ["enable"] }
tracing = "0.1"
tracing-subscriber = { version ="0.3.17", features=["env-filter"] }
pyo3-log = "0.8.0"
indicatif = "0.17.2"
highs-sys = { git = "https://github.com/jetuk/highs-sys", branch="fix-build-libz-linking", optional = true }
# highs-sys = { path = "../../highs-sys" }
pyo3 = { version = "0.19.0" }
rayon = "1.6.1"
polars = { version = "0.33.2", features = ["lazy", "rows", "ndarray"] }
pyo3-polars = "0.7.0"
pywr-schema = { git = "https://github.com/pywr/pywr-schema/", tag="v0.7.0" }
rhai = { version="1.12.0", features=["sync"] }

# OpenCL
ocl = { version = "0.19", optional = true }

# Binary dependencies
clap = { version="4.0", features=["derive"] }
anyhow = "1.0.69"

rand = "0.8.5"
rand_distr = "0.4.3"
rand_chacha = "0.3.1"

[dev-dependencies]
tempfile = "3.3.0"
criterion = "0.5"

[lib]
name = "pywr"
path = "src/lib.rs"
crate-type = ["cdylib", "rlib"]


[features]
extension-module = ["pyo3/extension-module"]
default = ["extension-module"]
highs = ["dep:highs-sys"]
ipm-ocl = ["dep:ipm-ocl", "dep:ocl"]
ipm-simd = ["dep:ipm-simd"]

[[bin]]
name = "pywr-cli"
path = "src/main.rs"


[workspace]
resolver = "2"
members = [
"ipm-common",
"ipm-ocl",
"ipm-simd",
"clp-sys",
"pywr-core",
"pywr-schema",
"pywr-cli",
"pywr-python",
]
exclude = [
"tests/models/simple-wasm/simple-wasm-parameter"
]
# IPM packages are not default because they require nightly (portable_simd).
default-members = [
"clp-sys",
"pywr-core",
"pywr-schema",
"pywr-cli",
"pywr-python",
]


[profile.release]
opt-level = 3 # fast and small wasm
Expand All @@ -96,6 +33,15 @@ opt-level = 3 # fast and small wasm
# debug = true


[[bench]]
name = "random_models"
harness = false
[workspace.dependencies]
serde = { version = "1", features = ["derive"] }
serde_json = "1.0"
thiserror = "1.0.25"
time = { version = "0.3", features = ["serde", "serde-well-known", "serde-human-readable", "macros"] }
num = "0.4.0"
ndarray = "0.15.3"
pyo3 = { version = "0.19.0" }
tracing = "0.1"
csv = "1.1"
hdf5 = { version="0.8.1" }
pywr-v1-schema = { git = "https://github.com/pywr/pywr-schema/", tag="v0.7.0", package = "pywr-schema" }
21 changes: 15 additions & 6 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -117,10 +117,11 @@ git submodule init
git submodule update
```

Rust is required for installation. To create a development installation requires first compiling the
Rust library and then installing the Python package in editable model.
Rust is required for installation of the Python extension. To create a Python development installation
requires first compiling the Rust library and then installing the Python package in editable model.

```bash
cd pywr-python
maturin develop
pip install -e .
```
Expand All @@ -143,19 +144,19 @@ python -m pywr

### Rust CLI

A basic command line interface is included such that you can use this version of Pywr without Python. Running this
currently requires disabling the default features that create the Python extension.
A basic command line interface is included such that you can use this version of Pywr without Python.
This CLI is in the `pywr-cli` crate.

To see the CLI commands available run the following:

```bash
cargo run --no-default-features -- --help
cargo run -p pywr-cli -- --help
```

To run a Pywr v2 model use the following:

```bash
cargo run --no-default-features -- run tests/models/simple1.json
cargo run -p pywr-cli -- run tests/models/simple1.json
```

### Python CLI
Expand Down Expand Up @@ -191,6 +192,14 @@ Feedback on porting models is very welcome, so please open an issue with any que

<p align="right">(<a href="#readme-top">back to top</a>)</p>

<!-- CRATES -->
## Crates
This repository contains the following crates:

- `pywr-core`: A low-level Rust library for constructing network models. This crate interfaces with linear program solvers.
- `pywr-schema`: A Rust library for validating Pywr JSON files against a schema, and then building a model from the schema using `pywr-core`.
- `pywr-cli`: A command line interface for running Pywr models.
- `pywr-python`: A Python package (and extension) for constructing and running Pywr models.


<!-- ROADMAP -->
Expand Down
20 changes: 20 additions & 0 deletions pywr-cli/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
[package]
name = "pywr-cli"
version = "0.1.0"
edition = "2021"

# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html

[dependencies]
clap = { version="4.0", features=["derive"] }
anyhow = "1.0.69"

rand = "0.8.5"
rand_chacha = "0.3.1"
time = { workspace = true, features = ["serde", "serde-well-known", "serde-human-readable", "macros"] }
serde = { workspace = true }
serde_json = { workspace = true }
pywr-v1-schema = { workspace = true }

pywr-core = { path = "../pywr-core" }
pywr-schema = { path = "../pywr-schema" }
32 changes: 14 additions & 18 deletions src/main.rs → pywr-cli/src/main.rs
Original file line number Diff line number Diff line change
@@ -1,19 +1,19 @@
use anyhow::{Context, Result};
use clap::{Parser, Subcommand, ValueEnum};
use pywr::model::Model;
use pywr::schema::model::PywrModel;
use pywr::schema::ConversionError;
use pywr_core::model::Model;
#[cfg(feature = "ipm-ocl")]
use pywr::solvers::{ClIpmF32Solver, ClIpmF64Solver, ClIpmSolverSettings};
use pywr::solvers::{ClpSolver, ClpSolverSettings};
use pywr_core::solvers::{ClIpmF32Solver, ClIpmF64Solver, ClIpmSolverSettings};
use pywr_core::solvers::{ClpSolver, ClpSolverSettings};
#[cfg(feature = "highs")]
use pywr::solvers::{HighsSolver, HighsSolverSettings};
use pywr_core::solvers::{HighsSolver, HighsSolverSettings};
#[cfg(feature = "ipm-simd")]
use pywr::solvers::{SimdIpmF64Solver, SimdIpmSolverSettings};
use pywr::test_utils::make_random_model;
use pywr::timestep::Timestepper;
use pywr::tracing::setup_tracing;
use pywr::PywrError;
use pywr_core::solvers::{SimdIpmF64Solver, SimdIpmSolverSettings};
use pywr_core::test_utils::make_random_model;
use pywr_core::timestep::Timestepper;
use pywr_core::tracing::setup_tracing;
use pywr_core::PywrError;
use pywr_schema::model::PywrModel;
use pywr_schema::ConversionError;
use rand::SeedableRng;
use rand_chacha::ChaCha8Rng;
use std::fmt::{Display, Formatter};
Expand Down Expand Up @@ -138,15 +138,11 @@ fn convert(path: &Path) -> Result<()> {
&& (path.extension().unwrap() == "json")
&& (!path.file_stem().unwrap().to_str().unwrap().contains("_v2"))
{
v1_to_v2(&path)
.map_err(PywrError::Conversion)
.with_context(|| format!("Could not convert model: `{:?}`", &path))?;
v1_to_v2(&path).with_context(|| format!("Could not convert model: `{:?}`", &path))?;
}
}
} else {
v1_to_v2(path)
.map_err(PywrError::Conversion)
.with_context(|| format!("Could not convert model: `{:?}`", path))?;
v1_to_v2(path).with_context(|| format!("Could not convert model: `{:?}`", path))?;
}

Ok(())
Expand All @@ -156,7 +152,7 @@ fn v1_to_v2(path: &Path) -> std::result::Result<(), ConversionError> {
println!("Model: {}", path.display());

let data = std::fs::read_to_string(path).unwrap();
let schema: pywr_schema::PywrModel = serde_json::from_str(data.as_str()).unwrap();
let schema: pywr_v1_schema::PywrModel = serde_json::from_str(data.as_str()).unwrap();
let schema_v2: PywrModel = schema.try_into()?;

// There must be a better way to do this!!
Expand Down
63 changes: 63 additions & 0 deletions pywr-core/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
[package]
name = "pywr-core"
version = "2.0.0-dev"
authors = ["James Tomlinson <[email protected]>"]
edition = "2021"
rust-version = "1.60"
description = "A generalised water resource allocation model."
readme = "README.md"
repository = "https://github.com/pywr/pywr-next/"
license = "MIT OR Apache-2.0"
license-file = "LICENSE"
keywords = ["water", "modelling"]
categories = ["science", "simulation"]

# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html

[dependencies]
libc = "0.2.97"
thiserror = { workspace = true }
ndarray = { workspace = true }
num = { workspace = true }
float-cmp = "0.9.0"
hdf5 = { workspace = true }
csv = { workspace = true }
clp-sys = { path = "../clp-sys" }
ipm-ocl = { path = "../ipm-ocl", optional = true }
ipm-simd = { path = "../ipm-simd", optional = true }
wasmer = "4.0.0"
time = { workspace = true, features = ["macros"] }
tracing = { workspace = true }
tracing-subscriber = { version ="0.3.17", features=["env-filter"] }
highs-sys = { git = "https://github.com/jetuk/highs-sys", branch="fix-build-libz-linking", optional = true }
# highs-sys = { path = "../../highs-sys" }

pyo3 = { workspace = true }


rayon = "1.6.1"


rhai = { version="1.12.0", features=["sync"] }

# OpenCL
ocl = { version = "0.19", optional = true }

rand = "0.8.5"
rand_distr = "0.4.3"
rand_chacha = "0.3.1"

[dev-dependencies]
criterion = "0.5"

[features]
highs = ["dep:highs-sys"]
ipm-ocl = ["dep:ipm-ocl", "dep:ocl"]
ipm-simd = ["dep:ipm-simd"]
default = []



[[bench]]
name = "random_models"
harness = false
12 changes: 6 additions & 6 deletions benches/random_models.rs → pywr-core/benches/random_models.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,14 +9,14 @@
/// input flows) and number of CPU threads.
use criterion::{criterion_group, criterion_main, BenchmarkId, Criterion, Throughput};
#[cfg(feature = "ipm-ocl")]
use pywr::solvers::{ClIpmF64Solver, ClIpmSolverSettings, ClIpmSolverSettingsBuilder};
use pywr::solvers::{ClpSolver, ClpSolverSettings, ClpSolverSettingsBuilder};
use pywr_core::solvers::{ClIpmF64Solver, ClIpmSolverSettings, ClIpmSolverSettingsBuilder};
use pywr_core::solvers::{ClpSolver, ClpSolverSettings, ClpSolverSettingsBuilder};
#[cfg(feature = "highs")]
use pywr::solvers::{HighsSolver, HighsSolverSettings};
use pywr_core::solvers::{HighsSolver, HighsSolverSettings};
#[cfg(feature = "ipm-simd")]
use pywr::solvers::{SimdIpmF64Solver, SimdIpmSolverSettings, SimdIpmSolverSettingsBuilder};
use pywr::test_utils::make_random_model;
use pywr::timestep::Timestepper;
use pywr_core::solvers::{SimdIpmF64Solver, SimdIpmSolverSettings, SimdIpmSolverSettingsBuilder};
use pywr_core::test_utils::make_random_model;
use pywr_core::timestep::Timestepper;
use rand::SeedableRng;
use rand_chacha::ChaCha8Rng;
use std::num::NonZeroUsize;
Expand Down
File renamed without changes.
File renamed without changes.
File renamed without changes.
Loading

0 comments on commit 9739076

Please sign in to comment.