#![allow(clippy::needless_range_loop)]
#![allow(dead_code)]
use crate::thresholds::EXACT_MATCH_TOLERANCE;
use std::f64::consts::PI;
const MAX_ITERATIONS: usize = 10000;
pub const DEFAULT_PSLQ_PRECISION: usize = 50;
#[derive(Debug, Clone)]
pub struct IntegerRelation {
pub coefficients: Vec<i64>,
pub basis_names: Vec<String>,
pub residual: f64,
pub is_exact: bool,
}
impl IntegerRelation {
pub fn format(&self) -> String {
let terms: Vec<String> = self
.coefficients
.iter()
.zip(self.basis_names.iter())
.filter(|(c, _)| **c != 0)
.map(|(c, name)| {
if *c == 1 {
name.clone()
} else if *c == -1 {
format!("-{}", name)
} else {
format!("{}*{}", c, name)
}
})
.collect();
if terms.is_empty() {
"0".to_string()
} else {
terms.join(" + ").replace("+ -", "- ")
}
}
}
pub fn standard_constants() -> Vec<(String, f64)> {
vec![
("1".to_string(), 1.0),
("π".to_string(), PI),
("π²".to_string(), PI * PI),
("π³".to_string(), PI * PI * PI),
("e".to_string(), std::f64::consts::E),
("e²".to_string(), std::f64::consts::E * std::f64::consts::E),
("e^π".to_string(), std::f64::consts::E.powf(PI)),
("ln(2)".to_string(), (2.0f64).ln()),
("ln(π)".to_string(), PI.ln()),
("√2".to_string(), std::f64::consts::SQRT_2),
("√π".to_string(), PI.sqrt()),
("φ".to_string(), (1.0 + 5.0f64.sqrt()) / 2.0), ("γ".to_string(), 0.5772156649015329), ("ζ(2)".to_string(), PI * PI / 6.0), ("ζ(3)".to_string(), 1.202056903159594), ("G".to_string(), 0.915965594177219), ]
}
pub fn extended_constants() -> Vec<(String, f64)> {
let mut constants = standard_constants();
constants.extend(vec![
("√3".to_string(), 3.0f64.sqrt()),
("√5".to_string(), 5.0f64.sqrt()),
("√7".to_string(), 7.0f64.sqrt()),
("ln(3)".to_string(), (3.0f64).ln()),
("ln(5)".to_string(), (5.0f64).ln()),
("ln(7)".to_string(), (7.0f64).ln()),
("π*√2".to_string(), PI * std::f64::consts::SQRT_2),
("e+π".to_string(), std::f64::consts::E + PI),
("e*π".to_string(), std::f64::consts::E * PI),
("2^π".to_string(), 2.0f64.powf(PI)),
("π^e".to_string(), PI.powf(std::f64::consts::E)),
]);
constants
}
#[derive(Debug, Clone)]
pub struct PslqConfig {
pub max_coefficient: i64,
pub max_iterations: usize,
pub tolerance: f64,
pub extended_constants: bool,
}
impl Default for PslqConfig {
fn default() -> Self {
Self {
max_coefficient: 1000,
max_iterations: MAX_ITERATIONS,
tolerance: EXACT_MATCH_TOLERANCE,
extended_constants: false,
}
}
}
pub fn find_integer_relation(target: f64, config: &PslqConfig) -> Option<IntegerRelation> {
let constants = if config.extended_constants {
extended_constants()
} else {
standard_constants()
};
let _n = constants.len() + 1;
let mut x: Vec<f64> = vec![target];
for (_, val) in &constants {
x.push(*val);
}
let coefficients = pslq(&x, config)?;
if coefficients[0] == 0 {
return None;
}
let mut residual = 0.0;
for (i, c) in coefficients.iter().enumerate() {
residual += (*c as f64) * x[i];
}
residual = residual.abs();
if residual > config.tolerance * target.abs().max(1.0) {
return None;
}
let mut basis_names = vec!["x".to_string()];
for (name, _) in &constants {
basis_names.push(name.clone());
}
Some(IntegerRelation {
coefficients,
basis_names,
residual,
is_exact: residual < EXACT_MATCH_TOLERANCE,
})
}
fn pslq(x: &[f64], config: &PslqConfig) -> Option<Vec<i64>> {
let n = x.len();
if n < 2 {
return None;
}
let _gamma = (4.0 / 3.0_f64).sqrt();
let mut s: Vec<f64> = vec![0.0; n];
s[n - 1] = x[n - 1].abs();
for i in (0..n - 1).rev() {
s[i] = (s[i + 1].powi(2) + x[i].powi(2)).sqrt();
}
let scale = s[0];
if scale == 0.0 {
return None;
}
let mut y: Vec<f64> = x.iter().map(|xi| xi / scale).collect();
for i in 0..n {
s[i] /= scale;
}
let mut h: Vec<Vec<f64>> = vec![vec![0.0; n]; n - 1];
for i in 0..n - 1 {
for j in 0..=i.min(n - 2) {
if i == j {
h[i][j] = s[j + 1] / s[j];
} else {
h[i][j] = -y[j] * y[i + 1] / (s[i] * s[i + 1]);
}
}
}
let mut a: Vec<Vec<i64>> = vec![vec![0; n]; n];
let mut b: Vec<Vec<i64>> = vec![vec![0; n]; n];
for i in 0..n {
a[i][i] = 1;
b[i][i] = 1;
}
for _iteration in 0..config.max_iterations {
let mut max_val = 0.0;
let mut max_idx = 0;
for i in 0..n - 1 {
let val = h[i][i].abs();
if val > max_val {
max_val = val;
max_idx = i;
}
}
if max_idx < n - 1 {
y.swap(max_idx, max_idx + 1);
a.swap(max_idx, max_idx + 1);
b.swap(max_idx, max_idx + 1);
for j in 0..n - 1 {
let tmp = h[max_idx][j];
h[max_idx][j] = h[max_idx + 1][j];
h[max_idx + 1][j] = tmp;
}
}
for i in (1..n).rev() {
if max_idx < n - 1 && h[max_idx][max_idx].abs() > 1e-50 {
let t = (h[i - 1][max_idx] / h[max_idx][max_idx]).round();
y[i - 1] -= t * y[max_idx];
for j in 0..n - 1 {
h[i - 1][j] -= t * h[max_idx][j];
}
for j in 0..n {
a[i - 1][j] -= (t as i64) * a[max_idx][j];
b[j][i - 1] -= (t as i64) * b[j][max_idx];
}
}
}
for i in 0..n {
if y[i].abs() < config.tolerance {
let coeffs: Vec<i64> = (0..n).map(|j| b[j][i]).collect();
if coeffs.iter().all(|&c| c.abs() <= config.max_coefficient) {
return Some(coeffs);
}
}
}
let mut min_diag = f64::MAX;
for i in 0..n - 1 {
if h[i][i].abs() < min_diag {
min_diag = h[i][i].abs();
}
}
if min_diag < config.tolerance {
let coeffs: Vec<i64> = (0..n).map(|j| b[j][0]).collect();
if coeffs.iter().all(|&c| c.abs() <= config.max_coefficient) {
return Some(coeffs);
}
}
}
None
}
pub fn find_rational_approximation(x: f64, max_denominator: i64) -> Option<(i64, i64)> {
let a0 = x.floor() as i64;
let mut remainder = x - a0 as f64;
if remainder.abs() < EXACT_MATCH_TOLERANCE {
return Some((a0, 1));
}
let mut h_prev = 1i64;
let mut h_curr = a0;
let mut k_prev = 0i64;
let mut k_curr = 1i64;
for _ in 0..100 {
if remainder.abs() < EXACT_MATCH_TOLERANCE {
break;
}
let reciprocal = 1.0 / remainder;
let a = reciprocal.floor() as i64;
remainder = reciprocal - a as f64;
let h_next = a * h_curr + h_prev;
let k_next = a * k_curr + k_prev;
if k_next > max_denominator {
break;
}
h_prev = h_curr;
h_curr = h_next;
k_prev = k_curr;
k_curr = k_next;
let approx = h_curr as f64 / k_curr as f64;
if (approx - x).abs() < EXACT_MATCH_TOLERANCE {
return Some((h_curr, k_curr));
}
}
if k_curr > 0 && k_curr <= max_denominator {
let approx = h_curr as f64 / k_curr as f64;
if (approx - x).abs() < x.abs() * 0.01 {
return Some((h_curr, k_curr));
}
}
None
}
pub fn find_minimal_polynomial(x: f64, max_degree: usize, max_coeff: i64) -> Option<Vec<i64>> {
for degree in 1..=max_degree {
if let Some(coeffs) = try_polynomial_degree(x, degree, max_coeff) {
return Some(coeffs);
}
}
None
}
fn try_polynomial_degree(x: f64, degree: usize, max_coeff: i64) -> Option<Vec<i64>> {
if degree == 0 {
return None;
}
let mut powers = vec![1.0; degree + 1];
for i in 1..=degree {
powers[i] = powers[i - 1] * x;
}
let mut best_coeffs: Option<Vec<i64>> = None;
let mut best_error = f64::MAX;
let search_range = (-(max_coeff / 10).max(1)..=(max_coeff / 10).max(1)).collect::<Vec<_>>();
if degree <= 2 {
for coeffs in coefficients_product(&search_range, degree + 1) {
let mut value = 0.0;
for (i, c) in coeffs.iter().enumerate() {
value += (*c as f64) * powers[i];
}
let error = value.abs();
if error < best_error && error < EXACT_MATCH_TOLERANCE * 100.0 {
best_error = error;
best_coeffs = Some(coeffs);
}
}
}
best_coeffs
}
fn coefficients_product(ranges: &[i64], count: usize) -> Vec<Vec<i64>> {
if count == 0 {
return vec![vec![]];
}
let mut result = Vec::new();
let rest = coefficients_product(ranges, count - 1);
for r in rest {
for &val in ranges {
let mut combo = r.clone();
combo.push(val);
result.push(combo);
}
}
result
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_rational_approximation_pi() {
let result = find_rational_approximation(PI, 1000);
assert!(result.is_some());
let (num, den) = result.unwrap();
assert_eq!(num, 355);
assert_eq!(den, 113);
}
#[test]
fn test_rational_approximation_sqrt2() {
let result = find_rational_approximation(std::f64::consts::SQRT_2, 200);
assert!(result.is_some());
let (num, den) = result.unwrap();
let approx = num as f64 / den as f64;
assert!((approx - std::f64::consts::SQRT_2).abs() < 0.001);
}
#[test]
fn test_integer_relation_simple() {
let config = PslqConfig::default();
let result = find_integer_relation(2.0 * PI, &config);
if let Some(rel) = result {
assert!(rel.residual < 0.01);
}
}
#[test]
fn test_minimal_polynomial_sqrt2() {
let result = find_minimal_polynomial(std::f64::consts::SQRT_2, 4, 100);
if let Some(coeffs) = result {
let value: f64 = coeffs
.iter()
.enumerate()
.map(|(i, c)| *c as f64 * std::f64::consts::SQRT_2.powi(i as i32))
.sum();
assert!(value.abs() < 0.01);
}
}
}