Line data Source code
1 : /*
2 : * Copyright (C) 2020-2025 MEmilio
3 : *
4 : * Authors: David Kerkmann, Sascha Korf, Khoa Nguyen
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 :
21 : #include "abm/infection.h"
22 : #include <math.h>
23 :
24 : namespace mio
25 : {
26 : namespace abm
27 : {
28 :
29 504 : Infection::Infection(PersonalRandomNumberGenerator& rng, VirusVariant virus, AgeGroup age, const Parameters& params,
30 504 : TimePoint init_date, InfectionState init_state, ProtectionEvent latest_protection, bool detected)
31 504 : : m_virus_variant(virus)
32 1008 : , m_detected(detected)
33 : {
34 504 : assert(age.get() < params.get_num_groups());
35 504 : m_viral_load.start_date = draw_infection_course(rng, age, params, init_date, init_state, latest_protection);
36 :
37 504 : auto vl_params = params.get<ViralLoadDistributions>()[{virus, age}];
38 504 : ScalarType high_viral_load_factor = 1;
39 504 : if (latest_protection.type != ProtectionType::NoProtection) {
40 9 : high_viral_load_factor -=
41 18 : params.get<HighViralLoadProtectionFactor>()[{latest_protection.type, age, virus}](
42 9 : init_date.days() - latest_protection.time.days());
43 : }
44 504 : m_viral_load.peak = vl_params.viral_load_peak.get_distribution_instance()(rng, vl_params.viral_load_peak.params) *
45 : high_viral_load_factor;
46 504 : m_viral_load.incline =
47 504 : vl_params.viral_load_incline.get_distribution_instance()(rng, vl_params.viral_load_incline.params);
48 504 : m_viral_load.decline =
49 504 : vl_params.viral_load_decline.get_distribution_instance()(rng, vl_params.viral_load_decline.params);
50 504 : m_viral_load.end_date =
51 2016 : m_viral_load.start_date +
52 2520 : days(int(m_viral_load.peak / m_viral_load.incline - m_viral_load.peak / m_viral_load.decline));
53 :
54 504 : auto inf_params = params.get<InfectivityDistributions>()[{virus, age}];
55 504 : m_log_norm_alpha =
56 504 : inf_params.infectivity_alpha.get_distribution_instance()(rng, inf_params.infectivity_alpha.params);
57 504 : m_log_norm_beta = inf_params.infectivity_beta.get_distribution_instance()(rng, inf_params.infectivity_beta.params);
58 504 : }
59 :
60 828 : ScalarType Infection::get_viral_load(TimePoint t) const
61 : {
62 828 : if (t >= m_viral_load.start_date && t <= m_viral_load.end_date) {
63 693 : if (t.days() <= m_viral_load.start_date.days() + m_viral_load.peak / m_viral_load.incline) {
64 207 : return m_viral_load.incline * (t - m_viral_load.start_date).days();
65 : }
66 : else {
67 486 : return m_viral_load.peak + m_viral_load.decline * (t.days() - m_viral_load.peak / m_viral_load.incline -
68 486 : m_viral_load.start_date.days());
69 : }
70 : }
71 : else {
72 135 : return 0.;
73 : }
74 : }
75 :
76 1692 : ScalarType Infection::get_infectivity(TimePoint t) const
77 : {
78 1692 : if (m_viral_load.start_date >= t || get_infection_state(t) == InfectionState::Exposed)
79 864 : return 0;
80 828 : return 1 / (1 + exp(-(m_log_norm_alpha + m_log_norm_beta * get_viral_load(t))));
81 : }
82 :
83 846 : VirusVariant Infection::get_virus_variant() const
84 : {
85 846 : return m_virus_variant;
86 : }
87 :
88 14094 : InfectionState Infection::get_infection_state(TimePoint t) const
89 : {
90 14094 : if (t < m_infection_course[0].first)
91 18 : return InfectionState::Susceptible;
92 :
93 14076 : return (*std::prev(std::upper_bound(m_infection_course.begin(), m_infection_course.end(), t,
94 36351 : [](const TimePoint& s, std::pair<TimePoint, InfectionState> state) {
95 36351 : return state.first > s;
96 28152 : })))
97 14076 : .second;
98 : }
99 :
100 90 : void Infection::set_detected()
101 : {
102 90 : m_detected = true;
103 90 : }
104 :
105 9 : bool Infection::is_detected() const
106 : {
107 9 : return m_detected;
108 : }
109 :
110 9 : TimePoint Infection::get_start_date() const
111 : {
112 9 : return m_viral_load.start_date;
113 : }
114 :
115 504 : TimePoint Infection::draw_infection_course(PersonalRandomNumberGenerator& rng, AgeGroup age, const Parameters& params,
116 : TimePoint init_date, InfectionState init_state,
117 : ProtectionEvent latest_protection)
118 : {
119 504 : assert(age.get() < params.get_num_groups());
120 504 : TimePoint start_date = draw_infection_course_backward(rng, age, params, init_date, init_state);
121 504 : draw_infection_course_forward(rng, age, params, init_date, init_state, latest_protection);
122 1008 : return start_date;
123 : }
124 :
125 504 : void Infection::draw_infection_course_forward(PersonalRandomNumberGenerator& rng, AgeGroup age,
126 : const Parameters& params, TimePoint init_date, InfectionState start_state,
127 : ProtectionEvent latest_protection)
128 : {
129 504 : assert(age.get() < params.get_num_groups());
130 504 : auto t = init_date;
131 504 : TimeSpan time_period{}; // time period for current infection state
132 504 : InfectionState next_state{start_state}; // next state to enter
133 504 : m_infection_course.push_back(std::pair<TimePoint, InfectionState>(t, next_state));
134 504 : auto& uniform_dist = UniformDistribution<double>::get_instance();
135 : ScalarType v; // random draws
136 1116 : while ((next_state != InfectionState::Recovered && next_state != InfectionState::Dead)) {
137 612 : switch (next_state) {
138 72 : case InfectionState::Exposed:
139 : // roll out how long until infected without symptoms
140 72 : time_period = days(params.get<IncubationPeriod>()[{m_virus_variant, age}]); // subject to change
141 72 : next_state = InfectionState::InfectedNoSymptoms;
142 72 : break;
143 207 : case InfectionState::InfectedNoSymptoms:
144 : // roll out next infection step
145 207 : v = uniform_dist(rng);
146 207 : if (v < 0.5) { // TODO: subject to change
147 : time_period =
148 108 : days(params.get<InfectedNoSymptomsToSymptoms>()[{m_virus_variant, age}]); // TODO: subject to change
149 108 : next_state = InfectionState::InfectedSymptoms;
150 : }
151 : else {
152 297 : time_period = days(
153 297 : params.get<InfectedNoSymptomsToRecovered>()[{m_virus_variant, age}]); // TODO: subject to change
154 99 : next_state = InfectionState::Recovered;
155 : }
156 :
157 207 : break;
158 234 : case InfectionState::InfectedSymptoms:
159 : // roll out next infection step
160 : {
161 234 : ScalarType severity_protection_factor = 0.5;
162 234 : v = uniform_dist(rng);
163 234 : if (latest_protection.type != ProtectionType::NoProtection) {
164 : severity_protection_factor =
165 18 : params.get<SeverityProtectionFactor>()[{latest_protection.type, age, m_virus_variant}](
166 9 : t.days() - latest_protection.time.days());
167 : }
168 234 : if (v < (1 - severity_protection_factor) * 0.5) {
169 : time_period =
170 27 : days(params.get<InfectedSymptomsToSevere>()[{m_virus_variant, age}]); // TODO: subject to change
171 27 : next_state = InfectionState::InfectedSevere;
172 : }
173 : else {
174 621 : time_period = days(
175 621 : params.get<InfectedSymptomsToRecovered>()[{m_virus_variant, age}]); // TODO: subject to change
176 207 : next_state = InfectionState::Recovered;
177 : }
178 234 : break;
179 : }
180 63 : case InfectionState::InfectedSevere:
181 : // roll out next infection step
182 63 : v = uniform_dist(rng);
183 63 : if (v < 0.5) { // TODO: subject to change
184 18 : time_period = days(params.get<SevereToCritical>()[{m_virus_variant, age}]); // TODO: subject to change
185 18 : next_state = InfectionState::InfectedCritical;
186 : }
187 : else {
188 45 : time_period = days(params.get<SevereToRecovered>()[{m_virus_variant, age}]); // TODO: subject to change
189 45 : next_state = InfectionState::Recovered;
190 : }
191 63 : break;
192 36 : case InfectionState::InfectedCritical:
193 : // roll out next infection step
194 36 : v = uniform_dist(rng);
195 36 : if (v < 0.5) { // TODO: subject to change
196 18 : time_period = days(params.get<CriticalToDead>()[{m_virus_variant, age}]); // TODO: subject to change
197 18 : next_state = InfectionState::Dead;
198 : }
199 : else {
200 : time_period =
201 18 : days(params.get<CriticalToRecovered>()[{m_virus_variant, age}]); // TODO: subject to change
202 18 : next_state = InfectionState::Recovered;
203 : }
204 36 : break;
205 0 : default:
206 0 : break;
207 : }
208 612 : t = t + time_period;
209 612 : m_infection_course.push_back({t, next_state});
210 : }
211 504 : }
212 :
213 504 : TimePoint Infection::draw_infection_course_backward(PersonalRandomNumberGenerator& rng, AgeGroup age,
214 : const Parameters& params, TimePoint init_date,
215 : InfectionState init_state)
216 : {
217 504 : assert(age.get() < params.get_num_groups());
218 504 : auto start_date = init_date;
219 504 : TimeSpan time_period{}; // time period for current infection state
220 504 : InfectionState previous_state{init_state}; // next state to enter
221 504 : auto& uniform_dist = UniformDistribution<double>::get_instance();
222 : ScalarType v; // random draws
223 :
224 1566 : while ((previous_state != InfectionState::Exposed)) {
225 1062 : switch (previous_state) {
226 :
227 432 : case InfectionState::InfectedNoSymptoms:
228 432 : time_period = days(params.get<IncubationPeriod>()[{m_virus_variant, age}]); // TODO: subject to change
229 432 : previous_state = InfectionState::Exposed;
230 432 : break;
231 :
232 288 : case InfectionState::InfectedSymptoms:
233 : time_period =
234 288 : days(params.get<InfectedNoSymptomsToSymptoms>()[{m_virus_variant, age}]); // TODO: subject to change
235 288 : previous_state = InfectionState::InfectedNoSymptoms;
236 288 : break;
237 :
238 135 : case InfectionState::InfectedSevere:
239 : time_period =
240 135 : days(params.get<InfectedSymptomsToSevere>()[{m_virus_variant, age}]); // TODO: subject to change
241 135 : previous_state = InfectionState::InfectedSymptoms;
242 135 : break;
243 :
244 90 : case InfectionState::InfectedCritical:
245 90 : time_period = days(params.get<SevereToCritical>()[{m_virus_variant, age}]); // TODO: subject to change
246 90 : previous_state = InfectionState::InfectedSevere;
247 90 : break;
248 :
249 81 : case InfectionState::Recovered:
250 : // roll out next infection step
251 81 : v = uniform_dist(rng);
252 81 : if (v < 0.25) {
253 27 : time_period = days(
254 27 : params.get<InfectedNoSymptomsToRecovered>()[{m_virus_variant, age}]); // TODO: subject to change
255 9 : previous_state = InfectionState::InfectedNoSymptoms;
256 : }
257 72 : else if (v < 0.5) { // TODO: subject to change
258 : time_period =
259 27 : days(params.get<InfectedSymptomsToRecovered>()[{m_virus_variant, age}]); // TODO: subject to change
260 27 : previous_state = InfectionState::InfectedSymptoms;
261 : }
262 45 : else if (v < 0.75) {
263 9 : time_period = days(params.get<SevereToRecovered>()[{m_virus_variant, age}]); // TODO: subject to change
264 9 : previous_state = InfectionState::InfectedSevere;
265 : }
266 : else {
267 : time_period =
268 36 : days(params.get<CriticalToRecovered>()[{m_virus_variant, age}]); // TODO: subject to change
269 36 : previous_state = InfectionState::InfectedCritical;
270 : }
271 81 : break;
272 :
273 36 : case InfectionState::Dead:
274 36 : time_period = days(params.get<CriticalToDead>()[{m_virus_variant, age}]); // TODO: subject to change
275 36 : previous_state = InfectionState::InfectedCritical;
276 36 : break;
277 :
278 0 : default:
279 0 : break;
280 : }
281 1062 : start_date = start_date - time_period;
282 1062 : m_infection_course.insert(m_infection_course.begin(), {start_date, previous_state});
283 : }
284 1008 : return start_date;
285 : }
286 :
287 : } // namespace abm
288 : } // namespace mio
|