2020-06-15 06:29:07 -05:00
|
|
|
//! Partitions a list of files into disjoint subsets.
|
|
|
|
//!
|
|
|
|
//! Files which do not belong to any explicitly configured `FileSet` belong to
|
|
|
|
//! the default `FileSet`.
|
2020-06-19 08:07:32 -05:00
|
|
|
use std::{fmt, iter};
|
2020-06-15 06:29:07 -05:00
|
|
|
|
|
|
|
use paths::AbsPathBuf;
|
|
|
|
use rustc_hash::FxHashMap;
|
|
|
|
|
|
|
|
use crate::{FileId, Vfs, VfsPath};
|
|
|
|
|
|
|
|
#[derive(Default, Clone, Eq, PartialEq)]
|
|
|
|
pub struct FileSet {
|
|
|
|
files: FxHashMap<VfsPath, FileId>,
|
|
|
|
paths: FxHashMap<FileId, VfsPath>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FileSet {
|
|
|
|
pub fn resolve_path(&self, anchor: FileId, path: &str) -> Option<FileId> {
|
|
|
|
let mut base = self.paths[&anchor].clone();
|
|
|
|
base.pop();
|
|
|
|
let path = base.join(path);
|
|
|
|
let res = self.files.get(&path).copied();
|
|
|
|
res
|
|
|
|
}
|
|
|
|
pub fn insert(&mut self, file_id: FileId, path: VfsPath) {
|
|
|
|
self.files.insert(path.clone(), file_id);
|
|
|
|
self.paths.insert(file_id, path);
|
|
|
|
}
|
|
|
|
pub fn iter(&self) -> impl Iterator<Item = FileId> + '_ {
|
|
|
|
self.paths.keys().copied()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Debug for FileSet {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
f.debug_struct("FileSet").field("n_files", &self.files.len()).finish()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct FileSetConfig {
|
|
|
|
n_file_sets: usize,
|
|
|
|
roots: Vec<(AbsPathBuf, usize)>,
|
|
|
|
}
|
|
|
|
|
2020-06-19 08:07:32 -05:00
|
|
|
impl Default for FileSetConfig {
|
|
|
|
fn default() -> Self {
|
|
|
|
FileSetConfig::builder().build()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-15 06:29:07 -05:00
|
|
|
impl FileSetConfig {
|
|
|
|
pub fn builder() -> FileSetConfigBuilder {
|
|
|
|
FileSetConfigBuilder::default()
|
|
|
|
}
|
|
|
|
pub fn partition(&self, vfs: &Vfs) -> Vec<FileSet> {
|
|
|
|
let mut res = vec![FileSet::default(); self.len()];
|
|
|
|
for (file_id, path) in vfs.iter() {
|
|
|
|
let root = self.classify(&path);
|
|
|
|
res[root].insert(file_id, path)
|
|
|
|
}
|
|
|
|
res
|
|
|
|
}
|
|
|
|
fn len(&self) -> usize {
|
|
|
|
self.n_file_sets
|
|
|
|
}
|
|
|
|
fn classify(&self, path: &VfsPath) -> usize {
|
2020-06-19 08:07:32 -05:00
|
|
|
let path = match path.as_path() {
|
|
|
|
Some(it) => it,
|
|
|
|
None => return self.len() - 1,
|
|
|
|
};
|
|
|
|
let idx = match self.roots.binary_search_by(|(p, _)| p.as_path().cmp(path)) {
|
|
|
|
Ok(it) => it,
|
|
|
|
Err(it) => it.saturating_sub(1),
|
|
|
|
};
|
|
|
|
if path.starts_with(&self.roots[idx].0) {
|
|
|
|
self.roots[idx].1
|
|
|
|
} else {
|
|
|
|
self.len() - 1
|
2020-06-15 06:29:07 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct FileSetConfigBuilder {
|
|
|
|
roots: Vec<Vec<AbsPathBuf>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for FileSetConfigBuilder {
|
|
|
|
fn default() -> Self {
|
|
|
|
FileSetConfigBuilder { roots: Vec::new() }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FileSetConfigBuilder {
|
2020-06-19 08:07:32 -05:00
|
|
|
pub fn len(&self) -> usize {
|
|
|
|
self.roots.len()
|
|
|
|
}
|
2020-06-15 06:29:07 -05:00
|
|
|
pub fn add_file_set(&mut self, roots: Vec<AbsPathBuf>) {
|
|
|
|
self.roots.push(roots)
|
|
|
|
}
|
|
|
|
pub fn build(self) -> FileSetConfig {
|
|
|
|
let n_file_sets = self.roots.len() + 1;
|
|
|
|
let mut roots: Vec<(AbsPathBuf, usize)> = self
|
|
|
|
.roots
|
|
|
|
.into_iter()
|
|
|
|
.enumerate()
|
|
|
|
.flat_map(|(i, paths)| paths.into_iter().zip(iter::repeat(i)))
|
|
|
|
.collect();
|
2020-06-19 08:07:32 -05:00
|
|
|
roots.sort();
|
2020-06-15 06:29:07 -05:00
|
|
|
FileSetConfig { n_file_sets, roots }
|
|
|
|
}
|
|
|
|
}
|