1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96
use comments_docsource::CommentDocumentation; use core::fmt; use options_docsource::OptionDocumentation; use rayon::iter::{IntoParallelRefIterator, ParallelIterator}; use std::path::PathBuf; use thiserror::Error; use xml_docsource::XmlFuncDocumentation; pub mod comments_docsource; pub mod nixpkgs_tree_docsource; pub mod options_docsource; pub mod xml_docsource; #[derive(Error, Debug)] pub enum Errors { #[error("IO Error for file {}: {}", .filename, .err)] FileIo { filename: String, err: std::io::Error, }, #[error("Failed to perform IO on a cache file")] CacheFileIo(#[from] std::io::Error), #[error("Failed to serialize/deserialize cache(bincode)")] Bincode(#[from] bincode::Error), #[error("Failed to serialize/deserialize cache(serde_json)")] SerdeJson(#[from] serde_json::Error), #[error("XML parsing error for file {}: {}", .filename, .err)] XmlParse { filename: String, err: roxmltree::Error, }, } pub enum DocEntry { OptionDoc(OptionDocumentation), CommentDoc(CommentDocumentation), XmlFuncDoc(XmlFuncDocumentation), NixpkgsTreeDoc(String), } impl DocEntry { pub fn name(&self) -> String { match self { DocEntry::OptionDoc(x) => x.name(), DocEntry::CommentDoc(x) => x.name(), DocEntry::XmlFuncDoc(x) => x.name(), DocEntry::NixpkgsTreeDoc(x) => x.clone(), } } pub fn pretty_printed(&self) -> String { match self { DocEntry::OptionDoc(x) => x.pretty_printed(), DocEntry::CommentDoc(x) => x.pretty_printed(), DocEntry::XmlFuncDoc(x) => x.pretty_printed(), DocEntry::NixpkgsTreeDoc(x) => x.clone(), } } } pub trait DocSource { fn all_keys(&self) -> Vec<&str>; fn search(&self, query: &str) -> Vec<DocEntry>; fn search_liberal(&self, query: &str) -> Vec<DocEntry>; } #[derive(Default)] pub struct AggregateDocSource { sources: Vec<Box<dyn DocSource + Sync>>, } impl AggregateDocSource { pub fn add_source(&mut self, source: Box<dyn DocSource + Sync>) { self.sources.push(source) } } impl DocSource for AggregateDocSource { fn all_keys(&self) -> Vec<&str> { self.sources .par_iter() .flat_map(|source| source.all_keys()) .collect() } fn search(&self, query: &str) -> Vec<DocEntry> { self.sources .par_iter() .flat_map(|source| source.search(query)) .collect() } fn search_liberal(&self, query: &str) -> Vec<DocEntry> { self.sources .par_iter() .flat_map(|source| source.search_liberal(query)) .collect() } }