Skip to content

Commit

Permalink
Impl, test ReplicateDemesHighestFecundity
Browse files Browse the repository at this point in the history
This criterion prevents demes from gaming
ReplicateDemesHighestBirthCount by spamming nonviable offspring
  • Loading branch information
mmore500 committed Nov 27, 2023
1 parent 8d41137 commit 04a5341
Show file tree
Hide file tree
Showing 14 changed files with 1,382 additions and 0 deletions.
110 changes: 110 additions & 0 deletions avida-core/source/actions/PopulationActions.cc
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,7 @@
#include <limits>
#include <numeric>
#include <set>
#include <utility>
#include <vector>

#include "stdlib.h"
Expand Down Expand Up @@ -5471,6 +5472,114 @@ class cActionReplicateDemesHighestBirthCount : public cAction
};


/*
* Replicate deme(s) with the highest fecundity
* (cell count with ties broken by birth count)
*/
class cActionReplicateDemesHighestFecundity : public cAction
{
private:
double m_replprob;
int m_replmax;
public:
cActionReplicateDemesHighestFecundity(cWorld *world, const cString &args, Feedback &) : cAction(world, args), m_replprob(0.01), m_replmax(std::numeric_limits<int>::max())
{
cString largs(args);
if (largs.GetSize()) m_replprob = largs.PopWord().AsDouble();
if (largs.GetSize()) m_replmax = largs.PopWord().AsInt();

assert(m_replprob >= 0);
}

static const cString GetDescription() { return "Arguments: [double replprob=0.01] [int replmax = intmax]"; }

void Process(cAvidaContext &ctx)
{
const int part_size = m_world->GetConfig().DEMES_PARTITION_INTERVAL.Get();
const int num_demes = m_world->GetPopulation().GetNumDemes();
if (part_size)
{
for (int i = 0; i < num_demes; i += part_size)
{
const int begin = i;
const int end = std::min(i + part_size, num_demes);
ProcessPartition(begin, end, ctx);
}
}
else
{
ProcessPartition(0, num_demes, ctx);
}
}

void ProcessPartition(const int begin, const int end, cAvidaContext &ctx)
{
cPopulation& pop = m_world->GetPopulation();
const int num_demes = end - begin;
std::vector<int> deme_indices(num_demes);
std::iota(std::begin(deme_indices), std::end(deme_indices), begin);

const int binomial_draw = ctx.GetRandom().GetRandBinomial(
num_demes,
m_replprob
);
const int repl_quota = std::min(binomial_draw, m_replmax);
if (repl_quota == 0) return;

if (repl_quota != binomial_draw) {
std::cout << "warning: capped repl quota at " << repl_quota << " from " << binomial_draw << " binomial sample with " << num_demes << " eligible and repl prob " << m_replprob << std::endl;
}

using fecundity_t = std::pair<int, int>;
struct GetFecundity {
cPopulation &pop;
GetFecundity(cPopulation &pop) : pop(pop) {}
fecundity_t operator()(const int d) {
return fecundity_t(
pop.GetDeme(d).GetOrgCount(), pop.GetDeme(d).GetBirthCount()
);
}
};
std::vector<fecundity_t> fecundities(num_demes);
std::transform(
std::begin(deme_indices), std::end(deme_indices),
std::begin(fecundities),
GetFecundity(pop)
);

struct Comp {
std::vector<fecundity_t> &fecundities;
const int begin;
Comp(std::vector<fecundity_t> &fecundities, const int begin)
: fecundities(fecundities), begin(begin) {}
bool operator()(const int d1, const int d2)
{
return fecundities[d1 - begin] > fecundities[d2 - begin];
}
};
std::partial_sort(
std::begin(deme_indices),
std::next(std::begin(deme_indices), repl_quota),
std::end(deme_indices),
Comp(fecundities, begin)
);

struct DoRepl {
cPopulation &pop;
cAvidaContext &ctx;
DoRepl(cPopulation &pop, cAvidaContext &ctx) : pop(pop), ctx(ctx) {}
void operator()(const int d) { pop.ReplicateDeme(pop.GetDeme(d), ctx); }
};
std::for_each(
std::begin(deme_indices),
std::next(std::begin(deme_indices), repl_quota),
DoRepl(pop, ctx)
);

} // End Process()
};


/*
Set the ages at which treatable demes can be treated
Expand Down Expand Up @@ -6143,6 +6252,7 @@ void RegisterPopulationActions(cActionLibrary* action_lib)
action_lib->Register<cActionSetDemeTreatmentAges>("SetDemeTreatmentAges");
action_lib->Register<cActionKillDemesHighestParasiteLoad>("KillDemesHighestParasiteLoad");
action_lib->Register<cActionReplicateDemesHighestBirthCount>("ReplicateDemesHighestBirthCount");
action_lib->Register<cActionReplicateDemesHighestFecundity>("ReplicateDemesHighestFecundity");
action_lib->Register<cActionKillMeanBelowThresholdPaintable>("KillMeanBelowThresholdPaintable");

action_lib->Register<cActionDiffuseHGTGenomeFragments>("DiffuseHGTGenomeFragments");
Expand Down
689 changes: 689 additions & 0 deletions avida-core/tests/demes_ReplicateDemesHighestFecundity/config/avida.cfg

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
#filetype germline_data
#format deme_id hw_type inst_set sequence deme_birth_count deme_parasite_memory_score
# Mode 1 Germline Save
# Tue Aug 29 01:14:50 2023
# 1: Deme ID
# 2: Hardware Type ID
# 3: Inst Set Name
# 4: Genome Sequence
# 5: Deme Birth Count
# 6: Deme Parasite Memory Score

0 2 transsmt ycdBCiEdimFjfCDaknmsAjempczypqvcrGxab 42 24.8
1 2 transsmt ycdBCiEdimFjfCDaknmsAjemEEcgccgssmhEEcsdseDcAcBcggclEEcDEgcvrsAmlzessjhcdcggkhamtmciEEvjDdhjidzoAyndvmEdbgznjDmcjohohooayaxdyalbcekzebjcogEtjgjacblDvubADnslyyocgsAcjCbobffhmvnnAdbDfkmxcagBFfndytqhutjdzfdjsnflfoqCwcvhsjcvbmlsqcjrgyiDivvnFhrArcsmifbClvluDqmCBbtiDhiEfACcarpEczijdljujACbfzuDEFyaqqekizDosbbzjgmpczypqvcrGxab 33 24.8
1 2 transsmt ycdBCiEdimFjfCDaknmsAjemEEcgccgssmhEEcsdseDcAcBcggclEEcDEgcvrsAmlzessjhcdcggkhamtmciEEvjDdhjidzoAyndvmEdbgznjDmcjohohooayaxdyalbcekzebjcogEtjgjacblDvubADnslyyocgsAcjCbobffhmvnnAdbDfkmxcagBFfndytqhutjdzfdjsnflfoqCwcvhsjcvbmlsqcjrgyiDivvnFhrArcsmifbClvluDqmCBbtiDhiEfACcarpEczijdljujACbfzuDEFyaqqekizDosbbzjgmpczypqvcrGxab 33 24.8
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
REACTION NOT not process:value=1.0:type=pow requisite:max_count=1
REACTION NAND nand process:value=1.0:type=pow requisite:max_count=1
REACTION AND and process:value=2.0:type=pow requisite:max_count=1
REACTION ORN orn process:value=2.0:type=pow requisite:max_count=1
REACTION OR or process:value=3.0:type=pow requisite:max_count=1
REACTION ANDN andn process:value=3.0:type=pow requisite:max_count=1
REACTION NOR nor process:value=4.0:type=pow requisite:max_count=1
REACTION XOR xor process:value=4.0:type=pow requisite:max_count=1
REACTION EQU equ process:value=5.0:type=pow requisite:max_count=1
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
u begin InjectAll filename=evolved-not.org
u begin LoadGermlines detailgermlines-500.sgerm
u begin LoadBirthCounts detailgermlines-500.sgerm
u begin SavePopulation
u begin SaveGermlines filename=evolved.sgerm:birthcounts=1
u 2 ReplicateDemesHighestFecundity 1.0 1
u 2 SaveGermlines filename=evolved.sgerm:birthcounts=1
u 2 SavePopulation
u 3 Exit
Loading

0 comments on commit 04a5341

Please sign in to comment.