use crate::{CausalTensor, Tensor};
use deep_causality_haft::{Applicative, BoundedComonad, Foldable, Functor, HKT, Monad};
use deep_causality_num::Zero;
pub struct CausalTensorWitness;
impl HKT for CausalTensorWitness {
type Type<T> = CausalTensor<T>;
}
impl Applicative<CausalTensorWitness> for CausalTensorWitness {
fn pure<T>(value: T) -> CausalTensor<T> {
CausalTensor::new(vec![value], vec![]).expect("Scalar tensor creation should not fail")
}
fn apply<A, B, Func>(f_ab: CausalTensor<Func>, f_a: CausalTensor<A>) -> CausalTensor<B>
where
Func: FnMut(A) -> B,
{
if f_ab.shape().is_empty() && f_ab.len() == 1 {
let shape = f_a.shape().to_vec(); let func = f_ab.data.into_iter().next().unwrap(); let new_data = f_a.data.into_iter().map(func).collect();
CausalTensor::new(new_data, shape).expect("Shape should remain valid after apply")
} else {
CausalTensor::new(Vec::new(), vec![0])
.expect("Creating an empty tensor should not fail")
}
}
}
impl Foldable<CausalTensorWitness> for CausalTensorWitness {
fn fold<A, B, Func>(fa: CausalTensor<A>, init: B, f: Func) -> B
where
Func: FnMut(B, A) -> B,
{
fa.data.into_iter().fold(init, f)
}
}
impl Functor<CausalTensorWitness> for CausalTensorWitness {
fn fmap<A, B, Func>(m_a: CausalTensor<A>, f: Func) -> CausalTensor<B>
where
Func: FnMut(A) -> B,
{
let shape = m_a.shape().to_vec(); let new_data = m_a.data.into_iter().map(f).collect();
CausalTensor::new(new_data, shape).expect("Shape should remain valid after fmap")
}
}
impl Monad<CausalTensorWitness> for CausalTensorWitness {
fn bind<A, B, Func>(m_a: CausalTensor<A>, mut f: Func) -> CausalTensor<B>
where
Func: FnMut(A) -> CausalTensor<B>,
{
let result_data: Vec<B> = m_a
.data
.into_iter()
.flat_map(|val_a| f(val_a).data.into_iter())
.collect();
let result_len = result_data.len();
CausalTensor::new(result_data, vec![result_len])
.expect("Concatenated tensor creation should not fail")
}
}
impl BoundedComonad<CausalTensorWitness> for CausalTensorWitness {
fn extract<A>(fa: &CausalTensor<A>) -> A
where
A: Clone,
{
if fa.num_dim() == 0 && !fa.is_empty() {
fa.data[0].clone()
} else if fa.is_empty() {
panic!("CoMonad::extract cannot be called on an empty CausalTensor.");
} else {
fa.data[0].clone()
}
}
fn extend<A, B, Func>(fa: &CausalTensor<A>, mut f: Func) -> CausalTensor<B>
where
Func: FnMut(&CausalTensor<A>) -> B,
A: Zero + Copy + Clone,
B: Zero + Copy + Clone,
{
let new_data: Vec<B> = (0..fa.data.len())
.map(|i| {
let focused_view = fa.shifted_view(i);
f(&focused_view)
})
.collect();
CausalTensor::new(new_data, fa.shape().to_vec()).expect("Shape mismatch")
}
}
use deep_causality_haft::BoundedAdjunction;
use std::ops::{Add, Mul};
impl BoundedAdjunction<CausalTensorWitness, CausalTensorWitness, Vec<usize>>
for CausalTensorWitness
{
fn left_adjunct<A, B, F>(ctx: &Vec<usize>, a: A, f: F) -> CausalTensor<B>
where
F: Fn(CausalTensor<A>) -> B,
A: Clone + Zero + Copy + PartialEq,
B: Clone,
{
let t_t_a = Self::unit(ctx, a);
<Self as Functor<Self>>::fmap(t_t_a, f)
}
fn right_adjunct<A, B, F>(ctx: &Vec<usize>, la: CausalTensor<A>, f: F) -> B
where
F: FnMut(A) -> CausalTensor<B>,
A: Clone + Zero,
B: Clone + Zero + Add<Output = B> + Mul<Output = B>,
{
let mapped = <Self as Functor<Self>>::fmap(la, f);
Self::counit(ctx, mapped)
}
fn unit<A>(ctx: &Vec<usize>, a: A) -> CausalTensor<CausalTensor<A>>
where
A: Clone + Zero + Copy + PartialEq,
{
if !ctx.is_empty() {
panic!(
"BoundedAdjunction::unit for CausalTensor requires an empty shape vector (Scalar). Provided shape: {:?}",
ctx
);
}
let inner = CausalTensor::new(vec![a], ctx.clone()).expect("Inner tensor creation failed");
CausalTensor::new(vec![inner], vec![]).expect("Outer tensor creation failed")
}
fn counit<B>(_ctx: &Vec<usize>, lrb: CausalTensor<CausalTensor<B>>) -> B
where
B: Clone + Zero + Add<Output = B> + Mul<Output = B>,
{
let flattened = <Self as Monad<Self>>::bind(lrb, |x| x);
<Self as BoundedComonad<Self>>::extract(&flattened)
}
}