LCOV - code coverage report
Current view: top level - models/ode_secirvvs - parameter_space.h (source / functions) Hit Total Coverage
Test: coverage.info Lines: 100 105 95.2 %
Date: 2025-01-17 12:16:22 Functions: 3 3 100.0 %

          Line data    Source code
       1             : /* 
       2             : * Copyright (C) 2020-2025 MEmilio
       3             : *
       4             : * Authors: Wadim Koslow, Daniel Abele, Martin J. Kühn
       5             : *
       6             : * Contact: Martin J. Kuehn <Martin.Kuehn@DLR.de>
       7             : *
       8             : * Licensed under the Apache License, Version 2.0 (the "License");
       9             : * you may not use this file except in compliance with the License.
      10             : * You may obtain a copy of the License at
      11             : *
      12             : *     http://www.apache.org/licenses/LICENSE-2.0
      13             : *
      14             : * Unless required by applicable law or agreed to in writing, software
      15             : * distributed under the License is distributed on an "AS IS" BASIS,
      16             : * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
      17             : * See the License for the specific language governing permissions and
      18             : * limitations under the License.
      19             : */
      20             : #ifndef MIO_ODE_SECIRVVS_PARAMETER_SPACE_H
      21             : #define MIO_ODE_SECIRVVS_PARAMETER_SPACE_H
      22             : 
      23             : #include "memilio/mobility/metapopulation_mobility_instant.h"
      24             : #include "memilio/utils/logging.h"
      25             : #include "ode_secirvvs/model.h"
      26             : #include "ode_secirvvs/infection_state.h"
      27             : 
      28             : #include <assert.h>
      29             : 
      30             : namespace mio
      31             : {
      32             : namespace osecirvvs
      33             : {
      34             : /**
      35             :      * draws a sample from the specified distributions for all parameters related to the demographics, e.g. population.
      36             :      * @tparam FP floating point type, e.g., double
      37             :      * @param[inout] model Model including contact patterns for alle age groups
      38             :      */
      39             : template <typename FP = double>
      40         108 : void draw_sample_demographics(Model<FP>& model)
      41             : {
      42         108 :     model.parameters.template get<ICUCapacity<FP>>().draw_sample();
      43         108 :     model.parameters.template get<TestAndTraceCapacity<FP>>().draw_sample();
      44             : 
      45         666 :     for (auto i = AgeGroup(0); i < model.parameters.get_num_groups(); i++) {
      46         558 :         double group_total = model.populations.get_group_total(i);
      47             : 
      48             :         //sample initial compartments (with exceptions)
      49       15624 :         for (auto inf_state = Index<InfectionState>(0); inf_state < InfectionState::Count; ++inf_state) {
      50       30132 :             if (inf_state != InfectionState::SusceptibleNaive && //not sampled, fixed after sampling everything else
      51       29574 :                 inf_state != InfectionState::DeadNaive && //not sampled, fixed from data
      52       73098 :                 inf_state != InfectionState::DeadPartialImmunity && //not sampled, fixed from data
      53       28458 :                 inf_state != InfectionState::DeadImprovedImmunity) { //not sampled, fixed from data
      54       12834 :                 model.populations[{i, inf_state}].draw_sample();
      55             :             }
      56             :         }
      57             : 
      58             :         //set susceptibles so the total number stays the same as before sampling.
      59             :         //if the new total without susceptibles is already bigger than the previous total
      60             :         //subtract the overflow from SusceptibleImprovedImmunity, susceptibles will then be approximately zero.
      61         558 :         model.populations[{i, InfectionState::SusceptibleNaive}] = 0;
      62         558 :         double diff                                              = model.populations.get_group_total(i) - group_total;
      63         558 :         if (diff > 0) {
      64           0 :             model.populations[{i, InfectionState::SusceptibleImprovedImmunity}] -= diff;
      65           0 :             if (model.populations[{i, InfectionState::SusceptibleImprovedImmunity}] < 0.0) {
      66           0 :                 log_error("Negative Compartment after sampling.");
      67             :             }
      68           0 :             assert(std::abs(group_total - model.populations.get_group_total(i)) < 1e-10 && "Sanity check.");
      69             :         }
      70         558 :         model.populations.template set_difference_from_group_total<AgeGroup>({i, InfectionState::SusceptibleNaive},
      71             :                                                                              group_total);
      72             :     }
      73         108 : }
      74             : 
      75             : /**
      76             :      * draws a sample from the specified distributions for all parameters related to the infection.
      77             :      * @tparam FP floating point type, e.g., double
      78             :      * @param[inout] model Model including contact patterns for alle age groups
      79             :      */
      80             : template <typename FP = double>
      81          99 : void draw_sample_infection(Model<FP>& model)
      82             : {
      83          99 :     model.parameters.template get<Seasonality<FP>>().draw_sample();
      84             : 
      85             :     //not age dependent
      86          99 :     model.parameters.template get<TimeExposed<FP>>()[AgeGroup(0)].draw_sample();
      87          99 :     model.parameters.template get<TimeInfectedNoSymptoms<FP>>()[AgeGroup(0)].draw_sample();
      88          99 :     model.parameters.template get<RelativeTransmissionNoSymptoms<FP>>()[AgeGroup(0)].draw_sample();
      89          99 :     model.parameters.template get<RiskOfInfectionFromSymptomatic<FP>>()[AgeGroup(0)].draw_sample();
      90          99 :     model.parameters.template get<MaxRiskOfInfectionFromSymptomatic<FP>>()[AgeGroup(0)].draw_sample();
      91             : 
      92          99 :     model.parameters.template get<ReducExposedPartialImmunity<FP>>()[AgeGroup(0)].draw_sample();
      93          99 :     model.parameters.template get<ReducExposedImprovedImmunity<FP>>()[AgeGroup(0)].draw_sample();
      94          99 :     model.parameters.template get<ReducInfectedSymptomsPartialImmunity<FP>>()[AgeGroup(0)].draw_sample();
      95          99 :     model.parameters.template get<ReducInfectedSymptomsImprovedImmunity<FP>>()[AgeGroup(0)].draw_sample();
      96          99 :     model.parameters.template get<ReducInfectedSevereCriticalDeadPartialImmunity<FP>>()[AgeGroup(0)].draw_sample();
      97          99 :     model.parameters.template get<ReducInfectedSevereCriticalDeadImprovedImmunity<FP>>()[AgeGroup(0)].draw_sample();
      98          99 :     model.parameters.template get<ReducTimeInfectedMild<FP>>()[AgeGroup(0)].draw_sample();
      99             : 
     100         603 :     for (auto i = AgeGroup(0); i < model.parameters.get_num_groups(); i++) {
     101             :         //not age dependent
     102         504 :         model.parameters.template get<TimeExposed<FP>>()[i] =
     103        1008 :             model.parameters.template get<TimeExposed<FP>>()[AgeGroup(0)];
     104         504 :         model.parameters.template get<TimeInfectedNoSymptoms<FP>>()[i] =
     105        1008 :             model.parameters.template get<TimeInfectedNoSymptoms<FP>>()[AgeGroup(0)];
     106         504 :         model.parameters.template get<RelativeTransmissionNoSymptoms<FP>>()[i] =
     107        1008 :             model.parameters.template get<RelativeTransmissionNoSymptoms<FP>>()[AgeGroup(0)];
     108         504 :         model.parameters.template get<RiskOfInfectionFromSymptomatic<FP>>()[i] =
     109        1008 :             model.parameters.template get<RiskOfInfectionFromSymptomatic<FP>>()[AgeGroup(0)];
     110         504 :         model.parameters.template get<MaxRiskOfInfectionFromSymptomatic<FP>>()[i] =
     111        1008 :             model.parameters.template get<MaxRiskOfInfectionFromSymptomatic<FP>>()[AgeGroup(0)];
     112             : 
     113         504 :         model.parameters.template get<ReducExposedPartialImmunity<FP>>()[i] =
     114        1008 :             model.parameters.template get<ReducExposedPartialImmunity<FP>>()[AgeGroup(0)];
     115         504 :         model.parameters.template get<ReducExposedImprovedImmunity<FP>>()[i] =
     116        1008 :             model.parameters.template get<ReducExposedImprovedImmunity<FP>>()[AgeGroup(0)];
     117         504 :         model.parameters.template get<ReducInfectedSymptomsPartialImmunity<FP>>()[i] =
     118        1008 :             model.parameters.template get<ReducInfectedSymptomsPartialImmunity<FP>>()[AgeGroup(0)];
     119         504 :         model.parameters.template get<ReducInfectedSymptomsImprovedImmunity<FP>>()[i] =
     120        1008 :             model.parameters.template get<ReducInfectedSymptomsImprovedImmunity<FP>>()[AgeGroup(0)];
     121         504 :         model.parameters.template get<ReducInfectedSevereCriticalDeadPartialImmunity<FP>>()[i] =
     122        1008 :             model.parameters.template get<ReducInfectedSevereCriticalDeadPartialImmunity<FP>>()[AgeGroup(0)];
     123         504 :         model.parameters.template get<ReducInfectedSevereCriticalDeadImprovedImmunity<FP>>()[i] =
     124        1008 :             model.parameters.template get<ReducInfectedSevereCriticalDeadImprovedImmunity<FP>>()[AgeGroup(0)];
     125         504 :         model.parameters.template get<ReducTimeInfectedMild<FP>>()[i] =
     126        1008 :             model.parameters.template get<ReducTimeInfectedMild<FP>>()[AgeGroup(0)];
     127             : 
     128             :         //age dependent
     129         504 :         model.parameters.template get<TimeInfectedSymptoms<FP>>()[i].draw_sample();
     130         504 :         model.parameters.template get<TimeInfectedSevere<FP>>()[i].draw_sample();
     131         504 :         model.parameters.template get<TimeInfectedCritical<FP>>()[i].draw_sample();
     132             : 
     133         504 :         model.parameters.template get<TransmissionProbabilityOnContact<FP>>()[i].draw_sample();
     134         504 :         model.parameters.template get<RecoveredPerInfectedNoSymptoms<FP>>()[i].draw_sample();
     135         504 :         model.parameters.template get<DeathsPerCritical<FP>>()[i].draw_sample();
     136         504 :         model.parameters.template get<SeverePerInfectedSymptoms<FP>>()[i].draw_sample();
     137         504 :         model.parameters.template get<CriticalPerSevere<FP>>()[i].draw_sample();
     138             :     }
     139          99 : }
     140             : 
     141             : /** Draws a sample from Model parameter distributions and stores sample values
     142             :     * as Parameters parameter values (cf. UncertainValue and Parameters classes)
     143             :     * @tparam FP floating point type, e.g., double
     144             :     * @param[inout] model Model including contact patterns for alle age groups
     145             :     */
     146             : template <typename FP = double>
     147             : void draw_sample(Model<FP>& model)
     148             : {
     149             :     draw_sample_infection(model);
     150             :     draw_sample_demographics(model);
     151             :     model.parameters.template get<ContactPatterns<FP>>().draw_sample();
     152             :     model.apply_constraints();
     153             : }
     154             : 
     155             : /**
     156             :     * Draws samples for each model node in a graph.
     157             :     * Some parameters are shared between nodes and only sampled once.
     158             :     * @tparam FP floating point type, e.g., double
     159             :     * @param graph Graph to be sampled.
     160             :     * @param variant_high If true, use high value for infectiousness of variant.
     161             :     * @return Graph with nodes and edges from the input graph sampled.
     162             :     */
     163             : template <typename FP = double>
     164          99 : Graph<Model<FP>, MobilityParameters<FP>> draw_sample(Graph<Model<FP>, MobilityParameters<FP>>& graph, bool variant_high)
     165             : {
     166          99 :     Graph<Model<FP>, MobilityParameters<FP>> sampled_graph;
     167             : 
     168             :     //sample global parameters
     169          99 :     auto& shared_params_model = graph.nodes()[0].property;
     170          99 :     draw_sample_infection(shared_params_model);
     171          99 :     auto& shared_contacts = shared_params_model.parameters.template get<ContactPatterns<FP>>();
     172          99 :     shared_contacts.draw_sample_dampings();
     173          99 :     auto& shared_dynamic_npis = shared_params_model.parameters.template get<DynamicNPIsInfectedSymptoms<FP>>();
     174          99 :     shared_dynamic_npis.draw_sample();
     175          99 :     auto& shared_dynamic_npis_delay = shared_params_model.parameters.template get<DynamicNPIsImplementationDelay<FP>>();
     176          99 :     shared_dynamic_npis_delay.draw_sample();
     177             : 
     178             :     double delta_fac;
     179          99 :     if (variant_high) {
     180          99 :         delta_fac = 1.6;
     181             :     }
     182             :     else {
     183           0 :         delta_fac = 1.4;
     184             :     }
     185             : 
     186             :     //infectiousness of virus variants is not sampled independently but depend on base infectiousness
     187         603 :     for (auto i = AgeGroup(0); i < shared_params_model.parameters.get_num_groups(); ++i) {
     188         504 :         shared_params_model.parameters.template get<InfectiousnessNewVariant<FP>>()[i] = delta_fac;
     189             :     }
     190             : 
     191         315 :     for (auto& params_node : graph.nodes()) {
     192         108 :         auto& node_model = params_node.property;
     193             : 
     194             :         //sample local parameters
     195         108 :         draw_sample_demographics(params_node.property);
     196             : 
     197             :         //copy global parameters
     198             :         //save demographic parameters so they aren't overwritten
     199         108 :         auto local_icu_capacity = node_model.parameters.template get<ICUCapacity<FP>>();
     200         108 :         auto local_tnt_capacity = node_model.parameters.template get<TestAndTraceCapacity<FP>>();
     201         108 :         auto local_holidays     = node_model.parameters.template get<ContactPatterns<FP>>().get_school_holidays();
     202         108 :         auto local_daily_v1     = node_model.parameters.template get<DailyPartialVaccinations<FP>>();
     203         108 :         auto local_daily_v2     = node_model.parameters.template get<DailyFullVaccinations<FP>>();
     204         108 :         node_model.parameters   = shared_params_model.parameters;
     205         108 :         node_model.parameters.template get<ICUCapacity<FP>>()                           = local_icu_capacity;
     206         108 :         node_model.parameters.template get<TestAndTraceCapacity<FP>>()                  = local_tnt_capacity;
     207         108 :         node_model.parameters.template get<ContactPatterns<FP>>().get_school_holidays() = local_holidays;
     208         108 :         node_model.parameters.template get<DailyPartialVaccinations<FP>>()              = local_daily_v1;
     209         108 :         node_model.parameters.template get<DailyFullVaccinations<FP>>()                 = local_daily_v2;
     210             : 
     211         108 :         node_model.parameters.template get<ContactPatterns<FP>>().make_matrix();
     212         108 :         node_model.apply_constraints();
     213             : 
     214         108 :         sampled_graph.add_node(params_node.id, node_model);
     215             :     }
     216             : 
     217         117 :     for (auto& edge : graph.edges()) {
     218           9 :         auto edge_params = edge.property;
     219             :         //no dynamic NPIs
     220             :         //TODO: add switch to optionally enable dynamic NPIs to edges
     221           9 :         sampled_graph.add_edge(edge.start_node_idx, edge.end_node_idx, edge_params);
     222             :     }
     223             : 
     224         198 :     return sampled_graph;
     225          99 : }
     226             : } // namespace osecirvvs
     227             : } // namespace mio
     228             : 
     229             : #endif // MIO_ODE_SECIRVVS_PARAMETER_SPACE_H

Generated by: LCOV version 1.14