feat(tvix/castore/directory/traverse): use castore Paths
This switches from using std::path::Path to using castore paths. We can drop some error handling in descend_to, as absolute (or redundant) paths are not representable. We however now need to convert from a std::path::Path to our representation, and decide to accept .. canonicalization, as paths in EvalIO might contain this. Dealing .. to hop into another store path, if we encounter this, should be dealt with in a previous step. Change-Id: I5e94693808420c5d56587c68731252b54755bf93 Reviewed-on: https://cl.tvl.fyi/c/depot/+/11575 Autosubmit: flokli <flokli@flokli.de> Reviewed-by: Connor Brewster <cbrewster@hey.com> Tested-by: BuildkiteCI
This commit is contained in:
parent
4033d4c50f
commit
abc0553eb8
2 changed files with 17 additions and 52 deletions
|
@ -1,30 +1,20 @@
|
||||||
use super::DirectoryService;
|
use super::DirectoryService;
|
||||||
use crate::{proto::NamedNode, B3Digest, Error};
|
use crate::{proto::NamedNode, B3Digest, Error, Path};
|
||||||
use std::os::unix::ffi::OsStrExt;
|
|
||||||
use tracing::{instrument, warn};
|
use tracing::{instrument, warn};
|
||||||
|
|
||||||
/// This descends from a (root) node to the given (sub)path, returning the Node
|
/// This descends from a (root) node to the given (sub)path, returning the Node
|
||||||
/// at that path, or none, if there's nothing at that path.
|
/// at that path, or none, if there's nothing at that path.
|
||||||
#[instrument(skip(directory_service))]
|
#[instrument(skip(directory_service, path), fields(%path))]
|
||||||
pub async fn descend_to<DS>(
|
pub async fn descend_to<DS>(
|
||||||
directory_service: DS,
|
directory_service: DS,
|
||||||
root_node: crate::proto::node::Node,
|
root_node: crate::proto::node::Node,
|
||||||
path: &std::path::Path,
|
path: impl AsRef<Path> + std::fmt::Display,
|
||||||
) -> Result<Option<crate::proto::node::Node>, Error>
|
) -> Result<Option<crate::proto::node::Node>, Error>
|
||||||
where
|
where
|
||||||
DS: AsRef<dyn DirectoryService>,
|
DS: AsRef<dyn DirectoryService>,
|
||||||
{
|
{
|
||||||
// strip a possible `/` prefix from the path.
|
|
||||||
let path = {
|
|
||||||
if path.starts_with("/") {
|
|
||||||
path.strip_prefix("/").unwrap()
|
|
||||||
} else {
|
|
||||||
path
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut cur_node = root_node;
|
let mut cur_node = root_node;
|
||||||
let mut it = path.components();
|
let mut it = path.as_ref().components();
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
match it.next() {
|
match it.next() {
|
||||||
|
@ -59,9 +49,8 @@ where
|
||||||
// look for first_component in the [Directory].
|
// look for first_component in the [Directory].
|
||||||
// FUTUREWORK: as the nodes() iterator returns in a sorted fashion, we
|
// FUTUREWORK: as the nodes() iterator returns in a sorted fashion, we
|
||||||
// could stop as soon as e.name is larger than the search string.
|
// could stop as soon as e.name is larger than the search string.
|
||||||
let child_node = directory.nodes().find(|n| {
|
let child_node =
|
||||||
n.get_name() == first_component.as_os_str().as_bytes()
|
directory.nodes().find(|n| n.get_name() == first_component);
|
||||||
});
|
|
||||||
|
|
||||||
match child_node {
|
match child_node {
|
||||||
// child node not found means there's no such element inside the directory.
|
// child node not found means there's no such element inside the directory.
|
||||||
|
@ -85,11 +74,10 @@ where
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use std::path::PathBuf;
|
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
directoryservice,
|
directoryservice,
|
||||||
fixtures::{DIRECTORY_COMPLICATED, DIRECTORY_WITH_KEEP},
|
fixtures::{DIRECTORY_COMPLICATED, DIRECTORY_WITH_KEEP},
|
||||||
|
PathBuf,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::descend_to;
|
use super::descend_to;
|
||||||
|
@ -132,7 +120,7 @@ mod tests {
|
||||||
let resp = descend_to(
|
let resp = descend_to(
|
||||||
&directory_service,
|
&directory_service,
|
||||||
node_directory_complicated.clone(),
|
node_directory_complicated.clone(),
|
||||||
&PathBuf::from(""),
|
"".parse::<PathBuf>().unwrap(),
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
.expect("must succeed");
|
.expect("must succeed");
|
||||||
|
@ -145,7 +133,7 @@ mod tests {
|
||||||
let resp = descend_to(
|
let resp = descend_to(
|
||||||
&directory_service,
|
&directory_service,
|
||||||
node_directory_complicated.clone(),
|
node_directory_complicated.clone(),
|
||||||
&PathBuf::from("keep"),
|
"keep".parse::<PathBuf>().unwrap(),
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
.expect("must succeed");
|
.expect("must succeed");
|
||||||
|
@ -158,7 +146,7 @@ mod tests {
|
||||||
let resp = descend_to(
|
let resp = descend_to(
|
||||||
&directory_service,
|
&directory_service,
|
||||||
node_directory_complicated.clone(),
|
node_directory_complicated.clone(),
|
||||||
&PathBuf::from("keep/.keep"),
|
"keep/.keep".parse::<PathBuf>().unwrap(),
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
.expect("must succeed");
|
.expect("must succeed");
|
||||||
|
@ -166,25 +154,12 @@ mod tests {
|
||||||
assert_eq!(Some(node_file_keep.clone()), resp);
|
assert_eq!(Some(node_file_keep.clone()), resp);
|
||||||
}
|
}
|
||||||
|
|
||||||
// traversal to `keep/.keep` should return the node for the .keep file
|
|
||||||
{
|
|
||||||
let resp = descend_to(
|
|
||||||
&directory_service,
|
|
||||||
node_directory_complicated.clone(),
|
|
||||||
&PathBuf::from("/keep/.keep"),
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.expect("must succeed");
|
|
||||||
|
|
||||||
assert_eq!(Some(node_file_keep), resp);
|
|
||||||
}
|
|
||||||
|
|
||||||
// traversal to `void` should return None (doesn't exist)
|
// traversal to `void` should return None (doesn't exist)
|
||||||
{
|
{
|
||||||
let resp = descend_to(
|
let resp = descend_to(
|
||||||
&directory_service,
|
&directory_service,
|
||||||
node_directory_complicated.clone(),
|
node_directory_complicated.clone(),
|
||||||
&PathBuf::from("void"),
|
"void".parse::<PathBuf>().unwrap(),
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
.expect("must succeed");
|
.expect("must succeed");
|
||||||
|
@ -192,12 +167,12 @@ mod tests {
|
||||||
assert_eq!(None, resp);
|
assert_eq!(None, resp);
|
||||||
}
|
}
|
||||||
|
|
||||||
// traversal to `void` should return None (doesn't exist)
|
// traversal to `v/oid` should return None (doesn't exist)
|
||||||
{
|
{
|
||||||
let resp = descend_to(
|
let resp = descend_to(
|
||||||
&directory_service,
|
&directory_service,
|
||||||
node_directory_complicated.clone(),
|
node_directory_complicated.clone(),
|
||||||
&PathBuf::from("//v/oid"),
|
"v/oid".parse::<PathBuf>().unwrap(),
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
.expect("must succeed");
|
.expect("must succeed");
|
||||||
|
@ -211,25 +186,12 @@ mod tests {
|
||||||
let resp = descend_to(
|
let resp = descend_to(
|
||||||
&directory_service,
|
&directory_service,
|
||||||
node_directory_complicated.clone(),
|
node_directory_complicated.clone(),
|
||||||
&PathBuf::from("keep/.keep/foo"),
|
"keep/.keep/foo".parse::<PathBuf>().unwrap(),
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
.expect("must succeed");
|
.expect("must succeed");
|
||||||
|
|
||||||
assert_eq!(None, resp);
|
assert_eq!(None, resp);
|
||||||
}
|
}
|
||||||
|
|
||||||
// traversal to a subpath of '/' should return the root node.
|
|
||||||
{
|
|
||||||
let resp = descend_to(
|
|
||||||
&directory_service,
|
|
||||||
node_directory_complicated.clone(),
|
|
||||||
&PathBuf::from("/"),
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.expect("must succeed");
|
|
||||||
|
|
||||||
assert_eq!(Some(node_directory_complicated), resp);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -305,6 +305,9 @@ impl TvixStoreIO {
|
||||||
};
|
};
|
||||||
|
|
||||||
// now with the root_node and sub_path, descend to the node requested.
|
// now with the root_node and sub_path, descend to the node requested.
|
||||||
|
// We convert sub_path to the castore model here.
|
||||||
|
let sub_path = tvix_castore::PathBuf::from_host_path(sub_path, true)?;
|
||||||
|
|
||||||
directoryservice::descend_to(&self.directory_service, root_node, sub_path)
|
directoryservice::descend_to(&self.directory_service, root_node, sub_path)
|
||||||
.await
|
.await
|
||||||
.map_err(|e| std::io::Error::new(io::ErrorKind::Other, e))
|
.map_err(|e| std::io::Error::new(io::ErrorKind::Other, e))
|
||||||
|
|
Loading…
Reference in a new issue