quantrs2-sim 0.1.3

Quantum circuit simulators for the QuantRS2 framework
Documentation
//! Tests for the simplified optimized quantum simulator
//!
//! This module provides tests for the simplified optimized simulator implementation
//! to ensure correctness and compatibility.

use scirs2_core::Complex64;
use std::f64::consts::FRAC_1_SQRT_2;

use quantrs2_circuit::builder::{Circuit, Simulator};
use quantrs2_core::register::Register;

use crate::optimized_simulator_simple::OptimizedSimulatorSimple;
use crate::statevector::StateVectorSimulator;

/// Create a bell state circuit
fn create_bell_circuit<const N: usize>() -> Circuit<N> {
    let mut circuit = Circuit::new();

    // Apply Hadamard to qubit 0
    circuit.h(0).expect("Failed to apply H gate");

    // Apply CNOT with qubit 0 as control and qubit 1 as target
    circuit.cnot(0, 1).expect("Failed to apply CNOT gate");

    circuit
}

/// Create a GHZ state circuit for N qubits
fn create_ghz_circuit<const N: usize>() -> Circuit<N> {
    let mut circuit = Circuit::new();

    // Apply Hadamard to qubit 0
    circuit.h(0).expect("Failed to apply H gate");

    // Apply CNOT gates to entangle all qubits
    for i in 1..N {
        circuit.cnot(0, i).expect("Failed to apply CNOT gate");
    }

    circuit
}

/// Create a random circuit with the specified number of gates
fn create_random_circuit<const N: usize>(num_gates: usize) -> Circuit<N> {
    use scirs2_core::random::prelude::*;
    use std::f64::consts::PI;

    let mut circuit = Circuit::new();
    let mut rng = StdRng::seed_from_u64(42); // Use fixed seed for reproducibility

    for _ in 0..num_gates {
        let gate_type = rng.random_range(0..5);

        match gate_type {
            0 => {
                // Hadamard gate
                let target = rng.random_range(0..N);
                circuit.h(target).expect("Failed to apply H gate");
            }
            1 => {
                // Pauli-X gate
                let target = rng.random_range(0..N);
                circuit.x(target).expect("Failed to apply X gate");
            }
            2 => {
                // Rotation-Z gate
                let target = rng.random_range(0..N);
                let angle = rng.random_range(0.0..2.0 * PI);
                circuit.rz(target, angle).expect("Failed to apply RZ gate");
            }
            3 => {
                // CNOT gate
                let control = rng.random_range(0..N);
                let mut target = rng.random_range(0..N);
                while target == control {
                    target = rng.random_range(0..N);
                }
                circuit
                    .cnot(control, target)
                    .expect("Failed to apply CNOT gate");
            }
            4 => {
                // CZ gate
                let control = rng.random_range(0..N);
                let mut target = rng.random_range(0..N);
                while target == control {
                    target = rng.random_range(0..N);
                }
                circuit
                    .cz(control, target)
                    .expect("Failed to apply CZ gate");
            }
            _ => unreachable!(),
        }
    }

    circuit
}

/// Compare results between standard and optimized simulators
fn compare_simulators<const N: usize>(circuit: &Circuit<N>, epsilon: f64) -> bool {
    let standard_sim = StateVectorSimulator::new();
    let optimized_sim = OptimizedSimulatorSimple::new();

    let standard_result = standard_sim
        .run(circuit)
        .expect("Standard simulator failed");
    let optimized_result = optimized_sim
        .run(circuit)
        .expect("Optimized simulator failed");

    let standard_state = standard_result.amplitudes();
    let optimized_state = optimized_result.amplitudes();

    // Check that the dimensions match
    if standard_state.len() != optimized_state.len() {
        println!("State vector dimensions don't match");
        return false;
    }

    // Check each amplitude
    for (i, (std_amp, opt_amp)) in standard_state
        .iter()
        .zip(optimized_state.iter())
        .enumerate()
    {
        let diff = (std_amp - opt_amp).norm();
        if diff > epsilon {
            println!("Amplitude {i} differs: standard={std_amp}, optimized={opt_amp}, diff={diff}");
            return false;
        }
    }

    true
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn test_bell_state() {
        const N: usize = 2;
        let circuit = create_bell_circuit::<N>();

        // Run the circuit with both simulators
        let standard_sim = StateVectorSimulator::new();
        let optimized_sim = OptimizedSimulatorSimple::new();

        let standard_result = standard_sim
            .run(&circuit)
            .expect("Standard simulator failed");
        let optimized_result = optimized_sim
            .run(&circuit)
            .expect("Optimized simulator failed");

        // Expected result: (|00> + |11>) / sqrt(2)
        let expected_amplitudes = [
            Complex64::new(FRAC_1_SQRT_2, 0.0),
            Complex64::new(0.0, 0.0),
            Complex64::new(0.0, 0.0),
            Complex64::new(FRAC_1_SQRT_2, 0.0),
        ];

        // Check standard simulator
        let standard_state = standard_result.amplitudes();
        for (i, (actual, expected)) in standard_state
            .iter()
            .zip(expected_amplitudes.iter())
            .enumerate()
        {
            let diff = (actual - expected).norm();
            assert!(
                diff < 1e-10,
                "Standard simulator: state[{i}] differs by {diff}"
            );
        }

        // Check optimized simulator
        let optimized_state = optimized_result.amplitudes();
        for (i, (actual, expected)) in optimized_state
            .iter()
            .zip(expected_amplitudes.iter())
            .enumerate()
        {
            let diff = (actual - expected).norm();
            assert!(
                diff < 1e-10,
                "Optimized simulator: state[{i}] differs by {diff}"
            );
        }
    }

    #[test]
    fn test_ghz_state() {
        const N: usize = 3;
        let circuit = create_ghz_circuit::<N>();

        // Run the circuit with both simulators
        let standard_sim = StateVectorSimulator::new();
        let optimized_sim = OptimizedSimulatorSimple::new();

        let standard_result = standard_sim
            .run(&circuit)
            .expect("Standard simulator failed");
        let optimized_result = optimized_sim
            .run(&circuit)
            .expect("Optimized simulator failed");

        // Expected result: (|000> + |111>) / sqrt(2)
        let mut expected_amplitudes = [Complex64::new(0.0, 0.0); 1 << N];
        expected_amplitudes[0] = Complex64::new(FRAC_1_SQRT_2, 0.0);
        expected_amplitudes[7] = Complex64::new(FRAC_1_SQRT_2, 0.0);

        // Check standard simulator
        let standard_state = standard_result.amplitudes();
        for (i, (actual, expected)) in standard_state
            .iter()
            .zip(expected_amplitudes.iter())
            .enumerate()
        {
            let diff = (actual - expected).norm();
            assert!(
                diff < 1e-10,
                "Standard simulator: state[{i}] differs by {diff}"
            );
        }

        // Check optimized simulator
        let optimized_state = optimized_result.amplitudes();
        for (i, (actual, expected)) in optimized_state
            .iter()
            .zip(expected_amplitudes.iter())
            .enumerate()
        {
            let diff = (actual - expected).norm();
            assert!(
                diff < 1e-10,
                "Optimized simulator: state[{i}] differs by {diff}"
            );
        }
    }

    #[test]
    fn test_random_circuit_4qubits() {
        const N: usize = 4;
        let circuit = create_random_circuit::<N>(20);

        assert!(
            compare_simulators(&circuit, 1e-10),
            "4-qubit random circuit: simulators disagree"
        );
    }

    #[test]
    fn test_random_circuit_8qubits() {
        const N: usize = 8;
        let circuit = create_random_circuit::<N>(20);

        assert!(
            compare_simulators(&circuit, 1e-10),
            "8-qubit random circuit: simulators disagree"
        );
    }

    #[test]
    #[ignore] // This test is resource-intensive, so we'll ignore it by default
    fn test_random_circuit_16qubits() {
        const N: usize = 16;
        let circuit = create_random_circuit::<N>(10);

        assert!(
            compare_simulators(&circuit, 1e-10),
            "16-qubit random circuit: simulators disagree"
        );
    }

    #[test]
    #[ignore] // This test is even more resource-intensive
    fn test_random_circuit_20qubits() {
        const N: usize = 20;
        let circuit = create_random_circuit::<N>(5);

        assert!(
            compare_simulators(&circuit, 1e-10),
            "20-qubit random circuit: simulators disagree"
        );
    }
}