pflow_metamodel/lib.rs
1//! > **Metamodel - A Rust Library for Abstract State Machine Modeling**
2//!
3//! See the code below for an example of how to create a Petri-net model using the `pflow_metamodel` library.
4//!
5//! Also see: [Macro pflow_metamodel::pflow_dsl](macro.pflow_dsl.html) for creating models using an internal rust DSL rather than json.
6//! ```
7//! use pflow_metamodel::*;
8//!
9//! let model: Model = pflow_json!{{
10//! "modelType": "petriNet",
11//! "version": "v0",
12//! "places": {
13//! "place0": { "offset": 0, "capacity": 3, "x": 100, "y": 180 }
14//! },
15//! "transitions": {
16//! "txn0": { "role": "role0", "x": 20, "y": 100 },
17//! "txn1": { "role": "role0", "x": 180, "y": 100 },
18//! "txn2": { "role": "role0", "x": 20, "y": 260 },
19//! "txn3": { "role": "role0", "x": 180, "y": 260 }
20//! },
21//! "arcs": [
22//! { "source": "txn0", "target": "place0" },
23//! { "source": "place0", "target": "txn1", "weight": 3 },
24//! { "source": "txn2", "target": "place0", "weight": 3, "inhibit": true },
25//! { "source": "place0", "target": "txn3", "inhibit": true }
26//! ]
27//! }};
28//!
29//! let state = model.vm.initial_vector();
30//! assert_eq!(state, vec![0]);
31//! let res = model.vm.transform(&state, "txn0", 1);
32//! assert!(res.ok);
33//! assert_eq!(state, vec![0]); // input state is _not_ mutated
34//! assert_eq!(res.output, vec![1]);
35//! let t = model.net.transitions.get("txn0");
36//! assert!(t.is_some());
37//! ```
38//!
39//! - Provides a DSL-driven framework for modeling and simulating Petri-nets, wf-nets, and DFAs.
40//! - State machine data types are executed as a [Vector Addition State Machine (VASM)](https://en.wikipedia.org/wiki/Vector_addition_system).
41//! - Data models are viewable / shareable in browsers by using [https://pflow.dev](https://pflow.dev/p/zb2rhkizUC1o2JuvgwhbH1XrLZkdK8x66pP1KR7sWAEw9c5FE/) JSON format.
42//! - Models can be compressed and shared as base64 encoded blobs.
43//! - Models can be consistently hashed and shared as CIDs.
44//!
45//! ![pflow][pflow]
46//!
47//! [pflow]: 
48//!
49//! # Dining Philosophers Example
50//!
51//! The following example demonstrates the dining philosophers problem using the `PetriNet` struct.
52//!
53//! - read more about the [dining philosophers problem](https://pflow.dev/examples-dining-philosophers).
54//! - interact with the model [dining philosophers model](https://pflow.dev/p/zb2rhimQLDKMY6yBXMLV2DJyCPqseb9kTJKdjiwKgzQEgwGvt/)
55//! ```rust
56//! use pflow_metamodel::*;
57//!
58//! let model = pflow_json!{{
59//! "modelType": "petriNet",
60//! "version": "v0",
61//! "places": {
62//! "chopstick0": { "offset": 0, "initial": 1, "x": 403, "y": 340 },
63//! "chopstick1": { "offset": 1, "initial": 1, "x": 534, "y": 345 },
64//! "chopstick2": { "offset": 2, "initial": 1, "x": 358, "y": 467 },
65//! "chopstick3": { "offset": 3, "initial": 1, "x": 547, "y": 461 },
66//! "chopstick4": { "offset": 4, "initial": 1, "x": 451, "y": 536 },
67//! "0right": { "offset": 5, "x": 415, "y": 181 },
68//! "0left": { "offset": 6, "x": 545, "y": 177 },
69//! "1right": { "offset": 7, "x": 719, "y": 288 },
70//! "1left": { "offset": 8, "x": 769, "y": 404 },
71//! "2left": { "offset": 9, "x": 686, "y": 584 },
72//! "2right": { "offset": 10, "x": 594, "y": 678 },
73//! "3left": { "offset": 11, "x": 315, "y": 679 },
74//! "3right": { "offset": 12, "x": 216, "y": 608 },
75//! "4right": { "offset": 13, "x": 148, "y": 397 },
76//! "4left": { "offset": 14, "x": 183, "y": 289 }
77//! },
78//! "transitions": {
79//! "0think": { "x": 478, "y": 106 },
80//! "0eat": { "x": 473, "y": 247 },
81//! "1eat": { "x": 654, "y": 396 },
82//! "2eat": { "x": 574, "y": 573 },
83//! "3eat": { "x": 333, "y": 556 },
84//! "4eat": { "x": 267, "y": 370 },
85//! "1think": { "x": 842, "y": 304 },
86//! "4think": { "x": 72, "y": 314 },
87//! "3think": { "x": 200, "y": 726 },
88//! "2think": { "x": 740, "y": 699 }
89//! },
90//! "arcs": [
91//! { "source": "chopstick0", "target": "0eat" },
92//! { "source": "chopstick1", "target": "0eat" },
93//! { "source": "chopstick0", "target": "4eat" },
94//! { "source": "chopstick2", "target": "4eat" },
95//! { "source": "chopstick1", "target": "1eat" },
96//! { "source": "chopstick3", "target": "1eat" },
97//! { "source": "chopstick2", "target": "3eat" },
98//! { "source": "chopstick4", "target": "3eat" },
99//! { "source": "chopstick4", "target": "2eat" },
100//! { "source": "chopstick3", "target": "2eat" },
101//! { "source": "0eat", "target": "0right" },
102//! { "source": "0eat", "target": "0left" },
103//! { "source": "1eat", "target": "1right" },
104//! { "source": "1eat", "target": "1left" },
105//! { "source": "2eat", "target": "2left" },
106//! { "source": "2eat", "target": "2right" },
107//! { "source": "3eat", "target": "3right" },
108//! { "source": "3eat", "target": "3left" },
109//! { "source": "4eat", "target": "4right" },
110//! { "source": "4eat", "target": "4left" },
111//! { "source": "0right", "target": "0think" },
112//! { "source": "0left", "target": "0think" },
113//! { "source": "1right", "target": "1think" },
114//! { "source": "1left", "target": "1think" },
115//! { "source": "2left", "target": "2think" },
116//! { "source": "2right", "target": "2think" },
117//! { "source": "4left", "target": "4think" },
118//! { "source": "4right", "target": "4think" },
119//! { "source": "3right", "target": "3think" },
120//! { "source": "3left", "target": "3think" },
121//! { "source": "4think", "target": "chopstick0" },
122//! { "source": "4think", "target": "chopstick2" },
123//! { "source": "0think", "target": "chopstick0" },
124//! { "source": "0think", "target": "chopstick1" },
125//! { "source": "1think", "target": "chopstick1" },
126//! { "source": "1think", "target": "chopstick3" },
127//! { "source": "2think", "target": "chopstick3" },
128//! { "source": "2think", "target": "chopstick4" },
129//! { "source": "3think", "target": "chopstick2" },
130//! { "source": "3think", "target": "chopstick4" }
131//! ]
132//! }};
133//! ```
134//!
135//! ![dining_philosophers][dining_philosophers]
136//!
137//! [dining_philosophers]: 
138//!
139
140// Rustc lints
141// <https://doc.rust-lang.org/rustc/lints/listing/allowed-by-default.html>
142#![warn(
143 anonymous_parameters,
144 bare_trait_objects,
145 elided_lifetimes_in_paths,
146 missing_copy_implementations,
147 rust_2018_idioms,
148 single_use_lifetimes,
149 trivial_casts,
150 trivial_numeric_casts,
151 unsafe_code,
152 unused_extern_crates,
153 unused_import_braces,
154)]
155
156// Clippy lints
157// <https://rust-lang.github.io/rust-clippy/current/>
158#![warn(
159 clippy::all,
160 clippy::cargo,
161 clippy::dbg_macro,
162 clippy::float_cmp_const,
163 clippy::get_unwrap,
164 clippy::mem_forget,
165 clippy::nursery,
166 clippy::pedantic,
167 clippy::todo,
168 clippy::unwrap_used,
169)]
170
171// Allow some clippy lints
172#![allow(
173 clippy::cargo_common_metadata,
174 clippy::default_trait_access,
175 clippy::doc_markdown,
176 clippy::enum_glob_use,
177 clippy::enum_variant_names,
178 clippy::if_not_else,
179 clippy::missing_errors_doc,
180 clippy::must_use_candidate,
181 clippy::needless_pass_by_value,
182 clippy::struct_excessive_bools,
183 clippy::use_self,
184 clippy::multiple_crate_versions,
185 clippy::struct_field_names,
186 clippy::similar_names,
187)]
188
189/// The `petri_net` module contains the definition and implementation of the `PetriNet` struct.
190pub mod petri_net;
191
192/// The `oid` module is used to generate CID's for the zipped blobs.
193pub mod oid;
194
195/// The `compression` module contains functions for zipping/unzipping models as sharable base64 blobs.
196pub mod compression;
197
198/// The `vasm` module contains the implementation of a Vector Addition State Machine (VASM).
199pub mod vasm;
200
201/// The `dsl` module contains `FlowDsl` and `Builder` traits for defining Petri-nets.
202pub mod dsl;
203
204/// The `zblob` contains utilities to facilitate loading zipped blob data as petri-nets.
205pub mod zblob;
206
207/// The `model` encapsulates the `PetriNet` and `Vasm` objects into a single `Model` object.
208pub mod model;
209
210pub use crate::model::*;
211pub use crate::vasm::*;
212
213
214/// Create a model using the pflow DSL
215/// This is the primary way to create a model for most use cases
216///
217/// ![pflow][pflow]
218///
219/// [pflow]: 
220/// # Example
221///
222/// ```
223/// use pflow_metamodel::*;
224///
225/// let model: Model = pflow_dsl!{
226/// declare "petriNet"
227/// cell "place0", 0, 3, [100, 180]
228/// func "txn0", "default", [20, 100]
229/// func "txn1", "default", [180, 100]
230/// func "txn2", "default", [20, 260]
231/// func "txn3", "default", [180, 260]
232/// arrow "txn0", "place0", 1
233/// arrow "place0", "txn1", 3
234/// guard "txn2", "place0", 3
235/// guard "place0", "txn3", 1
236/// };
237///
238/// let state = model.vm.initial_vector();
239/// assert_eq!(state, vec![0]);
240/// let res = model.vm.transform(&state, "txn0", 1);
241/// assert!(res.ok);
242/// assert_eq!(state, vec![0]); // input state is _not_ mutated
243/// assert_eq!(res.output, vec![1]);
244/// let t = model.net.transitions.get("txn0");
245/// assert!(t.is_some());
246/// ```
247#[macro_export]
248macro_rules! pflow_dsl {
249 ($($name:ident $($args:expr),*)*) => {{
250 declaration_function! {
251 |p: &mut dyn dsl::Dsl| {
252 #[allow(unused)]
253 fn declare(p: &mut dyn dsl::Dsl, model_type: &str) {
254 p.model_type(model_type);
255 }
256 #[allow(unused)]
257 fn cell(p: &mut dyn dsl::Dsl, name: &str, initial: i32, capacity: i32, coords: [i32; 2]) {
258 p.cell(name, Option::from(initial), Option::from(capacity), coords[0],coords[1]);
259 }
260 #[allow(unused)]
261 fn func(p: &mut dyn dsl::Dsl, name: &str, role: &str, coords: [i32; 2]) {
262 p.func(name, role, coords[0], coords[1]);
263 }
264 #[allow(unused)]
265 fn arrow(p: &mut dyn dsl::Dsl, source: &str, target: &str, weight: i32) {
266 p.arrow(source, target, weight);
267 }
268 #[allow(unused)]
269 fn guard(p: &mut dyn dsl::Dsl, source: &str, target: &str, weight: i32) {
270 p.guard(source, target, weight);
271 }
272 $(
273 $name(p, $($args),*);
274 )*
275 }
276 }
277 }};
278}
279
280/// Create a model using the internal DSL functions without macro rewriting
281/// Generally not used directly, but may be useful for deeper integration with other libraries
282///
283/// ![pflow][pflow]
284///
285/// [pflow]: 
286///
287/// # Example
288///
289/// ```
290/// use pflow_metamodel::*;
291///
292/// let model = declaration_function! {
293/// |p: &mut dyn dsl::Dsl| {
294/// p.model_type("petriNet");
295/// p.cell("place0", Option::from(0), Option::from(3), 100, 180);
296/// p.func("txn0", "default", 20, 100);
297/// p.func("txn1", "default", 180, 100);
298/// p.func("txn2", "default", 20, 260);
299/// p.func("txn3", "default", 180, 260);
300/// p.arrow("txn0", "place0", 1);
301/// p.arrow("place0", "txn1", 3);
302/// p.guard("txn2", "place0", 3);
303/// p.guard("place0", "txn3", 1);
304/// }
305/// };
306///
307/// let state = model.vm.initial_vector();
308/// assert_eq!(state, vec![0]);
309/// let res = model.vm.transform(&state, "txn0", 1);
310/// assert!(res.ok);
311/// assert_eq!(state, vec![0]); // input state is _not_ mutated
312/// assert_eq!(res.output, vec![1]);
313/// let t = model.net.transitions.get("txn0");
314/// assert!(t.is_some());
315/// ```
316#[macro_export]
317macro_rules! declaration_function {
318 ($($flow_dsl:tt)*) => {{
319 let model = model::Model::new(
320 $($flow_dsl)*
321 );
322 model
323 }};
324}
325
326/// Create a model from a JSON string compatible with pflow.xyz
327///
328/// ![pflow][pflow]
329///
330/// [pflow]: 
331///
332///
333/// # Example
334///
335/// ```
336/// use pflow_metamodel::*;
337///
338/// let model: Model = pflow_json!{{
339/// "modelType": "petriNet",
340/// "version": "v0",
341/// "places": {
342/// "place0": { "offset": 0, "capacity": 3, "x": 100, "y": 180 }
343/// },
344/// "transitions": {
345/// "txn0": { "role": "role0", "x": 20, "y": 100 },
346/// "txn1": { "role": "role0", "x": 180, "y": 100 },
347/// "txn2": { "role": "role0", "x": 20, "y": 260 },
348/// "txn3": { "role": "role0", "x": 180, "y": 260 }
349/// },
350/// "arcs": [
351/// { "source": "txn0", "target": "place0" },
352/// { "source": "place0", "target": "txn1", "weight": 3 },
353/// { "source": "txn2", "target": "place0", "weight": 3, "inhibit": true },
354/// { "source": "place0", "target": "txn3", "inhibit": true }
355/// ]
356/// }};
357///
358/// let state = model.vm.initial_vector();
359/// assert_eq!(state, vec![0]);
360/// let res = model.vm.transform(&state, "txn0", 1);
361/// assert!(res.ok);
362/// assert_eq!(state, vec![0]); // input state is _not_ mutated
363/// assert_eq!(res.output, vec![1]);
364/// let t = model.net.transitions.get("txn0");
365/// assert!(t.is_some());
366/// ```
367#[macro_export]
368macro_rules! pflow_json {
369 ($($flow_json:tt)*) => {{
370 let mut net = petri_net::PetriNet::from_json_value(
371 serde_json::json!($($flow_json)*)
372 ).expect("json fault");
373
374 let sm = vasm::StateMachine::from_model(&mut net);
375 model::Model {
376 net,
377 declaration: Vec::new(),
378 vm: Box::new(sm),
379 }
380 }};
381}
382
383/// Create a model from a diagram string
384///
385/// Example:
386///
387/// ```rust
388/// use pflow_metamodel::*;
389///
390/// // NOTICE: use uppercase for states vs lowercase for transitions
391/// let model = pflow_diagram!{ ModelType::Workflow;
392/// Water --> boil_water;
393/// boil_water --> BoiledWater;
394/// CoffeeBeans --> grind_beans;
395/// grind_beans --> GroundCoffee;
396/// BoiledWater --> brew_coffee;
397/// GroundCoffee --> brew_coffee;
398/// Filter --> brew_coffee;
399/// brew_coffee --> CoffeeInPot;
400/// CoffeeInPot --> pour_coffee;
401/// Cup --> pour_coffee;
402/// };
403/// println!("https://pflow.dev?z={}", model.net.to_zblob().base64_zipped);
404///
405/// // NOTICE: only specify states in a diagram not providing a ModelType::
406/// let state_model = pflow_diagram! {
407/// Crash --> [*];
408/// Moving --> Crash;
409/// Moving --> Still;
410/// Still --> Moving;
411/// Still --> [*];
412/// [*] --> Still;
413/// };
414///
415/// println!("https://pflow.dev?z={}", state_model.net.to_zblob().base64_zipped);
416/// ```
417#[macro_export]
418macro_rules! pflow_diagram {
419 ($($workflow_declaration:tt)*) => {
420 {
421 Model::from_diagram(stringify!($($workflow_declaration)*).to_string())
422 }
423 };
424}
425
426#[cfg(test)]
427mod tests {
428 use super::*;
429 use std::sync::{Arc, Mutex};
430
431 #[test]
432 fn test_pflow_diagram() {
433 let coffee_machine = pflow_diagram! { ModelType::Workflow;
434 Water --> boil_water;
435 boil_water --> BoiledWater;
436 CoffeeBeans --> grind_beans;
437 grind_beans --> GroundCoffee;
438 BoiledWater --> brew_coffee;
439 GroundCoffee --> brew_coffee;
440 Filter --> brew_coffee;
441 brew_coffee --> CoffeeInPot;
442 CoffeeInPot --> pour_coffee;
443 Cup --> pour_coffee;
444 };
445 let zblob = coffee_machine.net.to_zblob();
446 println!("https://pflow.dev?z={}", zblob.base64_zipped);
447 }
448
449 #[test]
450 fn test_dsl() {
451 let j = pflow_dsl! {
452 declare "petriNet"
453 cell "b", 1, 0, [1, 1]
454 cell "a", 1, 0, [1, 2]
455 func "f", "default", [2, 1]
456 func "g", "default", [2, 2]
457 arrow "a", "f", 1
458 guard "b", "g", 1
459 };
460
461 let initial = j.vm.initial_vector();
462 assert_eq!(initial, vec![1, 1]);
463
464 j.net.transitions.get("f").expect("expected transition");
465
466 let res = j.vm.transform(&initial, "f", 1);
467 assert!(res.ok);
468 assert_eq!(initial, vec![1, 1]); // input state is _not_ mutated
469 assert_eq!(res.output, vec![1, 0]);
470 }
471
472 #[test]
473 fn test_json_dsl() {
474 let j = pflow_json! {
475 {
476 "modelType": "petriNet",
477 "version": "v0",
478 "places": {
479 "a": { "offset": 0, "initial": 1, "capacity": 1, "x": 0, "y": 0 },
480 "b": { "offset": 1, "initial": 1, "capacity": 1, "x": 0, "y": 0 }
481 },
482 "transitions": {
483 "f": { "role": "default", "x": 0, "y": 0 }
484 },
485 "arcs": [
486 { "source": "a", "target": "f", "weight": 1 },
487 { "source": "b", "target": "f", "weight": 1 }
488 ]
489 }
490 };
491
492 let initial = j.vm.initial_vector();
493 assert_eq!(initial, vec![1, 1]);
494 j.net.transitions.get("f").expect("expected transition");
495 }
496
497 #[test]
498 fn test_model() {
499 let model = declaration_function! {
500 |p: &mut dyn dsl::Dsl| {
501 p.model_type("petriNet");
502 p.cell("b", Some(1), None, 0, 0);
503 p.func("f", "default", 1, 1);
504 p.cell("a", Some(1), None, 0, 0);
505 p.func("g", "default", 1, 1);
506 p.arrow("a", "f", 1);
507 p.guard("b", "g", 1);
508 }
509 };
510
511 assert_eq!(model.net.model_type, "petriNet");
512 let zblob = model.net.to_zblob();
513 assert_eq!(
514 zblob.ipfs_cid,
515 "zb2rhXMTtKZq96QpdSzkSYmEPKttirMw4okCG8c5QxwygAvWG"
516 );
517 }
518
519 #[test]
520 fn test_workflow_model() {
521 let model = pflow_dsl! {
522 declare "workflow"
523 cell "Water", 0, 1, [100, 300]
524 cell "CoffeeBeans", 0, 1, [180, 300]
525 cell "BoiledWater", 0, 1, [195, 397]
526 cell "GroundCoffee", 0, 1, [250, 339]
527 cell "Filter", 0, 1, [290, 280]
528 cell "CoffeeInPot", 0, 1, [328, 366]
529 cell "Cup", 0, 1, [365, 312]
530 cell "step0", 1, 1, [100, 100]
531 cell "step1", 0, 1, [180, 100]
532 cell "step2", 0, 1, [260, 100]
533 cell "step3", 0, 1, [340, 100]
534 cell "step4", 0, 1, [420, 100]
535 func "boil_water", "coffee_machine", [100, 200]
536 func "brew_coffee", "coffee_machine", [260, 200]
537 func "grind_beans", "coffee_machine", [180, 200]
538 func "pour_coffee", "coffee_machine", [340, 200]
539 arrow "Water", "boil_water", 1
540 arrow "CoffeeBeans", "grind_beans", 1
541 arrow "BoiledWater", "brew_coffee", 1
542 arrow "GroundCoffee", "brew_coffee", 1
543 arrow "Filter", "brew_coffee", 1
544 arrow "CoffeeInPot", "pour_coffee", 1
545 arrow "Cup", "pour_coffee", 1
546 arrow "step0", "boil_water", 1
547 arrow "boil_water", "step1", 1
548 arrow "step1", "grind_beans", 1
549 arrow "grind_beans", "step2", 1
550 arrow "step2", "brew_coffee", 1
551 arrow "brew_coffee", "step3", 1
552 arrow "step3", "pour_coffee", 1
553 arrow "pour_coffee", "step4", 1
554 };
555
556 let zb = model.net.to_zblob();
557 println!("https://pflow.dev?z={}", zb.base64_zipped);
558 assert_eq!(
559 zb.ipfs_cid,
560 "zb2rhcgvzu3CJ7KaRmySuR253VD2DFPqyQftHhDMKAPaQRzjE"
561 );
562
563 let state = Arc::new(Mutex::new(model.vm.initial_vector()));
564 {
565 let mut state_lock = state.lock().expect("state lock");
566 let res = model.vm.transform(&state_lock, "boil_water", 1);
567 if res.ok {
568 println!("{res:?}");
569 *state_lock = res.output;
570 drop(state_lock);
571 } else {
572 panic!("expected ok");
573 }
574 }
575 }
576}