wip
This commit is contained in:
83
'
83
'
@@ -1,83 +0,0 @@
|
|||||||
use std::{
|
|
||||||
sync::{Arc, Mutex},
|
|
||||||
thread,
|
|
||||||
};
|
|
||||||
|
|
||||||
use chrono::Local;
|
|
||||||
use cron_tab::Cron;
|
|
||||||
use gpio_cdev::{Chip, LineRequestFlags};
|
|
||||||
|
|
||||||
use ringer::Ringer;
|
|
||||||
|
|
||||||
use crate::{ringer::BeepRinger, scheduler::Scheduler};
|
|
||||||
|
|
||||||
mod handler;
|
|
||||||
mod router;
|
|
||||||
mod scheduler;
|
|
||||||
mod ringer;
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
struct AppState {
|
|
||||||
chip: Arc<Mutex<Chip>>,
|
|
||||||
cron: Arc<Mutex<Cron<Local>>>,
|
|
||||||
scheduler: Arc<Scheduler<BeepRinger>>
|
|
||||||
}
|
|
||||||
|
|
||||||
fn app_state() -> AppState {
|
|
||||||
let chip = Arc::new(Mutex::new(Chip::new("/dev/gpiochip0").unwrap()));
|
|
||||||
let cron = Arc::new(Mutex::new(Cron::new(chrono::Local)));
|
|
||||||
AppState {
|
|
||||||
chip: chip.clone(),
|
|
||||||
cron: cron.clone(),
|
|
||||||
scheduler: Arc::new(Scheduler::new(
|
|
||||||
BeepRinger::new(chip.clone()),
|
|
||||||
cron.clone()
|
|
||||||
))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tokio::main]
|
|
||||||
async fn main() -> Result<(), Box<dyn std::error::Error>> {
|
|
||||||
let chip = Arc::new(Mutex::new(Chip::new("/dev/gpiochip0")?));
|
|
||||||
let mut cron = Cron::new(chrono::Local);
|
|
||||||
let static_time = "0 30 7 * * *";
|
|
||||||
let app_state = app_state();
|
|
||||||
|
|
||||||
println!("Adding alarm {}", static_time);
|
|
||||||
let job_result = cron.add_fn(static_time, move || {
|
|
||||||
let mut guard = chip.lock().unwrap();
|
|
||||||
alarm(&mut *guard).unwrap();
|
|
||||||
});
|
|
||||||
|
|
||||||
let _ = job_result.expect("Failed to add job");
|
|
||||||
|
|
||||||
cron.start();
|
|
||||||
|
|
||||||
start_server(app_state).await;
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
async fn start_server(app_state: AppState) {
|
|
||||||
let router = router::router(app_state);
|
|
||||||
let listener = tokio::net::TcpListener::bind("0.0.0.0:8080").await.unwrap();
|
|
||||||
|
|
||||||
println!("Listening on http://0.0.0.0:8080");
|
|
||||||
axum::serve(listener, router).await.unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
fn alarm(chip: &mut Chip) -> Result<(), Box<dyn std::error::Error>> {
|
|
||||||
let beeper = chip
|
|
||||||
.get_line(17)?
|
|
||||||
.request(LineRequestFlags::OUTPUT, 0, "my-gpio")?;
|
|
||||||
|
|
||||||
for _ in 0..5 {
|
|
||||||
beeper.set_value(1)?;
|
|
||||||
thread::sleep(std::time::Duration::from_secs(1));
|
|
||||||
beeper.set_value(0)?;
|
|
||||||
thread::sleep(std::time::Duration::from_secs(1));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
2
Cargo.lock
generated
2
Cargo.lock
generated
@@ -586,6 +586,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "9a8e94ea7f378bd32cbbd37198a4a91436180c5bb472411e48b5ec2e2124ae9e"
|
checksum = "9a8e94ea7f378bd32cbbd37198a4a91436180c5bb472411e48b5ec2e2124ae9e"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"serde_core",
|
"serde_core",
|
||||||
|
"serde_derive",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -679,6 +680,7 @@ dependencies = [
|
|||||||
"chrono",
|
"chrono",
|
||||||
"cron_tab",
|
"cron_tab",
|
||||||
"gpio-cdev",
|
"gpio-cdev",
|
||||||
|
"serde",
|
||||||
"tokio",
|
"tokio",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|||||||
@@ -9,3 +9,4 @@ cron_tab = { version = "0.2", features = ["async"] }
|
|||||||
chrono = "0.4.42"
|
chrono = "0.4.42"
|
||||||
axum = { version = "0.8.8", features = ["macros"] }
|
axum = { version = "0.8.8", features = ["macros"] }
|
||||||
tokio = { version = "1.49.0", features = ["full"] }
|
tokio = { version = "1.49.0", features = ["full"] }
|
||||||
|
serde = { version = "1.0.228", features = ["derive"] }
|
||||||
|
|||||||
@@ -1,4 +1,20 @@
|
|||||||
#[axum::debug_handler]
|
use axum::{Json, extract::State};
|
||||||
pub async fn create_alarm() -> String {
|
use serde::{Deserialize, Serialize};
|
||||||
"hello world".to_string()
|
|
||||||
|
use crate::{AppState};
|
||||||
|
|
||||||
|
#[derive(Deserialize, Serialize)]
|
||||||
|
pub struct CreateAlarmRequest {
|
||||||
|
hour: String,
|
||||||
|
minute: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[axum::debug_handler]
|
||||||
|
pub async fn create_alarm(State(state): State<AppState>, Json(alarm): Json<CreateAlarmRequest>) -> String {
|
||||||
|
let result = state.scheduler.add_alarm(alarm.hour.as_str(), alarm.minute.as_str());
|
||||||
|
|
||||||
|
match result {
|
||||||
|
Ok(_) => "Alarm created".to_string(),
|
||||||
|
Err(e) => e.to_string(),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ use std::{
|
|||||||
sync::{Arc, Mutex},
|
sync::{Arc, Mutex},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use chrono::{DateTime, Local};
|
||||||
use cron_tab::Cron;
|
use cron_tab::Cron;
|
||||||
use gpio_cdev::Chip;
|
use gpio_cdev::Chip;
|
||||||
|
|
||||||
@@ -12,6 +13,11 @@ mod router;
|
|||||||
mod scheduler;
|
mod scheduler;
|
||||||
mod ringer;
|
mod ringer;
|
||||||
|
|
||||||
|
struct Alarm {
|
||||||
|
enabled: bool,
|
||||||
|
time: DateTime<Local>
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
struct AppState {
|
struct AppState {
|
||||||
scheduler: Arc<Scheduler<BeepRinger>>
|
scheduler: Arc<Scheduler<BeepRinger>>
|
||||||
@@ -30,10 +36,8 @@ fn app_state() -> AppState {
|
|||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() -> Result<(), Box<dyn std::error::Error>> {
|
async fn main() -> Result<(), Box<dyn std::error::Error>> {
|
||||||
let static_time = "0 30 7 * * *";
|
|
||||||
let app_state = app_state();
|
let app_state = app_state();
|
||||||
|
|
||||||
app_state.scheduler.schedule(static_time)?;
|
|
||||||
app_state.scheduler.start();
|
app_state.scheduler.start();
|
||||||
|
|
||||||
start_server(app_state).await;
|
start_server(app_state).await;
|
||||||
|
|||||||
@@ -9,11 +9,16 @@ use crate::ringer::Ringer;
|
|||||||
pub struct Scheduler<T: Ringer + 'static> {
|
pub struct Scheduler<T: Ringer + 'static> {
|
||||||
ringer: Arc<Mutex<T>>,
|
ringer: Arc<Mutex<T>>,
|
||||||
cron: Arc<Mutex<Cron<Local>>>,
|
cron: Arc<Mutex<Cron<Local>>>,
|
||||||
|
alarms: Arc<Mutex<Vec<String>>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: Ringer> Scheduler<T> {
|
impl<T: Ringer> Scheduler<T> {
|
||||||
pub fn new(ringer: Arc<Mutex<T>>, cron: Arc<Mutex<Cron<Local>>>) -> Self {
|
pub fn new(ringer: Arc<Mutex<T>>, cron: Arc<Mutex<Cron<Local>>>) -> Self {
|
||||||
Self { ringer, cron }
|
Self {
|
||||||
|
ringer,
|
||||||
|
cron,
|
||||||
|
alarms: Arc::new(Mutex::new(Vec::new())),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn schedule(&self, cron_schedule: &str) -> Result<(), String> {
|
pub fn schedule(&self, cron_schedule: &str) -> Result<(), String> {
|
||||||
@@ -34,6 +39,15 @@ impl<T: Ringer> Scheduler<T> {
|
|||||||
todo!()
|
todo!()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn add_alarm(&self, hour: &str, minute: &str) -> Result<(), String> {
|
||||||
|
let mut alarms = self.alarms.lock().map_err(|e| e.to_string())?;
|
||||||
|
let cron_schedule = format!("{} {} {} * * *", "*", minute, hour);
|
||||||
|
alarms.push(cron_schedule.clone());
|
||||||
|
println!("Added alarm {}", cron_schedule);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
pub fn start(&self) {
|
pub fn start(&self) {
|
||||||
self.cron.lock().expect("Failed to lock cron").start();
|
self.cron.lock().expect("Failed to lock cron").start();
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user