mirror of
https://github.com/Start9Labs/start-os.git
synced 2026-03-26 10:21:52 +00:00
* wip: Pulling in the features of the refactor since march * chore: Fixes to make the system able to build * chore: Adding in the documentation for the manager stuff * feat: Restarting and wait for stop * feat: Add a soft shutdown not commit to db. * chore: Remove the comments of bluj * chore: Clean up some of the linting errors * chore: Clean up the signal * chore: Some more cleanup * fix: The configure * fix: A missing config * fix: typo * chore: Remove a comment of BLUJ that needed to be removed
167 lines
4.7 KiB
Rust
167 lines
4.7 KiB
Rust
use std::convert::Infallible;
|
|
use std::net::{Ipv4Addr, Ipv6Addr, SocketAddr};
|
|
use std::path::Path;
|
|
|
|
use async_stream::try_stream;
|
|
use color_eyre::eyre::eyre;
|
|
use futures::stream::BoxStream;
|
|
use futures::{StreamExt, TryStreamExt};
|
|
use ipnet::{Ipv4Net, Ipv6Net};
|
|
use tokio::net::{TcpListener, TcpStream};
|
|
use tokio::process::Command;
|
|
|
|
use crate::util::Invoke;
|
|
use crate::Error;
|
|
|
|
fn parse_iface_ip(output: &str) -> Result<Vec<&str>, Error> {
|
|
let output = output.trim();
|
|
if output.is_empty() {
|
|
return Ok(Vec::new());
|
|
}
|
|
let mut res = Vec::new();
|
|
for line in output.lines() {
|
|
if let Some(ip) = line.split_ascii_whitespace().nth(3) {
|
|
res.push(ip)
|
|
} else {
|
|
return Err(Error::new(
|
|
eyre!("malformed output from `ip`"),
|
|
crate::ErrorKind::Network,
|
|
));
|
|
}
|
|
}
|
|
Ok(res)
|
|
}
|
|
|
|
pub async fn get_iface_ipv4_addr(iface: &str) -> Result<Option<(Ipv4Addr, Ipv4Net)>, Error> {
|
|
Ok(parse_iface_ip(&String::from_utf8(
|
|
Command::new("ip")
|
|
.arg("-4")
|
|
.arg("-o")
|
|
.arg("addr")
|
|
.arg("show")
|
|
.arg(iface)
|
|
.invoke(crate::ErrorKind::Network)
|
|
.await?,
|
|
)?)?
|
|
.into_iter()
|
|
.map(|s| Ok::<_, Error>((s.split("/").next().unwrap().parse()?, s.parse()?)))
|
|
.next()
|
|
.transpose()?)
|
|
}
|
|
|
|
pub async fn get_iface_ipv6_addr(iface: &str) -> Result<Option<(Ipv6Addr, Ipv6Net)>, Error> {
|
|
Ok(parse_iface_ip(&String::from_utf8(
|
|
Command::new("ip")
|
|
.arg("-6")
|
|
.arg("-o")
|
|
.arg("addr")
|
|
.arg("show")
|
|
.arg(iface)
|
|
.invoke(crate::ErrorKind::Network)
|
|
.await?,
|
|
)?)?
|
|
.into_iter()
|
|
.find(|ip| !ip.starts_with("fe80::"))
|
|
.map(|s| Ok::<_, Error>((s.split("/").next().unwrap().parse()?, s.parse()?)))
|
|
.transpose()?)
|
|
}
|
|
|
|
pub async fn iface_is_physical(iface: &str) -> bool {
|
|
tokio::fs::metadata(Path::new("/sys/class/net").join(iface).join("device"))
|
|
.await
|
|
.is_ok()
|
|
}
|
|
|
|
pub async fn iface_is_wireless(iface: &str) -> bool {
|
|
tokio::fs::metadata(Path::new("/sys/class/net").join(iface).join("wireless"))
|
|
.await
|
|
.is_ok()
|
|
}
|
|
|
|
pub fn list_interfaces() -> BoxStream<'static, Result<String, Error>> {
|
|
try_stream! {
|
|
let mut ifaces = tokio::fs::read_dir("/sys/class/net").await?;
|
|
while let Some(iface) = ifaces.next_entry().await? {
|
|
if let Some(iface) = iface.file_name().into_string().ok() {
|
|
yield iface;
|
|
}
|
|
}
|
|
}
|
|
.boxed()
|
|
}
|
|
|
|
pub async fn find_wifi_iface() -> Result<Option<String>, Error> {
|
|
let mut ifaces = list_interfaces();
|
|
while let Some(iface) = ifaces.try_next().await? {
|
|
if iface_is_wireless(&iface).await {
|
|
return Ok(Some(iface));
|
|
}
|
|
}
|
|
Ok(None)
|
|
}
|
|
|
|
pub async fn find_eth_iface() -> Result<String, Error> {
|
|
let mut ifaces = list_interfaces();
|
|
while let Some(iface) = ifaces.try_next().await? {
|
|
if iface_is_physical(&iface).await && !iface_is_wireless(&iface).await {
|
|
return Ok(iface);
|
|
}
|
|
}
|
|
Err(Error::new(
|
|
eyre!("Could not detect ethernet interface"),
|
|
crate::ErrorKind::Network,
|
|
))
|
|
}
|
|
|
|
#[pin_project::pin_project]
|
|
pub struct SingleAccept<T>(Option<T>);
|
|
impl<T> SingleAccept<T> {
|
|
pub fn new(conn: T) -> Self {
|
|
Self(Some(conn))
|
|
}
|
|
}
|
|
impl<T> hyper::server::accept::Accept for SingleAccept<T> {
|
|
type Conn = T;
|
|
type Error = Infallible;
|
|
fn poll_accept(
|
|
self: std::pin::Pin<&mut Self>,
|
|
_cx: &mut std::task::Context<'_>,
|
|
) -> std::task::Poll<Option<Result<Self::Conn, Self::Error>>> {
|
|
std::task::Poll::Ready(self.project().0.take().map(Ok))
|
|
}
|
|
}
|
|
|
|
pub struct TcpListeners {
|
|
listeners: Vec<TcpListener>,
|
|
}
|
|
impl TcpListeners {
|
|
pub fn new(listeners: impl IntoIterator<Item = TcpListener>) -> Self {
|
|
Self {
|
|
listeners: listeners.into_iter().collect(),
|
|
}
|
|
}
|
|
|
|
pub async fn accept(&self) -> std::io::Result<(TcpStream, SocketAddr)> {
|
|
futures::future::select_all(self.listeners.iter().map(|l| Box::pin(l.accept())))
|
|
.await
|
|
.0
|
|
}
|
|
}
|
|
impl hyper::server::accept::Accept for TcpListeners {
|
|
type Conn = TcpStream;
|
|
type Error = std::io::Error;
|
|
|
|
fn poll_accept(
|
|
self: std::pin::Pin<&mut Self>,
|
|
cx: &mut std::task::Context<'_>,
|
|
) -> std::task::Poll<Option<Result<Self::Conn, Self::Error>>> {
|
|
for listener in self.listeners.iter() {
|
|
let poll = listener.poll_accept(cx);
|
|
if poll.is_ready() {
|
|
return poll.map(|a| a.map(|a| a.0)).map(Some);
|
|
}
|
|
}
|
|
std::task::Poll::Pending
|
|
}
|
|
}
|