chore: some cleanup
parent
d1245ab365
commit
7c06906718
|
@ -1,9 +1,10 @@
|
||||||
mod models;
|
mod models;
|
||||||
|
|
||||||
pub use models::*;
|
|
||||||
use chrono::NaiveDate;
|
use chrono::NaiveDate;
|
||||||
|
pub use models::*;
|
||||||
|
|
||||||
const USER_AGENT: &str = "User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:109.0) Gecko/20100101 Firefox/112.0";
|
const USER_AGENT: &str =
|
||||||
|
"User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:109.0) Gecko/20100101 Firefox/112.0";
|
||||||
|
|
||||||
pub struct AffluencesClient {
|
pub struct AffluencesClient {
|
||||||
client: reqwest::Client,
|
client: reqwest::Client,
|
||||||
|
@ -13,22 +14,61 @@ impl AffluencesClient {
|
||||||
pub fn new() -> Self {
|
pub fn new() -> Self {
|
||||||
Self {
|
Self {
|
||||||
client: reqwest::Client::builder()
|
client: reqwest::Client::builder()
|
||||||
.user_agent(USER_AGENT).build().unwrap(),
|
.user_agent(USER_AGENT)
|
||||||
|
.build()
|
||||||
|
.unwrap(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn available(&self, site_id: uuid::Uuid, date: NaiveDate, resource_type: u32) -> reqwest::Result<Vec<Resource>> {
|
pub async fn available(
|
||||||
let url = format!("https://reservation.affluences.com/api/resources/{}/available", site_id);
|
&self,
|
||||||
self.client.get(url).query(&[("date", date.format("%Y-%m-%d").to_string()), ("type", resource_type.to_string())]).send().await?.json::<Vec<Resource>>().await
|
site_id: uuid::Uuid,
|
||||||
|
date: NaiveDate,
|
||||||
|
resource_type: u32,
|
||||||
|
) -> reqwest::Result<Vec<Resource>> {
|
||||||
|
let url = format!(
|
||||||
|
"https://reservation.affluences.com/api/resources/{}/available",
|
||||||
|
site_id
|
||||||
|
);
|
||||||
|
self.client
|
||||||
|
.get(url)
|
||||||
|
.query(&[
|
||||||
|
("date", date.format("%Y-%m-%d").to_string()),
|
||||||
|
("type", resource_type.to_string()),
|
||||||
|
])
|
||||||
|
.send()
|
||||||
|
.await?
|
||||||
|
.json::<Vec<Resource>>()
|
||||||
|
.await
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn site_data(&self, slug: &str) -> reqwest::Result<SiteData> {
|
pub async fn site_data(&self, slug: &str) -> reqwest::Result<SiteData> {
|
||||||
let url = format!("https://api.affluences.com/app/v3/sites/{}", slug);
|
let url = format!("https://api.affluences.com/app/v3/sites/{}", slug);
|
||||||
Ok(self.client.get(url).send().await?.json::<Data<SiteData>>().await?.data)
|
Ok(self
|
||||||
|
.client
|
||||||
|
.get(url)
|
||||||
|
.send()
|
||||||
|
.await?
|
||||||
|
.json::<Data<SiteData>>()
|
||||||
|
.await?
|
||||||
|
.data)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn make_reservation(&self, resource_id: u32, reservation: &Reservation) -> reqwest::Result<ReservationResponse> {
|
pub async fn make_reservation(
|
||||||
let url = format!("https://reservation.affluences.com/api/reserve/{}", resource_id);
|
&self,
|
||||||
self.client.post(url).json(reservation).send().await?.json::<ReservationResponse>().await
|
resource_id: u32,
|
||||||
|
reservation: &Reservation,
|
||||||
|
) -> reqwest::Result<ReservationResponse> {
|
||||||
|
let url = format!(
|
||||||
|
"https://reservation.affluences.com/api/reserve/{}",
|
||||||
|
resource_id
|
||||||
|
);
|
||||||
|
self.client
|
||||||
|
.post(url)
|
||||||
|
.json(reservation)
|
||||||
|
.send()
|
||||||
|
.await?
|
||||||
|
.json::<ReservationResponse>()
|
||||||
|
.await
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,206 +0,0 @@
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
use chrono::{NaiveTime, NaiveDate};
|
|
||||||
|
|
||||||
mod time_format {
|
|
||||||
use chrono::NaiveTime;
|
|
||||||
use serde::{self, Deserialize, Serializer, Deserializer};
|
|
||||||
|
|
||||||
const FORMAT: &'static str = "%H:%M";
|
|
||||||
|
|
||||||
pub fn serialize<S>(
|
|
||||||
time: &NaiveTime,
|
|
||||||
serializer: S,
|
|
||||||
) -> Result<S::Ok, S::Error>
|
|
||||||
where
|
|
||||||
S: Serializer,
|
|
||||||
{
|
|
||||||
let s = format!("{}", time.format(FORMAT));
|
|
||||||
serializer.serialize_str(&s)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn deserialize<'de, D>(
|
|
||||||
deserializer: D,
|
|
||||||
) -> Result<NaiveTime, D::Error>
|
|
||||||
where
|
|
||||||
D: Deserializer<'de>,
|
|
||||||
{
|
|
||||||
let s = String::deserialize(deserializer)?;
|
|
||||||
NaiveTime::parse_from_str(&s, FORMAT).map_err(serde::de::Error::custom)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug, Clone, Copy)]
|
|
||||||
pub struct Hour {
|
|
||||||
#[serde(with = "time_format")]
|
|
||||||
pub hour: NaiveTime,
|
|
||||||
pub state: u32,
|
|
||||||
// reservations
|
|
||||||
pub granularity: u32,
|
|
||||||
pub person_count: u32,
|
|
||||||
pub places_available: u32,
|
|
||||||
pub places_bookable: u32,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct Resource {
|
|
||||||
pub resource_id: u32,
|
|
||||||
pub resource_name: String,
|
|
||||||
pub resource_type: u32,
|
|
||||||
pub granularity: u32,
|
|
||||||
pub time_slot_count: u32,
|
|
||||||
pub static_time_slot: bool,
|
|
||||||
// reservations_by_timeslot
|
|
||||||
pub note_available: bool,
|
|
||||||
pub note_required: bool,
|
|
||||||
pub note_description: String,
|
|
||||||
pub description: String,
|
|
||||||
pub capacity: u32,
|
|
||||||
pub site_timezone: String,
|
|
||||||
pub user_name_required: bool,
|
|
||||||
pub user_phone_required: bool,
|
|
||||||
pub user_name_available: bool,
|
|
||||||
pub user_phone_available: bool,
|
|
||||||
// time_before_reservations_closed
|
|
||||||
// min_places_per_reservation
|
|
||||||
// max_places_per_reservation
|
|
||||||
pub image_url: Option<String>,
|
|
||||||
// services
|
|
||||||
pub slots_state: u32,
|
|
||||||
pub hours: Vec<Hour>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct Data<T> {
|
|
||||||
pub data: T
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct SiteDataCategory {
|
|
||||||
pub id: u32,
|
|
||||||
pub name: String,
|
|
||||||
pub name_plural: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct SiteDataLocationCoordinates {
|
|
||||||
pub latitude: f64,
|
|
||||||
pub longitude: f64,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct SiteDataLocationAddress {
|
|
||||||
pub route: String,
|
|
||||||
pub city: String,
|
|
||||||
pub zip_code: String,
|
|
||||||
pub region: String,
|
|
||||||
pub country_code: String
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct SiteDataLocation {
|
|
||||||
pub coordinates: SiteDataLocationCoordinates,
|
|
||||||
pub address: SiteDataLocationAddress
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct SiteDataForecast {
|
|
||||||
pub opened: bool,
|
|
||||||
pub occupancy: u32,
|
|
||||||
// waiting_time
|
|
||||||
pub waiting_time_overflow: bool
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct SiteDataNotice {
|
|
||||||
pub message: String,
|
|
||||||
pub url: Option<String>
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct SiteDataService {
|
|
||||||
pub id: u32,
|
|
||||||
pub name: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct SiteDataInfo {
|
|
||||||
pub title: String,
|
|
||||||
pub description: String,
|
|
||||||
pub url: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct SiteDataStatus {
|
|
||||||
pub state: String,
|
|
||||||
pub text: String,
|
|
||||||
pub color: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct SiteData {
|
|
||||||
pub id: uuid::Uuid,
|
|
||||||
pub slug: String,
|
|
||||||
pub parent: Option<String>,
|
|
||||||
pub primary_name: String,
|
|
||||||
pub secondary_name: String,
|
|
||||||
pub concat_name: String,
|
|
||||||
pub categories: Vec<SiteDataCategory>,
|
|
||||||
pub time_zone: String,
|
|
||||||
pub location: SiteDataLocation,
|
|
||||||
pub phone_number: Option<String>,
|
|
||||||
pub email: Option<String>,
|
|
||||||
pub url: Option<String>,
|
|
||||||
pub notices: Vec<SiteDataNotice>,
|
|
||||||
// messages
|
|
||||||
pub estimated_distance: f64,
|
|
||||||
pub current_forecast: SiteDataForecast,
|
|
||||||
pub today_forecasts: Vec<SiteDataForecast>,
|
|
||||||
// events
|
|
||||||
pub children: Vec<SiteData>,
|
|
||||||
// actions
|
|
||||||
pub services: Vec<SiteDataService>,
|
|
||||||
pub infos: Vec<SiteDataInfo>,
|
|
||||||
pub poster_image: String,
|
|
||||||
pub image: Option<Vec<String>>,
|
|
||||||
// status
|
|
||||||
pub closed: bool,
|
|
||||||
pub booking_available: bool,
|
|
||||||
pub extended_forecasts: bool,
|
|
||||||
pub booking_url: Option<String>,
|
|
||||||
pub validated: bool,
|
|
||||||
#[serde(rename = "validationStatus")]
|
|
||||||
pub validation_status: String,
|
|
||||||
#[serde(rename = "publicationStatus")]
|
|
||||||
pub publication_status: String
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Serialize, Debug)]
|
|
||||||
pub struct Reservation {
|
|
||||||
// This string might not be correct
|
|
||||||
pub auth_type: Option<String>,
|
|
||||||
pub email: String,
|
|
||||||
pub date: NaiveDate,
|
|
||||||
#[serde(with = "time_format")]
|
|
||||||
pub start_time: NaiveTime,
|
|
||||||
#[serde(with = "time_format")]
|
|
||||||
pub end_time: NaiveTime,
|
|
||||||
pub note: String,
|
|
||||||
pub user_firstname: String,
|
|
||||||
pub user_lastname: String,
|
|
||||||
pub user_phone: Option<String>,
|
|
||||||
pub person_count: u32
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
|
||||||
pub struct ReservationResponse {
|
|
||||||
pub reservation_id: u32,
|
|
||||||
// This string might not be correct
|
|
||||||
pub auth_type: Option<String>,
|
|
||||||
pub user_validation: bool,
|
|
||||||
// ticket_payload
|
|
||||||
pub email: String,
|
|
||||||
pub success: String,
|
|
||||||
#[serde(rename = "successMessage")]
|
|
||||||
pub success_message: String,
|
|
||||||
// cancellation_token
|
|
||||||
}
|
|
|
@ -0,0 +1,43 @@
|
||||||
|
use super::hh_mm_time_format;
|
||||||
|
use chrono::NaiveTime;
|
||||||
|
use serde::Deserialize;
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug, Clone, Copy)]
|
||||||
|
pub struct HourBlock {
|
||||||
|
#[serde(with = "hh_mm_time_format")]
|
||||||
|
pub hour: NaiveTime,
|
||||||
|
pub state: u32,
|
||||||
|
// reservations
|
||||||
|
pub granularity: u32,
|
||||||
|
pub person_count: u32,
|
||||||
|
pub places_available: u32,
|
||||||
|
pub places_bookable: u32,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct Resource {
|
||||||
|
pub resource_id: u32,
|
||||||
|
pub resource_name: String,
|
||||||
|
pub resource_type: u32,
|
||||||
|
pub granularity: u32,
|
||||||
|
pub time_slot_count: u32,
|
||||||
|
pub static_time_slot: bool,
|
||||||
|
// reservations_by_timeslot
|
||||||
|
pub note_available: bool,
|
||||||
|
pub note_required: bool,
|
||||||
|
pub note_description: String,
|
||||||
|
pub description: String,
|
||||||
|
pub capacity: u32,
|
||||||
|
pub site_timezone: String,
|
||||||
|
pub user_name_required: bool,
|
||||||
|
pub user_phone_required: bool,
|
||||||
|
pub user_name_available: bool,
|
||||||
|
pub user_phone_available: bool,
|
||||||
|
// time_before_reservations_closed
|
||||||
|
// min_places_per_reservation
|
||||||
|
// max_places_per_reservation
|
||||||
|
pub image_url: Option<String>,
|
||||||
|
// services
|
||||||
|
pub slots_state: u32,
|
||||||
|
pub hours: Vec<HourBlock>,
|
||||||
|
}
|
|
@ -0,0 +1,20 @@
|
||||||
|
use chrono::NaiveTime;
|
||||||
|
use serde::{self, Deserialize, Deserializer, Serializer};
|
||||||
|
|
||||||
|
const FORMAT: &'static str = "%H:%M";
|
||||||
|
|
||||||
|
pub fn serialize<S>(time: &NaiveTime, serializer: S) -> Result<S::Ok, S::Error>
|
||||||
|
where
|
||||||
|
S: Serializer,
|
||||||
|
{
|
||||||
|
let s = format!("{}", time.format(FORMAT));
|
||||||
|
serializer.serialize_str(&s)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn deserialize<'de, D>(deserializer: D) -> Result<NaiveTime, D::Error>
|
||||||
|
where
|
||||||
|
D: Deserializer<'de>,
|
||||||
|
{
|
||||||
|
let s = String::deserialize(deserializer)?;
|
||||||
|
NaiveTime::parse_from_str(&s, FORMAT).map_err(serde::de::Error::custom)
|
||||||
|
}
|
|
@ -0,0 +1,8 @@
|
||||||
|
mod available;
|
||||||
|
mod hh_mm_time_format;
|
||||||
|
mod reservation;
|
||||||
|
mod site_data;
|
||||||
|
|
||||||
|
pub use available::*;
|
||||||
|
pub use reservation::*;
|
||||||
|
pub use site_data::*;
|
|
@ -0,0 +1,34 @@
|
||||||
|
use super::hh_mm_time_format;
|
||||||
|
use chrono::{NaiveDate, NaiveTime};
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
#[derive(Serialize, Debug)]
|
||||||
|
pub struct Reservation {
|
||||||
|
// This string might not be correct
|
||||||
|
pub auth_type: Option<String>,
|
||||||
|
pub email: String,
|
||||||
|
pub date: NaiveDate,
|
||||||
|
#[serde(with = "hh_mm_time_format")]
|
||||||
|
pub start_time: NaiveTime,
|
||||||
|
#[serde(with = "hh_mm_time_format")]
|
||||||
|
pub end_time: NaiveTime,
|
||||||
|
pub note: String,
|
||||||
|
pub user_firstname: String,
|
||||||
|
pub user_lastname: String,
|
||||||
|
pub user_phone: Option<String>,
|
||||||
|
pub person_count: u32,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct ReservationResponse {
|
||||||
|
pub reservation_id: u32,
|
||||||
|
// This string might not be correct
|
||||||
|
pub auth_type: Option<String>,
|
||||||
|
pub user_validation: bool,
|
||||||
|
// ticket_payload
|
||||||
|
pub email: String,
|
||||||
|
pub success: String,
|
||||||
|
#[serde(rename = "successMessage")]
|
||||||
|
pub success_message: String,
|
||||||
|
// cancellation_token
|
||||||
|
}
|
|
@ -0,0 +1,106 @@
|
||||||
|
use serde::Deserialize;
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct Data<T> {
|
||||||
|
pub data: T,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct SiteDataCategory {
|
||||||
|
pub id: u32,
|
||||||
|
pub name: String,
|
||||||
|
pub name_plural: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct SiteDataLocationCoordinates {
|
||||||
|
pub latitude: f64,
|
||||||
|
pub longitude: f64,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct SiteDataLocationAddress {
|
||||||
|
pub route: String,
|
||||||
|
pub city: String,
|
||||||
|
pub zip_code: String,
|
||||||
|
pub region: String,
|
||||||
|
pub country_code: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct SiteDataLocation {
|
||||||
|
pub coordinates: SiteDataLocationCoordinates,
|
||||||
|
pub address: SiteDataLocationAddress,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct SiteDataForecast {
|
||||||
|
pub opened: bool,
|
||||||
|
pub occupancy: u32,
|
||||||
|
// waiting_time
|
||||||
|
pub waiting_time_overflow: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct SiteDataNotice {
|
||||||
|
pub message: String,
|
||||||
|
pub url: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct SiteDataService {
|
||||||
|
pub id: u32,
|
||||||
|
pub name: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct SiteDataInfo {
|
||||||
|
pub title: String,
|
||||||
|
pub description: String,
|
||||||
|
pub url: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct SiteDataStatus {
|
||||||
|
pub state: String,
|
||||||
|
pub text: String,
|
||||||
|
pub color: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Debug)]
|
||||||
|
pub struct SiteData {
|
||||||
|
pub id: uuid::Uuid,
|
||||||
|
pub slug: String,
|
||||||
|
pub parent: Option<String>,
|
||||||
|
pub primary_name: String,
|
||||||
|
pub secondary_name: String,
|
||||||
|
pub concat_name: String,
|
||||||
|
pub categories: Vec<SiteDataCategory>,
|
||||||
|
pub time_zone: String,
|
||||||
|
pub location: SiteDataLocation,
|
||||||
|
pub phone_number: Option<String>,
|
||||||
|
pub email: Option<String>,
|
||||||
|
pub url: Option<String>,
|
||||||
|
pub notices: Vec<SiteDataNotice>,
|
||||||
|
// messages
|
||||||
|
pub estimated_distance: f64,
|
||||||
|
pub current_forecast: SiteDataForecast,
|
||||||
|
pub today_forecasts: Vec<SiteDataForecast>,
|
||||||
|
// events
|
||||||
|
pub children: Vec<SiteData>,
|
||||||
|
// actions
|
||||||
|
pub services: Vec<SiteDataService>,
|
||||||
|
pub infos: Vec<SiteDataInfo>,
|
||||||
|
pub poster_image: String,
|
||||||
|
pub image: Option<Vec<String>>,
|
||||||
|
// status
|
||||||
|
pub closed: bool,
|
||||||
|
pub booking_available: bool,
|
||||||
|
pub extended_forecasts: bool,
|
||||||
|
pub booking_url: Option<String>,
|
||||||
|
pub validated: bool,
|
||||||
|
#[serde(rename = "validationStatus")]
|
||||||
|
pub validation_status: String,
|
||||||
|
#[serde(rename = "publicationStatus")]
|
||||||
|
pub publication_status: String,
|
||||||
|
}
|
|
@ -1,8 +1,7 @@
|
||||||
use crate::{Context, Error};
|
use crate::{Context, Error};
|
||||||
use chrono::{NaiveDate, Duration};
|
use affluences_api::HourBlock;
|
||||||
|
use chrono::{Duration, NaiveDate};
|
||||||
use uuid::{uuid, Uuid};
|
use uuid::{uuid, Uuid};
|
||||||
use affluences_api::Hour;
|
|
||||||
use poise::serenity_prelude as serenity;
|
|
||||||
|
|
||||||
const STERRE_BIB_ID: Uuid = uuid!("4737e57a-ee05-4f7b-901a-7bb541eeb297");
|
const STERRE_BIB_ID: Uuid = uuid!("4737e57a-ee05-4f7b-901a-7bb541eeb297");
|
||||||
const TIME_FORMAT: &'static str = "%H:%M";
|
const TIME_FORMAT: &'static str = "%H:%M";
|
||||||
|
@ -85,24 +84,25 @@ pub async fn getvotes(
|
||||||
|
|
||||||
/// List available timeslots for day
|
/// List available timeslots for day
|
||||||
#[poise::command(prefix_command, slash_command)]
|
#[poise::command(prefix_command, slash_command)]
|
||||||
pub async fn available(
|
pub async fn available(ctx: Context<'_>, date: NaiveDate) -> Result<(), Error> {
|
||||||
ctx: Context<'_>,
|
|
||||||
date: NaiveDate,
|
|
||||||
) -> Result<(), Error> {
|
|
||||||
let client = &ctx.data().client;
|
let client = &ctx.data().client;
|
||||||
let resources = client.available(STERRE_BIB_ID, date, 1).await?;
|
let resources = client.available(STERRE_BIB_ID, date, 1).await?;
|
||||||
let mut fields: Vec<(String, String, bool)> = Default::default();
|
let mut fields: Vec<(String, String, bool)> = Default::default();
|
||||||
|
|
||||||
for resource in &resources {
|
for resource in &resources {
|
||||||
if resource.hours.len() == 0 {
|
if resource.hours.len() == 0 {
|
||||||
fields.push((resource.resource_name.clone(), "Nothing available.".to_string(), false));
|
fields.push((
|
||||||
|
resource.resource_name.clone(),
|
||||||
|
"Nothing available.".to_string(),
|
||||||
|
false,
|
||||||
|
));
|
||||||
|
|
||||||
continue
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut lines: Vec<String> = Default::default();
|
let mut lines: Vec<String> = Default::default();
|
||||||
|
|
||||||
let mut start_hour_opt: Option<&Hour> = None;
|
let mut start_hour_opt: Option<&HourBlock> = None;
|
||||||
let mut duration = Duration::seconds(0);
|
let mut duration = Duration::seconds(0);
|
||||||
|
|
||||||
for hour in &resource.hours {
|
for hour in &resource.hours {
|
||||||
|
@ -111,7 +111,13 @@ pub async fn available(
|
||||||
duration = duration + Duration::minutes(hour.granularity.into());
|
duration = duration + Duration::minutes(hour.granularity.into());
|
||||||
} else {
|
} else {
|
||||||
let end_hour = start_hour.hour + duration;
|
let end_hour = start_hour.hour + duration;
|
||||||
lines.push(format!("{} - {} ({:02}:{:02})", start_hour.hour.format(TIME_FORMAT), end_hour.format(TIME_FORMAT), duration.num_hours(), duration.num_minutes() % 60));
|
lines.push(format!(
|
||||||
|
"{} - {} ({:02}:{:02})",
|
||||||
|
start_hour.hour.format(TIME_FORMAT),
|
||||||
|
end_hour.format(TIME_FORMAT),
|
||||||
|
duration.num_hours(),
|
||||||
|
duration.num_minutes() % 60
|
||||||
|
));
|
||||||
start_hour_opt = None;
|
start_hour_opt = None;
|
||||||
}
|
}
|
||||||
} else if hour.state == 1 {
|
} else if hour.state == 1 {
|
||||||
|
@ -123,16 +129,32 @@ pub async fn available(
|
||||||
// Print final entry if present
|
// Print final entry if present
|
||||||
if let Some(start_hour) = start_hour_opt {
|
if let Some(start_hour) = start_hour_opt {
|
||||||
let end_hour = start_hour.hour + duration;
|
let end_hour = start_hour.hour + duration;
|
||||||
lines.push(format!("{} - {} ({:02}:{:02})", start_hour.hour.format(TIME_FORMAT), end_hour.format(TIME_FORMAT), duration.num_hours(), duration.num_minutes() % 60));
|
lines.push(format!(
|
||||||
|
"{} - {} ({:02}:{:02})",
|
||||||
|
start_hour.hour.format(TIME_FORMAT),
|
||||||
|
end_hour.format(TIME_FORMAT),
|
||||||
|
duration.num_hours(),
|
||||||
|
duration.num_minutes() % 60
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
fields.push((resource.resource_name.clone(), lines.join("\n"), false));
|
fields.push((resource.resource_name.clone(), lines.join("\n"), false));
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx.send(|f|
|
ctx.send(|f| {
|
||||||
f.embed(|e|
|
f.embed(|e| {
|
||||||
e.description(format!("Available booking dates for {}.", date))
|
e.description(format!("Available booking dates for {}.", date))
|
||||||
.fields(fields))).await?;
|
.fields(fields)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.await?;
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Create a reservation
|
||||||
|
// #[poise::command(prefix_command, slash_command)]
|
||||||
|
// pub async fn reserve(
|
||||||
|
// ctx: Context<'_>,
|
||||||
|
// date: NaiveDate,
|
||||||
|
// ) -> Result<(), Error> {
|
||||||
|
|
|
@ -36,7 +36,12 @@ async fn main() {
|
||||||
// FrameworkOptions contains all of poise's configuration option in one struct
|
// FrameworkOptions contains all of poise's configuration option in one struct
|
||||||
// Every option can be omitted to use its default value
|
// Every option can be omitted to use its default value
|
||||||
let options = poise::FrameworkOptions {
|
let options = poise::FrameworkOptions {
|
||||||
commands: vec![commands::help(), commands::vote(), commands::getvotes(), commands::available()],
|
commands: vec![
|
||||||
|
commands::help(),
|
||||||
|
commands::vote(),
|
||||||
|
commands::getvotes(),
|
||||||
|
commands::available(),
|
||||||
|
],
|
||||||
prefix_options: poise::PrefixFrameworkOptions {
|
prefix_options: poise::PrefixFrameworkOptions {
|
||||||
prefix: Some("~".into()),
|
prefix: Some("~".into()),
|
||||||
edit_tracker: Some(poise::EditTracker::for_timespan(Duration::from_secs(3600))),
|
edit_tracker: Some(poise::EditTracker::for_timespan(Duration::from_secs(3600))),
|
||||||
|
|
Loading…
Reference in New Issue