Generalize constraints to observables #48

Merged
glen merged 25 commits from Vectornaut/dyna3:observables_on_main into main 2025-03-10 23:43:25 +00:00
4 changed files with 65 additions and 65 deletions
Showing only changes of commit 309b0881df - Show all commits

View file

@ -1,15 +1,14 @@
use nalgebra::{DMatrix, DVector, DVectorView, Vector3};
use rustc_hash::FxHashMap;
use slab::Slab;
use std::{
collections::BTreeSet,
num::ParseFloatError,
sync::atomic::{AtomicU64, Ordering}
};
use std::{collections::BTreeSet, sync::atomic::{AtomicU64, Ordering}};
use sycamore::prelude::*;
use web_sys::{console, wasm_bindgen::JsValue}; /* DEBUG */
use crate::engine::{Q, local_unif_to_std, realize_gram, ConfigSubspace, PartialMatrix};
use crate::{
engine::{Q, local_unif_to_std, realize_gram, ConfigSubspace, PartialMatrix},
specified::{SpecifiedValue, SpecifiedValue::{Absent, Present}}
};
// the types of the keys we use to access an assembly's elements and regulators
pub type ElementKey = usize;
@ -118,61 +117,6 @@ impl Element {
}
}
// 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
}
}
use SpecifiedValue::*;
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()
}
}
fn is_present(&self) -> bool {
match self {
Absent => false,
Present { .. } => true
}
}
}
// 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
impl TryFrom<String> for SpecifiedValue {
type Error = ParseFloatError;
fn try_from(spec: String) -> Result<Self, Self::Error> {
if spec.is_empty() {
Ok(Absent)
} else {
spec.parse::<f64>().map(
|value| Present { spec: spec, value: value }
)
}
}
}
#[derive(Clone, Copy)]
pub struct Regulator {
pub subjects: (ElementKey, ElementKey),

View file

@ -3,6 +3,7 @@ mod assembly;
mod display;
mod engine;
mod outline;
mod specified;
use rustc_hash::FxHashSet;
use sycamore::prelude::*;

View file

@ -12,10 +12,9 @@ use crate::{
assembly::{
ElementKey,
Regulator,
RegulatorKey,
SpecifiedValue,
SpecifiedValue::*
}
RegulatorKey
},
specified::{SpecifiedValue, SpecifiedValue::{Absent, Present}}
};
// an editable view of a regulator

View file

@ -0,0 +1,56 @@
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
}
}
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)
glen marked this conversation as resolved
Review
  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.
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.
Review
  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.
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.
Review

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 spec(&self) -> String {
match self {
Absent => String::new(),
Present { spec, .. } => spec.clone()
}
}
pub fn is_present(&self) -> bool {
match self {
Absent => false,
Present { .. } => true
}
}
}
// 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
impl TryFrom<String> for SpecifiedValue {
type Error = ParseFloatError;
fn try_from(spec: String) -> Result<Self, Self::Error> {
if spec.is_empty() {
Ok(Absent)
} else {
spec.parse::<f64>().map(
|value| Present { spec: spec, value: value }
)
}
}
}