#[cfg(test)]
mod test;
use crate::math::{Tensor, TensorRank0, TensorRank1, TensorRank2, tensor::list::TensorList};
use std::ops::Mul;
#[cfg(test)]
use crate::math::tensor::test::ErrorTensor;
pub type TensorRank1List<const D: usize, const I: usize, const N: usize> =
TensorList<TensorRank1<D, I>, N>;
impl<const D: usize, const I: usize, const N: usize> From<[[TensorRank0; D]; N]>
for TensorRank1List<D, I, N>
{
fn from(array: [[TensorRank0; D]; N]) -> Self {
array.into_iter().map(|entry| entry.into()).collect()
}
}
impl<const D: usize, const N: usize> From<TensorRank1List<D, 9, N>> for TensorRank1List<D, 0, N> {
fn from(tensor_rank_1_list: TensorRank1List<D, 9, N>) -> Self {
tensor_rank_1_list
.into_iter()
.map(|entry| entry.into())
.collect()
}
}
impl<const D: usize, const N: usize> From<TensorRank1List<D, 0, N>> for TensorRank1List<D, 1, N> {
fn from(tensor_rank_1_list: TensorRank1List<D, 0, N>) -> Self {
tensor_rank_1_list
.into_iter()
.map(|entry| entry.into())
.collect()
}
}
impl<const D: usize, const I: usize, const J: usize, const W: usize> Mul<TensorRank1List<D, J, W>>
for TensorRank1List<D, I, W>
{
type Output = TensorRank2<D, I, J>;
fn mul(self, tensor_rank_1_list: TensorRank1List<D, J, W>) -> Self::Output {
self.into_iter()
.zip(tensor_rank_1_list)
.map(|(self_entry, entry)| Self::Output::from((self_entry, entry)))
.sum()
}
}
impl<const D: usize, const I: usize, const J: usize, const W: usize> Mul<&TensorRank1List<D, J, W>>
for TensorRank1List<D, I, W>
{
type Output = TensorRank2<D, I, J>;
fn mul(self, tensor_rank_1_list: &TensorRank1List<D, J, W>) -> Self::Output {
self.into_iter()
.zip(tensor_rank_1_list.iter())
.map(|(self_entry, entry)| Self::Output::from((self_entry, entry)))
.sum()
}
}
impl<const D: usize, const I: usize, const J: usize, const W: usize> Mul<TensorRank1List<D, J, W>>
for &TensorRank1List<D, I, W>
{
type Output = TensorRank2<D, I, J>;
fn mul(self, tensor_rank_1_list: TensorRank1List<D, J, W>) -> Self::Output {
self.iter()
.zip(tensor_rank_1_list)
.map(|(self_entry, entry)| Self::Output::from((self_entry, entry)))
.sum()
}
}
impl<const D: usize, const I: usize, const J: usize, const W: usize> Mul<&TensorRank1List<D, J, W>>
for &TensorRank1List<D, I, W>
{
type Output = TensorRank2<D, I, J>;
fn mul(self, tensor_rank_1_list: &TensorRank1List<D, J, W>) -> Self::Output {
self.iter()
.zip(tensor_rank_1_list.iter())
.map(|(self_entry, entry)| Self::Output::from((self_entry, entry)))
.sum()
}
}
#[cfg(test)]
impl<const D: usize, const I: usize, const W: usize> ErrorTensor for TensorRank1List<D, I, W> {
fn error_fd(&self, comparator: &Self, epsilon: TensorRank0) -> Option<(bool, usize)> {
let error_count = self
.iter()
.zip(comparator.iter())
.map(|(entry, comparator_entry)| {
entry
.iter()
.zip(comparator_entry.iter())
.filter(|&(&entry_i, &comparator_entry_i)| {
(entry_i / comparator_entry_i - 1.0).abs() >= epsilon
&& (entry_i.abs() >= epsilon || comparator_entry_i.abs() >= epsilon)
})
.count()
})
.sum();
if error_count > 0 {
let auxiliary = self
.iter()
.zip(comparator.iter())
.map(|(entry, comparator_entry)| {
entry
.iter()
.zip(comparator_entry.iter())
.filter(|&(&entry_i, &comparator_entry_i)| {
(entry_i / comparator_entry_i - 1.0).abs() >= epsilon
&& (entry_i - comparator_entry_i).abs() >= epsilon
&& (entry_i.abs() >= epsilon || comparator_entry_i.abs() >= epsilon)
})
.count()
})
.sum::<usize>()
> 0;
Some((auxiliary, error_count))
} else {
None
}
}
}