use super::{MixedHamiltonian, MixedLindbladNoiseOperator, OperateOnMixedSystems};
use crate::{OpenSystem, OperateOnDensityMatrix, StruqtureError};
use qoqo_calculator::CalculatorFloat;
use serde::{Deserialize, Serialize};
use std::fmt;
use std::ops;
#[derive(Debug, Clone, PartialEq, Deserialize, Serialize, Default)]
#[cfg_attr(feature = "json_schema", derive(schemars::JsonSchema))]
#[cfg_attr(feature = "json_schema", schemars(deny_unknown_fields))]
pub struct MixedLindbladOpenSystem {
system: MixedHamiltonian,
noise: MixedLindbladNoiseOperator,
}
impl crate::SerializationSupport for MixedLindbladOpenSystem {
fn struqture_type() -> crate::StruqtureType {
crate::StruqtureType::MixedLindbladOpenSystem
}
}
impl OpenSystem<'_> for MixedLindbladOpenSystem {
type System = MixedHamiltonian;
type Noise = MixedLindbladNoiseOperator;
fn noise(&self) -> &Self::Noise {
&self.noise
}
fn system(&self) -> &Self::System {
&self.system
}
fn noise_mut(&mut self) -> &mut Self::Noise {
&mut self.noise
}
fn system_mut(&mut self) -> &mut Self::System {
&mut self.system
}
fn ungroup(self) -> (Self::System, Self::Noise) {
(self.system, self.noise)
}
fn group(system: Self::System, noise: Self::Noise) -> Result<Self, StruqtureError> {
if system.n_spins != noise.n_spins
|| system.n_bosons != noise.n_bosons
|| system.n_fermions != noise.n_fermions
{
return Err(StruqtureError::MismatchedNumberSubsystems {
target_number_spin_subsystems: system.n_spins,
target_number_boson_subsystems: system.n_bosons,
target_number_fermion_subsystems: system.n_fermions,
actual_number_spin_subsystems: noise.n_spins,
actual_number_boson_subsystems: noise.n_bosons,
actual_number_fermion_subsystems: noise.n_fermions,
});
}
Ok(Self { system, noise })
}
fn empty_clone(&self) -> Self {
Self::group(self.system.empty_clone(None), self.noise.empty_clone(None)).expect(
"Internal error: Number of spins in system and noise unexpectedly does not match.",
)
}
}
impl OperateOnMixedSystems<'_> for MixedLindbladOpenSystem {
fn current_number_spins(&self) -> Vec<usize> {
self.system
.current_number_spins()
.iter()
.zip(self.noise.current_number_spins().iter())
.map(|(s, n)| *(s.max(n)))
.collect()
}
fn current_number_bosonic_modes(&self) -> Vec<usize> {
self.system
.current_number_bosonic_modes()
.iter()
.zip(self.noise.current_number_bosonic_modes().iter())
.map(|(s, n)| *(s.max(n)))
.collect()
}
fn current_number_fermionic_modes(&self) -> Vec<usize> {
self.system
.current_number_fermionic_modes()
.iter()
.zip(self.noise.current_number_fermionic_modes().iter())
.map(|(s, n)| *(s.max(n)))
.collect()
}
}
impl MixedLindbladOpenSystem {
pub fn new(number_spins: usize, number_bosons: usize, number_fermions: usize) -> Self {
MixedLindbladOpenSystem {
system: MixedHamiltonian::new(number_spins, number_bosons, number_fermions),
noise: MixedLindbladNoiseOperator::new(number_spins, number_bosons, number_fermions),
}
}
#[cfg(feature = "struqture_1_export")]
pub fn to_struqture_1(
&self,
) -> Result<struqture_1::mixed_systems::MixedLindbladOpenSystem, StruqtureError> {
let new_system = self.system().to_struqture_1()?;
let new_noise = self.noise().to_struqture_1()?;
struqture_1::OpenSystem::group(new_system, new_noise).map_err(
|err| StruqtureError::GenericError { msg:
format!("Could not convert struqture 2.x MixedLindbladOpenSystem to 1.x MixedLindbladOpenSystem, group function failed: {err:?}.")
}
)
}
#[cfg(feature = "struqture_1_import")]
pub fn from_struqture_1(
value: &struqture_1::mixed_systems::MixedLindbladOpenSystem,
) -> Result<Self, StruqtureError> {
let (system_one, noise_one) = struqture_1::OpenSystem::ungroup(value.clone());
let new_system = MixedHamiltonian::from_struqture_1(&system_one)?;
let new_noise = MixedLindbladNoiseOperator::from_struqture_1(&noise_one)?;
Self::group(new_system, new_noise)
}
}
impl ops::Neg for MixedLindbladOpenSystem {
type Output = Self;
fn neg(self) -> Self {
let (self_sys, self_noise) = self.ungroup();
Self {
system: self_sys.neg(),
noise: self_noise.neg(),
}
}
}
impl ops::Add<MixedLindbladOpenSystem> for MixedLindbladOpenSystem {
type Output = Result<Self, StruqtureError>;
fn add(self, other: MixedLindbladOpenSystem) -> Self::Output {
let (self_sys, self_noise) = self.ungroup();
let (other_sys, other_noise) = other.ungroup();
Self::group((self_sys + other_sys)?, self_noise + other_noise)
}
}
impl ops::Sub<MixedLindbladOpenSystem> for MixedLindbladOpenSystem {
type Output = Result<Self, StruqtureError>;
fn sub(self, other: MixedLindbladOpenSystem) -> Self::Output {
let (self_sys, self_noise) = self.ungroup();
let (other_sys, other_noise) = other.ungroup();
Self::group((self_sys - other_sys)?, self_noise - other_noise)
}
}
impl ops::Mul<CalculatorFloat> for MixedLindbladOpenSystem {
type Output = Self;
fn mul(self, rhs: CalculatorFloat) -> Self::Output {
Self {
system: self.system * rhs.clone(),
noise: self.noise * rhs,
}
}
}
impl fmt::Display for MixedLindbladOpenSystem {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let mut output = "MixedLindbladOpenSystem{\n".to_string();
output.push_str("System: {\n");
output.push_str(format!("{}", self.system()).as_str());
output.push_str("}\n");
output.push_str("Noise: {\n");
output.push_str(format!("{}", self.noise()).as_str());
output.push_str("}\n");
output.push('}');
write!(f, "{output}")
}
}