use crate::{
analyzers::{helpers, Analyzer, Outcome},
EventKind, Log,
};
pub const CAIRN_CM_BUFF: u32 = 38_098;
#[derive(Debug, Clone, Copy)]
pub struct Cairn<'log> {
log: &'log Log,
}
impl<'log> Cairn<'log> {
pub fn new(log: &'log Log) -> Self {
Cairn { log }
}
}
impl<'log> Analyzer for Cairn<'log> {
fn log(&self) -> &Log {
self.log
}
fn is_cm(&self) -> bool {
helpers::buff_present(self.log, CAIRN_CM_BUFF)
}
fn outcome(&self) -> Option<Outcome> {
check_reward!(self.log);
Outcome::from_bool(helpers::boss_is_dead(self.log))
}
}
pub const MO_CM_HEALTH: u64 = 30_000_000;
#[derive(Debug, Clone, Copy)]
pub struct MursaatOverseer<'log> {
log: &'log Log,
}
impl<'log> MursaatOverseer<'log> {
pub fn new(log: &'log Log) -> Self {
MursaatOverseer { log }
}
}
impl<'log> Analyzer for MursaatOverseer<'log> {
fn log(&self) -> &Log {
self.log
}
fn is_cm(&self) -> bool {
helpers::boss_health(self.log)
.map(|h| h >= MO_CM_HEALTH)
.unwrap_or(false)
}
fn outcome(&self) -> Option<Outcome> {
check_reward!(self.log);
Outcome::from_bool(helpers::boss_is_dead(self.log))
}
}
pub const SAMAROG_CM_HEALTH: u64 = 40_000_000;
#[derive(Debug, Clone, Copy)]
pub struct Samarog<'log> {
log: &'log Log,
}
impl<'log> Samarog<'log> {
pub fn new(log: &'log Log) -> Self {
Samarog { log }
}
}
impl<'log> Analyzer for Samarog<'log> {
fn log(&self) -> &Log {
self.log
}
fn is_cm(&self) -> bool {
helpers::boss_health(self.log)
.map(|h| h >= SAMAROG_CM_HEALTH)
.unwrap_or(false)
}
fn outcome(&self) -> Option<Outcome> {
check_reward!(self.log);
Outcome::from_bool(helpers::boss_is_dead(self.log))
}
}
pub const DEIMOS_CM_HEALTH: u64 = 42_000_000;
#[derive(Debug, Clone, Copy)]
pub struct Deimos<'log> {
log: &'log Log,
}
impl<'log> Deimos<'log> {
pub fn new(log: &'log Log) -> Self {
Deimos { log }
}
}
impl<'log> Analyzer for Deimos<'log> {
fn log(&self) -> &Log {
self.log
}
fn is_cm(&self) -> bool {
helpers::boss_health(self.log)
.map(|h| h >= DEIMOS_CM_HEALTH)
.unwrap_or(false)
}
fn outcome(&self) -> Option<Outcome> {
check_reward!(self.log);
let split_time = deimos_10_time(self.log);
if split_time == 0 {
return Some(Outcome::Failure);
}
let at_address = deimos_at_address(self.log);
if at_address == 0 {
return Some(Outcome::Failure);
}
let mut player_exit = 0u64;
let mut at_exit = 0u64;
for event in self.log.events() {
match event.kind() {
EventKind::ExitCombat { agent_addr }
if self
.log
.agent_by_addr(*agent_addr)
.map(|a| a.kind().is_player())
.unwrap_or(false)
&& event.time() >= player_exit =>
{
player_exit = event.time();
}
EventKind::Targetable {
agent_addr,
targetable,
} if *agent_addr == at_address && !targetable && event.time() >= at_exit => {
at_exit = event.time();
}
_ => (),
}
}
Outcome::from_bool(player_exit > at_exit + 1000)
}
}
fn deimos_10_time(log: &Log) -> u64 {
let mut first_aware = 0u64;
for event in log.events() {
if let EventKind::Targetable { targetable, .. } = event.kind() {
if *targetable {
first_aware = event.time();
}
}
}
first_aware
}
fn deimos_at_address(log: &Log) -> u64 {
for event in log.events().iter().rev() {
if let EventKind::AttackTarget {
agent_addr,
parent_agent_addr,
..
} = event.kind()
{
let parent = log.agent_by_addr(*parent_agent_addr);
if let Some(parent) = parent {
if Some("Deimos") == parent.as_gadget().map(|g| g.name()) {
return *agent_addr;
}
}
}
}
0
}