1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
use crate::action::Action;
use crate::comm::SignalId;
use crate::resource::ResourceAddr;
use crate::server::{config::OptionalConfig, Config, State};
use crate::util::Hash;
use eyre::{eyre, Result};
use serde::{Deserialize, Serialize};
use serde_cbor::ser::to_vec_packed;
use serde_cbor::Value;
use std::collections::BTreeMap;
#[derive(Deserialize, Serialize, Debug)]
pub struct Block {
name: String,
#[serde(default)]
#[serde(alias = "cfg")]
config: OptionalConfig,
tree: Box<dyn Action>,
#[serde(default)]
state: BTreeMap<SignalId, Value>,
}
impl Block {
pub fn init(&mut self) -> Result<()> {
self.verify_name()?;
self.verify_connections()?;
Ok(())
}
fn verify_name(&self) -> Result<()> {
if self.name.is_empty() {
Err(eyre!("Block `name` cannot be the empty string."))
} else if !self
.name
.chars()
.all(|c| c.is_alphabetic() || c.is_alphanumeric() | "+-_ ".contains(c))
{
Err(eyre!(
"Block `name` characters need to be alphanumeric or one of (+-_ ): '{}'",
self.name
))
} else {
Ok(())
}
}
fn verify_connections(&self) -> Result<()> {
let mut in_signals = self.tree.in_signals();
let mut out_signals = self.tree.out_signals();
in_signals.insert(0);
out_signals.insert(0);
let in_not_out: Vec<_> = in_signals.difference(&out_signals).collect();
let out_not_in: Vec<_> = out_signals.difference(&in_signals).collect();
if !in_not_out.is_empty() {
Err(eyre!("Consumed signals are never produced: {in_not_out:?}"))
} else if !out_not_in.is_empty() {
Err(eyre!("Produced signals are never consumed: {out_not_in:?}"))
} else {
Ok(())
}
}
pub fn resources(&self, config: &Config) -> Vec<ResourceAddr> {
self.tree.resources(config)
}
#[inline(always)]
pub fn action_tree(&self) -> &dyn Action {
&*self.tree
}
#[inline(always)]
pub fn action_tree_vec(&self) -> Vec<u8> {
to_vec_packed(&self.tree).unwrap()
}
#[inline(always)]
pub fn default_state(&self) -> &State {
&self.state
}
#[inline(always)]
pub fn label(&self) -> &str {
&self.name
}
#[inline]
pub fn config(&self, base_config: &Config) -> Config {
self.config.fill_blanks(base_config)
}
}
impl Hash for Block {
fn hash(&self) -> String {
use sha2::{Digest, Sha256};
let mut hasher = Sha256::default();
hasher.update(&serde_cbor::to_vec(&(&self.tree, &self.config)).unwrap());
hex::encode(hasher.finalize())
}
}