def test_model__with_two_symmetric_stratifications(): """ Adding two strata with the same properties should yield the exact same infection dynamics and outputs as having no strata at all. This does not test strains directly, but if this doesn't work then further testing is pointless. """ model = CompartmentalModel(times=[0, 5], compartments=["S", "I", "R"], infectious_compartments=["I"]) model.set_initial_population(distribution={"S": 900, "I": 100}) model.add_infection_frequency_flow("infection", 0.2, "S", "I") model.add_sojourn_flow("recovery", 10, "I", "R") # Do pre-run force of infection calcs. model._prepare_to_run() model._prepare_time_step(0, model.initial_population) # Check infectiousness multipliers susceptible = model.compartments[0] infectious = model.compartments[1] assert model._get_infection_density_multiplier(susceptible, infectious) == 100.0 assert model._get_infection_frequency_multiplier(susceptible, infectious) == 0.1 model.run() # Create a stratified model where the two non-strain strata are symmetric stratified_model = CompartmentalModel(times=[0, 5], compartments=["S", "I", "R"], infectious_compartments=["I"]) stratified_model.set_initial_population(distribution={"S": 900, "I": 100}) stratified_model.add_infection_frequency_flow("infection", 0.2, "S", "I") stratified_model.add_sojourn_flow("recovery", 10, "I", "R") strat = Stratification("clinical", ["home", "hospital"], ["I"]) stratified_model.stratify_with(strat) stratified_model.run() # Ensure stratified model has the same results as the unstratified model. merged_outputs = np.zeros_like(model.outputs) merged_outputs[:, 0] = stratified_model.outputs[:, 0] merged_outputs[:, 1] = stratified_model.outputs[:, 1] + stratified_model.outputs[:, 2] merged_outputs[:, 2] = stratified_model.outputs[:, 3] assert_allclose(merged_outputs, model.outputs, atol=0.01, rtol=0.01, verbose=True)
def test_strains__with_two_symmetric_strains(): """ Adding two strains with the same properties should yield the same infection dynamics and outputs as having no strains at all. We expect the force of infection for each strain to be 1/2 of the unstratified model, but the stratification process will not apply the usual conservation fraction to the fan out flows. """ # Create an unstratified model model = CompartmentalModel(times=[0, 5], compartments=["S", "I", "R"], infectious_compartments=["I"]) model.set_initial_population(distribution={"S": 900, "I": 100}) model.add_infection_frequency_flow("infection", 0.2, "S", "I") model.add_sojourn_flow("recovery", 10, "I", "R") # Do pre-run force of infection calcs. model._prepare_to_run() model._prepare_time_step(0, model.initial_population) # Check infectiousness multipliers susceptible = model.compartments[0] infectious = model.compartments[1] assert model._get_infection_density_multiplier(susceptible, infectious) == 100.0 assert model._get_infection_frequency_multiplier(susceptible, infectious) == 0.1 model.run() # Create a stratified model where the two strain strata are symmetric strain_model = CompartmentalModel(times=[0, 5], compartments=["S", "I", "R"], infectious_compartments=["I"]) strain_model.set_initial_population(distribution={"S": 900, "I": 100}) strain_model.add_infection_frequency_flow("infection", 0.2, "S", "I") strain_model.add_sojourn_flow("recovery", 10, "I", "R") strat = StrainStratification("strain", ["a", "b"], ["I"]) strain_model.stratify_with(strat) strain_model.run() # Ensure stratified model has the same results as the unstratified model. merged_outputs = np.zeros_like(model.outputs) merged_outputs[:, 0] = strain_model.outputs[:, 0] merged_outputs[:, 1] = strain_model.outputs[:, 1] + strain_model.outputs[:, 2] merged_outputs[:, 2] = strain_model.outputs[:, 3] assert_allclose(merged_outputs, model.outputs, atol=0.01, rtol=0.01, verbose=True)
def test_strain__with_flow_adjustments(): """ Test infectious multiplier and flow rate calculations for 3 strains which have different flow adjustments. These flow adjustments would correspond to some physical process that we're modelling, and they should be effectively the same as applying infectiousness multipliers. """ model = CompartmentalModel(times=[0, 5], compartments=["S", "I", "R"], infectious_compartments=["I"]) model.set_initial_population(distribution={"S": 900, "I": 100}) contact_rate = 0.2 model.add_infection_frequency_flow("infection", contact_rate, "S", "I") strat = StrainStratification("strain", ["a", "b", "c"], ["I"]) strat.set_population_split({ "a": 0.7, # 70 people "b": 0.2, # 20 people "c": 0.1, # 10 people }) strat.add_flow_adjustments( "infection", { "a": adjust.Multiply(0.5), # 0.5x as susceptible "b": adjust.Multiply(3), # 3x as susceptible "c": adjust.Multiply(2), # 2x as susceptible }, ) model.stratify_with(strat) # Do pre-run force of infection calcs. model._prepare_to_run() assert_array_equal(model._compartment_infectiousness["a"], np.array([0, 1, 0, 0, 0])) assert_array_equal(model._compartment_infectiousness["b"], np.array([0, 0, 1, 0, 0])) assert_array_equal(model._compartment_infectiousness["c"], np.array([0, 0, 0, 1, 0])) assert model._category_lookup == {0: 0, 1: 0, 2: 0, 3: 0, 4: 0} assert_array_equal(model._category_matrix, np.array([[1, 1, 1, 1, 1]])) # Do pre-iteration force of infection calcs model._prepare_time_step(0, model.initial_population) assert_array_equal(model._category_populations, np.array([1000])) assert_array_equal(model._infection_density["a"], np.array([70])) assert_array_equal(model._infection_density["b"], np.array([20])) assert_array_equal(model._infection_density["c"], np.array([10])) assert_array_equal(model._infection_frequency["a"], np.array([70 / 1000])) assert_array_equal(model._infection_frequency["b"], np.array([20 / 1000])) assert_array_equal(model._infection_frequency["c"], np.array([10 / 1000])) # Get multipliers susceptible = model.compartments[0] infectious_a = model.compartments[1] infectious_b = model.compartments[2] infectious_c = model.compartments[3] assert model._get_infection_density_multiplier(susceptible, infectious_a) == 70 assert model._get_infection_density_multiplier(susceptible, infectious_b) == 20 assert model._get_infection_density_multiplier(susceptible, infectious_c) == 10 assert model._get_infection_frequency_multiplier(susceptible, infectious_a) == 70 / 1000 assert model._get_infection_frequency_multiplier(susceptible, infectious_b) == 20 / 1000 assert model._get_infection_frequency_multiplier(susceptible, infectious_c) == 10 / 1000 # Get infection flow rates flow_rates = model._get_compartment_rates(model.initial_population, 0) sus_pop = 900 flow_to_a = sus_pop * contact_rate * (70 * 0.5 / 1000) flow_to_b = sus_pop * contact_rate * (20 * 3 / 1000) flow_to_c = sus_pop * contact_rate * (10 * 2 / 1000) expected_flow_rates = np.array([ -flow_to_a - flow_to_b - flow_to_c, flow_to_a, flow_to_b, flow_to_c, 0.0 ]) assert_allclose(expected_flow_rates, flow_rates, verbose=True)
def test_strat_infectiousness__with_adjustments(): """ Ensure multiply infectiousness adjustment is applied. """ # Create a model model = CompartmentalModel( times=[0, 5], compartments=["S", "I", "R"], infectious_compartments=["I"] ) model.set_initial_population(distribution={"S": 900, "I": 100}) strat = Stratification("age", ["baby", "child", "adult"], ["S", "I", "R"]) strat.set_population_split({"baby": 0.1, "child": 0.3, "adult": 0.6}) strat.add_infectiousness_adjustments( "I", {"child": adjust.Multiply(3), "adult": adjust.Multiply(0.5), "baby": None} ) model.stratify_with(strat) assert_array_equal( model.initial_population, np.array([90, 270, 540, 10, 30, 60, 0, 0, 0]), ) # Do pre-run force of infection calcs. model._prepare_to_run() assert_array_equal( model._compartment_infectiousness["default"], np.array([0, 0, 0, 1, 3, 0.5, 0, 0, 0]), ) # Do pre-iteration force of infection calcs model._prepare_time_step(0, model.initial_population) # Get multipliers infectees = model.compartments[0:3] infectors = model.compartments[3:6] expected_density = 10 * 1 + 30 * 3 + 60 * 0.5 expected_frequency = expected_density / 1000 for infectee, infector in zip(infectees, infectors): assert model._get_infection_density_multiplier(infectee, infector) == expected_density for infectee, infector in zip(infectees, infectors): assert model._get_infection_frequency_multiplier(infectee, infector) == expected_frequency # Stratify again, now with overwrites strat = Stratification("location", ["urban", "rural"], ["S", "I", "R"]) strat.add_infectiousness_adjustments( "I", {"urban": adjust.Overwrite(1), "rural": adjust.Multiply(7)} ) model.stratify_with(strat) assert_array_equal( model.initial_population, np.array([45, 45, 135, 135, 270.0, 270, 5, 5, 15, 15, 30, 30, 0, 0, 0, 0, 0, 0]), ) # Do pre-run force of infection calcs. model._prepare_to_run() assert_array_equal( model._compartment_infectiousness["default"], np.array([0, 0, 0, 0, 0, 0, 1, 7, 1, 21, 1, 3.5, 0, 0, 0, 0, 0, 0]), ) # Do pre-iteration force of infection calcs model._prepare_time_step(0, model.initial_population) # Get multipliers infectees = model.compartments[0:6] infectors = model.compartments[6:12] expected_density = 5 * 1 + 5 * 7 + 15 * 1 + 15 * 21 + 30 * 1 + 30 * 3.5 expected_frequency = expected_density / 1000 for infectee, infector in zip(infectees, infectors): assert model._get_infection_density_multiplier(infectee, infector) == expected_density for infectee, infector in zip(infectees, infectors): assert model._get_infection_frequency_multiplier(infectee, infector) == expected_frequency