2025-07-11 01:01:39 -03:00
|
|
|
use std::sync::Arc;
|
2025-07-17 21:19:17 -03:00
|
|
|
use tokio_stream::StreamExt as _;
|
2025-07-11 01:01:39 -03:00
|
|
|
use tracing::trace;
|
|
|
|
|
|
|
|
|
|
pub struct HostedBus {
|
|
|
|
|
peers: Arc<busd::peers::Peers>,
|
|
|
|
|
guid: zbus::OwnedGuid,
|
|
|
|
|
next_id: usize,
|
|
|
|
|
_self_conn: zbus::Connection,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl HostedBus {
|
|
|
|
|
pub async fn new() -> eyre::Result<Self> {
|
|
|
|
|
let guid: zbus::OwnedGuid = zbus::Guid::generate().into();
|
|
|
|
|
|
|
|
|
|
let peers = busd::peers::Peers::new();
|
|
|
|
|
|
|
|
|
|
let dbus = busd::fdo::DBus::new(peers.clone(), guid.clone());
|
|
|
|
|
let monitoring = busd::fdo::Monitoring::new(peers.clone());
|
|
|
|
|
|
|
|
|
|
// Create a peer for ourselves.
|
|
|
|
|
trace!("Creating self-dial connection.");
|
|
|
|
|
let (client_socket, peer_socket) = zbus::connection::socket::Channel::pair();
|
|
|
|
|
let service_conn =
|
|
|
|
|
zbus::connection::Builder::authenticated_socket(client_socket, guid.clone())?
|
|
|
|
|
.p2p()
|
|
|
|
|
.unique_name(busd::fdo::BUS_NAME)?
|
|
|
|
|
.name(busd::fdo::BUS_NAME)?
|
|
|
|
|
.serve_at(busd::fdo::DBus::PATH, dbus)?
|
|
|
|
|
.serve_at(busd::fdo::Monitoring::PATH, monitoring)?
|
|
|
|
|
.build()
|
|
|
|
|
.await?;
|
|
|
|
|
let peer_conn = zbus::connection::Builder::authenticated_socket(peer_socket, guid.clone())?
|
|
|
|
|
.p2p()
|
|
|
|
|
.build()
|
|
|
|
|
.await?;
|
|
|
|
|
|
|
|
|
|
peers.add_us(peer_conn).await;
|
|
|
|
|
trace!("Self-dial connection created.");
|
|
|
|
|
Ok(HostedBus {
|
|
|
|
|
peers,
|
|
|
|
|
guid,
|
|
|
|
|
next_id: 0,
|
|
|
|
|
_self_conn: service_conn,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn server_guid(&self) -> &zbus::Guid<'_> {
|
|
|
|
|
self.guid.inner()
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-17 19:47:50 -03:00
|
|
|
pub async fn connect_channel(&mut self, skip_hello: bool) -> eyre::Result<zbus::Connection> {
|
2025-07-11 01:01:39 -03:00
|
|
|
let id = self.next_id();
|
|
|
|
|
self.peers
|
2025-07-17 19:47:50 -03:00
|
|
|
.add_channel(&self.guid, id, skip_hello)
|
2025-07-11 01:01:39 -03:00
|
|
|
.await
|
|
|
|
|
.map_err(|err| eyre::eyre!(Box::new(err))) // https://github.com/eyre-rs/eyre/issues/31 XXX: busd should not use anyhow!
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub async fn connect_unix(&mut self, socket: tokio::net::UnixStream) -> eyre::Result<()> {
|
|
|
|
|
let id = self.next_id();
|
|
|
|
|
self.peers
|
|
|
|
|
.add(&self.guid, id, socket.into(), zbus::AuthMechanism::External)
|
|
|
|
|
.await
|
|
|
|
|
.map_err(|err| eyre::eyre!(Box::new(err)))
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-17 21:19:17 -03:00
|
|
|
pub async fn name_owner_changes(&mut self) -> eyre::Result<NameOwnerStream> {
|
|
|
|
|
let conn = self.connect_channel(false).await?;
|
|
|
|
|
NameOwnerStream::new(conn).await
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-11 01:01:39 -03:00
|
|
|
fn next_id(&mut self) -> usize {
|
|
|
|
|
self.next_id += 1;
|
|
|
|
|
self.next_id
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-07-17 21:19:17 -03:00
|
|
|
|
|
|
|
|
pub struct NameOwnerStream {
|
|
|
|
|
_conn: zbus::Connection,
|
|
|
|
|
stream: zbus::MessageStream,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl NameOwnerStream {
|
|
|
|
|
pub async fn new(conn: zbus::Connection) -> eyre::Result<Self> {
|
|
|
|
|
// we're subscribing and waiting for the server to confirm the subscription *here*
|
|
|
|
|
// so there should be no missing events if an owner we're waiting for is spawned
|
|
|
|
|
// "too soon", even if we start polling `next()` after it's connected
|
|
|
|
|
// (to test: `tokio::time::sleep(Duration::from_secs(1)).await` before `wait_for_acquisition()` :))
|
|
|
|
|
let stream = zbus::MessageStream::for_match_rule(
|
|
|
|
|
zbus::MatchRule::builder()
|
|
|
|
|
.msg_type(zbus::message::Type::Signal)
|
|
|
|
|
.path(busd::fdo::DBus::PATH)?
|
|
|
|
|
.interface(busd::fdo::DBus::INTERFACE)?
|
|
|
|
|
.member("NameOwnerChanged")?
|
|
|
|
|
.build()
|
|
|
|
|
.into_owned(),
|
|
|
|
|
&conn,
|
|
|
|
|
None,
|
|
|
|
|
)
|
|
|
|
|
.await?;
|
|
|
|
|
Ok(NameOwnerStream {
|
|
|
|
|
_conn: conn,
|
|
|
|
|
stream,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub async fn next(&mut self) -> eyre::Result<busd::name_registry::NameOwnerChanged> {
|
|
|
|
|
match self.stream.next().await {
|
|
|
|
|
None => Err(eyre::eyre!("stream ended")),
|
|
|
|
|
Some(Err(err)) => Err(err.into()),
|
|
|
|
|
Some(Ok(msg)) => Ok({
|
|
|
|
|
let body = msg.body();
|
|
|
|
|
let (name, old_owner, new_owner): (
|
|
|
|
|
zbus::names::BusName,
|
|
|
|
|
zbus::zvariant::Optional<zbus::names::UniqueName>,
|
|
|
|
|
zbus::zvariant::Optional<zbus::names::UniqueName>,
|
|
|
|
|
) = body.deserialize()?;
|
|
|
|
|
busd::name_registry::NameOwnerChanged {
|
|
|
|
|
name: name.into_owned().into(),
|
|
|
|
|
old_owner: Option::from(old_owner).map(|x: zbus::names::UniqueName| x.into()),
|
|
|
|
|
new_owner: Option::from(new_owner).map(|x: zbus::names::UniqueName| x.into()),
|
|
|
|
|
}
|
|
|
|
|
}),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub async fn wait_for_acquisition(
|
|
|
|
|
&mut self,
|
|
|
|
|
name: zbus::names::OwnedWellKnownName,
|
|
|
|
|
) -> eyre::Result<()> {
|
|
|
|
|
while let Ok(change) = self.next().await {
|
|
|
|
|
if let zbus::names::BusName::WellKnown(ref well_known) = *change.name {
|
|
|
|
|
if *well_known == name {
|
|
|
|
|
return Ok(());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Err(eyre::eyre!(
|
|
|
|
|
"non-Ok response when waiting for name acquisition!"
|
|
|
|
|
))
|
|
|
|
|
}
|
|
|
|
|
}
|