dicetest 0.3.1

Framework for writing tests with randomly generated test data
Documentation
use std::hash::BuildHasher;
#[allow(deprecated)]
use std::hash::SipHasher;
use std::num::Wrapping;

use crate::util::conversion;
use crate::Seed;

/// A pseudorandom number generator.
///
/// The algorithms are based on [this article] by Bob Jenkins.
///
/// [this article]: http://burtleburtle.net/bob/rand/smallprng.html
#[derive(Debug, Clone, Eq, PartialEq)]
pub struct Prng {
    state: (u64, u64, u64, u64),
}

impl Prng {
    /// Creates a [`Prng`] whose internal state is initialized with the given seed.
    ///
    /// The result has a satisfying cycle length.
    pub fn from_seed(seed: Seed) -> Prng {
        let state = (0xf1ea_5eed, seed.0, seed.0, seed.0);
        let mut prng = Prng { state };
        for _ in 0..20 {
            prng.next_number();
        }
        prng
    }

    /// Creates a [`Prng`] using the given byte array as internal state.
    ///
    /// This function is a left and right inverse for [`to_bytes`].
    ///
    /// A satisfying cycle length is only guaranteed for bytes from [`to_bytes`] called
    /// with an [`Prng`] that has a satisfying cycle length. Other bytes should not be passed to this
    /// function. For initializing an [`Prng`] with an arbitrary seed, use [`from_seed`] instead.
    ///
    /// [`to_bytes`]: Prng::to_bytes
    /// [`from_seed`]: Prng::from_seed
    pub fn from_bytes(state_bytes: [u8; 32]) -> Prng {
        #[rustfmt::skip]
        let [
            a0, a1, a2, a3, a4, a5, a6, a7,
            b0, b1, b2, b3, b4, b5, b6, b7,
            c0, c1, c2, c3, c4, c5, c6, c7,
            d0, d1, d2, d3, d4, d5, d6, d7,
        ] = state_bytes;

        let a = conversion::bytes_to_u64([a0, a1, a2, a3, a4, a5, a6, a7]);
        let b = conversion::bytes_to_u64([b0, b1, b2, b3, b4, b5, b6, b7]);
        let c = conversion::bytes_to_u64([c0, c1, c2, c3, c4, c5, c6, c7]);
        let d = conversion::bytes_to_u64([d0, d1, d2, d3, d4, d5, d6, d7]);

        let state = (a, b, c, d);
        Prng { state }
    }

    /// Returns the internal state as a byte array.
    ///
    /// This function is a left and right inverse for `from_bytes`.
    ///
    /// [`from_bytes`]: Prng::from_bytes
    pub fn to_bytes(&self) -> [u8; 32] {
        let (a, b, c, d) = self.state;

        let [a0, a1, a2, a3, a4, a5, a6, a7] = conversion::u64_to_bytes(a);
        let [b0, b1, b2, b3, b4, b5, b6, b7] = conversion::u64_to_bytes(b);
        let [c0, c1, c2, c3, c4, c5, c6, c7] = conversion::u64_to_bytes(c);
        let [d0, d1, d2, d3, d4, d5, d6, d7] = conversion::u64_to_bytes(d);

        #[rustfmt::skip]
        let state_bytes = [
            a0, a1, a2, a3, a4, a5, a6, a7,
            b0, b1, b2, b3, b4, b5, b6, b7,
            c0, c1, c2, c3, c4, c5, c6, c7,
            d0, d1, d2, d3, d4, d5, d6, d7,
        ];

        state_bytes
    }

    #[allow(clippy::many_single_char_names)]
    /// Returns the next pseudorandom number.
    pub fn next_number(&mut self) -> u64 {
        let (a, b, c, d) = self.state;

        // We use `Wrapping` because overflow and underflow is intended
        let Wrapping(e) = Wrapping(a) - Wrapping(b.rotate_left(7));
        let Wrapping(f) = Wrapping(b) ^ Wrapping(c.rotate_left(13));
        let Wrapping(g) = Wrapping(c) + Wrapping(d.rotate_left(37));
        let Wrapping(h) = Wrapping(d) + Wrapping(e);
        let Wrapping(i) = Wrapping(e) + Wrapping(a);

        self.state = (f, g, h, i);
        i
    }

    #[allow(clippy::many_single_char_names)]
    /// Reinitialize the internal state of `self` using the current internal state and the given
    /// seed.
    pub fn reseed(&mut self, seed: Seed) {
        let (a, b, c, d) = self.state;

        // The implementation is inspired by ScalaCheck.
        let n0 = (seed.0 >> 32) & 0xffff_ffff;
        let n1 = seed.0 & 0xffff_ffff;

        self.state = (a ^ n0, b ^ n1, c, d);

        for _ in 0..16 {
            self.next_number();
        }
    }

    /// Splits off a new [`Prng`] from `self`. The internal state of the new [`Prng`] is generated
    /// with `self`.
    pub fn fork(&mut self) -> Prng {
        let random_number = self.next_number();
        let mut reseeded_prng = self.clone();
        reseeded_prng.reseed(random_number.into());
        reseeded_prng
    }
}

impl BuildHasher for Prng {
    #[allow(deprecated)]
    type Hasher = SipHasher;

    fn build_hasher(&self) -> Self::Hasher {
        let mut prng = self.clone();
        let (key0, key1) = (prng.next_number(), prng.next_number());
        #[allow(deprecated)]
        SipHasher::new_with_keys(key0, key1)
    }
}

#[cfg(feature = "rand_core")]
impl rand_core::RngCore for Prng {
    fn next_u32(&mut self) -> u32 {
        self.next_number() as u32
    }

    fn next_u64(&mut self) -> u64 {
        self.next_number()
    }

    fn fill_bytes(&mut self, dest: &mut [u8]) {
        rand_core::impls::fill_bytes_via_next(self, dest)
    }

    fn try_fill_bytes(&mut self, dest: &mut [u8]) -> Result<(), rand_core::Error> {
        self.fill_bytes(dest);
        Ok(())
    }
}

#[cfg(test)]
mod tests {
    use crate::prelude::*;
    use crate::{asserts, Prng};

    #[test]
    fn from_seed_must_not_have_cycle_length_zero() {
        Dicetest::repeatedly().run(|mut fate| {
            let seed = fate.roll(dice::u64(..));

            let prng_init = Prng::from_seed(seed.into());
            let mut prng_next = prng_init.clone();
            let _ = prng_next.next_number();
            let cycle_length_is_zero = prng_init == prng_next;

            hint_debug!(seed);
            hint_debug!(prng_init);
            hint_debug!(prng_next);

            assert!(!cycle_length_is_zero);
        })
    }

    #[test]
    fn from_bytes_is_left_inverse() {
        Dicetest::repeatedly().run(|fate| {
            asserts::left_inverse(
                fate,
                dice::from_fn(|mut fate| fate.fork_prng()),
                |prng| prng.to_bytes(),
                Prng::from_bytes,
            );
        })
    }

    #[test]
    fn to_bytes_is_left_inverse() {
        Dicetest::repeatedly().run(|fate| {
            asserts::left_inverse(fate, dice::array(dice::u8(..)), Prng::from_bytes, |prng| {
                prng.to_bytes()
            });
        })
    }

    #[test]
    fn reseed_changes_prng() {
        Dicetest::repeatedly().run(|mut fate| {
            let prng = fate.fork_prng();
            let seed = fate.roll(dice::u64(..)).into();

            let mut prng_reseeded = prng.clone();
            prng_reseeded.reseed(seed);

            hint_debug!(prng);
            hint_debug!(seed);
            hint_debug!(prng_reseeded);

            assert_ne!(prng, prng_reseeded);
        })
    }
}