joecalsend/src/app/mod.rs

279 lines
8.7 KiB
Rust
Raw Normal View History

use std::{collections::BTreeMap, net::SocketAddr, time::Duration};
use crossterm::event::{Event, EventStream, KeyCode, KeyEvent, KeyEventKind};
use futures::{FutureExt, StreamExt};
use joecalsend::{
Config, JoecalService, JoecalUploadRequest, Listeners, TransferEvent, UploadDialog,
error::Result, models::Device,
};
2025-08-01 16:16:05 +00:00
use julid::Julid;
use log::{LevelFilter, debug, error, info, warn};
2025-08-01 20:52:13 +00:00
use ratatui::{DefaultTerminal, Frame, widgets::TableState};
use tokio::{
sync::mpsc::{UnboundedReceiver, unbounded_channel},
task::JoinSet,
};
2025-07-28 22:51:00 +00:00
pub mod widgets;
pub type Peers = BTreeMap<SocketAddr, (String, String)>;
pub struct App {
pub service: JoecalService,
pub screen: Vec<CurrentScreen>,
pub events: EventStream,
// addr -> (alias, fingerprint)
pub peers: Peers,
2025-08-01 16:16:05 +00:00
pub uploads: BTreeMap<Julid, JoecalUploadRequest>,
2025-08-01 20:52:13 +00:00
upload_state: TableState,
// for getting messages back from the web server or web client about things we've done; the
// other end is held by the state
event_listener: UnboundedReceiver<TransferEvent>,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum CurrentScreen {
Main,
Sending,
Receiving,
Stopping,
Logging,
}
#[tokio::main]
pub async fn start_and_run(
terminal: &mut DefaultTerminal,
config: Config,
device: Device,
) -> Result<()> {
let (event_tx, event_listener) = unbounded_channel();
let service = JoecalService::new(device, event_tx)
.await
.expect("Could not create JoecalService");
let mut app = App::new(service, event_listener);
let mut handles = JoinSet::new();
app.service.start(&config, &mut handles).await;
loop {
terminal.draw(|frame| app.draw(frame))?;
app.handle_events().await?;
if let Some(&top) = app.screen.last()
&& top == CurrentScreen::Stopping
{
app.service.stop().await;
break;
}
let peers = app.service.peers.lock().await;
app.peers.clear();
peers.iter().for_each(|(fingerprint, (addr, device))| {
let alias = device.alias.clone();
app.peers
.insert(addr.to_owned(), (alias, fingerprint.to_owned()));
});
let mut stale_uploads = Vec::new();
let now = chrono::Utc::now().timestamp_millis() as u64;
for (id, request) in app.uploads.iter() {
if request.tx.is_closed() || (now - id.timestamp()) > 60_000 {
stale_uploads.push(*id);
}
}
for id in stale_uploads {
app.uploads.remove(&id);
}
}
shutdown(&mut handles).await;
Ok(())
}
impl App {
pub fn new(service: JoecalService, event_listener: UnboundedReceiver<TransferEvent>) -> Self {
App {
service,
event_listener,
screen: vec![CurrentScreen::Main],
events: Default::default(),
2025-08-01 20:52:13 +00:00
peers: Default::default(),
2025-08-01 16:16:05 +00:00
uploads: Default::default(),
2025-08-01 20:52:13 +00:00
upload_state: Default::default(),
}
}
async fn handle_events(&mut self) -> Result<()> {
tokio::select! {
event = self.events.next().fuse() => {
if let Some(Ok(evt)) = event {
match evt {
Event::Key(key)
if key.kind == KeyEventKind::Press
=> self.handle_key_event(key),
Event::Mouse(_) => {}
Event::Resize(_, _) => {}
_ => {}
}
}
}
transfer_event = self.event_listener.recv() => {
if let Some(event) = transfer_event {
2025-08-01 17:24:25 +00:00
debug!("got transferr event {event:?}");
match event {
2025-08-01 16:16:05 +00:00
TransferEvent::UploadRequest { id, request } => {
self.uploads.insert(id, request);
}
2025-08-01 23:33:00 +00:00
TransferEvent::Cancelled(id) | TransferEvent::Received(id) => {
2025-08-01 16:16:05 +00:00
self.uploads.remove(&id);
}
}
}
}
2025-07-15 22:46:13 +00:00
_ = tokio::time::sleep(Duration::from_millis(200)) => {}
}
Ok(())
}
fn handle_key_event(&mut self, key_event: KeyEvent) {
2025-07-30 05:04:20 +00:00
match self.screen.last().unwrap() {
CurrentScreen::Logging => match key_event.code {
KeyCode::Esc => self.pop(),
KeyCode::Left => change_log_level(-1),
KeyCode::Right => change_log_level(1),
KeyCode::Char('q') => self.exit(),
_ => {}
},
CurrentScreen::Receiving => match key_event.code {
KeyCode::Up => self.upload_state.select_previous(),
KeyCode::Down => self.upload_state.select_next(),
KeyCode::Char('a') => self.accept(),
KeyCode::Char('d') => self.deny(),
KeyCode::Esc => self.pop(),
KeyCode::Char('q') => self.exit(),
_ => {}
},
2025-07-30 05:04:20 +00:00
_ => match key_event.code {
KeyCode::Char('q') => self.exit(),
KeyCode::Char('s') => self.send(),
KeyCode::Char('r') => self.recv(),
KeyCode::Char('l') => self.logs(),
KeyCode::Esc => self.pop(),
_ => {}
},
}
}
fn accept(&mut self) {
let Some(idx) = self.upload_state.selected() else {
return;
};
// keys are sorted, so we can use the table selection index
let keys: Vec<_> = self.uploads.keys().collect();
let Some(key) = keys.get(idx) else {
warn!("could not get id from selection index {idx}");
return;
};
let Some(req) = self.uploads.get(key) else {
return;
};
if let Err(e) = req.tx.send(UploadDialog::UploadConfirm) {
error!("got error sending upload confirmation: {e:?}");
};
}
fn deny(&mut self) {
let Some(idx) = self.upload_state.selected() else {
return;
};
// keys are sorted, so we can use the table selection index
let keys: Vec<_> = self.uploads.keys().cloned().collect();
let Some(key) = keys.get(idx) else {
warn!("could not get id from selection index {idx}");
return;
};
let Some(req) = self.uploads.get(key).cloned() else {
return;
};
if let Err(e) = req.tx.send(UploadDialog::UploadDeny) {
error!("got error sending upload confirmation: {e:?}");
};
self.uploads.remove(key);
}
2025-08-01 20:52:13 +00:00
fn draw(&mut self, frame: &mut Frame) {
2025-07-29 04:49:57 +00:00
frame.render_widget(self, frame.area());
}
fn exit(&mut self) {
self.screen.push(CurrentScreen::Stopping);
}
fn send(&mut self) {
2025-07-15 22:46:13 +00:00
let last = self.screen.last();
match last {
Some(CurrentScreen::Sending) => {}
_ => self.screen.push(CurrentScreen::Sending),
}
}
fn recv(&mut self) {
2025-07-15 22:46:13 +00:00
let last = self.screen.last();
match last {
Some(CurrentScreen::Receiving) => {}
_ => self.screen.push(CurrentScreen::Receiving),
}
2025-07-09 05:27:00 +00:00
}
fn logs(&mut self) {
let last = self.screen.last();
match last {
Some(CurrentScreen::Logging) => {}
_ => self.screen.push(CurrentScreen::Logging),
}
}
2025-07-09 05:27:00 +00:00
fn pop(&mut self) {
2025-07-15 22:46:13 +00:00
self.screen.pop();
if self.screen.last().is_none() {
self.screen.push(CurrentScreen::Main);
2025-07-09 05:27:00 +00:00
}
}
}
2025-07-30 19:41:18 +00:00
fn change_log_level(delta: isize) {
let level = log::max_level() as isize;
let max = log::LevelFilter::max() as isize;
let level = (level + delta).clamp(0, max) as usize;
// levelfilter is repr(usize) so this is safe
let level = unsafe { std::mem::transmute::<usize, LevelFilter>(level) };
log::set_max_level(level);
2025-07-30 05:04:20 +00:00
}
2025-07-16 00:28:40 +00:00
async fn shutdown(handles: &mut JoinSet<Listeners>) {
let mut alarm = tokio::time::interval(tokio::time::Duration::from_secs(5));
alarm.tick().await;
loop {
tokio::select! {
join_result = handles.join_next() => {
match join_result {
Some(handle) => match handle {
Ok(h) => info!("Stopped {h:?}"),
Err(e) => error!("Got error {e:?}"),
2025-07-16 00:28:40 +00:00
}
None => break,
}
}
_ = alarm.tick() => {
info!("Exit timeout reached, aborting all unjoined tasks");
2025-07-16 00:28:40 +00:00
handles.abort_all();
break;
},
}
}
}