CGGItemSets/src/cgg_data_source.rs

199 lines
5.9 KiB
Rust
Raw Normal View History

2018-06-16 10:48:46 +02:00
use indexmap::IndexMap;
use lazy_static::lazy_static;
2018-06-16 10:48:46 +02:00
use regex::Regex;
2021-03-13 23:02:26 +01:00
use serde_derive::Deserialize;
use serde_json::{json, Value};
2021-03-14 10:52:27 +01:00
use std::collections::HashMap;
use std::sync::Mutex;
2018-06-16 10:48:46 +02:00
2019-09-26 22:39:25 +02:00
use crate::data_source::DataSource;
2021-03-13 23:02:26 +01:00
use crate::ChampInfo;
2018-06-16 10:48:46 +02:00
2021-03-13 23:02:26 +01:00
#[derive(Deserialize, Debug)]
struct BuildResponse {
lol: Lol
}
#[derive(Deserialize, Debug)]
struct Lol {
#[serde(rename = "championsReport")]
champions_report: Vec<ChampionReport>
}
#[derive(Deserialize, Debug)]
struct ChampionReport {
champion_id: u32,
role: String,
patch: String,
stats: Stats
}
#[derive(Deserialize, Debug)]
struct Stats {
starting_items: Build,
2021-03-14 09:49:18 +01:00
core_builds: Build,
big_item_builds: Build,
skills: Build,
most_common_starting_items: Build,
most_common_core_builds: Build,
most_common_big_item_builds: Build,
most_common_skills: Build
2021-03-13 23:02:26 +01:00
}
#[derive(Deserialize, Debug)]
struct Build {
build: Vec<u32>,
win_rate: f64,
games: u32
}
2021-03-14 10:52:27 +01:00
#[derive(Deserialize, Debug, Clone)]
struct ChampStat {
champion_id: Option<u32>,
stats: Option<Info>,
matchups: Option<Vec<Info>>,
#[serde(rename = "winRate")]
win_rate: Option<f64>,
games: Option<u32>,
kda: Option<f64>,
}
#[derive(Deserialize, Debug, Clone)]
struct Info {
id: String
}
lazy_static! {
2021-03-14 11:01:37 +01:00
static ref CHAMPIONS_REPORT: Mutex<Vec<ChampionReport>> = Mutex::new(Vec::new());
2021-03-14 10:52:27 +01:00
static ref CHAMPIONS_STATS: Mutex<HashMap<String, ChampStat>> = Mutex::new(HashMap::new());
}
2018-06-16 10:48:46 +02:00
pub struct CGGDataSource;
impl CGGDataSource {
2021-03-13 23:02:26 +01:00
fn make_item_set(&self, build: &Build, label: &str) -> Value {
json!({
2021-03-14 09:49:18 +01:00
"items": build.build.iter().map(|x| json!({"id": x.to_string(), "count": 1})).collect::<Vec<Value>>(),
2021-03-13 23:02:26 +01:00
"type": format!("{} ({:.2}% - {} games)", label, build.win_rate * 100., build.games)
})
}
fn make_item_set_from_list(
2021-03-14 09:49:18 +01:00
&self, build: &Build, label: &str, skills: &Build,
) -> Value {
2021-03-14 09:49:18 +01:00
let key_order = skills.build
.iter()
2021-03-14 09:49:18 +01:00
.map(|x| { x.to_string() })
.collect::<Vec<String>>()
.join("");
self.make_item_set(build, [label, &key_order.as_str()].join(" ").as_str())
}
}
2018-06-16 10:48:46 +02:00
impl DataSource for CGGDataSource {
fn get_alias(&self) -> &str {
"CGG"
}
2021-03-14 09:49:18 +01:00
fn get_timeout(&self) -> u64 {
0
}
2018-06-16 10:48:46 +02:00
fn get_champs_with_positions_and_patch(
2018-06-16 12:48:14 +02:00
&self,
2021-03-10 12:22:00 +01:00
client: &ureq::Agent,
2018-06-16 10:48:46 +02:00
) -> (IndexMap<String, Vec<String>>, String) {
2021-03-13 23:02:26 +01:00
let req = client
2018-06-16 10:48:46 +02:00
.get("https://champion.gg")
2021-03-10 12:22:00 +01:00
.call()
2018-06-16 10:48:46 +02:00
.unwrap();
2021-03-13 23:02:26 +01:00
lazy_static! {
static ref RE: Regex =
Regex::new(r"(?m)^\s+window.__PRELOADED_STATE__ = (.*);$").unwrap();
}
2021-03-14 10:52:27 +01:00
lazy_static! {
static ref RE2: Regex =
Regex::new(r"(?m)^\s+window.__FLASH_CMS_APOLLO_STATE__ = (.*);$").unwrap();
}
let page = &req.into_string().unwrap();
2021-03-13 23:02:26 +01:00
let datas: BuildResponse =
2021-03-14 10:52:27 +01:00
serde_json::from_str(&RE.captures(&page).unwrap()[1].replace("undefined", "null")).unwrap();
let champs_stats: HashMap<String, ChampStat> = serde_json::from_str(&RE2.captures(&page).unwrap()[1].replace("undefined", "null")).unwrap();
for entry in champs_stats.iter() {
CHAMPIONS_STATS.lock().unwrap().insert(entry.0.clone(), entry.1.clone());
}
2021-03-13 23:02:26 +01:00
2021-03-14 09:24:09 +01:00
let patch = datas.lol.champions_report[0].patch.clone();
2018-06-16 10:48:46 +02:00
2021-03-14 09:24:09 +01:00
let mut champions: IndexMap<String, Vec<String>> = IndexMap::new();
2021-03-13 23:02:26 +01:00
for champ in &datas.lol.champions_report {
let id = champ.champion_id.to_string();
2021-03-14 09:24:09 +01:00
let mut roles: Vec<String> = Vec::new();
if champions.contains_key(&id) {
let c = champions.get(&id).unwrap();
let mut new_roles = c.clone();
new_roles.push(champ.role.clone());
} else {
roles.push(champ.role.clone());
}
champions.insert(id, roles);
2018-06-16 10:48:46 +02:00
}
2021-03-14 11:01:37 +01:00
for report in datas.lol.champions_report {
CHAMPIONS_REPORT.lock().unwrap().push(report);
2021-03-13 23:02:26 +01:00
}
2018-06-16 10:48:46 +02:00
(champions, patch)
}
fn get_champ_data_with_win_pourcentage(
&self,
2021-03-13 23:02:26 +01:00
champ: &ChampInfo,
position: &str,
2021-03-13 23:02:26 +01:00
_client: &ureq::Agent,
) -> Option<(Vec<Value>, f64)> {
2021-03-13 23:02:26 +01:00
let mut some_champ: Option<&ChampionReport> = None;
2021-03-14 11:01:37 +01:00
let reports = CHAMPIONS_REPORT.lock().unwrap();
for champion in reports.iter() {
if champion.champion_id.to_string() == champ.key && champion.role == position {
some_champ = Some(champion);
}
2021-03-13 23:02:26 +01:00
}
if let Some(champ) = some_champ {
let mut blocks = vec![];
2021-03-14 09:49:18 +01:00
blocks.push(self.make_item_set_from_list(&champ.stats.starting_items, "Highest % Win Starting Items | Skills: ", &champ.stats.skills));
2021-03-13 23:02:26 +01:00
blocks.push(self.make_item_set(&champ.stats.core_builds, "Highest % Win Core Build Path:"));
2021-03-14 09:49:18 +01:00
blocks.push(self.make_item_set(&champ.stats.big_item_builds, "Highest % Win Big Items:"));
blocks.push(self.make_item_set_from_list(&champ.stats.most_common_starting_items, "Most Frequent Starting Items | Skills: ", &champ.stats.most_common_skills));
blocks.push(self.make_item_set(&champ.stats.most_common_core_builds, "Most Frequent Build Path"));
blocks.push(self.make_item_set(&champ.stats.most_common_big_item_builds, "Most Frequent Big Items:"));
2021-03-14 10:52:27 +01:00
let mut key: String = String::new();
let champs_stats = CHAMPIONS_STATS.lock().unwrap();
for val in champs_stats.values() {
if val.champion_id.is_some() && val.champion_id.unwrap() == champ.champion_id {
key = val.stats.as_ref().unwrap().id.clone();
}
}
let win_rate = champs_stats.get(&key).unwrap().win_rate;
return Some((blocks, win_rate.unwrap()));
2018-06-16 10:48:46 +02:00
}
2021-03-13 23:02:26 +01:00
None
2018-06-16 10:48:46 +02:00
}
}