use super::{Context, Module, ModuleConfig}; use crate::formatter::VersionFormatter; use crate::configs::cmake::CMakeConfig; use crate::formatter::StringFormatter; /// Creates a module with the current `CMake` version pub fn module<'a>(context: &'a Context) -> Option> { let mut module = context.new_module("cmake"); let config = CMakeConfig::try_load(module.config); let is_cmake_project = context .try_begin_scan()? .set_files(&config.detect_files) .set_extensions(&config.detect_extensions) .set_folders(&config.detect_folders) .is_match(); if !is_cmake_project { return None; } let parsed = StringFormatter::new(config.format).and_then(|formatter| { formatter .map_meta(|variable, _| match variable { "symbol" => Some(config.symbol), _ => None, }) .map_style(|variable| match variable { "style" => Some(Ok(config.style)), _ => None, }) .map(|variable| match variable { "version" => { let cmake_version = parse_cmake_version(&context.exec_cmd("cmake", &["--version"])?.stdout)?; VersionFormatter::format_module_version( module.get_name(), &cmake_version, config.version_format, ) .map(Ok) } _ => None, }) .parse(None, Some(context)) }); module.set_segments(match parsed { Ok(segments) => segments, Err(error) => { log::warn!("Error in module `cmake`: \n{}", error); return None; } }); Some(module) } fn parse_cmake_version(cmake_version: &str) -> Option { Some( cmake_version //split into ["cmake" "version" "3.10.2", ...] .split_whitespace() // get down to "3.10.2" .nth(2)? .to_string(), ) } #[cfg(test)] mod tests { use crate::test::ModuleRenderer; use nu_ansi_term::Color; use std::fs::File; use std::io; #[test] fn folder_without_cmake_lists() -> io::Result<()> { let dir = tempfile::tempdir()?; let actual = ModuleRenderer::new("cmake").path(dir.path()).collect(); let expected = None; assert_eq!(expected, actual); dir.close() } #[test] fn folder_with_cmake_lists() -> io::Result<()> { let dir = tempfile::tempdir()?; File::create(dir.path().join("CMakeLists.txt"))?.sync_all()?; let actual = ModuleRenderer::new("cmake").path(dir.path()).collect(); let expected = Some(format!("via {}", Color::Blue.bold().paint("△ v3.17.3 "))); assert_eq!(expected, actual); dir.close() } #[test] fn buildfolder_with_cmake_cache() -> io::Result<()> { let dir = tempfile::tempdir()?; File::create(dir.path().join("CMakeCache.txt"))?.sync_all()?; let actual = ModuleRenderer::new("cmake").path(dir.path()).collect(); let expected = Some(format!("via {}", Color::Blue.bold().paint("△ v3.17.3 "))); assert_eq!(expected, actual); dir.close() } }