Skip to content

Commit

Permalink
Add get_parent_item method
Browse files Browse the repository at this point in the history
  • Loading branch information
Peter554 committed Dec 30, 2024
1 parent de1f6c6 commit df3ce7c
Show file tree
Hide file tree
Showing 2 changed files with 88 additions and 3 deletions.
22 changes: 19 additions & 3 deletions src/package_info/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ new_key_type! {
}

/// A python package.
#[derive(Debug, Clone)]
#[derive(Debug, Clone, PartialEq)]
pub struct Package {
/// The absolute filesystem path to this package.
pub path: PathBuf,
Expand Down Expand Up @@ -71,7 +71,7 @@ impl Package {
}

/// A python module.
#[derive(Debug, Clone)]
#[derive(Debug, Clone, PartialEq)]
pub struct Module {
/// The absolute filesystem path to this module.
pub path: PathBuf,
Expand Down Expand Up @@ -157,7 +157,7 @@ pub struct PackageInfo {
}

/// A unified representation of an item within a package.
#[derive(Debug, Clone)]
#[derive(Debug, Clone, PartialEq)]
pub enum PackageItem<'a> {
/// A package.
Package(&'a Package),
Expand Down Expand Up @@ -259,6 +259,22 @@ impl<'a> PackageItem<'a> {
PackageItem::Module(m) => m.token.into(),
}
}

/// The filesystem path for this package item.
pub fn path(&'a self) -> &Path {
match self {
PackageItem::Package(p) => &p.path,
PackageItem::Module(m) => &m.path,
}
}

/// The pypath for this package item.
pub fn pypath(&'a self) -> &Pypath {
match self {
PackageItem::Package(p) => &p.pypath,
PackageItem::Module(m) => &m.pypath,
}
}
}

impl PackageInfo {
Expand Down
69 changes: 69 additions & 0 deletions src/package_info/queries.rs
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,25 @@ impl PackageInfo {
}

/// Get a package item via the associated pypath.
///
/// ```
/// # use anyhow::Result;
/// # use pyimports::{testpackage,TestPackage};
/// use pyimports::PackageInfo;
///
/// # fn main() -> Result<()> {
/// let test_package = testpackage! {
/// "__init__.py" => "",
/// "foo.py" => ""
/// };
///
/// let package_info = PackageInfo::build(&test_package.path())?;
///
/// let foo = package_info.get_item_by_pypath("testpackage.foo")?;
/// assert!(foo.is_some());
/// # Ok(())
/// # }
/// ```
pub fn get_item_by_pypath<T: IntoPypath>(&self, pypath: T) -> Result<Option<PackageItem>> {
let pypath = pypath.into_pypath()?;
if let Some(package) = self.packages_by_pypath.get(pypath.borrow()) {
Expand Down Expand Up @@ -120,6 +139,25 @@ impl PackageInfo {
self.get_package(self.root).unwrap()
}

/// Get the parent package of the passed package item.
pub fn get_parent_item(&self, token: PackageItemToken) -> Result<Option<&Package>> {
let item = self.get_item(token)?;
let parent = match item {
PackageItem::Package(package) => match package.parent {
Some(parent) => Some(self.get_item(parent.into())?),
None => None,
},
PackageItem::Module(module) => Some(self.get_item(module.parent.into())?),
};
match parent {
Some(parent) => match parent {
PackageItem::Package(parent) => Ok(Some(parent)),
PackageItem::Module(_) => panic!(),
},
None => Ok(None),
}
}

/// Get an iterator over the child items of the passed package.
///
/// ```
Expand Down Expand Up @@ -228,6 +266,37 @@ mod tests {
})
}

#[test]
fn test_get_parent_item() -> Result<()> {
let test_package = create_test_package()?;
let package_info = PackageInfo::build(test_package.path())?;

let root_package = package_info.get_root();
let colors_package = package_info
.get_item_by_pypath("testpackage.colors")?
.unwrap();
let main = package_info
.get_item_by_pypath("testpackage.main")?
.unwrap();

assert_eq!(
package_info.get_parent_item(colors_package.token())?,
Some(root_package)
);

assert_eq!(
package_info.get_parent_item(main.token())?,
Some(root_package)
);

assert_eq!(
package_info.get_parent_item(root_package.token.into())?,
None
);

Ok(())
}

#[test]
fn test_get_child_items() -> Result<()> {
let test_package = create_test_package()?;
Expand Down

0 comments on commit df3ce7c

Please sign in to comment.