use std::collections::HashMap;
use std::sync::Arc;
use crate::lockorder::RankedMutex;
use super::{
ActivityStore, EpochMs, LastFetchedAt, ProfileActivity, REFRESH_INTERVAL_MS, TokenEntry,
clear_activity, mark_activity, partition_due,
};
fn token(name: &str) -> TokenEntry {
TokenEntry {
name: name.to_string(),
access_token: "access".to_string(),
refresh_token: Some("refresh".to_string()),
}
}
#[test]
fn partition_due_uses_fixed_interval() {
let last_fetched: LastFetchedAt = Arc::new(RankedMutex::new(HashMap::new()));
let activity: ActivityStore = Arc::new(RankedMutex::new(HashMap::new()));
let snapshot = vec![token("a")];
let base = 1_700_000_000_000u64;
let (due, next) = partition_due(&snapshot, base, &last_fetched, &activity);
assert_eq!(due.len(), 1, "a never-fetched profile is due");
assert_eq!(next.get("a").copied(), Some(REFRESH_INTERVAL_MS));
last_fetched
.lock()
.unwrap()
.insert("a".to_string(), EpochMs::from_millis(base));
let (due, next) = partition_due(&snapshot, base + 1, &last_fetched, &activity);
assert!(due.is_empty(), "not due one ms after a fetch");
assert_eq!(next.get("a").copied(), Some(base + REFRESH_INTERVAL_MS));
let (due, _) = partition_due(
&snapshot,
base + REFRESH_INTERVAL_MS,
&last_fetched,
&activity,
);
assert_eq!(due.len(), 1, "due once the fixed interval has elapsed");
}
#[test]
fn partition_due_excludes_switching_and_refreshing() {
let last_fetched: LastFetchedAt = Arc::new(RankedMutex::new(HashMap::new()));
let activity: ActivityStore = Arc::new(RankedMutex::new(HashMap::new()));
let snapshot = vec![token("a"), token("b")];
mark_activity(&activity, "a", ProfileActivity::Switching);
mark_activity(&activity, "b", ProfileActivity::Refreshing);
let (due, next) = partition_due(&snapshot, REFRESH_INTERVAL_MS + 1, &last_fetched, &activity);
assert!(due.is_empty(), "switching/refreshing profiles are excluded");
assert!(
next.contains_key("a") && next.contains_key("b"),
"countdown still publishes for excluded profiles"
);
}
#[test]
fn activity_cleared_on_worker_panic() {
let activity: ActivityStore = Arc::new(RankedMutex::new(HashMap::new()));
let name = "test-profile";
mark_activity(&activity, name, ProfileActivity::Fetching);
assert!(
!activity.lock().unwrap().is_empty(),
"slot must be set after mark_activity"
);
let h = std::thread::spawn(|| -> () { panic!("simulated worker panic") });
match h.join() {
Ok(_) => panic!("expected panic in worker"),
Err(_) => clear_activity(&activity, name),
}
assert!(
activity.lock().unwrap().is_empty(),
"activity slot must be cleared after worker panic"
);
}
#[test]
fn cached_fallback_does_not_clobber_store() {
use super::{FetchOutcome, FetchStatus, StatusStore, apply_outcome};
use crate::usage::{UsageInfo, UsageWindow};
let store: super::UsageStore = Arc::new(RankedMutex::new(HashMap::new()));
let status: StatusStore = Arc::new(RankedMutex::new(HashMap::new()));
let last_fetched: LastFetchedAt = Arc::new(RankedMutex::new(HashMap::new()));
let live = UsageInfo {
five_hour: Some(UsageWindow {
utilization: 1.0,
resets_at: Some("2999-01-01T00:00:00+00:00".to_string()),
}),
..Default::default()
};
store.lock().unwrap().insert("a".to_string(), live);
let stale_windowless = UsageInfo::default();
apply_outcome(
FetchOutcome {
name: "a".to_string(),
info: Some(stale_windowless.clone()),
status: FetchStatus::RateLimited,
rotated: None,
from_fetch: false,
retry_after: None,
},
&store,
&status,
&last_fetched,
);
assert!(
store.lock().unwrap().get("a").unwrap().five_hour.is_some(),
"a cache fallback must not overwrite a newer store entry"
);
assert_eq!(
status.lock().unwrap().get("a").copied(),
Some(FetchStatus::RateLimited),
"the RateLimited status still surfaces"
);
apply_outcome(
FetchOutcome {
name: "b".to_string(),
info: Some(stale_windowless),
status: FetchStatus::Cached,
rotated: None,
from_fetch: false,
retry_after: None,
},
&store,
&status,
&last_fetched,
);
assert!(
store.lock().unwrap().contains_key("b"),
"a cache fallback still cold-fills an absent entry"
);
}
#[test]
fn mark_window_open_synthesizes_only_when_not_live() {
use super::mark_window_open;
use crate::usage::{UsageInfo, UsageWindow, iso_to_epoch_secs};
let store: super::UsageStore = Arc::new(RankedMutex::new(HashMap::new()));
let now = 1_780_000_000i64;
mark_window_open(&store, "a", now);
let resets = store.lock().unwrap()["a"]
.five_hour
.as_ref()
.and_then(|w| w.resets_at.as_deref())
.and_then(iso_to_epoch_secs);
assert_eq!(
resets,
Some(now + 5 * 3600),
"synthetic window opens at +5h"
);
let live_resets = "2999-01-01T00:00:00+00:00";
store.lock().unwrap().insert(
"b".to_string(),
UsageInfo {
five_hour: Some(UsageWindow {
utilization: 42.0,
resets_at: Some(live_resets.to_string()),
}),
..Default::default()
},
);
mark_window_open(&store, "b", now);
let kept = store.lock().unwrap()["b"].five_hour.clone().unwrap();
assert_eq!(kept.resets_at.as_deref(), Some(live_resets));
assert_eq!(kept.utilization, 42.0);
store.lock().unwrap().insert(
"c".to_string(),
UsageInfo {
five_hour: Some(UsageWindow {
utilization: 88.0,
resets_at: Some("2020-01-01T00:00:00+00:00".to_string()),
}),
..Default::default()
},
);
mark_window_open(&store, "c", now);
let replaced = store.lock().unwrap()["c"].five_hour.clone().unwrap();
assert_eq!(
replaced.resets_at.as_deref().and_then(iso_to_epoch_secs),
Some(now + 5 * 3600)
);
assert_eq!(replaced.utilization, 0.0, "fresh window starts at zero");
}
#[test]
fn retry_after_defers_next_fetch_slot() {
use std::time::Duration;
use super::{
FetchOutcome, FetchStatus, MAX_RETRY_AFTER_MS, StatusStore, apply_outcome, now_ms,
};
let store: super::UsageStore = Arc::new(RankedMutex::new(HashMap::new()));
let status: StatusStore = Arc::new(RankedMutex::new(HashMap::new()));
let last_fetched: LastFetchedAt = Arc::new(RankedMutex::new(HashMap::new()));
let outcome = |name: &str, retry_after: Option<Duration>| FetchOutcome {
name: name.to_string(),
info: None,
status: FetchStatus::RateLimited,
rotated: None,
from_fetch: false,
retry_after,
};
let stamp = |name: &str| {
last_fetched
.lock()
.unwrap()
.get(name)
.copied()
.expect("stamp present")
.as_millis()
};
let before = now_ms();
apply_outcome(
outcome("a", Some(Duration::from_secs(300))),
&store,
&status,
&last_fetched,
);
let after = now_ms();
let extra = 300_000 - REFRESH_INTERVAL_MS;
let a = stamp("a");
assert!(
(before + extra..=after + extra).contains(&a),
"deferred stamp must sit retry_after - interval ahead of now"
);
let snapshot = vec![token("a")];
let activity: ActivityStore = Arc::new(RankedMutex::new(HashMap::new()));
let (due, next) = partition_due(
&snapshot,
a + REFRESH_INTERVAL_MS - 1,
&last_fetched,
&activity,
);
assert!(due.is_empty(), "not due before the deferred slot");
assert_eq!(
next.get("a").copied(),
Some(a + REFRESH_INTERVAL_MS),
"countdown publishes the deferred slot"
);
let (due, _) = partition_due(&snapshot, a + REFRESH_INTERVAL_MS, &last_fetched, &activity);
assert_eq!(due.len(), 1, "due once the deferred slot arrives");
let before = now_ms();
apply_outcome(outcome("b", None), &store, &status, &last_fetched);
let after = now_ms();
assert!((before..=after).contains(&stamp("b")));
let before = now_ms();
apply_outcome(
outcome("c", Some(Duration::from_secs(5))),
&store,
&status,
&last_fetched,
);
let after = now_ms();
assert!((before..=after).contains(&stamp("c")));
let before = now_ms();
apply_outcome(
outcome("d", Some(Duration::from_secs(86_400))),
&store,
&status,
&last_fetched,
);
let after = now_ms();
let capped = MAX_RETRY_AFTER_MS - REFRESH_INTERVAL_MS;
assert!(
(before + capped..=after + capped).contains(&stamp("d")),
"huge retry-after clamps to MAX_RETRY_AFTER_MS"
);
}