-
Notifications
You must be signed in to change notification settings - Fork 21
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Adds a small, non-executable Demo class. This class shows some important features of the framework in a condensed way.
- Loading branch information
David Legg
committed
Dec 19, 2023
1 parent
1b222ac
commit 8f98784
Showing
1 changed file
with
182 additions
and
0 deletions.
There are no files selected for viewing
182 changes: 182 additions & 0 deletions
182
contrib/src/main/java/gov/nasa/jpl/aerie/contrib/streamline/modeling/Demo.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,182 @@ | ||
package gov.nasa.jpl.aerie.contrib.streamline.modeling; | ||
|
||
import gov.nasa.jpl.aerie.contrib.streamline.modeling.black_box.*; | ||
import gov.nasa.jpl.aerie.contrib.streamline.modeling.black_box.monads.UnstructuredResourceApplicative; | ||
import gov.nasa.jpl.aerie.contrib.streamline.modeling.discrete.Discrete; | ||
import gov.nasa.jpl.aerie.contrib.streamline.modeling.discrete.DiscreteResources; | ||
import gov.nasa.jpl.aerie.contrib.streamline.modeling.linear.Linear; | ||
import gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.Polynomial; | ||
import gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.PolynomialEffects; | ||
import gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.PolynomialResources; | ||
import gov.nasa.jpl.aerie.contrib.streamline.unit_aware.UnitAware; | ||
import gov.nasa.jpl.aerie.contrib.streamline.core.Resource; | ||
import gov.nasa.jpl.aerie.contrib.streamline.core.MutableResource; | ||
import gov.nasa.jpl.aerie.merlin.protocol.types.Duration; | ||
import org.apache.commons.lang3.tuple.Pair; | ||
import org.apache.commons.math3.geometry.euclidean.threed.Vector3D; | ||
|
||
import java.util.Optional; | ||
|
||
import static gov.nasa.jpl.aerie.contrib.streamline.core.Resources.currentValue; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.black_box.Approximation.approximate; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.black_box.DifferentiableResources.asDifferentiable; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.black_box.DifferentiableResources.divide; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.black_box.DivergenceEstimators.byBoundingError; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.black_box.IntervalFunctions.byBoundingError; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.black_box.IntervalFunctions.byUniformSampling; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.black_box.SecantApproximation.ErrorEstimates.errorByOptimization; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.black_box.SecantApproximation.ErrorEstimates.errorByQuadraticApproximation; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.black_box.SecantApproximation.secantApproximation; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.black_box.UnstructuredResources.asUnstructured; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.discrete.DiscreteEffects.set; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.discrete.DiscreteEffects.toggle; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.discrete.DiscreteEffects.using; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.discrete.DiscreteResources.assertThat; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.discrete.DiscreteResources.discreteResource; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.discrete.monads.DiscreteResourceMonad.map; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.PolynomialEffects.consume; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.PolynomialResources.asPolynomial; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.PolynomialResources.asUnitAwarePolynomial; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.PolynomialResources.clamp; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.PolynomialResources.constant; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.PolynomialResources.integrate; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.PolynomialResources.lessThan; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.PolynomialResources.lessThan$; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.PolynomialResources.polynomialResource; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.modeling.polynomial.PolynomialResources.unitAware; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.unit_aware.Quantities.quantity; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.unit_aware.StandardUnits.*; | ||
import static gov.nasa.jpl.aerie.merlin.framework.ModelActions.*; | ||
import static gov.nasa.jpl.aerie.contrib.streamline.core.MutableResource.resource; | ||
|
||
public final class Demo { | ||
|
||
// Unit-naive version of a model, to demonstrate some core concepts: | ||
|
||
// Consumable, continuous: | ||
MutableResource<Polynomial> fuel_kg = PolynomialResources.polynomialResource(20.0); | ||
// Non-consumable, discrete: | ||
MutableResource<Discrete<Double>> power_w = discreteResource(120.0); | ||
// Atomic non-consumable: | ||
MutableResource<Discrete<Integer>> rwaControl = DiscreteResources.discreteResource(1); | ||
// Settable / enum state: | ||
MutableResource<Discrete<OnOff>> enumSwitch = DiscreteResources.discreteResource(OnOff.ON); | ||
// Toggle / flag: | ||
MutableResource<Discrete<Boolean>> boolSwitch = DiscreteResources.discreteResource(true); | ||
|
||
// Derived states: | ||
Resource<Discrete<OnOff>> derivedEnumSwitch = map(boolSwitch, b -> b ? OnOff.ON : OnOff.OFF); | ||
Resource<Polynomial> batterySOC_J = integrate(asPolynomial(power_w), 100); | ||
Resource<Discrete<Double>> clampedPower_w = map(power_w, p -> p < 0 ? 0 : p); | ||
Resource<Polynomial> clampedBatterySOC_J = clamp(batterySOC_J, constant(0), constant(100)); | ||
Resource<Discrete<Boolean>> lowPower = lessThan(batterySOC_J, 20); | ||
Resource<Discrete<Boolean>> badness = map( | ||
lowPower, enumSwitch, | ||
(lowPower$, switch$) -> | ||
lowPower$ && switch$ == OnOff.OFF); | ||
|
||
{ | ||
using(power_w, 10, () -> { | ||
using(rwaControl, () -> { | ||
// Consume 5.4 kg of fuel over the next minute, linearly | ||
PolynomialEffects.consumeUniformly(fuel_kg, 5.4, Duration.MINUTE); | ||
// Separately, we could be doing things during that minute. | ||
delay(Duration.MINUTE); | ||
}); | ||
set(enumSwitch, OnOff.OFF); | ||
toggle(boolSwitch); | ||
}); | ||
|
||
set(boolSwitch, false); | ||
} | ||
|
||
|
||
// The exact same model again, but this time made unit-aware throughout. | ||
// States without units have been re-used instead of being re-defined | ||
|
||
// Consumable, continuous: | ||
// CellResource<Polynomial> fuel_kg = polynomialCellResource(20.0); | ||
UnitAware<MutableResource<Polynomial>> fuel = unitAware( | ||
PolynomialResources.polynomialResource(20.0), KILOGRAM); | ||
// Non-consumable, discrete: | ||
UnitAware<MutableResource<Discrete<Double>>> power = DiscreteResources.unitAware( | ||
discreteResource(120.0), WATT); | ||
|
||
UnitAware<Resource<Polynomial>> batterySOC = integrate(asUnitAwarePolynomial(power), quantity(100, JOULE)); | ||
UnitAware<Resource<Discrete<Double>>> clampedPower = DiscreteResources.unitAware(map(power.value(WATT), p -> p < 0 ? 0 : p), WATT); | ||
UnitAware<Resource<Discrete<Double>>> clampedPower_v2 = /* map(power, p -> lessThan(p, quantity(0, WATT)) ? quantity(0, WATT) : p) */ | ||
null; | ||
UnitAware<Resource<Polynomial>> clampedBatterySOC = clamp(batterySOC, constant(quantity(0, JOULE)), constant(quantity(100, JOULE))); | ||
Resource<Discrete<Boolean>> lowPower$ = lessThan$(batterySOC, quantity(20, JOULE)); | ||
|
||
{ | ||
using(power, quantity(10, WATT), () -> { | ||
using(rwaControl, () -> { | ||
// Consume 5.4 kg of fuel over the next minute, linearly | ||
consume(fuel, quantity(5.4, KILOGRAM), Duration.MINUTE); | ||
// Separately, we could be doing things during that minute. | ||
delay(Duration.MINUTE); | ||
}); | ||
set(enumSwitch, OnOff.OFF); | ||
toggle(boolSwitch); | ||
}); | ||
} | ||
|
||
// Example of using unstructured resources + approximation to represent functions that aren't | ||
// easily represented by analytic derivations | ||
Resource<Polynomial> p = PolynomialResources.polynomialResource(1, 2, 3); | ||
Resource<Polynomial> q = PolynomialResources.polynomialResource(6, 5, 4); | ||
Resource<Unstructured<Double>> quotient = UnstructuredResourceApplicative.map(asUnstructured(p), asUnstructured(q), (p$, q$) -> p$ / q$); | ||
Resource<Linear> approxQuotient = approximate(quotient, secantApproximation(IntervalFunctions.<Unstructured<Double>>byBoundingError( | ||
1e-6, Duration.SECOND, Duration.HOUR.times(24), errorByOptimization()))); | ||
|
||
Resource<Unstructured<Pair<Vector3D, Vector3D>>> positionAndVelocity = resource(Unstructured.timeBased(t -> /* some spice call */ null)); | ||
Resource<Discrete<Pair<Vector3D, Vector3D>>> approxPosVel = approximate( | ||
positionAndVelocity, | ||
DiscreteApproximation.<Pair<Vector3D, Vector3D>, Unstructured<Pair<Vector3D, Vector3D>>>discreteApproximation( | ||
byBoundingError( | ||
1e-6, | ||
Duration.SECOND, | ||
Duration.HOUR.times(24), | ||
(u, v) -> Math.max( | ||
u.getLeft().distance(v.getLeft()) / v.getLeft().getNorm(), | ||
u.getRight().distance(v.getRight()) / v.getRight().getNorm())))); | ||
|
||
// Example of the semi-structured "differentiable" resources, and using the additional information to approximate: | ||
Resource<Differentiable> pDiff = asDifferentiable(p); | ||
Resource<Differentiable> qDiff = asDifferentiable(q); | ||
Resource<Differentiable> quotient2 = divide(pDiff, qDiff); | ||
Resource<Linear> approxQuotient2 = approximate( | ||
quotient2, | ||
secantApproximation(byBoundingError( | ||
1e-6, | ||
Duration.SECOND, | ||
Duration.HOUR.times(24), | ||
errorByQuadraticApproximation()))); | ||
|
||
// Another example, pushing a polynomial down to a linear for registering: | ||
Resource<Polynomial> r; | ||
Resource<Linear> approxR = approximate(r, SecantApproximation.<Polynomial>secantApproximation(byUniformSampling(Duration.HOUR))); | ||
|
||
|
||
// Example of a locking state: | ||
|
||
MutableResource<Discrete<Integer>> importantHardware = DiscreteResources.discreteResource(42); | ||
MutableResource<Discrete<Optional<Integer>>> importantHardwareLock = DiscreteResources.discreteResource(Optional.empty()); | ||
Resource<Discrete<Boolean>> importantHardwareLockAssertion = assertThat( | ||
"Important hardware does not change state while locked", | ||
map(importantHardwareLock, importantHardware, (lock, state) -> lock.map(state::equals).orElse(true))); | ||
void lockImportantHardware() { | ||
if (currentValue(importantHardwareLock).isPresent()) { | ||
throw new IllegalStateException("Already locked!"); | ||
} else { | ||
set(importantHardwareLock, Optional.of(currentValue(importantHardware))); | ||
} | ||
} | ||
|
||
void unlockImportantHardware() { | ||
set(importantHardwareLock, Optional.empty()); | ||
} | ||
|
||
public enum OnOff { ON, OFF } | ||
} |