diff --git a/.vscode/settings.json b/.vscode/settings.json index 6ac87e9e5..639f0a55d 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -63,7 +63,12 @@ "thread": "cpp", "typeinfo": "cpp", "variant": "cpp", - "semaphore": "cpp" + "semaphore": "cpp", + "list": "cpp", + "unordered_set": "cpp", + "shared_mutex": "cpp", + "cfenv": "cpp", + "cinttypes": "cpp" }, "intel-corporation.oneapi-analysis-configurator.binary-path": "/home/george/Documents/development/epiworld/tests/01c.o" } \ No newline at end of file diff --git a/README.md b/README.md index 38086ab53..b12b6e249 100644 --- a/README.md +++ b/README.md @@ -137,7 +137,7 @@ int main() virus.set_status(1, 2); - model.add_virus_n(virus, 1000); + model.default_add_virusn(virus, 1000); // Generating a random pop from a smallworld network model.agents_smallworld(100000, 4L, false, .01); diff --git a/epiworld-hpp.R b/epiworld-hpp.R index f14430bc1..8f36341e2 100644 --- a/epiworld-hpp.R +++ b/epiworld-hpp.R @@ -37,7 +37,7 @@ unfolder <- function(txt, rel = "include/epiworld/") { loc <- heads[h] - fn <- paste0(rel, fns[h]) + fn <- trimws(paste0(rel, fns[h])) tmp_lines <- readLines(fn, warn = FALSE) # Extracting relative path diff --git a/epiworld.hpp b/epiworld.hpp index c263f4098..a75502370 100644 --- a/epiworld.hpp +++ b/epiworld.hpp @@ -38,7 +38,7 @@ namespace epiworld { #define EPIWORLD_MAXNEIGHBORS 1048576 #endif -#ifdef _OPENMP +#if defined(_OPENMP) || defined(__OPENMP) #include // #else // #define omp_get_thread_num() 0 @@ -288,7 +288,9 @@ struct Action { if (a) \ {\ throw EPI_DEBUG_ERROR(std::logic_error, b); \ - } + } + + #define epiexception(a) std::logic_error #else #define EPI_DEBUG_PRINTF(fmt, ...) #define EPI_DEBUG_ERROR(fmt, ...) @@ -300,9 +302,10 @@ struct Action { #define EPI_DEBUG_FAIL_AT_TRUE(a, b) \ if (a) \ return false; + #define epiexception(a) a #endif -#ifdef EPI_DEBUG_NO_THREAD_ID +#if defined(EPI_DEBUG_NO_THREAD_ID) || (!defined(__OPENMP) && !defined(_OPENMP)) #define EPI_GET_THREAD_ID() 0 #else #define EPI_GET_THREAD_ID() omp_get_thread_num() @@ -2655,6 +2658,9 @@ inline void default_rm_virus(Action & a, Model * m); template inline void default_rm_tool(Action & a, Model * m); +template +inline void default_change_state(Action & a, Model * m); + /** * @brief Statistical data about the process * @@ -2667,6 +2673,7 @@ class DataBase { friend void default_add_tool(Action & a, Model * m); friend void default_rm_virus(Action & a, Model * m); friend void default_rm_tool(Action & a, Model * m); + friend void default_change_state(Action & a, Model * m); private: Model * model; @@ -6059,7 +6066,6 @@ class Model { bool using_backup = true; std::vector< Agent > population_backup = {}; - /** * @name Auxiliary variables for AgentsSample iterators * @@ -6125,8 +6131,13 @@ class Model { epiworld_fast_uint ndays = 0; Progress pb; - std::vector< UpdateFun > state_fun = {}; - std::vector< std::string > states_labels = {}; + std::vector< UpdateFun > state_fun = {}; ///< Functions to update states + std::vector< std::string > states_labels = {}; ///< Labels of the states + + /** Function to distribute states. Goes along with the function */ + std::function*)> initial_states_fun = [](Model * /**/) + -> void {}; + epiworld_fast_uint nstates = 0u; bool verbose = true; @@ -6176,20 +6187,13 @@ class Model { VirusPtr virus_, ToolPtr tool_, Entity * entity_, - epiworld_fast_uint new_state_, + epiworld_fast_int new_state_, epiworld_fast_int queue_, ActionFun call_, int idx_agent_, int idx_object_ ); - /** - * @brief Executes the stored action - * - * @param model_ Model over which it will be executed. - */ - void actions_run(); - /** * @name Tool Mixers * @@ -6213,12 +6217,13 @@ class Model { public: + std::vector array_double_tmp; std::vector * > array_virus_tmp; Model(); Model(const Model & m); - Model(Model & m) = delete; + Model(Model & m); Model(Model && m); Model & operator=(const Model & m); @@ -6356,7 +6361,7 @@ class Model { std::vector< Entity > & get_entities(); - void agents_smallworld( + Model & agents_smallworld( epiworld_fast_uint n = 1000, epiworld_fast_uint k = 5, bool d = false, @@ -6378,7 +6383,7 @@ class Model { void update_state(); void mutate_virus(); void next(); - virtual void run( + virtual Model & run( epiworld_fast_uint ndays, int seed = -1 ); ///< Runs the simulation (after initialization) @@ -6393,14 +6398,14 @@ class Model { ); ///@} - size_t get_n_viruses() const; - size_t get_n_tools() const; + size_t get_n_viruses() const; ///< Number of viruses in the model + size_t get_n_tools() const; ///< Number of tools in the model epiworld_fast_uint get_ndays() const; epiworld_fast_uint get_n_replicates() const; void set_ndays(epiworld_fast_uint ndays); bool get_verbose() const; - void verbose_off(); - void verbose_on(); + Model & verbose_off(); + Model & verbose_on(); int today() const; ///< The current time of the model /** @@ -6480,7 +6485,7 @@ class Model { * */ virtual void reset(); - void print(bool lite = false) const; + const Model & print(bool lite = false) const; Model && clone() const; @@ -6505,6 +6510,19 @@ class Model { void print_state_codes() const; ///@} + /** + * @name Initial states + * + * @details These functions are called before the simulation starts. + * + * @param proportions_ Vector of proportions for each state. + * @param queue_ Vector of queue for each state. + */ + virtual Model & initial_states( + std::vector< double > /*proportions_*/, + std::vector< int > /*queue_*/ + ) {return *this;}; + /** * @name Setting and accessing parameters from the model * @@ -6610,7 +6628,7 @@ class Model { */ ////@{ void queuing_on(); ///< Activates the queuing system (default.) - void queuing_off(); ///< Deactivates the queuing system. + Model & queuing_off(); ///< Deactivates the queuing system. bool is_queuing_on() const; ///< Query if the queuing system is on. Queue & get_queue(); ///< Retrieve the `Queue` object. ///@} @@ -6629,6 +6647,8 @@ class Model { ///@} const std::vector< VirusPtr > & get_viruses() const; + const std::vector< epiworld_double > & get_prevalence_virus() const; + const std::vector< bool > & get_prevalence_virus_as_proportion() const; const std::vector< ToolPtr > & get_tools() const; Virus & get_virus(size_t id); Tool & get_tool(size_t id); @@ -6659,6 +6679,14 @@ class Model { bool operator==(const Model & other) const; bool operator!=(const Model & other) const {return !operator==(other);}; + /** + * @brief Executes the stored action + * + * @param model_ Model over which it will be executed. + */ + void actions_run(); + + }; #endif @@ -6683,8 +6711,6 @@ class Model { #ifndef EPIWORLD_MODEL_MEAT_HPP #define EPIWORLD_MODEL_MEAT_HPP - - /** * @brief Function factory for saving model runs * @@ -6832,38 +6858,31 @@ inline std::function*)> make_save_run( return saver; } + template inline void Model::actions_add( Agent * agent_, VirusPtr virus_, ToolPtr tool_, Entity * entity_, - epiworld_fast_uint new_state_, + epiworld_fast_int new_state_, epiworld_fast_int queue_, ActionFun call_, int idx_agent_, int idx_object_ ) { - + ++nactions; #ifdef EPI_DEBUG if (nactions == 0) throw std::logic_error("Actions cannot be zero!!"); - - if ((virus_ != nullptr) && idx_agent_ >= 0) - { - if (idx_agent_ >= static_cast(virus_->get_agent()->get_n_viruses())) - throw std::logic_error( - "The virus to add is out of range in the host agent." - ); - } #endif if (nactions > actions.size()) { - actions.push_back( + actions.emplace_back( Action( agent_, virus_, tool_, entity_, new_state_, queue_, call_, idx_agent_, idx_object_ @@ -6879,7 +6898,7 @@ inline void Model::actions_add( A.virus = virus_; A.tool = tool_; A.entity = entity_; - A.new_state = new_state_; + A.new_state = new_state_; A.queue = queue_; A.call = call_; A.idx_agent = idx_agent_; @@ -6895,76 +6914,46 @@ template inline void Model::actions_run() { // Making the call - while (nactions != 0u) + size_t nactions_tmp = 0; + while (nactions_tmp < nactions) { - Action a = actions[--nactions]; + Action & a = actions[nactions_tmp++]; Agent * p = a.agent; - // Applying function - if (a.call) - { - a.call(a, this); - } - - // Updating state - if (static_cast(p->state) != a.new_state) - { - - if (a.new_state >= static_cast(nstates)) - throw std::range_error( - "The proposed state " + std::to_string(a.new_state) + " is out of range. " + - "The model currently has " + std::to_string(nstates - 1) + " states."); - - // Figuring out if we need to undo a change - // If the agent has made a change in the state recently, then we - // need to undo the accounting, e.g., if A->B was made, we need to - // undo it and set B->A so that the daily accounting is right. - if (p->state_last_changed == today()) - { - - // Updating accounting - db.update_state(p->state_prev, p->state, true); // Undoing - db.update_state(p->state_prev, a.new_state); - - for (size_t v = 0u; v < p->n_viruses; ++v) - { - db.update_virus(p->viruses[v]->id, p->state, p->state_prev); // Undoing - db.update_virus(p->viruses[v]->id, p->state_prev, a.new_state); - } - - for (size_t t = 0u; t < p->n_tools; ++t) - { - db.update_tool(p->tools[t]->id, p->state, p->state_prev); // Undoing - db.update_tool(p->tools[t]->id, p->state_prev, a.new_state); - } - - // Changing to the new state, we won't update the - // previous state in case we need to undo the change - p->state = a.new_state; - - } else { - - // Updating accounting - db.update_state(p->state, a.new_state); - - for (size_t v = 0u; v < p->n_viruses; ++v) - db.update_virus(p->viruses[v]->id, p->state, a.new_state); + #ifdef EPI_DEBUG + if (a.new_state >= static_cast(nstates)) + throw std::range_error( + "The proposed state " + std::to_string(a.new_state) + " is out of range. " + + "The model currently has " + std::to_string(nstates - 1) + " states."); - for (size_t t = 0u; t < p->n_tools; ++t) - db.update_tool(p->tools[t]->id, p->state, a.new_state); + if (a.new_state < 0) + throw std::range_error( + "The proposed state " + std::to_string(a.new_state) + " is out of range. " + + "The state cannot be negative."); + #endif - // Saving the last state and setting the new one - p->state_prev = p->state; - p->state = a.new_state; + // Undoing the change in the transition matrix + if ((p->state_last_changed == today()) && (static_cast(p->state) != a.new_state)) + { + // Undoing state change in the transition matrix + // The previous state is already recorded + db.update_state(p->state_prev, p->state, true); - // It used to be a day before, but we still - p->state_last_changed = today(); + } else + p->state_prev = p->state; // Recording the previous state - } - + // Applying function after the fact. This way, if there were + // updates, they can be recorded properly, before losing the information + p->state = a.new_state; + if (a.call) + { + a.call(a, this); } + // Registering that the last change was today + p->state_last_changed = today(); + #ifdef EPI_DEBUG if (static_cast(p->state) >= static_cast(nstates)) throw std::range_error( @@ -6993,6 +6982,9 @@ inline void Model::actions_run() } + // Go back to square 1 + nactions = 0u; + return; } @@ -7117,6 +7109,7 @@ inline Model::Model(const Model & model) : pb(model.pb), state_fun(model.state_fun), states_labels(model.states_labels), + initial_states_fun(model.initial_states_fun), nstates(model.nstates), verbose(model.verbose), current_date(model.current_date), @@ -7157,6 +7150,10 @@ inline Model::Model(const Model & model) : } +template +inline Model::Model(Model & model) : + Model(dynamic_cast< const Model & >(model)) {} + template inline Model::Model(Model && model) : name(std::move(model.name)), @@ -7197,6 +7194,7 @@ inline Model::Model(Model && model) : pb(std::move(model.pb)), state_fun(std::move(model.state_fun)), states_labels(std::move(model.states_labels)), + initial_states_fun(std::move(model.initial_states_fun)), nstates(model.nstates), verbose(model.verbose), current_date(std::move(model.current_date)), @@ -7268,6 +7266,7 @@ inline Model & Model::operator=(const Model & m) state_fun = m.state_fun; states_labels = m.states_labels; + initial_states_fun = m.initial_states_fun; nstates = m.nstates; verbose = m.verbose; @@ -7330,7 +7329,7 @@ inline std::vector< Viruses_const > Model::get_agents_viruses() cons std::vector< Viruses_const > viruses(population.size()); for (size_t i = 0u; i < population.size(); ++i) - viruses[i] = population[i].get_viruses(); + viruses[i] = population[i].get_virus(); return viruses; @@ -7343,7 +7342,7 @@ inline std::vector< Viruses > Model::get_agents_viruses() std::vector< Viruses > viruses(population.size()); for (size_t i = 0u; i < population.size(); ++i) - viruses[i] = population[i].get_viruses(); + viruses[i] = population[i].get_virus(); return viruses; @@ -7356,7 +7355,7 @@ inline std::vector> & Model::get_entities() } template -inline void Model::agents_smallworld( +inline Model & Model::agents_smallworld( epiworld_fast_uint n, epiworld_fast_uint k, bool d, @@ -7366,6 +7365,8 @@ inline void Model::agents_smallworld( agents_from_adjlist( rgraph_smallworld(n, k, p, d, *this) ); + + return *this; } template @@ -7452,10 +7453,9 @@ inline void Model::dist_virus() // Starting first infection int n = size(); - std::vector< size_t > idx(n); - - int n_left = n; + std::vector< size_t > idx(n, 0u); std::iota(idx.begin(), idx.end(), 0); + int n_left = idx.size(); for (size_t v = 0u; v < viruses.size(); ++v) { @@ -7493,7 +7493,7 @@ inline void Model::dist_virus() Agent & agent = population[idx[loc]]; // Adding action - agent.add_virus( + agent.set_virus( virus, const_cast * >(this), virus->state_init, @@ -8170,7 +8170,7 @@ inline void Model::next() { } template -inline void Model::run( +inline Model & Model::run( epiworld_fast_uint ndays, int seed ) @@ -8276,6 +8276,8 @@ inline void Model::run( chrono_end(); + return *this; + } template @@ -8506,6 +8508,15 @@ inline void Model::update_state() { template inline void Model::mutate_virus() { + // Checking if any virus has mutation + size_t nmutates = 0u; + for (const auto & v: viruses) + if (v->mutation_fun) + nmutates++; + + if (nmutates == 0u) + return; + if (use_queuing) { @@ -8516,9 +8527,8 @@ inline void Model::mutate_virus() { if (queue[++i] == 0) continue; - if (p.n_viruses > 0u) - for (auto & v : p.get_viruses()) - v->mutate(this); + if (p.virus != nullptr) + p.virus->mutate(this); } @@ -8529,9 +8539,8 @@ inline void Model::mutate_virus() { for (auto & p: population) { - if (p.n_viruses > 0u) - for (auto & v : p.get_viruses()) - v->mutate(this); + if (p.virus != nullptr) + p.virus->mutate(this); } @@ -8572,13 +8581,15 @@ inline bool Model::get_verbose() const { } template -inline void Model::verbose_on() { +inline Model & Model::verbose_on() { verbose = true; + return *this; } template -inline void Model::verbose_off() { +inline Model & Model::verbose_off() { verbose = false; + return *this; } template @@ -8787,6 +8798,9 @@ inline void Model::reset() { dist_virus(); dist_tools(); + // Distributing initial state, if specified + initial_states_fun(this); + // Recording the original state (at time 0) and advancing // to time 1 next(); @@ -8808,7 +8822,7 @@ inline void Model::reset() { #define EPIWORLD_MODEL_MEAT_PRINT_HPP template -inline void Model::print(bool lite) const +inline const Model & Model::print(bool lite) const { // Horizontal line @@ -8864,7 +8878,7 @@ inline void Model::print(bool lite) const printf_epiworld(" The model hasn't been run yet.\n"); } - return; + return *this; } printf_epiworld("%s\n%s\n\n",line.c_str(), "SIMULATION STUDY"); @@ -9130,7 +9144,7 @@ inline void Model::print(bool lite) const if (today() != 0) (void) db.transition_probability(true); - return; + return *this; } @@ -9561,9 +9575,10 @@ inline void Model::queuing_on() } template -inline void Model::queuing_off() +inline Model & Model::queuing_off() { use_queuing = false; + return *this; } template @@ -9584,6 +9599,18 @@ inline const std::vector< VirusPtr > & Model::get_viruses() const return viruses; } +template +inline const std::vector< epiworld_double > & Model::get_prevalence_virus() const +{ + return prevalence_virus; +} + +template +inline const std::vector< bool > & Model::get_prevalence_virus_as_proportion() const +{ + return prevalence_virus_as_proportion; +} + template const std::vector< ToolPtr > & Model::get_tools() const { @@ -10130,8 +10157,6 @@ class Virus { private: Agent * agent = nullptr; - int pos_in_agent = -99; ///< Location in the agent - int agent_exposure_number = -99; std::shared_ptr baseline_sequence = nullptr; std::shared_ptr virus_name = nullptr; @@ -10146,7 +10171,6 @@ class Virus { VirusFun incubation_fun = nullptr; // Setup parameters - std::vector< epiworld_double * > params = {}; std::vector< epiworld_double > data = {}; epiworld_fast_int state_init = -99; ///< Change of state when added to agent. @@ -10167,7 +10191,7 @@ class Virus { void set_sequence(TSeq sequence); Agent * get_agent(); - void set_agent(Agent * p, epiworld_fast_uint idx); + void set_agent(Agent * p); void set_date(int d); int get_date() const; @@ -10347,7 +10371,9 @@ inline VirusFun virus_fun_logit( size_t K = coefs_f.size(); epiworld_double res = 0.0; + #if defined(__OPENMP) || defined(_OPENMP) #pragma omp simd reduction(+:res) + #endif for (size_t i = 0u; i < K; ++i) res += agent->operator[](vars.at(i)) * coefs_f.at(i); @@ -10410,21 +10436,9 @@ inline Agent * Virus::get_agent() } template -inline void Virus::set_agent(Agent * p, epiworld_fast_uint idx) +inline void Virus::set_agent(Agent * p) { - - #ifdef EPI_DEBUG - if (idx >= p->viruses.size()) - { - printf_epiworld( - "[epi-debug]Virus::set_agent id to set up is outside of range." - ); - } - #endif - - agent = p; - pos_in_agent = static_cast(idx); - + agent = p; } template @@ -11425,7 +11439,9 @@ inline ToolFun tool_fun_logit( size_t K = coefs_f.size(); epiworld_double res = 0.0; + #if defined(__OPENMP) || defined(_OPENMP) #pragma omp simd reduction(+:res) + #endif for (size_t i = 0u; i < K; ++i) res += agent->operator[](vars.at(i)) * coefs_f.at(i); @@ -11997,28 +12013,6 @@ class Entity { #ifndef EPIWORLD_ENTITY_MEAT_HPP #define EPIWORLD_ENTITY_MEAT_HPP -// template -// inline Entity::Entity(const Entity & e) : -// model(e.model), -// id(e.id), -// agents(0u), -// agents_location(0u), -// n_agents(0), -// sampled_agents(0u), -// sampled_agents_n(0u), -// sampled_agents_left(0u), -// sampled_agents_left_n(0u), -// max_capacity(e.max_capacity), -// entity_name(e.entity_name), -// location(e.location), -// state_init(e.state_init), -// state_post(e.state_post), -// queue_init(e.queue_init), -// queue_post(e.queue_post) -// { - -// } - template inline void Entity::add_agent( Agent & p, @@ -12546,37 +12540,31 @@ inline std::function*,Model*)> make_update_susceptible( [](Agent * p, Model * m) -> void { - if (p->get_n_viruses() > 0u) + if (p->get_virus() != nullptr) throw std::logic_error( std::string("Using the -default_update_susceptible- on agents WITH viruses makes no sense! ") + std::string("Agent id ") + std::to_string(p->get_id()) + - std::string(" has ") + std::to_string(p->get_n_viruses()) + - std::string(" viruses.") + std::string(" has a virus.") ); // This computes the prob of getting any neighbor variant size_t nviruses_tmp = 0u; for (auto & neighbor: p->get_neighbors()) { - - for (const VirusPtr & v : neighbor->get_viruses()) - { - - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) - ; - m->array_virus_tmp[nviruses_tmp++] = &(*v); + auto & v = neighbor->get_virus(); + if (v == nullptr) + continue; + + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); - } } // No virus to compute @@ -12589,7 +12577,7 @@ inline std::function*,Model*)> make_update_susceptible( if (which < 0) return; - p->add_virus(*m->array_virus_tmp[which], m); + p->set_virus(*m->array_virus_tmp[which], m); return; }; @@ -12630,12 +12618,11 @@ inline std::function*,Model*)> make_update_susceptible( } - if (p->get_n_viruses() > 0u) + if (p->get_virus() != nullptr) throw std::logic_error( std::string("Using the -default_update_susceptible- on agents WITH viruses makes no sense! ") + std::string("Agent id ") + std::to_string(p->get_id()) + - std::string(" has ") + std::to_string(p->get_n_viruses()) + - std::string(" viruses.") + std::string(" has a virus.") ); // This computes the prob of getting any neighbor variant @@ -12646,26 +12633,21 @@ inline std::function*,Model*)> make_update_susceptible( // If the state is in the list, exclude it if (exclude_agent_bool->operator[](neighbor->get_state())) continue; - - for (const VirusPtr & v : neighbor->get_viruses()) - { - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - - #endif + auto & v = neighbor->get_virus(); + if (v == nullptr) + continue; - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) - ; + + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); - m->array_virus_tmp[nviruses_tmp++] = &(*v); - - } } // No virus to compute @@ -12678,7 +12660,7 @@ inline std::function*,Model*)> make_update_susceptible( if (which < 0) return; - p->add_virus(*m->array_virus_tmp[which], m); + p->set_virus(*m->array_virus_tmp[which], m); return; @@ -12714,37 +12696,37 @@ inline std::function*(Agent*,Model*)> make_sample_virus_ std::function*(Agent*,Model*)> res = [](Agent * p, Model * m) -> Virus* { - if (p->get_n_viruses() > 0u) + if (p->get_virus() != nullptr) throw std::logic_error( std::string("Using the -default_update_susceptible- on agents WITH viruses makes no sense! ") + std::string("Agent id ") + std::to_string(p->get_id()) + - std::string(" has ") + std::to_string(p->get_n_viruses()) + - std::string(" viruses.") + std::string(" has a virus.") ); // This computes the prob of getting any neighbor variant size_t nviruses_tmp = 0u; for (auto & neighbor: p->get_neighbors()) { - - for (const VirusPtr & v : neighbor->get_viruses()) - { - - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) - ; - m->array_virus_tmp[nviruses_tmp++] = &(*v); + if (neighbor->get_virus() == nullptr) + continue; + + auto & v = neighbor->get_virus(); + + #ifdef EPI_DEBUG + if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif - } + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); + } // No virus to compute @@ -12798,12 +12780,11 @@ inline std::function*(Agent*,Model*)> make_sample_virus_ } - if (p->get_n_viruses() > 0u) + if (p->get_virus() != nullptr) throw std::logic_error( std::string("Using the -default_update_susceptible- on agents WITH viruses makes no sense! ") + std::string("Agent id ") + std::to_string(p->get_id()) + - std::string(" has ") + std::to_string(p->get_n_viruses()) + - std::string(" viruses.") + std::string(" has a virus.") ); // This computes the prob of getting any neighbor variant @@ -12814,25 +12795,26 @@ inline std::function*(Agent*,Model*)> make_sample_virus_ // If the state is in the list, exclude it if (exclude_agent_bool->operator[](neighbor->get_state())) continue; - - for (const VirusPtr & v : neighbor->get_viruses()) - { - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif + if (neighbor->get_virus() == nullptr) + continue; + + auto & v = neighbor->get_virus(); - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) - ; - - m->array_virus_tmp[nviruses_tmp++] = &(*v); + #ifdef EPI_DEBUG + if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif - } + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); + } // No virus to compute @@ -12875,12 +12857,11 @@ template inline Virus * sample_virus_single(Agent * p, Model * m) { - if (p->get_n_viruses() > 0u) + if (p->get_virus() != nullptr) throw std::logic_error( - std::string("Using the -default_update_susceptible- on agents WITH viruses makes no sense! ") + + std::string("Using the -default_update_susceptible- on agents WITH viruses makes no sense!") + std::string("Agent id ") + std::to_string(p->get_id()) + - std::string(" has ") + std::to_string(p->get_n_viruses()) + - std::string(" viruses.") + std::string(" has a virus.") ); // This computes the prob of getting any neighbor variant @@ -12889,40 +12870,42 @@ inline Virus * sample_virus_single(Agent * p, Model * m) { #ifdef EPI_DEBUG int _vcount_neigh = 0; - #endif - for (const VirusPtr & v : neighbor->get_viruses()) - { + #endif - #ifdef EPI_DEBUG - if (nviruses_tmp >= m->array_virus_tmp.size()) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) - ; - - m->array_virus_tmp[nviruses_tmp++] = &(*v); + if (neighbor->get_virus() == nullptr) + continue; - #ifdef EPI_DEBUG - if ( - (m->array_double_tmp[nviruses_tmp - 1] < 0.0) | - (m->array_double_tmp[nviruses_tmp - 1] > 1.0) - ) - { - printf_epiworld( - "[epi-debug] Agent %i's virus %i has transmission prob outside of [0, 1]: %.4f!\n", - static_cast(neighbor->get_id()), - static_cast(_vcount_neigh++), - m->array_double_tmp[nviruses_tmp - 1] - ); - } - #endif + auto & v = neighbor->get_virus(); + + #ifdef EPI_DEBUG + if (nviruses_tmp >= m->array_virus_tmp.size()) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif + + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); + + #ifdef EPI_DEBUG + if ( + (m->array_double_tmp[nviruses_tmp - 1] < 0.0) | + (m->array_double_tmp[nviruses_tmp - 1] > 1.0) + ) + { + printf_epiworld( + "[epi-debug] Agent %i's virus %i has transmission prob outside of [0, 1]: %.4f!\n", + static_cast(neighbor->get_id()), + static_cast(_vcount_neigh++), + m->array_double_tmp[nviruses_tmp - 1] + ); + } + #endif - } } @@ -12974,7 +12957,7 @@ inline void default_update_susceptible( if (virus == nullptr) return; - p->add_virus(*virus, m); + p->set_virus(*virus, m); return; @@ -12983,59 +12966,37 @@ inline void default_update_susceptible( template inline void default_update_exposed(Agent * p, Model * m) { - if (p->get_n_viruses() == 0u) + if (p->get_virus() == nullptr) throw std::logic_error( std::string("Using the -default_update_exposed- on agents WITHOUT viruses makes no sense! ") + std::string("Agent id ") + std::to_string(p->get_id()) + std::string(" has no virus registered.") ); - // Odd: Die, Even: Recover - epiworld_fast_uint n_events = 0u; - for (const auto & v : p->get_viruses()) - { - - // Die - m->array_double_tmp[n_events++] = - v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); + // Die + auto & virus = p->get_virus(); + m->array_double_tmp[0u] = + virus->get_prob_death(m) * (1.0 - p->get_death_reduction(virus, m)); - // Recover - m->array_double_tmp[n_events++] = - 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); - - } - - #ifdef EPI_DEBUG - if (n_events == 0u) - { - printf_epiworld( - "[epi-debug] agent %i has 0 possible events!!\n", - static_cast(p->get_id()) - ); - throw std::logic_error("Zero events in exposed."); - } - #else - if (n_events == 0u) - return; - #endif + // Recover + m->array_double_tmp[1u] = + 1.0 - (1.0 - virus->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(virus, m)); // Running the roulette - int which = roulette(n_events, m); + int which = roulette(2u, m); if (which < 0) return; // Which roulette happen? - if ((which % 2) == 0) // If odd + if (which == 0u) // If odd { - size_t which_v = std::ceil(which / 2); - p->rm_agent_by_virus(which_v, m); + p->rm_agent_by_virus(m); } else { - size_t which_v = std::floor(which / 2); - p->rm_virus(which_v, m); + p->rm_virus(m); } @@ -13116,6 +13077,9 @@ inline void default_rm_tool(Action & a, Model * m); template inline void default_rm_entity(Action & a, Model * m); +template +inline void default_change_state(Action & a, Model * m); + /** @@ -13127,8 +13091,6 @@ template class Agent { friend class Model; friend class Virus; - friend class Viruses; - friend class Viruses_const; friend class Tool; friend class Tools; friend class Tools_const; @@ -13141,6 +13103,7 @@ class Agent { friend void default_rm_virus(Action & a, Model * m); friend void default_rm_tool(Action & a, Model * m); friend void default_rm_entity(Action & a, Model * m); + friend void default_change_state(Action & a, Model * m); private: Model * model; @@ -13159,22 +13122,11 @@ class Agent { int state_last_changed = -1; ///< Last time the agent was updated. int id = -1; - std::vector< VirusPtr > viruses; - epiworld_fast_uint n_viruses = 0u; + VirusPtr virus = nullptr; std::vector< ToolPtr > tools; epiworld_fast_uint n_tools = 0u; - ActionFun add_virus_ = default_add_virus; - ActionFun add_tool_ = default_add_tool; - ActionFun add_entity_ = default_add_entity; - - ActionFun rm_virus_ = default_rm_virus; - ActionFun rm_tool_ = default_rm_tool; - ActionFun rm_entity_ = default_rm_entity; - - epiworld_fast_uint action_counter = 0u; - std::vector< Agent * > sampled_agents; size_t sampled_agents_n = 0u; std::vector< size_t > sampled_agents_left; @@ -13213,14 +13165,14 @@ class Agent { epiworld_fast_int queue = -99 ); - void add_virus( + void set_virus( VirusPtr virus, Model * model, epiworld_fast_int state_new = -99, epiworld_fast_int queue = -99 ); - void add_virus( + void set_virus( Virus virus, Model * model, epiworld_fast_int state_new = -99, @@ -13249,14 +13201,6 @@ class Agent { ); void rm_virus( - epiworld_fast_uint virus_idx, - Model * model, - epiworld_fast_int state_new = -99, - epiworld_fast_int queue = -99 - ); - - void rm_virus( - VirusPtr & virus, Model * model, epiworld_fast_int state_new = -99, epiworld_fast_int queue = -99 @@ -13277,14 +13221,6 @@ class Agent { ); void rm_agent_by_virus( - epiworld_fast_uint virus_idx, - Model * model, - epiworld_fast_int state_new = -99, - epiworld_fast_int queue = -99 - ); ///< Agent removed by virus - - void rm_agent_by_virus( - VirusPtr & virus, Model * model, epiworld_fast_int state_new = -99, epiworld_fast_int queue = -99 @@ -13306,10 +13242,7 @@ class Agent { int get_id() const; ///< Id of the individual - VirusPtr & get_virus(int i); - Viruses get_viruses(); - const Viruses_const get_viruses() const; - size_t get_n_viruses() const noexcept; + VirusPtr & get_virus(); ToolPtr & get_tool(int i); Tools get_tools(); @@ -13415,9 +13348,6 @@ class Agent { #ifndef EPIWORLD_PERSON_MEAT_HPP #define EPIWORLD_PERSON_MEAT_HPP -// template -// inline bool IN(Ta & a, std::vector< Ta > & b); - #define CHECK_COALESCE_(proposed_, virus_tool_, alt_) \ if (static_cast(proposed_) == -99) {\ if (static_cast(virus_tool_) == -99) \ @@ -13444,9 +13374,6 @@ inline void default_add_virus(Action & a, Model * m) Agent * p = a.agent; VirusPtr v = a.virus; - CHECK_COALESCE_(a.new_state, v->state_init, p->get_state()) - CHECK_COALESCE_(a.queue, v->queue_init, 1) - // Has a agent? If so, we need to register the transmission if (v->get_agent()) { @@ -13462,30 +13389,22 @@ inline void default_add_virus(Action & a, Model * m) } - // Update virus accounting - p->n_viruses++; - size_t n_viruses = p->n_viruses; - - if (n_viruses <= p->viruses.size()) - p->viruses[n_viruses - 1] = std::make_shared< Virus >(*v); - else - p->viruses.push_back(std::make_shared< Virus >(*v)); + p->virus = std::make_shared< Virus >(*v); + p->virus->set_date(m->today()); + p->virus->set_agent(p); - n_viruses--; - - // Notice that both agent and date can be changed in this case - // as only the sequence is a shared_ptr itself. - #ifdef EPI_DEBUG - if (n_viruses >= p->viruses.size()) + // Change of state needs to be recorded and updated on the + // tools. + if (p->state_prev != p->state) { - throw std::logic_error( - "[epi-debug]::default_add_virus Index for new virus out of range." - ); + auto & db = m->get_db(); + db.update_state(p->state_prev, p->state); + + for (size_t i = 0u; i < p->n_tools; ++i) + db.update_tool(p->tools[i]->get_id(), p->state_prev, p->state); } - #endif - p->viruses[n_viruses]->set_agent(p, n_viruses); - p->viruses[n_viruses]->set_date(m->today()); + // Lastly, we increase the daily count of the virus #ifdef EPI_DEBUG m->get_db().today_virus.at(v->get_id()).at(p->state)++; #else @@ -13500,9 +13419,6 @@ inline void default_add_tool(Action & a, Model * m) Agent * p = a.agent; ToolPtr t = a.tool; - - CHECK_COALESCE_(a.new_state, t->state_init, p->get_state()) - CHECK_COALESCE_(a.queue, t->queue_init, Queue::NoOne) // Update tool accounting p->n_tools++; @@ -13518,47 +13434,64 @@ inline void default_add_tool(Action & a, Model * m) p->tools[n_tools]->set_date(m->today()); p->tools[n_tools]->set_agent(p, n_tools); + // Change of state needs to be recorded and updated on the + // tools. + if (p->state_prev != p->state) + { + auto & db = m->get_db(); + db.update_state(p->state_prev, p->state); + + if (p->virus) + db.update_virus(p->virus->get_id(), p->state_prev, p->state); + } + m->get_db().today_tool[t->get_id()][p->state]++; + } template inline void default_rm_virus(Action & a, Model * model) { - Agent * p = a.agent; - VirusPtr & v = a.agent->viruses[a.virus->pos_in_agent]; - - CHECK_COALESCE_(a.new_state, v->state_post, p->get_state()) - CHECK_COALESCE_(a.queue, v->queue_post, -Queue::Everyone) + Agent * p = a.agent; + VirusPtr & v = a.virus; - if (--p->n_viruses > 0) - { - // The new virus will change positions - p->viruses[p->n_viruses]->pos_in_agent = v->pos_in_agent; - std::swap( - p->viruses[p->n_viruses], // Moving to the end - p->viruses[v->pos_in_agent] // Moving to the beginning - ); - } - // Calling the virus action over the removed virus v->post_recovery(model); + p->virus = nullptr; + + // Change of state needs to be recorded and updated on the + // tools. + if (p->state_prev != p->state) + { + auto & db = model->get_db(); + db.update_state(p->state_prev, p->state); + + for (size_t i = 0u; i < p->n_tools; ++i) + db.update_tool(p->tools[i]->get_id(), p->state_prev, p->state); + } + + // The counters of the virus only needs to decrease + #ifdef EPI_DEBUG + model->get_db().today_virus.at(v->get_id()).at(p->state_prev)--; + #else + model->get_db().today_virus[v->get_id()][p->state_prev]--; + #endif + + return; } template -inline void default_rm_tool(Action & a, Model * /*m*/) +inline void default_rm_tool(Action & a, Model * m) { Agent * p = a.agent; ToolPtr & t = a.agent->tools[a.tool->pos_in_agent]; - CHECK_COALESCE_(a.new_state, t->state_post, p->get_state()) - CHECK_COALESCE_(a.queue, t->queue_post, Queue::NoOne) - if (--p->n_tools > 0) { p->tools[p->n_tools]->pos_in_agent = t->pos_in_agent; @@ -13568,10 +13501,49 @@ inline void default_rm_tool(Action & a, Model * /*m*/) ); } + // Change of state needs to be recorded and updated on the + // tools. + if (p->state_prev != p->state) + { + auto & db = m->get_db(); + db.update_state(p->state_prev, p->state); + + if (p->virus) + db.update_virus(p->virus->get_id(), p->state_prev, p->state); + } + + // Lastly, we increase the daily count of the tool + #ifdef EPI_DEBUG + m->get_db().today_tool.at(t->get_id()).at(p->state_prev)--; + #else + m->get_db().today_tool[t->get_id()][p->state_prev]--; + #endif + return; } +template +inline void default_change_state(Action & a, Model * m) +{ + + Agent * p = a.agent; + + if (p->state_prev != p->state) + { + auto & db = m->get_db(); + db.update_state(p->state_prev, p->state); + + if (p->virus) + db.update_virus(p->virus->get_id(), p->state_prev, p->state); + + for (size_t i = 0u; i < p->n_tools; ++i) + db.update_tool(p->tools[i]->get_id(), p->state_prev, p->state); + + } + +} + template inline void default_add_entity(Action & a, Model *) { @@ -13579,9 +13551,6 @@ inline void default_add_entity(Action & a, Model *) Agent * p = a.agent; Entity * e = a.entity; - CHECK_COALESCE_(a.new_state, e->state_post, p->get_state()) - CHECK_COALESCE_(a.queue, e->queue_post, Queue::NoOne) - // Checking the agent and the entity are not linked if ((p->get_n_entities() > 0) && (e->size() > 0)) { @@ -13644,9 +13613,6 @@ inline void default_rm_entity(Action & a, Model * m) size_t idx_agent_in_entity = a.idx_agent; size_t idx_entity_in_agent = a.idx_object; - CHECK_COALESCE_(a.new_state, e->state_post, p->get_state()) - CHECK_COALESCE_(a.queue, e->queue_post, Queue::NoOne) - if (--p->n_entities > 0) { @@ -13720,36 +13686,21 @@ inline Agent::Agent(Agent && p) : state_prev(p.state_prev), state_last_changed(p.state_last_changed), id(p.id), - viruses(std::move(p.viruses)), /// Needs to be adjusted - n_viruses(p.n_viruses), tools(std::move(p.tools)), /// Needs to be adjusted - n_tools(p.n_tools), - add_virus_(std::move(p.add_virus_)), - add_tool_(std::move(p.add_tool_)), - add_entity_(std::move(p.add_entity_)), - rm_virus_(std::move(p.rm_virus_)), - rm_tool_(std::move(p.rm_tool_)), - rm_entity_(std::move(p.rm_entity_)), - action_counter(p.action_counter) + n_tools(p.n_tools) { state = p.state; id = p.id; // Dealing with the virus - - int loc = 0; - for (auto & v : viruses) + if (p.virus != nullptr) { - - // Will create a copy of the virus, with the exeption of - // the virus code - v->agent = this; - v->pos_in_agent = loc++; - + virus = std::move(p.virus); + virus->set_agent(this); } - loc = 0; + int loc = 0; for (auto & t : tools) { @@ -13781,34 +13732,24 @@ inline Agent::Agent(const Agent & p) : id = p.id; // Dealing with the virus - viruses.resize(p.get_n_viruses(), nullptr); - n_viruses = viruses.size(); - for (size_t i = 0u; i < n_viruses; ++i) + if (p.virus != nullptr) { - - // Will create a copy of the virus, with the exeption of - // the virus code - viruses[i] = std::make_shared>(*p.viruses[i]); - viruses[i]->set_agent(this, i); - + virus = std::make_shared>(*p.virus); + virus->set_agent(this); } + - tools.resize(p.get_n_tools(), nullptr); + tools.reserve(p.get_n_tools()); n_tools = tools.size(); for (size_t i = 0u; i < n_tools; ++i) { // Will create a copy of the virus, with the exeption of // the virus code - tools[i] = std::make_shared>(*p.tools[i]); - tools[i]->set_agent(this, i); + tools.emplace_back(std::make_shared>(*p.tools[i])); + tools.back()->set_agent(this, i); } - - add_virus_ = p.add_virus_; - add_tool_ = p.add_tool_; - rm_virus_ = p.rm_virus_; - rm_tool_ = p.rm_tool_; } @@ -13842,14 +13783,12 @@ inline Agent & Agent::operator=( state_last_changed = other_agent.state_last_changed; id = other_agent.id; - // viruses = other_agent.viruses; - n_viruses = other_agent.n_viruses; - viruses.resize(n_viruses, nullptr); - for (size_t i = 0u; i < n_viruses; ++i) + if (other_agent.virus != nullptr) { - viruses[i] = std::make_shared>(*other_agent.viruses[i]); - viruses[i]->set_agent(this, i); - } + virus = std::make_shared>(*other_agent.virus); + virus->set_agent(this); + } else + virus = nullptr; // tools = other_agent.tools; n_tools = other_agent.n_tools; @@ -13858,14 +13797,6 @@ inline Agent & Agent::operator=( tools[i] = std::make_shared>(*other_agent.tools[i]); tools[i]->set_agent(this, i); } - - add_virus_ = other_agent.add_virus_; - add_tool_ = other_agent.add_tool_; - add_entity_ = other_agent.add_entity_; - rm_virus_ = other_agent.rm_virus_; - rm_tool_ = other_agent.rm_tool_; - rm_entity_ = other_agent.rm_entity_; - action_counter = other_agent.action_counter; return *this; @@ -13884,10 +13815,12 @@ inline void Agent::add_tool( throw std::range_error("The tool with id: " + std::to_string(tool->get_id()) + " has not been registered. There are only " + std::to_string(model->get_n_tools()) + " included in the model."); - + + CHECK_COALESCE_(state_new, tool->state_init, state); + CHECK_COALESCE_(queue, tool->queue_init, Queue::NoOne); model->actions_add( - this, nullptr, tool, nullptr, state_new, queue, add_tool_, -1, -1 + this, nullptr, tool, nullptr, state_new, queue, default_add_tool, -1, -1 ); } @@ -13905,7 +13838,7 @@ inline void Agent::add_tool( } template -inline void Agent::add_virus( +inline void Agent::set_virus( VirusPtr virus, Model * model, epiworld_fast_int state_new, @@ -13919,14 +13852,17 @@ inline void Agent::add_virus( " has not been registered. There are only " + std::to_string(model->get_n_viruses()) + " included in the model."); + CHECK_COALESCE_(state_new, virus->state_init, state); + CHECK_COALESCE_(queue, virus->queue_init, Queue::NoOne); + model->actions_add( - this, virus, nullptr, nullptr, state_new, queue, add_virus_, -1, -1 + this, virus, nullptr, nullptr, state_new, queue, default_add_virus, -1, -1 ); } template -inline void Agent::add_virus( +inline void Agent::set_virus( Virus virus, Model * model, epiworld_fast_int state_new, @@ -13934,7 +13870,7 @@ inline void Agent::add_virus( ) { VirusPtr virus_ptr = std::make_shared< Virus >(virus); - add_virus(virus_ptr, model, state_new, queue); + set_virus(virus_ptr, model, state_new, queue); } template @@ -13946,11 +13882,14 @@ inline void Agent::add_entity( ) { + CHECK_COALESCE_(state_new, entity.state_init, state); + CHECK_COALESCE_(queue, entity.queue_init, Queue::NoOne); + if (model != nullptr) { model->actions_add( - this, nullptr, nullptr, &entity, state_new, queue, add_entity_, -1, -1 + this, nullptr, nullptr, &entity, state_new, queue, default_add_entity, -1, -1 ); } @@ -13959,11 +13898,11 @@ inline void Agent::add_entity( { Action a( - this, nullptr, nullptr, &entity, state_new, queue, add_entity_, + this, nullptr, nullptr, &entity, state_new, queue, default_add_entity, -1, -1 ); - default_add_entity(a, model); /* passing model makes nothing */ + // default_add_entity(a, model); /* passing model makes nothing */ } @@ -13978,6 +13917,9 @@ inline void Agent::rm_tool( ) { + CHECK_COALESCE_(state_new, tools[tool_idx]->state_post, state); + CHECK_COALESCE_(queue, tools[tool_idx]->queue_post, Queue::NoOne); + if (tool_idx >= n_tools) throw std::range_error( "The Tool you want to remove is out of range. This Agent only has " + @@ -13985,7 +13927,7 @@ inline void Agent::rm_tool( ); model->actions_add( - this, nullptr, tools[tool_idx], nullptr, state_new, queue, rm_tool_, -1, -1 + this, nullptr, tools[tool_idx], nullptr, state_new, queue, default_rm_tool, -1, -1 ); } @@ -14003,63 +13945,32 @@ inline void Agent::rm_tool( throw std::logic_error("Cannot remove a virus from another agent!"); model->actions_add( - this, nullptr, tool, nullptr, state_new, queue, rm_tool_, -1, -1 + this, nullptr, tool, nullptr, state_new, queue, default_rm_tool, -1, -1 ); } template inline void Agent::rm_virus( - epiworld_fast_uint virus_idx, Model * model, epiworld_fast_int state_new, epiworld_fast_int queue ) { - if (virus_idx >= n_viruses) - throw std::range_error( - "The Virus you want to remove is out of range. This Agent only has " + - std::to_string(n_viruses) + " viruses." - ); - else if (n_viruses == 0u) - throw std::logic_error( - "There is no virus to remove here!" - ); - #ifdef EPI_DEBUG - if (viruses[virus_idx]->pos_in_agent >= static_cast(n_viruses)) - { + if (virus == nullptr) throw std::logic_error( - "[epi-debug]::rm_virus the position in the agent is wrong." - ); - } - #endif - - model->actions_add( - this, viruses[virus_idx], nullptr, nullptr, state_new, queue, - default_rm_virus, -1, -1 + "There is no virus to remove here!" ); - -} -template -inline void Agent::rm_virus( - VirusPtr & virus, - Model * model, - epiworld_fast_int state_new, - epiworld_fast_int queue -) -{ - - if (virus->agent != this) - throw std::logic_error("Cannot remove a virus from another agent!"); + CHECK_COALESCE_(state_new, virus->state_post, state); + CHECK_COALESCE_(queue, virus->queue_post, Queue::Everyone); model->actions_add( this, virus, nullptr, nullptr, state_new, queue, default_rm_virus, -1, -1 ); - - + } template @@ -14081,9 +13992,12 @@ inline void Agent::rm_entity( "There is entity to remove here!" ); + CHECK_COALESCE_(state_new, model->entities[entity_idx].state_post, state); + CHECK_COALESCE_(queue, model->entities[entity_idx].queue_post, Queue::NoOne); + model->actions_add( this, nullptr, nullptr, model->entities[entity_idx], state_new, queue, - default_rm_entity, entities_locations[entity_idx], entity_idx + default_rm_entity, entities_locations[entity_idx], entity_idx ); } @@ -14110,82 +14024,30 @@ inline void Agent::rm_entity( entity.get_name() + "\"." ); + CHECK_COALESCE_(state_new, entity.state_post, state); + CHECK_COALESCE_(queue, entity.queue_post, Queue::NoOne); model->actions_add( this, nullptr, nullptr, entities[entity_idx], state_new, queue, - default_rm_entity, entities_locations[entity_idx], entity_idx + default_rm_entity, entities_locations[entity_idx], entity_idx ); } template inline void Agent::rm_agent_by_virus( - epiworld_fast_uint virus_idx, Model * model, epiworld_fast_int state_new, epiworld_fast_int queue ) { - if (state_new == -99) - state_new = state; - - if (virus_idx >= n_viruses) - throw std::range_error( - std::string("The virus trying to remove the agent is out of range. ") + - std::string("This agent has only ") + std::to_string(n_viruses) + - std::string(" and you are trying to remove virus # ") + - std::to_string(virus_idx) + std::string(".") - ); - - // Removing viruses - for (size_t i = 0u; i < n_viruses; ++i) - { - if (i != virus_idx) - rm_virus(i, model); - } - - // Changing state to new_state - VirusPtr & v = viruses[virus_idx]; - epiworld_fast_int dead_state, dead_queue; - v->get_state(nullptr, nullptr, &dead_state); - v->get_queue(nullptr, nullptr, &dead_queue); + CHECK_COALESCE_(state_new, virus->state_removed, state); + CHECK_COALESCE_(queue, virus->queue_removed, Queue::Everyone); - if (queue != -99) - dead_queue = queue; - - change_state( - model, - // Either preserve the current state or apply a new one - (dead_state < 0) ? state : static_cast(dead_state), - - // By default, it will be removed from the queue... unless the user - // says the contrary! - (dead_queue == -99) ? Queue::NoOne : dead_queue - ); - -} - -template -inline void Agent::rm_agent_by_virus( - VirusPtr & virus, - Model * model, - epiworld_fast_int state_new, - epiworld_fast_int queue -) -{ - - if (virus->get_agent() == nullptr) - throw std::logic_error("The virus trying to remove the agent has no host."); - - if (virus->get_agent()->id != id) - throw std::logic_error("Viruses can only remove their hosts'."); - - rm_agent_by_virus( - virus->pos_in_agent, - model, - state_new, - queue - ); + model->actions_add( + this, virus, nullptr, nullptr, state_new, queue, + default_rm_virus, -1, -1 + ); } @@ -14229,29 +14091,10 @@ inline int Agent::get_id() const } template -inline Viruses Agent::get_viruses() { - - return Viruses(*this); - +inline VirusPtr & Agent::get_virus() { + return virus; } -template -inline const Viruses_const Agent::get_viruses() const { - - return Viruses_const(*this); - -} - -template -inline VirusPtr & Agent::get_virus(int i) { - return viruses.at(i); -} - -template -inline size_t Agent::get_n_viruses() const noexcept -{ - return n_viruses; -} template inline Tools Agent::get_tools() { @@ -14279,8 +14122,7 @@ template inline void Agent::mutate_virus() { - for (auto & v : viruses) - v->mutate(); + virus->mutate(); } @@ -14406,7 +14248,8 @@ inline void Agent::change_state( { model->actions_add( - this, nullptr, nullptr, nullptr, new_state, queue, nullptr, -1, -1 + this, nullptr, nullptr, nullptr, new_state, queue, + default_change_state, -1, -1 ); return; @@ -14422,8 +14265,7 @@ template inline void Agent::reset() { - this->viruses.clear(); - n_viruses = 0u; + this->virus = nullptr; this->tools.clear(); n_tools = 0u; @@ -14470,9 +14312,8 @@ inline bool Agent::has_tool(const Tool & tool) const template inline bool Agent::has_virus(epiworld_fast_uint t) const { - for (auto & v : viruses) - if (v->get_id() == static_cast(t)) - return true; + if (virus->get_id() == static_cast(t)) + return true; return false; } @@ -14481,9 +14322,8 @@ template inline bool Agent::has_virus(std::string name) const { - for (auto & v : viruses) - if (v->get_name() == name) - return true; + if (virus->get_name() == name) + return true; return false; @@ -14507,15 +14347,18 @@ inline void Agent::print( if (compressed) { printf_epiworld( - "Agent: %i, state: %s (%lu), Nvirus: %lu, NTools: %lu, NNeigh: %lu\n", - id, model->states_labels[state].c_str(), state, n_viruses, n_tools, neighbors.size() + "Agent: %i, state: %s (%lu), Has virus: %s, NTools: %lu, NNeigh: %lu\n", + id, model->states_labels[state].c_str(), state, + virus == nullptr ? std::string("no").c_str() : std::string("yes").c_str(), + n_tools, neighbors.size() ); } else { printf_epiworld("Information about agent id %i\n", this->id); printf_epiworld(" State : %s (%lu)\n", model->states_labels[state].c_str(), state); - printf_epiworld(" Virus count : %lu\n", n_viruses); + printf_epiworld(" Has virus : %s\n", virus == nullptr ? + std::string("no").c_str() : std::string("yes").c_str()); printf_epiworld(" Tool count : %lu\n", n_tools); printf_epiworld(" Neigh. count : %lu\n", neighbors.size()); @@ -14666,24 +14509,21 @@ inline bool Agent::operator==(const Agent & other) const // state_last_changed != other.state_last_changed, // "Agent:: state_last_changed don't match" // ) ///< Last time the agent was updated. - - + EPI_DEBUG_FAIL_AT_TRUE( - n_viruses != other.n_viruses, - "Agent:: n_viruses don't match" - ) - + ((virus == nullptr) && (other.virus != nullptr)) || + ((virus != nullptr) && (other.virus == nullptr)), + "Agent:: virus don't match" + ) - for (size_t i = 0u; i < n_viruses; ++i) + if ((virus != nullptr) && (other.virus != nullptr)) { - EPI_DEBUG_FAIL_AT_TRUE( - *viruses[i] != *other.viruses[i], - "Agent:: viruses[i] don't match" + *virus != *other.virus, + "Agent:: virus doesn't match" ) - } - + EPI_DEBUG_FAIL_AT_TRUE(n_tools != other.n_tools, "Agent:: n_tools don't match") for (size_t i = 0u; i < n_tools; ++i) @@ -14765,6 +14605,7 @@ class AgentsSample { Agent * agent = nullptr; int sample_type = SAMPLETYPE::AGENT; + std::vector< size_t > states = {}; void sample_n(size_t n); ///< Backbone function for sampling @@ -14776,9 +14617,23 @@ class AgentsSample { AgentsSample(const AgentsSample & a) = delete; ///< Copy constructor AgentsSample(AgentsSample && a) = delete; ///< Move constructor - AgentsSample(Model & model_, size_t n, bool truncate = false); - AgentsSample(Model * model, Entity & entity_, size_t n, bool truncate = false); - AgentsSample(Model * model, Agent & agent_, size_t n, bool truncate = false); + AgentsSample( + Model & model_, size_t n, + std::vector< size_t > states_ = {}, + bool truncate = false + ); + + AgentsSample( + Model * model, Entity & entity_, size_t n, + std::vector< size_t > states_ = {}, + bool truncate = false + ); + + AgentsSample( + Model * model, Agent & agent_, size_t n, + std::vector< size_t > states_ = {}, + bool truncate = false + ); ~AgentsSample(); @@ -14795,9 +14650,12 @@ template inline AgentsSample::AgentsSample( Model & model_, size_t n, + std::vector< size_t > states_, bool truncate ) { + states = states_; + if (truncate) { @@ -14809,12 +14667,12 @@ inline AgentsSample::AgentsSample( "There are only " + std::to_string(model_.size()) + " agents. You cannot " + "sample " + std::to_string(n)); - sample_size = n; - model = &model_; - sample_type = SAMPLETYPE::MODEL; + sample_size = n; + model = &model_; + sample_type = SAMPLETYPE::MODEL; - agents = &model_.sampled_population; - agents_n = &model_.sampled_population_n; + agents = &model_.sampled_population; + agents_n = &model_.sampled_population_n; agents_left = &model_.population_left; agents_left_n = &model_.population_left_n; @@ -14829,9 +14687,13 @@ template inline AgentsSample::AgentsSample( Model * model, Entity & entity_, - size_t n, bool truncate + size_t n, + std::vector< size_t > states_, + bool truncate ) { + states = states_; + if (truncate) { @@ -14843,12 +14705,12 @@ inline AgentsSample::AgentsSample( "There are only " + std::to_string(entity_.size()) + " agents. You cannot " + "sample " + std::to_string(n)); - sample_size = n; - model = &entity_.model; - sample_type = SAMPLETYPE::ENTITY; + sample_size = n; + model = &entity_.model; + sample_type = SAMPLETYPE::ENTITY; - agents = &entity_.sampled_agents; - agents_n = &entity_.sampled_agents_n; + agents = &entity_.sampled_agents; + agents_n = &entity_.sampled_agents_n; agents_left = &entity_.sampled_agents_left; agents_left_n = &entity_.sampled_agents_left_n; @@ -14876,16 +14738,18 @@ inline AgentsSample::AgentsSample( Model * model, Agent & agent_, size_t n, + std::vector< size_t > states_, bool truncate ) { + states = states_; sample_type = SAMPLETYPE::AGENT; - agent = &agent_; + agent = &agent_; - agents = &agent_.sampled_agents; - agents_n = &agent_.sampled_agents_n; + agents = &agent_.sampled_agents; + agents_n = &agent_.sampled_agents_n; agents_left = &agent_.sampled_agents_left; agents_left_n = &agent_.sampled_agents_left_n; @@ -14894,7 +14758,7 @@ inline AgentsSample::AgentsSample( size_t agents_in_entities = 0; Entities entities_a = agent->get_entities(); - std::vector< int > cum_agents_count(entities_a.size(), 0); + std::vector< size_t > cum_agents_count(entities_a.size(), 0); int idx = -1; for (auto & e : entities_a) { @@ -14917,15 +14781,18 @@ inline AgentsSample::AgentsSample( } else if (n > agents_in_entities) throw std::logic_error( - "There are only " + std::to_string(agents_in_entities) + " agents. You cannot " + - "sample " + std::to_string(n)); + "There are only " + std::to_string(agents_in_entities) + + " agents. You cannot " + + "sample " + std::to_string(n) + ); sample_size = n; if (agents->size() < n) agents->resize(n); - for (size_t i = 0u; i < n; ++i) + size_t i_obs = 0u; + for (size_t i = 0u; i < agents_in_entities; ++i) { int jth = std::floor(model->runif() * agents_in_entities); for (size_t e = 0u; e < cum_agents_count.size(); ++e) @@ -14933,11 +14800,24 @@ inline AgentsSample::AgentsSample( // Are we in the limit? if (jth <= cum_agents_count[e]) { + size_t agent_idx = 0u; if (e == 0) // From the first group - agents->operator[](i) = entities_a[e][jth]; + agent_idx = entities_a[e][jth]; else - agents->operator[](i) = entities_a[e][jth - cum_agents_count[e - 1]]; + agent_idx = entities_a[e][jth - cum_agents_count[e - 1]]; + + // Getting the state + size_t state = model->population[agent_idx].get_state(); + + // Checking if states was specified + if (states.size()) + { + if (std::find(states.begin(), states.end(), state) != states.end()) + continue; + } + agents->operator[](i_obs++) = agent_idx; + break; } @@ -14999,59 +14879,143 @@ template inline void AgentsSample::sample_n(size_t n) { - // Checking if the size of the entity has changed (or hasn't been initialized) - if (sample_type == SAMPLETYPE::MODEL) + // Reducing size + if (states.size()) { + + // Getting the number of agents left + agents_left->clear(); - if (model->size() != agents_left->size()) + if (sample_type == SAMPLETYPE::MODEL) { - agents_left->resize(model->size(), 0u); - std::iota(agents_left->begin(), agents_left->end(), 0u); - } + // Making some room + agents_left->reserve(model->size()); - } else if (sample_type == SAMPLETYPE::ENTITY) { + // Iterating through the agents in the population + for (size_t a_i = 0u; a_i < model->population.size(); ++a_i) + { + + // If the agent is within the selected set of states, + // then we add it to the list of agents left + size_t s = model->population[a_i].get_state(); + if (std::find(states.begin(), states.end(), s) != states.end()) + agents_left->push_back(a_i); + + } - if (entity->size() != agents_left->size()) + } + else if (sample_type == SAMPLETYPE::ENTITY) { - agents_left->resize(entity->size(), 0u); - std::iota(agents_left->begin(), agents_left->end(), 0u); + // Making room + agents_left->reserve(entity->size()); + + // Iterating through the agents in the entity + for (size_t a_i = 0u; a_i < entity->size(); ++a_i) + { + size_t s = model->population[entity->agents[a_i]].get_state(); + if (std::find(states.begin(), states.end(), s) != states.end()) + agents_left->push_back(a_i); + + } } - } + } else { + + // Checking if the size of the entity has changed (or hasn't been initialized) + if (sample_type == SAMPLETYPE::MODEL) + { + + if (model->size() != agents_left->size()) + { + agents_left->resize(model->size(), 0u); + std::iota(agents_left->begin(), agents_left->end(), 0u); + } + + } else if (sample_type == SAMPLETYPE::ENTITY) { + + if (entity->size() != agents_left->size()) + { + + agents_left->resize(entity->size(), 0u); + std::iota(agents_left->begin(), agents_left->end(), 0u); + + } + + } + + } // Restart the counter of agents left *agents_left_n = agents_left->size(); + // Making sure we have enough room for the sample of agents if (agents->size() < sample_size) agents->resize(sample_size, nullptr); if (sample_type == SAMPLETYPE::MODEL) { + #ifdef EPI_DEBUG + std::vector< bool > __sampled(model->size(), true); + for (auto & a_i: *agents_left) + __sampled[a_i] = false; + #endif + for (size_t i = 0u; i < n; ++i) { - size_t ith = agents_left->operator[](model->runif() * ((*agents_left_n)--)); + // Sampling from 0 to (agents_left_n - 1) + size_t ith_ = static_cast(model->runif() * ((*agents_left_n)--)); + + // Getting the id of the agent and adding it to the list of agents + size_t ith = agents_left->operator[](ith_); agents->operator[](i) = &model->population[ith]; + #ifdef EPI_DEBUG + if (__sampled[ith]) + throw std::logic_error("The same agent was sampled twice."); + else + __sampled[ith] = true; + #endif + // Updating list - std::swap(agents_left->operator[](ith), agents_left->operator[](*agents_left_n)); + std::swap( + agents_left->operator[](ith_), + agents_left->operator[](*agents_left_n) + ); } - } else if (sample_type == SAMPLETYPE::ENTITY) { + + } + else if (sample_type == SAMPLETYPE::ENTITY) + { + + #ifdef EPI_DEBUG + std::vector< bool > __sampled(entity->size(), true); + for (auto & a_i: *agents_left) + __sampled[a_i] = false; + #endif for (size_t i = 0u; i < n; ++i) { - size_t ith = agents_left->operator[](model->runif() * (--(*agents_left_n))); - agents->operator[](i) = entity->agents[ith]; + size_t ith_ = static_cast(model->runif() * ((*agents_left_n)--)); + size_t ith = agents_left->operator[](ith_); + agents->operator[](i) = &model->population[entity->agents[ith]]; + + #ifdef EPI_DEBUG + if (__sampled[ith]) + throw std::logic_error("The same agent was sampled twice."); + else + __sampled[ith] = true; + #endif // Updating list - std::swap(agents_left->operator[](ith), agents_left->operator[](*agents_left_n)); + std::swap(agents_left->operator[](ith_), agents_left->operator[](*agents_left_n)); } @@ -15085,7 +15049,367 @@ inline void AgentsSample::sample_n(size_t n) #define EPIWORLD_MODELS_HPP namespace epimodels { - + +/*////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// + + Start of -include/epiworld//models/init-functions.hpp- + +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////*/ + + +#ifndef EPIWORLD_MODELS_INIT_FUNCTIONS_HPP +#define EPIWORLD_MODELS_INIT_FUNCTIONS_HPP + +/** + * @brief Creates an initial function for the SIR-like models + * The function is used for the initial states of the model. +*/ +template +inline std::function*)> create_init_function_sir( + std::vector< double > proportions_ +) { + + // Checking widths + if (proportions_.size() != 1u) + throw std::invalid_argument( + "The vector of proportions must have a single element." + ); + + // Proportion should be within [0, 1] + if ((proportions_[0] < 0.0) || (proportions_[0] > 1.0)) + throw std::invalid_argument( + "The proportion must be within (0, 1)." + ); + + double prop = proportions_[0u]; + + std::function*)> fun = + [prop] (epiworld::Model * model) -> void { + + // Figuring out information about the viruses + double tot = 0.0; + const auto & vpreval = model->get_prevalence_virus(); + const auto & vprop = model->get_prevalence_virus_as_proportion(); + double n = static_cast(model->size()); + for (size_t i = 0u; i < model->get_n_viruses(); ++i) + { + if (vprop[i]) + tot += vpreval[i]; + else + tot += vpreval[i] / n; + } + + // Putting the total into context + double tot_left = 1.0 - tot; + + // Since susceptible and infected are "fixed," + // we only need to change recovered + size_t nrecovered = prop * tot_left * n; + + epiworld::AgentsSample sample( + *model, + nrecovered, + {0u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample) + agent->change_state(model, 2, Queue::NoOne); + + // Running the actions + model->actions_run(); + + return; + + }; + + return fun; + +} + +/** + * @brief Creates an initial function for the SIR-like models + * The function is used for the initial states of the model. +*/ +template +inline std::function*)> create_init_function_sird( + std::vector< double > prop +) { + + // Check length of prop equals two + if (prop.size() != 2u) + throw std::invalid_argument( + "The vector of proportions must have two elements." + ); + + // Check elements in prop are within [0, 1] and sum up to 1 + double tot = 0.0; + for (auto & v : prop) + { + if ((v < 0.0) || (v > 1.0)) + throw std::invalid_argument( + "The proportion must be within (0, 1)." + ); + tot += v; + } + + if (tot >= 1.0) + throw std::invalid_argument( + "The proportions must sum up to 1." + ); + + std::function*)> fun = + [prop] (epiworld::Model * model) -> void { + + // Figuring out information about the viruses + double tot = 0.0; + const auto & vpreval = model->get_prevalence_virus(); + const auto & vprop = model->get_prevalence_virus_as_proportion(); + double n = static_cast(model->size()); + for (size_t i = 0u; i < model->get_n_viruses(); ++i) + { + if (vprop[i]) + tot += vpreval[i]; + else + tot += vpreval[i] / n; + } + + // Putting the total into context + double tot_left = 1.0 - tot; + + // Since susceptible and infected are "fixed," + // we only need to change recovered + size_t nrecovered = prop[0u] * tot_left * n; + size_t ndeceased = prop[01] * tot_left * n; + + epiworld::AgentsSample sample_recover( + *model, + nrecovered, + {0u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_recover) + agent->change_state(model, 2, Queue::NoOne); + + epiworld::AgentsSample sample_deceased( + *model, + ndeceased, + {0u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_deceased) + agent->change_state(model, 3, Queue::NoOne); + + // Running the actions + model->actions_run(); + + return; + + }; + + return fun; + +} + + +/** + * @brief Creates an initial function for the SEIR-like models + * The function is used for the initial states of the model. +*/ +template +inline std::function*)> create_init_function_seir( + std::vector< double > proportions_ +) { + + // Checking widths + if (proportions_.size() != 2u) { + throw std::invalid_argument("-proportions_- must have two entries."); + } + + // proportions_ are values between 0 and 1, otherwise error + for (auto & v : proportions_) + if ((v < 0.0) || (v > 1.0)) + throw std::invalid_argument( + "-proportions_- must have values between 0 and 1." + ); + + + std::function*)> fun = + [proportions_] (epiworld::Model * model) -> void { + + // Figuring out information about the viruses + double tot = 0.0; + const auto & vpreval = model->get_prevalence_virus(); + const auto & vprop = model->get_prevalence_virus_as_proportion(); + double n = static_cast(model->size()); + for (size_t i = 0u; i < model->get_n_viruses(); ++i) + { + if (vprop[i]) + tot += vpreval[i]; + else + tot += vpreval[i] / n; + } + + // Putting the total into context + double tot_left = 1.0 - tot; + + // Since susceptible and infected are "fixed," + // we only need to change recovered + size_t nexposed = proportions_[0u] * tot * n; + size_t nrecovered = proportions_[1u] * tot_left * n; + + epiworld::AgentsSample sample_suscept( + *model, + nrecovered, + {0u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_suscept) + agent->change_state(model, 3, Queue::NoOne); + + epiworld::AgentsSample sample_exposed( + *model, + nexposed, + {1u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_exposed) + agent->change_state(model, 2, Queue::NoOne); + + // Running the actions + model->actions_run(); + + return; + + }; + + return fun; + +} + +/** + * @brief Creates an initial function for the SEIR-like models + * The function is used for the initial states of the model. +*/ +template +inline std::function*)> create_init_function_seird( + std::vector< double > proportions_ +) { + + // Checking widths + if (proportions_.size() != 3u) { + throw std::invalid_argument("-proportions_- must have three entries."); + } + + // proportions_ are values between 0 and 1, otherwise error + for (auto & v : proportions_) + if ((v < 0.0) || (v > 1.0)) + throw std::invalid_argument( + "-proportions_- must have values between 0 and 1." + ); + + // Last first two terms shouldn't add up to more than 1 + if ((proportions_[1u] + proportions_[2u]) > 1.0) + throw std::invalid_argument( + "The last two terms of -proportions_- must add up to less than 1." + ); + + std::function*)> fun = + [proportions_] (epiworld::Model * model) -> void { + + // Figuring out information about the viruses + double tot = 0.0; + const auto & vpreval = model->get_prevalence_virus(); + const auto & vprop = model->get_prevalence_virus_as_proportion(); + double n = static_cast(model->size()); + for (size_t i = 0u; i < model->get_n_viruses(); ++i) + { + if (vprop[i]) + tot += vpreval[i]; + else + tot += vpreval[i] / n; + } + + // Putting the total into context + double tot_left = 1.0 - tot; + + // Since susceptible and infected are "fixed," + // we only need to change recovered + size_t nexposed = proportions_[0u] * tot * n; + size_t nrecovered = proportions_[1u] * tot_left * n; + size_t ndeceased = proportions_[2u] * tot_left * n; + + epiworld::AgentsSample sample_suscept( + *model, + nrecovered, + {0u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_suscept) + agent->change_state(model, 3, Queue::NoOne); + + epiworld::AgentsSample sample_exposed( + *model, + nexposed, + {1u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_exposed) + agent->change_state(model, 2, Queue::NoOne); + + // Running the actions + model->actions_run(); + + // Setting the initial states for the deceased + epiworld::AgentsSample sample_deceased( + *model, + ndeceased, + {0u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_deceased) + agent->change_state(model, 4, Queue::NoOne); + + // Running the actions + model->actions_run(); + + return; + + }; + + return fun; + +} + + +#endif +/*////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// + + End of -include/epiworld//models/init-functions.hpp- + +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////*/ + + + /*////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// @@ -15178,7 +15502,9 @@ inline std::function*)> globalaction_tool_logit( // Computing the probability using a logit. Uses OpenMP reduction // to sum the coefficients. double p = 0.0; + #if defined(__OPENMP) || defined(_OPENMP) #pragma omp parallel for reduction(+:p) + #endif for (size_t i = 0u; i < coefs.size(); ++i) p += coefs.at(i) * agent(vars[i]); @@ -15398,6 +15724,16 @@ class ModelSIR : public epiworld::Model epiworld_double transmission_rate, epiworld_double recovery_rate ); + + /** + * @brief Set the initial states of the model + * @param proportions_ Double vector with a single element: + * - The proportion of non-infected individuals who have recovered. + */ + ModelSIR & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); }; @@ -15456,6 +15792,20 @@ inline ModelSIR::ModelSIR( } +template +inline ModelSIR & ModelSIR::initial_states( + std::vector< double > proportions_, + std::vector< int > /* queue_ */ +) { + + Model::initial_states_fun = + create_init_function_sir(proportions_) + ; + + return *this; + +} + #endif /*////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// @@ -15523,7 +15873,7 @@ class ModelSEIR : public epiworld::Model ) -> void { // Getting the virus - auto v = p->get_virus(0); + auto v = p->get_virus(); // Does the agent become infected? if (m->runif() < 1.0/(v->get_incubation(m))) @@ -15539,11 +15889,22 @@ class ModelSEIR : public epiworld::Model ) -> void { // Does the agent recover? if (m->runif() < (m->par("Recovery rate"))) - p->rm_virus(0, m); + p->rm_virus(m); return; }; + /** + * @brief Set up the initial states of the model. + * @param proportions_ Double vector with the following values: + * - 0: Proportion of non-infected agents who are removed. + * - 1: Proportion of exposed agents to be set as infected. + */ + ModelSEIR & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); + }; @@ -15609,7 +15970,19 @@ inline ModelSEIR::ModelSEIR( } +template +inline ModelSEIR & ModelSEIR::initial_states( + std::vector< double > proportions_, + std::vector< int > /**/ +) { + + Model::initial_states_fun = + create_init_function_seir(proportions_) + ; + + return *this; +} #endif /*////////////////////////////////////////////////////////////////////////////// @@ -15754,22 +16127,20 @@ inline ModelSURV::ModelSURV( for (auto & neighbor: p->get_neighbors()) { - for (size_t i = 0u; i < neighbor->get_n_viruses(); ++i) - { + auto & v = neighbor->get_virus(); - auto & v = neighbor->get_virus(i); - - /* And it is a function of susceptibility_reduction as well */ - epiworld_double tmp_transmission = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) - ; - - m->array_double_tmp[nviruses_tmp] = tmp_transmission; - m->array_virus_tmp[nviruses_tmp++] = &(*v); + if (v == nullptr) + continue; - } + /* And it is a function of susceptibility_reduction as well */ + epiworld_double tmp_transmission = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) + ; + + m->array_double_tmp[nviruses_tmp] = tmp_transmission; + m->array_virus_tmp[nviruses_tmp++] = &(*v); } // No virus to compute on @@ -15782,7 +16153,7 @@ inline ModelSURV::ModelSURV( if (which < 0) return; - p->add_virus(*m->array_virus_tmp[which], m); + p->set_virus(*m->array_virus_tmp[which], m); return; }; @@ -15792,7 +16163,7 @@ inline ModelSURV::ModelSURV( [](epiworld::Agent * p, epiworld::Model * m) -> void { - epiworld::VirusPtr & v = p->get_virus(0u); + epiworld::VirusPtr & v = p->get_virus(); epiworld_double p_die = v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); epiworld_fast_uint days_since_exposed = m->today() - v->get_date(); @@ -15816,7 +16187,7 @@ inline ModelSURV::ModelSURV( // If past days infected + latent, then bye. if (days_since_exposed >= v->get_data()[1u]) { - p->rm_virus(0, m); + p->rm_virus(m); return; } @@ -16053,12 +16424,7 @@ class ModelSIRCONN : public epiworld::Model public: - ModelSIRCONN() { - - // tracked_agents_infected.reserve(1e4); - // tracked_agents_infected_next.reserve(1e4); - - }; + ModelSIRCONN() {}; ModelSIRCONN( ModelSIRCONN & model, @@ -16079,17 +16445,7 @@ class ModelSIRCONN : public epiworld::Model epiworld_double recovery_rate ); - // Tracking who is infected and who is not - // std::vector< epiworld::Agent* > tracked_agents_infected = {}; - // std::vector< epiworld::Agent* > tracked_agents_infected_next = {}; - // std::vector< epiworld_double > tracked_agents_weight = {}; - // std::vector< epiworld_double > tracked_agents_weight_next = {}; - - // int tracked_ninfected = 0; - // int tracked_ninfected_next = 0; - // epiworld_double tracked_current_infect_prob = 0.0; - - void run( + ModelSIRCONN & run( epiworld_fast_uint ndays, int seed = -1 ); @@ -16098,24 +16454,28 @@ class ModelSIRCONN : public epiworld::Model Model * clone_ptr(); + /** + * @brief Set the initial states of the model + * @param proportions_ Double vector with a single element: + * - The proportion of non-infected individuals who have recovered. + */ + ModelSIRCONN & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); + }; template -inline void ModelSIRCONN::run( +inline ModelSIRCONN & ModelSIRCONN::run( epiworld_fast_uint ndays, int seed ) { - // tracked_agents_infected.clear(); - // tracked_agents_infected_next.clear(); - - // tracked_ninfected = 0; - // tracked_ninfected_next = 0; - // tracked_current_infect_prob = 0.0; - Model::run(ndays, seed); + return *this; } @@ -16124,14 +16484,6 @@ inline void ModelSIRCONN::reset() { Model::reset(); - - // Model::set_rand_binom( - // Model::size(), - // static_cast( - // Model::par("Contact rate"))/ - // static_cast(Model::size()) - // ); - return; } @@ -16171,8 +16523,6 @@ inline ModelSIRCONN::ModelSIRCONN( ) { - - epiworld::UpdateFun update_susceptible = []( epiworld::Agent * p, epiworld::Model * m ) -> void @@ -16219,24 +16569,25 @@ inline ModelSIRCONN::ModelSIRCONN( if (neighbor.get_state() == ModelSIRCONN::INFECTED) { - for (const VirusPtr & v : neighbor.get_viruses()) - { + if (neighbor.get_virus() == nullptr) + continue; - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor.get_transmission_reduction(v, m)) - ; - - m->array_virus_tmp[nviruses_tmp++] = &(*v); + auto & v = neighbor.get_virus(); + + #ifdef EPI_DEBUG + if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif - } + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor.get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); + } } @@ -16251,7 +16602,7 @@ inline ModelSIRCONN::ModelSIRCONN( if (which < 0) return; - p->add_virus(*m->array_virus_tmp[which], m); + p->set_virus(*m->array_virus_tmp[which], m); return; @@ -16270,14 +16621,10 @@ inline ModelSIRCONN::ModelSIRCONN( // Odd: Die, Even: Recover epiworld_fast_uint n_events = 0u; - for (const auto & v : p->get_viruses()) - { - - // Recover - m->array_double_tmp[n_events++] = - 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); - - } + // Recover + m->array_double_tmp[n_events++] = + 1.0 - (1.0 - p->get_virus()->get_prob_recovery(m)) * + (1.0 - p->get_recovery_enhancer(p->get_virus(), m)); #ifdef EPI_DEBUG if (n_events == 0u) @@ -16301,13 +16648,14 @@ inline ModelSIRCONN::ModelSIRCONN( return; // Which roulette happen? - size_t which_v = std::floor(which / 2); - p->rm_virus(which_v, m); + p->rm_virus(m); return ; } else - throw std::logic_error("This function can only be applied to infected individuals. (SIR)") ; + throw std::logic_error( + "This function can only be applied to infected individuals. (SIR)" + ) ; return; @@ -16367,6 +16715,20 @@ inline ModelSIRCONN::ModelSIRCONN( } +template +inline ModelSIRCONN & ModelSIRCONN::initial_states( + std::vector< double > proportions_, + std::vector< int > /**/ +) { + + Model::initial_states_fun = + create_init_function_sir(proportions_) + ; + + return *this; + +} + #endif /*////////////////////////////////////////////////////////////////////////////// @@ -16424,7 +16786,7 @@ class ModelSEIRCONN : public epiworld::Model epiworld_double recovery_rate ); - void run( + ModelSEIRCONN & run( epiworld_fast_uint ndays, int seed = -1 ); @@ -16433,10 +16795,20 @@ class ModelSEIRCONN : public epiworld::Model Model * clone_ptr(); + /** + * @brief Set the initial states of the model + * @param proportions_ Double vector with a single element: + * - The proportion of non-infected individuals who have recovered. + */ + ModelSEIRCONN & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); + }; template -inline void ModelSEIRCONN::run( +inline ModelSEIRCONN & ModelSEIRCONN::run( epiworld_fast_uint ndays, int seed ) @@ -16444,6 +16816,8 @@ inline void ModelSEIRCONN::run( Model::run(ndays, seed); + return *this; + } template @@ -16538,24 +16912,21 @@ inline ModelSEIRCONN::ModelSEIRCONN( if (neighbor.get_state() == ModelSEIRCONN::INFECTED) { - for (const VirusPtr & v : neighbor.get_viruses()) - { + auto & v = neighbor.get_virus(); - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor.get_transmission_reduction(v, m)) - ; - - m->array_virus_tmp[nviruses_tmp++] = &(*v); + #ifdef EPI_DEBUG + if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif - } + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor.get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); } } @@ -16570,7 +16941,7 @@ inline ModelSEIRCONN::ModelSEIRCONN( if (which < 0) return; - p->add_virus( + p->set_virus( *m->array_virus_tmp[which], m, ModelSEIRCONN::EXPOSED @@ -16590,7 +16961,7 @@ inline ModelSEIRCONN::ModelSEIRCONN( { // Getting the virus - auto & v = p->get_virus(0u); + auto & v = p->get_virus(); // Does the agent become infected? if (m->runif() < 1.0/(v->get_incubation(m))) @@ -16608,14 +16979,11 @@ inline ModelSEIRCONN::ModelSEIRCONN( // Odd: Die, Even: Recover epiworld_fast_uint n_events = 0u; - for (const auto & v : p->get_viruses()) - { - - // Recover - m->array_double_tmp[n_events++] = - 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); + const auto & v = p->get_virus(); - } + // Recover + m->array_double_tmp[n_events++] = + 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); #ifdef EPI_DEBUG if (n_events == 0u) @@ -16639,8 +17007,7 @@ inline ModelSEIRCONN::ModelSEIRCONN( return; // Which roulette happen? - size_t which_v = std::floor(which / 2); - p->rm_virus(which_v, m); + p->rm_virus(m); return ; @@ -16716,6 +17083,21 @@ inline ModelSEIRCONN::ModelSEIRCONN( } +template +inline ModelSEIRCONN & ModelSEIRCONN::initial_states( + std::vector< double > proportions_, + std::vector< int > /* queue_ */ +) +{ + + Model::initial_states_fun = + create_init_function_seir(proportions_) + ; + + return *this; + +} + #endif /*////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// @@ -16740,13 +17122,6 @@ inline ModelSEIRCONN::ModelSEIRCONN( /** * @brief Template for a Susceptible-Infected-Removed-Deceased (SIRD) model - * - * @param model A Model object where to set up the SIRD. - * @param vname std::string Name of the virus - * @param initial_prevalence epiworld_double Initial prevalence - * @param initial_efficacy epiworld_double Initial susceptibility_reduction of the immune system - * @param initial_recovery epiworld_double Initial recovery_rate rate of the immune system - * @param initial_death epiworld_double Initial death_rate of the immune system */ template class ModelSIRD : public epiworld::Model @@ -16755,6 +17130,18 @@ class ModelSIRD : public epiworld::Model ModelSIRD() {}; + + /** + * @brief Constructs a new SIRD model with the given parameters. + * + * @param model The SIRD model to copy from. + * @param vname The name of the vertex associated with this model. + * @param prevalence The initial prevalence of the disease in the population. + * @param transmission_rate The rate at which the disease spreads from infected to susceptible individuals. + * @param recovery_rate The rate at which infected individuals recover and become immune. + * @param death_rate The rate at which infected individuals die. + */ + ///@{ ModelSIRD( ModelSIRD & model, std::string vname, @@ -16771,6 +17158,18 @@ class ModelSIRD : public epiworld::Model epiworld_double recovery_rate, epiworld_double death_rate ); + ///@} + + /** + * @brief Set the initial states of the model + * @param proportions_ Double vector with two elements: + * - The proportion of non-infected individuals who have recovered. + * - The proportion of non-infected individuals who have died. + */ + ModelSIRD & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); }; @@ -16835,6 +17234,20 @@ inline ModelSIRD::ModelSIRD( } +template +inline ModelSIRD & ModelSIRD::initial_states( + std::vector< double > proportions_, + std::vector< int > /**/ +) { + + Model::initial_states_fun = + create_init_function_sird(proportions_) + ; + + return *this; + +} + #endif /*////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// @@ -16977,15 +17390,7 @@ inline ModelSISD::ModelSISD( /** * @brief Template for a Susceptible-Exposed-Infected-Removed-Deceased (SEIRD) model - * - * @param model A Model object where to set up the SIR. - * @param vname std::string Name of the virus - * @param prevalence epiworld_double Initial prevalence the immune system - * @param transmission_rate epiworld_double Transmission rate of the virus - * @param avg_incubation_days epiworld_double Average incubation days of the virus - * @param recovery_rate epiworld_double Recovery rate of the virus. - * @param death_rate epiworld_double Death rate of the virus. - */ +*/ template class ModelSEIRD : public epiworld::Model { @@ -16999,6 +17404,18 @@ class ModelSEIRD : public epiworld::Model ModelSEIRD() {}; + /** + * @brief Constructor for the SEIRD model. + * + * @tparam TSeq Type of the sequence used in the model. + * @param model Reference to the SEIRD model. + * @param vname Name of the model. + * @param prevalence Prevalence of the disease. + * @param transmission_rate Transmission rate of the disease. + * @param avg_incubation_days Average incubation period of the disease. + * @param recovery_rate Recovery rate of the disease. + * @param death_rate Death rate of the disease. + */ ModelSEIRD( ModelSEIRD & model, std::string vname, @@ -17009,6 +17426,16 @@ class ModelSEIRD : public epiworld::Model epiworld_double death_rate ); + /** + * @brief Constructor for the SEIRD model. + * + * @param vname Name of the model. + * @param prevalence Initial prevalence of the disease. + * @param transmission_rate Transmission rate of the disease. + * @param avg_incubation_days Average incubation period of the disease. + * @param recovery_rate Recovery rate of the disease. + * @param death_rate Death rate of the disease. + */ ModelSEIRD( std::string vname, epiworld_double prevalence, @@ -17024,7 +17451,7 @@ class ModelSEIRD : public epiworld::Model ) -> void { // Getting the virus - auto v = p->get_virus(0); + auto v = p->get_virus(); // Does the agent become infected? if (m->runif() < 1.0/(v->get_incubation(m))) @@ -17037,23 +17464,20 @@ class ModelSEIRD : public epiworld::Model epiworld::UpdateFun update_infected = []( epiworld::Agent * p, epiworld::Model * m ) -> void { - - auto state = p->get_state(); - + // Odd: Die, Even: Recover epiworld_fast_uint n_events = 0u; - for (const auto & v : p->get_viruses()) - { - - // Die - m->array_double_tmp[n_events++] = - v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); - - // Recover - m->array_double_tmp[n_events++] = - 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); + + const auto & v = p->get_virus(); - } + // Die + m->array_double_tmp[n_events++] = + v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); + + // Recover + m->array_double_tmp[n_events++] = + 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); + #ifdef EPI_DEBUG if (n_events == 0u) @@ -17080,13 +17504,11 @@ class ModelSEIRD : public epiworld::Model if ((which % 2) == 0) // If odd { - size_t which_v = std::ceil(which / 2); - p->rm_agent_by_virus(which_v, m); + p->rm_agent_by_virus(m); } else { - size_t which_v = std::floor(which / 2); - p->rm_virus(which_v, m); + p->rm_virus(m); } @@ -17097,6 +17519,12 @@ class ModelSEIRD : public epiworld::Model return; }; + + ModelSEIRD & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); + }; @@ -17169,6 +17597,19 @@ inline ModelSEIRD::ModelSEIRD( } +template +inline ModelSEIRD & ModelSEIRD::initial_states( + std::vector< double > proportions_, + std::vector< int > /**/ +) { + + Model::initial_states_fun = + create_init_function_seird(proportions_) + ; + + return *this; + +} #endif @@ -17240,7 +17681,7 @@ class ModelSIRDCONN : public epiworld::Model // int tracked_ninfected_next = 0; // epiworld_double tracked_current_infect_prob = 0.0; - void run( + ModelSIRDCONN & run( epiworld_fast_uint ndays, int seed = -1 ); @@ -17253,21 +17694,16 @@ class ModelSIRDCONN : public epiworld::Model }; template -inline void ModelSIRDCONN::run( +inline ModelSIRDCONN & ModelSIRDCONN::run( epiworld_fast_uint ndays, int seed ) { - // tracked_agents_infected.clear(); - // tracked_agents_infected_next.clear(); - - // tracked_ninfected = 0; - // tracked_ninfected_next = 0; - // tracked_current_infect_prob = 0.0; - Model::run(ndays, seed); + return *this; + } template @@ -17372,24 +17808,21 @@ inline ModelSIRDCONN::ModelSIRDCONN( if (neighbor.get_state() == ModelSIRDCONN::INFECTED) { - for (const VirusPtr & v : neighbor.get_viruses()) - { - - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor.get_transmission_reduction(v, m)) - ; + const auto & v = neighbor.get_virus(); - m->array_virus_tmp[nviruses_tmp++] = &(*v); + #ifdef EPI_DEBUG + if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif - } + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor.get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); } } @@ -17404,7 +17837,7 @@ inline ModelSIRDCONN::ModelSIRDCONN( if (which < 0) return; - p->add_virus(*m->array_virus_tmp[which], m); + p->set_virus(*m->array_virus_tmp[which], m); return; @@ -17421,55 +17854,50 @@ inline ModelSIRDCONN::ModelSIRDCONN( { - // Odd: Die, Even: Recover - epiworld_fast_uint n_events = 0u; - for (const auto & v : p->get_viruses()) - { - + // Odd: Die, Even: Recover + epiworld_fast_uint n_events = 0u; + const auto & v = p->get_virus(); + // Die m->array_double_tmp[n_events++] = - v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); + v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); // Recover m->array_double_tmp[n_events++] = - 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); + 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); - } - -#ifdef EPI_DEBUG - if (n_events == 0u) - { - printf_epiworld( - "[epi-debug] agent %i has 0 possible events!!\n", - static_cast(p->get_id()) - ); - throw std::logic_error("Zero events in exposed."); - } -#else - if (n_events == 0u) - return; -#endif - - - // Running the roulette - int which = roulette(n_events, m); - - if (which < 0) - return; - - // Which roulette happen? - if ((which % 2) == 0) // If odd - { + #ifdef EPI_DEBUG + if (n_events == 0u) + { + printf_epiworld( + "[epi-debug] agent %i has 0 possible events!!\n", + static_cast(p->get_id()) + ); + throw std::logic_error("Zero events in exposed."); + } + #else + if (n_events == 0u) + return; + #endif - size_t which_v = std::ceil(which / 2); - p->rm_agent_by_virus(which_v, m); - } else { + // Running the roulette + int which = roulette(n_events, m); - size_t which_v = std::floor(which / 2); - p->rm_virus(which_v, m); + if (which < 0) + return; - } + // Which roulette happen? + if ((which % 2) == 0) // If odd + { + + p->rm_agent_by_virus(m); + + } else { + + p->rm_virus(m); + + } return ; @@ -17600,7 +18028,7 @@ class ModelSEIRDCONN : public epiworld::Model epiworld_double death_rate ); - void run( + ModelSEIRDCONN & run( epiworld_fast_uint ndays, int seed = -1 ); @@ -17609,10 +18037,21 @@ class ModelSEIRDCONN : public epiworld::Model Model * clone_ptr(); + /** + * @brief Set up the initial states of the model. + * @param proportions_ Double vector with the following values: + * - 0: Proportion of non-infected agents who are removed. + * - 1: Proportion of exposed agents to be set as infected. + */ + ModelSEIRDCONN & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); + }; template -inline void ModelSEIRDCONN::run( +inline ModelSEIRDCONN & ModelSEIRDCONN::run( epiworld_fast_uint ndays, int seed ) @@ -17620,6 +18059,8 @@ inline void ModelSEIRDCONN::run( Model::run(ndays, seed); + return *this; + } template @@ -17716,25 +18157,23 @@ inline ModelSEIRDCONN::ModelSEIRDCONN( if (neighbor.get_state() == ModelSEIRDCONN::INFECTED) { - for (const VirusPtr & v : neighbor.get_viruses()) - { + const auto & v = neighbor.get_virus(); - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor.get_transmission_reduction(v, m)) - ; - - m->array_virus_tmp[nviruses_tmp++] = &(*v); + + #ifdef EPI_DEBUG + if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif - } - + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor.get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); + } } @@ -17748,7 +18187,7 @@ inline ModelSEIRDCONN::ModelSEIRDCONN( if (which < 0) return; - p->add_virus( + p->set_virus( *m->array_virus_tmp[which], m, ModelSEIRDCONN::EXPOSED @@ -17768,7 +18207,7 @@ inline ModelSEIRDCONN::ModelSEIRDCONN( { // Getting the virus - auto & v = p->get_virus(0u); + auto & v = p->get_virus(); // Does the agent become infected? if (m->runif() < 1.0/(v->get_incubation(m))) @@ -17783,56 +18222,50 @@ inline ModelSEIRDCONN::ModelSEIRDCONN( } else if (state == ModelSEIRDCONN::INFECTED) { - - // Odd: Die, Even: Recover - epiworld_fast_uint n_events = 0u; - for (const auto & v : p->get_viruses()) - { + // Odd: Die, Even: Recover + epiworld_fast_uint n_events = 0u; + const auto & v = p->get_virus(); // Die m->array_double_tmp[n_events++] = - v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); + v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); // Recover m->array_double_tmp[n_events++] = - 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); - - } - - #ifdef EPI_DEBUG - if (n_events == 0u) - { + 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); + + #ifdef EPI_DEBUG + if (n_events == 0u) + { printf_epiworld( - "[epi-debug] agent %i has 0 possible events!!\n", - static_cast(p->get_id()) + "[epi-debug] agent %i has 0 possible events!!\n", + static_cast(p->get_id()) ); throw std::logic_error("Zero events in exposed."); - } - #else - if (n_events == 0u) + } + #else + if (n_events == 0u) return; - #endif - - - // Running the roulette - int which = roulette(n_events, m); - - if (which < 0) + #endif + + + // Running the roulette + int which = roulette(n_events, m); + + if (which < 0) return; - - // Which roulette happen? - if ((which % 2) == 0) // If odd - { - size_t which_v = std::ceil(which / 2); - p->rm_agent_by_virus(which_v, m); + // Which roulette happen? + if ((which % 2) == 0) // If odd + { + + p->rm_agent_by_virus(m); - } else { + } else { - size_t which_v = std::floor(which / 2); - p->rm_virus(which_v, m); + p->rm_virus(m); - } + } return ; @@ -17912,6 +18345,20 @@ inline ModelSEIRDCONN::ModelSEIRDCONN( } +template +inline ModelSEIRDCONN & ModelSEIRDCONN::initial_states( + std::vector< double > proportions_, + std::vector< int > /**/ +) { + + Model::initial_states_fun = + create_init_function_seir(proportions_) + ; + + return *this; + +} + #endif /*////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// @@ -17984,7 +18431,7 @@ class ModelSIRLogit : public epiworld::Model epiworld_double prevalence ); - void run( + ModelSIRLogit & run( epiworld_fast_uint ndays, int seed = -1 ); @@ -18003,13 +18450,14 @@ class ModelSIRLogit : public epiworld::Model template -inline void ModelSIRLogit::run( +inline ModelSIRLogit & ModelSIRLogit::run( epiworld_fast_uint ndays, int seed ) { Model::run(ndays, seed); + return *this; } @@ -18122,30 +18570,30 @@ inline ModelSIRLogit::ModelSIRLogit( for (auto & neighbor: p->get_neighbors()) { - for (const VirusPtr & v : neighbor->get_viruses()) - { + if (neighbor->get_virus() == nullptr) + continue; - #ifdef EPI_DEBUG - if (nviruses_tmp >= m->array_virus_tmp.size()) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - baseline + - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) * - coef_exposure - ; + auto & v = neighbor->get_virus(); - // Applying the plogis function - m->array_double_tmp[nviruses_tmp] = 1.0/ - (1.0 + std::exp(-m->array_double_tmp[nviruses_tmp])); - - m->array_virus_tmp[nviruses_tmp++] = &(*v); + #ifdef EPI_DEBUG + if (nviruses_tmp >= m->array_virus_tmp.size()) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif + + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + baseline + + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) * + coef_exposure + ; - } + // Applying the plogis function + m->array_double_tmp[nviruses_tmp] = 1.0/ + (1.0 + std::exp(-m->array_double_tmp[nviruses_tmp])); + + m->array_virus_tmp[nviruses_tmp++] = &(*v); } @@ -18159,7 +18607,7 @@ inline ModelSIRLogit::ModelSIRLogit( if (which < 0) return; - p->add_virus(*m->array_virus_tmp[which], m); + p->set_virus(*m->array_virus_tmp[which], m); return; @@ -18175,7 +18623,9 @@ inline ModelSIRLogit::ModelSIRLogit( // Computing recovery probability once double prob = 0.0; + #if defined(__OPENMP) || defined(_OPENMP) #pragma omp simd reduction(+:prob) + #endif for (size_t i = 0u; i < _m->coefs_recover.size(); ++i) prob += p->operator[](i) * _m->coefs_recover[i]; @@ -18183,7 +18633,7 @@ inline ModelSIRLogit::ModelSIRLogit( prob = 1.0/(1.0 + std::exp(-prob)); if (prob > m->runif()) - p->rm_virus(0, m); + p->rm_virus(m); return; @@ -18367,24 +18817,25 @@ inline ModelDiffNet::ModelDiffNet( if (neighbor->get_state() == ModelDiffNet::ADOPTER) { - for (const VirusPtr & v : neighbor->get_viruses()) - { - - /* And it is a function of susceptibility_reduction as well */ - double p_i = - (1.0 - agent.get_susceptibility_reduction(v, m)) * - (1.0 - agent.get_transmission_reduction(v, m)) - ; + auto & v = neighbor->get_virus(); - size_t vid = v->get_id(); - if (!stored[vid]) - { - stored[vid] = true; - innovations[vid] = &(*v); - } - exposure[vid] += p_i; - - } + if (v == nullptr) + continue; + + /* And it is a function of susceptibility_reduction as well */ + double p_i = + (1.0 - agent.get_susceptibility_reduction(v, m)) * + (1.0 - agent.get_transmission_reduction(v, m)) + ; + + size_t vid = v->get_id(); + if (!stored[vid]) + { + stored[vid] = true; + innovations[vid] = &(*v); + } + exposure[vid] += p_i; + } @@ -18417,7 +18868,7 @@ inline ModelDiffNet::ModelDiffNet( return; // Otherwise, it is adopted from any of the neighbors - agent.add_virus( + agent.set_virus( *innovations.at(which), m, ModelDiffNet::ADOPTER @@ -18507,8 +18958,6 @@ inline ModelDiffNet::ModelDiffNet( - - } #endif diff --git a/examples/00-hello-world/Makefile b/examples/00-hello-world/Makefile index 1dc0bde3a..cedd25a24 100755 --- a/examples/00-hello-world/Makefile +++ b/examples/00-hello-world/Makefile @@ -1,5 +1,6 @@ main.o: main.cpp - g++ -std=c++11 -Wall -pedantic -g -O2 -ftree-vectorize main.cpp -o main.o + g++ -std=c++11 -Wall -pedantic -fopenmp -g -O2 -ftree-vectorize main.cpp -o main.o + README.md: main.o echo "## Example: 00-hello-world" > README.md && \ echo "" >> README.md && \ diff --git a/examples/00-hello-world/README.md b/examples/00-hello-world/README.md index be1fdfbe0..28aeeb110 100644 --- a/examples/00-hello-world/README.md +++ b/examples/00-hello-world/README.md @@ -3,40 +3,45 @@ Output from the program: ``` -Running the model... _________________________________________________________________________ +Running the model... ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||| done. -[epiworld-debug] DEBUGGING ON (compiled with EPI_DEBUG defined) - + done. +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY +Name of the model : (none) Population size : 10000 +Agents' data : (none) Number of entities : 0 Days (duration) : 100 (of 100) -Number of variants : 1 -Last run elapsed t : 40.00ms +Number of viruses : 1 +Last run elapsed t : 16.00ms +Last run speed : 59.75 million agents x day / second Rewiring : off +Global actions: + (none) + Virus(es): - covid 19 (baseline prevalence: 50 seeds) Tool(s): - vaccine (baseline prevalence: 50.00%) - - Immunity (covid 19) (originated in the model...) Model parameters: (none) Distribution of the population at time 100: - - (0) Susceptible : 9950 -> 70 - - (1) Exposed : 50 -> 70 - - (2) Recovered : 0 -> 9271 - - (3) Removed : 0 -> 589 + - (0) Susceptible : 9950 -> 0 + - (1) Exposed : 50 -> 0 + - (2) Recovered : 0 -> 9399 + - (3) Removed : 0 -> 601 Transition Probabilities: - - Susceptible 0.95 0.05 0.00 0.00 - - Exposed 0.00 0.85 0.14 0.01 + - Susceptible 0.87 0.13 0.00 0.00 + - Exposed 0.00 0.83 0.15 0.01 - Recovered 0.00 0.00 1.00 0.00 - Removed 0.00 0.00 0.00 1.00 diff --git a/examples/00-hello-world/main.cpp b/examples/00-hello-world/main.cpp index 1809a7f3e..dc4d93fcc 100644 --- a/examples/00-hello-world/main.cpp +++ b/examples/00-hello-world/main.cpp @@ -1,5 +1,5 @@ -#define EPI_DEBUG +// #define EPI_DEBUG #include "../../include/epiworld/epiworld.hpp" using namespace epiworld; @@ -27,7 +27,7 @@ int main() model.add_tool(tool, .5); // Generating a random pop - model.agents_smallworld(10000); + model.agents_smallworld(10000, 20, false, .01); // Running the model model.run(100, 123); @@ -42,7 +42,8 @@ int main() "total_hist.txt", "transmissions.txt", "transitions.txt", - "reproductive.txt" + "reproductive.txt", + "generation_time.txt" ); } diff --git a/examples/01-seir/Makefile b/examples/01-seir/Makefile index 53a5528e6..7f55b84ed 100755 --- a/examples/01-seir/Makefile +++ b/examples/01-seir/Makefile @@ -1,5 +1,5 @@ main.o: main.cpp - g++ -std=c++11 -Wall -pedantic -g -O2 -mtune=native main.cpp -o main.o + g++ -std=c++14 -Wall -pedantic -fopenmp -g -O2 -mtune=native main.cpp -o main.o README.md: main.o echo "## Example: 01-sir" > README.md && \ echo "" >> README.md && \ diff --git a/examples/01-seir/main.cpp b/examples/01-seir/main.cpp index 7728f5f89..ae071a85a 100644 --- a/examples/01-seir/main.cpp +++ b/examples/01-seir/main.cpp @@ -14,7 +14,7 @@ int main() { // Adding a bernoulli graph as step 0 model.agents_from_adjlist( - rgraph_smallworld(500000, 5, .001, false, model) + rgraph_smallworld(1000000, 5, .001, false, model) ); // Running and checking the results diff --git a/examples/01-sir/README.md b/examples/01-sir/README.md index 97d24b3b4..40e3367a5 100644 --- a/examples/01-sir/README.md +++ b/examples/01-sir/README.md @@ -4,19 +4,24 @@ Output from the program: ``` Running the model... -_________________________________________________________________________ -||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||| done. - +DEBUGGING ON (compiled with EPI_DEBUG defined) +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY -Population size : 10000 +Name of the model : Susceptible-Infected-Recovered (SIR) +Population size : 50000 +Agents' data : (none) Number of entities : 0 -Days (duration) : 100 (of 100) -Number of variants : 1 -Last run elapsed t : 17.00ms +Days (duration) : 50 (of 50) +Number of viruses : 1 +Last run elapsed t : 70.00ms +Last run speed : 35.62 million agents x day / second Rewiring : off +Global actions: + (none) + Virus(es): - a virus (baseline prevalence: 1.00%) @@ -24,17 +29,17 @@ Tool(s): (none) Model parameters: - - Infectiousness : 0.9000 - - Prob. of Recovery : 0.5000 + - Recovery rate : 0.5000 + - Transmission rate : 0.9000 -Distribution of the population at time 100: - - (0) Susceptible : 9900 -> 0 - - (1) Infected : 100 -> 0 - - (2) Recovered : 0 -> 10000 +Distribution of the population at time 50: + - (0) Susceptible : 49500 -> 0 + - (1) Infected : 500 -> 0 + - (2) Recovered : 0 -> 50000 Transition Probabilities: - - Susceptible 0.89 0.11 0.00 - - Infected 0.00 0.50 0.50 + - Susceptible 0.81 0.19 0.00 + - Infected 0.00 0.49 0.51 - Recovered 0.00 0.00 1.00 ``` diff --git a/examples/01-sir/main.cpp b/examples/01-sir/main.cpp index 58aabb069..c5091066e 100644 --- a/examples/01-sir/main.cpp +++ b/examples/01-sir/main.cpp @@ -1,3 +1,4 @@ +#define EPI_DEBUG #include "../../include/epiworld/epiworld.hpp" using namespace epiworld; @@ -13,7 +14,7 @@ int main() { // Adding a bernoulli graph as step 0 model.agents_from_adjlist( - rgraph_smallworld(100000, 5, .001, false, model) + rgraph_smallworld(50000, 20, .01, false, model) ); // Running and checking the results diff --git a/examples/02-sir_multiple_runs/README.md b/examples/02-sir_multiple_runs/README.md index eb553f714..fc86d9e55 100644 --- a/examples/02-sir_multiple_runs/README.md +++ b/examples/02-sir_multiple_runs/README.md @@ -3,31 +3,6 @@ Output from the program: ``` - -________________________________________________________________________________ -SIMULATION STUDY - -Population size : 1000 -Number of entities : 0 -Days (duration) : 0 (of 60) -Number of variants : 1 -Last run elapsed t : - -Rewiring : off - -Virus(es): - - a virus (baseline prevalence: 1.00%) - -Tool(s): - (none) - -Model parameters: - - Infectiousness : 0.9000 - - Prob. of Recovery : 0.3000 - -Distribution of the population at time 0: - - (0) Susceptible : 990 - - (1) Infected : 10 - - (2) Recovered : 0 Replicate 0 done Replicate 1 done Replicate 2 done @@ -129,18 +104,18 @@ Replicate 97 done Replicate 98 done Replicate 99 done last run elapsed time : 0.00ms -total elapsed time : 101.00ms +total elapsed time : 56.00ms total runs : 100 -mean run elapsed time : 1.01ms +mean run elapsed time : 0.56ms Susceptible, Infected, Recovered, -0, 0, 1000, -0, 0, 1000, -0, 0, 1000, -0, 0, 1000, -0, 0, 1000, -0, 0, 1000, -0, 0, 1000, -0, 0, 1000, -0, 0, 1000, -0, 0, 1000, +167, 25, 808, +71, 13, 916, +302, 13, 685, +0, 5, 995, +126, 15, 859, +187, 24, 789, +13, 9, 978, +146, 9, 845, +7, 13, 980, +76, 26, 898, ``` diff --git a/examples/02b-sir_multiple_runs/README.md b/examples/02b-sir_multiple_runs/README.md index 29a939ec3..926cdf45f 100644 --- a/examples/02b-sir_multiple_runs/README.md +++ b/examples/02b-sir_multiple_runs/README.md @@ -3,17 +3,30 @@ Output from the program: ``` - +Starting multiple runs (100) +_________________________________________________________________________ +_________________________________________________________________________ +||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||| done. + done. +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY +Name of the model : Susceptible-Infected-Recovered (SIR) Population size : 10000 +Agents' data : (none) Number of entities : 0 -Days (duration) : 0 (of 60) -Number of variants : 1 -Last run elapsed t : - +Days (duration) : 60 (of 60) +Number of viruses : 1 +Last run elapsed t : 5.00ms +Total elapsed t : 580.00ms (100 runs) +Last run speed : 104.73 million agents x day / second +Average run speed : 103.36 million agents x day / second Rewiring : off +Global actions: + (none) + Virus(es): - a virus (baseline prevalence: 1.00%) @@ -21,20 +34,21 @@ Tool(s): (none) Model parameters: - - Infectiousness : 0.9000 - - Prob. of Recovery : 0.3000 + - Recovery rate : 0.3000 + - Transmission rate : 0.9000 -Distribution of the population at time 0: - - (0) Susceptible : 9900 - - (1) Infected : 100 - - (2) Recovered : 0 -Starting multiple runs (100) -_________________________________________________________________________ -_________________________________________________________________________ -||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||| done. - done. -last run elapsed time : 0.00s -total elapsed time : 1.00s +Distribution of the population at time 60: + - (0) Susceptible : 9900 -> 1075 + - (1) Infected : 100 -> 152 + - (2) Recovered : 0 -> 8773 + +Transition Probabilities: + - Susceptible 0.96 0.04 0.00 + - Infected 0.00 0.70 0.30 + - Recovered 0.00 0.00 1.00 + +last run elapsed time : 5.00ms +total elapsed time : 580.00ms total runs : 100 -mean run elapsed time : 0.01s +mean run elapsed time : 5.80ms ``` diff --git a/examples/04-advanced-usage/README.md b/examples/04-advanced-usage/README.md index e55819137..8c0b98460 100644 --- a/examples/04-advanced-usage/README.md +++ b/examples/04-advanced-usage/README.md @@ -3,18 +3,23 @@ Output from the program: ``` -[epiworld-debug] DEBUGGING ON (compiled with EPI_DEBUG defined) - +DEBUGGING ON (compiled with EPI_DEBUG defined) +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY +Name of the model : (none) Population size : 1000 +Agents' data : (none) Number of entities : 0 -Days (duration) : 0 (of 60) -Number of variants : 1 +Days (duration) : 0 (of 0) +Number of viruses : 1 Last run elapsed t : - Rewiring : on (0.10) +Global actions: + (none) + Virus(es): - COVID19 (baseline prevalence: 1.00%) @@ -33,31 +38,27 @@ Model parameters: - vax death : 1.0e-04 - vax efficacy : 0.9000 - virus death : 0.0100 - -Distribution of the population at time 0: - - (0) Susceptible : 990 - - (1) Exposed : 10 - - (2) Recovered : 0 - - (3) Removed : 0 Running the model... -_________________________________________________________________________ -||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||| done. -[epiworld-debug] DEBUGGING ON (compiled with EPI_DEBUG defined) - +DEBUGGING ON (compiled with EPI_DEBUG defined) +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY +Name of the model : (none) Population size : 1000 +Agents' data : (none) Number of entities : 0 Days (duration) : 60 (of 60) -Number of variants : 3 -Last run elapsed t : 63.00ms +Number of viruses : 1 +Last run elapsed t : 35.00ms +Last run speed : 1.68 million agents x day / second Rewiring : on (0.10) +Global actions: + (none) + Virus(es): - COVID19 (baseline prevalence: 1.00%) - - COVID19 (originated in the model...) - - COVID19 (originated in the model...) Tool(s): - Immune system (baseline prevalence: 100.00%) @@ -76,14 +77,14 @@ Model parameters: - virus death : 0.0100 Distribution of the population at time 60: - - (0) Susceptible : 990 -> 751 - - (1) Exposed : 10 -> 20 - - (2) Recovered : 0 -> 220 - - (3) Removed : 0 -> 9 + - (0) Susceptible : 990 -> 901 + - (1) Exposed : 10 -> 0 + - (2) Recovered : 0 -> 97 + - (3) Removed : 0 -> 2 Transition Probabilities: - Susceptible 1.00 0.00 0.00 0.00 - - Exposed 0.00 0.78 0.21 0.01 + - Exposed 0.00 0.75 0.24 0.00 - Recovered 0.00 0.00 1.00 0.00 - Removed 0.00 0.00 0.00 1.00 diff --git a/examples/04-advanced-usage/main.cpp b/examples/04-advanced-usage/main.cpp index c52dc2ce7..8f750236d 100644 --- a/examples/04-advanced-usage/main.cpp +++ b/examples/04-advanced-usage/main.cpp @@ -121,7 +121,8 @@ int main() { "total.txt", "transmisions.txt", "transition.txt", - "reproductive.txt" + "reproductive.txt", + "" ); model.write_edgelist("simple-world-edgelist.txt"); diff --git a/examples/06-sir-omp/README.md b/examples/06-sir-omp/README.md index decbf7f63..c5ad1f761 100644 --- a/examples/06-sir-omp/README.md +++ b/examples/06-sir-omp/README.md @@ -4,17 +4,23 @@ Output from the program: ``` Generating random graph... done. - +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY +Name of the model : Susceptible-Infected-Recovered (SIR) Population size : 250000 +Agents' data : (none) Number of entities : 0 Days (duration) : 100 (of 100) -Number of variants : 1 -Last run elapsed t : 1.00s +Number of viruses : 1 +Last run elapsed t : 706.00ms +Last run speed : 35.37 million agents x day / second Rewiring : off +Global actions: + (none) + Virus(es): - a virus (baseline prevalence: 1.00%) @@ -22,30 +28,36 @@ Tool(s): (none) Model parameters: - - Infectiousness : 0.9000 - - Prob. of Recovery : 0.3000 + - Recovery rate : 0.3000 + - Transmission rate : 0.9000 Distribution of the population at time 100: - - (0) Susceptible : 247500 -> 0 - - (1) Infected : 2500 -> 0 - - (2) Recovered : 0 -> 250000 + - (0) Susceptible : 247500 -> 7640 + - (1) Infected : 2500 -> 908 + - (2) Recovered : 0 -> 241452 Transition Probabilities: - - Susceptible 0.87 0.13 0.00 - - Infected 0.00 0.71 0.29 + - Susceptible 0.97 0.03 0.00 + - Infected 0.00 0.70 0.30 - Recovered 0.00 0.00 1.00 - +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY +Name of the model : Susceptible-Infected-Recovered (SIR) Population size : 250000 +Agents' data : (none) Number of entities : 0 Days (duration) : 100 (of 100) -Number of variants : 1 -Last run elapsed t : 1.00s +Number of viruses : 1 +Last run elapsed t : 651.00ms +Last run speed : 38.34 million agents x day / second Rewiring : off +Global actions: + (none) + Virus(es): - a virus (baseline prevalence: 1.00%) @@ -53,30 +65,36 @@ Tool(s): (none) Model parameters: - - Infectiousness : 0.9000 - - Prob. of Recovery : 0.3000 + - Recovery rate : 0.3000 + - Transmission rate : 0.9000 Distribution of the population at time 100: - - (0) Susceptible : 247500 -> 0 - - (1) Infected : 2500 -> 0 - - (2) Recovered : 0 -> 250000 + - (0) Susceptible : 247500 -> 7167 + - (1) Infected : 2500 -> 876 + - (2) Recovered : 0 -> 241957 Transition Probabilities: - - Susceptible 0.87 0.13 0.00 + - Susceptible 0.97 0.03 0.00 - Infected 0.00 0.70 0.30 - Recovered 0.00 0.00 1.00 - +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY +Name of the model : Susceptible-Infected-Recovered (SIR) Population size : 250000 +Agents' data : (none) Number of entities : 0 Days (duration) : 100 (of 100) -Number of variants : 1 -Last run elapsed t : 1.00s +Number of viruses : 1 +Last run elapsed t : 710.00ms +Last run speed : 35.19 million agents x day / second Rewiring : off +Global actions: + (none) + Virus(es): - a virus (baseline prevalence: 1.00%) @@ -84,30 +102,36 @@ Tool(s): (none) Model parameters: - - Infectiousness : 0.9000 - - Prob. of Recovery : 0.3000 + - Recovery rate : 0.3000 + - Transmission rate : 0.9000 Distribution of the population at time 100: - - (0) Susceptible : 247500 -> 0 - - (1) Infected : 2500 -> 0 - - (2) Recovered : 0 -> 250000 + - (0) Susceptible : 247500 -> 6076 + - (1) Infected : 2500 -> 814 + - (2) Recovered : 0 -> 243110 Transition Probabilities: - - Susceptible 0.86 0.14 0.00 - - Infected 0.00 0.71 0.29 + - Susceptible 0.96 0.04 0.00 + - Infected 0.00 0.70 0.30 - Recovered 0.00 0.00 1.00 - +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY +Name of the model : Susceptible-Infected-Recovered (SIR) Population size : 250000 +Agents' data : (none) Number of entities : 0 Days (duration) : 100 (of 100) -Number of variants : 1 -Last run elapsed t : 1.00s +Number of viruses : 1 +Last run elapsed t : 651.00ms +Last run speed : 38.38 million agents x day / second Rewiring : off +Global actions: + (none) + Virus(es): - a virus (baseline prevalence: 1.00%) @@ -115,22 +139,22 @@ Tool(s): (none) Model parameters: - - Infectiousness : 0.9000 - - Prob. of Recovery : 0.3000 + - Recovery rate : 0.3000 + - Transmission rate : 0.9000 Distribution of the population at time 100: - - (0) Susceptible : 247500 -> 0 - - (1) Infected : 2500 -> 0 - - (2) Recovered : 0 -> 250000 + - (0) Susceptible : 247500 -> 5738 + - (1) Infected : 2500 -> 765 + - (2) Recovered : 0 -> 243497 Transition Probabilities: - - Susceptible 0.87 0.12 0.00 - - Infected 0.00 0.69 0.31 + - Susceptible 0.96 0.04 0.00 + - Infected 0.00 0.70 0.30 - Recovered 0.00 0.00 1.00 -last run elapsed time : 1040.00ms. -last run elapsed time : 1056.00ms. -last run elapsed time : 1101.00ms. -last run elapsed time : 1053.00ms. -Elapsed time: 1125 milliseconds +last run elapsed time : 706.00ms. +last run elapsed time : 651.00ms. +last run elapsed time : 710.00ms. +last run elapsed time : 651.00ms. +Elapsed time: 726 milliseconds ``` diff --git a/examples/06b-sir-omp/README.md b/examples/06b-sir-omp/README.md index 73e11bc4f..2faebb16e 100644 --- a/examples/06b-sir-omp/README.md +++ b/examples/06b-sir-omp/README.md @@ -3,27 +3,30 @@ Output from the program: ``` -Starting multiple runs (20) using 2 threads +Starting multiple runs (20) using 2 thread(s) _________________________________________________________________________ _________________________________________________________________________ ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||| done. done. -Elapsed time: 3887 milliseconds - +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY Name of the model : Susceptible-Infected-Recovered (SIR) -Population size : 100000 +Population size : 10000 +Agents' data : (none) Number of entities : 0 Days (duration) : 100 (of 100) -Number of variants : 1 -Last run elapsed t : 0.00s -Total elapsed t : 3.00s (20 runs) -Last run speed : 31.53 million agents x day / second -Average run speed : 63.60 million agents x day / second +Number of viruses : 1 +Last run elapsed t : 10.00ms +Total elapsed t : 105.00ms (20 runs) +Last run speed : 99.18 million agents x day / second +Average run speed : 189.38 million agents x day / second Rewiring : off +Global actions: + (none) + Virus(es): - a virus (baseline prevalence: 1.00%) @@ -31,17 +34,17 @@ Tool(s): (none) Model parameters: - - Infectiousness : 0.9000 - - Prob. of Recovery : 0.3000 + - Recovery rate : 0.3000 + - Transmission rate : 0.9000 Distribution of the population at time 100: - - (0) Susceptible : 99000 -> 3837 - - (1) Infected : 1000 -> 338 - - (2) Recovered : 0 -> 95825 + - (0) Susceptible : 9900 -> 226 + - (1) Infected : 100 -> 28 + - (2) Recovered : 0 -> 9746 Transition Probabilities: - - Susceptible 0.97 0.03 0.00 - - Infected 0.00 0.70 0.30 + - Susceptible 0.96 0.04 0.00 + - Infected 0.00 0.71 0.29 - Recovered 0.00 0.00 1.00 ``` diff --git a/examples/07-surveillance/README.md b/examples/07-surveillance/README.md index 1a40e48ab..c2b02c4f2 100644 --- a/examples/07-surveillance/README.md +++ b/examples/07-surveillance/README.md @@ -3,18 +3,23 @@ Output from the program: ``` -[epiworld-debug] DEBUGGING ON (compiled with EPI_DEBUG defined) - +DEBUGGING ON (compiled with EPI_DEBUG defined) +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY +Name of the model : Surveillance Population size : 10000 +Agents' data : (none) Number of entities : 0 -Days (duration) : 0 (of 100) -Number of variants : 1 +Days (duration) : 0 (of 0) +Number of viruses : 1 Last run elapsed t : - Rewiring : off +Global actions: + - Surveilance program (runs daily) + Virus(es): - Covid19 (baseline prevalence: 10 seeds) @@ -31,37 +36,30 @@ Model parameters: - Surveilance prob. : 0.0010 - Vax efficacy : 0.9000 - Vax redux transmission : 0.5000 - -Distribution of the population at time 0: - - (0) Susceptible : 9990 - - (1) Latent : 10 - - (2) Symptomatic : 0 - - (3) Symptomatic isolated : 0 - - (4) Asymptomatic : 0 - - (5) Asymptomatic isolated : 0 - - (6) Recovered : 0 - - (7) Removed : 0 Running the model... -_________________________________________________________________________ -||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||| done. -[epiworld-debug] DEBUGGING ON (compiled with EPI_DEBUG defined) - +DEBUGGING ON (compiled with EPI_DEBUG defined) +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY +Name of the model : Surveillance Population size : 10000 +Agents' data : (none) Number of entities : 0 Days (duration) : 100 (of 100) -Number of variants : 1 -Last run elapsed t : 12.00ms +Number of viruses : 1 +Last run elapsed t : 3.00ms +Last run speed : 252.72 million agents x day / second Rewiring : off +Global actions: + - Surveilance program (runs daily) + Virus(es): - Covid19 (baseline prevalence: 10 seeds) Tool(s): - Vaccine (baseline prevalence: 25.00%) - - Immunity (Covid19) (originated in the model...) Model parameters: - Infect period : 12.0000 @@ -75,22 +73,22 @@ Model parameters: - Vax redux transmission : 0.5000 Distribution of the population at time 100: - - (0) Susceptible : 9990 -> 8068 - - (1) Latent : 10 -> 46 - - (2) Symptomatic : 0 -> 158 - - (3) Symptomatic isolated : 0 -> 2 - - (4) Asymptomatic : 0 -> 70 - - (5) Asymptomatic isolated : 0 -> 1 - - (6) Recovered : 0 -> 1635 - - (7) Removed : 0 -> 20 + - (0) Susceptible : 9990 -> 9484 + - (1) Latent : 10 -> 13 + - (2) Symptomatic : 0 -> 23 + - (3) Symptomatic isolated : 0 -> 1 + - (4) Asymptomatic : 0 -> 14 + - (5) Asymptomatic isolated : 0 -> 0 + - (6) Recovered : 0 -> 457 + - (7) Removed : 0 -> 8 Transition Probabilities: - Susceptible 1.00 0.00 0.00 0.00 0.00 0.00 0.00 0.00 - - Latent 0.00 0.68 0.23 0.00 0.09 0.00 0.00 0.00 - - Symptomatic 0.00 0.00 0.94 0.00 0.00 0.00 0.06 0.00 - - Symptomatic isolated 0.00 0.00 0.00 0.92 0.00 0.00 0.08 0.00 - - Asymptomatic 0.00 0.00 0.00 0.00 0.94 0.00 0.06 0.00 - - Asymptomatic isolated 0.00 0.00 0.00 0.00 0.00 0.84 0.16 0.00 + - Latent 0.00 0.74 0.17 0.00 0.09 0.00 0.00 0.00 + - Symptomatic 0.00 0.00 0.91 0.00 0.00 0.00 0.09 0.00 + - Symptomatic isolated 0.00 0.00 0.00 0.87 0.00 0.00 0.13 0.00 + - Asymptomatic 0.00 0.00 0.00 0.00 0.90 0.00 0.09 0.00 + - Asymptomatic isolated 0.00 0.00 0.00 0.00 0.00 0.88 0.12 0.00 - Recovered 0.00 0.00 0.00 0.00 0.00 0.00 1.00 0.00 - Removed 0.00 0.00 0.00 0.00 0.00 0.00 0.00 1.00 diff --git a/examples/07-surveillance/main.cpp b/examples/07-surveillance/main.cpp index 01926404a..ea1a1c4cd 100644 --- a/examples/07-surveillance/main.cpp +++ b/examples/07-surveillance/main.cpp @@ -46,7 +46,7 @@ int main(int argc, char* argv[]) { model.print(); model.write_data( - "","", "", "", "07-surveillance_hist.txt", "", "", "" + "","", "", "", "07-surveillance_hist.txt", "", "", "", "" ); model.get_user_data().write("07-surveillance_user_data.txt"); diff --git a/examples/10-likelihood-free-mcmc/README.md b/examples/10-likelihood-free-mcmc/README.md index 500f77e71..eafa9ff08 100644 --- a/examples/10-likelihood-free-mcmc/README.md +++ b/examples/10-likelihood-free-mcmc/README.md @@ -3,17 +3,23 @@ Output from the program: ``` - +________________________________________________________________________________ ________________________________________________________________________________ SIMULATION STUDY +Name of the model : Susceptible-Infected-Recovered (SIR) Population size : 1000 +Agents' data : (none) Number of entities : 0 Days (duration) : 50 (of 50) -Number of variants : 1 -Last run elapsed t : 1.00ms +Number of viruses : 1 +Last run elapsed t : 685.00µs +Last run speed : 72.99 million agents x day / second Rewiring : off +Global actions: + (none) + Virus(es): - covid (baseline prevalence: 10.00%) @@ -21,17 +27,17 @@ Tool(s): (none) Model parameters: - - Infectiousness : 0.9000 - - Prob. of Recovery : 0.3000 + - Recovery rate : 0.3000 + - Transmission rate : 0.9000 Distribution of the population at time 50: - - (0) Susceptible : 900 -> 0 - - (1) Infected : 100 -> 0 - - (2) Recovered : 0 -> 1000 + - (0) Susceptible : 900 -> 0 + - (1) Infected : 100 -> 0 + - (2) Recovered : 0 -> 1000 Transition Probabilities: - - Susceptible 0.24 0.75 0.01 - - Infected 0.00 0.83 0.17 + - Susceptible 0.60 0.40 0.00 + - Infected 0.00 0.70 0.30 - Recovered 0.00 0.00 1.00 ___________________________________________ @@ -39,16 +45,16 @@ ___________________________________________ LIKELIHOOD-FREE MARKOV CHAIN MONTE CARLO N Samples : 1000 -Elapsed t : 1.00s +Elapsed t : 663.00ms Parameters: - -Immune recovery : 0.50 [ 0.15, 0.96] (initial : 0.50) - -Infectiousness : 0.70 [ 0.27, 0.99] (initial : 0.50) + -Immune recovery : 0.58 [ 0.13, 0.96] (initial : 0.50) + -Infectiousness : 0.87 [ 0.58, 0.99] (initial : 0.50) Statistics: - -Susceptible : 0.18 [ 0.00, 2.00] (Observed: 0.00) - -Infected : 0.03 [ 0.00, 1.00] (Observed: 0.00) - -Recovered : 994.79 [ 998.00, 1000.00] (Observed: 1000.00) + -Susceptible : 0.32 [ 0.00, 2.00] (Observed: 0.00) + -Infected : 0.08 [ 0.00, 1.00] (Observed: 0.00) + -Recovered : 998.61 [ 998.00, 1000.00] (Observed: 1000.00) ___________________________________________ ``` diff --git a/examples/10-likelihood-free-mcmc/main.cpp b/examples/10-likelihood-free-mcmc/main.cpp index cc3df3a7a..af2ba1819 100644 --- a/examples/10-likelihood-free-mcmc/main.cpp +++ b/examples/10-likelihood-free-mcmc/main.cpp @@ -13,8 +13,8 @@ std::vector< int > simfun( LFMCMC> * m ) { - model("Prob. of Recovery") = params[0u]; - model("Infectiousness") = params[1u]; + model("Recovery rate") = params[0u]; + model("Transmission rate") = params[1u]; model.reset(); model.run(50); diff --git a/include/epiworld/agent-actions-meat.hpp b/include/epiworld/agent-actions-meat.hpp index 20cc829cf..ef078cbbd 100644 --- a/include/epiworld/agent-actions-meat.hpp +++ b/include/epiworld/agent-actions-meat.hpp @@ -8,9 +8,6 @@ inline void default_add_virus(Action & a, Model * m) Agent * p = a.agent; VirusPtr v = a.virus; - CHECK_COALESCE_(a.new_state, v->state_init, p->get_state()) - CHECK_COALESCE_(a.queue, v->queue_init, 1) - // Has a agent? If so, we need to register the transmission if (v->get_agent()) { @@ -26,30 +23,22 @@ inline void default_add_virus(Action & a, Model * m) } - // Update virus accounting - p->n_viruses++; - size_t n_viruses = p->n_viruses; - - if (n_viruses <= p->viruses.size()) - p->viruses[n_viruses - 1] = std::make_shared< Virus >(*v); - else - p->viruses.push_back(std::make_shared< Virus >(*v)); - - n_viruses--; + p->virus = std::make_shared< Virus >(*v); + p->virus->set_date(m->today()); + p->virus->set_agent(p); - // Notice that both agent and date can be changed in this case - // as only the sequence is a shared_ptr itself. - #ifdef EPI_DEBUG - if (n_viruses >= p->viruses.size()) + // Change of state needs to be recorded and updated on the + // tools. + if (p->state_prev != p->state) { - throw std::logic_error( - "[epi-debug]::default_add_virus Index for new virus out of range." - ); + auto & db = m->get_db(); + db.update_state(p->state_prev, p->state); + + for (size_t i = 0u; i < p->n_tools; ++i) + db.update_tool(p->tools[i]->get_id(), p->state_prev, p->state); } - #endif - p->viruses[n_viruses]->set_agent(p, n_viruses); - p->viruses[n_viruses]->set_date(m->today()); + // Lastly, we increase the daily count of the virus #ifdef EPI_DEBUG m->get_db().today_virus.at(v->get_id()).at(p->state)++; #else @@ -64,9 +53,6 @@ inline void default_add_tool(Action & a, Model * m) Agent * p = a.agent; ToolPtr t = a.tool; - - CHECK_COALESCE_(a.new_state, t->state_init, p->get_state()) - CHECK_COALESCE_(a.queue, t->queue_init, Queue::NoOne) // Update tool accounting p->n_tools++; @@ -82,47 +68,64 @@ inline void default_add_tool(Action & a, Model * m) p->tools[n_tools]->set_date(m->today()); p->tools[n_tools]->set_agent(p, n_tools); + // Change of state needs to be recorded and updated on the + // tools. + if (p->state_prev != p->state) + { + auto & db = m->get_db(); + db.update_state(p->state_prev, p->state); + + if (p->virus) + db.update_virus(p->virus->get_id(), p->state_prev, p->state); + } + m->get_db().today_tool[t->get_id()][p->state]++; + } template inline void default_rm_virus(Action & a, Model * model) { - Agent * p = a.agent; - VirusPtr & v = a.agent->viruses[a.virus->pos_in_agent]; - - CHECK_COALESCE_(a.new_state, v->state_post, p->get_state()) - CHECK_COALESCE_(a.queue, v->queue_post, -Queue::Everyone) + Agent * p = a.agent; + VirusPtr & v = a.virus; - if (--p->n_viruses > 0) - { - // The new virus will change positions - p->viruses[p->n_viruses]->pos_in_agent = v->pos_in_agent; - std::swap( - p->viruses[p->n_viruses], // Moving to the end - p->viruses[v->pos_in_agent] // Moving to the beginning - ); - } - // Calling the virus action over the removed virus v->post_recovery(model); + p->virus = nullptr; + + // Change of state needs to be recorded and updated on the + // tools. + if (p->state_prev != p->state) + { + auto & db = model->get_db(); + db.update_state(p->state_prev, p->state); + + for (size_t i = 0u; i < p->n_tools; ++i) + db.update_tool(p->tools[i]->get_id(), p->state_prev, p->state); + } + + // The counters of the virus only needs to decrease + #ifdef EPI_DEBUG + model->get_db().today_virus.at(v->get_id()).at(p->state_prev)--; + #else + model->get_db().today_virus[v->get_id()][p->state_prev]--; + #endif + + return; } template -inline void default_rm_tool(Action & a, Model * /*m*/) +inline void default_rm_tool(Action & a, Model * m) { Agent * p = a.agent; ToolPtr & t = a.agent->tools[a.tool->pos_in_agent]; - CHECK_COALESCE_(a.new_state, t->state_post, p->get_state()) - CHECK_COALESCE_(a.queue, t->queue_post, Queue::NoOne) - if (--p->n_tools > 0) { p->tools[p->n_tools]->pos_in_agent = t->pos_in_agent; @@ -132,10 +135,49 @@ inline void default_rm_tool(Action & a, Model * /*m*/) ); } + // Change of state needs to be recorded and updated on the + // tools. + if (p->state_prev != p->state) + { + auto & db = m->get_db(); + db.update_state(p->state_prev, p->state); + + if (p->virus) + db.update_virus(p->virus->get_id(), p->state_prev, p->state); + } + + // Lastly, we increase the daily count of the tool + #ifdef EPI_DEBUG + m->get_db().today_tool.at(t->get_id()).at(p->state_prev)--; + #else + m->get_db().today_tool[t->get_id()][p->state_prev]--; + #endif + return; } +template +inline void default_change_state(Action & a, Model * m) +{ + + Agent * p = a.agent; + + if (p->state_prev != p->state) + { + auto & db = m->get_db(); + db.update_state(p->state_prev, p->state); + + if (p->virus) + db.update_virus(p->virus->get_id(), p->state_prev, p->state); + + for (size_t i = 0u; i < p->n_tools; ++i) + db.update_tool(p->tools[i]->get_id(), p->state_prev, p->state); + + } + +} + template inline void default_add_entity(Action & a, Model *) { @@ -143,9 +185,6 @@ inline void default_add_entity(Action & a, Model *) Agent * p = a.agent; Entity * e = a.entity; - CHECK_COALESCE_(a.new_state, e->state_post, p->get_state()) - CHECK_COALESCE_(a.queue, e->queue_post, Queue::NoOne) - // Checking the agent and the entity are not linked if ((p->get_n_entities() > 0) && (e->size() > 0)) { @@ -208,9 +247,6 @@ inline void default_rm_entity(Action & a, Model * m) size_t idx_agent_in_entity = a.idx_agent; size_t idx_entity_in_agent = a.idx_object; - CHECK_COALESCE_(a.new_state, e->state_post, p->get_state()) - CHECK_COALESCE_(a.queue, e->queue_post, Queue::NoOne) - if (--p->n_entities > 0) { diff --git a/include/epiworld/agent-bones.hpp b/include/epiworld/agent-bones.hpp index 422f071fb..06bc77ce7 100644 --- a/include/epiworld/agent-bones.hpp +++ b/include/epiworld/agent-bones.hpp @@ -52,6 +52,9 @@ inline void default_rm_tool(Action & a, Model * m); template inline void default_rm_entity(Action & a, Model * m); +template +inline void default_change_state(Action & a, Model * m); + /** @@ -63,8 +66,6 @@ template class Agent { friend class Model; friend class Virus; - friend class Viruses; - friend class Viruses_const; friend class Tool; friend class Tools; friend class Tools_const; @@ -77,6 +78,7 @@ class Agent { friend void default_rm_virus(Action & a, Model * m); friend void default_rm_tool(Action & a, Model * m); friend void default_rm_entity(Action & a, Model * m); + friend void default_change_state(Action & a, Model * m); private: Model * model; @@ -95,22 +97,11 @@ class Agent { int state_last_changed = -1; ///< Last time the agent was updated. int id = -1; - std::vector< VirusPtr > viruses; - epiworld_fast_uint n_viruses = 0u; + VirusPtr virus = nullptr; std::vector< ToolPtr > tools; epiworld_fast_uint n_tools = 0u; - ActionFun add_virus_ = default_add_virus; - ActionFun add_tool_ = default_add_tool; - ActionFun add_entity_ = default_add_entity; - - ActionFun rm_virus_ = default_rm_virus; - ActionFun rm_tool_ = default_rm_tool; - ActionFun rm_entity_ = default_rm_entity; - - epiworld_fast_uint action_counter = 0u; - std::vector< Agent * > sampled_agents; size_t sampled_agents_n = 0u; std::vector< size_t > sampled_agents_left; @@ -149,14 +140,14 @@ class Agent { epiworld_fast_int queue = -99 ); - void add_virus( + void set_virus( VirusPtr virus, Model * model, epiworld_fast_int state_new = -99, epiworld_fast_int queue = -99 ); - void add_virus( + void set_virus( Virus virus, Model * model, epiworld_fast_int state_new = -99, @@ -185,14 +176,6 @@ class Agent { ); void rm_virus( - epiworld_fast_uint virus_idx, - Model * model, - epiworld_fast_int state_new = -99, - epiworld_fast_int queue = -99 - ); - - void rm_virus( - VirusPtr & virus, Model * model, epiworld_fast_int state_new = -99, epiworld_fast_int queue = -99 @@ -213,14 +196,6 @@ class Agent { ); void rm_agent_by_virus( - epiworld_fast_uint virus_idx, - Model * model, - epiworld_fast_int state_new = -99, - epiworld_fast_int queue = -99 - ); ///< Agent removed by virus - - void rm_agent_by_virus( - VirusPtr & virus, Model * model, epiworld_fast_int state_new = -99, epiworld_fast_int queue = -99 @@ -242,10 +217,7 @@ class Agent { int get_id() const; ///< Id of the individual - VirusPtr & get_virus(int i); - Viruses get_viruses(); - const Viruses_const get_viruses() const; - size_t get_n_viruses() const noexcept; + VirusPtr & get_virus(); ToolPtr & get_tool(int i); Tools get_tools(); diff --git a/include/epiworld/agent-meat-state.hpp b/include/epiworld/agent-meat-state.hpp index e45c56465..91133f4ad 100644 --- a/include/epiworld/agent-meat-state.hpp +++ b/include/epiworld/agent-meat-state.hpp @@ -33,7 +33,7 @@ inline void default_update_susceptible( if (virus == nullptr) return; - p->add_virus(*virus, m); + p->set_virus(*virus, m); return; @@ -42,59 +42,37 @@ inline void default_update_susceptible( template inline void default_update_exposed(Agent * p, Model * m) { - if (p->get_n_viruses() == 0u) + if (p->get_virus() == nullptr) throw std::logic_error( std::string("Using the -default_update_exposed- on agents WITHOUT viruses makes no sense! ") + std::string("Agent id ") + std::to_string(p->get_id()) + std::string(" has no virus registered.") ); - // Odd: Die, Even: Recover - epiworld_fast_uint n_events = 0u; - for (const auto & v : p->get_viruses()) - { - - // Die - m->array_double_tmp[n_events++] = - v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); - - // Recover - m->array_double_tmp[n_events++] = - 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); - - } + // Die + auto & virus = p->get_virus(); + m->array_double_tmp[0u] = + virus->get_prob_death(m) * (1.0 - p->get_death_reduction(virus, m)); - #ifdef EPI_DEBUG - if (n_events == 0u) - { - printf_epiworld( - "[epi-debug] agent %i has 0 possible events!!\n", - static_cast(p->get_id()) - ); - throw std::logic_error("Zero events in exposed."); - } - #else - if (n_events == 0u) - return; - #endif + // Recover + m->array_double_tmp[1u] = + 1.0 - (1.0 - virus->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(virus, m)); // Running the roulette - int which = roulette(n_events, m); + int which = roulette(2u, m); if (which < 0) return; // Which roulette happen? - if ((which % 2) == 0) // If odd + if (which == 0u) // If odd { - size_t which_v = std::ceil(which / 2); - p->rm_agent_by_virus(which_v, m); + p->rm_agent_by_virus(m); } else { - size_t which_v = std::floor(which / 2); - p->rm_virus(which_v, m); + p->rm_virus(m); } diff --git a/include/epiworld/agent-meat-virus-sampling.hpp b/include/epiworld/agent-meat-virus-sampling.hpp index 820962064..d72899377 100644 --- a/include/epiworld/agent-meat-virus-sampling.hpp +++ b/include/epiworld/agent-meat-virus-sampling.hpp @@ -34,37 +34,31 @@ inline std::function*,Model*)> make_update_susceptible( [](Agent * p, Model * m) -> void { - if (p->get_n_viruses() > 0u) + if (p->get_virus() != nullptr) throw std::logic_error( std::string("Using the -default_update_susceptible- on agents WITH viruses makes no sense! ") + std::string("Agent id ") + std::to_string(p->get_id()) + - std::string(" has ") + std::to_string(p->get_n_viruses()) + - std::string(" viruses.") + std::string(" has a virus.") ); // This computes the prob of getting any neighbor variant size_t nviruses_tmp = 0u; for (auto & neighbor: p->get_neighbors()) { - - for (const VirusPtr & v : neighbor->get_viruses()) - { - - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) - ; - m->array_virus_tmp[nviruses_tmp++] = &(*v); + auto & v = neighbor->get_virus(); + if (v == nullptr) + continue; + + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); - } } // No virus to compute @@ -77,7 +71,7 @@ inline std::function*,Model*)> make_update_susceptible( if (which < 0) return; - p->add_virus(*m->array_virus_tmp[which], m); + p->set_virus(*m->array_virus_tmp[which], m); return; }; @@ -118,12 +112,11 @@ inline std::function*,Model*)> make_update_susceptible( } - if (p->get_n_viruses() > 0u) + if (p->get_virus() != nullptr) throw std::logic_error( std::string("Using the -default_update_susceptible- on agents WITH viruses makes no sense! ") + std::string("Agent id ") + std::to_string(p->get_id()) + - std::string(" has ") + std::to_string(p->get_n_viruses()) + - std::string(" viruses.") + std::string(" has a virus.") ); // This computes the prob of getting any neighbor variant @@ -134,26 +127,21 @@ inline std::function*,Model*)> make_update_susceptible( // If the state is in the list, exclude it if (exclude_agent_bool->operator[](neighbor->get_state())) continue; - - for (const VirusPtr & v : neighbor->get_viruses()) - { - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - - #endif + auto & v = neighbor->get_virus(); + if (v == nullptr) + continue; - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) - ; + + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); - m->array_virus_tmp[nviruses_tmp++] = &(*v); - - } } // No virus to compute @@ -166,7 +154,7 @@ inline std::function*,Model*)> make_update_susceptible( if (which < 0) return; - p->add_virus(*m->array_virus_tmp[which], m); + p->set_virus(*m->array_virus_tmp[which], m); return; @@ -202,37 +190,37 @@ inline std::function*(Agent*,Model*)> make_sample_virus_ std::function*(Agent*,Model*)> res = [](Agent * p, Model * m) -> Virus* { - if (p->get_n_viruses() > 0u) + if (p->get_virus() != nullptr) throw std::logic_error( std::string("Using the -default_update_susceptible- on agents WITH viruses makes no sense! ") + std::string("Agent id ") + std::to_string(p->get_id()) + - std::string(" has ") + std::to_string(p->get_n_viruses()) + - std::string(" viruses.") + std::string(" has a virus.") ); // This computes the prob of getting any neighbor variant size_t nviruses_tmp = 0u; for (auto & neighbor: p->get_neighbors()) { - - for (const VirusPtr & v : neighbor->get_viruses()) - { - - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) - ; - m->array_virus_tmp[nviruses_tmp++] = &(*v); + if (neighbor->get_virus() == nullptr) + continue; + + auto & v = neighbor->get_virus(); + + #ifdef EPI_DEBUG + if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif - } + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); + } // No virus to compute @@ -286,12 +274,11 @@ inline std::function*(Agent*,Model*)> make_sample_virus_ } - if (p->get_n_viruses() > 0u) + if (p->get_virus() != nullptr) throw std::logic_error( std::string("Using the -default_update_susceptible- on agents WITH viruses makes no sense! ") + std::string("Agent id ") + std::to_string(p->get_id()) + - std::string(" has ") + std::to_string(p->get_n_viruses()) + - std::string(" viruses.") + std::string(" has a virus.") ); // This computes the prob of getting any neighbor variant @@ -302,25 +289,26 @@ inline std::function*(Agent*,Model*)> make_sample_virus_ // If the state is in the list, exclude it if (exclude_agent_bool->operator[](neighbor->get_state())) continue; - - for (const VirusPtr & v : neighbor->get_viruses()) - { - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif + if (neighbor->get_virus() == nullptr) + continue; + + auto & v = neighbor->get_virus(); - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) - ; - - m->array_virus_tmp[nviruses_tmp++] = &(*v); + #ifdef EPI_DEBUG + if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif - } + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); + } // No virus to compute @@ -363,12 +351,11 @@ template inline Virus * sample_virus_single(Agent * p, Model * m) { - if (p->get_n_viruses() > 0u) + if (p->get_virus() != nullptr) throw std::logic_error( - std::string("Using the -default_update_susceptible- on agents WITH viruses makes no sense! ") + + std::string("Using the -default_update_susceptible- on agents WITH viruses makes no sense!") + std::string("Agent id ") + std::to_string(p->get_id()) + - std::string(" has ") + std::to_string(p->get_n_viruses()) + - std::string(" viruses.") + std::string(" has a virus.") ); // This computes the prob of getting any neighbor variant @@ -377,40 +364,42 @@ inline Virus * sample_virus_single(Agent * p, Model * m) { #ifdef EPI_DEBUG int _vcount_neigh = 0; - #endif - for (const VirusPtr & v : neighbor->get_viruses()) - { - - #ifdef EPI_DEBUG - if (nviruses_tmp >= m->array_virus_tmp.size()) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) - ; - - m->array_virus_tmp[nviruses_tmp++] = &(*v); - - #ifdef EPI_DEBUG - if ( - (m->array_double_tmp[nviruses_tmp - 1] < 0.0) | - (m->array_double_tmp[nviruses_tmp - 1] > 1.0) - ) - { - printf_epiworld( - "[epi-debug] Agent %i's virus %i has transmission prob outside of [0, 1]: %.4f!\n", - static_cast(neighbor->get_id()), - static_cast(_vcount_neigh++), - m->array_double_tmp[nviruses_tmp - 1] - ); - } - #endif + #endif + + if (neighbor->get_virus() == nullptr) + continue; + + auto & v = neighbor->get_virus(); + + #ifdef EPI_DEBUG + if (nviruses_tmp >= m->array_virus_tmp.size()) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif + + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); + + #ifdef EPI_DEBUG + if ( + (m->array_double_tmp[nviruses_tmp - 1] < 0.0) | + (m->array_double_tmp[nviruses_tmp - 1] > 1.0) + ) + { + printf_epiworld( + "[epi-debug] Agent %i's virus %i has transmission prob outside of [0, 1]: %.4f!\n", + static_cast(neighbor->get_id()), + static_cast(_vcount_neigh++), + m->array_double_tmp[nviruses_tmp - 1] + ); + } + #endif - } } diff --git a/include/epiworld/agent-meat.hpp b/include/epiworld/agent-meat.hpp index 47f696264..6bed922b8 100644 --- a/include/epiworld/agent-meat.hpp +++ b/include/epiworld/agent-meat.hpp @@ -1,9 +1,6 @@ #ifndef EPIWORLD_PERSON_MEAT_HPP #define EPIWORLD_PERSON_MEAT_HPP -// template -// inline bool IN(Ta & a, std::vector< Ta > & b); - #define CHECK_COALESCE_(proposed_, virus_tool_, alt_) \ if (static_cast(proposed_) == -99) {\ if (static_cast(virus_tool_) == -99) \ @@ -29,36 +26,21 @@ inline Agent::Agent(Agent && p) : state_prev(p.state_prev), state_last_changed(p.state_last_changed), id(p.id), - viruses(std::move(p.viruses)), /// Needs to be adjusted - n_viruses(p.n_viruses), tools(std::move(p.tools)), /// Needs to be adjusted - n_tools(p.n_tools), - add_virus_(std::move(p.add_virus_)), - add_tool_(std::move(p.add_tool_)), - add_entity_(std::move(p.add_entity_)), - rm_virus_(std::move(p.rm_virus_)), - rm_tool_(std::move(p.rm_tool_)), - rm_entity_(std::move(p.rm_entity_)), - action_counter(p.action_counter) + n_tools(p.n_tools) { state = p.state; id = p.id; // Dealing with the virus - - int loc = 0; - for (auto & v : viruses) + if (p.virus != nullptr) { - - // Will create a copy of the virus, with the exeption of - // the virus code - v->agent = this; - v->pos_in_agent = loc++; - + virus = std::move(p.virus); + virus->set_agent(this); } - loc = 0; + int loc = 0; for (auto & t : tools) { @@ -90,17 +72,12 @@ inline Agent::Agent(const Agent & p) : id = p.id; // Dealing with the virus - viruses.reserve(p.get_n_viruses()); - n_viruses = viruses.size(); - for (size_t i = 0u; i < n_viruses; ++i) + if (p.virus != nullptr) { - - // Will create a copy of the virus, with the exeption of - // the virus code - viruses.emplace_back(std::make_shared>(*p.viruses[i])); - viruses.back()->set_agent(this, i); - + virus = std::make_shared>(*p.virus); + virus->set_agent(this); } + tools.reserve(p.get_n_tools()); n_tools = tools.size(); @@ -113,11 +90,6 @@ inline Agent::Agent(const Agent & p) : tools.back()->set_agent(this, i); } - - add_virus_ = p.add_virus_; - add_tool_ = p.add_tool_; - rm_virus_ = p.rm_virus_; - rm_tool_ = p.rm_tool_; } @@ -151,14 +123,12 @@ inline Agent & Agent::operator=( state_last_changed = other_agent.state_last_changed; id = other_agent.id; - // viruses = other_agent.viruses; - n_viruses = other_agent.n_viruses; - viruses.resize(n_viruses, nullptr); - for (size_t i = 0u; i < n_viruses; ++i) + if (other_agent.virus != nullptr) { - viruses[i] = std::make_shared>(*other_agent.viruses[i]); - viruses[i]->set_agent(this, i); - } + virus = std::make_shared>(*other_agent.virus); + virus->set_agent(this); + } else + virus = nullptr; // tools = other_agent.tools; n_tools = other_agent.n_tools; @@ -167,14 +137,6 @@ inline Agent & Agent::operator=( tools[i] = std::make_shared>(*other_agent.tools[i]); tools[i]->set_agent(this, i); } - - add_virus_ = other_agent.add_virus_; - add_tool_ = other_agent.add_tool_; - add_entity_ = other_agent.add_entity_; - rm_virus_ = other_agent.rm_virus_; - rm_tool_ = other_agent.rm_tool_; - rm_entity_ = other_agent.rm_entity_; - action_counter = other_agent.action_counter; return *this; @@ -193,10 +155,12 @@ inline void Agent::add_tool( throw std::range_error("The tool with id: " + std::to_string(tool->get_id()) + " has not been registered. There are only " + std::to_string(model->get_n_tools()) + " included in the model."); - + + CHECK_COALESCE_(state_new, tool->state_init, state); + CHECK_COALESCE_(queue, tool->queue_init, Queue::NoOne); model->actions_add( - this, nullptr, tool, nullptr, state_new, queue, add_tool_, -1, -1 + this, nullptr, tool, nullptr, state_new, queue, default_add_tool, -1, -1 ); } @@ -214,7 +178,7 @@ inline void Agent::add_tool( } template -inline void Agent::add_virus( +inline void Agent::set_virus( VirusPtr virus, Model * model, epiworld_fast_int state_new, @@ -228,14 +192,17 @@ inline void Agent::add_virus( " has not been registered. There are only " + std::to_string(model->get_n_viruses()) + " included in the model."); + CHECK_COALESCE_(state_new, virus->state_init, state); + CHECK_COALESCE_(queue, virus->queue_init, Queue::NoOne); + model->actions_add( - this, virus, nullptr, nullptr, state_new, queue, add_virus_, -1, -1 + this, virus, nullptr, nullptr, state_new, queue, default_add_virus, -1, -1 ); } template -inline void Agent::add_virus( +inline void Agent::set_virus( Virus virus, Model * model, epiworld_fast_int state_new, @@ -243,7 +210,7 @@ inline void Agent::add_virus( ) { VirusPtr virus_ptr = std::make_shared< Virus >(virus); - add_virus(virus_ptr, model, state_new, queue); + set_virus(virus_ptr, model, state_new, queue); } template @@ -255,11 +222,14 @@ inline void Agent::add_entity( ) { + CHECK_COALESCE_(state_new, entity.state_init, state); + CHECK_COALESCE_(queue, entity.queue_init, Queue::NoOne); + if (model != nullptr) { model->actions_add( - this, nullptr, nullptr, &entity, state_new, queue, add_entity_, -1, -1 + this, nullptr, nullptr, &entity, state_new, queue, default_add_entity, -1, -1 ); } @@ -268,11 +238,11 @@ inline void Agent::add_entity( { Action a( - this, nullptr, nullptr, &entity, state_new, queue, add_entity_, + this, nullptr, nullptr, &entity, state_new, queue, default_add_entity, -1, -1 ); - default_add_entity(a, model); /* passing model makes nothing */ + // default_add_entity(a, model); /* passing model makes nothing */ } @@ -287,6 +257,9 @@ inline void Agent::rm_tool( ) { + CHECK_COALESCE_(state_new, tools[tool_idx]->state_post, state); + CHECK_COALESCE_(queue, tools[tool_idx]->queue_post, Queue::NoOne); + if (tool_idx >= n_tools) throw std::range_error( "The Tool you want to remove is out of range. This Agent only has " + @@ -294,7 +267,7 @@ inline void Agent::rm_tool( ); model->actions_add( - this, nullptr, tools[tool_idx], nullptr, state_new, queue, rm_tool_, -1, -1 + this, nullptr, tools[tool_idx], nullptr, state_new, queue, default_rm_tool, -1, -1 ); } @@ -312,63 +285,32 @@ inline void Agent::rm_tool( throw std::logic_error("Cannot remove a virus from another agent!"); model->actions_add( - this, nullptr, tool, nullptr, state_new, queue, rm_tool_, -1, -1 + this, nullptr, tool, nullptr, state_new, queue, default_rm_tool, -1, -1 ); } template inline void Agent::rm_virus( - epiworld_fast_uint virus_idx, Model * model, epiworld_fast_int state_new, epiworld_fast_int queue ) { - if (virus_idx >= n_viruses) - throw std::range_error( - "The Virus you want to remove is out of range. This Agent only has " + - std::to_string(n_viruses) + " viruses." - ); - else if (n_viruses == 0u) - throw std::logic_error( - "There is no virus to remove here!" - ); - #ifdef EPI_DEBUG - if (viruses[virus_idx]->pos_in_agent >= static_cast(n_viruses)) - { + if (virus == nullptr) throw std::logic_error( - "[epi-debug]::rm_virus the position in the agent is wrong." - ); - } - #endif - - model->actions_add( - this, viruses[virus_idx], nullptr, nullptr, state_new, queue, - default_rm_virus, -1, -1 + "There is no virus to remove here!" ); - -} -template -inline void Agent::rm_virus( - VirusPtr & virus, - Model * model, - epiworld_fast_int state_new, - epiworld_fast_int queue -) -{ - - if (virus->agent != this) - throw std::logic_error("Cannot remove a virus from another agent!"); + CHECK_COALESCE_(state_new, virus->state_post, state); + CHECK_COALESCE_(queue, virus->queue_post, Queue::Everyone); model->actions_add( this, virus, nullptr, nullptr, state_new, queue, default_rm_virus, -1, -1 ); - - + } template @@ -390,9 +332,12 @@ inline void Agent::rm_entity( "There is entity to remove here!" ); + CHECK_COALESCE_(state_new, model->entities[entity_idx].state_post, state); + CHECK_COALESCE_(queue, model->entities[entity_idx].queue_post, Queue::NoOne); + model->actions_add( this, nullptr, nullptr, model->entities[entity_idx], state_new, queue, - default_rm_entity, entities_locations[entity_idx], entity_idx + default_rm_entity, entities_locations[entity_idx], entity_idx ); } @@ -419,82 +364,30 @@ inline void Agent::rm_entity( entity.get_name() + "\"." ); + CHECK_COALESCE_(state_new, entity.state_post, state); + CHECK_COALESCE_(queue, entity.queue_post, Queue::NoOne); model->actions_add( this, nullptr, nullptr, entities[entity_idx], state_new, queue, - default_rm_entity, entities_locations[entity_idx], entity_idx + default_rm_entity, entities_locations[entity_idx], entity_idx ); } template inline void Agent::rm_agent_by_virus( - epiworld_fast_uint virus_idx, Model * model, epiworld_fast_int state_new, epiworld_fast_int queue ) { - if (state_new == -99) - state_new = state; - - if (virus_idx >= n_viruses) - throw std::range_error( - std::string("The virus trying to remove the agent is out of range. ") + - std::string("This agent has only ") + std::to_string(n_viruses) + - std::string(" and you are trying to remove virus # ") + - std::to_string(virus_idx) + std::string(".") - ); + CHECK_COALESCE_(state_new, virus->state_removed, state); + CHECK_COALESCE_(queue, virus->queue_removed, Queue::Everyone); - // Removing viruses - for (size_t i = 0u; i < n_viruses; ++i) - { - if (i != virus_idx) - rm_virus(i, model); - } - - // Changing state to new_state - VirusPtr & v = viruses[virus_idx]; - epiworld_fast_int dead_state, dead_queue; - v->get_state(nullptr, nullptr, &dead_state); - v->get_queue(nullptr, nullptr, &dead_queue); - - if (queue != -99) - dead_queue = queue; - - change_state( - model, - // Either preserve the current state or apply a new one - (dead_state < 0) ? state : static_cast(dead_state), - - // By default, it will be removed from the queue... unless the user - // says the contrary! - (dead_queue == -99) ? Queue::NoOne : dead_queue - ); - -} - -template -inline void Agent::rm_agent_by_virus( - VirusPtr & virus, - Model * model, - epiworld_fast_int state_new, - epiworld_fast_int queue -) -{ - - if (virus->get_agent() == nullptr) - throw std::logic_error("The virus trying to remove the agent has no host."); - - if (virus->get_agent()->id != id) - throw std::logic_error("Viruses can only remove their hosts'."); - - rm_agent_by_virus( - virus->pos_in_agent, - model, - state_new, - queue - ); + model->actions_add( + this, virus, nullptr, nullptr, state_new, queue, + default_rm_virus, -1, -1 + ); } @@ -538,29 +431,10 @@ inline int Agent::get_id() const } template -inline Viruses Agent::get_viruses() { - - return Viruses(*this); - -} - -template -inline const Viruses_const Agent::get_viruses() const { - - return Viruses_const(*this); - +inline VirusPtr & Agent::get_virus() { + return virus; } -template -inline VirusPtr & Agent::get_virus(int i) { - return viruses.at(i); -} - -template -inline size_t Agent::get_n_viruses() const noexcept -{ - return n_viruses; -} template inline Tools Agent::get_tools() { @@ -588,8 +462,7 @@ template inline void Agent::mutate_virus() { - for (auto & v : viruses) - v->mutate(); + virus->mutate(); } @@ -715,7 +588,8 @@ inline void Agent::change_state( { model->actions_add( - this, nullptr, nullptr, nullptr, new_state, queue, nullptr, -1, -1 + this, nullptr, nullptr, nullptr, new_state, queue, + default_change_state, -1, -1 ); return; @@ -731,8 +605,7 @@ template inline void Agent::reset() { - this->viruses.clear(); - n_viruses = 0u; + this->virus = nullptr; this->tools.clear(); n_tools = 0u; @@ -779,9 +652,8 @@ inline bool Agent::has_tool(const Tool & tool) const template inline bool Agent::has_virus(epiworld_fast_uint t) const { - for (auto & v : viruses) - if (v->get_id() == static_cast(t)) - return true; + if (virus->get_id() == static_cast(t)) + return true; return false; } @@ -790,9 +662,8 @@ template inline bool Agent::has_virus(std::string name) const { - for (auto & v : viruses) - if (v->get_name() == name) - return true; + if (virus->get_name() == name) + return true; return false; @@ -816,15 +687,18 @@ inline void Agent::print( if (compressed) { printf_epiworld( - "Agent: %i, state: %s (%lu), Nvirus: %lu, NTools: %lu, NNeigh: %lu\n", - id, model->states_labels[state].c_str(), state, n_viruses, n_tools, neighbors.size() + "Agent: %i, state: %s (%lu), Has virus: %s, NTools: %lu, NNeigh: %lu\n", + id, model->states_labels[state].c_str(), state, + virus == nullptr ? std::string("no").c_str() : std::string("yes").c_str(), + n_tools, neighbors.size() ); } else { printf_epiworld("Information about agent id %i\n", this->id); printf_epiworld(" State : %s (%lu)\n", model->states_labels[state].c_str(), state); - printf_epiworld(" Virus count : %lu\n", n_viruses); + printf_epiworld(" Has virus : %s\n", virus == nullptr ? + std::string("no").c_str() : std::string("yes").c_str()); printf_epiworld(" Tool count : %lu\n", n_tools); printf_epiworld(" Neigh. count : %lu\n", neighbors.size()); @@ -975,24 +849,21 @@ inline bool Agent::operator==(const Agent & other) const // state_last_changed != other.state_last_changed, // "Agent:: state_last_changed don't match" // ) ///< Last time the agent was updated. - - + EPI_DEBUG_FAIL_AT_TRUE( - n_viruses != other.n_viruses, - "Agent:: n_viruses don't match" - ) - + ((virus == nullptr) && (other.virus != nullptr)) || + ((virus != nullptr) && (other.virus == nullptr)), + "Agent:: virus don't match" + ) - for (size_t i = 0u; i < n_viruses; ++i) + if ((virus != nullptr) && (other.virus != nullptr)) { - EPI_DEBUG_FAIL_AT_TRUE( - *viruses[i] != *other.viruses[i], - "Agent:: viruses[i] don't match" + *virus != *other.virus, + "Agent:: virus doesn't match" ) - } - + EPI_DEBUG_FAIL_AT_TRUE(n_tools != other.n_tools, "Agent:: n_tools don't match") for (size_t i = 0u; i < n_tools; ++i) diff --git a/include/epiworld/agentssample-bones.hpp b/include/epiworld/agentssample-bones.hpp index 165c4985a..c8462c585 100644 --- a/include/epiworld/agentssample-bones.hpp +++ b/include/epiworld/agentssample-bones.hpp @@ -41,6 +41,7 @@ class AgentsSample { Agent * agent = nullptr; int sample_type = SAMPLETYPE::AGENT; + std::vector< size_t > states = {}; void sample_n(size_t n); ///< Backbone function for sampling @@ -52,9 +53,23 @@ class AgentsSample { AgentsSample(const AgentsSample & a) = delete; ///< Copy constructor AgentsSample(AgentsSample && a) = delete; ///< Move constructor - AgentsSample(Model & model_, size_t n, bool truncate = false); - AgentsSample(Model * model, Entity & entity_, size_t n, bool truncate = false); - AgentsSample(Model * model, Agent & agent_, size_t n, bool truncate = false); + AgentsSample( + Model & model_, size_t n, + std::vector< size_t > states_ = {}, + bool truncate = false + ); + + AgentsSample( + Model * model, Entity & entity_, size_t n, + std::vector< size_t > states_ = {}, + bool truncate = false + ); + + AgentsSample( + Model * model, Agent & agent_, size_t n, + std::vector< size_t > states_ = {}, + bool truncate = false + ); ~AgentsSample(); @@ -71,9 +86,12 @@ template inline AgentsSample::AgentsSample( Model & model_, size_t n, + std::vector< size_t > states_, bool truncate ) { + states = states_; + if (truncate) { @@ -85,12 +103,12 @@ inline AgentsSample::AgentsSample( "There are only " + std::to_string(model_.size()) + " agents. You cannot " + "sample " + std::to_string(n)); - sample_size = n; - model = &model_; - sample_type = SAMPLETYPE::MODEL; + sample_size = n; + model = &model_; + sample_type = SAMPLETYPE::MODEL; - agents = &model_.sampled_population; - agents_n = &model_.sampled_population_n; + agents = &model_.sampled_population; + agents_n = &model_.sampled_population_n; agents_left = &model_.population_left; agents_left_n = &model_.population_left_n; @@ -105,9 +123,13 @@ template inline AgentsSample::AgentsSample( Model * model, Entity & entity_, - size_t n, bool truncate + size_t n, + std::vector< size_t > states_, + bool truncate ) { + states = states_; + if (truncate) { @@ -119,12 +141,12 @@ inline AgentsSample::AgentsSample( "There are only " + std::to_string(entity_.size()) + " agents. You cannot " + "sample " + std::to_string(n)); - sample_size = n; - model = &entity_.model; - sample_type = SAMPLETYPE::ENTITY; + sample_size = n; + model = &entity_.model; + sample_type = SAMPLETYPE::ENTITY; - agents = &entity_.sampled_agents; - agents_n = &entity_.sampled_agents_n; + agents = &entity_.sampled_agents; + agents_n = &entity_.sampled_agents_n; agents_left = &entity_.sampled_agents_left; agents_left_n = &entity_.sampled_agents_left_n; @@ -152,16 +174,18 @@ inline AgentsSample::AgentsSample( Model * model, Agent & agent_, size_t n, + std::vector< size_t > states_, bool truncate ) { + states = states_; sample_type = SAMPLETYPE::AGENT; - agent = &agent_; + agent = &agent_; - agents = &agent_.sampled_agents; - agents_n = &agent_.sampled_agents_n; + agents = &agent_.sampled_agents; + agents_n = &agent_.sampled_agents_n; agents_left = &agent_.sampled_agents_left; agents_left_n = &agent_.sampled_agents_left_n; @@ -170,7 +194,7 @@ inline AgentsSample::AgentsSample( size_t agents_in_entities = 0; Entities entities_a = agent->get_entities(); - std::vector< int > cum_agents_count(entities_a.size(), 0); + std::vector< size_t > cum_agents_count(entities_a.size(), 0); int idx = -1; for (auto & e : entities_a) { @@ -193,15 +217,18 @@ inline AgentsSample::AgentsSample( } else if (n > agents_in_entities) throw std::logic_error( - "There are only " + std::to_string(agents_in_entities) + " agents. You cannot " + - "sample " + std::to_string(n)); + "There are only " + std::to_string(agents_in_entities) + + " agents. You cannot " + + "sample " + std::to_string(n) + ); sample_size = n; if (agents->size() < n) agents->resize(n); - for (size_t i = 0u; i < n; ++i) + size_t i_obs = 0u; + for (size_t i = 0u; i < agents_in_entities; ++i) { int jth = std::floor(model->runif() * agents_in_entities); for (size_t e = 0u; e < cum_agents_count.size(); ++e) @@ -209,11 +236,24 @@ inline AgentsSample::AgentsSample( // Are we in the limit? if (jth <= cum_agents_count[e]) { + size_t agent_idx = 0u; if (e == 0) // From the first group - agents->operator[](i) = entities_a[e][jth]; + agent_idx = entities_a[e][jth]; else - agents->operator[](i) = entities_a[e][jth - cum_agents_count[e - 1]]; + agent_idx = entities_a[e][jth - cum_agents_count[e - 1]]; + + // Getting the state + size_t state = model->population[agent_idx].get_state(); + + // Checking if states was specified + if (states.size()) + { + if (std::find(states.begin(), states.end(), state) != states.end()) + continue; + } + agents->operator[](i_obs++) = agent_idx; + break; } @@ -275,59 +315,143 @@ template inline void AgentsSample::sample_n(size_t n) { - // Checking if the size of the entity has changed (or hasn't been initialized) - if (sample_type == SAMPLETYPE::MODEL) + // Reducing size + if (states.size()) { + + // Getting the number of agents left + agents_left->clear(); - if (model->size() != agents_left->size()) + if (sample_type == SAMPLETYPE::MODEL) { - agents_left->resize(model->size(), 0u); - std::iota(agents_left->begin(), agents_left->end(), 0u); - } + // Making some room + agents_left->reserve(model->size()); + + // Iterating through the agents in the population + for (size_t a_i = 0u; a_i < model->population.size(); ++a_i) + { - } else if (sample_type == SAMPLETYPE::ENTITY) { + // If the agent is within the selected set of states, + // then we add it to the list of agents left + size_t s = model->population[a_i].get_state(); + if (std::find(states.begin(), states.end(), s) != states.end()) + agents_left->push_back(a_i); - if (entity->size() != agents_left->size()) + } + + } + else if (sample_type == SAMPLETYPE::ENTITY) { - agents_left->resize(entity->size(), 0u); - std::iota(agents_left->begin(), agents_left->end(), 0u); + // Making room + agents_left->reserve(entity->size()); + + // Iterating through the agents in the entity + for (size_t a_i = 0u; a_i < entity->size(); ++a_i) + { + size_t s = model->population[entity->agents[a_i]].get_state(); + if (std::find(states.begin(), states.end(), s) != states.end()) + agents_left->push_back(a_i); + + } } - } + } else { + + // Checking if the size of the entity has changed (or hasn't been initialized) + if (sample_type == SAMPLETYPE::MODEL) + { + + if (model->size() != agents_left->size()) + { + agents_left->resize(model->size(), 0u); + std::iota(agents_left->begin(), agents_left->end(), 0u); + } + + } else if (sample_type == SAMPLETYPE::ENTITY) { + + if (entity->size() != agents_left->size()) + { + + agents_left->resize(entity->size(), 0u); + std::iota(agents_left->begin(), agents_left->end(), 0u); + + } + + } + + } // Restart the counter of agents left *agents_left_n = agents_left->size(); + // Making sure we have enough room for the sample of agents if (agents->size() < sample_size) agents->resize(sample_size, nullptr); if (sample_type == SAMPLETYPE::MODEL) { + #ifdef EPI_DEBUG + std::vector< bool > __sampled(model->size(), true); + for (auto & a_i: *agents_left) + __sampled[a_i] = false; + #endif + for (size_t i = 0u; i < n; ++i) { - size_t ith = agents_left->operator[](model->runif() * ((*agents_left_n)--)); + // Sampling from 0 to (agents_left_n - 1) + size_t ith_ = static_cast(model->runif() * ((*agents_left_n)--)); + + // Getting the id of the agent and adding it to the list of agents + size_t ith = agents_left->operator[](ith_); agents->operator[](i) = &model->population[ith]; + #ifdef EPI_DEBUG + if (__sampled[ith]) + throw std::logic_error("The same agent was sampled twice."); + else + __sampled[ith] = true; + #endif + // Updating list - std::swap(agents_left->operator[](ith), agents_left->operator[](*agents_left_n)); + std::swap( + agents_left->operator[](ith_), + agents_left->operator[](*agents_left_n) + ); } - } else if (sample_type == SAMPLETYPE::ENTITY) { + + } + else if (sample_type == SAMPLETYPE::ENTITY) + { + + #ifdef EPI_DEBUG + std::vector< bool > __sampled(entity->size(), true); + for (auto & a_i: *agents_left) + __sampled[a_i] = false; + #endif for (size_t i = 0u; i < n; ++i) { - size_t ith = agents_left->operator[](model->runif() * (--(*agents_left_n))); - agents->operator[](i) = entity->agents[ith]; + size_t ith_ = static_cast(model->runif() * ((*agents_left_n)--)); + size_t ith = agents_left->operator[](ith_); + agents->operator[](i) = &model->population[entity->agents[ith]]; + + #ifdef EPI_DEBUG + if (__sampled[ith]) + throw std::logic_error("The same agent was sampled twice."); + else + __sampled[ith] = true; + #endif // Updating list - std::swap(agents_left->operator[](ith), agents_left->operator[](*agents_left_n)); + std::swap(agents_left->operator[](ith_), agents_left->operator[](*agents_left_n)); } diff --git a/include/epiworld/config.hpp b/include/epiworld/config.hpp index 8c65e0671..633dd0fd8 100644 --- a/include/epiworld/config.hpp +++ b/include/epiworld/config.hpp @@ -9,7 +9,7 @@ #define EPIWORLD_MAXNEIGHBORS 1048576 #endif -#ifdef _OPENMP +#if defined(_OPENMP) || defined(__OPENMP) #include // #else // #define omp_get_thread_num() 0 @@ -259,7 +259,9 @@ struct Action { if (a) \ {\ throw EPI_DEBUG_ERROR(std::logic_error, b); \ - } + } + + #define epiexception(a) std::logic_error #else #define EPI_DEBUG_PRINTF(fmt, ...) #define EPI_DEBUG_ERROR(fmt, ...) @@ -271,9 +273,10 @@ struct Action { #define EPI_DEBUG_FAIL_AT_TRUE(a, b) \ if (a) \ return false; + #define epiexception(a) a #endif -#ifdef EPI_DEBUG_NO_THREAD_ID +#if defined(EPI_DEBUG_NO_THREAD_ID) || (!defined(__OPENMP) && !defined(_OPENMP)) #define EPI_GET_THREAD_ID() 0 #else #define EPI_GET_THREAD_ID() omp_get_thread_num() diff --git a/include/epiworld/database-bones.hpp b/include/epiworld/database-bones.hpp index be62c6716..a57f37a1d 100644 --- a/include/epiworld/database-bones.hpp +++ b/include/epiworld/database-bones.hpp @@ -22,6 +22,9 @@ inline void default_rm_virus(Action & a, Model * m); template inline void default_rm_tool(Action & a, Model * m); +template +inline void default_change_state(Action & a, Model * m); + /** * @brief Statistical data about the process * @@ -34,6 +37,7 @@ class DataBase { friend void default_add_tool(Action & a, Model * m); friend void default_rm_virus(Action & a, Model * m); friend void default_rm_tool(Action & a, Model * m); + friend void default_change_state(Action & a, Model * m); private: Model * model; diff --git a/include/epiworld/entity-meat.hpp b/include/epiworld/entity-meat.hpp index 3220848e8..6912ec527 100644 --- a/include/epiworld/entity-meat.hpp +++ b/include/epiworld/entity-meat.hpp @@ -1,28 +1,6 @@ #ifndef EPIWORLD_ENTITY_MEAT_HPP #define EPIWORLD_ENTITY_MEAT_HPP -// template -// inline Entity::Entity(const Entity & e) : -// model(e.model), -// id(e.id), -// agents(0u), -// agents_location(0u), -// n_agents(0), -// sampled_agents(0u), -// sampled_agents_n(0u), -// sampled_agents_left(0u), -// sampled_agents_left_n(0u), -// max_capacity(e.max_capacity), -// entity_name(e.entity_name), -// location(e.location), -// state_init(e.state_init), -// state_post(e.state_post), -// queue_init(e.queue_init), -// queue_post(e.queue_post) -// { - -// } - template inline void Entity::add_agent( Agent & p, diff --git a/include/epiworld/epiworld.hpp b/include/epiworld/epiworld.hpp index 3644e5df0..583878ffa 100644 --- a/include/epiworld/epiworld.hpp +++ b/include/epiworld/epiworld.hpp @@ -71,8 +71,6 @@ namespace epiworld { #include "models/models.hpp" - - } #endif \ No newline at end of file diff --git a/include/epiworld/model-bones.hpp b/include/epiworld/model-bones.hpp index b913a3831..01efeea3d 100644 --- a/include/epiworld/model-bones.hpp +++ b/include/epiworld/model-bones.hpp @@ -104,7 +104,6 @@ class Model { bool using_backup = true; std::vector< Agent > population_backup = {}; - /** * @name Auxiliary variables for AgentsSample iterators * @@ -170,8 +169,13 @@ class Model { epiworld_fast_uint ndays = 0; Progress pb; - std::vector< UpdateFun > state_fun = {}; - std::vector< std::string > states_labels = {}; + std::vector< UpdateFun > state_fun = {}; ///< Functions to update states + std::vector< std::string > states_labels = {}; ///< Labels of the states + + /** Function to distribute states. Goes along with the function */ + std::function*)> initial_states_fun = [](Model * /**/) + -> void {}; + epiworld_fast_uint nstates = 0u; bool verbose = true; @@ -221,20 +225,13 @@ class Model { VirusPtr virus_, ToolPtr tool_, Entity * entity_, - epiworld_fast_uint new_state_, + epiworld_fast_int new_state_, epiworld_fast_int queue_, ActionFun call_, int idx_agent_, int idx_object_ ); - /** - * @brief Executes the stored action - * - * @param model_ Model over which it will be executed. - */ - void actions_run(); - /** * @name Tool Mixers * @@ -258,12 +255,13 @@ class Model { public: + std::vector array_double_tmp; std::vector * > array_virus_tmp; Model(); Model(const Model & m); - Model(Model & m) = delete; + Model(Model & m); Model(Model && m); Model & operator=(const Model & m); @@ -401,7 +399,7 @@ class Model { std::vector< Entity > & get_entities(); - void agents_smallworld( + Model & agents_smallworld( epiworld_fast_uint n = 1000, epiworld_fast_uint k = 5, bool d = false, @@ -423,7 +421,7 @@ class Model { void update_state(); void mutate_virus(); void next(); - virtual void run( + virtual Model & run( epiworld_fast_uint ndays, int seed = -1 ); ///< Runs the simulation (after initialization) @@ -438,14 +436,14 @@ class Model { ); ///@} - size_t get_n_viruses() const; - size_t get_n_tools() const; + size_t get_n_viruses() const; ///< Number of viruses in the model + size_t get_n_tools() const; ///< Number of tools in the model epiworld_fast_uint get_ndays() const; epiworld_fast_uint get_n_replicates() const; void set_ndays(epiworld_fast_uint ndays); bool get_verbose() const; - void verbose_off(); - void verbose_on(); + Model & verbose_off(); + Model & verbose_on(); int today() const; ///< The current time of the model /** @@ -525,7 +523,7 @@ class Model { * */ virtual void reset(); - void print(bool lite = false) const; + const Model & print(bool lite = false) const; Model && clone() const; @@ -550,6 +548,19 @@ class Model { void print_state_codes() const; ///@} + /** + * @name Initial states + * + * @details These functions are called before the simulation starts. + * + * @param proportions_ Vector of proportions for each state. + * @param queue_ Vector of queue for each state. + */ + virtual Model & initial_states( + std::vector< double > /*proportions_*/, + std::vector< int > /*queue_*/ + ) {return *this;}; + /** * @name Setting and accessing parameters from the model * @@ -655,7 +666,7 @@ class Model { */ ////@{ void queuing_on(); ///< Activates the queuing system (default.) - void queuing_off(); ///< Deactivates the queuing system. + Model & queuing_off(); ///< Deactivates the queuing system. bool is_queuing_on() const; ///< Query if the queuing system is on. Queue & get_queue(); ///< Retrieve the `Queue` object. ///@} @@ -674,6 +685,8 @@ class Model { ///@} const std::vector< VirusPtr > & get_viruses() const; + const std::vector< epiworld_double > & get_prevalence_virus() const; + const std::vector< bool > & get_prevalence_virus_as_proportion() const; const std::vector< ToolPtr > & get_tools() const; Virus & get_virus(size_t id); Tool & get_tool(size_t id); @@ -704,6 +717,14 @@ class Model { bool operator==(const Model & other) const; bool operator!=(const Model & other) const {return !operator==(other);}; + /** + * @brief Executes the stored action + * + * @param model_ Model over which it will be executed. + */ + void actions_run(); + + }; #endif diff --git a/include/epiworld/model-meat-print.hpp b/include/epiworld/model-meat-print.hpp index 673f51893..a5a419aa9 100644 --- a/include/epiworld/model-meat-print.hpp +++ b/include/epiworld/model-meat-print.hpp @@ -2,7 +2,7 @@ #define EPIWORLD_MODEL_MEAT_PRINT_HPP template -inline void Model::print(bool lite) const +inline const Model & Model::print(bool lite) const { // Horizontal line @@ -58,7 +58,7 @@ inline void Model::print(bool lite) const printf_epiworld(" The model hasn't been run yet.\n"); } - return; + return *this; } printf_epiworld("%s\n%s\n\n",line.c_str(), "SIMULATION STUDY"); @@ -324,7 +324,7 @@ inline void Model::print(bool lite) const if (today() != 0) (void) db.transition_probability(true); - return; + return *this; } diff --git a/include/epiworld/model-meat.hpp b/include/epiworld/model-meat.hpp index 431a58c24..ce7e59b8b 100644 --- a/include/epiworld/model-meat.hpp +++ b/include/epiworld/model-meat.hpp @@ -1,8 +1,6 @@ #ifndef EPIWORLD_MODEL_MEAT_HPP #define EPIWORLD_MODEL_MEAT_HPP - - /** * @brief Function factory for saving model runs * @@ -150,38 +148,31 @@ inline std::function*)> make_save_run( return saver; } + template inline void Model::actions_add( Agent * agent_, VirusPtr virus_, ToolPtr tool_, Entity * entity_, - epiworld_fast_uint new_state_, + epiworld_fast_int new_state_, epiworld_fast_int queue_, ActionFun call_, int idx_agent_, int idx_object_ ) { - + ++nactions; #ifdef EPI_DEBUG if (nactions == 0) throw std::logic_error("Actions cannot be zero!!"); - - if ((virus_ != nullptr) && idx_agent_ >= 0) - { - if (idx_agent_ >= static_cast(virus_->get_agent()->get_n_viruses())) - throw std::logic_error( - "The virus to add is out of range in the host agent." - ); - } #endif if (nactions > actions.size()) { - actions.push_back( + actions.emplace_back( Action( agent_, virus_, tool_, entity_, new_state_, queue_, call_, idx_agent_, idx_object_ @@ -197,7 +188,7 @@ inline void Model::actions_add( A.virus = virus_; A.tool = tool_; A.entity = entity_; - A.new_state = new_state_; + A.new_state = new_state_; A.queue = queue_; A.call = call_; A.idx_agent = idx_agent_; @@ -213,76 +204,46 @@ template inline void Model::actions_run() { // Making the call - while (nactions != 0u) + size_t nactions_tmp = 0; + while (nactions_tmp < nactions) { - Action a = actions[--nactions]; + Action & a = actions[nactions_tmp++]; Agent * p = a.agent; - // Applying function - if (a.call) - { - a.call(a, this); - } - - // Updating state - if (static_cast(p->state) != a.new_state) - { - - if (a.new_state >= static_cast(nstates)) - throw std::range_error( - "The proposed state " + std::to_string(a.new_state) + " is out of range. " + - "The model currently has " + std::to_string(nstates - 1) + " states."); - - // Figuring out if we need to undo a change - // If the agent has made a change in the state recently, then we - // need to undo the accounting, e.g., if A->B was made, we need to - // undo it and set B->A so that the daily accounting is right. - if (p->state_last_changed == today()) - { - - // Updating accounting - db.update_state(p->state_prev, p->state, true); // Undoing - db.update_state(p->state_prev, a.new_state); - - for (size_t v = 0u; v < p->n_viruses; ++v) - { - db.update_virus(p->viruses[v]->id, p->state, p->state_prev); // Undoing - db.update_virus(p->viruses[v]->id, p->state_prev, a.new_state); - } - - for (size_t t = 0u; t < p->n_tools; ++t) - { - db.update_tool(p->tools[t]->id, p->state, p->state_prev); // Undoing - db.update_tool(p->tools[t]->id, p->state_prev, a.new_state); - } - - // Changing to the new state, we won't update the - // previous state in case we need to undo the change - p->state = a.new_state; - - } else { - - // Updating accounting - db.update_state(p->state, a.new_state); - - for (size_t v = 0u; v < p->n_viruses; ++v) - db.update_virus(p->viruses[v]->id, p->state, a.new_state); + #ifdef EPI_DEBUG + if (a.new_state >= static_cast(nstates)) + throw std::range_error( + "The proposed state " + std::to_string(a.new_state) + " is out of range. " + + "The model currently has " + std::to_string(nstates - 1) + " states."); - for (size_t t = 0u; t < p->n_tools; ++t) - db.update_tool(p->tools[t]->id, p->state, a.new_state); + if (a.new_state < 0) + throw std::range_error( + "The proposed state " + std::to_string(a.new_state) + " is out of range. " + + "The state cannot be negative."); + #endif - // Saving the last state and setting the new one - p->state_prev = p->state; - p->state = a.new_state; + // Undoing the change in the transition matrix + if ((p->state_last_changed == today()) && (static_cast(p->state) != a.new_state)) + { + // Undoing state change in the transition matrix + // The previous state is already recorded + db.update_state(p->state_prev, p->state, true); - // It used to be a day before, but we still - p->state_last_changed = today(); + } else + p->state_prev = p->state; // Recording the previous state - } - + // Applying function after the fact. This way, if there were + // updates, they can be recorded properly, before losing the information + p->state = a.new_state; + if (a.call) + { + a.call(a, this); } + // Registering that the last change was today + p->state_last_changed = today(); + #ifdef EPI_DEBUG if (static_cast(p->state) >= static_cast(nstates)) throw std::range_error( @@ -311,6 +272,9 @@ inline void Model::actions_run() } + // Go back to square 1 + nactions = 0u; + return; } @@ -435,6 +399,7 @@ inline Model::Model(const Model & model) : pb(model.pb), state_fun(model.state_fun), states_labels(model.states_labels), + initial_states_fun(model.initial_states_fun), nstates(model.nstates), verbose(model.verbose), current_date(model.current_date), @@ -475,6 +440,10 @@ inline Model::Model(const Model & model) : } +template +inline Model::Model(Model & model) : + Model(dynamic_cast< const Model & >(model)) {} + template inline Model::Model(Model && model) : name(std::move(model.name)), @@ -515,6 +484,7 @@ inline Model::Model(Model && model) : pb(std::move(model.pb)), state_fun(std::move(model.state_fun)), states_labels(std::move(model.states_labels)), + initial_states_fun(std::move(model.initial_states_fun)), nstates(model.nstates), verbose(model.verbose), current_date(std::move(model.current_date)), @@ -586,6 +556,7 @@ inline Model & Model::operator=(const Model & m) state_fun = m.state_fun; states_labels = m.states_labels; + initial_states_fun = m.initial_states_fun; nstates = m.nstates; verbose = m.verbose; @@ -648,7 +619,7 @@ inline std::vector< Viruses_const > Model::get_agents_viruses() cons std::vector< Viruses_const > viruses(population.size()); for (size_t i = 0u; i < population.size(); ++i) - viruses[i] = population[i].get_viruses(); + viruses[i] = population[i].get_virus(); return viruses; @@ -661,7 +632,7 @@ inline std::vector< Viruses > Model::get_agents_viruses() std::vector< Viruses > viruses(population.size()); for (size_t i = 0u; i < population.size(); ++i) - viruses[i] = population[i].get_viruses(); + viruses[i] = population[i].get_virus(); return viruses; @@ -674,7 +645,7 @@ inline std::vector> & Model::get_entities() } template -inline void Model::agents_smallworld( +inline Model & Model::agents_smallworld( epiworld_fast_uint n, epiworld_fast_uint k, bool d, @@ -684,6 +655,8 @@ inline void Model::agents_smallworld( agents_from_adjlist( rgraph_smallworld(n, k, p, d, *this) ); + + return *this; } template @@ -770,10 +743,9 @@ inline void Model::dist_virus() // Starting first infection int n = size(); - std::vector< size_t > idx(n); - - int n_left = n; + std::vector< size_t > idx(n, 0u); std::iota(idx.begin(), idx.end(), 0); + int n_left = idx.size(); for (size_t v = 0u; v < viruses.size(); ++v) { @@ -811,7 +783,7 @@ inline void Model::dist_virus() Agent & agent = population[idx[loc]]; // Adding action - agent.add_virus( + agent.set_virus( virus, const_cast * >(this), virus->state_init, @@ -1488,7 +1460,7 @@ inline void Model::next() { } template -inline void Model::run( +inline Model & Model::run( epiworld_fast_uint ndays, int seed ) @@ -1594,6 +1566,8 @@ inline void Model::run( chrono_end(); + return *this; + } template @@ -1824,6 +1798,15 @@ inline void Model::update_state() { template inline void Model::mutate_virus() { + // Checking if any virus has mutation + size_t nmutates = 0u; + for (const auto & v: viruses) + if (v->mutation_fun) + nmutates++; + + if (nmutates == 0u) + return; + if (use_queuing) { @@ -1834,9 +1817,8 @@ inline void Model::mutate_virus() { if (queue[++i] == 0) continue; - if (p.n_viruses > 0u) - for (auto & v : p.get_viruses()) - v->mutate(this); + if (p.virus != nullptr) + p.virus->mutate(this); } @@ -1847,9 +1829,8 @@ inline void Model::mutate_virus() { for (auto & p: population) { - if (p.n_viruses > 0u) - for (auto & v : p.get_viruses()) - v->mutate(this); + if (p.virus != nullptr) + p.virus->mutate(this); } @@ -1890,13 +1871,15 @@ inline bool Model::get_verbose() const { } template -inline void Model::verbose_on() { +inline Model & Model::verbose_on() { verbose = true; + return *this; } template -inline void Model::verbose_off() { +inline Model & Model::verbose_off() { verbose = false; + return *this; } template @@ -2105,6 +2088,9 @@ inline void Model::reset() { dist_virus(); dist_tools(); + // Distributing initial state, if specified + initial_states_fun(this); + // Recording the original state (at time 0) and advancing // to time 1 next(); @@ -2531,9 +2517,10 @@ inline void Model::queuing_on() } template -inline void Model::queuing_off() +inline Model & Model::queuing_off() { use_queuing = false; + return *this; } template @@ -2554,6 +2541,18 @@ inline const std::vector< VirusPtr > & Model::get_viruses() const return viruses; } +template +inline const std::vector< epiworld_double > & Model::get_prevalence_virus() const +{ + return prevalence_virus; +} + +template +inline const std::vector< bool > & Model::get_prevalence_virus_as_proportion() const +{ + return prevalence_virus_as_proportion; +} + template const std::vector< ToolPtr > & Model::get_tools() const { diff --git a/include/epiworld/models/diffnet.hpp b/include/epiworld/models/diffnet.hpp index ee9a99672..b7ab07243 100644 --- a/include/epiworld/models/diffnet.hpp +++ b/include/epiworld/models/diffnet.hpp @@ -91,24 +91,25 @@ inline ModelDiffNet::ModelDiffNet( if (neighbor->get_state() == ModelDiffNet::ADOPTER) { - for (const VirusPtr & v : neighbor->get_viruses()) - { - - /* And it is a function of susceptibility_reduction as well */ - double p_i = - (1.0 - agent.get_susceptibility_reduction(v, m)) * - (1.0 - agent.get_transmission_reduction(v, m)) - ; + auto & v = neighbor->get_virus(); - size_t vid = v->get_id(); - if (!stored[vid]) - { - stored[vid] = true; - innovations[vid] = &(*v); - } - exposure[vid] += p_i; - - } + if (v == nullptr) + continue; + + /* And it is a function of susceptibility_reduction as well */ + double p_i = + (1.0 - agent.get_susceptibility_reduction(v, m)) * + (1.0 - agent.get_transmission_reduction(v, m)) + ; + + size_t vid = v->get_id(); + if (!stored[vid]) + { + stored[vid] = true; + innovations[vid] = &(*v); + } + exposure[vid] += p_i; + } @@ -141,7 +142,7 @@ inline ModelDiffNet::ModelDiffNet( return; // Otherwise, it is adopted from any of the neighbors - agent.add_virus( + agent.set_virus( *innovations.at(which), m, ModelDiffNet::ADOPTER diff --git a/include/epiworld/models/globalactions.hpp b/include/epiworld/models/globalactions.hpp index f0e20fe14..432928d49 100644 --- a/include/epiworld/models/globalactions.hpp +++ b/include/epiworld/models/globalactions.hpp @@ -81,7 +81,9 @@ inline std::function*)> globalaction_tool_logit( // Computing the probability using a logit. Uses OpenMP reduction // to sum the coefficients. double p = 0.0; + #if defined(__OPENMP) || defined(_OPENMP) #pragma omp parallel for reduction(+:p) + #endif for (size_t i = 0u; i < coefs.size(); ++i) p += coefs.at(i) * agent(vars[i]); diff --git a/include/epiworld/models/init-functions.hpp b/include/epiworld/models/init-functions.hpp new file mode 100644 index 000000000..660d5db84 --- /dev/null +++ b/include/epiworld/models/init-functions.hpp @@ -0,0 +1,341 @@ +#ifndef EPIWORLD_MODELS_INIT_FUNCTIONS_HPP +#define EPIWORLD_MODELS_INIT_FUNCTIONS_HPP + +/** + * @brief Creates an initial function for the SIR-like models + * The function is used for the initial states of the model. +*/ +template +inline std::function*)> create_init_function_sir( + std::vector< double > proportions_ +) { + + // Checking widths + if (proportions_.size() != 1u) + throw std::invalid_argument( + "The vector of proportions must have a single element." + ); + + // Proportion should be within [0, 1] + if ((proportions_[0] < 0.0) || (proportions_[0] > 1.0)) + throw std::invalid_argument( + "The proportion must be within (0, 1)." + ); + + double prop = proportions_[0u]; + + std::function*)> fun = + [prop] (epiworld::Model * model) -> void { + + // Figuring out information about the viruses + double tot = 0.0; + const auto & vpreval = model->get_prevalence_virus(); + const auto & vprop = model->get_prevalence_virus_as_proportion(); + double n = static_cast(model->size()); + for (size_t i = 0u; i < model->get_n_viruses(); ++i) + { + if (vprop[i]) + tot += vpreval[i]; + else + tot += vpreval[i] / n; + } + + // Putting the total into context + double tot_left = 1.0 - tot; + + // Since susceptible and infected are "fixed," + // we only need to change recovered + size_t nrecovered = prop * tot_left * n; + + epiworld::AgentsSample sample( + *model, + nrecovered, + {0u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample) + agent->change_state(model, 2, Queue::NoOne); + + // Running the actions + model->actions_run(); + + return; + + }; + + return fun; + +} + +/** + * @brief Creates an initial function for the SIR-like models + * The function is used for the initial states of the model. +*/ +template +inline std::function*)> create_init_function_sird( + std::vector< double > prop +) { + + // Check length of prop equals two + if (prop.size() != 2u) + throw std::invalid_argument( + "The vector of proportions must have two elements." + ); + + // Check elements in prop are within [0, 1] and sum up to 1 + double tot = 0.0; + for (auto & v : prop) + { + if ((v < 0.0) || (v > 1.0)) + throw std::invalid_argument( + "The proportion must be within (0, 1)." + ); + tot += v; + } + + if (tot >= 1.0) + throw std::invalid_argument( + "The proportions must sum up to 1." + ); + + std::function*)> fun = + [prop] (epiworld::Model * model) -> void { + + // Figuring out information about the viruses + double tot = 0.0; + const auto & vpreval = model->get_prevalence_virus(); + const auto & vprop = model->get_prevalence_virus_as_proportion(); + double n = static_cast(model->size()); + for (size_t i = 0u; i < model->get_n_viruses(); ++i) + { + if (vprop[i]) + tot += vpreval[i]; + else + tot += vpreval[i] / n; + } + + // Putting the total into context + double tot_left = 1.0 - tot; + + // Since susceptible and infected are "fixed," + // we only need to change recovered + size_t nrecovered = prop[0u] * tot_left * n; + size_t ndeceased = prop[01] * tot_left * n; + + epiworld::AgentsSample sample_recover( + *model, + nrecovered, + {0u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_recover) + agent->change_state(model, 2, Queue::NoOne); + + epiworld::AgentsSample sample_deceased( + *model, + ndeceased, + {0u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_deceased) + agent->change_state(model, 3, Queue::NoOne); + + // Running the actions + model->actions_run(); + + return; + + }; + + return fun; + +} + + +/** + * @brief Creates an initial function for the SEIR-like models + * The function is used for the initial states of the model. +*/ +template +inline std::function*)> create_init_function_seir( + std::vector< double > proportions_ +) { + + // Checking widths + if (proportions_.size() != 2u) { + throw std::invalid_argument("-proportions_- must have two entries."); + } + + // proportions_ are values between 0 and 1, otherwise error + for (auto & v : proportions_) + if ((v < 0.0) || (v > 1.0)) + throw std::invalid_argument( + "-proportions_- must have values between 0 and 1." + ); + + + std::function*)> fun = + [proportions_] (epiworld::Model * model) -> void { + + // Figuring out information about the viruses + double tot = 0.0; + const auto & vpreval = model->get_prevalence_virus(); + const auto & vprop = model->get_prevalence_virus_as_proportion(); + double n = static_cast(model->size()); + for (size_t i = 0u; i < model->get_n_viruses(); ++i) + { + if (vprop[i]) + tot += vpreval[i]; + else + tot += vpreval[i] / n; + } + + // Putting the total into context + double tot_left = 1.0 - tot; + + // Since susceptible and infected are "fixed," + // we only need to change recovered + size_t nexposed = proportions_[0u] * tot * n; + size_t nrecovered = proportions_[1u] * tot_left * n; + + epiworld::AgentsSample sample_suscept( + *model, + nrecovered, + {0u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_suscept) + agent->change_state(model, 3, Queue::NoOne); + + epiworld::AgentsSample sample_exposed( + *model, + nexposed, + {1u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_exposed) + agent->change_state(model, 2, Queue::NoOne); + + // Running the actions + model->actions_run(); + + return; + + }; + + return fun; + +} + +/** + * @brief Creates an initial function for the SEIR-like models + * The function is used for the initial states of the model. +*/ +template +inline std::function*)> create_init_function_seird( + std::vector< double > proportions_ +) { + + // Checking widths + if (proportions_.size() != 3u) { + throw std::invalid_argument("-proportions_- must have three entries."); + } + + // proportions_ are values between 0 and 1, otherwise error + for (auto & v : proportions_) + if ((v < 0.0) || (v > 1.0)) + throw std::invalid_argument( + "-proportions_- must have values between 0 and 1." + ); + + // Last first two terms shouldn't add up to more than 1 + if ((proportions_[1u] + proportions_[2u]) > 1.0) + throw std::invalid_argument( + "The last two terms of -proportions_- must add up to less than 1." + ); + + std::function*)> fun = + [proportions_] (epiworld::Model * model) -> void { + + // Figuring out information about the viruses + double tot = 0.0; + const auto & vpreval = model->get_prevalence_virus(); + const auto & vprop = model->get_prevalence_virus_as_proportion(); + double n = static_cast(model->size()); + for (size_t i = 0u; i < model->get_n_viruses(); ++i) + { + if (vprop[i]) + tot += vpreval[i]; + else + tot += vpreval[i] / n; + } + + // Putting the total into context + double tot_left = 1.0 - tot; + + // Since susceptible and infected are "fixed," + // we only need to change recovered + size_t nexposed = proportions_[0u] * tot * n; + size_t nrecovered = proportions_[1u] * tot_left * n; + size_t ndeceased = proportions_[2u] * tot_left * n; + + epiworld::AgentsSample sample_suscept( + *model, + nrecovered, + {0u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_suscept) + agent->change_state(model, 3, Queue::NoOne); + + epiworld::AgentsSample sample_exposed( + *model, + nexposed, + {1u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_exposed) + agent->change_state(model, 2, Queue::NoOne); + + // Running the actions + model->actions_run(); + + // Setting the initial states for the deceased + epiworld::AgentsSample sample_deceased( + *model, + ndeceased, + {0u}, + true + ); + + // Setting up the initial states + for (auto & agent : sample_deceased) + agent->change_state(model, 4, Queue::NoOne); + + // Running the actions + model->actions_run(); + + return; + + }; + + return fun; + +} + + +#endif \ No newline at end of file diff --git a/include/epiworld/models/models.hpp b/include/epiworld/models/models.hpp index 2e27bfef4..6c01abe48 100644 --- a/include/epiworld/models/models.hpp +++ b/include/epiworld/models/models.hpp @@ -2,7 +2,9 @@ #define EPIWORLD_MODELS_HPP namespace epimodels { - + + #include "init-functions.hpp" + #include "globalactions.hpp" #include "sis.hpp" #include "sir.hpp" diff --git a/include/epiworld/models/seir.hpp b/include/epiworld/models/seir.hpp index e310a1697..301796ee9 100644 --- a/include/epiworld/models/seir.hpp +++ b/include/epiworld/models/seir.hpp @@ -46,7 +46,7 @@ class ModelSEIR : public epiworld::Model ) -> void { // Getting the virus - auto v = p->get_virus(0); + auto v = p->get_virus(); // Does the agent become infected? if (m->runif() < 1.0/(v->get_incubation(m))) @@ -62,11 +62,22 @@ class ModelSEIR : public epiworld::Model ) -> void { // Does the agent recover? if (m->runif() < (m->par("Recovery rate"))) - p->rm_virus(0, m); + p->rm_virus(m); return; }; + /** + * @brief Set up the initial states of the model. + * @param proportions_ Double vector with the following values: + * - 0: Proportion of non-infected agents who are removed. + * - 1: Proportion of exposed agents to be set as infected. + */ + ModelSEIR & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); + }; @@ -132,6 +143,18 @@ inline ModelSEIR::ModelSEIR( } +template +inline ModelSEIR & ModelSEIR::initial_states( + std::vector< double > proportions_, + std::vector< int > /**/ +) { + + Model::initial_states_fun = + create_init_function_seir(proportions_) + ; + + return *this; +} #endif diff --git a/include/epiworld/models/seirconnected.hpp b/include/epiworld/models/seirconnected.hpp index 4a8ed8e41..93f0eeea8 100644 --- a/include/epiworld/models/seirconnected.hpp +++ b/include/epiworld/models/seirconnected.hpp @@ -35,7 +35,7 @@ class ModelSEIRCONN : public epiworld::Model epiworld_double recovery_rate ); - void run( + ModelSEIRCONN & run( epiworld_fast_uint ndays, int seed = -1 ); @@ -44,10 +44,20 @@ class ModelSEIRCONN : public epiworld::Model Model * clone_ptr(); + /** + * @brief Set the initial states of the model + * @param proportions_ Double vector with a single element: + * - The proportion of non-infected individuals who have recovered. + */ + ModelSEIRCONN & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); + }; template -inline void ModelSEIRCONN::run( +inline ModelSEIRCONN & ModelSEIRCONN::run( epiworld_fast_uint ndays, int seed ) @@ -55,6 +65,8 @@ inline void ModelSEIRCONN::run( Model::run(ndays, seed); + return *this; + } template @@ -149,24 +161,21 @@ inline ModelSEIRCONN::ModelSEIRCONN( if (neighbor.get_state() == ModelSEIRCONN::INFECTED) { - for (const VirusPtr & v : neighbor.get_viruses()) - { - - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor.get_transmission_reduction(v, m)) - ; - - m->array_virus_tmp[nviruses_tmp++] = &(*v); + auto & v = neighbor.get_virus(); + + #ifdef EPI_DEBUG + if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif - } + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor.get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); } } @@ -181,7 +190,7 @@ inline ModelSEIRCONN::ModelSEIRCONN( if (which < 0) return; - p->add_virus( + p->set_virus( *m->array_virus_tmp[which], m, ModelSEIRCONN::EXPOSED @@ -201,7 +210,7 @@ inline ModelSEIRCONN::ModelSEIRCONN( { // Getting the virus - auto & v = p->get_virus(0u); + auto & v = p->get_virus(); // Does the agent become infected? if (m->runif() < 1.0/(v->get_incubation(m))) @@ -219,14 +228,11 @@ inline ModelSEIRCONN::ModelSEIRCONN( // Odd: Die, Even: Recover epiworld_fast_uint n_events = 0u; - for (const auto & v : p->get_viruses()) - { + const auto & v = p->get_virus(); - // Recover - m->array_double_tmp[n_events++] = - 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); - - } + // Recover + m->array_double_tmp[n_events++] = + 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); #ifdef EPI_DEBUG if (n_events == 0u) @@ -250,8 +256,7 @@ inline ModelSEIRCONN::ModelSEIRCONN( return; // Which roulette happen? - size_t which_v = std::floor(which / 2); - p->rm_virus(which_v, m); + p->rm_virus(m); return ; @@ -327,4 +332,19 @@ inline ModelSEIRCONN::ModelSEIRCONN( } +template +inline ModelSEIRCONN & ModelSEIRCONN::initial_states( + std::vector< double > proportions_, + std::vector< int > /* queue_ */ +) +{ + + Model::initial_states_fun = + create_init_function_seir(proportions_) + ; + + return *this; + +} + #endif diff --git a/include/epiworld/models/seirconnected_logit.hpp b/include/epiworld/models/seirconnected_logit.hpp index 6718f678b..397aa95bb 100644 --- a/include/epiworld/models/seirconnected_logit.hpp +++ b/include/epiworld/models/seirconnected_logit.hpp @@ -113,7 +113,7 @@ inline ModelSEIRCONNLogit::ModelSEIRCONNLogit( for (auto & p: *_tracked_agents_infected) { - if (p->get_n_viruses() == 0) + if (p->get_virus() == nullptr) throw std::logic_error("Cannot be infected and have no viruses."); } @@ -154,7 +154,7 @@ inline ModelSEIRCONNLogit::ModelSEIRCONNLogit( // Infecting the individual #ifdef EPI_DEBUG - if (_tracked_agents_infected->operator[](which)->get_n_viruses() == 0) + if (_tracked_agents_infected->operator[](which)->get_virus() == nullptr) { printf_epiworld("[epi-debug] date: %i\n", m->today()); @@ -167,8 +167,8 @@ inline ModelSEIRCONNLogit::ModelSEIRCONNLogit( ); } #endif - p->add_virus( - _tracked_agents_infected->operator[](which)->get_virus(0u), + p->set_virus( + _tracked_agents_infected->operator[](which)->get_virus(), ModelSEIRCONNLogit::EXPOSED ); @@ -216,7 +216,7 @@ inline ModelSEIRCONNLogit::ModelSEIRCONNLogit( { *_tracked_ninfected_next -= 1; - p->rm_virus(0, m); + p->rm_virus(m); return; } diff --git a/include/epiworld/models/seird.hpp b/include/epiworld/models/seird.hpp index a89e038b7..fe6089347 100644 --- a/include/epiworld/models/seird.hpp +++ b/include/epiworld/models/seird.hpp @@ -3,15 +3,7 @@ /** * @brief Template for a Susceptible-Exposed-Infected-Removed-Deceased (SEIRD) model - * - * @param model A Model object where to set up the SIR. - * @param vname std::string Name of the virus - * @param prevalence epiworld_double Initial prevalence the immune system - * @param transmission_rate epiworld_double Transmission rate of the virus - * @param avg_incubation_days epiworld_double Average incubation days of the virus - * @param recovery_rate epiworld_double Recovery rate of the virus. - * @param death_rate epiworld_double Death rate of the virus. - */ +*/ template class ModelSEIRD : public epiworld::Model { @@ -25,6 +17,18 @@ class ModelSEIRD : public epiworld::Model ModelSEIRD() {}; + /** + * @brief Constructor for the SEIRD model. + * + * @tparam TSeq Type of the sequence used in the model. + * @param model Reference to the SEIRD model. + * @param vname Name of the model. + * @param prevalence Prevalence of the disease. + * @param transmission_rate Transmission rate of the disease. + * @param avg_incubation_days Average incubation period of the disease. + * @param recovery_rate Recovery rate of the disease. + * @param death_rate Death rate of the disease. + */ ModelSEIRD( ModelSEIRD & model, std::string vname, @@ -35,6 +39,16 @@ class ModelSEIRD : public epiworld::Model epiworld_double death_rate ); + /** + * @brief Constructor for the SEIRD model. + * + * @param vname Name of the model. + * @param prevalence Initial prevalence of the disease. + * @param transmission_rate Transmission rate of the disease. + * @param avg_incubation_days Average incubation period of the disease. + * @param recovery_rate Recovery rate of the disease. + * @param death_rate Death rate of the disease. + */ ModelSEIRD( std::string vname, epiworld_double prevalence, @@ -50,7 +64,7 @@ class ModelSEIRD : public epiworld::Model ) -> void { // Getting the virus - auto v = p->get_virus(0); + auto v = p->get_virus(); // Does the agent become infected? if (m->runif() < 1.0/(v->get_incubation(m))) @@ -63,23 +77,20 @@ class ModelSEIRD : public epiworld::Model epiworld::UpdateFun update_infected = []( epiworld::Agent * p, epiworld::Model * m ) -> void { - - auto state = p->get_state(); - + // Odd: Die, Even: Recover epiworld_fast_uint n_events = 0u; - for (const auto & v : p->get_viruses()) - { - - // Die - m->array_double_tmp[n_events++] = - v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); - - // Recover - m->array_double_tmp[n_events++] = - 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); + + const auto & v = p->get_virus(); - } + // Die + m->array_double_tmp[n_events++] = + v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); + + // Recover + m->array_double_tmp[n_events++] = + 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); + #ifdef EPI_DEBUG if (n_events == 0u) @@ -106,13 +117,11 @@ class ModelSEIRD : public epiworld::Model if ((which % 2) == 0) // If odd { - size_t which_v = std::ceil(which / 2); - p->rm_agent_by_virus(which_v, m); + p->rm_agent_by_virus(m); } else { - size_t which_v = std::floor(which / 2); - p->rm_virus(which_v, m); + p->rm_virus(m); } @@ -123,6 +132,12 @@ class ModelSEIRD : public epiworld::Model return; }; + + ModelSEIRD & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); + }; @@ -195,6 +210,19 @@ inline ModelSEIRD::ModelSEIRD( } +template +inline ModelSEIRD & ModelSEIRD::initial_states( + std::vector< double > proportions_, + std::vector< int > /**/ +) { + + Model::initial_states_fun = + create_init_function_seird(proportions_) + ; + + return *this; + +} #endif \ No newline at end of file diff --git a/include/epiworld/models/seirdconnected.hpp b/include/epiworld/models/seirdconnected.hpp index bffdd0e32..148801b8c 100644 --- a/include/epiworld/models/seirdconnected.hpp +++ b/include/epiworld/models/seirdconnected.hpp @@ -38,7 +38,7 @@ class ModelSEIRDCONN : public epiworld::Model epiworld_double death_rate ); - void run( + ModelSEIRDCONN & run( epiworld_fast_uint ndays, int seed = -1 ); @@ -47,10 +47,21 @@ class ModelSEIRDCONN : public epiworld::Model Model * clone_ptr(); + /** + * @brief Set up the initial states of the model. + * @param proportions_ Double vector with the following values: + * - 0: Proportion of non-infected agents who are removed. + * - 1: Proportion of exposed agents to be set as infected. + */ + ModelSEIRDCONN & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); + }; template -inline void ModelSEIRDCONN::run( +inline ModelSEIRDCONN & ModelSEIRDCONN::run( epiworld_fast_uint ndays, int seed ) @@ -58,6 +69,8 @@ inline void ModelSEIRDCONN::run( Model::run(ndays, seed); + return *this; + } template @@ -154,25 +167,23 @@ inline ModelSEIRDCONN::ModelSEIRDCONN( if (neighbor.get_state() == ModelSEIRDCONN::INFECTED) { - for (const VirusPtr & v : neighbor.get_viruses()) - { - - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor.get_transmission_reduction(v, m)) - ; - - m->array_virus_tmp[nviruses_tmp++] = &(*v); - - } + const auto & v = neighbor.get_virus(); + + #ifdef EPI_DEBUG + if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif + + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor.get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); + } } @@ -186,7 +197,7 @@ inline ModelSEIRDCONN::ModelSEIRDCONN( if (which < 0) return; - p->add_virus( + p->set_virus( *m->array_virus_tmp[which], m, ModelSEIRDCONN::EXPOSED @@ -206,7 +217,7 @@ inline ModelSEIRDCONN::ModelSEIRDCONN( { // Getting the virus - auto & v = p->get_virus(0u); + auto & v = p->get_virus(); // Does the agent become infected? if (m->runif() < 1.0/(v->get_incubation(m))) @@ -221,56 +232,50 @@ inline ModelSEIRDCONN::ModelSEIRDCONN( } else if (state == ModelSEIRDCONN::INFECTED) { - - // Odd: Die, Even: Recover - epiworld_fast_uint n_events = 0u; - for (const auto & v : p->get_viruses()) - { + // Odd: Die, Even: Recover + epiworld_fast_uint n_events = 0u; + const auto & v = p->get_virus(); // Die m->array_double_tmp[n_events++] = - v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); + v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); // Recover m->array_double_tmp[n_events++] = - 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); - - } - - #ifdef EPI_DEBUG - if (n_events == 0u) - { + 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); + + #ifdef EPI_DEBUG + if (n_events == 0u) + { printf_epiworld( - "[epi-debug] agent %i has 0 possible events!!\n", - static_cast(p->get_id()) + "[epi-debug] agent %i has 0 possible events!!\n", + static_cast(p->get_id()) ); throw std::logic_error("Zero events in exposed."); - } - #else - if (n_events == 0u) + } + #else + if (n_events == 0u) return; - #endif - - - // Running the roulette - int which = roulette(n_events, m); - - if (which < 0) + #endif + + + // Running the roulette + int which = roulette(n_events, m); + + if (which < 0) return; - - // Which roulette happen? - if ((which % 2) == 0) // If odd - { - size_t which_v = std::ceil(which / 2); - p->rm_agent_by_virus(which_v, m); + // Which roulette happen? + if ((which % 2) == 0) // If odd + { + + p->rm_agent_by_virus(m); - } else { + } else { - size_t which_v = std::floor(which / 2); - p->rm_virus(which_v, m); + p->rm_virus(m); - } + } return ; @@ -350,4 +355,18 @@ inline ModelSEIRDCONN::ModelSEIRDCONN( } +template +inline ModelSEIRDCONN & ModelSEIRDCONN::initial_states( + std::vector< double > proportions_, + std::vector< int > /**/ +) { + + Model::initial_states_fun = + create_init_function_seird(proportions_) + ; + + return *this; + +} + #endif diff --git a/include/epiworld/models/sir.hpp b/include/epiworld/models/sir.hpp index 25bce99ae..ae9ca2e30 100644 --- a/include/epiworld/models/sir.hpp +++ b/include/epiworld/models/sir.hpp @@ -31,6 +31,16 @@ class ModelSIR : public epiworld::Model epiworld_double transmission_rate, epiworld_double recovery_rate ); + + /** + * @brief Set the initial states of the model + * @param proportions_ Double vector with a single element: + * - The proportion of non-infected individuals who have recovered. + */ + ModelSIR & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); }; @@ -89,4 +99,18 @@ inline ModelSIR::ModelSIR( } +template +inline ModelSIR & ModelSIR::initial_states( + std::vector< double > proportions_, + std::vector< int > /* queue_ */ +) { + + Model::initial_states_fun = + create_init_function_sir(proportions_) + ; + + return *this; + +} + #endif diff --git a/include/epiworld/models/sirconnected.hpp b/include/epiworld/models/sirconnected.hpp index 84cf844da..dd7ec49ea 100644 --- a/include/epiworld/models/sirconnected.hpp +++ b/include/epiworld/models/sirconnected.hpp @@ -11,12 +11,7 @@ class ModelSIRCONN : public epiworld::Model public: - ModelSIRCONN() { - - // tracked_agents_infected.reserve(1e4); - // tracked_agents_infected_next.reserve(1e4); - - }; + ModelSIRCONN() {}; ModelSIRCONN( ModelSIRCONN & model, @@ -37,17 +32,7 @@ class ModelSIRCONN : public epiworld::Model epiworld_double recovery_rate ); - // Tracking who is infected and who is not - // std::vector< epiworld::Agent* > tracked_agents_infected = {}; - // std::vector< epiworld::Agent* > tracked_agents_infected_next = {}; - // std::vector< epiworld_double > tracked_agents_weight = {}; - // std::vector< epiworld_double > tracked_agents_weight_next = {}; - - // int tracked_ninfected = 0; - // int tracked_ninfected_next = 0; - // epiworld_double tracked_current_infect_prob = 0.0; - - void run( + ModelSIRCONN & run( epiworld_fast_uint ndays, int seed = -1 ); @@ -56,24 +41,28 @@ class ModelSIRCONN : public epiworld::Model Model * clone_ptr(); + /** + * @brief Set the initial states of the model + * @param proportions_ Double vector with a single element: + * - The proportion of non-infected individuals who have recovered. + */ + ModelSIRCONN & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); + }; template -inline void ModelSIRCONN::run( +inline ModelSIRCONN & ModelSIRCONN::run( epiworld_fast_uint ndays, int seed ) { - // tracked_agents_infected.clear(); - // tracked_agents_infected_next.clear(); - - // tracked_ninfected = 0; - // tracked_ninfected_next = 0; - // tracked_current_infect_prob = 0.0; - Model::run(ndays, seed); + return *this; } @@ -82,14 +71,6 @@ inline void ModelSIRCONN::reset() { Model::reset(); - - // Model::set_rand_binom( - // Model::size(), - // static_cast( - // Model::par("Contact rate"))/ - // static_cast(Model::size()) - // ); - return; } @@ -129,8 +110,6 @@ inline ModelSIRCONN::ModelSIRCONN( ) { - - epiworld::UpdateFun update_susceptible = []( epiworld::Agent * p, epiworld::Model * m ) -> void @@ -177,24 +156,25 @@ inline ModelSIRCONN::ModelSIRCONN( if (neighbor.get_state() == ModelSIRCONN::INFECTED) { - for (const VirusPtr & v : neighbor.get_viruses()) - { - - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor.get_transmission_reduction(v, m)) - ; - - m->array_virus_tmp[nviruses_tmp++] = &(*v); + if (neighbor.get_virus() == nullptr) + continue; + + auto & v = neighbor.get_virus(); + + #ifdef EPI_DEBUG + if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif - } + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor.get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); + } } @@ -209,7 +189,7 @@ inline ModelSIRCONN::ModelSIRCONN( if (which < 0) return; - p->add_virus(*m->array_virus_tmp[which], m); + p->set_virus(*m->array_virus_tmp[which], m); return; @@ -228,14 +208,10 @@ inline ModelSIRCONN::ModelSIRCONN( // Odd: Die, Even: Recover epiworld_fast_uint n_events = 0u; - for (const auto & v : p->get_viruses()) - { - - // Recover - m->array_double_tmp[n_events++] = - 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); - - } + // Recover + m->array_double_tmp[n_events++] = + 1.0 - (1.0 - p->get_virus()->get_prob_recovery(m)) * + (1.0 - p->get_recovery_enhancer(p->get_virus(), m)); #ifdef EPI_DEBUG if (n_events == 0u) @@ -259,13 +235,14 @@ inline ModelSIRCONN::ModelSIRCONN( return; // Which roulette happen? - size_t which_v = std::floor(which / 2); - p->rm_virus(which_v, m); + p->rm_virus(m); return ; } else - throw std::logic_error("This function can only be applied to infected individuals. (SIR)") ; + throw std::logic_error( + "This function can only be applied to infected individuals. (SIR)" + ) ; return; @@ -325,5 +302,19 @@ inline ModelSIRCONN::ModelSIRCONN( } +template +inline ModelSIRCONN & ModelSIRCONN::initial_states( + std::vector< double > proportions_, + std::vector< int > /**/ +) { + + Model::initial_states_fun = + create_init_function_sir(proportions_) + ; + + return *this; + +} + #endif diff --git a/include/epiworld/models/sird.hpp b/include/epiworld/models/sird.hpp index 6109e94ac..f9f518469 100644 --- a/include/epiworld/models/sird.hpp +++ b/include/epiworld/models/sird.hpp @@ -3,13 +3,6 @@ /** * @brief Template for a Susceptible-Infected-Removed-Deceased (SIRD) model - * - * @param model A Model object where to set up the SIRD. - * @param vname std::string Name of the virus - * @param initial_prevalence epiworld_double Initial prevalence - * @param initial_efficacy epiworld_double Initial susceptibility_reduction of the immune system - * @param initial_recovery epiworld_double Initial recovery_rate rate of the immune system - * @param initial_death epiworld_double Initial death_rate of the immune system */ template class ModelSIRD : public epiworld::Model @@ -18,6 +11,18 @@ class ModelSIRD : public epiworld::Model ModelSIRD() {}; + + /** + * @brief Constructs a new SIRD model with the given parameters. + * + * @param model The SIRD model to copy from. + * @param vname The name of the vertex associated with this model. + * @param prevalence The initial prevalence of the disease in the population. + * @param transmission_rate The rate at which the disease spreads from infected to susceptible individuals. + * @param recovery_rate The rate at which infected individuals recover and become immune. + * @param death_rate The rate at which infected individuals die. + */ + ///@{ ModelSIRD( ModelSIRD & model, std::string vname, @@ -34,6 +39,18 @@ class ModelSIRD : public epiworld::Model epiworld_double recovery_rate, epiworld_double death_rate ); + ///@} + + /** + * @brief Set the initial states of the model + * @param proportions_ Double vector with two elements: + * - The proportion of non-infected individuals who have recovered. + * - The proportion of non-infected individuals who have died. + */ + ModelSIRD & initial_states( + std::vector< double > proportions_, + std::vector< int > queue_ = {} + ); }; @@ -98,4 +115,18 @@ inline ModelSIRD::ModelSIRD( } +template +inline ModelSIRD & ModelSIRD::initial_states( + std::vector< double > proportions_, + std::vector< int > /**/ +) { + + Model::initial_states_fun = + create_init_function_sird(proportions_) + ; + + return *this; + +} + #endif diff --git a/include/epiworld/models/sirdconnected.hpp b/include/epiworld/models/sirdconnected.hpp index 4caa1cbfa..68f2db0e2 100644 --- a/include/epiworld/models/sirdconnected.hpp +++ b/include/epiworld/models/sirdconnected.hpp @@ -48,7 +48,7 @@ class ModelSIRDCONN : public epiworld::Model // int tracked_ninfected_next = 0; // epiworld_double tracked_current_infect_prob = 0.0; - void run( + ModelSIRDCONN & run( epiworld_fast_uint ndays, int seed = -1 ); @@ -61,21 +61,16 @@ class ModelSIRDCONN : public epiworld::Model }; template -inline void ModelSIRDCONN::run( +inline ModelSIRDCONN & ModelSIRDCONN::run( epiworld_fast_uint ndays, int seed ) { - // tracked_agents_infected.clear(); - // tracked_agents_infected_next.clear(); - - // tracked_ninfected = 0; - // tracked_ninfected_next = 0; - // tracked_current_infect_prob = 0.0; - Model::run(ndays, seed); + return *this; + } template @@ -180,24 +175,21 @@ inline ModelSIRDCONN::ModelSIRDCONN( if (neighbor.get_state() == ModelSIRDCONN::INFECTED) { - for (const VirusPtr & v : neighbor.get_viruses()) - { - - #ifdef EPI_DEBUG - if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor.get_transmission_reduction(v, m)) - ; + const auto & v = neighbor.get_virus(); - m->array_virus_tmp[nviruses_tmp++] = &(*v); + #ifdef EPI_DEBUG + if (nviruses_tmp >= static_cast(m->array_virus_tmp.size())) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif - } + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor.get_transmission_reduction(v, m)) + ; + + m->array_virus_tmp[nviruses_tmp++] = &(*v); } } @@ -212,7 +204,7 @@ inline ModelSIRDCONN::ModelSIRDCONN( if (which < 0) return; - p->add_virus(*m->array_virus_tmp[which], m); + p->set_virus(*m->array_virus_tmp[which], m); return; @@ -229,55 +221,50 @@ inline ModelSIRDCONN::ModelSIRDCONN( { - // Odd: Die, Even: Recover - epiworld_fast_uint n_events = 0u; - for (const auto & v : p->get_viruses()) - { - + // Odd: Die, Even: Recover + epiworld_fast_uint n_events = 0u; + const auto & v = p->get_virus(); + // Die m->array_double_tmp[n_events++] = - v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); + v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); // Recover m->array_double_tmp[n_events++] = - 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); + 1.0 - (1.0 - v->get_prob_recovery(m)) * (1.0 - p->get_recovery_enhancer(v, m)); - } - -#ifdef EPI_DEBUG - if (n_events == 0u) - { - printf_epiworld( - "[epi-debug] agent %i has 0 possible events!!\n", - static_cast(p->get_id()) - ); - throw std::logic_error("Zero events in exposed."); - } -#else - if (n_events == 0u) - return; -#endif - - - // Running the roulette - int which = roulette(n_events, m); - - if (which < 0) - return; - - // Which roulette happen? - if ((which % 2) == 0) // If odd - { + #ifdef EPI_DEBUG + if (n_events == 0u) + { + printf_epiworld( + "[epi-debug] agent %i has 0 possible events!!\n", + static_cast(p->get_id()) + ); + throw std::logic_error("Zero events in exposed."); + } + #else + if (n_events == 0u) + return; + #endif - size_t which_v = std::ceil(which / 2); - p->rm_agent_by_virus(which_v, m); - } else { + // Running the roulette + int which = roulette(n_events, m); - size_t which_v = std::floor(which / 2); - p->rm_virus(which_v, m); + if (which < 0) + return; - } + // Which roulette happen? + if ((which % 2) == 0) // If odd + { + + p->rm_agent_by_virus(m); + + } else { + + p->rm_virus(m); + + } return ; diff --git a/include/epiworld/models/sirlogit.hpp b/include/epiworld/models/sirlogit.hpp index 00eff4e7e..81ed8616b 100644 --- a/include/epiworld/models/sirlogit.hpp +++ b/include/epiworld/models/sirlogit.hpp @@ -51,7 +51,7 @@ class ModelSIRLogit : public epiworld::Model epiworld_double prevalence ); - void run( + ModelSIRLogit & run( epiworld_fast_uint ndays, int seed = -1 ); @@ -70,13 +70,14 @@ class ModelSIRLogit : public epiworld::Model template -inline void ModelSIRLogit::run( +inline ModelSIRLogit & ModelSIRLogit::run( epiworld_fast_uint ndays, int seed ) { Model::run(ndays, seed); + return *this; } @@ -189,30 +190,30 @@ inline ModelSIRLogit::ModelSIRLogit( for (auto & neighbor: p->get_neighbors()) { - for (const VirusPtr & v : neighbor->get_viruses()) - { - - #ifdef EPI_DEBUG - if (nviruses_tmp >= m->array_virus_tmp.size()) - throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); - #endif - - /* And it is a function of susceptibility_reduction as well */ - m->array_double_tmp[nviruses_tmp] = - baseline + - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) * - coef_exposure - ; - - // Applying the plogis function - m->array_double_tmp[nviruses_tmp] = 1.0/ - (1.0 + std::exp(-m->array_double_tmp[nviruses_tmp])); - - m->array_virus_tmp[nviruses_tmp++] = &(*v); - - } + if (neighbor->get_virus() == nullptr) + continue; + + auto & v = neighbor->get_virus(); + + #ifdef EPI_DEBUG + if (nviruses_tmp >= m->array_virus_tmp.size()) + throw std::logic_error("Trying to add an extra element to a temporal array outside of the range."); + #endif + + /* And it is a function of susceptibility_reduction as well */ + m->array_double_tmp[nviruses_tmp] = + baseline + + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) * + coef_exposure + ; + + // Applying the plogis function + m->array_double_tmp[nviruses_tmp] = 1.0/ + (1.0 + std::exp(-m->array_double_tmp[nviruses_tmp])); + + m->array_virus_tmp[nviruses_tmp++] = &(*v); } @@ -226,7 +227,7 @@ inline ModelSIRLogit::ModelSIRLogit( if (which < 0) return; - p->add_virus(*m->array_virus_tmp[which], m); + p->set_virus(*m->array_virus_tmp[which], m); return; @@ -242,7 +243,9 @@ inline ModelSIRLogit::ModelSIRLogit( // Computing recovery probability once double prob = 0.0; + #if defined(__OPENMP) || defined(_OPENMP) #pragma omp simd reduction(+:prob) + #endif for (size_t i = 0u; i < _m->coefs_recover.size(); ++i) prob += p->operator[](i) * _m->coefs_recover[i]; @@ -250,7 +253,7 @@ inline ModelSIRLogit::ModelSIRLogit( prob = 1.0/(1.0 + std::exp(-prob)); if (prob > m->runif()) - p->rm_virus(0, m); + p->rm_virus(m); return; diff --git a/include/epiworld/models/surveillance.hpp b/include/epiworld/models/surveillance.hpp index 695f0a7f4..81fd83b97 100644 --- a/include/epiworld/models/surveillance.hpp +++ b/include/epiworld/models/surveillance.hpp @@ -122,22 +122,20 @@ inline ModelSURV::ModelSURV( for (auto & neighbor: p->get_neighbors()) { - for (size_t i = 0u; i < neighbor->get_n_viruses(); ++i) - { + auto & v = neighbor->get_virus(); - auto & v = neighbor->get_virus(i); - - /* And it is a function of susceptibility_reduction as well */ - epiworld_double tmp_transmission = - (1.0 - p->get_susceptibility_reduction(v, m)) * - v->get_prob_infecting(m) * - (1.0 - neighbor->get_transmission_reduction(v, m)) - ; - - m->array_double_tmp[nviruses_tmp] = tmp_transmission; - m->array_virus_tmp[nviruses_tmp++] = &(*v); + if (v == nullptr) + continue; - } + /* And it is a function of susceptibility_reduction as well */ + epiworld_double tmp_transmission = + (1.0 - p->get_susceptibility_reduction(v, m)) * + v->get_prob_infecting(m) * + (1.0 - neighbor->get_transmission_reduction(v, m)) + ; + + m->array_double_tmp[nviruses_tmp] = tmp_transmission; + m->array_virus_tmp[nviruses_tmp++] = &(*v); } // No virus to compute on @@ -150,7 +148,7 @@ inline ModelSURV::ModelSURV( if (which < 0) return; - p->add_virus(*m->array_virus_tmp[which], m); + p->set_virus(*m->array_virus_tmp[which], m); return; }; @@ -160,7 +158,7 @@ inline ModelSURV::ModelSURV( [](epiworld::Agent * p, epiworld::Model * m) -> void { - epiworld::VirusPtr & v = p->get_virus(0u); + epiworld::VirusPtr & v = p->get_virus(); epiworld_double p_die = v->get_prob_death(m) * (1.0 - p->get_death_reduction(v, m)); epiworld_fast_uint days_since_exposed = m->today() - v->get_date(); @@ -184,7 +182,7 @@ inline ModelSURV::ModelSURV( // If past days infected + latent, then bye. if (days_since_exposed >= v->get_data()[1u]) { - p->rm_virus(0, m); + p->rm_virus(m); return; } diff --git a/include/epiworld/tool-meat.hpp b/include/epiworld/tool-meat.hpp index e70d2f60b..a4925c552 100644 --- a/include/epiworld/tool-meat.hpp +++ b/include/epiworld/tool-meat.hpp @@ -66,7 +66,9 @@ inline ToolFun tool_fun_logit( size_t K = coefs_f.size(); epiworld_double res = 0.0; + #if defined(__OPENMP) || defined(_OPENMP) #pragma omp simd reduction(+:res) + #endif for (size_t i = 0u; i < K; ++i) res += agent->operator[](vars.at(i)) * coefs_f.at(i); diff --git a/include/epiworld/virus-bones.hpp b/include/epiworld/virus-bones.hpp index 1a34f8e05..8397e4539 100644 --- a/include/epiworld/virus-bones.hpp +++ b/include/epiworld/virus-bones.hpp @@ -30,8 +30,6 @@ class Virus { private: Agent * agent = nullptr; - int pos_in_agent = -99; ///< Location in the agent - int agent_exposure_number = -99; std::shared_ptr baseline_sequence = nullptr; std::shared_ptr virus_name = nullptr; @@ -46,7 +44,6 @@ class Virus { VirusFun incubation_fun = nullptr; // Setup parameters - std::vector< epiworld_double * > params = {}; std::vector< epiworld_double > data = {}; epiworld_fast_int state_init = -99; ///< Change of state when added to agent. @@ -67,7 +64,7 @@ class Virus { void set_sequence(TSeq sequence); Agent * get_agent(); - void set_agent(Agent * p, epiworld_fast_uint idx); + void set_agent(Agent * p); void set_date(int d); int get_date() const; diff --git a/include/epiworld/virus-meat.hpp b/include/epiworld/virus-meat.hpp index d5a10ae40..c0fae7a1e 100644 --- a/include/epiworld/virus-meat.hpp +++ b/include/epiworld/virus-meat.hpp @@ -65,7 +65,9 @@ inline VirusFun virus_fun_logit( size_t K = coefs_f.size(); epiworld_double res = 0.0; + #if defined(__OPENMP) || defined(_OPENMP) #pragma omp simd reduction(+:res) + #endif for (size_t i = 0u; i < K; ++i) res += agent->operator[](vars.at(i)) * coefs_f.at(i); @@ -128,21 +130,9 @@ inline Agent * Virus::get_agent() } template -inline void Virus::set_agent(Agent * p, epiworld_fast_uint idx) +inline void Virus::set_agent(Agent * p) { - - #ifdef EPI_DEBUG - if (idx >= p->viruses.size()) - { - printf_epiworld( - "[epi-debug]Virus::set_agent id to set up is outside of range." - ); - } - #endif - - agent = p; - pos_in_agent = static_cast(idx); - + agent = p; } template diff --git a/readme.cpp b/readme.cpp index 11986b13d..e02c5d09d 100644 --- a/readme.cpp +++ b/readme.cpp @@ -27,7 +27,7 @@ int main() virus.set_status(1, 2); - model.add_virus_n(virus, 1000); + model.default_add_virusn(virus, 1000); // Generating a random pop from a smallworld network model.agents_smallworld(100000, 4L, false, .01); diff --git a/tests/00-cloning-model.cpp b/tests/00-cloning-model.cpp index 5daab5526..fd466a112 100644 --- a/tests/00-cloning-model.cpp +++ b/tests/00-cloning-model.cpp @@ -6,20 +6,21 @@ EPIWORLD_TEST_CASE("Cloning", "[clone]") { epiworld::Model m; - m.add_status("Susceptible", default_update_susceptible); - m.add_status("Recovered"); + m.add_state("Susceptible", default_update_susceptible); + m.add_state("Recovered"); epiworld::Virus v; epiworld::Tool t; - v.set_status(0, 1); + v.set_state(0, 1); - m.agents_from_adjlist("edgelist.txt", 1000); + m.seed(1333); + m.agents_smallworld(1000); m.add_virus(v, .5); m.add_tool(t, .5); // Cloning - epiworld::Modelm2(m); + epiworld::Model m2 = m; // Printing the addresses std::cout << @@ -38,13 +39,13 @@ EPIWORLD_TEST_CASE("Cloning", "[clone]") { std::cout << n << ", "; std::cout << std::endl; - std::cout << "Agent[0] in m viruses and tools : " << - m.get_agents()[0u].get_virus(0u)->get_agent() << ", " << - m.get_agents()[0u].get_tool(0u)->get_agent() << std::endl; + // std::cout << "Agent[0] in m tools : " << + // // m.get_agents()[0u].get_virus()->get_agent() << ", " << + // m.get_agents()[0u].get_tool(0u)->get_agent() << std::endl; - std::cout << "Agent[0] in m2 viruses and tools : " << - m2.get_agents()[0u].get_virus(0u)->get_agent() << ", " << - m2.get_agents()[0u].get_tool(0u)->get_agent() << std::endl; + // std::cout << "Agent[0] in m2 tools : " << + // // m2.get_agents()[0u].get_virus()->get_agent() << ", " << + // m2.get_agents()[0u].get_tool(0u)->get_agent() << std::endl; #ifndef CATCH_CONFIG_MAIN diff --git a/tests/01-sir.cpp b/tests/01-sir.cpp index b9425082c..f176fde9b 100644 --- a/tests/01-sir.cpp +++ b/tests/01-sir.cpp @@ -58,8 +58,8 @@ EPIWORLD_TEST_CASE("SIR", "[SIR]") { REQUIRE(out_of_range_0 == 0); REQUIRE(out_of_range_1 == 0); #else - model_0.print(true); - model_1.print(true); + model_0.print(false); + model_1.print(false); #endif #ifndef CATCH_CONFIG_MAIN diff --git a/tests/04-initial-dist.cpp b/tests/04-initial-dist.cpp new file mode 100644 index 000000000..a91b53696 --- /dev/null +++ b/tests/04-initial-dist.cpp @@ -0,0 +1,212 @@ +#ifndef CATCH_CONFIG_MAIN +#define EPI_DEBUG +#endif + +#include "tests.hpp" + +using namespace epiworld; + +EPIWORLD_TEST_CASE("SIR dist", "[SIR-dist]") { + + // Queuing doesn't matter and get results that are meaningful + epimodels::ModelSIR<> model_0( + "a virus", 0.01, .9, .3 + ); + + model_0.initial_states({0.5}). + agents_smallworld(10000, 5, false, 0.01). + verbose_off(). + run(100, 1231); + + epimodels::ModelSIR<> model_1( + "a virus", 0.01, .9, .3 + ); + + model_1.initial_states({.5}). + agents_smallworld(10000, 5, false, 0.01). + queuing_off(). + verbose_off(). + run(100, 1231); + + std::vector< int > h_0, h_1; + model_0.get_db().get_hist_total(nullptr, nullptr, &h_0); + model_1.get_db().get_hist_total(nullptr, nullptr, &h_1); + + // Getting transition matrix + auto tmat_0 = model_0.get_db().transition_probability(false); + int out_of_range_0 = 0; + + for (auto & v: tmat_0) + if (v < 0.0 | v > 1.0) + out_of_range_0++; + + auto tmat_1 = model_1.get_db().transition_probability(false); + int out_of_range_1 = 0; + + for (auto & v: tmat_1) + if (v < 0.0 | v > 1.0) + out_of_range_1++; + + std::vector< epiworld_double > tmat_expected = { + 0.9988896435, 0.0, 0.0, 0.0011103565, 0.6434485983, + 0.0, 0.0, 0.3565514017, 1.0 + }; + + #ifdef CATCH_CONFIG_MAIN + REQUIRE_THAT(tmat_0, Catch::Approx(tmat_expected).margin(0.025)); + REQUIRE_THAT(tmat_1, Catch::Equals(tmat_0)); + REQUIRE_THAT(h_0, Catch::Equals(h_1)); + REQUIRE(out_of_range_0 == 0); + REQUIRE(out_of_range_1 == 0); + #else + model_0.print(false); + model_1.print(false); + #endif + + // Checking that the history matches the expected values + #ifdef CATCH_CONFIG_MAIN + REQUIRE_FALSE( + ((h_0[0u] != 4950) || (h_0[1u] != 100) || (h_0[2u] != 4950)) + ); + #endif + + // Trying out the SEIR model now ------------------------------------------- + epimodels::ModelSEIR<> model_2( + "a virus", 0.01, .5, 7.0, .1 + ); + + + model_2.agents_smallworld(10000, 5, false, 0.01). + queuing_off(). + verbose_off(). + initial_states({.3, .5}, {}). + run(100, 1231). + print(false); + + h_0.clear(); + model_2.get_db().get_hist_total(nullptr, nullptr, &h_0); + + #ifdef CATCH_CONFIG_MAIN + REQUIRE_FALSE( + ((h_0[0u] != 4950) || (h_0[1u] != 70) || (h_0[2u] != 30) || (h_0[3u] != 4950)) + ); + #endif + + // Trying SIRCONN ---------------------------------------------------------- + epimodels::ModelSIRCONN<> model_3( + "a virus", 10000, 0.01, 2, .5, .3 + ); + + model_3.initial_states({.5}). + verbose_off(). + run(100, 222). + print(false); + + h_0.clear(); + model_3.get_db().get_hist_total(nullptr, nullptr, &h_0); + + #ifdef CATCH_CONFIG_MAIN + REQUIRE_FALSE( + ((h_0[0u] != 4950) || (h_0[1u] != 100) || (h_0[2u] != 4950)) + ); + #endif + + // Trying SEIRCONN --------------------------------------------------------- + epimodels::ModelSEIRCONN<> model_4( + "a virus", 10000, 0.01, 2, .5, 7, .1 + ); + + model_4.initial_states({.3, .5}). + verbose_off(). + run(100, 222). + print(false); + + h_0.clear(); + model_4.get_db().get_hist_total(nullptr, nullptr, &h_0); + + #ifdef CATCH_CONFIG_MAIN + REQUIRE_FALSE( + ((h_0[0u] != 4950) || (h_0[1u] != 70) || (h_0[2u] != 30) || (h_0[3u] != 4950)) + ); + #endif + + // Trying SIRD ------------------------------------------------------------- + epimodels::ModelSIRD<> model_5( + "a virus", 0.01, .9, .3, .1 + ); + + model_5.initial_states({0.5, .05}). + agents_smallworld(10000, 5, false, 0.01). + verbose_off(). + run(100, 1231). + print(false); + + h_0.clear(); + model_5.get_db().get_hist_total(nullptr, nullptr, &h_0); + + #ifdef CATCH_CONFIG_MAIN + REQUIRE_FALSE( + ( + moreless(h_0[0u], 4700) || + moreless(h_0[1u], 100) || + moreless(h_0[2u], 4705) || + moreless(h_0[3u], 495)) + ); + #endif + + // Trying SEIRD ------------------------------------------------------------- + epimodels::ModelSEIRD<> model_6( + "a virus", 0.1, .9, 7, .3, .1 + ); + + model_6.initial_states({0.5, .1, .1}). + agents_smallworld(10000, 5, false, 0.01). + verbose_off(). + run(100, 1231). + print(false); + + h_0.clear(); + model_6.get_db().get_hist_total(nullptr, nullptr, &h_0); + + #ifdef CATCH_CONFIG_MAIN + REQUIRE_FALSE( + ( + moreless(h_0[0u], 7200) || + moreless(h_0[1u], 500) || + moreless(h_0[2u], 500) || + moreless(h_0[3u], 900) || + moreless(h_0[4u], 900) + ) + ); + #endif + + // Trying SEIRDCONN -------------------------------------------------------- + epimodels::ModelSEIRDCONN<> model_7( + "a virus", 10000, 0.1, 4, .9, 7, .3, .1 + ); + + model_7.initial_states({0.5, .1, .1}). + verbose_off(). + run(100, 1231). + print(false); + + h_0.clear(); + model_7.get_db().get_hist_total(nullptr, nullptr, &h_0); + + #ifdef CATCH_CONFIG_MAIN + REQUIRE_FALSE( + ( + moreless(h_0[0u], 7200) || + moreless(h_0[1u], 500) || + moreless(h_0[2u], 500) || + moreless(h_0[3u], 900) || + moreless(h_0[4u], 900) + ) + ); + #endif + + #ifndef CATCH_CONFIG_MAIN + return 0; + #endif + +} diff --git a/tests/Makefile b/tests/Makefile index 021a15945..344658e29 100644 --- a/tests/Makefile +++ b/tests/Makefile @@ -1,7 +1,7 @@ all: main.o main.o: main.cpp clean - g++ -std=c++14 -Wall -Wextra -Wnull-dereference -fdelete-null-pointer-checks -O2 -fopenmp -g -pedantic main.cpp -o main.o + g++ -std=c++14 -Wunused -Wall -Wextra -Wnull-dereference -fdelete-null-pointer-checks -O2 -fopenmp -g -pedantic main.cpp -o main.o main.a: main.cpp clean clang++ -std=c++14 -Wall -Wextra -fopenmp -Wpedantic -O2 main.cpp -o main.a diff --git a/tests/main.cpp b/tests/main.cpp index 67dcad636..510a6bfaa 100644 --- a/tests/main.cpp +++ b/tests/main.cpp @@ -6,11 +6,12 @@ #include "tests.hpp" - +#include "00-cloning-model.cpp" #include "00-lfmcmc.cpp" #include "01-sir.cpp" -// #include "01b-sir.cpp" +#include "01b-sir.cpp" #include "01c-sir.cpp" #include "01-sirconnected.cpp" #include "02-reproducible-sir.cpp" -#include "02-reproducible-sirconn.cpp" \ No newline at end of file +#include "02-reproducible-sirconn.cpp" +#include "04-initial-dist.cpp" \ No newline at end of file diff --git a/tests/tests.hpp b/tests/tests.hpp index ccab1b5d6..303759ff5 100644 --- a/tests/tests.hpp +++ b/tests/tests.hpp @@ -9,6 +9,12 @@ #include #include "../include/epiworld/epiworld.hpp" +template +inline bool moreless(T a, T b, T eps = static_cast(1)) +{ + return(std::abs(a-b) > eps); +} + std::string file_reader(std::string fname) { // Create a text string, which is used to output the text file @@ -50,4 +56,5 @@ std::string file_reader(std::string fname) #endif + #endif \ No newline at end of file