use std::{fs, io, path, thread, time};
use crate::error::Error;
use super::conn_expr::ConnectionExpr;
#[derive(Debug)]
pub enum ConnectionExprSource {
Direct(ConnectionExpr),
PortFile {
path: Option<path::PathBuf>,
wait_for: Option<time::Duration>,
},
}
impl From<ConnectionExpr> for ConnectionExprSource {
fn from(e: ConnectionExpr) -> Self {
ConnectionExprSource::Direct(e)
}
}
impl From<&ConnectionExpr> for ConnectionExprSource {
fn from(e: &ConnectionExpr) -> Self {
e.clone().into()
}
}
impl ConnectionExprSource {
pub fn resolve_expr(&self) -> Result<ConnectionExpr, Error> {
const THROTTLING_DELAY: time::Duration = time::Duration::from_millis(50);
match self {
ConnectionExprSource::Direct(e) => Ok(e.clone()),
ConnectionExprSource::PortFile {
path,
wait_for: None,
} => try_load_from_port_file(path.as_ref()),
ConnectionExprSource::PortFile {
path,
wait_for: Some(duration),
} => {
let deadline = time::SystemTime::now() + *duration;
loop {
match try_load_from_port_file(path.as_ref()) {
Ok(r) => return Ok(r),
Err(e) => match e {
Error::NotSpecified | Error::NotFound(_) => {
if time::SystemTime::now() >= deadline {
return Err(Error::PortFileTimeout);
}
}
_ => return Err(e),
},
}
thread::sleep(THROTTLING_DELAY);
}
}
}
}
}
fn try_load_from_port_file(
given: Option<impl AsRef<path::Path>>,
) -> Result<ConnectionExpr, Error> {
if let Some(f) = given {
load_from_port_file(f)
} else if let Some(ref f) = find_port_file().map_err(|_| Error::Unknown)? {
load_from_port_file(f)
} else {
Err(Error::NotSpecified)
}
}
fn find_port_file() -> io::Result<Option<path::PathBuf>> {
let current_dir = path::PathBuf::from(".").canonicalize()?;
for dir in current_dir.ancestors() {
let mut path = path::PathBuf::from(dir);
path.push(".nrepl-port");
if path.is_file() {
return Ok(Some(path));
}
}
Ok(None)
}
fn load_from_port_file(
path: impl AsRef<path::Path>,
) -> Result<ConnectionExpr, Error> {
let path = path.as_ref();
fs::read_to_string(path)
.map_err(|e| {
if e.kind() == io::ErrorKind::NotFound {
Error::NotFound(path.to_string_lossy().into())
} else {
Error::CannotReadFile(path.to_string_lossy().into())
}
})?
.trim()
.parse()
.map_err(|_| Error::CannotParsePortFile(path.to_string_lossy().into()))
}