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 : #include "memilio/config.h"
21 :
22 : #ifdef MEMILIO_HAS_JSONCPP
23 :
24 : #include "ode_secirvvs/parameters_io.h"
25 :
26 : namespace mio
27 : {
28 : namespace osecirvvs
29 : {
30 : namespace details
31 : {
32 : //gets the county or state id of the entry if available, 0 (for whole country) otherwise
33 : //used for comparisons of entry to integer region id
34 : template <class EpiDataEntry>
35 72036 : int get_region_id(const EpiDataEntry& rki_entry)
36 : {
37 164772 : return rki_entry.county_id ? rki_entry.county_id->get()
38 62100 : : (rki_entry.state_id ? rki_entry.state_id->get()
39 113436 : : (rki_entry.district_id ? rki_entry.district_id->get() : 0));
40 : }
41 :
42 9 : IOResult<void> read_confirmed_cases_data(
43 : std::string const& path, std::vector<int> const& vregion, Date date, std::vector<std::vector<double>>& vnum_Exposed,
44 : std::vector<std::vector<double>>& vnum_InfectedNoSymptoms, std::vector<std::vector<double>>& vnum_InfectedSymptoms,
45 : std::vector<std::vector<double>>& vnum_InfectedSevere, std::vector<std::vector<double>>& vnum_icu,
46 : std::vector<std::vector<double>>& vnum_death, std::vector<std::vector<double>>& vnum_rec,
47 : const std::vector<std::vector<int>>& vt_Exposed, const std::vector<std::vector<int>>& vt_InfectedNoSymptoms,
48 : const std::vector<std::vector<int>>& vt_InfectedSymptoms, const std::vector<std::vector<int>>& vt_InfectedSevere,
49 : const std::vector<std::vector<int>>& vt_InfectedCritical, const std::vector<std::vector<double>>& vmu_C_R,
50 : const std::vector<std::vector<double>>& vmu_I_H, const std::vector<std::vector<double>>& vmu_H_U,
51 : const std::vector<double>& scaling_factor_inf)
52 : {
53 9 : BOOST_OUTCOME_TRY(auto&& rki_data, mio::read_confirmed_cases_data(path));
54 : return read_confirmed_cases_data(rki_data, vregion, date, vnum_Exposed, vnum_InfectedNoSymptoms,
55 : vnum_InfectedSymptoms, vnum_InfectedSevere, vnum_icu, vnum_death, vnum_rec,
56 : vt_Exposed, vt_InfectedNoSymptoms, vt_InfectedSymptoms, vt_InfectedSevere,
57 9 : vt_InfectedCritical, vmu_C_R, vmu_I_H, vmu_H_U, scaling_factor_inf);
58 9 : }
59 :
60 279 : IOResult<void> read_confirmed_cases_data(
61 : const std::vector<ConfirmedCasesDataEntry>& rki_data, std::vector<int> const& vregion, Date date,
62 : std::vector<std::vector<double>>& vnum_Exposed, std::vector<std::vector<double>>& vnum_InfectedNoSymptoms,
63 : std::vector<std::vector<double>>& vnum_InfectedSymptoms, std::vector<std::vector<double>>& vnum_InfectedSevere,
64 : std::vector<std::vector<double>>& vnum_icu, std::vector<std::vector<double>>& vnum_death,
65 : std::vector<std::vector<double>>& vnum_rec, const std::vector<std::vector<int>>& vt_Exposed,
66 : const std::vector<std::vector<int>>& vt_InfectedNoSymptoms,
67 : const std::vector<std::vector<int>>& vt_InfectedSymptoms, const std::vector<std::vector<int>>& vt_InfectedSevere,
68 : const std::vector<std::vector<int>>& vt_InfectedCritical, const std::vector<std::vector<double>>& vmu_C_R,
69 : const std::vector<std::vector<double>>& vmu_I_H, const std::vector<std::vector<double>>& vmu_H_U,
70 : const std::vector<double>& scaling_factor_inf)
71 : {
72 279 : auto max_date_entry = std::max_element(rki_data.begin(), rki_data.end(), [](auto&& a, auto&& b) {
73 153729 : return a.date < b.date;
74 : });
75 279 : if (max_date_entry == rki_data.end()) {
76 0 : log_error("RKI data file is empty.");
77 0 : return failure(StatusCode::InvalidValue, "RKI data is empty.");
78 : }
79 279 : auto max_date = max_date_entry->date;
80 279 : if (max_date < date) {
81 0 : log_error("Specified date does not exist in RKI data");
82 0 : return failure(StatusCode::OutOfRange, "RKI data does not contain specified date.");
83 : }
84 :
85 : // shifts the initilization to the recent past if simulation starts
86 : // around current day and data of the future would be required.
87 : // Only needed for preinfection compartments, exposed and InfectedNoSymptoms.
88 279 : auto days_surplus = get_offset_in_days(max_date, date) - 6; // 6 > T_E + T_C
89 279 : if (days_surplus > 0) {
90 279 : days_surplus = 0;
91 : }
92 :
93 154287 : for (auto&& entry : rki_data) {
94 203688 : auto it = std::find_if(vregion.begin(), vregion.end(), [&entry](auto r) {
95 203688 : return r == 0 || get_region_id(entry) == r;
96 : });
97 154008 : if (it != vregion.end()) {
98 154008 : auto region_idx = size_t(it - vregion.begin());
99 :
100 154008 : auto& t_Exposed = vt_Exposed[region_idx];
101 154008 : auto& t_InfectedNoSymptoms = vt_InfectedNoSymptoms[region_idx];
102 154008 : auto& t_InfectedSymptoms = vt_InfectedSymptoms[region_idx];
103 154008 : auto& t_InfectedSevere = vt_InfectedSevere[region_idx];
104 154008 : auto& t_InfectedCritical = vt_InfectedCritical[region_idx];
105 :
106 154008 : auto& num_InfectedNoSymptoms = vnum_InfectedNoSymptoms[region_idx];
107 154008 : auto& num_InfectedSymptoms = vnum_InfectedSymptoms[region_idx];
108 154008 : auto& num_rec = vnum_rec[region_idx];
109 154008 : auto& num_Exposed = vnum_Exposed[region_idx];
110 154008 : auto& num_InfectedSevere = vnum_InfectedSevere[region_idx];
111 154008 : auto& num_death = vnum_death[region_idx];
112 154008 : auto& num_icu = vnum_icu[region_idx];
113 :
114 154008 : auto& mu_C_R = vmu_C_R[region_idx];
115 154008 : auto& mu_I_H = vmu_I_H[region_idx];
116 154008 : auto& mu_H_U = vmu_H_U[region_idx];
117 :
118 154008 : bool read_icu = false; // params.populations.get({age, SecirCompartments::U}) == 0;
119 :
120 154008 : auto age = (size_t)entry.age_group;
121 154008 : if (entry.date == offset_date_by_days(date, 0)) {
122 1188 : num_InfectedSymptoms[age] += scaling_factor_inf[age] * entry.num_confirmed;
123 1188 : num_rec[age] += entry.num_confirmed;
124 : }
125 154008 : if (entry.date == offset_date_by_days(date, t_InfectedNoSymptoms[age] + days_surplus)) {
126 1188 : num_InfectedNoSymptoms[age] += 1 / (1 - mu_C_R[age]) * scaling_factor_inf[age] * entry.num_confirmed;
127 1188 : num_Exposed[age] -= 1 / (1 - mu_C_R[age]) * scaling_factor_inf[age] * entry.num_confirmed;
128 : }
129 154008 : if (entry.date == offset_date_by_days(date, days_surplus)) {
130 1188 : num_InfectedNoSymptoms[age] -= 1 / (1 - mu_C_R[age]) * scaling_factor_inf[age] * entry.num_confirmed;
131 : }
132 154008 : if (entry.date == offset_date_by_days(date, t_Exposed[age] + t_InfectedNoSymptoms[age] + days_surplus)) {
133 1188 : num_Exposed[age] += 1 / (1 - mu_C_R[age]) * scaling_factor_inf[age] * entry.num_confirmed;
134 : }
135 154008 : if (entry.date == offset_date_by_days(date, -t_InfectedSymptoms[age])) {
136 1188 : num_InfectedSymptoms[age] -= scaling_factor_inf[age] * entry.num_confirmed;
137 1188 : num_InfectedSevere[age] += mu_I_H[age] * scaling_factor_inf[age] * entry.num_confirmed;
138 : }
139 154008 : if (entry.date == offset_date_by_days(date, -t_InfectedSymptoms[age] - t_InfectedSevere[age])) {
140 1188 : num_InfectedSevere[age] -= mu_I_H[age] * scaling_factor_inf[age] * entry.num_confirmed;
141 1188 : if (read_icu) {
142 0 : num_icu[age] += mu_I_H[age] * mu_H_U[age] * scaling_factor_inf[age] * entry.num_confirmed;
143 : }
144 : }
145 308016 : if (entry.date ==
146 308016 : offset_date_by_days(date, -t_InfectedSymptoms[age] - t_InfectedSevere[age] - t_InfectedCritical[age])) {
147 1188 : num_death[age] += entry.num_deaths;
148 1188 : if (read_icu) {
149 0 : num_icu[age] -= mu_I_H[age] * mu_H_U[age] * scaling_factor_inf[age] * entry.num_confirmed;
150 : }
151 : }
152 : }
153 : }
154 :
155 558 : for (size_t region_idx = 0; region_idx < vregion.size(); ++region_idx) {
156 279 : auto region = vregion[region_idx];
157 :
158 279 : auto& num_InfectedNoSymptoms = vnum_InfectedNoSymptoms[region_idx];
159 279 : auto& num_InfectedSymptoms = vnum_InfectedSymptoms[region_idx];
160 279 : auto& num_rec = vnum_rec[region_idx];
161 279 : auto& num_Exposed = vnum_Exposed[region_idx];
162 279 : auto& num_InfectedSevere = vnum_InfectedSevere[region_idx];
163 279 : auto& num_death = vnum_death[region_idx];
164 279 : auto& num_icu = vnum_icu[region_idx];
165 :
166 279 : size_t num_groups = ConfirmedCasesDataEntry::age_group_names.size();
167 1953 : for (size_t i = 0; i < num_groups; i++) {
168 : // subtract infected confirmed cases which are not yet recovered
169 : // and remove dark figure scaling factor
170 1674 : num_rec[i] -= num_InfectedSymptoms[i] / scaling_factor_inf[i];
171 1674 : num_rec[i] -= num_InfectedSevere[i] / scaling_factor_inf[i];
172 1674 : num_rec[i] -=
173 3348 : num_icu[i] /
174 1674 : scaling_factor_inf[i]; // TODO: this has to be adapted for scaling_factor_inf != 1 or != ***_icu
175 1674 : num_rec[i] -= num_death[i] / scaling_factor_inf[i];
176 1674 : auto try_fix_constraints = [region, i](double& value, double error, auto str) {
177 11718 : if (value < error) {
178 : // this should probably return a failure
179 : // but the algorithm is not robust enough to avoid large negative
180 : // values and there are tests that rely on it
181 0 : log_error("{:s} for age group {:s} is {:.4f} for region {:d}, "
182 : "exceeds expected negative value.",
183 0 : str, ConfirmedCasesDataEntry::age_group_names[i], value, region);
184 0 : value = 0.0;
185 : }
186 11718 : else if (value < 0) {
187 0 : log_info("{:s} for age group {:s} is {:.4f} for region {:d}, "
188 : "automatically corrected",
189 0 : str, ConfirmedCasesDataEntry::age_group_names[i], value, region);
190 0 : value = 0.0;
191 : }
192 11718 : };
193 :
194 1674 : try_fix_constraints(num_InfectedSymptoms[i], -5, "InfectedSymptoms");
195 1674 : try_fix_constraints(num_InfectedNoSymptoms[i], -5, "InfectedNoSymptoms");
196 1674 : try_fix_constraints(num_Exposed[i], -5, "Exposed");
197 1674 : try_fix_constraints(num_InfectedSevere[i], -5, "InfectedSevere");
198 1674 : try_fix_constraints(num_death[i], -5, "Dead");
199 1674 : try_fix_constraints(num_icu[i], -5, "InfectedCritical");
200 1674 : try_fix_constraints(num_rec[i], -20, "Recovered or vaccinated");
201 : }
202 : }
203 :
204 558 : return success();
205 : }
206 :
207 0 : IOResult<void> read_confirmed_cases_data_fix_recovered(std::string const& path, std::vector<int> const& vregion,
208 : Date date, std::vector<std::vector<double>>& vnum_rec,
209 : double delay)
210 : {
211 0 : BOOST_OUTCOME_TRY(auto&& rki_data, mio::read_confirmed_cases_data(path));
212 0 : return read_confirmed_cases_data_fix_recovered(rki_data, vregion, date, vnum_rec, delay);
213 0 : }
214 :
215 108 : IOResult<void> read_confirmed_cases_data_fix_recovered(const std::vector<ConfirmedCasesDataEntry>& rki_data,
216 : std::vector<int> const& vregion, Date date,
217 : std::vector<std::vector<double>>& vnum_rec, double delay)
218 : {
219 108 : auto max_date_entry = std::max_element(rki_data.begin(), rki_data.end(), [](auto&& a, auto&& b) {
220 59508 : return a.date < b.date;
221 : });
222 108 : if (max_date_entry == rki_data.end()) {
223 0 : log_error("RKI data is empty.");
224 0 : return failure(StatusCode::InvalidValue, "RKI data is empty.");
225 : }
226 108 : auto max_date = max_date_entry->date;
227 108 : if (max_date < date) {
228 0 : log_error("Specified date does not exist in RKI data");
229 0 : return failure(StatusCode::OutOfRange, "RKI data does not contain specified date.");
230 : }
231 :
232 : // shifts the initilization to the recent past if simulation starts
233 : // around current day and data of the future would be required.
234 : // Only needed for preinfection compartments, exposed and InfectedNoSymptoms.
235 108 : auto days_surplus = get_offset_in_days(max_date, date) - 6; // 6 > T_E^C + T_C^I
236 108 : if (days_surplus > 0) {
237 108 : days_surplus = 0;
238 : }
239 :
240 59724 : for (auto&& rki_entry : rki_data) {
241 79488 : auto it = std::find_if(vregion.begin(), vregion.end(), [&rki_entry](auto r) {
242 79488 : return r == 0 || get_region_id(rki_entry) == r;
243 : });
244 59616 : if (it != vregion.end()) {
245 54648 : auto region_idx = size_t(it - vregion.begin());
246 54648 : if (rki_entry.date == offset_date_by_days(date, int(-delay))) {
247 378 : vnum_rec[region_idx][size_t(rki_entry.age_group)] = rki_entry.num_confirmed;
248 : }
249 : }
250 : }
251 :
252 216 : for (size_t region_idx = 0; region_idx < vregion.size(); ++region_idx) {
253 108 : auto region = vregion[region_idx];
254 108 : auto& num_rec = vnum_rec[region_idx];
255 :
256 108 : size_t num_groups = ConfirmedCasesDataEntry::age_group_names.size();
257 756 : for (size_t i = 0; i < num_groups; i++) {
258 648 : auto try_fix_constraints = [region, i](double& value, double error, auto str) {
259 648 : if (value < error) {
260 : // this should probably return a failure
261 : // but the algorithm is not robust enough to avoid large negative
262 : // values and there are tests that rely on it
263 0 : log_error("{:s} for age group {:s} is {:.4f} for region {:d}, "
264 : "exceeds expected negative value.",
265 0 : str, ConfirmedCasesDataEntry::age_group_names[i], value, region);
266 0 : value = 0.0;
267 : }
268 648 : else if (value < 0) {
269 0 : log_info("{:s} for age group {:s} is {:.4f} for region {:d}, "
270 : "automatically corrected",
271 0 : str, ConfirmedCasesDataEntry::age_group_names[i], value, region);
272 0 : value = 0.0;
273 : }
274 648 : };
275 648 : try_fix_constraints(num_rec[i], 0, "Recovered");
276 : }
277 : }
278 :
279 216 : return success();
280 : }
281 :
282 63 : IOResult<void> read_divi_data(const std::string& path, const std::vector<int>& vregion, Date date,
283 : std::vector<double>& vnum_icu)
284 : {
285 63 : BOOST_OUTCOME_TRY(auto&& divi_data, mio::read_divi_data(path));
286 63 : return read_divi_data(divi_data, vregion, date, vnum_icu);
287 63 : }
288 :
289 63 : IOResult<void> read_divi_data(const std::vector<DiviEntry>& divi_data, const std::vector<int>& vregion, Date date,
290 : std::vector<double>& vnum_icu)
291 : {
292 63 : auto max_date_entry = std::max_element(divi_data.begin(), divi_data.end(), [](auto&& a, auto&& b) {
293 5733 : return a.date < b.date;
294 : });
295 63 : if (max_date_entry == divi_data.end()) {
296 0 : log_error("DIVI data is empty.");
297 0 : return failure(StatusCode::InvalidValue, "DIVI data is empty.");
298 : }
299 63 : auto max_date = max_date_entry->date;
300 63 : if (max_date < date) {
301 0 : log_error("DIVI data does not contain the specified date.");
302 0 : return failure(StatusCode::OutOfRange, "DIVI data does not contain the specified date.");
303 : }
304 :
305 5859 : for (auto&& entry : divi_data) {
306 8280 : auto it = std::find_if(vregion.begin(), vregion.end(), [&entry](auto r) {
307 8280 : return r == 0 || r == get_region_id(entry);
308 : });
309 5796 : auto date_df = entry.date;
310 5796 : if (it != vregion.end() && date_df == date) {
311 63 : auto region_idx = size_t(it - vregion.begin());
312 63 : vnum_icu[region_idx] = entry.num_icu;
313 : }
314 : }
315 :
316 126 : return success();
317 : }
318 :
319 135 : IOResult<std::vector<std::vector<double>>> read_population_data(const std::string& path,
320 : const std::vector<int>& vregion)
321 : {
322 135 : BOOST_OUTCOME_TRY(auto&& population_data, mio::read_population_data(path));
323 135 : return read_population_data(population_data, vregion);
324 135 : }
325 :
326 135 : IOResult<std::vector<std::vector<double>>> read_population_data(const std::vector<PopulationDataEntry>& population_data,
327 : const std::vector<int>& vregion)
328 : {
329 135 : std::vector<std::vector<double>> vnum_population(
330 540 : vregion.size(), std::vector<double>(ConfirmedCasesDataEntry::age_group_names.size(), 0.0));
331 :
332 54270 : for (auto&& county_entry : population_data) {
333 : //accumulate population of states or country from population of counties
334 54135 : if (!county_entry.county_id && !county_entry.district_id) {
335 0 : return failure(StatusCode::InvalidFileFormat, "File with county population expected.");
336 : }
337 : //find region that this county belongs to
338 : //all counties belong to the country (id = 0)
339 191241 : auto it = std::find_if(vregion.begin(), vregion.end(), [&county_entry](auto r) {
340 14436 : return r == 0 ||
341 54135 : (county_entry.county_id &&
342 97443 : regions::StateId(r) == regions::get_state_id(int(*county_entry.county_id))) ||
343 187650 : (county_entry.county_id && regions::CountyId(r) == *county_entry.county_id) ||
344 158742 : (county_entry.district_id && regions::DistrictId(r) == *county_entry.district_id);
345 : });
346 54135 : if (it != vregion.end()) {
347 39726 : auto region_idx = size_t(it - vregion.begin());
348 39726 : auto& num_population = vnum_population[region_idx];
349 278082 : for (size_t age = 0; age < num_population.size(); age++) {
350 238356 : num_population[age] += county_entry.population[AgeGroup(age)];
351 : }
352 : }
353 : }
354 :
355 135 : return success(vnum_population);
356 135 : }
357 :
358 : } // namespace details
359 : } // namespace osecirvvs
360 : } // namespace mio
361 :
362 : #endif // MEMILIO_HAS_JSONCPP
|