properly do uptime windows
This commit is contained in:
parent
c5a3ae279c
commit
d2e11d2b4f
4 changed files with 240 additions and 39 deletions
13
src/clock.rs
13
src/clock.rs
|
@ -1,6 +1,6 @@
|
||||||
// Mostly nicked from https://github.com/icykoneko/ff14-fish-tracker-app/blob/master/js/app/time.js. Thanks!
|
// Mostly nicked from https://github.com/icykoneko/ff14-fish-tracker-app/blob/master/js/app/time.js. Thanks!
|
||||||
|
|
||||||
use chrono::{DateTime, Timelike, Utc};
|
use chrono::{DateTime, Duration, Timelike, Utc};
|
||||||
|
|
||||||
const EARTH_TO_EORZEA: f64 = 3600. / 175.;
|
const EARTH_TO_EORZEA: f64 = 3600. / 175.;
|
||||||
const EORZEA_TO_EARTH: f64 = 1. / EARTH_TO_EORZEA;
|
const EORZEA_TO_EARTH: f64 = 1. / EARTH_TO_EORZEA;
|
||||||
|
@ -36,6 +36,17 @@ pub fn set_hm_from_float(date: &DateTime<Utc>, hm: f32) -> DateTime<Utc> {
|
||||||
.and_utc()
|
.and_utc()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Calculates the length between a window start time and end time, and returns it as a Duration.
|
||||||
|
pub fn get_window_duration(start_time: f32, end_time: f32) -> Duration {
|
||||||
|
let diff_float = if end_time > start_time {
|
||||||
|
end_time - start_time
|
||||||
|
} else {
|
||||||
|
(24. - start_time) + end_time
|
||||||
|
};
|
||||||
|
let minutes = (diff_float - (diff_float as u32 as f32) * 60.) as u32;
|
||||||
|
Duration::minutes(((diff_float as u32) * 60 + minutes) as i64)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn display_eorzea_time(date: &DateTime<Utc>) -> String {
|
pub fn display_eorzea_time(date: &DateTime<Utc>) -> String {
|
||||||
date.format("%H:%M").to_string()
|
date.format("%H:%M").to_string()
|
||||||
}
|
}
|
||||||
|
|
214
src/data.rs
214
src/data.rs
|
@ -1,6 +1,6 @@
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
|
|
||||||
use chrono::{DateTime, Duration, Timelike, Utc};
|
use chrono::{DateTime, Datelike, Duration, Timelike, Utc};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
@ -93,14 +93,47 @@ pub struct CombinedFish<'a> {
|
||||||
pub entry: &'a FishEntry,
|
pub entry: &'a FishEntry,
|
||||||
pub meta: &'a FishMeta,
|
pub meta: &'a FishMeta,
|
||||||
pub is_up: bool,
|
pub is_up: bool,
|
||||||
pub next_uptime: DateTime<Utc>,
|
pub is_always_up: bool,
|
||||||
|
pub windows: Vec<Window>,
|
||||||
|
pub rarity: f32,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct Window {
|
||||||
|
pub start_time: DateTime<Utc>,
|
||||||
|
pub duration: Duration,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> CombinedFish<'a> {
|
impl<'a> CombinedFish<'a> {
|
||||||
/// Fills in the rest of the struct.
|
/// Fills in the rest of the struct.
|
||||||
pub fn find_uptime(&mut self, data: &Data) {
|
pub fn find_uptime(&mut self, data: &Data) {
|
||||||
self.is_up = self.is_up_now(data);
|
// If the start and end times are equal, and there's no weather constraint, the fish is always up.
|
||||||
self.next_uptime = self.get_next_uptime(data);
|
if self
|
||||||
|
.entry
|
||||||
|
.start_hour
|
||||||
|
.is_some_and(|f| self.entry.end_hour.is_some_and(|g| f == 0. && g == 24.))
|
||||||
|
&& self.entry.weather_set.is_empty()
|
||||||
|
{
|
||||||
|
self.is_always_up = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
self.windows = self.get_next_uptimes(data);
|
||||||
|
let now = clock::get_current_eorzea_date();
|
||||||
|
let first_window = self.windows.first();
|
||||||
|
self.is_up = if let Some(window) = first_window {
|
||||||
|
window.start_time < now && (window.start_time + window.duration) > now
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
};
|
||||||
|
|
||||||
|
// Find the rarity of the fish. We define this as the percentage of uptime in the next 10 days.
|
||||||
|
let duration = clock::to_eorzea_time(Utc::now() + Duration::days(10))
|
||||||
|
- clock::get_current_eorzea_date();
|
||||||
|
let up_durations = self
|
||||||
|
.windows
|
||||||
|
.iter()
|
||||||
|
.fold(Duration::zero(), |acc, w| acc + w.duration);
|
||||||
|
self.rarity = up_durations.num_minutes() as f32 / duration.num_minutes() as f32;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Finds out whether the fish is technically up at the given time (discounting factors like weather).
|
/// Finds out whether the fish is technically up at the given time (discounting factors like weather).
|
||||||
|
@ -202,32 +235,157 @@ impl<'a> CombinedFish<'a> {
|
||||||
results
|
results
|
||||||
}
|
}
|
||||||
|
|
||||||
// Go until you find the next uptime slot, and return its date.
|
// Given a forecast and a start date of a window, find how long that weather stays.
|
||||||
pub fn get_next_uptime(&self, data: &Data) -> DateTime<Utc> {
|
pub fn get_weather_duration(&self, forecast: &Forecast, date: &DateTime<Utc>) -> Duration {
|
||||||
// Is this a time-only fish? If so, just return the next time.
|
let mut length = 0;
|
||||||
|
let mut date = date.clone();
|
||||||
|
let mut cur_pattern = forecast.weather_at(date).weather_id;
|
||||||
|
while self.entry.weather_set.iter().any(|w| w == &cur_pattern) {
|
||||||
|
length += 8;
|
||||||
|
date = date + Duration::hours(8);
|
||||||
|
cur_pattern = forecast.weather_at(date).weather_id;
|
||||||
|
}
|
||||||
|
Duration::hours(length)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculate the uptime windows occurring in the next 10 days.
|
||||||
|
pub fn get_next_uptimes(&self, data: &Data) -> Vec<Window> {
|
||||||
|
let mut idx_date = round_to_last_weather_time(clock::get_current_eorzea_date());
|
||||||
|
let goal_date = clock::to_eorzea_time(clock::to_earth_time(idx_date) + Duration::days(10));
|
||||||
|
|
||||||
|
// If it's a time-only fish, the procedure is a bit simpler.
|
||||||
if self.entry.weather_set.is_empty() && self.entry.start_hour.is_some() {
|
if self.entry.weather_set.is_empty() && self.entry.start_hour.is_some() {
|
||||||
return clock::to_earth_time(self.get_next_time_uptime());
|
let mut idx_date = clock::get_current_eorzea_date(); // We don't need to round to a weather time here.
|
||||||
}
|
let mut results = Vec::new();
|
||||||
|
|
||||||
// If not, check if there's a weather-only fish
|
|
||||||
let next_weather_uptimes = self.get_next_weather_uptimes(data, 50);
|
|
||||||
if !self.entry.weather_set.is_empty() && self.entry.start_hour.is_none() {
|
|
||||||
return clock::to_earth_time(next_weather_uptimes.first().unwrap().clone());
|
|
||||||
}
|
|
||||||
|
|
||||||
// Now it gets a little bit tricky: We need to find a weather slot that also includes the start point for the
|
|
||||||
// time window.
|
|
||||||
for date in next_weather_uptimes {
|
|
||||||
// Does the start time lie within this window?
|
|
||||||
let uptime_end = date + Duration::hours(8);
|
|
||||||
let start_hour = self.entry.start_hour.unwrap();
|
let start_hour = self.entry.start_hour.unwrap();
|
||||||
if date.hour() as f32 <= start_hour && start_hour <= uptime_end.hour() as f32 {
|
let end_hour = self.entry.end_hour.unwrap();
|
||||||
return clock::to_earth_time(clock::set_hm_from_float(&date, start_hour));
|
while idx_date < goal_date {
|
||||||
|
// Because there will be a window every single day, just set our clock to when it starts.
|
||||||
|
let next_start =
|
||||||
|
clock::set_hm_from_float(&idx_date, self.entry.start_hour.unwrap());
|
||||||
|
let duration = clock::get_window_duration(start_hour, end_hour);
|
||||||
|
// Safely push the full duration.
|
||||||
|
results.push(Window {
|
||||||
|
start_time: next_start,
|
||||||
|
duration,
|
||||||
|
});
|
||||||
|
// If the end time is on the same day, push the clock ahead by a day so that we don't use the same
|
||||||
|
// window the next iteration.
|
||||||
|
let next_end = next_start + duration;
|
||||||
|
if next_start.day() == next_end.day() {
|
||||||
|
idx_date = idx_date + Duration::days(1);
|
||||||
|
} else {
|
||||||
|
idx_date = next_end;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If our fish is weather-affected, we need to be a bit more careful.
|
||||||
|
let forecast = self.get_forecast(data).unwrap();
|
||||||
|
let has_time_constraint =
|
||||||
|
!(self.entry.start_hour.unwrap() == 0. && self.entry.end_hour.unwrap() == 24.);
|
||||||
|
let mut results = Vec::new();
|
||||||
|
// No use in running this loop if the fish is up all the time.
|
||||||
|
if !has_time_constraint && self.entry.weather_set.is_empty() {
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
while idx_date < goal_date {
|
||||||
|
// Is our fish technically in favorable weather here?
|
||||||
|
let has_weather = self.is_in_correct_weather_at(forecast, idx_date);
|
||||||
|
// If we don't have a time constraint, and the weather is good, add to our results.
|
||||||
|
|
||||||
|
if !has_time_constraint && has_weather {
|
||||||
|
// The weather might drag on for more than one cycle, find out when it ends.
|
||||||
|
// However: A weather constraint with a previous weather set will never have two consecutive uptime cycles.
|
||||||
|
let duration = if self.entry.previous_weather_set.is_empty() {
|
||||||
|
self.get_weather_duration(forecast, &idx_date)
|
||||||
|
} else {
|
||||||
|
Duration::hours(8)
|
||||||
|
};
|
||||||
|
results.push(Window {
|
||||||
|
start_time: idx_date,
|
||||||
|
duration,
|
||||||
|
});
|
||||||
|
|
||||||
|
idx_date = idx_date + duration;
|
||||||
|
} else if has_time_constraint && has_weather {
|
||||||
|
// Calculate start and end times for both the weather uptime and the time-based uptime.
|
||||||
|
let w_duration = if self.entry.previous_weather_set.is_empty() {
|
||||||
|
self.get_weather_duration(forecast, &idx_date)
|
||||||
|
} else {
|
||||||
|
Duration::hours(8)
|
||||||
|
};
|
||||||
|
let w_start_date = idx_date;
|
||||||
|
let w_end_date = w_start_date + w_duration;
|
||||||
|
let start_hour = self.entry.start_hour.unwrap();
|
||||||
|
let end_hour = self.entry.end_hour.unwrap();
|
||||||
|
// If the current weather cycle is the midnight one, and the timespan wraps a day, start
|
||||||
|
// at the previous day.
|
||||||
|
let t_start_date = if idx_date.hour() == 0 && start_hour > end_hour {
|
||||||
|
clock::set_hm_from_float(
|
||||||
|
&(idx_date - Duration::days(1)),
|
||||||
|
self.entry.start_hour.unwrap(),
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
clock::set_hm_from_float(&idx_date, self.entry.start_hour.unwrap())
|
||||||
|
};
|
||||||
|
let t_duration = clock::get_window_duration(
|
||||||
|
self.entry.start_hour.unwrap(),
|
||||||
|
self.entry.end_hour.unwrap(),
|
||||||
|
);
|
||||||
|
let t_end_date = t_start_date + t_duration;
|
||||||
|
// We have to check four cases:
|
||||||
|
if t_start_date <= w_start_date && t_end_date >= w_end_date {
|
||||||
|
// CASE 1: The time-based uptime exceeds the weather-based uptime at start and end
|
||||||
|
// This is the easiest: Just save the full weather-based uptime.
|
||||||
|
results.push(Window {
|
||||||
|
start_time: w_start_date,
|
||||||
|
duration: w_duration,
|
||||||
|
});
|
||||||
|
} else if w_start_date <= t_start_date && w_end_date >= t_end_date {
|
||||||
|
// CASE 2: Vice-versa, the weather-based uptime exceeds the time-based uptime.
|
||||||
|
// Also just save the time-based uptime.
|
||||||
|
results.push(Window {
|
||||||
|
start_time: t_start_date,
|
||||||
|
duration: t_duration,
|
||||||
|
})
|
||||||
|
} else if w_start_date <= t_start_date
|
||||||
|
&& t_end_date >= w_end_date
|
||||||
|
&& w_end_date > t_start_date
|
||||||
|
{
|
||||||
|
// CASE 3: The weather starts before the time, but the doesn't last beyond it.
|
||||||
|
// In this case, save the time-based start time, and the duration between time-based start time
|
||||||
|
// and weather-based end time.
|
||||||
|
let diff = w_end_date - t_start_date;
|
||||||
|
results.push(Window {
|
||||||
|
start_time: t_start_date,
|
||||||
|
duration: diff,
|
||||||
|
});
|
||||||
|
} else if t_start_date <= w_start_date
|
||||||
|
&& w_end_date >= t_end_date
|
||||||
|
&& t_end_date > w_start_date
|
||||||
|
{
|
||||||
|
// CASE 4: The time starts before the weather, but doesn't last beyond it.
|
||||||
|
// Do the opposite of Case 3 here.
|
||||||
|
let diff = t_end_date - w_start_date;
|
||||||
|
results.push(Window {
|
||||||
|
start_time: w_start_date,
|
||||||
|
duration: diff,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
idx_date = idx_date + w_duration;
|
||||||
|
} else {
|
||||||
|
idx_date = idx_date + Duration::hours(8);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: This should never be called?
|
if self.meta.name_en == "The Drowned Sniper" {
|
||||||
Utc::now()
|
dbg!(&results[1]);
|
||||||
|
dbg!(clock::to_earth_time(results[1].start_time));
|
||||||
|
}
|
||||||
|
results
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -269,8 +427,10 @@ impl Data {
|
||||||
let mut cfish = CombinedFish {
|
let mut cfish = CombinedFish {
|
||||||
entry: v,
|
entry: v,
|
||||||
meta: m,
|
meta: m,
|
||||||
is_up: false, // fake default values for now
|
is_up: false, // fake default values for now
|
||||||
next_uptime: Utc::now(), // dito
|
is_always_up: false, // dito
|
||||||
|
windows: Vec::new(), // dito
|
||||||
|
rarity: 1., // dito
|
||||||
};
|
};
|
||||||
cfish.find_uptime(self);
|
cfish.find_uptime(self);
|
||||||
(k.clone(), cfish)
|
(k.clone(), cfish)
|
||||||
|
|
|
@ -1,13 +1,9 @@
|
||||||
use std::sync::Arc;
|
use std::{cmp::Ordering, sync::Arc};
|
||||||
|
|
||||||
use axum::extract::State;
|
use axum::extract::State;
|
||||||
use maud::{html, Markup, DOCTYPE};
|
use maud::{html, Markup, DOCTYPE};
|
||||||
|
|
||||||
use crate::{
|
use crate::{clock, data::CombinedFish, AppState};
|
||||||
clock,
|
|
||||||
data::{self, CombinedFish},
|
|
||||||
AppState,
|
|
||||||
};
|
|
||||||
|
|
||||||
pub fn layout(content: Markup) -> Markup {
|
pub fn layout(content: Markup) -> Markup {
|
||||||
html! {
|
html! {
|
||||||
|
@ -27,22 +23,41 @@ pub fn layout(content: Markup) -> Markup {
|
||||||
pub fn main_page(state: State<Arc<AppState>>) -> Markup {
|
pub fn main_page(state: State<Arc<AppState>>) -> Markup {
|
||||||
let meta = state.data.fish_with_meta();
|
let meta = state.data.fish_with_meta();
|
||||||
let mut values: Vec<&CombinedFish> = meta.values().filter(|f| f.entry.big_fish).collect();
|
let mut values: Vec<&CombinedFish> = meta.values().filter(|f| f.entry.big_fish).collect();
|
||||||
values.sort_by(|afish, bfish| bfish.is_up.cmp(&afish.is_up));
|
values.sort_by(|afish, bfish| {
|
||||||
|
bfish
|
||||||
|
.is_up
|
||||||
|
.cmp(&afish.is_up)
|
||||||
|
.then(bfish.is_always_up.cmp(&afish.is_always_up))
|
||||||
|
.then(bfish.rarity.total_cmp(&afish.rarity).reverse())
|
||||||
|
// if (afish.is_up || afish.is_always_up) && !(bfish.is_up && bfish.is_always_up) {
|
||||||
|
// Ordering::Less
|
||||||
|
// } else {
|
||||||
|
// Ordering::Greater
|
||||||
|
// }
|
||||||
|
});
|
||||||
layout(html! {
|
layout(html! {
|
||||||
h1 { "Hello! Current ET: " (clock::get_current_eorzea_date().format("%H:%M")) }
|
h1 { "Hello! Current ET: " (clock::get_current_eorzea_date().format("%H:%M")) }
|
||||||
@for fish in values {
|
@for fish in values {
|
||||||
section.up[fish.is_up] {
|
section.up[fish.is_up || fish.is_always_up] {
|
||||||
.title {
|
.title {
|
||||||
h3 { (fish.meta.name_en) }
|
h3 { (fish.meta.name_en) }
|
||||||
|
.subtitle {
|
||||||
|
(fish.entry.patch)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
.meta {
|
.meta {
|
||||||
@if fish.entry.start_hour.is_some() && fish.entry.end_hour.is_some() {
|
@if fish.entry.start_hour.is_some() && fish.entry.end_hour.is_some() {
|
||||||
div {
|
div {
|
||||||
(clock::display_eorzea_time(&clock::set_hm_from_float(&clock::get_current_eorzea_date(), fish.entry.start_hour.unwrap())))
|
@if !fish.is_always_up {
|
||||||
" to "
|
(clock::display_eorzea_time(&clock::set_hm_from_float(&clock::get_current_eorzea_date(), fish.entry.start_hour.unwrap())))
|
||||||
(clock::display_eorzea_time(&clock::set_hm_from_float(&clock::get_current_eorzea_date(), fish.entry.end_hour.unwrap())))
|
" to "
|
||||||
|
(clock::display_eorzea_time(&clock::set_hm_from_float(&clock::get_current_eorzea_date(), fish.entry.end_hour.unwrap())))
|
||||||
|
} @else {
|
||||||
|
"always up!"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
div { "Rarity: " (format!("{:.2}", fish.rarity * 100.)) "%" }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,21 @@ section {
|
||||||
padding: 0 10px;
|
padding: 0 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
padding: 5px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title h3 {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title .subtitle {
|
||||||
|
margin: 0;
|
||||||
|
font-size: 14px;
|
||||||
|
color: gray;
|
||||||
|
}
|
||||||
|
|
||||||
section.up {
|
section.up {
|
||||||
background-color: greenyellow;
|
background-color: greenyellow;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue