water closet (actually a word count)
This commit is contained in:
parent
07d069e2ef
commit
3e5f17018c
4 changed files with 83 additions and 2 deletions
|
@ -2,6 +2,7 @@ pub mod fix;
|
||||||
pub mod generate;
|
pub mod generate;
|
||||||
mod parse;
|
mod parse;
|
||||||
pub mod serve;
|
pub mod serve;
|
||||||
|
pub mod wc;
|
||||||
|
|
||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
|
@ -37,6 +38,9 @@ pub enum Command {
|
||||||
serve: ServeArgs,
|
serve: ServeArgs,
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/// Count words in the treehouse's branches.
|
||||||
|
Wc(#[clap(flatten)] WcArgs),
|
||||||
|
|
||||||
/// Generates a new ulid and prints it to stdout.
|
/// Generates a new ulid and prints it to stdout.
|
||||||
Ulid,
|
Ulid,
|
||||||
}
|
}
|
||||||
|
@ -75,6 +79,13 @@ pub struct ServeArgs {
|
||||||
pub port: u16,
|
pub port: u16,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Args)]
|
||||||
|
pub struct WcArgs {
|
||||||
|
/// A list of paths to report the word counts of.
|
||||||
|
/// If no paths are provided, the entire tree is word-counted.
|
||||||
|
pub paths: Vec<PathBuf>,
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct Paths<'a> {
|
pub struct Paths<'a> {
|
||||||
pub target_dir: &'a Path,
|
pub target_dir: &'a Path,
|
||||||
|
|
69
crates/treehouse/src/cli/wc.rs
Normal file
69
crates/treehouse/src/cli/wc.rs
Normal file
|
@ -0,0 +1,69 @@
|
||||||
|
use std::{ffi::OsStr, path::Path};
|
||||||
|
|
||||||
|
use anyhow::Context;
|
||||||
|
use treehouse_format::ast::{Branch, Roots};
|
||||||
|
use walkdir::WalkDir;
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
cli::parse::parse_tree_with_diagnostics,
|
||||||
|
state::{Source, Treehouse},
|
||||||
|
};
|
||||||
|
|
||||||
|
use super::WcArgs;
|
||||||
|
|
||||||
|
fn wc_branch(source: &str, branch: &Branch) -> usize {
|
||||||
|
let word_count = source[branch.content.clone()].split_whitespace().count();
|
||||||
|
word_count
|
||||||
|
+ branch
|
||||||
|
.children
|
||||||
|
.iter()
|
||||||
|
.map(|branch| wc_branch(source, branch))
|
||||||
|
.sum::<usize>()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn wc_roots(source: &str, roots: &Roots) -> usize {
|
||||||
|
roots
|
||||||
|
.branches
|
||||||
|
.iter()
|
||||||
|
.map(|branch| wc_branch(source, branch))
|
||||||
|
.sum()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn wc_cli(content_dir: &Path, mut wc_args: WcArgs) -> anyhow::Result<()> {
|
||||||
|
if wc_args.paths.is_empty() {
|
||||||
|
for entry in WalkDir::new(content_dir) {
|
||||||
|
let entry = entry?;
|
||||||
|
if entry.file_type().is_file() && entry.path().extension() == Some(OsStr::new("tree")) {
|
||||||
|
wc_args.paths.push(entry.into_path());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut treehouse = Treehouse::new();
|
||||||
|
|
||||||
|
let mut total = 0;
|
||||||
|
|
||||||
|
for path in &wc_args.paths {
|
||||||
|
let file = std::fs::read_to_string(path)
|
||||||
|
.with_context(|| format!("cannot read file to word count: {path:?}"))?;
|
||||||
|
let path_without_ext = path.with_extension("");
|
||||||
|
let utf8_filename = path_without_ext
|
||||||
|
.strip_prefix(content_dir)
|
||||||
|
.expect("paths should be rooted within the content directory")
|
||||||
|
.to_string_lossy();
|
||||||
|
|
||||||
|
let file_id = treehouse.add_file(utf8_filename.into_owned(), Source::Other(file));
|
||||||
|
if let Ok(parsed) = parse_tree_with_diagnostics(&mut treehouse, file_id) {
|
||||||
|
let source = treehouse.source(file_id);
|
||||||
|
let word_count = wc_roots(source.input(), &parsed);
|
||||||
|
println!("{word_count:>8} {}", treehouse.filename(file_id));
|
||||||
|
total += word_count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
println!("{total:>8} total");
|
||||||
|
|
||||||
|
treehouse.report_diagnostics()?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
|
@ -5,6 +5,7 @@ use cli::{
|
||||||
fix::{fix_all_cli, fix_file_cli},
|
fix::{fix_all_cli, fix_file_cli},
|
||||||
generate::regenerate_or_report_error,
|
generate::regenerate_or_report_error,
|
||||||
serve::serve,
|
serve::serve,
|
||||||
|
wc::wc_cli,
|
||||||
Command, Paths, ProgramArgs,
|
Command, Paths, ProgramArgs,
|
||||||
};
|
};
|
||||||
use log::{error, info, warn};
|
use log::{error, info, warn};
|
||||||
|
@ -50,6 +51,8 @@ async fn fallible_main() -> anyhow::Result<()> {
|
||||||
Command::Fix(fix_args) => fix_file_cli(fix_args)?,
|
Command::Fix(fix_args) => fix_file_cli(fix_args)?,
|
||||||
Command::FixAll(fix_args) => fix_all_cli(fix_args, &paths)?,
|
Command::FixAll(fix_args) => fix_all_cli(fix_args, &paths)?,
|
||||||
|
|
||||||
|
Command::Wc(wc_args) => wc_cli(paths.content_dir, wc_args)?,
|
||||||
|
|
||||||
Command::Ulid => {
|
Command::Ulid => {
|
||||||
let mut rng = rand::thread_rng();
|
let mut rng = rand::thread_rng();
|
||||||
let ulid = ulid::Generator::new()
|
let ulid = ulid::Generator::new()
|
||||||
|
|
|
@ -144,7 +144,6 @@ pub struct SemaBranch {
|
||||||
pub file_id: FileId,
|
pub file_id: FileId,
|
||||||
|
|
||||||
pub indent_level: usize,
|
pub indent_level: usize,
|
||||||
pub raw_attributes: Option<treehouse_format::pull::Attributes>,
|
|
||||||
pub kind: BranchKind,
|
pub kind: BranchKind,
|
||||||
pub kind_span: Range<usize>,
|
pub kind_span: Range<usize>,
|
||||||
pub content: Range<usize>,
|
pub content: Range<usize>,
|
||||||
|
@ -170,7 +169,6 @@ impl SemaBranch {
|
||||||
let branch = Self {
|
let branch = Self {
|
||||||
file_id,
|
file_id,
|
||||||
indent_level: branch.indent_level,
|
indent_level: branch.indent_level,
|
||||||
raw_attributes: branch.attributes,
|
|
||||||
kind: branch.kind,
|
kind: branch.kind,
|
||||||
kind_span: branch.kind_span,
|
kind_span: branch.kind_span,
|
||||||
content: branch.content,
|
content: branch.content,
|
||||||
|
|
Loading…
Reference in a new issue