Generalize constraints to observables #48
|
@ -13,6 +13,7 @@ itertools = "0.13.0"
|
|||
js-sys = "0.3.70"
|
||||
lazy_static = "1.5.0"
|
||||
nalgebra = "0.33.0"
|
||||
readonly = "0.2.12"
|
||||
rustc-hash = "2.0.0"
|
||||
slab = "0.4.9"
|
||||
sycamore = "0.9.0-beta.3"
|
||||
|
|
|
@ -7,7 +7,7 @@ use web_sys::{console, wasm_bindgen::JsValue}; /* DEBUG */
|
|||
|
||||
use crate::{
|
||||
engine::{Q, local_unif_to_std, realize_gram, ConfigSubspace, PartialMatrix},
|
||||
specified::{SpecifiedValue, SpecifiedValue::{Absent, Present}}
|
||||
specified::SpecifiedValue
|
||||
};
|
||||
|
||||
// the types of the keys we use to access an assembly's elements and regulators
|
||||
|
@ -239,7 +239,7 @@ impl Assembly {
|
|||
reps.0.dot(&(&*Q * reps.1))
|
||||
}
|
||||
);
|
||||
let set_point = create_signal(Absent);
|
||||
let set_point = create_signal(SpecifiedValue::from_empty_spec());
|
||||
self.insert_regulator(Regulator {
|
||||
subjects: subjects,
|
||||
measurement: measurement,
|
||||
|
@ -256,9 +256,9 @@ impl Assembly {
|
|||
&JsValue::from(reg.subjects.0),
|
||||
&JsValue::from(reg.subjects.1),
|
||||
&JsValue::from(":"),
|
||||
&JsValue::from(reg.set_point.with_untracked(
|
||||
|set_pt| set_pt.spec()
|
||||
))
|
||||
®.set_point.with_untracked(
|
||||
|set_pt| JsValue::from(set_pt.spec.as_str())
|
||||
)
|
||||
);
|
||||
}
|
||||
});
|
||||
|
@ -269,7 +269,7 @@ impl Assembly {
|
|||
console::log_1(&JsValue::from(
|
||||
format!("Updated constraint with subjects ({}, {})", subjects.0, subjects.1)
|
||||
));
|
||||
if set_point.with(|set_pt| matches!(set_pt, Present { .. })) {
|
||||
if set_point.with(|set_pt| set_pt.is_present()) {
|
||||
self.realize();
|
||||
}
|
||||
});
|
||||
glen marked this conversation as resolved
Outdated
|
||||
|
@ -292,11 +292,11 @@ impl Assembly {
|
|||
self.regulators.with_untracked(|regs| {
|
||||
for (_, reg) in regs {
|
||||
reg.set_point.with_untracked(|set_pt| {
|
||||
if let Present { value, .. } = set_pt {
|
||||
if let Some(val) = set_pt.value {
|
||||
let subjects = reg.subjects;
|
||||
glen marked this conversation as resolved
Outdated
glen
commented
Please illuminate me: Why do we not need to write Please illuminate me: Why do we not need to write `SpecifiedValue::Absent` here? I mean, I like the brevity, but I think I am unclear about when the bare variants are in scope...
Vectornaut
commented
The declaration
puts all variants in scope. I currently have it just after the definition of The declaration
```
use SpecifiedValue::*;
```
puts all variants in scope. I currently have it just after the definition of `SpecifiedValue`, because it felt out of context elsewhere. For example, I usually expect use declarations at the top of the file to be external.
glen
commented
Ah, the fact that Ah, the fact that `use` is sort of "lost" in the midst of the file suggests that SpecifiedValue should be in its own source file and imported here, so the relevant `use` is up where one would generally look. assembly.rs is getting pretty huge anyway... What are your thought about that factoring?
Vectornaut
commented
Specified values and assemblies do seem like pretty independent concepts, so putting Specified values and assemblies do seem like pretty independent concepts, so putting `SpecifiedValue` in its own module might be reasonable.
glen
commented
Are you doing so in this PR? That's my (relatively mild) recommendation. Are you doing so in this PR? That's my (relatively mild) recommendation.
Vectornaut
commented
Yes, I'll do this—I was just waiting for a go-ahead from you. I should've mentioned that in the last comment. Yes, I'll do this—I was just waiting for a go-ahead from you. I should've mentioned that in the last comment.
Vectornaut
commented
Done ( Done (309b088).
|
||||
let row = elts[subjects.0].column_index.unwrap();
|
||||
let col = elts[subjects.1].column_index.unwrap();
|
||||
gram_to_be.push_sym(row, col, *value);
|
||||
gram_to_be.push_sym(row, col, val);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
|
|
@ -14,7 +14,7 @@ use crate::{
|
|||
Regulator,
|
||||
RegulatorKey
|
||||
},
|
||||
specified::{SpecifiedValue, SpecifiedValue::{Absent, Present}}
|
||||
specified::SpecifiedValue
|
||||
};
|
||||
|
||||
// an editable view of a regulator
|
||||
|
@ -22,7 +22,7 @@ use crate::{
|
|||
fn RegulatorInput(regulator: Regulator) -> View {
|
||||
let valid = create_signal(true);
|
||||
let value = create_signal(
|
||||
regulator.set_point.with_untracked(|set_pt| set_pt.spec())
|
||||
regulator.set_point.with_untracked(|set_pt| set_pt.spec.clone())
|
||||
);
|
||||
|
||||
// this closure resets the input value to the regulator's set point
|
||||
|
@ -30,7 +30,7 @@ fn RegulatorInput(regulator: Regulator) -> View {
|
|||
let reset_value = move || {
|
||||
batch(|| {
|
||||
valid.set(true);
|
||||
value.set(regulator.set_point.with(|set_pt| set_pt.spec()));
|
||||
value.set(regulator.set_point.with(|set_pt| set_pt.spec.clone()));
|
||||
})
|
||||
};
|
||||
|
||||
|
@ -44,9 +44,10 @@ fn RegulatorInput(regulator: Regulator) -> View {
|
|||
class=move || {
|
||||
if valid.get() {
|
||||
regulator.set_point.with(|set_pt| {
|
||||
match set_pt {
|
||||
Absent => "regulator-input",
|
||||
Present { .. } => "regulator-input constraint"
|
||||
if set_pt.is_present() {
|
||||
"regulator-input constraint"
|
||||
} else {
|
||||
"regulator-input"
|
||||
}
|
||||
})
|
||||
} else {
|
||||
|
|
|
@ -1,48 +1,43 @@
|
|||
use std::num::ParseFloatError;
|
||||
|
||||
// to construct a `SpecifiedValue` that might be `Present`, use the associated
|
||||
// function `try_from`. this ensures that `spec` is always a valid specification
|
||||
// of `value` according to the format discussed at the implementation of
|
||||
// `TryFrom<String>`
|
||||
pub enum SpecifiedValue {
|
||||
Absent,
|
||||
Present {
|
||||
spec: String,
|
||||
value: f64
|
||||
}
|
||||
// a real number described by a specification string. since the structure is
|
||||
// read-only, we can guarantee that `spec` always specifies `value` in the
|
||||
// following format
|
||||
// ┌──────────────────────────────────────────────────────┬───────────┐
|
||||
// │ `spec` │ `value` │
|
||||
// ┝━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┿━━━━━━━━━━━┥
|
||||
// │ a string that parses to the floating-point value `x` │ `Some(x)` │
|
||||
// ├──────────────────────────────────────────────────────┼───────────┤
|
||||
// │ the empty string │ `None` │
|
||||
// └──────────────────────────────────────────────────────┴───────────┘
|
||||
#[readonly::make]
|
||||
pub struct SpecifiedValue {
|
||||
pub spec: String,
|
||||
pub value: Option<f64>
|
||||
}
|
||||
|
||||
use SpecifiedValue::{Absent, Present};
|
||||
|
||||
impl SpecifiedValue {
|
||||
// get the specification for this value. the associated function `try_from`
|
||||
// is essentially a left inverse of this method:
|
||||
//
|
||||
// SpecifiedValue::try_from(x.spec()) == Ok(x)
|
||||
//
|
||||
pub fn spec(&self) -> String {
|
||||
match self {
|
||||
Absent => String::new(),
|
||||
Present { spec, .. } => spec.clone()
|
||||
}
|
||||
pub fn from_empty_spec() -> SpecifiedValue {
|
||||
SpecifiedValue { spec: String::new(), value: None }
|
||||
glen marked this conversation as resolved
glen
commented
1) Would "none" or "empty" or "absent" be a simpler name for this function, and perhaps also closer to the actual semantics? It seems to me that what one wants is a a SpecifiedValue that is _not_ present, rather than being tied to the representation as an empty string.
2) Do you feel some non-DRYness between this and `try_from("")`? Wouldn't it be cleaner reuse either to implement this as the appropriate unwrapping of try_from(String::new()), or else in the try_from case implement the empty string case by returning the result of this function? We really only want the format of the canonical absent SpecifiedValue to be contained in one place, it seems to me.
Vectornaut
commented
1. The meaning I intended was "the value specified by the empty string," rather than "the canonically specified absent value." I think this works fine in both of the places where the function is used as of commit 08ec838. If we eventually have more than one way to specify an absent value, it might make sense to switch to a "canonically specified absent value" function, but ensuring consistency with `try_from` would get more complicated. For that reason, I'd want to make that switch in the context of actually wanting it, rather than guessing at what we might want in the future.
2. Yes, `from_empty_spec` did have potential for inconsistency with `try_from`. Commit 08ec838 implements your suggestion of having `try_from` call `from_empty_spec` when the specification is empty.
glen
commented
All right. Having the syntactic "trivial" generator of an unspecified SpecifiedValue (i.e. the one specified by an empty string) rather than a semantic one like "the canonical absent SpecifiedValue" seems a slightly odd emphasis to me (on syntax over semantics), but as you say, at the moment they are more or less equivalent because in fact there is only one syntactically correct specification of an absent SpecifiedValue. Hence resolving. All right. Having the _syntactic_ "trivial" generator of an unspecified SpecifiedValue (i.e. the one specified by an empty string) rather than a _semantic_ one like "the canonical absent SpecifiedValue" seems a slightly odd emphasis to me (on syntax over semantics), but as you say, at the moment they are more or less equivalent because in fact there is only one syntactically correct specification of an absent SpecifiedValue. Hence resolving.
|
||||
}
|
||||
|
||||
pub fn is_present(&self) -> bool {
|
||||
matches!(self.value, Some(_))
|
||||
}
|
||||
}
|
||||
|
||||
// we can try to turn a specification string into a `SpecifiedValue`. if the
|
||||
// specification is empty, the `SpecifiedValue` is `Absent`. if the
|
||||
// specification parses to a floating-point value `x`, the `SpecifiedValue` is
|
||||
// `Present`, with a `value` of `x`, and the specification is stored in `spec`.
|
||||
// these are currently the only valid specifications; any other produces an
|
||||
// error
|
||||
// a `SpecifiedValue` can be constructed from a specification string, formatted
|
||||
// as described in the comment on the structure definition. the result is `Ok`
|
||||
// if the specification is properly formatted, and `Error` if not
|
||||
impl TryFrom<String> for SpecifiedValue {
|
||||
type Error = ParseFloatError;
|
||||
|
||||
fn try_from(spec: String) -> Result<Self, Self::Error> {
|
||||
if spec.is_empty() {
|
||||
Ok(Absent)
|
||||
Ok(SpecifiedValue { spec: spec, value: None })
|
||||
} else {
|
||||
spec.parse::<f64>().map(
|
||||
|value| Present { spec: spec, value: value }
|
||||
|value| SpecifiedValue { spec: spec, value: Some(value) }
|
||||
)
|
||||
}
|
||||
}
|
||||
|
|
Should this really be
pub
? Where would a client of Assembly get a "free-floating" regulator to insert? Seems, at least for the moment, like a private helper forinsert_new_regulator
, and indeed, at the moment it is only called from there.Good point! I've made
insert_regulator
private for now (c368a38
).