use ndarray::Array1;
use rand_distr::Distribution;
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::process::cpoisson::CompoundPoisson;
use crate::traits::FloatExt;
use crate::traits::ProcessExt;
pub struct Kou<T, D, S: SeedExt = Unseeded>
where
T: FloatExt,
D: Distribution<T> + Send + Sync,
{
pub alpha: T,
pub sigma: T,
pub lambda: T,
pub theta: T,
pub n: usize,
pub x0: Option<T>,
pub t: Option<T>,
pub cpoisson: CompoundPoisson<T, D>,
pub seed: S,
}
impl<T, D> Kou<T, D>
where
T: FloatExt,
D: Distribution<T> + Send + Sync,
{
pub fn new(
alpha: T,
sigma: T,
lambda: T,
theta: T,
n: usize,
x0: Option<T>,
t: Option<T>,
cpoisson: CompoundPoisson<T, D>,
) -> Self {
Self {
alpha,
sigma,
lambda,
theta,
n,
x0,
t,
cpoisson,
seed: Unseeded,
}
}
}
impl<T, D> Kou<T, D, Deterministic>
where
T: FloatExt,
D: Distribution<T> + Send + Sync,
{
pub fn seeded(
alpha: T,
sigma: T,
lambda: T,
theta: T,
n: usize,
x0: Option<T>,
t: Option<T>,
cpoisson: CompoundPoisson<T, D>,
seed: u64,
) -> Self {
Self {
alpha,
sigma,
lambda,
theta,
n,
x0,
t,
cpoisson,
seed: Deterministic::new(seed),
}
}
}
impl<T, D, S: SeedExt> ProcessExt<T> for Kou<T, D, S>
where
T: FloatExt,
D: Distribution<T> + Send + Sync,
{
type Output = Array1<T>;
fn sample(&self) -> Self::Output {
let dt = if self.n > 1 {
self.t.unwrap_or(T::one()) / T::from_usize_(self.n - 1)
} else {
T::zero()
};
let jump_increments = self.cpoisson.sample_grid_increments(self.n, dt);
let sqrt_dt = dt.sqrt();
let mut gn = Array1::<T>::zeros(self.n.saturating_sub(1));
if let Some(gn_slice) = gn.as_slice_mut() {
let normal = SimdNormal::<T>::from_seed_source(T::zero(), sqrt_dt, &self.seed);
normal.fill_slice_fast(gn_slice);
}
let mut merton = Array1::<T>::zeros(self.n);
if self.n == 0 {
return merton;
}
merton[0] = self.x0.unwrap_or(T::zero());
for i in 1..self.n {
merton[i] = merton[i - 1]
+ (self.alpha
- self.sigma.powf(T::from_usize_(2)) / T::from_usize_(2)
- self.lambda * self.theta)
* dt
+ self.sigma * gn[i - 1]
+ jump_increments[i];
}
merton
}
}
#[cfg(feature = "python")]
#[pyo3::prelude::pyclass]
pub struct PyKou {
inner_f32: Option<Kou<f32, crate::traits::CallableDist<f32>>>,
inner_f64: Option<Kou<f64, crate::traits::CallableDist<f64>>>,
seeded_f32: Option<Kou<f32, crate::traits::CallableDist<f32>, crate::simd_rng::Deterministic>>,
seeded_f64: Option<Kou<f64, crate::traits::CallableDist<f64>, crate::simd_rng::Deterministic>>,
}
#[cfg(feature = "python")]
#[pyo3::prelude::pymethods]
impl PyKou {
#[new]
#[pyo3(signature = (alpha, sigma, lambda_, theta, distribution, n, x0=None, t=None, seed=None, dtype=None))]
fn new(
alpha: f64,
sigma: f64,
lambda_: f64,
theta: f64,
distribution: pyo3::Py<pyo3::PyAny>,
n: usize,
x0: Option<f64>,
t: Option<f64>,
seed: Option<u64>,
dtype: Option<&str>,
) -> Self {
use crate::process::poisson::Poisson;
let mut s = Self {
inner_f32: None,
inner_f64: None,
seeded_f32: None,
seeded_f64: None,
};
match dtype.unwrap_or("f64") {
"f32" => {
let cpoisson = CompoundPoisson::new(
crate::traits::CallableDist::new(distribution),
Poisson::new(lambda_ as f32, Some(n), t.map(|v| v as f32)),
);
match seed {
Some(sd) => {
s.seeded_f32 = Some(Kou::seeded(
alpha as f32,
sigma as f32,
lambda_ as f32,
theta as f32,
n,
x0.map(|v| v as f32),
t.map(|v| v as f32),
cpoisson,
sd,
));
}
None => {
s.inner_f32 = Some(Kou::new(
alpha as f32,
sigma as f32,
lambda_ as f32,
theta as f32,
n,
x0.map(|v| v as f32),
t.map(|v| v as f32),
cpoisson,
));
}
}
}
_ => {
let cpoisson = CompoundPoisson::new(
crate::traits::CallableDist::new(distribution),
Poisson::new(lambda_, Some(n), t),
);
match seed {
Some(sd) => {
s.seeded_f64 = Some(Kou::seeded(
alpha, sigma, lambda_, theta, n, x0, t, cpoisson, sd,
));
}
None => {
s.inner_f64 = Some(Kou::new(alpha, sigma, lambda_, theta, n, x0, t, cpoisson));
}
}
}
}
s
}
fn sample<'py>(&self, py: pyo3::Python<'py>) -> pyo3::Py<pyo3::PyAny> {
use numpy::IntoPyArray;
use pyo3::IntoPyObjectExt;
use crate::traits::ProcessExt;
py_dispatch!(self, |inner| inner
.sample()
.into_pyarray(py)
.into_py_any(py)
.unwrap())
}
fn sample_par<'py>(&self, py: pyo3::Python<'py>, m: usize) -> pyo3::Py<pyo3::PyAny> {
use numpy::IntoPyArray;
use numpy::ndarray::Array2;
use pyo3::IntoPyObjectExt;
use crate::traits::ProcessExt;
py_dispatch!(self, |inner| {
let paths = inner.sample_par(m);
let n = paths[0].len();
let mut result = Array2::zeros((m, n));
for (i, path) in paths.iter().enumerate() {
result.row_mut(i).assign(path);
}
result.into_pyarray(py).into_py_any(py).unwrap()
})
}
}