use ndarray::Array1;
use stochastic_rs_core::simd_rng::Deterministic;
use stochastic_rs_core::simd_rng::SeedExt;
use stochastic_rs_core::simd_rng::Unseeded;
use stochastic_rs_distributions::normal::SimdNormal;
use crate::traits::FloatExt;
use crate::traits::ProcessExt;
#[derive(Debug, Clone)]
pub struct Arch<T: FloatExt, S: SeedExt = Unseeded> {
pub omega: T,
pub alpha: Array1<T>,
pub n: usize,
pub seed: S,
}
impl<T: FloatExt> Arch<T> {
pub fn new(omega: T, alpha: Array1<T>, n: usize) -> Self {
assert!(omega > T::zero(), "Arch requires omega > 0");
Self {
omega,
alpha,
n,
seed: Unseeded,
}
}
}
impl<T: FloatExt> Arch<T, Deterministic> {
pub fn seeded(omega: T, alpha: Array1<T>, n: usize, seed: u64) -> Self {
assert!(omega > T::zero(), "Arch requires omega > 0");
Self {
omega,
alpha,
n,
seed: Deterministic::new(seed),
}
}
}
impl<T: FloatExt, S: SeedExt> ProcessExt<T> for Arch<T, S> {
type Output = Array1<T>;
fn sample(&self) -> Self::Output {
let m = self.alpha.len();
let mut z = Array1::<T>::zeros(self.n);
if self.n > 0 {
let slice = z.as_slice_mut().expect("contiguous");
let normal = SimdNormal::<T>::from_seed_source(T::zero(), T::one(), &self.seed);
normal.fill_slice_fast(slice);
}
let mut x = Array1::<T>::zeros(self.n);
let var_floor = T::from_f64_fast(1e-12);
for t in 0..self.n {
let mut var_t = self.omega;
for i in 1..=m {
if t >= i {
let x_lag = x[t - i];
var_t += self.alpha[i - 1] * x_lag.powi(2);
}
}
assert!(
var_t.is_finite() && var_t > T::zero(),
"Arch produced non-positive or non-finite conditional variance at t={}",
t
);
let sigma_t = var_t.max(var_floor).sqrt();
x[t] = sigma_t * z[t];
}
x
}
}
py_process_1d!(PyArch, Arch,
sig: (omega, alpha, n, seed=None, dtype=None),
params: (omega: f64, alpha: Vec<f64>, n: usize)
);