84a7000759
Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
158 lines
4.6 KiB
Rust
158 lines
4.6 KiB
Rust
mod interfaces;
|
|
mod inventory;
|
|
|
|
use std::{
|
|
env,
|
|
error::Error,
|
|
process,
|
|
sync::Arc,
|
|
};
|
|
|
|
use interfaces::{BlockDeviceInterface, DriveInterface, ObjectManagerRoot, UDisksManager};
|
|
use inventory::{Inventory, MANAGER_PATH, ROOT_PATH};
|
|
use tokio::runtime::Builder as RuntimeBuilder;
|
|
use zbus::{
|
|
Address,
|
|
connection::Builder as ConnectionBuilder,
|
|
zvariant::OwnedObjectPath,
|
|
};
|
|
|
|
const BUS_NAME: &str = "org.freedesktop.UDisks2";
|
|
|
|
enum Command {
|
|
Run,
|
|
Help,
|
|
}
|
|
|
|
fn usage() -> &'static str {
|
|
"Usage: redbear-udisks [--help]"
|
|
}
|
|
|
|
fn parse_args() -> Result<Command, String> {
|
|
let mut args = env::args().skip(1);
|
|
|
|
match args.next() {
|
|
None => Ok(Command::Run),
|
|
Some(arg) if arg == "--help" || arg == "-h" => {
|
|
if args.next().is_some() {
|
|
return Err(String::from("unexpected extra arguments after --help"));
|
|
}
|
|
|
|
Ok(Command::Help)
|
|
}
|
|
Some(arg) => Err(format!("unrecognized argument '{arg}'")),
|
|
}
|
|
}
|
|
|
|
fn parse_object_path(path: &str) -> Result<OwnedObjectPath, Box<dyn Error>> {
|
|
Ok(OwnedObjectPath::try_from(path.to_owned())?)
|
|
}
|
|
|
|
fn system_connection_builder() -> Result<ConnectionBuilder<'static>, Box<dyn Error>> {
|
|
if let Ok(address) = env::var("DBUS_STARTER_ADDRESS") {
|
|
Ok(ConnectionBuilder::address(Address::try_from(address.as_str())?)?)
|
|
} else {
|
|
Ok(ConnectionBuilder::address(Address::try_from("unix:path=/run/dbus/system_bus_socket")?)?)
|
|
}
|
|
}
|
|
|
|
#[cfg(all(unix, not(target_os = "redox")))]
|
|
async fn wait_for_shutdown() -> Result<(), Box<dyn Error>> {
|
|
use tokio::signal::unix::{SignalKind, signal};
|
|
|
|
let mut terminate = signal(SignalKind::terminate())?;
|
|
|
|
tokio::select! {
|
|
_ = terminate.recv() => Ok(()),
|
|
_ = tokio::signal::ctrl_c() => Ok(()),
|
|
}
|
|
}
|
|
|
|
#[cfg(target_os = "redox")]
|
|
async fn wait_for_shutdown() -> Result<(), Box<dyn Error>> {
|
|
std::future::pending::<()>().await;
|
|
#[allow(unreachable_code)]
|
|
Ok(())
|
|
}
|
|
|
|
#[cfg(all(not(unix), not(target_os = "redox")))]
|
|
async fn wait_for_shutdown() -> Result<(), Box<dyn Error>> {
|
|
tokio::signal::ctrl_c().await?;
|
|
Ok(())
|
|
}
|
|
|
|
async fn run_daemon() -> Result<(), Box<dyn Error>> {
|
|
eprintln!("redbear-udisks: startup begin");
|
|
let _root_path = parse_object_path(ROOT_PATH)?;
|
|
let _manager_path = parse_object_path(MANAGER_PATH)?;
|
|
eprintln!("redbear-udisks: object paths parsed");
|
|
let inventory = Arc::new(Inventory::scan());
|
|
eprintln!(
|
|
"redbear-udisks: inventory scanned drives={} blocks={}",
|
|
inventory.drives().len(),
|
|
inventory.blocks().len()
|
|
);
|
|
let block_sizes_known = inventory
|
|
.blocks()
|
|
.iter()
|
|
.filter(|block| block.logical_block_size > 0)
|
|
.count();
|
|
|
|
eprintln!("redbear-udisks: starter address={:?}", env::var("DBUS_STARTER_ADDRESS").ok());
|
|
eprintln!("redbear-udisks: building D-Bus connection");
|
|
let mut builder = system_connection_builder()?
|
|
.name(BUS_NAME)?
|
|
.serve_at(ROOT_PATH, ObjectManagerRoot::new(inventory.clone()))?
|
|
.serve_at(MANAGER_PATH, UDisksManager::new(inventory.clone()))?;
|
|
|
|
for drive in inventory.drives() {
|
|
builder = builder.serve_at(drive.object_path.as_str(), DriveInterface::new(drive.clone()))?;
|
|
}
|
|
|
|
for block in inventory.blocks() {
|
|
builder = builder.serve_at(block.object_path.as_str(), BlockDeviceInterface::new(block.clone()))?;
|
|
}
|
|
|
|
let connection = builder.build().await?;
|
|
|
|
eprintln!(
|
|
"redbear-udisks: registered {BUS_NAME} on the system bus with {} drives, {} block devices, {} metadata-backed block sizes",
|
|
inventory.drives().len(),
|
|
inventory.blocks().len(),
|
|
block_sizes_known,
|
|
);
|
|
|
|
wait_for_shutdown().await?;
|
|
drop(connection);
|
|
eprintln!("redbear-udisks: received shutdown signal, exiting cleanly");
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn main() {
|
|
match parse_args() {
|
|
Ok(Command::Help) => {
|
|
println!("{}", usage());
|
|
}
|
|
Ok(Command::Run) => {
|
|
let runtime = match RuntimeBuilder::new_multi_thread().enable_all().build() {
|
|
Ok(runtime) => runtime,
|
|
Err(err) => {
|
|
eprintln!("redbear-udisks: failed to create tokio runtime: {err}");
|
|
process::exit(1);
|
|
}
|
|
};
|
|
|
|
if let Err(err) = runtime.block_on(run_daemon()) {
|
|
eprintln!("redbear-udisks: fatal error: {err}");
|
|
process::exit(1);
|
|
}
|
|
}
|
|
Err(err) => {
|
|
eprintln!("redbear-udisks: {err}");
|
|
eprintln!("{}", usage());
|
|
process::exit(1);
|
|
}
|
|
}
|
|
}
|