Score refactor, add scores to feed_channels

This commit is contained in:
Julia Lange 2026-02-03 17:51:09 -08:00
parent 91229287a8
commit 25f00d1665
Signed by: Julia
SSH key fingerprint: SHA256:5DJcfxa5/fKCYn57dcabJa2vN2e6eT0pBerYi5SUbto
4 changed files with 166 additions and 78 deletions

View file

@ -51,6 +51,8 @@ CREATE TABLE feed_items (
item_id INTEGER NOT NULL, item_id INTEGER NOT NULL,
feed_id INTEGER NOT NULL, feed_id INTEGER NOT NULL,
score INTEGER NOT NULL, score INTEGER NOT NULL,
last_updated TEXT NOT NULL,
boosted_at TEXT,
archived BOOLEAN DEFAULT FALSE, archived BOOLEAN DEFAULT FALSE,
PRIMARY KEY (item_id, feed_id), PRIMARY KEY (item_id, feed_id),
FOREIGN KEY (feed_id) REFERENCES feeds(id), FOREIGN KEY (feed_id) REFERENCES feeds(id),

View file

@ -115,7 +115,7 @@ impl Channel {
) -> Result<Vec<FeedChannel>> { ) -> Result<Vec<FeedChannel>> {
let feeds: Result<Vec<FeedChannel>> = sqlx::query_as!( let feeds: Result<Vec<FeedChannel>> = sqlx::query_as!(
UnparsedFeedChannel, UnparsedFeedChannel,
"SELECT channel_id, feed_id "SELECT channel_id, feed_id, initial_score, gravity, boost
FROM feed_channels FROM feed_channels
WHERE channel_id = ?", WHERE channel_id = ?",
self.id.0 self.id.0

View file

@ -8,17 +8,29 @@ use crate::{
FeedId, FeedId,
Item, Item,
}, },
score::{
Score,
Gravity,
Boost,
},
}; };
use chrono::{Utc, DateTime};
pub struct UnparsedFeedChannel { pub struct UnparsedFeedChannel {
pub channel_id: i64, pub channel_id: i64,
pub feed_id: i64, pub feed_id: i64,
pub initial_score: Option<i64>,
pub gravity: Option<i64>,
pub boost: Option<i64>,
} }
impl UnparsedFeedChannel { impl UnparsedFeedChannel {
pub fn parse(self) -> Result<FeedChannel> { pub fn parse(self) -> Result<FeedChannel> {
Ok(FeedChannel { Ok(FeedChannel {
channel_id: ChannelId(self.channel_id), channel_id: ChannelId(self.channel_id),
feed_id: FeedId(self.feed_id) feed_id: FeedId(self.feed_id),
initial_score: Score::new(self.initial_score),
gravity: Gravity::new(self.gravity),
boost: Boost::new(self.boost),
}) })
} }
} }
@ -26,6 +38,9 @@ impl UnparsedFeedChannel {
pub struct FeedChannel { pub struct FeedChannel {
channel_id: ChannelId, channel_id: ChannelId,
feed_id: FeedId, feed_id: FeedId,
initial_score: Score,
gravity: Gravity,
boost: Boost,
} }
impl FeedChannel { impl FeedChannel {
@ -38,17 +53,26 @@ impl FeedChannel {
pub async fn add_item( pub async fn add_item(
&self, pool: &AdapterPool, item: &Item &self, pool: &AdapterPool, item: &Item
) -> Result<()> {
self.add_item_at(pool, item, Utc::now()).await
}
async fn add_item_at(
&self, pool: &AdapterPool, item: &Item, add_at: DateTime<Utc>
) -> Result<()> { ) -> Result<()> {
let int_item_id = i64::from(item.id()); let int_item_id = i64::from(item.id());
let int_feed_id = i64::from(self.feed_id); let int_feed_id = i64::from(self.feed_id);
let int_initial_score = i64::from(self.initial_score);
let string_last_updated = add_at.to_rfc2822();
sqlx::query!( sqlx::query!(
"INSERT OR IGNORE INTO feed_items (feed_id, item_id, score) "INSERT OR IGNORE INTO feed_items (feed_id, item_id, score, last_updated)
VALUES (?, ?, 5)", // TODO: Add in scoring featuress VALUES (?, ?, ?, ?)",
int_feed_id, int_item_id int_feed_id, int_item_id, int_initial_score, string_last_updated
).execute(&pool.0).await?; ).execute(&pool.0).await?;
Ok(()) Ok(())
} }
} }
#[cfg(test)] #[cfg(test)]
@ -62,23 +86,32 @@ mod tests {
User User
}, },
test_utils::{ test_utils::{
FEED1, setup_adapter, FEED1, setup_adapter, get_datetime
}, },
}; };
#[test] #[test]
fn parse() { fn parse() {
const CID: i64 = 1; const CID: i64 = 1;
const FID: i64 = 1; const FID: i64 = 2;
const IS: i64 = 3;
const G: i64 = 4;
const B: i64 = 5;
let ufc = UnparsedFeedChannel { let ufc = UnparsedFeedChannel {
channel_id: CID, channel_id: CID,
feed_id: FID, feed_id: FID,
initial_score: Some(IS),
gravity: Some(G),
boost: Some(B),
}; };
let fc = ufc.parse().unwrap(); let fc = ufc.parse().unwrap();
assert_eq!(fc.channel_id.0, CID); assert_eq!(fc.channel_id.0, CID);
assert_eq!(fc.feed_id.0, FID); assert_eq!(fc.feed_id.0, FID);
assert_eq!(i64::from(fc.initial_score), IS);
assert_eq!(i64::from(fc.gravity), G);
assert_eq!(i64::from(fc.boost), B);
} }
// FeedChannel Tests // FeedChannel Tests
@ -93,6 +126,9 @@ mod tests {
let fc = FeedChannel { let fc = FeedChannel {
channel_id: channel.id(), channel_id: channel.id(),
feed_id: FeedId(1), // Fake Feed feed_id: FeedId(1), // Fake Feed
initial_score: Score::new(None),
gravity: Gravity::new(None),
boost: Boost::new(None),
}; };
let channel_from_fc = fc.get_channel(pool).await.unwrap(); let channel_from_fc = fc.get_channel(pool).await.unwrap();
@ -110,6 +146,9 @@ mod tests {
let fc = FeedChannel { let fc = FeedChannel {
channel_id: ChannelId(1), // Fake Channel channel_id: ChannelId(1), // Fake Channel
feed_id: feed.id(), feed_id: feed.id(),
initial_score: Score::new(None),
gravity: Gravity::new(None),
boost: Boost::new(None),
}; };
let feed_from_fc = fc.get_feed(pool).await.unwrap(); let feed_from_fc = fc.get_feed(pool).await.unwrap();
@ -118,6 +157,7 @@ mod tests {
#[tokio::test] #[tokio::test]
pub async fn add_item() { pub async fn add_item() {
let dt = get_datetime();
let adapter = setup_adapter().await; let adapter = setup_adapter().await;
let pool = adapter.get_pool(); let pool = adapter.get_pool();
@ -128,10 +168,13 @@ mod tests {
let fc = FeedChannel { let fc = FeedChannel {
channel_id: channel.id(), channel_id: channel.id(),
feed_id: feed.id(), feed_id: feed.id(),
initial_score: Score::new(None),
gravity: Gravity::new(None),
boost: Boost::new(None),
}; };
let item = Item::get_or_create(pool, channel.id(), "item-guid").await.unwrap(); let item = Item::get_or_create(pool, channel.id(), "item-guid").await.unwrap();
fc.add_item(pool, &item).await.unwrap(); fc.add_item_at(pool, &item, dt).await.unwrap();
let items = feed.get_items(pool, 1, 0).await.unwrap(); let items = feed.get_items(pool, 1, 0).await.unwrap();
assert_eq!(items[0].id(), item.id()); assert_eq!(items[0].id(), item.id());

View file

@ -1,5 +1,6 @@
use chrono::{DateTime, Utc, TimeDelta}; use chrono::{DateTime, Utc, TimeDelta};
use crate::{Result}; use crate::{Result};
use std::ops::{Add, Sub};
mod default { mod default {
use crate::score::SECONDS_IN_A_DAY; use crate::score::SECONDS_IN_A_DAY;
@ -12,10 +13,16 @@ mod default {
const SECONDS_IN_A_DAY: i64 = 60 * 60 * 24; const SECONDS_IN_A_DAY: i64 = 60 * 60 * 24;
macro_rules! rich_i64 { macro_rules! rich_i64 {
($name:ident, $default:expr) => { ($name:ident) => {
#[derive(PartialEq, Debug, Copy, Clone)] #[derive(PartialOrd, PartialEq, Debug, Copy, Clone)]
pub struct $name(i64); pub struct $name(i64);
impl From<$name> for i64 { fn from(id: $name) -> Self { id.0 } } impl From<$name> for i64 { fn from(id: $name) -> Self { id.0 } }
};
}
macro_rules! defaulting_i64 {
($name:ident, $default:expr) => {
rich_i64!($name);
impl $name { impl $name {
pub fn new(value: Option<i64>) -> Self { pub fn new(value: Option<i64>) -> Self {
Self(value.unwrap_or($default)) Self(value.unwrap_or($default))
@ -24,45 +31,73 @@ macro_rules! rich_i64 {
}; };
} }
rich_i64!(Gravity, default::GRAVITY); macro_rules! addable_i64s {
rich_i64!(Boost, default::BOOST); ($lhs:ident, $rhs:ident) => {
pub struct UnparsedScore { impl Add<$rhs> for $lhs {
type Output = Self;
fn add(self, other: $rhs) -> Self::Output { Self(self.0 + other.0) }
}
}
}
defaulting_i64!(Score, default::INITIAL_SCORE);
addable_i64s!(Score, Score);
addable_i64s!(Score, Boost);
impl Sub<Boost> for Score {
type Output = Self;
fn sub(self, other: Boost) -> Self::Output { Self(self.0 - other.0) }
}
addable_i64s!(Score, GravityOverDuration);
defaulting_i64!(Boost, default::BOOST);
defaulting_i64!(Gravity, default::GRAVITY);
rich_i64!(GravityOverDuration);
impl Gravity {
fn over_duration(
&self, start: DateTime<Utc>, end: DateTime<Utc>
) -> GravityOverDuration {
let elapsed_time = end.signed_duration_since(start);
GravityOverDuration(
self.0 * (elapsed_time.num_seconds() / SECONDS_IN_A_DAY)
)
}
}
pub struct UnparsedTimedScore {
pub value: i64, pub value: i64,
pub last_updated: DateTime<Utc>, pub last_updated: DateTime<Utc>,
pub last_boosted: Option<DateTime<Utc>>, pub last_boosted: Option<DateTime<Utc>>,
} }
impl UnparsedScore { impl UnparsedTimedScore {
pub fn parse(self) -> Score { pub fn parse(self) -> TimedScore {
match self.last_boosted { match self.last_boosted {
None => Score::Decaying(DecayingScore { None => TimedScore::Decaying(DecayingScore {
value: self.value, value: Score(self.value),
last_updated: self.last_updated, last_updated: self.last_updated,
}), }),
Some(last_boosted) => Score::Boosted(BoostedScore { Some(last_boosted) => TimedScore::Boosted(BoostedScore {
value: self.value, value: Score(self.value),
boosted_at: last_boosted, boosted_at: last_boosted,
}), }),
} }
} }
} }
pub enum Score { pub enum TimedScore {
Decaying(DecayingScore), Decaying(DecayingScore),
Boosted(BoostedScore), Boosted(BoostedScore),
} }
impl Score { impl TimedScore {
pub fn new() -> DecayingScore { pub fn new() -> DecayingScore {
Self::new_with_initial(default::INITIAL_SCORE) Self::new_with_initial(Score::new(None))
} }
pub fn new_with_initial(initial_score: i64) -> DecayingScore { pub fn new_with_initial(initial_score: Score) -> DecayingScore {
Self::new_with_initial_and_time(initial_score, Utc::now()) Self::new_with_initial_and_time(initial_score, Utc::now())
} }
pub fn new_with_initial_and_time( pub fn new_with_initial_and_time(
initial: i64, time: DateTime<Utc> initial: Score, time: DateTime<Utc>
) -> DecayingScore { ) -> DecayingScore {
DecayingScore { DecayingScore {
value: initial, value: initial,
@ -70,7 +105,7 @@ impl Score {
} }
} }
pub fn get_score(&self) -> i64 { pub fn get_score(&self) -> Score {
match self { match self {
Self::Decaying(s) => s.get_score(), Self::Decaying(s) => s.get_score(),
Self::Boosted(b) => b.get_score(), Self::Boosted(b) => b.get_score(),
@ -83,16 +118,16 @@ impl Score {
fn update_score_at_time(self, gravity: Gravity, time: DateTime<Utc>) -> Self { fn update_score_at_time(self, gravity: Gravity, time: DateTime<Utc>) -> Self {
match self { match self {
Self::Decaying(d) => Score::Decaying( Self::Decaying(d) => TimedScore::Decaying(
d.apply_gravity_to_time(gravity, time) d.apply_gravity_to_time(gravity, time)
), ),
Self::Boosted(b) => { Self::Boosted(b) => {
let try_unfrozen = b.try_unfreeze_at_time(time); let try_unfrozen = b.try_unfreeze_at_time(time);
match try_unfrozen { match try_unfrozen {
Self::Decaying(s) => Score::Decaying( Self::Decaying(s) => TimedScore::Decaying(
s.apply_gravity_to_time(gravity, time) s.apply_gravity_to_time(gravity, time)
), ),
Self::Boosted(b) => Score::Boosted(b), Self::Boosted(b) => TimedScore::Boosted(b),
} }
} }
} }
@ -114,30 +149,26 @@ impl Score {
} }
pub struct BoostedScore { pub struct BoostedScore {
value: i64, value: Score,
boosted_at: DateTime<Utc>, boosted_at: DateTime<Utc>,
} }
pub struct DecayingScore { pub struct DecayingScore {
value: i64, value: Score,
last_updated: DateTime<Utc>, last_updated: DateTime<Utc>,
} }
impl DecayingScore { impl DecayingScore {
fn get_score(&self) -> i64 { fn get_score(&self) -> Score {
self.value self.value
} }
fn apply_gravity_to_time( fn apply_gravity_to_time(
self, gravity: Gravity, update_time: DateTime<Utc> self, gravity: Gravity, update_time: DateTime<Utc>
) -> Self { ) -> Self {
let elapsed_time: TimeDelta = update_time.signed_duration_since(self.last_updated);
let new_value = if elapsed_time <= TimeDelta::zero() { self.value } else {
self.value + i64::from(gravity) * (elapsed_time.num_seconds() / SECONDS_IN_A_DAY)
};
Self { Self {
last_updated: update_time, last_updated: update_time,
value: new_value, value: self.value + gravity.over_duration(self.last_updated, update_time),
} }
} }
@ -147,33 +178,33 @@ impl DecayingScore {
fn boost_at_time(self, boost: Boost, boost_time: DateTime<Utc>) -> BoostedScore { fn boost_at_time(self, boost: Boost, boost_time: DateTime<Utc>) -> BoostedScore {
BoostedScore { BoostedScore {
value: self.value + i64::from(boost), value: self.value + boost,
boosted_at: boost_time, boosted_at: boost_time,
} }
} }
} }
impl BoostedScore { impl BoostedScore {
fn get_score(&self) -> i64 { fn get_score(&self) -> Score {
self.value self.value
} }
pub fn unboost(self, boost: Boost) -> DecayingScore { pub fn unboost(self, boost: Boost) -> DecayingScore {
DecayingScore { DecayingScore {
value: self.value - i64::from(boost), value: self.value - boost,
last_updated: self.boosted_at, last_updated: self.boosted_at,
} }
} }
fn try_unfreeze_at_time(self, update_time: DateTime<Utc>) -> Score { fn try_unfreeze_at_time(self, update_time: DateTime<Utc>) -> TimedScore {
let boost_end = self.boosted_at + TimeDelta::seconds(default::BOOST_FREEZE_IN_SECONDS); let boost_end = self.boosted_at + TimeDelta::seconds(default::BOOST_FREEZE_IN_SECONDS);
if boost_end < update_time { if boost_end < update_time {
Score::Decaying(DecayingScore { TimedScore::Decaying(DecayingScore {
value: self.value, value: self.value,
last_updated: boost_end, last_updated: boost_end,
}) })
} else { } else {
Score::Boosted(self) TimedScore::Boosted(self)
} }
} }
} }
@ -199,7 +230,7 @@ mod tests {
#[test] #[test]
fn parse_decaying() { fn parse_decaying() {
let ups = UnparsedScore { let ups = UnparsedTimedScore {
value: 10, value: 10,
last_updated: get_datetime(), last_updated: get_datetime(),
last_boosted: None, last_boosted: None,
@ -211,7 +242,7 @@ mod tests {
#[test] #[test]
fn parse_boosted() { fn parse_boosted() {
let dt = get_datetime(); let dt = get_datetime();
let ups = UnparsedScore { let ups = UnparsedTimedScore {
value: 10, value: 10,
last_updated: dt, last_updated: dt,
last_boosted: Some(dt), last_boosted: Some(dt),
@ -222,22 +253,24 @@ mod tests {
#[test] #[test]
fn new() { fn new() {
let score = Score::new(); let score = TimedScore::new();
assert_eq!(score.value, default::INITIAL_SCORE); assert_eq!(score.value, Score(default::INITIAL_SCORE));
} }
#[test] #[test]
fn new_with_values() { fn new_with_values() {
let dt = get_datetime(); let dt = get_datetime();
let score = Score::new_with_initial_and_time(10, dt); let score = TimedScore::new_with_initial_and_time(Score(10), dt);
assert_eq!(score.value, 10); assert_eq!(score.value, Score(10));
assert_eq!(score.last_updated, dt); assert_eq!(score.last_updated, dt);
} }
#[test] #[test]
fn update_score_stays_decaying() { fn update_score_stays_decaying() {
let dt = get_datetime(); let dt = get_datetime();
let score = Score::Decaying(Score::new_with_initial_and_time(10, dt)); let score = TimedScore::Decaying(
TimedScore::new_with_initial_and_time(Score(10), dt)
);
let gravity = Gravity::new(None); let gravity = Gravity::new(None);
let dt2 = dt + TimeDelta::seconds(SECONDS_IN_A_DAY); let dt2 = dt + TimeDelta::seconds(SECONDS_IN_A_DAY);
@ -248,7 +281,9 @@ mod tests {
#[test] #[test]
fn update_score_stays_frozen() { fn update_score_stays_frozen() {
let dt = get_datetime(); let dt = get_datetime();
let score = Score::Boosted(BoostedScore { value: 10, boosted_at: dt }); let score = TimedScore::Boosted(
BoostedScore { value: Score(10), boosted_at: dt }
);
let gravity = Gravity::new(None); let gravity = Gravity::new(None);
let dt2 = dt + TimeDelta::seconds(default::BOOST_FREEZE_IN_SECONDS); let dt2 = dt + TimeDelta::seconds(default::BOOST_FREEZE_IN_SECONDS);
@ -259,7 +294,9 @@ mod tests {
#[test] #[test]
fn update_score_thaws_and_decays() { fn update_score_thaws_and_decays() {
let dt = get_datetime(); let dt = get_datetime();
let score = Score::Boosted(BoostedScore { value: 10, boosted_at: dt }); let score = TimedScore::Boosted(
BoostedScore { value: Score(10), boosted_at: dt }
);
let gravity = Gravity::new(None); let gravity = Gravity::new(None);
let dt2 = dt + TimeDelta::seconds( let dt2 = dt + TimeDelta::seconds(
@ -268,13 +305,15 @@ mod tests {
let updated = score.update_score_at_time(gravity, dt2) let updated = score.update_score_at_time(gravity, dt2)
.get_decaying().unwrap(); .get_decaying().unwrap();
assert!(updated.value < 10) assert!(updated.value < Score(10))
} }
#[test] #[test]
fn get_decaying_success() { fn get_decaying_success() {
let dt = get_datetime(); let dt = get_datetime();
let score = Score::Decaying(Score::new_with_initial_and_time(10, dt)); let score = TimedScore::Decaying(
TimedScore::new_with_initial_and_time(Score(10), dt)
);
score.get_decaying().unwrap(); score.get_decaying().unwrap();
} }
@ -283,7 +322,9 @@ mod tests {
#[should_panic = "Attempted to get_boosted() of a decaying score"] #[should_panic = "Attempted to get_boosted() of a decaying score"]
fn get_boosted_failure() { fn get_boosted_failure() {
let dt = get_datetime(); let dt = get_datetime();
let score = Score::Decaying(Score::new_with_initial_and_time(10, dt)); let score = TimedScore::Decaying(
TimedScore::new_with_initial_and_time(Score(10), dt)
);
score.get_boosted().unwrap(); score.get_boosted().unwrap();
} }
@ -293,8 +334,9 @@ mod tests {
fn get_decaying_failure() { fn get_decaying_failure() {
let dt = get_datetime(); let dt = get_datetime();
let boost = Boost::new(None); let boost = Boost::new(None);
let score = Score::Boosted( let score = TimedScore::Boosted(
Score::new_with_initial_and_time(10, dt).boost_at_time(boost, dt) TimedScore::new_with_initial_and_time(Score(10), dt)
.boost_at_time(boost, dt)
); );
score.get_decaying().unwrap(); score.get_decaying().unwrap();
@ -304,8 +346,9 @@ mod tests {
fn get_boosted_success() { fn get_boosted_success() {
let dt = get_datetime(); let dt = get_datetime();
let boost = Boost::new(None); let boost = Boost::new(None);
let score = Score::Boosted( let score = TimedScore::Boosted(
Score::new_with_initial_and_time(10, dt).boost_at_time(boost, dt) TimedScore::new_with_initial_and_time(Score(10), dt)
.boost_at_time(boost, dt)
); );
score.get_boosted().unwrap(); score.get_boosted().unwrap();
@ -316,37 +359,37 @@ mod tests {
#[test] #[test]
fn apply_gravity_to_future() { fn apply_gravity_to_future() {
let dt = get_datetime(); let dt = get_datetime();
let score = DecayingScore { value: 10, last_updated: dt }; let score = DecayingScore { value: Score(10), last_updated: dt };
let future = dt + TimeDelta::seconds(SECONDS_IN_A_DAY); let future = dt + TimeDelta::seconds(SECONDS_IN_A_DAY);
let gravity = Gravity::new(None); let gravity = Gravity::new(None);
let updated = score.apply_gravity_to_time(gravity, future); let updated = score.apply_gravity_to_time(gravity, future);
assert_eq!(updated.value, 10 + default::GRAVITY); assert!(updated.value < Score(10));
assert_eq!(updated.last_updated, future); assert_eq!(updated.last_updated, future);
} }
#[test] #[test]
fn apply_gravity_to_past() { fn apply_gravity_to_past() {
let dt = get_datetime(); let dt = get_datetime();
let score = DecayingScore { value: 10, last_updated: dt }; let score = DecayingScore { value: Score(10), last_updated: dt };
let past = dt - TimeDelta::seconds(SECONDS_IN_A_DAY); let past = dt - TimeDelta::seconds(SECONDS_IN_A_DAY);
let gravity = Gravity::new(None); let gravity = Gravity::new(None);
let updated = score.apply_gravity_to_time(gravity, past); let updated = score.apply_gravity_to_time(gravity, past);
assert_eq!(updated.value, 10); assert!(updated.value > Score(10));
assert_eq!(updated.last_updated, past); assert_eq!(updated.last_updated, past);
} }
#[test] #[test]
fn boost() { fn boost() {
let dt = get_datetime(); let dt = get_datetime();
let score = DecayingScore { value: 10, last_updated: dt }; let score = DecayingScore { value: Score(10), last_updated: dt };
let boost = Boost::new(None); let boost = Boost::new(None);
let boosted = score.boost_at_time(boost, dt); let boosted = score.boost_at_time(boost, dt);
assert_eq!(boosted.value, 10 + default::BOOST); assert_eq!(boosted.value, Score(10) + Boost(default::BOOST));
assert_eq!(boosted.boosted_at, dt); assert_eq!(boosted.boosted_at, dt);
} }
@ -355,19 +398,19 @@ mod tests {
#[test] #[test]
fn unboost() { fn unboost() {
let dt = get_datetime(); let dt = get_datetime();
let score = DecayingScore { value: 10, last_updated: dt }; let score = DecayingScore { value: Score(10), last_updated: dt };
let boost = Boost::new(None); let boost = Boost::new(None);
let boosted = score.boost_at_time(boost, dt); let boosted = score.boost_at_time(boost, dt);
let unboosted = boosted.unboost(boost); let unboosted = boosted.unboost(boost);
assert_eq!(unboosted.value, 10); assert_eq!(unboosted.value, Score(10));
assert_eq!(unboosted.last_updated, dt); assert_eq!(unboosted.last_updated, dt);
} }
#[test] #[test]
fn boosted_stays_frozen() { fn boosted_stays_frozen() {
let dt = get_datetime(); let dt = get_datetime();
let score = BoostedScore { value: 10, boosted_at: dt }; let score = BoostedScore { value: Score(10), boosted_at: dt };
let last_second = dt + TimeDelta::seconds(default::BOOST_FREEZE_IN_SECONDS); let last_second = dt + TimeDelta::seconds(default::BOOST_FREEZE_IN_SECONDS);
@ -377,7 +420,7 @@ mod tests {
#[test] #[test]
fn boosted_thaws() { fn boosted_thaws() {
let dt = get_datetime(); let dt = get_datetime();
let score = BoostedScore { value: 10, boosted_at: dt }; let score = BoostedScore { value: Score(10), boosted_at: dt };
let first_second = dt + TimeDelta::days(default::BOOST_FREEZE_IN_SECONDS+1); let first_second = dt + TimeDelta::days(default::BOOST_FREEZE_IN_SECONDS+1);