Skip to content

Commit

Permalink
Fix/memorization calculation and due date handling in simulation (#265)
Browse files Browse the repository at this point in the history
Co-authored-by: Luc Mcgrady <[email protected]>
  • Loading branch information
L-M-Sherlock and Luc-Mcgrady authored Dec 25, 2024
1 parent 7da138a commit 937d73b
Show file tree
Hide file tree
Showing 3 changed files with 28 additions and 33 deletions.
2 changes: 1 addition & 1 deletion Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[package]
name = "fsrs"
version = "1.4.8"
version = "1.4.9"
authors = ["Open Spaced Repetition"]
categories = ["algorithms", "science"]
edition = "2021"
Expand Down
57 changes: 26 additions & 31 deletions src/optimal_retention.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,6 @@ use rand::Rng;
use rand::{distributions::WeightedIndex, rngs::StdRng, SeedableRng};
use rayon::iter::IntoParallelIterator;
use rayon::iter::ParallelIterator;
use std::cmp::min;
use std::collections::HashMap;

trait Round {
Expand Down Expand Up @@ -192,15 +191,6 @@ pub fn simulate(
.into_iter()
.filter(|card| card.stability > 1e-9),
);

for card in &cards {
let upper = min(card.due as usize, learn_span);
let elapsed_days = -card.last_date;
for i in 0..upper {
memorized_cnt_per_day[i] +=
power_forgetting_curve(elapsed_days + i as f32, card.stability);
}
}
}

if learn_limit > 0 {
Expand Down Expand Up @@ -233,16 +223,26 @@ pub fn simulate(

let is_learn = card.last_date == f32::NEG_INFINITY;

let last_date_index = card.last_date as usize;

// Guards
if card.due >= learn_span as f32 {
if !is_learn {
let delta_t = learn_span.max(last_date_index) - last_date_index;
let pre_sim_days = (-card.last_date) as usize;
for i in 0..delta_t {
memorized_cnt_per_day[last_date_index + i] +=
power_forgetting_curve((pre_sim_days + i) as f32, card.stability);
}
}
card_priorities.pop();
continue;
}
if (!is_learn && review_cnt_per_day[day_index] + 1 > review_limit)
|| (is_learn && learn_cnt_per_day[day_index] + 1 > learn_limit)
|| (cost_per_day[day_index] + fail_cost > max_cost_perday)
{
card.due += 1.;
card.due = day_index as f32 + 1.0;
card_priorities.change_priority(&card_index, card_priority(card, is_learn));
continue;
}
Expand Down Expand Up @@ -270,10 +270,11 @@ pub fn simulate(
} else {
// For review cards
// Updating delta_t for 'has_learned' cards
let delta_t = card.due - card.last_date;
let elapsed_days = card.due - card.last_date;
let last_stability = card.stability;

// Calculate retrievability for entries where has_learned is true
let retrievability = power_forgetting_curve(delta_t, card.stability);
let retrievability = power_forgetting_curve(elapsed_days, card.stability);

// Create 'forget' mask
let forget = !rng.gen_bool(retrievability as f64);
Expand All @@ -290,10 +291,10 @@ pub fn simulate(
// Update stability
card.stability = if forget {
let post_lapse_stab =
stability_after_failure(w, card.stability, retrievability, card.difficulty);
stability_after_failure(w, last_stability, retrievability, card.difficulty);
stability_short_term(w, post_lapse_stab, forget_rating_offset, forget_session_len)
} else {
stability_after_success(w, card.stability, retrievability, card.difficulty, rating)
stability_after_success(w, last_stability, retrievability, card.difficulty, rating)
};

// Update difficulty for review cards
Expand All @@ -315,23 +316,17 @@ pub fn simulate(
// Update days statistics
review_cnt_per_day[day_index] += 1;
cost_per_day[day_index] += cost;
}

let upper = min(ivl as usize, learn_span - day_index);
for i in 0..upper {
memorized_cnt_per_day[day_index + i] +=
power_forgetting_curve(i as f32, card.stability);
let delta_t = day_index - last_date_index;
let pre_sim_days = (-card.last_date) as usize;
for i in 0..delta_t {
memorized_cnt_per_day[last_date_index + i] +=
power_forgetting_curve((pre_sim_days + i) as f32, last_stability);
}
}

// dbg!(ivl);

// Update 'cost' based on 'forget' and 'rating'

// dbg!(&review_cnt_per_day);

// +1 because the day index is one less than the actual day as today is not graphed.
card.last_date = card.due;
card.due += ivl;
card.last_date = day_index as f32;
card.due = day_index as f32 + ivl;

card_priorities.change_priority(&card_index, card_priority(card, false));
}
Expand Down Expand Up @@ -913,7 +908,7 @@ mod tests {
simulate(&config, &DEFAULT_PARAMETERS, 0.9, None, None)?;
assert_eq!(
memorized_cnt_per_day[memorized_cnt_per_day.len() - 1],
6781.4946
6781.493
);
Ok(())
}
Expand Down Expand Up @@ -1080,7 +1075,7 @@ mod tests {
..Default::default()
};
let results = simulate(&config, &DEFAULT_PARAMETERS, 0.9, None, None)?;
assert_eq!(results.0[results.0.len() - 1], 6484.7144);
assert_eq!(results.0[results.0.len() - 1], 6487.4004);
Ok(())
}

Expand Down

0 comments on commit 937d73b

Please sign in to comment.