use super::pagination::Bound;
use crate::util::{encode, validate_limit};
#[derive(Clone, Debug)]
pub enum LeaderboardType {
League,
Xp,
Ar,
}
impl LeaderboardType {
pub(crate) fn to_param(&self) -> String {
match self {
LeaderboardType::League => "league".to_string(),
LeaderboardType::Xp => "xp".to_string(),
LeaderboardType::Ar => "ar".to_string(),
}
}
}
#[derive(Clone, Debug, Default)]
pub struct SearchCriteria {
pub bound: Option<Bound>,
pub limit: Option<u8>,
pub country: Option<String>,
}
impl SearchCriteria {
pub fn new() -> Self {
Self::default()
}
pub fn init(&mut self) {
self.bound = None;
self.limit = None;
self.country = None;
}
pub fn after(self, bound: [f64; 3]) -> Self {
Self {
bound: Some(Bound::After(bound)),
..self
}
}
pub fn before(self, bound: [f64; 3]) -> Self {
Self {
bound: Some(Bound::Before(bound)),
..self
}
}
pub fn limit(self, limit: u8) -> Self {
validate_limit(limit);
Self {
limit: Some(limit),
..self
}
}
pub fn country(self, country: &str) -> Self {
Self {
country: Some(country.to_owned()),
..self
}
}
pub(crate) fn validate_limit(&self) {
if let Some(self_limit) = self.limit {
validate_limit(self_limit)
}
}
pub(crate) fn build(self) -> Vec<(String, String)> {
let mut result = Vec::new();
if let Some(b) = self.bound {
result.push(b.to_query_param());
}
if let Some(l) = self.limit {
result.push(("limit".to_string(), l.to_string()));
}
if let Some(c) = self.country {
result.push(("country".to_string(), encode(c.to_uppercase())));
}
result
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn leaderboard_type_to_param_converts_into_param_str() {
assert_eq!(LeaderboardType::League.to_param(), "league");
assert_eq!(LeaderboardType::Xp.to_param(), "xp");
assert_eq!(LeaderboardType::Ar.to_param(), "ar");
}
#[test]
fn search_criteria_new_creates_default() {
let criteria = SearchCriteria::new();
assert!(criteria.bound.is_none());
assert!(criteria.limit.is_none());
assert!(criteria.country.is_none());
}
#[test]
fn search_criteria_init_initializes() {
let mut criteria = SearchCriteria::new()
.after([15200.0, 0.0, 0.0])
.limit(3)
.country("jp");
criteria.init();
assert!(criteria.bound.is_none());
assert!(criteria.limit.is_none());
assert!(criteria.country.is_none());
}
#[test]
fn search_criteria_after_sets_upper_bound() {
let criteria = SearchCriteria::new().after([15200.0, 0.0, 0.0]);
assert!(matches!(
criteria.bound,
Some(Bound::After([15200.0, 0.0, 0.0]))
));
}
#[test]
fn search_criteria_before_sets_lower_bound() {
let criteria = SearchCriteria::new().before([15200.0, 0.0, 0.0]);
assert!(matches!(
criteria.bound,
Some(Bound::Before([15200.0, 0.0, 0.0]))
));
}
#[test]
fn search_criteria_limit_sets_valid_limit() {
for i in 1..=100 {
let criteria = SearchCriteria::new().limit(i);
assert_eq!(criteria.limit, Some(i));
}
}
#[test]
#[should_panic]
fn search_criteria_limit_panics_if_out_of_range() {
SearchCriteria::new().limit(0);
SearchCriteria::new().limit(101);
}
#[test]
fn search_criteria_country_sets_country() {
let criteria = SearchCriteria::new().country("jp");
assert_eq!(criteria.country, Some("jp".to_string()));
}
#[test]
#[should_panic]
fn search_criteria_validate_limit_panics_if_out_of_range() {
SearchCriteria {
limit: Some(0),
..SearchCriteria::default()
}
.validate_limit();
SearchCriteria {
limit: Some(101),
..SearchCriteria::default()
}
.validate_limit();
}
#[test]
fn search_criteria_build_creates_query_params() {
let criteria = SearchCriteria::new()
.after([15200.0, 0.0, 0.0])
.limit(3)
.country("JP");
let query_params = criteria.build();
assert_eq!(
query_params,
vec![
("after".to_string(), "15200:0:0".to_string()),
("limit".to_string(), "3".to_string()),
("country".to_string(), "JP".to_string())
]
);
}
#[test]
fn search_criteria_build_returns_empty_vec_if_no_params() {
let criteria = SearchCriteria::new();
let query_params = criteria.build();
assert!(query_params.is_empty());
}
}