mirror of
				https://github.com/Zedfrigg/ironbar.git
				synced 2025-10-26 11:41:54 +01:00 
			
		
		
		
	This splits most CLI/IPC commands into two categories: - `var` for ironvars - `bar` for controlling individual bars. It also introduces more commands for visibility, as well as breaking existing ones. New commands: - `show` - `hide` - `toggle_visible` - `set_popup_visible` - `get_popup_visible` Lastly, the implementation of some existing commands has been improved. BREAKING CHANGE: All IPC commands have changed. Namely, `type` has been changed to `command`, and bar/var related commands are now under a `subcommand`. The full spec can be found on the wiki. BREAKING CHANGE: Several CLI commands are now located under the `var` and `bar` categories. Usage of any commands to get/set Ironvars or control bar visibility will need to be updated. BREAKING CHANGE: The `open_popup` and `close_popup` IPC commands are now called `show_popup` and `hide_popup` respectively. BREAKING CHANGE: The popup `name` argument has been renamed to `widget_name` on all IPC commands. BREAKING CHANGE: The `set-visibility` CLI command now takes a `true`/`false` positional argument in place of the `-v` flag. BREAKING CHANGE: `ok_value` responses will no longer print `ok` as the first line when using the CLI
		
			
				
	
	
		
			164 lines
		
	
	
	
		
			5.1 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			164 lines
		
	
	
	
		
			5.1 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| mod bar;
 | |
| mod ironvar;
 | |
| 
 | |
| use std::fs;
 | |
| use std::path::Path;
 | |
| use std::rc::Rc;
 | |
| 
 | |
| use color_eyre::{Report, Result};
 | |
| use gtk::prelude::*;
 | |
| use gtk::Application;
 | |
| use tokio::io::{AsyncReadExt, AsyncWriteExt};
 | |
| use tokio::net::{UnixListener, UnixStream};
 | |
| use tokio::sync::mpsc::{self, Receiver, Sender};
 | |
| use tracing::{debug, error, info, warn};
 | |
| 
 | |
| use crate::ipc::{Command, Response};
 | |
| use crate::style::load_css;
 | |
| use crate::{glib_recv_mpsc, send_async, spawn, try_send, Ironbar};
 | |
| 
 | |
| use super::Ipc;
 | |
| 
 | |
| impl Ipc {
 | |
|     /// Starts the IPC server on its socket.
 | |
|     ///
 | |
|     /// Once started, the server will begin accepting connections.
 | |
|     pub fn start(&self, application: &Application, ironbar: Rc<Ironbar>) {
 | |
|         let (cmd_tx, cmd_rx) = mpsc::channel(32);
 | |
|         let (res_tx, mut res_rx) = mpsc::channel(32);
 | |
| 
 | |
|         let path = self.path.clone();
 | |
| 
 | |
|         if path.exists() {
 | |
|             warn!("Socket already exists. Did Ironbar exit abruptly?");
 | |
|             warn!("Attempting IPC shutdown to allow binding to address");
 | |
|             Self::shutdown(&path);
 | |
|         }
 | |
| 
 | |
|         spawn(async move {
 | |
|             info!("Starting IPC on {}", path.display());
 | |
| 
 | |
|             let listener = match UnixListener::bind(&path) {
 | |
|                 Ok(listener) => listener,
 | |
|                 Err(err) => {
 | |
|                     error!(
 | |
|                         "{:?}",
 | |
|                         Report::new(err).wrap_err("Unable to start IPC server")
 | |
|                     );
 | |
|                     return;
 | |
|                 }
 | |
|             };
 | |
| 
 | |
|             loop {
 | |
|                 match listener.accept().await {
 | |
|                     Ok((stream, _addr)) => {
 | |
|                         if let Err(err) =
 | |
|                             Self::handle_connection(stream, &cmd_tx, &mut res_rx).await
 | |
|                         {
 | |
|                             error!("{err:?}");
 | |
|                         }
 | |
|                     }
 | |
|                     Err(err) => {
 | |
|                         error!("{err:?}");
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         let application = application.clone();
 | |
|         glib_recv_mpsc!(cmd_rx, command => {
 | |
|             let res = Self::handle_command(command, &application, &ironbar);
 | |
|             try_send!(res_tx, res);
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     /// Takes an incoming connections,
 | |
|     /// reads the command message, and sends the response.
 | |
|     ///
 | |
|     /// The connection is closed once the response has been written.
 | |
|     async fn handle_connection(
 | |
|         mut stream: UnixStream,
 | |
|         cmd_tx: &Sender<Command>,
 | |
|         res_rx: &mut Receiver<Response>,
 | |
|     ) -> Result<()> {
 | |
|         let (mut stream_read, mut stream_write) = stream.split();
 | |
| 
 | |
|         let mut read_buffer = vec![0; 1024];
 | |
|         let bytes = stream_read.read(&mut read_buffer).await?;
 | |
| 
 | |
|         // FIXME: Error on invalid command
 | |
|         let command = serde_json::from_slice::<Command>(&read_buffer[..bytes])?;
 | |
| 
 | |
|         debug!("Received command: {command:?}");
 | |
| 
 | |
|         send_async!(cmd_tx, command);
 | |
|         let res = res_rx
 | |
|             .recv()
 | |
|             .await
 | |
|             .unwrap_or(Response::Err { message: None });
 | |
|         let res = serde_json::to_vec(&res)?;
 | |
| 
 | |
|         stream_write.write_all(&res).await?;
 | |
|         stream_write.shutdown().await?;
 | |
| 
 | |
|         Ok(())
 | |
|     }
 | |
| 
 | |
|     /// Takes an input command, runs it and returns with the appropriate response.
 | |
|     ///
 | |
|     /// This runs on the main thread, allowing commands to interact with GTK.
 | |
|     fn handle_command(
 | |
|         command: Command,
 | |
|         application: &Application,
 | |
|         ironbar: &Rc<Ironbar>,
 | |
|     ) -> Response {
 | |
|         match command {
 | |
|             Command::Ping => Response::Ok,
 | |
|             Command::Inspect => {
 | |
|                 gtk::Window::set_interactive_debugging(true);
 | |
|                 Response::Ok
 | |
|             }
 | |
|             Command::Reload => {
 | |
|                 info!("Closing existing bars");
 | |
|                 ironbar.bars.borrow_mut().clear();
 | |
| 
 | |
|                 let windows = application.windows();
 | |
|                 for window in windows {
 | |
|                     window.close();
 | |
|                 }
 | |
| 
 | |
|                 let wl = ironbar.clients.borrow_mut().wayland();
 | |
|                 let outputs = wl.output_info_all();
 | |
| 
 | |
|                 ironbar.reload_config();
 | |
| 
 | |
|                 for output in outputs {
 | |
|                     match crate::load_output_bars(ironbar, application, &output) {
 | |
|                         Ok(mut bars) => ironbar.bars.borrow_mut().append(&mut bars),
 | |
|                         Err(err) => error!("{err:?}"),
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 Response::Ok
 | |
|             }
 | |
|             Command::LoadCss { path } => {
 | |
|                 if path.exists() {
 | |
|                     load_css(path);
 | |
|                     Response::Ok
 | |
|                 } else {
 | |
|                     Response::error("File not found")
 | |
|                 }
 | |
|             }
 | |
|             Command::Var(cmd) => ironvar::handle_command(cmd),
 | |
|             Command::Bar(cmd) => bar::handle_command(cmd, ironbar),
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /// Shuts down the IPC server,
 | |
|     /// removing the socket file in the process.
 | |
|     ///
 | |
|     /// Note this is static as the `Ipc` struct is not `Send`.
 | |
|     pub fn shutdown<P: AsRef<Path>>(path: P) {
 | |
|         fs::remove_file(&path).ok();
 | |
|     }
 | |
| }
 |