import unittest
from collections import Counter

from utils import evenly_distributed_crash_with_survival


class TestUtils(unittest.TestCase):
    def setUp(self):
        """Set up test parameters."""
        self.alive_nodes = [1, 2, 3]
        self.num_trials = 10000  # For probability distribution testing

    def test_at_least_one_alive(self):
        """Ensure that at least one node remains alive in every run."""
        for _ in range(1000):  # Run multiple times to check consistency
            crashed_nodes = evenly_distributed_crash_with_survival(self.alive_nodes)
            self.assertTrue(len(crashed_nodes) < len(self.alive_nodes))

    def test_output_is_subset(self):
        """Ensure that the returned crashed nodes are always a subset of the input alive nodes."""
        for _ in range(1000):
            crashed_nodes = evenly_distributed_crash_with_survival(self.alive_nodes)
            self.assertTrue(set(crashed_nodes).issubset(self.alive_nodes))

    def test_uniform_distribution(self):
        """Test that all valid crash scenarios appear with approximately equal probability."""
        crash_counts = Counter()

        for _ in range(self.num_trials):
            crashed_nodes = frozenset(evenly_distributed_crash_with_survival(self.alive_nodes))
            crash_counts[crashed_nodes] += 1

        # Expected uniform probability (each valid crash scenario should be around total_trials / num_valid_cases)
        num_valid_cases = 2 ** len(self.alive_nodes) - 1
        expected_probability = self.num_trials / num_valid_cases

        # Check each scenario's occurrence is roughly equal
        for scenario, count in crash_counts.items():
            self.assertAlmostEqual(count / expected_probability, 1.0, delta=0.1)
