Setup daemon tracking.
Track original path before resolution for later git copying and messaging.pull/3/head
parent
4fa6654423
commit
7db0656e89
|
@ -10,7 +10,7 @@ use std::io::Write;
|
||||||
pub fn write_config(mut pending: PathConfig) -> config::Result<()> {
|
pub fn write_config(mut pending: PathConfig) -> config::Result<()> {
|
||||||
println!(
|
println!(
|
||||||
"Generating config at {}...\n",
|
"Generating config at {}...\n",
|
||||||
Success.paint(pending.0.display().to_string())
|
Success.paint(pending.0.unresolved().display().to_string())
|
||||||
);
|
);
|
||||||
|
|
||||||
print!(
|
print!(
|
||||||
|
@ -45,9 +45,9 @@ pub fn write_config(mut pending: PathConfig) -> config::Result<()> {
|
||||||
pub fn list_packages(config: PathConfig) {
|
pub fn list_packages(config: PathConfig) {
|
||||||
println!(
|
println!(
|
||||||
"Listing packages in {}...\n",
|
"Listing packages in {}...\n",
|
||||||
Success.paint(config.0.display().to_string())
|
Success.paint(config.0.unresolved().display().to_string())
|
||||||
);
|
);
|
||||||
// TODO(jrpotter): Alphabetize the output list.
|
// Alphabetical ordered ensured by B-tree implementation.
|
||||||
for (k, _) in config.1.packages {
|
for (k, _) in config.1.packages {
|
||||||
println!("• {}", k);
|
println!("• {}", k);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
use super::path;
|
use super::path::ResPathBuf;
|
||||||
use super::path::{NormalPathBuf, Normalize};
|
|
||||||
use serde_derive::{Deserialize, Serialize};
|
use serde_derive::{Deserialize, Serialize};
|
||||||
use std::collections::HashMap;
|
use std::collections::BTreeMap;
|
||||||
use std::io::Write;
|
use std::io::Write;
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use std::{error, fmt, fs, io};
|
use std::{error, fmt, fs, io};
|
||||||
|
@ -61,7 +60,7 @@ pub struct Package {
|
||||||
#[derive(Debug, Deserialize, Serialize)]
|
#[derive(Debug, Deserialize, Serialize)]
|
||||||
pub struct Config {
|
pub struct Config {
|
||||||
pub remote: Remote,
|
pub remote: Remote,
|
||||||
pub packages: HashMap<String, Package>,
|
pub packages: BTreeMap<String, Package>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Config {
|
impl Config {
|
||||||
|
@ -71,10 +70,10 @@ impl Config {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct PathConfig(pub NormalPathBuf, pub Config);
|
pub struct PathConfig(pub ResPathBuf, pub Config);
|
||||||
|
|
||||||
impl PathConfig {
|
impl PathConfig {
|
||||||
pub fn new(path: &NormalPathBuf, config: Option<Config>) -> Self {
|
pub fn new(path: &ResPathBuf, config: Option<Config>) -> Self {
|
||||||
PathConfig(
|
PathConfig(
|
||||||
path.clone(),
|
path.clone(),
|
||||||
config.unwrap_or(Config {
|
config.unwrap_or(Config {
|
||||||
|
@ -82,7 +81,7 @@ impl PathConfig {
|
||||||
owner: "example-user".to_owned(),
|
owner: "example-user".to_owned(),
|
||||||
name: "home-config".to_owned(),
|
name: "home-config".to_owned(),
|
||||||
},
|
},
|
||||||
packages: HashMap::new(),
|
packages: BTreeMap::new(),
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -111,7 +110,7 @@ pub fn default_paths() -> Vec<PathBuf> {
|
||||||
DEFAULT_PATHS.iter().map(|s| PathBuf::from(s)).collect()
|
DEFAULT_PATHS.iter().map(|s| PathBuf::from(s)).collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn load(candidates: &Vec<NormalPathBuf>) -> Result<PathConfig> {
|
pub fn load(candidates: &Vec<ResPathBuf>) -> Result<PathConfig> {
|
||||||
// When trying our paths, the only acceptable error is a `NotFound` file.
|
// When trying our paths, the only acceptable error is a `NotFound` file.
|
||||||
// Anything else should be surfaced to the end user.
|
// Anything else should be surfaced to the end user.
|
||||||
for candidate in candidates {
|
for candidate in candidates {
|
||||||
|
@ -126,3 +125,9 @@ pub fn load(candidates: &Vec<NormalPathBuf>) -> Result<PathConfig> {
|
||||||
}
|
}
|
||||||
Err(Error::MissingConfig)
|
Err(Error::MissingConfig)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn reload(config: &PathConfig) -> Result<PathConfig> {
|
||||||
|
// TODO(jrpotter): Let's add a proper logging solution.
|
||||||
|
println!("Configuration reloaded.");
|
||||||
|
load(&vec![config.0.clone()])
|
||||||
|
}
|
||||||
|
|
249
src/daemon.rs
249
src/daemon.rs
|
@ -1,99 +1,210 @@
|
||||||
use super::config;
|
use super::config;
|
||||||
use super::config::PathConfig;
|
use super::config::PathConfig;
|
||||||
use super::path;
|
use super::path;
|
||||||
use super::path::{NormalPathBuf, Normalize};
|
use super::path::ResPathBuf;
|
||||||
use notify::{RecommendedWatcher, Watcher};
|
use notify::{DebouncedEvent, RecommendedWatcher, RecursiveMode, Watcher};
|
||||||
use std::collections::HashSet;
|
use std::collections::HashSet;
|
||||||
|
use std::error::Error;
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use std::sync::mpsc::channel;
|
use std::sync::mpsc::channel;
|
||||||
|
use std::sync::mpsc::{Receiver, Sender, TryRecvError};
|
||||||
|
use std::thread;
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
// TODO(jrpotter): Add logging.
|
// TODO(jrpotter): Add logging.
|
||||||
// TODO(jrpotter): Add pid file to only allow one daemon at a time.
|
// TODO(jrpotter): Add pid file to only allow one daemon at a time.
|
||||||
|
|
||||||
|
// Used for both polling and debouncing file system events.
|
||||||
|
const WATCH_FREQUENCY: Duration = Duration::from_secs(5);
|
||||||
|
|
||||||
// ========================================
|
// ========================================
|
||||||
// State
|
// Polling
|
||||||
// ========================================
|
// ========================================
|
||||||
|
|
||||||
struct WatchState {
|
enum PollEvent {
|
||||||
// Paths that we were not able to watch properly but could potentially do so
|
Pending(PathBuf),
|
||||||
// in the future. These include paths that did not exist at the time of
|
Clear,
|
||||||
// canonicalization or did not have environment variables defined that may
|
|
||||||
// be defined later on.
|
|
||||||
pending: HashSet<PathBuf>,
|
|
||||||
// Paths that we are currently watching.
|
|
||||||
watching: HashSet<NormalPathBuf>,
|
|
||||||
// Paths that are not valid and will never become valid. These may include
|
|
||||||
// paths that include prefixes or refer to directories that could never be
|
|
||||||
// reached (e.g. parent of root).
|
|
||||||
invalid: HashSet<PathBuf>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl WatchState {
|
fn resolve_pending(tx: &Sender<DebouncedEvent>, pending: &HashSet<PathBuf>) -> Vec<PathBuf> {
|
||||||
pub fn new(config: &PathConfig) -> notify::Result<Self> {
|
let mut to_remove = vec![];
|
||||||
let mut pending: HashSet<PathBuf> = HashSet::new();
|
for path in pending {
|
||||||
let mut watching: HashSet<NormalPathBuf> = HashSet::new();
|
match path::resolve(&path) {
|
||||||
watching.insert(config.0.clone());
|
Ok(Some(resolved)) => {
|
||||||
// We try and resolve our configuration again here. We want to
|
to_remove.push(path.clone());
|
||||||
// specifically track any new configs that may pop up with higher
|
tx.send(DebouncedEvent::Create(resolved.into()))
|
||||||
// priority.
|
.expect("File watcher channel closed.");
|
||||||
for path in config::default_paths() {
|
|
||||||
match path::normalize(&path)? {
|
|
||||||
// TODO(jrpotter): Check if the path can be canonicalized.
|
|
||||||
Normalize::Done(p) => watching.insert(p),
|
|
||||||
Normalize::Pending => pending.insert(path),
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
Ok(None) => (),
|
||||||
|
Err(e) => {
|
||||||
|
to_remove.push(path.clone());
|
||||||
|
eprintln!(
|
||||||
|
"Encountered unexpected error {} when processing path {}",
|
||||||
|
e,
|
||||||
|
path.display()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
to_remove
|
||||||
|
}
|
||||||
|
|
||||||
|
fn poll_pending(tx: Sender<DebouncedEvent>, rx: Receiver<PollEvent>) {
|
||||||
|
let mut pending = HashSet::new();
|
||||||
|
loop {
|
||||||
|
match rx.try_recv() {
|
||||||
|
Ok(PollEvent::Pending(path)) => {
|
||||||
|
pending.insert(path);
|
||||||
|
}
|
||||||
|
Ok(PollEvent::Clear) => pending.clear(),
|
||||||
|
Err(TryRecvError::Empty) => {
|
||||||
|
resolve_pending(&tx, &pending).iter().for_each(|r| {
|
||||||
|
pending.remove(r);
|
||||||
|
});
|
||||||
|
thread::sleep(WATCH_FREQUENCY);
|
||||||
|
}
|
||||||
|
Err(TryRecvError::Disconnected) => panic!("Polling channel closed."),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ========================================
|
||||||
|
// File Watcher
|
||||||
|
// ========================================
|
||||||
|
|
||||||
|
struct WatchState<'a> {
|
||||||
|
poll_tx: Sender<PollEvent>,
|
||||||
|
watcher: &'a mut RecommendedWatcher,
|
||||||
|
watching: HashSet<ResPathBuf>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> WatchState<'a> {
|
||||||
|
pub fn new(
|
||||||
|
poll_tx: Sender<PollEvent>,
|
||||||
|
watcher: &'a mut RecommendedWatcher,
|
||||||
|
) -> notify::Result<Self> {
|
||||||
Ok(WatchState {
|
Ok(WatchState {
|
||||||
pending,
|
poll_tx,
|
||||||
watching,
|
watcher,
|
||||||
invalid: HashSet::new(),
|
watching: HashSet::new(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn send_poll(&self, event: PollEvent) {
|
||||||
|
self.poll_tx.send(event).expect("Polling channel closed.");
|
||||||
|
}
|
||||||
|
|
||||||
|
fn watch(&mut self, path: ResPathBuf) {
|
||||||
|
match self.watcher.watch(&path, RecursiveMode::NonRecursive) {
|
||||||
|
Ok(()) => {
|
||||||
|
self.watching.insert(path);
|
||||||
|
}
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!(
|
||||||
|
"Encountered unexpected error {} when watching path {}",
|
||||||
|
e,
|
||||||
|
path.unresolved().display()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Reads in the new path config, updating all watched and pending files
|
||||||
|
/// according to the packages in the specified config.
|
||||||
|
pub fn update(&mut self, config: &PathConfig) {
|
||||||
|
self.send_poll(PollEvent::Clear);
|
||||||
|
for path in &self.watching {
|
||||||
|
match self.watcher.unwatch(&path) {
|
||||||
|
Ok(()) => (),
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!(
|
||||||
|
"Encountered unexpected error {} when unwatching path {}",
|
||||||
|
e,
|
||||||
|
path.unresolved().display()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
self.watching.clear();
|
||||||
|
for (_, package) in &config.1.packages {
|
||||||
|
for path in &package.configs {
|
||||||
|
match path::resolve(&path) {
|
||||||
|
Ok(None) => self.send_poll(PollEvent::Pending(path.clone())),
|
||||||
|
Ok(Some(n)) => self.watch(n),
|
||||||
|
Err(_) => (),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ========================================
|
// ========================================
|
||||||
// Daemon
|
// Daemon
|
||||||
// ========================================
|
// ========================================
|
||||||
|
|
||||||
fn reload_config(config: &PathConfig) -> notify::Result<WatchState> {
|
pub fn launch(mut config: PathConfig) -> Result<(), Box<dyn Error>> {
|
||||||
let state = WatchState::new(config)?;
|
let (poll_tx, poll_rx) = channel();
|
||||||
Ok(state)
|
let (watch_tx, watch_rx) = channel();
|
||||||
}
|
let watch_tx1 = watch_tx.clone();
|
||||||
|
// `notify-rs` internally uses `fs::canonicalize` on each path we try to
|
||||||
pub fn launch(config: PathConfig) -> notify::Result<()> {
|
// watch, but this fails if no file exists at the given path. In these
|
||||||
let (tx, rx) = channel();
|
// cases, we rely on a basic polling strategy to check if the files ever
|
||||||
// Create a "debounced" watcher. Events will not trigger until after the
|
// come into existence.
|
||||||
// specified duration has passed with no additional changes.
|
thread::spawn(move || poll_pending(watch_tx, poll_rx));
|
||||||
let mut watcher: RecommendedWatcher = Watcher::new(tx, Duration::from_secs(2))?;
|
// Track our original config file separately from the other files that may
|
||||||
// for (_, package) in &config.1.packages {
|
// be defined in the config. We want to make sure we're always alerted on
|
||||||
// for path in &package.configs {
|
// changes to it for hot reloading purposes, and not worry that our wrapper
|
||||||
// match normalize_path(&path) {
|
// will ever clear it from its watch state.
|
||||||
// // `notify-rs` is not able to handle files that do not exist and
|
let mut watcher: RecommendedWatcher = Watcher::new(watch_tx1, WATCH_FREQUENCY)?;
|
||||||
// // are then created. This is handled internally by the library
|
watcher.watch(&config.0, RecursiveMode::NonRecursive)?;
|
||||||
// // via the `fs::canonicalize` which fails on missing paths. So
|
let mut state = WatchState::new(poll_tx, &mut watcher)?;
|
||||||
// // track which paths end up missing and apply polling on them.
|
state.update(&config);
|
||||||
// Ok(normalized) => match watcher.watch(&normalized, RecursiveMode::NonRecursive) {
|
|
||||||
// Ok(_) => {
|
|
||||||
// tracked_paths.insert(normalized);
|
|
||||||
// }
|
|
||||||
// Err(notify::Error::PathNotFound) => {
|
|
||||||
// missing_paths.insert(normalized);
|
|
||||||
// }
|
|
||||||
// Err(e) => return Err(e),
|
|
||||||
// },
|
|
||||||
// // TODO(jrpotter): Retry even in cases where environment
|
|
||||||
// // variables are not defined.
|
|
||||||
// Err(e) => eprintln!("{}", e),
|
|
||||||
// };
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// This is a simple loop, but you may want to use more complex logic here,
|
|
||||||
// for example to handle I/O.
|
|
||||||
loop {
|
loop {
|
||||||
match rx.recv() {
|
// Received paths should always be the fully resolved ones so safe to
|
||||||
Ok(event) => println!("{:?}", event),
|
// compare against our current config path.
|
||||||
Err(e) => println!("watch error: {:?}", e),
|
match watch_rx.recv() {
|
||||||
|
Ok(DebouncedEvent::NoticeWrite(_)) => {
|
||||||
|
// Intentionally ignore in favor of stronger signals.
|
||||||
|
}
|
||||||
|
Ok(DebouncedEvent::NoticeRemove(_)) => {
|
||||||
|
// Intentionally ignore in favor of stronger signals.
|
||||||
|
}
|
||||||
|
Ok(DebouncedEvent::Create(p)) => {
|
||||||
|
if config.0 == p {
|
||||||
|
config = config::reload(&config)?;
|
||||||
|
state.update(&config);
|
||||||
|
}
|
||||||
|
println!("Create {}", p.display());
|
||||||
|
}
|
||||||
|
Ok(DebouncedEvent::Write(p)) => {
|
||||||
|
if config.0 == p {
|
||||||
|
config = config::reload(&config)?;
|
||||||
|
state.update(&config);
|
||||||
|
}
|
||||||
|
println!("Write {}", p.display());
|
||||||
|
}
|
||||||
|
// Do not try reloading our primary config in any of the following
|
||||||
|
// cases since it may lead to undesired behavior. If our config has
|
||||||
|
// e.g. been removed, let's just keep using what we have in memory
|
||||||
|
// in the chance it may be added back.
|
||||||
|
Ok(DebouncedEvent::Chmod(p)) => {
|
||||||
|
println!("Chmod {}", p.display());
|
||||||
|
}
|
||||||
|
Ok(DebouncedEvent::Remove(p)) => {
|
||||||
|
println!("Remove {}", p.display());
|
||||||
|
}
|
||||||
|
Ok(DebouncedEvent::Rename(src, dst)) => {
|
||||||
|
println!("Rename {} {}", src.display(), dst.display())
|
||||||
|
}
|
||||||
|
Ok(DebouncedEvent::Rescan) => {
|
||||||
|
println!("Rescanning");
|
||||||
|
}
|
||||||
|
Ok(DebouncedEvent::Error(e, _maybe_path)) => {
|
||||||
|
println!("Error {}", e);
|
||||||
|
}
|
||||||
|
Err(e) => {
|
||||||
|
println!("watch error: {:?}", e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,7 @@ pub mod daemon;
|
||||||
pub mod path;
|
pub mod path;
|
||||||
|
|
||||||
use config::PathConfig;
|
use config::PathConfig;
|
||||||
use path::NormalPathBuf;
|
use path::ResPathBuf;
|
||||||
use std::error::Error;
|
use std::error::Error;
|
||||||
use std::io;
|
use std::io;
|
||||||
|
|
||||||
|
@ -18,7 +18,7 @@ pub fn run_daemon(config: PathConfig) -> Result<(), Box<dyn Error>> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn run_init(candidates: Vec<NormalPathBuf>) -> Result<(), config::Error> {
|
pub fn run_init(candidates: Vec<ResPathBuf>) -> Result<(), config::Error> {
|
||||||
debug_assert!(!candidates.is_empty(), "Empty candidates found in `init`.");
|
debug_assert!(!candidates.is_empty(), "Empty candidates found in `init`.");
|
||||||
if candidates.is_empty() {
|
if candidates.is_empty() {
|
||||||
return Err(config::Error::FileError(io::Error::new(
|
return Err(config::Error::FileError(io::Error::new(
|
||||||
|
|
14
src/main.rs
14
src/main.rs
|
@ -1,5 +1,5 @@
|
||||||
use clap::{App, AppSettings, Arg};
|
use clap::{App, AppSettings, Arg};
|
||||||
use homesync::path::{NormalPathBuf, Normalize};
|
use homesync::path::ResPathBuf;
|
||||||
use std::error::Error;
|
use std::error::Error;
|
||||||
use std::io;
|
use std::io;
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
|
@ -54,18 +54,18 @@ fn dispatch(matches: clap::ArgMatches) -> Result<(), Box<dyn Error>> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn find_candidates(matches: &clap::ArgMatches) -> Result<Vec<NormalPathBuf>, Box<dyn Error>> {
|
fn find_candidates(matches: &clap::ArgMatches) -> Result<Vec<ResPathBuf>, io::Error> {
|
||||||
let candidates = match matches.value_of("config") {
|
let candidates = match matches.value_of("config") {
|
||||||
Some(config_match) => vec![PathBuf::from(config_match)],
|
Some(config_match) => vec![PathBuf::from(config_match)],
|
||||||
None => homesync::config::default_paths(),
|
None => homesync::config::default_paths(),
|
||||||
};
|
};
|
||||||
let mut normals = vec![];
|
let mut resolved = vec![];
|
||||||
for candidate in candidates {
|
for candidate in candidates {
|
||||||
if let Ok(Normalize::Done(n)) = homesync::path::normalize(&candidate) {
|
if let Ok(Some(r)) = homesync::path::resolve(&candidate) {
|
||||||
normals.push(n);
|
resolved.push(r);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if normals.is_empty() {
|
if resolved.is_empty() {
|
||||||
if let Some(config_match) = matches.value_of("config") {
|
if let Some(config_match) = matches.value_of("config") {
|
||||||
Err(io::Error::new(
|
Err(io::Error::new(
|
||||||
io::ErrorKind::NotFound,
|
io::ErrorKind::NotFound,
|
||||||
|
@ -79,6 +79,6 @@ fn find_candidates(matches: &clap::ArgMatches) -> Result<Vec<NormalPathBuf>, Box
|
||||||
))?
|
))?
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Ok(normals)
|
Ok(resolved)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
100
src/path.rs
100
src/path.rs
|
@ -9,45 +9,67 @@ use std::path::{Component, Path, PathBuf};
|
||||||
// Path
|
// Path
|
||||||
// ========================================
|
// ========================================
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct NormalPathBuf(PathBuf);
|
pub struct ResPathBuf {
|
||||||
|
inner: PathBuf,
|
||||||
|
unresolved: PathBuf,
|
||||||
|
}
|
||||||
|
|
||||||
impl NormalPathBuf {
|
impl ResPathBuf {
|
||||||
pub fn display(&self) -> std::path::Display {
|
pub fn display(&self) -> std::path::Display {
|
||||||
self.0.display()
|
self.inner.display()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn unresolved(&self) -> &PathBuf {
|
||||||
|
return &self.unresolved;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AsRef<Path> for NormalPathBuf {
|
impl PartialEq<ResPathBuf> for ResPathBuf {
|
||||||
|
fn eq(&self, other: &Self) -> bool {
|
||||||
|
self.inner == other.inner
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PartialEq<PathBuf> for ResPathBuf {
|
||||||
|
fn eq(&self, other: &PathBuf) -> bool {
|
||||||
|
self.inner == *other
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Eq for ResPathBuf {}
|
||||||
|
|
||||||
|
impl From<ResPathBuf> for PathBuf {
|
||||||
|
fn from(path: ResPathBuf) -> PathBuf {
|
||||||
|
path.inner
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AsRef<Path> for ResPathBuf {
|
||||||
fn as_ref(&self) -> &Path {
|
fn as_ref(&self) -> &Path {
|
||||||
&self.0
|
&self.inner
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AsRef<PathBuf> for NormalPathBuf {
|
impl AsRef<PathBuf> for ResPathBuf {
|
||||||
fn as_ref(&self) -> &PathBuf {
|
fn as_ref(&self) -> &PathBuf {
|
||||||
&self.0
|
&self.inner
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Hash for NormalPathBuf {
|
impl Hash for ResPathBuf {
|
||||||
fn hash<H: Hasher>(&self, h: &mut H) {
|
fn hash<H: Hasher>(&self, h: &mut H) {
|
||||||
for component in self.0.components() {
|
for component in self.inner.components() {
|
||||||
component.hash(h);
|
component.hash(h);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub enum Normalize {
|
/// Find environment variables found within the argument and expand them if
|
||||||
Done(NormalPathBuf), // An instance of a fully resolved path.
|
/// possible.
|
||||||
Pending, // An instance of a path that cannot yet be normalized.
|
///
|
||||||
}
|
/// Returns `None` in the case an environment variable present within the
|
||||||
|
/// argument is not defined.
|
||||||
// Find environment variables found within the argument and expand them if
|
|
||||||
// possible.
|
|
||||||
//
|
|
||||||
// Returns `None` in the case an environment variable present within the
|
|
||||||
// argument is not defined.
|
|
||||||
fn expand_env(s: &OsStr) -> Option<OsString> {
|
fn expand_env(s: &OsStr) -> Option<OsString> {
|
||||||
let re = Regex::new(r"\$(?P<env>[[:alnum:]]+)").unwrap();
|
let re = Regex::new(r"\$(?P<env>[[:alnum:]]+)").unwrap();
|
||||||
let lossy = s.to_string_lossy();
|
let lossy = s.to_string_lossy();
|
||||||
|
@ -59,17 +81,14 @@ fn expand_env(s: &OsStr) -> Option<OsString> {
|
||||||
Some(path.into())
|
Some(path.into())
|
||||||
}
|
}
|
||||||
|
|
||||||
// Normalizes the provided path, returning a new instance.
|
/// Attempt to resolve the provided path, returning a fully resolved path
|
||||||
//
|
/// instance.
|
||||||
// There currently does not exist a method that yields some canonical path for
|
///
|
||||||
// files that do not exist (at least in the parts of the standard library I've
|
/// If the provided file does not exist but could potentially exist in the
|
||||||
// looked in). We create a consistent view of every path so as to avoid watching
|
/// future (e.g. for paths with environment variables defined), this will
|
||||||
// the same path multiple times, which would duplicate messages on changes.
|
/// return a `None` instead of an error.
|
||||||
//
|
pub fn resolve(path: &Path) -> io::Result<Option<ResPathBuf>> {
|
||||||
// Note this does not actually prevent the issue fully. We could have two paths
|
let mut expanded = env::current_dir()?;
|
||||||
// that refer to the same real path - normalization would not catch this.
|
|
||||||
pub fn normalize(path: &Path) -> io::Result<Normalize> {
|
|
||||||
let mut pb = env::current_dir()?;
|
|
||||||
for comp in path.components() {
|
for comp in path.components() {
|
||||||
match comp {
|
match comp {
|
||||||
Component::Prefix(_) => {
|
Component::Prefix(_) => {
|
||||||
|
@ -79,12 +98,12 @@ pub fn normalize(path: &Path) -> io::Result<Normalize> {
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
Component::RootDir => {
|
Component::RootDir => {
|
||||||
pb.clear();
|
expanded.clear();
|
||||||
pb.push(Component::RootDir)
|
expanded.push(Component::RootDir)
|
||||||
}
|
}
|
||||||
Component::CurDir => (), // Make no changes.
|
Component::CurDir => (), // Make no changes.
|
||||||
Component::ParentDir => {
|
Component::ParentDir => {
|
||||||
if !pb.pop() {
|
if !expanded.pop() {
|
||||||
return Err(io::Error::new(
|
return Err(io::Error::new(
|
||||||
io::ErrorKind::InvalidInput,
|
io::ErrorKind::InvalidInput,
|
||||||
"Cannot take parent of root.",
|
"Cannot take parent of root.",
|
||||||
|
@ -92,12 +111,19 @@ pub fn normalize(path: &Path) -> io::Result<Normalize> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Component::Normal(c) => match expand_env(c) {
|
Component::Normal(c) => match expand_env(c) {
|
||||||
Some(c) => pb.push(Component::Normal(&c)),
|
Some(c) => expanded.push(Component::Normal(&c)),
|
||||||
// The environment variable isn't defined yet but might be in
|
// The environment variable isn't defined yet but might be in
|
||||||
// the future.
|
// the future.
|
||||||
None => return Ok(Normalize::Pending),
|
None => return Ok(None),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(Normalize::Done(NormalPathBuf(pb)))
|
match expanded.canonicalize() {
|
||||||
|
Ok(resolved) => Ok(Some(ResPathBuf {
|
||||||
|
inner: resolved,
|
||||||
|
unresolved: path.to_path_buf(),
|
||||||
|
})),
|
||||||
|
Err(e) if e.kind() == io::ErrorKind::NotFound => Ok(None),
|
||||||
|
Err(e) => Err(e),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue