use std::path::PathBuf;
use cha_core::SourceFile;
use cha_parser::parse_file;
fn fixture(name: &str) -> SourceFile {
let path = PathBuf::from(format!("tests/fixtures/{name}"));
let content = std::fs::read_to_string(&path).unwrap();
SourceFile::new(path, content)
}
#[test]
fn rust_simple_fn() {
let model = parse_file(&fixture("simple.rs")).unwrap();
assert_eq!(model.language, "rust");
assert_eq!(model.total_lines, 3);
assert_eq!(model.functions.len(), 1);
assert_eq!(model.functions[0].name, "hello");
assert_eq!(model.functions[0].line_count, 3);
assert_eq!(model.functions[0].complexity, 1);
assert_eq!(model.classes.len(), 0);
assert_eq!(model.imports.len(), 0);
}
#[test]
fn rust_complex_fn() {
let model = parse_file(&fixture("complex.rs")).unwrap();
assert_eq!(model.functions.len(), 1);
let f = &model.functions[0];
assert_eq!(f.name, "decide");
assert_eq!(f.line_count, 17);
assert_eq!(f.complexity, 10);
}
#[test]
fn rust_structs_and_impl() {
let model = parse_file(&fixture("structs.rs")).unwrap();
assert_eq!(model.classes.len(), 1);
assert_eq!(model.classes[0].name, "Point");
assert_eq!(model.functions.len(), 2);
assert_eq!(model.functions[0].name, "new");
assert_eq!(model.functions[1].name, "distance");
}
#[test]
fn rust_imports() {
let model = parse_file(&fixture("imports.rs")).unwrap();
assert_eq!(model.imports.len(), 3);
assert_eq!(model.imports[0].source, "std::collections::HashMap");
assert_eq!(model.imports[1].source, "std::path::PathBuf");
assert_eq!(model.imports[2].source, "std::io::Read");
assert_eq!(model.imports[0].line, 1);
}
#[test]
fn rust_exports() {
let model = parse_file(&fixture("exports.rs")).unwrap();
assert_eq!(model.functions.len(), 2);
assert!(model.functions[0].is_exported); assert!(!model.functions[1].is_exported); assert_eq!(model.classes.len(), 2);
assert!(model.classes[0].is_exported); assert!(!model.classes[1].is_exported); }
#[test]
fn ts_simple() {
let model = parse_file(&fixture("simple.ts")).unwrap();
assert_eq!(model.language, "typescript");
assert_eq!(model.total_lines, 18);
assert_eq!(model.functions.len(), 1);
assert_eq!(model.functions[0].name, "greet");
assert_eq!(model.classes.len(), 1);
assert_eq!(model.classes[0].name, "Greeter");
assert_eq!(model.classes[0].method_count, 2);
assert_eq!(model.imports.len(), 2);
}
#[test]
fn tsx_exports() {
let model = parse_file(&fixture("exports.tsx")).unwrap();
assert_eq!(model.functions.len(), 2);
assert!(model.functions[0].is_exported); assert!(!model.functions[1].is_exported); assert_eq!(model.classes.len(), 2);
assert!(model.classes[0].is_exported); assert!(!model.classes[1].is_exported); }
#[test]
fn unsupported_extension() {
let file = SourceFile::new(PathBuf::from("test.rb"), "def foo; end".into());
assert!(parse_file(&file).is_none());
}
#[test]
fn python_simple() {
let model = parse_file(&fixture("simple.py")).unwrap();
assert_eq!(model.language, "python");
assert_eq!(model.functions.len(), 13); let top_fn = model
.functions
.iter()
.find(|f| f.name == "simple_function")
.unwrap();
assert_eq!(top_fn.parameter_count, 2);
assert_eq!(top_fn.complexity, 1);
let complex_fn = model
.functions
.iter()
.find(|f| f.name == "complex_function")
.unwrap();
assert!(complex_fn.complexity >= 4); assert_eq!(complex_fn.optional_param_count, 1);
assert!(complex_fn.comment_lines >= 1);
}
#[test]
fn python_classes() {
let model = parse_file(&fixture("simple.py")).unwrap();
assert_eq!(model.classes.len(), 3);
let animal = model.classes.iter().find(|c| c.name == "Animal").unwrap();
assert_eq!(animal.method_count, 4);
assert!(animal.field_count >= 2); assert!(animal.has_listener_field);
assert!(animal.has_notify_method);
assert!(animal.has_behavior);
let dog = model.classes.iter().find(|c| c.name == "Dog").unwrap();
assert_eq!(dog.parent_name.as_deref(), Some("Animal"));
assert!(dog.has_behavior);
let iface = model
.classes
.iter()
.find(|c| c.name == "EmptyInterface")
.unwrap();
assert!(iface.is_interface);
}
#[test]
fn python_imports() {
let model = parse_file(&fixture("simple.py")).unwrap();
assert!(model.imports.len() >= 4); assert!(model.imports.iter().any(|i| i.source == "os"));
assert!(model.imports.iter().any(|i| i.source.contains("Path")));
}
#[test]
fn python_chain_depth() {
let model = parse_file(&fixture("simple.py")).unwrap();
let chain_fn = model
.functions
.iter()
.find(|f| f.name == "long_chain_example")
.unwrap();
assert!(chain_fn.chain_depth >= 4);
}
#[test]
fn python_delegating() {
let model = parse_file(&fixture("simple.py")).unwrap();
let del_fn = model
.functions
.iter()
.find(|f| f.name == "delegating")
.unwrap();
assert!(del_fn.is_delegating);
}
#[test]
fn go_simple() {
let model = parse_file(&fixture("simple.go")).unwrap();
assert_eq!(model.language, "go");
assert_eq!(model.functions.len(), 2);
assert_eq!(model.functions[0].name, "Hello");
assert!(model.functions[0].is_exported);
assert_eq!(model.functions[1].name, "add");
assert!(!model.functions[1].is_exported);
assert_eq!(model.imports.len(), 1);
}
#[test]
fn go_structs() {
let model = parse_file(&fixture("structs.go")).unwrap();
assert!(model.classes.len() >= 2);
let server = model.classes.iter().find(|c| c.name == "Server").unwrap();
assert!(server.is_exported);
assert_eq!(server.field_count, 2);
let handler = model.classes.iter().find(|c| c.name == "Handler").unwrap();
assert!(handler.is_interface);
assert!(model.functions.iter().any(|f| f.name == "Start"));
}
#[test]
fn go_complex() {
let model = parse_file(&fixture("complex.go")).unwrap();
let f = &model.functions[0];
assert_eq!(f.name, "Complex");
assert!(f.complexity > 1);
assert!(f.switch_arms >= 4);
}
#[test]
fn c_simple() {
let model = parse_file(&fixture("simple.c")).unwrap();
assert_eq!(model.language, "c");
assert_eq!(model.functions.len(), 2);
assert_eq!(model.functions[0].name, "add");
assert_eq!(model.functions[0].parameter_count, 2);
assert_eq!(model.imports.len(), 2);
assert!(
model.classes.len() >= 2,
"expected >= 2 structs, got {}",
model.classes.len()
);
assert!(
model.classes.iter().any(|c| c.name == "Point"),
"missing typedef struct Point"
);
assert!(
model.classes.iter().any(|c| c.name == "Color"),
"missing struct Color"
);
}
#[test]
fn cpp_simple() {
let model = parse_file(&fixture("simple.cpp")).unwrap();
assert_eq!(model.language, "cpp");
assert!(model.functions.len() >= 1);
let factorial = model
.functions
.iter()
.find(|f| f.name == "factorial")
.unwrap();
assert!(factorial.complexity > 1);
assert!(model.classes.len() >= 1);
let animal = model.classes.iter().find(|c| c.name == "Animal").unwrap();
assert!(animal.field_count >= 2);
}
#[test]
fn cpp_macro_class() {
let model = parse_file(&fixture("classes.cpp")).unwrap();
assert_eq!(model.language, "cpp");
assert!(
model.classes.iter().any(|c| c.name == "Animal"),
"missing normal class Animal"
);
assert!(
model.classes.iter().any(|c| c.name == "Shape"),
"missing macro-decorated class Shape"
);
}
#[test]
fn cpp_h_sniffing() {
let model = parse_file(&fixture("header.h")).unwrap();
assert_eq!(model.language, "cpp");
assert!(
model.classes.iter().any(|c| c.name == "Widget"),
"missing class Widget from .h file"
);
assert!(
model.classes.iter().any(|c| c.name == "Point"),
"missing struct Point from .h file"
);
}
#[test]
fn go_cognitive_complexity() {
let model = parse_file(&fixture("cognitive.go")).unwrap();
let f = model
.functions
.iter()
.find(|f| f.name == "SumOfPrimes")
.unwrap();
assert!(
f.cognitive_complexity >= 5,
"expected cognitive complexity >= 5, got {}",
f.cognitive_complexity
);
assert!(
f.cognitive_complexity > f.complexity || f.cognitive_complexity == f.complexity,
"cognitive should generally be >= cyclomatic for nested code"
);
}
fn parse_rust(content: &str) -> cha_core::SourceModel {
use std::path::PathBuf;
let file = cha_core::SourceFile::new(PathBuf::from("test.rs"), content.into());
parse_file(&file).unwrap()
}
#[test]
fn rust_param_origin_external() {
let src = "use tree_sitter::Node;\nfn handle(n: &Node) {}\n";
let m = parse_rust(src);
let tr = &m.functions[0].parameter_types[0];
assert_eq!(tr.name, "Node");
assert_eq!(
tr.origin,
cha_core::TypeOrigin::External("tree_sitter".into())
);
}
#[test]
fn rust_param_origin_local_via_crate() {
let src = "use crate::model::Finding;\nfn handle(f: &Finding) {}\n";
let m = parse_rust(src);
let tr = &m.functions[0].parameter_types[0];
assert_eq!(tr.origin, cha_core::TypeOrigin::Local);
}
#[test]
fn rust_param_origin_primitive_std() {
let src = "use std::collections::HashMap;\nfn handle(m: HashMap) {}\n";
let m = parse_rust(src);
let tr = &m.functions[0].parameter_types[0];
assert_eq!(tr.origin, cha_core::TypeOrigin::Primitive);
}
#[test]
fn rust_param_origin_rename_alias() {
let src = "use tree_sitter::Node as TsNode;\nfn handle(n: &TsNode) {}\n";
let m = parse_rust(src);
let tr = &m.functions[0].parameter_types[0];
assert_eq!(
tr.origin,
cha_core::TypeOrigin::External("tree_sitter".into())
);
}
#[test]
fn rust_param_origin_unknown_no_import() {
let src = "fn handle(n: SomeType) {}\n";
let m = parse_rust(src);
let tr = &m.functions[0].parameter_types[0];
assert_eq!(tr.origin, cha_core::TypeOrigin::Unknown);
}
#[test]
fn rust_param_origin_group_use() {
let src = "use tree_sitter::{Node, Parser};\nfn handle(n: &Node, p: &Parser) {}\n";
let m = parse_rust(src);
assert_eq!(
m.functions[0].parameter_types[0].origin,
cha_core::TypeOrigin::External("tree_sitter".into())
);
assert_eq!(
m.functions[0].parameter_types[1].origin,
cha_core::TypeOrigin::External("tree_sitter".into())
);
}
fn parse_ts(content: &str) -> cha_core::SourceModel {
use std::path::PathBuf;
let file = cha_core::SourceFile::new(PathBuf::from("test.ts"), content.into());
parse_file(&file).unwrap()
}
#[test]
fn ts_param_origin_external() {
let src = "import { Foo } from 'pkg';\nfunction handle(x: Foo): void {}\n";
let m = parse_ts(src);
let tr = &m.functions[0].parameter_types[0];
assert_eq!(tr.name, "Foo");
assert_eq!(tr.origin, cha_core::TypeOrigin::External("pkg".into()));
}
#[test]
fn ts_param_origin_local_relative() {
let src = "import { Helper } from './helper';\nfunction handle(h: Helper): void {}\n";
let m = parse_ts(src);
assert_eq!(
m.functions[0].parameter_types[0].origin,
cha_core::TypeOrigin::Local
);
}
fn parse_py(content: &str) -> cha_core::SourceModel {
use std::path::PathBuf;
let file = cha_core::SourceFile::new(PathBuf::from("test.py"), content.into());
parse_file(&file).unwrap()
}
#[test]
fn py_param_origin_external() {
let src = "from pydantic import BaseModel\n\ndef handle(m: BaseModel) -> None:\n pass\n";
let m = parse_py(src);
let tr = &m.functions[0].parameter_types[0];
assert_eq!(tr.origin, cha_core::TypeOrigin::External("pydantic".into()));
}
#[test]
fn py_param_origin_stdlib_primitive() {
let src = "from typing import List\n\ndef handle(xs: List) -> None:\n pass\n";
let m = parse_py(src);
assert_eq!(
m.functions[0].parameter_types[0].origin,
cha_core::TypeOrigin::Primitive
);
}
#[test]
fn py_param_origin_relative_local() {
let src = "from .models import User\n\ndef handle(u: User) -> None:\n pass\n";
let m = parse_py(src);
assert_eq!(
m.functions[0].parameter_types[0].origin,
cha_core::TypeOrigin::Local
);
}
fn parse_go(content: &str) -> cha_core::SourceModel {
use std::path::PathBuf;
let file = cha_core::SourceFile::new(PathBuf::from("test.go"), content.into());
parse_file(&file).unwrap()
}
#[test]
fn go_param_origin_builtin() {
let src = "package p\n\nfunc handle(x int) {}\n";
let m = parse_go(src);
assert_eq!(
m.functions[0].parameter_types[0].origin,
cha_core::TypeOrigin::Primitive
);
}
#[test]
fn go_param_origin_stdlib() {
let src = "package p\n\nimport \"context\"\n\nfunc handle(ctx context.Context) {}\n";
let m = parse_go(src);
let tr = &m.functions[0].parameter_types[0];
assert_eq!(tr.origin, cha_core::TypeOrigin::Primitive);
assert_eq!(tr.name, "Context");
}
#[test]
fn go_param_origin_third_party() {
let src = "package p\n\nimport \"github.com/foo/bar\"\n\nfunc handle(x bar.Thing) {}\n";
let m = parse_go(src);
let tr = &m.functions[0].parameter_types[0];
assert_eq!(
tr.origin,
cha_core::TypeOrigin::External("github.com/foo/bar".into())
);
assert_eq!(tr.name, "Thing");
}
#[test]
fn go_param_origin_pointer() {
let src = "package p\n\nimport \"github.com/foo/bar\"\n\nfunc handle(x *bar.Thing) {}\n";
let m = parse_go(src);
assert_eq!(
m.functions[0].parameter_types[0].origin,
cha_core::TypeOrigin::External("github.com/foo/bar".into())
);
}
fn parse_c(content: &str) -> cha_core::SourceModel {
use std::path::PathBuf;
let file = cha_core::SourceFile::new(PathBuf::from("test.c"), content.into());
parse_file(&file).unwrap()
}
#[test]
fn c_param_origin_primitive() {
let src = "void handle(int x, size_t y) { }\n";
let m = parse_c(src);
assert_eq!(
m.functions[0].parameter_types[0].origin,
cha_core::TypeOrigin::Primitive
);
assert_eq!(
m.functions[0].parameter_types[1].origin,
cha_core::TypeOrigin::Primitive
);
}
#[test]
fn c_param_origin_unknown_struct() {
let src = "void handle(cmark_node_t *n) { }\n";
let m = parse_c(src);
let tr = &m.functions[0].parameter_types[0];
assert_eq!(tr.origin, cha_core::TypeOrigin::Unknown);
}