cabytcini/src/main.rs

144 lines
3.3 KiB
Rust
Raw Normal View History

2020-04-07 12:56:56 +00:00
mod config;
2020-04-07 01:09:09 +00:00
mod xsetroot;
use chrono::prelude::*;
2020-04-07 12:56:56 +00:00
use log::*;
2020-04-07 12:03:07 +00:00
use std::{
fmt,
io::{prelude::*, BufReader},
net::Shutdown,
os::unix::net::{UnixListener, UnixStream},
sync::{Arc, Mutex},
thread, time,
};
2020-04-07 01:09:09 +00:00
use xsetroot::XSetRoot;
2020-04-07 12:56:56 +00:00
pub type MTState = Arc<Mutex<State>>;
2020-04-07 01:09:09 +00:00
fn handle_client(stream: UnixStream, st: MTState) {
let mut rdr = BufReader::new(&stream);
let mut msg = String::new();
let _ = rdr.read_line(&mut msg);
let _ = write!(&stream, "OK");
let mut data = st.lock().unwrap();
2020-04-07 12:03:07 +00:00
data.msg = msg.trim().to_string();
2020-04-07 12:56:56 +00:00
info!("new message: {}", data.msg);
2020-04-07 12:03:07 +00:00
stream.shutdown(Shutdown::Both).expect("socket to close");
2020-04-07 01:09:09 +00:00
}
2020-04-07 12:03:07 +00:00
const UPDATE_FREQUENCY: u64 = 250;
fn update_every_so_often(st: MTState) {
loop {
let so_often = time::Duration::from_millis(UPDATE_FREQUENCY);
thread::sleep(so_often);
{
let data = st.lock().unwrap();
data.show();
}
}
}
2020-04-07 12:56:56 +00:00
const FIVE_MINUTES: u64 = 60 * 5;
fn front_update(st: MTState, cfg: config::Config) {
loop {
match reqwest::blocking::get(&cfg.front_url) {
Ok(who) => {
let mut data = st.lock().unwrap();
let who = who.text().unwrap().trim().to_string();
if who != data.front {
data.front = who;
info!("new front: {}", data.front);
}
}
Err(why) => {
error!("front error: {:?}", why);
}
}
let dur = time::Duration::new(FIVE_MINUTES, 0);
thread::sleep(dur);
}
}
2020-04-07 12:03:07 +00:00
fn main() -> std::io::Result<()> {
let st: State = State::init();
2020-04-07 01:09:09 +00:00
st.show();
let mtst = Arc::new(Mutex::new(st));
2020-04-07 12:56:56 +00:00
env_logger::init();
let cfg = config::load()?;
2020-04-07 01:09:09 +00:00
2020-04-07 12:03:07 +00:00
// start update thread
{
let mtst = mtst.clone();
thread::spawn(move || update_every_so_often(mtst));
}
2020-04-07 12:56:56 +00:00
// start front thread
{
let mtst = mtst.clone();
let cfg = cfg.clone();
thread::spawn(move || front_update(mtst, cfg));
}
2020-04-07 12:03:07 +00:00
let _ = std::fs::remove_file("/home/cadey/tmp/cabytcini.sock")?;
2020-04-07 01:09:09 +00:00
let listener = UnixListener::bind("/home/cadey/tmp/cabytcini.sock")?;
for stream in listener.incoming() {
match stream {
Ok(stream) => {
2020-04-07 12:03:07 +00:00
let mtst = mtst.clone();
thread::spawn(move || handle_client(stream, mtst));
}
2020-04-07 01:09:09 +00:00
Err(err) => {
2020-04-07 12:56:56 +00:00
error!("unix listener error: {:?}", err);
2020-04-07 01:09:09 +00:00
break;
2020-04-07 12:03:07 +00:00
}
2020-04-07 01:09:09 +00:00
}
}
Ok(())
}
2020-04-07 12:56:56 +00:00
pub struct State {
2020-04-07 01:09:09 +00:00
msg: String,
2020-04-07 12:56:56 +00:00
front: String,
2020-04-07 01:09:09 +00:00
}
impl fmt::Display for State {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.msg)
}
}
impl State {
fn init() -> State {
State {
msg: "".to_string(),
2020-04-07 12:56:56 +00:00
front: "".to_string(),
2020-04-07 01:09:09 +00:00
}
}
fn show(&self) {
2020-04-07 12:03:07 +00:00
let now = Local::now().format("%H:%M M%m %-d %a");
2020-04-07 01:09:09 +00:00
let xsr = XSetRoot::init();
2020-04-07 12:56:56 +00:00
let mut msg = String::new();
if self.msg != "" {
msg.push_str(format!("{} | ", msg).as_str());
}
if self.front != "" {
msg.push_str(format!("{} | ", self.front).as_str());
2020-04-07 12:03:07 +00:00
}
2020-04-07 12:56:56 +00:00
msg.push_str(format!("{}", now).as_str());
xsr.render(msg);
2020-04-07 01:09:09 +00:00
}
}