ew/src/router/items.rs

529 lines
19 KiB
Rust

use json::{array, object, JsonValue};
use rand::Rng;
use actix_web::{HttpRequest, http::header::{HeaderMap, HeaderValue}};
use crate::encryption;
use crate::router::{userdata, global, databases};
pub fn remove_gems(user: &mut JsonValue, amount: i64) {
let mut amount = amount;
let mut free = user["gem"]["free"].as_i64().unwrap();
let mut paid = user["gem"]["charge"].as_i64().unwrap();
free -= amount;
if free < 0 {
amount = -free;
free = 0;
}
paid -= amount;
if paid < 0 {
paid = 0;
}
user["gem"]["free"] = free.into();
user["gem"]["charge"] = paid.into();
user["gem"]["total"] = (free + paid).into();
}
pub fn remove_paid_gems(user: &mut JsonValue, amount: i64) {
let free = user["gem"]["free"].as_i64().unwrap();
let mut paid = user["gem"]["charge"].as_i64().unwrap();
paid -= amount;
if paid < 0 {
paid = 0;
}
user["gem"]["charge"] = paid.into();
user["gem"]["total"] = (free + paid).into();
}
pub fn get_region(headers: &HeaderMap) -> bool {
let blank_header = HeaderValue::from_static("");
let asset_version = headers.get("aoharu-asset-version").unwrap_or(&blank_header).to_str().unwrap_or("");
asset_version == global::ASSET_VERSION_JP
}
pub fn check_for_region(user: &mut JsonValue, headers: &HeaderMap) {
let items = if user["data"]["updated_value_list"]["item_list"].is_empty() {user["data"]["item_list"].clone()} else {user["data"]["updated_value_list"]["item_list"].clone()};
let is_jp = get_region(headers);
if !is_jp || items.is_empty() {
return;
}
let mut id = 0;
for (i, data) in items.members().enumerate() {
if data["master_item_id"] == 15570008 {
id = i + 1;
break;
}
}
if id > 0 {
if user["data"]["updated_value_list"]["item_list"].is_empty() {
user["data"]["item_list"].array_remove(id - 1);
} else {
user["data"]["updated_value_list"]["item_list"].array_remove(id - 1);
}
}
}
// true - limit reached
// false - all good
const GIFT_LIMIT: usize = 100000;
const LIMIT_ITEMS: i64 = 200000000;
const LIMIT_COINS: i64 = 2000000000;
const LIMIT_PRIMOGEMS: i64 = 1000000;
pub fn give_shop(master_item_id: i64, count: i64, user: &mut JsonValue) -> bool {
let mut has = false;
for dataa in user["shop_list"].members_mut() {
if dataa["master_shop_item_id"].as_i64().unwrap() == master_item_id {
has = true;
let new_amount = dataa["count"].as_i64().unwrap() + count;
if new_amount > LIMIT_ITEMS {
return true;
}
dataa["count"] = new_amount.into();
break;
}
}
if !has {
user["shop_list"].push(object!{
master_shop_item_id: master_item_id,
count: count
}).unwrap();
}
false
}
pub fn give_item(master_item_id: i64, amount: i64, user: &mut JsonValue) -> bool {
let mut has = false;
for dataa in user["item_list"].members_mut() {
if dataa["master_item_id"].as_i64().unwrap() == master_item_id {
has = true;
let new_amount = dataa["amount"].as_i64().unwrap() + amount;
if new_amount > LIMIT_ITEMS {
return true;
}
dataa["amount"] = new_amount.into();
break;
}
}
if !has {
user["item_list"].push(object!{
id: master_item_id,
master_item_id: master_item_id,
amount: amount,
expire_date_time: null
}).unwrap();
}
false
}
pub fn use_item(item: &JsonValue, multiplier: i64, user: &mut JsonValue) {
if item["consumeType"] == 0 {
// Is anything really ever free...?
} else if item["consumeType"] == 1 {
remove_gems(user, item["amount"].as_i64().unwrap());
} else if item["consumeType"] == 2 {
remove_paid_gems(user, item["amount"].as_i64().unwrap());
} else if item["consumeType"] == 4 {
use_itemm(item["value"].as_i64().unwrap(), item["amount"].as_i64().unwrap() * multiplier, user);
} else {
println!("Unknown consume type {}", item["consumeType"]);
}
}
pub fn give_gift(data: &JsonValue, user: &mut JsonValue, missions: &mut JsonValue, clear_missions: &mut JsonValue, chats: &mut JsonValue) -> bool {
if data.is_empty() {
return false;
}
if data["reward_type"] == 1 {
// basically primogems!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
return !give_primogems(data["amount"].as_i64().unwrap(), user);
} else if data["reward_type"] == 2 {
//character
give_character(data["value"].as_i64().unwrap(), user, missions, clear_missions, chats);
return true;
} else if data["reward_type"] == 3 {
return !give_item(data["value"].as_i64().unwrap(), data["amount"].as_i64().unwrap(), user);
} else if data["reward_type"] == 4 {
// basically moraa!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
return !give_points(data["value"].as_i64().unwrap(), data["amount"].as_i64().unwrap(), user, missions, clear_missions);
} else if data["reward_type"] == 8 {
// title
let title = data["value"].as_i64().unwrap();
if !user["master_title_ids"].contains(title) {
user["master_title_ids"].push(title).unwrap();
return true;
}
return false;
}
println!("Redeeming reward not implemented for reward type {}", data["reward_type"]);
false
}
pub fn give_gift_basic(ty_pe: i32, id: i64, amount: i64, user: &mut JsonValue, missions: &mut JsonValue, clear_missions: &mut JsonValue, chats: &mut JsonValue) -> bool {
give_gift(&object!{
reward_type: ty_pe,
amount: amount,
value: id
}, user, missions, clear_missions, chats)
}
pub fn give_points(master_item_id: i64, amount: i64, user: &mut JsonValue, missions: &mut JsonValue, clear_missions: &mut JsonValue) -> bool {
if master_item_id == 1 {
let cleared = advance_variable_mission(1121001, 1121019, amount, missions);
for data in cleared.members() {
if !clear_missions.contains(data.as_i64().unwrap()) {
clear_missions.push(data.clone()).unwrap();
}
}
}
let mut has = false;
for data in user["point_list"].members_mut() {
if data["type"].as_i64().unwrap() == master_item_id {
has = true;
let new_amount = data["amount"].as_i64().unwrap() + amount;
if new_amount > LIMIT_COINS {
return true;
}
data["amount"] = new_amount.into();
break;
}
}
if !has {
user["point_list"].push(object!{
type: master_item_id,
amount: amount
}).unwrap();
}
false
}
pub fn use_itemm(master_item_id: i64, amount: i64, user: &mut JsonValue) {
for data in user["item_list"].members_mut() {
if data["master_item_id"].as_i64().unwrap() == master_item_id {
if data["amount"].as_i64().unwrap() >= amount {
data["amount"] = (data["amount"].as_i64().unwrap() - amount).into();
} else {
data["amount"] = (0).into();
}
break;
}
}
}
pub fn give_primogems(amount: i64, user: &mut JsonValue) -> bool {
let new_amount = user["gem"]["free"].as_i64().unwrap() + amount;
if new_amount > LIMIT_PRIMOGEMS {
return true;
}
user["gem"]["free"] = new_amount.into();
false
}
pub fn gift_item(item: &JsonValue, reason: &str, user: &mut JsonValue) -> JsonValue {
let to_push = object!{
id: item["id"].clone(),
reward_type: item["type"].clone(),
is_receive: 0,
reason_text: reason,
value: item["value"].clone(),
level: item["level"].clone(),
amount: item["amount"].clone(),
created_date_time: global::timestamp(),
expire_date_time: global::timestamp() + (5 * (24 * 60 * 60)),
received_date_time: 0
};
if user["home"]["gift_list"].len() >= GIFT_LIMIT {
return to_push;
}
user["home"]["gift_list"].push(to_push.clone()).unwrap();
to_push
}
fn random_number(lowest: usize, highest: usize) -> usize {
if lowest == highest {
return lowest;
}
assert!(lowest < highest);
rand::thread_rng().gen_range(lowest..highest + 1)
}
pub fn gift_item_basic(id: i32, value: i64, ty_pe: i32, reason: &str, user: &mut JsonValue) -> JsonValue {
gift_item(&object!{
id: random_number(0, global::timestamp_msec() as usize),
type: ty_pe,
level: 0,
amount: value,
value: id
}, reason, user)
}
pub fn lp_modification(user: &mut JsonValue, change_amount: u64, remove: bool) {
let max = get_user_rank_data(user["user"]["exp"].as_i64().unwrap())["maxLp"].as_u64().unwrap();
let speed = 285; //4 mins, 45 sec
let since_last = global::timestamp() - user["stamina"]["last_updated_time"].as_u64().unwrap();
let diff = since_last % speed;
let restored = (since_last - diff) / speed;
user["stamina"]["last_updated_time"] = (global::timestamp() - diff).into();
let mut stamina = user["stamina"]["stamina"].as_u64().unwrap();
if stamina < max {
stamina += restored;
if stamina > max {
stamina = max;
}
}
if remove {
stamina -= change_amount;
} else {
stamina += change_amount;
}
user["stamina"]["stamina"] = stamina.into();
}
pub fn get_rarity(id: i64) -> i32 {
databases::CARD_LIST[id.to_string()]["rarity"].as_i32().unwrap_or(0)
}
// true - added
// false - already has
pub fn give_character(id: i64, user: &mut JsonValue, missions: &mut JsonValue, clear_missions: &mut JsonValue, chats: &mut JsonValue) -> bool {
let character_rarity = get_rarity(id);
if character_rarity == 0 {
println!("Attempted to give user undefined card!! Card id: {}", id);
return false;
}
if !databases::CHARACTER_CHATS[id.to_string()]["51"].is_empty() {
let chat = &databases::CHARACTER_CHATS[id.to_string()]["51"];
let mission_id = databases::MISSION_REWARD[chat[0].to_string()]["value"].as_i64().unwrap();
if crate::router::chat::add_chat_from_chapter_id(mission_id, chats) {
update_mission_status(chat[1].as_i64().unwrap(), 0, true, true, 1, missions);
if !clear_missions.contains(chat[1].as_i64().unwrap()) {
clear_missions.push(chat[1].clone()).unwrap();
}
}
}
for data in user["card_list"].members() {
if data["master_card_id"] == id || data["id"] == id {
let amount = if character_rarity == 1 { 20 } else if character_rarity == 2 { 50 } else if character_rarity == 3 { 500 } else { 0 };
give_item(19100001, amount, user);
return false;
}
}
let cleared = advance_variable_mission(1112001, 1112033, 1, missions);
for data in cleared.members() {
if !clear_missions.contains(data.as_i64().unwrap()) {
clear_missions.push(data.clone()).unwrap();
}
}
let to_push = object!{
"id": id,
"master_card_id": id,
"exp": 0,
"skill_exp": 0,
"evolve": [],
"created_date_time": global::timestamp()
};
user["card_list"].push(to_push.clone()).unwrap();
true
}
pub fn get_user_rank_data(exp: i64) -> JsonValue {
for (i, rank) in databases::RANKS.members().enumerate() {
if exp < rank["exp"].as_i64().unwrap() {
return databases::RANKS[i - 1].clone();
}
}
databases::RANKS[databases::RANKS.len() - 1].clone()
}
pub fn give_exp(amount: i32, user: &mut JsonValue, mission: &mut JsonValue, rv: &mut JsonValue) {
let current_rank = get_user_rank_data(user["user"]["exp"].as_i64().unwrap());
user["user"]["exp"] = (user["user"]["exp"].as_i32().unwrap() + amount).into();
let new_rank = get_user_rank_data(user["user"]["exp"].as_i64().unwrap());
if current_rank["rank"] != new_rank["rank"] {
user["stamina"]["stamina"] = (user["stamina"]["stamina"].as_i64().unwrap() + new_rank["maxLp"].as_i64().unwrap()).into();
user["stamina"]["last_updated_time"] = global::timestamp().into();
let status = get_mission_status(get_variable_mission_num(1101001, 1101030, mission), mission);
if status.is_empty() {
return;
}
let to_advance = new_rank["rank"].as_i64().unwrap() - status["progress"].as_i64().unwrap();
let rvv = advance_variable_mission(1101001, 1101030, to_advance, mission);
for id in rvv.members() {
rv.push(id.as_i64().unwrap()).unwrap();
}
}
}
pub fn update_mission_status(master_mission_id: i64, expire: u64, completed: bool, claimed: bool, advance: i64, missions: &mut JsonValue) -> Option<i64> {
for mission in missions.members_mut() {
if mission["master_mission_id"].as_i64().unwrap() == master_mission_id {
let was_completed = mission["status"] == 2;
mission["status"] = if claimed { 3 } else if completed { 2 } else { 1 }.into();
if expire != 0 {
mission["expire_date_time"] = expire.into();
}
if (mission["expire_date_time"].as_u64().unwrap() < global::timestamp() || expire != 0) && (mission["expire_date_time"].as_u64().unwrap() != 0 || expire != 0) {
mission["progress"] = 0.into();
}
if advance > 0 {
mission["progress"] = (mission["progress"].as_i64().unwrap() + advance).into();
}
if completed && !claimed && !was_completed {
return Some(master_mission_id);
}
return None;
}
}
None
}
pub fn update_mission_status_multi(master_mission_id: JsonValue, expire: u64, completed: bool, claimed: bool, advance: i64, missions: &mut JsonValue) -> JsonValue {
let mut rv = array![];
for mission in master_mission_id.members() {
let val = update_mission_status(mission.as_i64().unwrap(), expire, completed, claimed, advance, missions);
if let Some(val2) = val {
rv.push(val2).unwrap();
}
}
rv
}
pub fn get_mission_status(id: i64, missions: &JsonValue) -> JsonValue {
for mission in missions.members() {
if mission["master_mission_id"].as_i64().unwrap() == id {
return mission.clone();
}
}
JsonValue::Null
}
pub fn change_mission_id(old: i64, new: i64, missions: &mut JsonValue) {
for mission in missions.members_mut() {
if mission["master_mission_id"].as_i64().unwrap() == old {
mission["master_mission_id"] = new.into();
return;
}
}
}
pub fn get_variable_mission_num(min: i64, max: i64, missions: &JsonValue) -> i64 {
for i in min..=max {
let mission_status = get_mission_status(i, missions);
if mission_status.is_empty() {
continue;
}
return i;
}
0
}
pub fn advance_variable_mission(min: i64, max: i64, count: i64, missions: &mut JsonValue) -> JsonValue {
let mut rv = array![];
for i in min..=max {
let mission_status = get_mission_status(i, missions);
if mission_status.is_empty() {
continue;
}
let mission_info = &databases::MISSION_LIST[i.to_string()];
if i == max && mission_info["conditionNumber"].as_i64().unwrap() <= mission_status["progress"].as_i64().unwrap() {
break;
}
if mission_info["conditionNumber"].as_i64().unwrap() > mission_status["progress"].as_i64().unwrap() + count {
if update_mission_status(i, 0, false, false, count, missions).is_some() {
rv.push(i).unwrap();
}
} else if update_mission_status(i, 0, true, false, count, missions).is_some() {
rv.push(i).unwrap();
}
break;
}
rv
}
pub fn advance_mission(id: i64, count: i64, max: i64, missions: &mut JsonValue) -> Option<i64> {
let mission = get_mission_status(id, missions);
if mission["status"].as_i32().unwrap() > 1 {
return None;
}
let mut new = mission["progress"].as_i64().unwrap() + count;
if new > max {
new = max;
}
let completed = new == max;
let advanced = new - mission["progress"].as_i64().unwrap();
if update_mission_status(id, 0, completed, false, advanced, missions).is_some() {
return Some(id);
}
None
}
pub fn completed_daily_mission(id: i64, missions: &mut JsonValue) -> JsonValue {
let all_daily_missions = array![1224003, 1253003, 1273009, 1273010, 1273011, 1273012];
let mission = get_mission_status(id, missions);
if mission["expire_date_time"].as_u64().unwrap_or(0) >= global::timestamp() && mission["status"].as_i32().unwrap() > 1 {
return array![];
}
let mut rv = array![];
if id == 1253003 {
rv = advance_variable_mission(1153001, 1153019, 1, missions);
}
let mut mission = get_mission_status(1224003, missions);
let next_reset = global::timestamp_since_midnight() + (24 * 60 * 60);
if mission["expire_date_time"].as_u64().unwrap_or(0) < global::timestamp() {
update_mission_status_multi(all_daily_missions, next_reset, false, false, 0, missions);
mission = get_mission_status(1224003, missions);
}
if mission["progress"].as_i32().unwrap_or(0) == 4 {
if update_mission_status(1224003, 0, true, false, 1, missions).is_some() {
rv.push(1224003).unwrap();
}
} else if update_mission_status(1224003, 0, false, false, 1, missions).is_some() {
rv.push(1224003).unwrap();
}
if update_mission_status(id, next_reset, true, false, 1, missions).is_some() {
rv.push(id).unwrap();
}
rv
}
pub fn use_item_req(req: HttpRequest, body: String) -> Option<JsonValue> {
let key = global::get_login(req.headers(), &body);
let body = json::parse(&encryption::decrypt_packet(&body).unwrap()).unwrap();
let mut user = userdata::get_acc(&key);
let item = &databases::ITEM_INFO[body["id"].to_string()];
let amount = body["amount"].as_i64().unwrap();
if item["effectType"].as_i32().unwrap() == 1 {
lp_modification(&mut user, item["effectValue"].as_u64().unwrap() * (amount as u64), false);
} else {
println!("Use item not implemented for effect type {}", item["effectType"]);
}
use_item(&object!{
value: body["id"].as_i64().unwrap(),
amount: 1,
consumeType: 4
}, amount, &mut user);
userdata::save_acc(&key, user.clone());
Some(object!{
item_list: user["item_list"].clone(),
stamina: user["stamina"].clone()
})
}