nbcl 0.1.0

Configuration language designed to be easy and understandable.
Documentation
use crate::error::{NbclError, Result};
use std::path::{Path, PathBuf};

/// Simple module resolver.
#[derive(Debug, Clone)]
pub struct FileModuleResolver {
    file_path: PathBuf,
}

impl FileModuleResolver {
    /// Create a new FileModuleResolver
    pub fn new(fpath: PathBuf) -> Self {
        Self { file_path: fpath }
    }

    /// Resolve a module based on relative string (e.g. test/node.nbl)
    pub fn find_target(&self, path_str: &str) -> Result<PathBuf> {
        // Determine the base directory
        // If file_path is "main.nbl", parent() might be empty, so we default to "."
        let current_dir = self
            .file_path
            .parent()
            .filter(|p| !p.as_os_str().is_empty())
            .unwrap_or_else(|| Path::new("."));

        // Join and attempt to resolve the absolute path
        let full_path = current_dir.join(path_str);

        full_path.canonicalize().map_err(|e| {
            let (message, hint) = match e.kind() {
                std::io::ErrorKind::NotFound => {
                    let msg = format!(
                        "Module not found: '{}' (searched in '{}')",
                        path_str,
                        current_dir.display()
                    );
                    let hint =
                        "Ensure that the module exists and try adjusting the path".to_string();

                    (msg, Some(hint))
                }
                std::io::ErrorKind::PermissionDenied => {
                    let msg = format!("Permission denied accessing module: '{}'", path_str);
                    let hint = "Set proper file permissions and try again.".to_string();

                    (msg, Some(hint))
                }
                _ => (format!("Failed to resolve module path '{}': {}", path_str, e), None),
            };

            NbclError::IO { message, hint, path: full_path }
        })
    }
}