feos-core 0.9.5

Core traits and functionalities for the `feos` project.
Documentation
use super::{DensityInitialization, State};
use crate::Total;
use crate::equation_of_state::Residual;
use crate::errors::FeosResult;
use nalgebra::DVector;
use quantity::*;

/// A simple tool to construct [State]s with arbitrary input parameters.
///
/// # Examples
/// ```
/// # use feos_core::{FeosResult, StateBuilder};
/// # use feos_core::cubic::{PengRobinson, PengRobinsonParameters};
/// # use quantity::*;
/// # use nalgebra::dvector;
/// # use approx::assert_relative_eq;
/// # use typenum::P3;
/// # fn main() -> FeosResult<()> {
/// // Create a state for given T,V,N
/// let eos = &PengRobinson::new(PengRobinsonParameters::new_simple(&[369.8], &[41.9 * 1e5], &[0.15], &[15.0])?);
/// let state = StateBuilder::new(&eos)
///                 .temperature(300.0 * KELVIN)
///                 .volume(12.5 * METER.powi::<P3>())
///                 .moles(&(dvector![2.5] * MOL))
///                 .build()?;
/// assert_eq!(state.density, 0.2 * MOL / METER.powi::<P3>());
///
/// // For a pure component, the composition does not need to be specified.
/// let eos = &PengRobinson::new(PengRobinsonParameters::new_simple(&[369.8], &[41.9 * 1e5], &[0.15], &[15.0])?);
/// let state = StateBuilder::new(&eos)
///                 .temperature(300.0 * KELVIN)
///                 .volume(12.5 * METER.powi::<P3>())
///                 .total_moles(2.5 * MOL)
///                 .build()?;
/// assert_eq!(state.density, 0.2 * MOL / METER.powi::<P3>());
///
/// // The state can be constructed without providing any extensive property.
/// let eos = &PengRobinson::new(
///     PengRobinsonParameters::new_simple(
///         &[369.8, 305.4],
///         &[41.9 * 1e5, 48.2 * 1e5],
///         &[0.15, 0.10],
///         &[15.0, 30.0]
///     )?
/// );
/// let state = StateBuilder::new(&eos)
///                 .temperature(300.0 * KELVIN)
///                 .partial_density(&(dvector![0.2, 0.6] * MOL / METER.powi::<P3>()))
///                 .build()?;
/// assert_relative_eq!(state.molefracs, dvector![0.25, 0.75]);
/// assert_relative_eq!(state.density, 0.8 * MOL / METER.powi::<P3>());
/// # Ok(())
/// # }
/// ```
#[derive(Clone)]
pub struct StateBuilder<'a, E, const IG: bool> {
    eos: &'a E,
    temperature: Option<Temperature>,
    volume: Option<Volume>,
    density: Option<Density>,
    partial_density: Option<&'a Density<DVector<f64>>>,
    total_moles: Option<Moles>,
    moles: Option<&'a Moles<DVector<f64>>>,
    molefracs: Option<&'a DVector<f64>>,
    pressure: Option<Pressure>,
    molar_enthalpy: Option<MolarEnergy>,
    molar_entropy: Option<MolarEntropy>,
    molar_internal_energy: Option<MolarEnergy>,
    density_initialization: Option<DensityInitialization>,
    initial_temperature: Option<Temperature>,
}

impl<'a, E: Residual> StateBuilder<'a, E, false> {
    /// Create a new `StateBuilder` for the given equation of state.
    pub fn new(eos: &'a E) -> Self {
        StateBuilder {
            eos,
            temperature: None,
            volume: None,
            density: None,
            partial_density: None,
            total_moles: None,
            moles: None,
            molefracs: None,
            pressure: None,
            molar_enthalpy: None,
            molar_entropy: None,
            molar_internal_energy: None,
            density_initialization: None,
            initial_temperature: None,
        }
    }
}

impl<'a, E: Residual, const IG: bool> StateBuilder<'a, E, IG> {
    /// Provide the temperature for the new state.
    pub fn temperature(mut self, temperature: Temperature) -> Self {
        self.temperature = Some(temperature);
        self
    }

    /// Provide the volume for the new state.
    pub fn volume(mut self, volume: Volume) -> Self {
        self.volume = Some(volume);
        self
    }

    /// Provide the density for the new state.
    pub fn density(mut self, density: Density) -> Self {
        self.density = Some(density);
        self
    }

    /// Provide partial densities for the new state.
    pub fn partial_density(mut self, partial_density: &'a Density<DVector<f64>>) -> Self {
        self.partial_density = Some(partial_density);
        self
    }

    /// Provide the total moles for the new state.
    pub fn total_moles(mut self, total_moles: Moles) -> Self {
        self.total_moles = Some(total_moles);
        self
    }

    /// Provide the moles for the new state.
    pub fn moles(mut self, moles: &'a Moles<DVector<f64>>) -> Self {
        self.moles = Some(moles);
        self
    }

    /// Provide the molefracs for the new state.
    pub fn molefracs(mut self, molefracs: &'a DVector<f64>) -> Self {
        self.molefracs = Some(molefracs);
        self
    }

    /// Provide the pressure for the new state.
    pub fn pressure(mut self, pressure: Pressure) -> Self {
        self.pressure = Some(pressure);
        self
    }

    /// Specify a vapor state.
    pub fn vapor(mut self) -> Self {
        self.density_initialization = Some(DensityInitialization::Vapor);
        self
    }

    /// Specify a liquid state.
    pub fn liquid(mut self) -> Self {
        self.density_initialization = Some(DensityInitialization::Liquid);
        self
    }

    /// Provide an initial density used in density iterations.
    pub fn initial_density(mut self, initial_density: Density) -> Self {
        self.density_initialization = Some(DensityInitialization::InitialDensity(initial_density));
        self
    }
}

impl<'a, E: Total, const IG: bool> StateBuilder<'a, E, IG> {
    /// Provide the molar enthalpy for the new state.
    pub fn molar_enthalpy(mut self, molar_enthalpy: MolarEnergy) -> StateBuilder<'a, E, true> {
        self.molar_enthalpy = Some(molar_enthalpy);
        self.convert()
    }

    /// Provide the molar entropy for the new state.
    pub fn molar_entropy(mut self, molar_entropy: MolarEntropy) -> StateBuilder<'a, E, true> {
        self.molar_entropy = Some(molar_entropy);
        self.convert()
    }

    /// Provide the molar internal energy for the new state.
    pub fn molar_internal_energy(
        mut self,
        molar_internal_energy: MolarEnergy,
    ) -> StateBuilder<'a, E, true> {
        self.molar_internal_energy = Some(molar_internal_energy);
        self.convert()
    }

    /// Provide an initial temperature used in the Newton solver.
    pub fn initial_temperature(
        mut self,
        initial_temperature: Temperature,
    ) -> StateBuilder<'a, E, true> {
        self.initial_temperature = Some(initial_temperature);
        self.convert()
    }

    fn convert(self) -> StateBuilder<'a, E, true> {
        StateBuilder {
            eos: self.eos,
            temperature: self.temperature,
            volume: self.volume,
            density: self.density,
            partial_density: self.partial_density,
            total_moles: self.total_moles,
            moles: self.moles,
            molefracs: self.molefracs,
            pressure: self.pressure,
            molar_enthalpy: self.molar_enthalpy,
            molar_entropy: self.molar_entropy,
            molar_internal_energy: self.molar_internal_energy,
            density_initialization: self.density_initialization,
            initial_temperature: self.initial_temperature,
        }
    }
}

impl<E: Residual> StateBuilder<'_, E, false> {
    /// Try to build the state with the given inputs.
    pub fn build(self) -> FeosResult<State<E>> {
        State::new(
            self.eos,
            self.temperature,
            self.volume,
            self.density,
            self.partial_density,
            self.total_moles,
            self.moles,
            self.molefracs,
            self.pressure,
            self.density_initialization,
        )
    }
}

impl<E: Total> StateBuilder<'_, E, true> {
    /// Try to build the state with the given inputs.
    pub fn build(self) -> FeosResult<State<E>> {
        State::new_full(
            self.eos,
            self.temperature,
            self.volume,
            self.density,
            self.partial_density,
            self.total_moles,
            self.moles,
            self.molefracs,
            self.pressure,
            self.molar_enthalpy,
            self.molar_entropy,
            self.molar_internal_energy,
            self.density_initialization,
            self.initial_temperature,
        )
    }
}