use super::{HermitianOperateOnSpins, OperateOnSpins, SpinSystem};
use crate::fermions::FermionHamiltonianSystem;
use crate::mappings::JordanWignerSpinToFermion;
use crate::spins::{
PauliProduct, SpinHamiltonian, ToSparseMatrixOperator, ToSparseMatrixSuperOperator,
};
use crate::{CooSparseMatrix, OperateOnDensityMatrix, OperateOnState, SpinIndex, StruqtureError};
#[cfg(feature = "indexed_map_iterators")]
use indexmap::map::{Iter, Keys, Values};
use num_complex::Complex64;
use qoqo_calculator::{CalculatorComplex, CalculatorFloat};
use serde::{Deserialize, Serialize};
#[cfg(not(feature = "indexed_map_iterators"))]
use std::collections::hash_map::{Iter, Keys, Values};
use std::fmt::{self, Write};
use std::iter::{FromIterator, IntoIterator};
use std::ops;
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize, Default)]
#[cfg_attr(feature = "json_schema", derive(schemars::JsonSchema))]
pub struct SpinHamiltonianSystem {
pub(crate) number_spins: Option<usize>,
pub(crate) hamiltonian: SpinHamiltonian,
}
impl crate::MinSupportedVersion for SpinHamiltonianSystem {}
impl<'a> OperateOnDensityMatrix<'a> for SpinHamiltonianSystem {
type Index = PauliProduct;
type Value = CalculatorFloat;
type IteratorType = Iter<'a, Self::Index, Self::Value>;
type KeyIteratorType = Keys<'a, Self::Index, Self::Value>;
type ValueIteratorType = Values<'a, Self::Index, Self::Value>;
fn get(&self, key: &Self::Index) -> &Self::Value {
self.hamiltonian.get(key)
}
fn iter(&'a self) -> Self::IteratorType {
self.hamiltonian.iter()
}
fn keys(&'a self) -> Self::KeyIteratorType {
self.hamiltonian.keys()
}
fn values(&'a self) -> Self::ValueIteratorType {
self.hamiltonian.values()
}
fn remove(&mut self, key: &Self::Index) -> Option<Self::Value> {
self.hamiltonian.remove(key)
}
fn empty_clone(&self, capacity: Option<usize>) -> Self {
match capacity {
Some(cap) => Self {
number_spins: self.number_spins,
hamiltonian: SpinHamiltonian::with_capacity(cap),
},
None => Self {
number_spins: self.number_spins,
hamiltonian: SpinHamiltonian::new(),
},
}
}
fn set(
&mut self,
key: Self::Index,
value: Self::Value,
) -> Result<Option<Self::Value>, StruqtureError> {
match self.number_spins {
Some(x) => {
if key.current_number_spins() <= x {
self.hamiltonian.set(key, value)
} else {
Err(StruqtureError::NumberSpinsExceeded)
}
}
None => self.hamiltonian.set(key, value),
}
}
fn add_operator_product(
&mut self,
key: Self::Index,
value: Self::Value,
) -> Result<(), StruqtureError> {
match self.number_spins {
Some(x) => {
if key.current_number_spins() <= x {
self.hamiltonian.add_operator_product(key, value)
} else {
Err(StruqtureError::NumberSpinsExceeded)
}
}
None => self.hamiltonian.add_operator_product(key, value),
}
}
}
impl OperateOnState<'_> for SpinHamiltonianSystem {
fn hermitian_conjugate(&self) -> Self {
self.clone()
}
}
impl OperateOnSpins<'_> for SpinHamiltonianSystem {
fn number_spins(&self) -> usize {
match self.number_spins {
Some(spins) => spins,
None => self.hamiltonian.current_number_spins(),
}
}
fn current_number_spins(&self) -> usize {
self.hamiltonian.current_number_spins()
}
}
impl HermitianOperateOnSpins<'_> for SpinHamiltonianSystem {}
impl ToSparseMatrixOperator<'_> for SpinHamiltonianSystem {}
impl<'a> ToSparseMatrixSuperOperator<'a> for SpinHamiltonianSystem {
fn sparse_matrix_superoperator_entries_on_row(
&'a self,
row: usize,
number_spins: usize,
) -> Result<std::collections::HashMap<usize, Complex64>, StruqtureError> {
<Self as ToSparseMatrixOperator>::sparse_matrix_superoperator_entries_on_row(
self,
row,
number_spins,
)
}
fn unitary_sparse_matrix_coo(&'a self) -> Result<CooSparseMatrix, StruqtureError> {
self.sparse_matrix_coo(self.number_spins)
}
fn sparse_lindblad_entries(
&'a self,
) -> Result<Vec<(CooSparseMatrix, CooSparseMatrix, Complex64)>, StruqtureError> {
let rate = Complex64::default();
let left: CooSparseMatrix = (vec![], (vec![], vec![]));
let right: CooSparseMatrix = (vec![], (vec![], vec![]));
Ok(vec![(left, right, rate)])
}
}
impl SpinHamiltonianSystem {
pub fn new(number_spins: Option<usize>) -> Self {
SpinHamiltonianSystem {
number_spins,
hamiltonian: SpinHamiltonian::new(),
}
}
pub fn with_capacity(number_spins: Option<usize>, capacity: usize) -> Self {
SpinHamiltonianSystem {
number_spins,
hamiltonian: SpinHamiltonian::with_capacity(capacity),
}
}
#[deprecated(note = "Use the hamiltonian() method instead.")]
pub fn operator(&self) -> &SpinHamiltonian {
&self.hamiltonian
}
pub fn hamiltonian(&self) -> &SpinHamiltonian {
&self.hamiltonian
}
pub fn from_hamiltonian(
hamiltonian: SpinHamiltonian,
number_spins: Option<usize>,
) -> Result<Self, StruqtureError> {
match number_spins {
Some(x) => {
if hamiltonian.current_number_spins() <= x {
Ok(SpinHamiltonianSystem {
number_spins: Some(x),
hamiltonian,
})
} else {
Err(StruqtureError::NumberSpinsExceeded)
}
}
None => Ok(SpinHamiltonianSystem {
number_spins: None,
hamiltonian,
}),
}
}
pub fn separate_into_n_terms(
&self,
number_spins: usize,
) -> Result<(Self, Self), StruqtureError> {
let mut separated = Self::default();
let mut remainder = Self::default();
for (prod, val) in self.iter() {
if prod.len() == number_spins {
separated.add_operator_product(prod.clone(), val.clone())?;
} else {
remainder.add_operator_product(prod.clone(), val.clone())?;
}
}
Ok((separated, remainder))
}
}
impl ops::Neg for SpinHamiltonianSystem {
type Output = Self;
fn neg(mut self) -> Self {
self.hamiltonian = self.hamiltonian.neg();
self
}
}
impl<T, V> ops::Add<T> for SpinHamiltonianSystem
where
T: IntoIterator<Item = (PauliProduct, V)>,
V: Into<CalculatorFloat>,
{
type Output = Result<Self, StruqtureError>;
fn add(mut self, other: T) -> Self::Output {
for (key, value) in other.into_iter() {
self.add_operator_product(key.clone(), Into::<CalculatorFloat>::into(value))?;
}
Ok(self)
}
}
impl<T, V> ops::Sub<T> for SpinHamiltonianSystem
where
T: IntoIterator<Item = (PauliProduct, V)>,
V: Into<CalculatorFloat>,
{
type Output = Result<Self, StruqtureError>;
fn sub(mut self, other: T) -> Self::Output {
for (key, value) in other.into_iter() {
self.add_operator_product(key.clone(), Into::<CalculatorFloat>::into(value) * -1.0)?;
}
Ok(self)
}
}
impl ops::Mul<CalculatorFloat> for SpinHamiltonianSystem {
type Output = Self;
fn mul(mut self, other: CalculatorFloat) -> Self {
self.hamiltonian = self.hamiltonian * other;
self
}
}
impl ops::Mul<CalculatorComplex> for SpinHamiltonianSystem {
type Output = SpinSystem;
fn mul(self, other: CalculatorComplex) -> Self::Output {
let mut system = SpinSystem::new(self.number_spins);
system.operator = self.hamiltonian * other;
system
}
}
impl ops::Mul<SpinHamiltonianSystem> for SpinHamiltonianSystem {
type Output = Result<SpinSystem, StruqtureError>;
fn mul(self, other: SpinHamiltonianSystem) -> Self::Output {
Ok(SpinSystem {
number_spins: Some(self.number_spins().max(other.number_spins())),
operator: self.hamiltonian * other.hamiltonian,
})
}
}
impl IntoIterator for SpinHamiltonianSystem {
type Item = (PauliProduct, CalculatorFloat);
#[cfg(not(feature = "indexed_map_iterators"))]
type IntoIter = std::collections::hash_map::IntoIter<PauliProduct, CalculatorFloat>;
#[cfg(feature = "indexed_map_iterators")]
type IntoIter = indexmap::map::IntoIter<PauliProduct, CalculatorFloat>;
fn into_iter(self) -> Self::IntoIter {
self.hamiltonian.into_iter()
}
}
impl<'a> IntoIterator for &'a SpinHamiltonianSystem {
type Item = (&'a PauliProduct, &'a CalculatorFloat);
type IntoIter = Iter<'a, PauliProduct, CalculatorFloat>;
fn into_iter(self) -> Self::IntoIter {
self.hamiltonian.iter()
}
}
impl FromIterator<(PauliProduct, CalculatorFloat)> for SpinHamiltonianSystem {
fn from_iter<I: IntoIterator<Item = (PauliProduct, CalculatorFloat)>>(iter: I) -> Self {
let mut so = SpinHamiltonianSystem::new(None);
for (pp, cc) in iter {
so.add_operator_product(pp.clone(), cc.clone())
.expect("Internal error in add_operator_product");
}
so
}
}
impl Extend<(PauliProduct, CalculatorFloat)> for SpinHamiltonianSystem {
fn extend<I: IntoIterator<Item = (PauliProduct, CalculatorFloat)>>(&mut self, iter: I) {
for (pp, cc) in iter {
self.add_operator_product(pp, cc)
.expect("Internal error in add_operator_product");
}
}
}
impl fmt::Display for SpinHamiltonianSystem {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let mut output = format!("SpinHamiltonianSystem({}){{\n", self.number_spins());
for (key, val) in self.iter() {
writeln!(output, "{key}: {val},")?;
}
output.push('}');
write!(f, "{output}")
}
}
impl JordanWignerSpinToFermion for SpinHamiltonianSystem {
type Output = FermionHamiltonianSystem;
fn jordan_wigner(&self) -> Self::Output {
FermionHamiltonianSystem::from_hamiltonian(
self.hamiltonian().jordan_wigner(),
self.number_spins,
)
.expect("Internal bug in jordan_wigner() for SpinHamiltonian. The number of modes in the resulting fermionic Hamiltonian should equal the number of spins of the spin Hamiltonian.")
}
}