conspire/math/integrate/
mod.rs#[cfg(test)]
mod test;
mod backward_euler;
mod bogacki_shampine;
mod dormand_prince;
mod verner_8;
mod verner_9;
pub use backward_euler::BackwardEuler;
pub use bogacki_shampine::BogackiShampine;
pub use dormand_prince::DormandPrince;
pub use verner_8::Verner8;
pub use verner_9::Verner9;
pub type Ode1be = BackwardEuler;
pub type Ode23 = BogackiShampine;
pub type Ode45 = DormandPrince;
pub type Ode78 = Verner8;
pub type Ode89 = Verner9;
use super::{
interpolate::InterpolateSolution, Tensor, TensorArray, TensorRank0, TensorVec, Vector,
};
use crate::defeat_message;
use std::{
fmt,
ops::{Div, Mul, Sub},
};
pub trait OdeSolver<Y, U>
where
Self: fmt::Debug,
Y: Tensor,
U: TensorVec<Item = Y>,
{
}
impl<A, Y, U> OdeSolver<Y, U> for A
where
A: std::fmt::Debug,
Y: Tensor,
U: TensorVec<Item = Y>,
{
}
pub trait Explicit<Y, U>: OdeSolver<Y, U>
where
Self: InterpolateSolution<Y, U>,
Y: Tensor + TensorArray,
for<'a> &'a Y: Mul<TensorRank0, Output = Y> + Sub<&'a Y, Output = Y>,
U: TensorVec<Item = Y>,
{
fn integrate(
&self,
function: impl Fn(&TensorRank0, &Y) -> Y,
time: &[TensorRank0],
initial_condition: Y,
) -> Result<(Vector, U), IntegrationError>;
}
pub trait Implicit<Y, J, U>: OdeSolver<Y, U>
where
Self: InterpolateSolution<Y, U>,
Y: Tensor + TensorArray + Div<J, Output = Y>,
for<'a> &'a Y: Mul<TensorRank0, Output = Y> + Sub<&'a Y, Output = Y>,
J: Tensor + TensorArray,
U: TensorVec<Item = Y>,
{
fn integrate(
&self,
function: impl Fn(&TensorRank0, &Y) -> Y,
jacobian: impl Fn(&TensorRank0, &Y) -> J,
time: &[TensorRank0],
initial_condition: Y,
) -> Result<(Vector, U), IntegrationError>;
}
pub enum IntegrationError {
InitialTimeNotLessThanFinalTime,
LengthTimeLessThanTwo,
}
impl From<&str> for IntegrationError {
fn from(string: &str) -> Self {
todo!("{}", string)
}
}
impl fmt::Debug for IntegrationError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let error = match self {
Self::InitialTimeNotLessThanFinalTime => {
"\x1b[1;91mThe initial time must precede the final time.".to_string()
}
Self::LengthTimeLessThanTwo => {
"\x1b[1;91mThe time must contain at least two entries.".to_string()
}
};
write!(f, "\n{}\n\x1b[0;2;31m{}\x1b[0m\n", error, defeat_message())
}
}
impl fmt::Display for IntegrationError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let error = match self {
Self::InitialTimeNotLessThanFinalTime => {
"\x1b[1;91mThe initial time must precede the final time.".to_string()
}
Self::LengthTimeLessThanTwo => {
"\x1b[1;91mThe time must contain at least two entries.".to_string()
}
};
write!(f, "{}\x1b[0m", error)
}
}