pub mod aggregator;
pub mod errors;
pub mod formatters;
pub mod models;
pub mod pipeline;
pub mod templates;
pub mod ticketed_stats;
pub use errors::{ReportError, Result};
pub use models::ReportData;
pub use pipeline::{ReportPipeline, ReportStats};
pub use ticketed_stats::{compute_ticketed_stats, TicketedStats};
#[cfg(test)]
mod tests {
use std::path::PathBuf;
use crate::core::config::{Config, OutputConfig, RepositoryConfig};
use crate::core::db::Database;
use super::aggregator::Aggregator;
use super::formatters::{csv as csv_fmt, json as json_fmt, markdown as md_fmt};
use super::pipeline::ReportPipeline;
fn seed_db() -> Database {
let db = Database::open_in_memory().expect("open db");
let conn = db.connection();
conn.execute(
"INSERT INTO classifications (id, category, subcategory, ticket_id, confidence, method) \
VALUES (1, 'feature', NULL, NULL, 0.9, 'exact_rule')",
[],
)
.expect("insert classification");
conn.execute(
"INSERT INTO commits (sha, author_name, author_email, timestamp, message, repository, \
files_changed, insertions, deletions, classification_id, confidence, is_merge) \
VALUES ('aaa111', 'Alice', 'alice@example.com', '2024-01-15T10:00:00+00:00', \
'feat: add login', 'repo-a', 3, 50, 5, 1, 0.9, 0)",
[],
)
.expect("insert commit 1");
conn.execute(
"INSERT INTO commits (sha, author_name, author_email, timestamp, message, repository, \
files_changed, insertions, deletions, classification_id, confidence, is_merge) \
VALUES ('bbb222', 'Bob', 'bob@example.com', '2024-01-22T11:00:00+00:00', \
'fix: edge case', 'repo-a', 1, 10, 2, NULL, NULL, 0)",
[],
)
.expect("insert commit 2");
db
}
fn baseline_config() -> Config {
Config {
repositories: vec![RepositoryConfig {
path: PathBuf::from("/tmp/repo-a"),
name: Some("repo-a".into()),
..Default::default()
}],
..Default::default()
}
}
#[test]
fn aggregator_builds_report_data() {
let db = seed_db();
let cfg = baseline_config();
let data = Aggregator::build(&db, &cfg).expect("aggregate");
assert_eq!(data.total_commits, 2);
assert_eq!(data.total_authors, 2);
assert!(data.period_start.is_some());
assert!(data.period_end.is_some());
assert_eq!(data.repositories.len(), 1);
assert_eq!(data.repositories[0].name, "repo-a");
assert_eq!(data.repositories[0].author_count, 2);
assert_eq!(data.category_breakdown.get("feature").copied(), Some(1));
assert_eq!(data.weekly_activity.len(), 2);
}
#[test]
fn aggregator_handles_empty_db() {
let db = Database::open_in_memory().expect("open db");
let cfg = baseline_config();
let data = Aggregator::build(&db, &cfg).expect("aggregate");
assert_eq!(data.total_commits, 0);
assert_eq!(data.total_authors, 0);
assert!(data.period_start.is_none());
}
fn tmp_dir(label: &str) -> PathBuf {
let mut path = std::env::temp_dir();
let unique = format!(
"tga-report-{label}-{}-{}",
std::process::id(),
std::time::SystemTime::now()
.duration_since(std::time::UNIX_EPOCH)
.map(|d| d.as_nanos())
.unwrap_or(0)
);
path.push(unique);
std::fs::create_dir_all(&path).expect("mkdir");
path
}
#[test]
fn csv_formatter_writes_files_with_headers() {
let db = seed_db();
let cfg = baseline_config();
let data = Aggregator::build(&db, &cfg).expect("aggregate");
let dir = tmp_dir("csv");
let authors_path = csv_fmt::write_author_csv(&data, &dir).expect("write authors");
let weekly_path = csv_fmt::write_weekly_csv(&data, &dir).expect("write weekly");
let authors_text = std::fs::read_to_string(&authors_path).expect("read");
assert!(authors_text.starts_with("name,email,commit_count"));
assert!(authors_text.contains("Alice"));
assert!(authors_text.contains("Bob"));
let weekly_text = std::fs::read_to_string(&weekly_path).expect("read");
assert!(weekly_text.starts_with("week,author,repository"));
std::fs::remove_dir_all(&dir).ok();
}
#[test]
fn json_formatter_writes_valid_json() {
let db = seed_db();
let cfg = baseline_config();
let data = Aggregator::build(&db, &cfg).expect("aggregate");
let dir = tmp_dir("json");
let path = json_fmt::write_json(&data, &dir).expect("write json");
let text = std::fs::read_to_string(&path).expect("read");
let parsed: serde_json::Value = serde_json::from_str(&text).expect("valid json");
assert_eq!(parsed["total_commits"], 2);
assert_eq!(parsed["total_authors"], 2);
std::fs::remove_dir_all(&dir).ok();
}
#[test]
fn markdown_formatter_emits_report_header() {
let db = seed_db();
let cfg = baseline_config();
let data = Aggregator::build(&db, &cfg).expect("aggregate");
let dir = tmp_dir("md");
let path = md_fmt::write_markdown(&data, &dir).expect("write md");
let text = std::fs::read_to_string(&path).expect("read");
assert!(text.contains("# Git Activity Report"));
assert!(text.contains("Alice"));
std::fs::remove_dir_all(&dir).ok();
}
#[test]
fn pipeline_constructs_without_panic() {
let cfg = baseline_config();
let _pipeline = ReportPipeline::new(cfg);
}
#[test]
fn aggregator_computes_summary_and_dora_and_quality() {
let db = seed_db();
let cfg = baseline_config();
let data = Aggregator::build(&db, &cfg).expect("aggregate");
let summary = data.summary.as_ref().expect("summary present");
assert_eq!(summary.total_commits, 2);
assert_eq!(summary.total_developers, 2);
assert!(summary.total_weeks >= 1);
assert!((summary.classification_coverage_pct - 50.0).abs() < 1e-6);
let dora = data.dora.as_ref().expect("dora present");
let lvl = dora.performance_level.as_str();
assert!(
matches!(lvl, "elite" | "high" | "medium" | "low"),
"unexpected performance_level: {lvl}"
);
let quality = data.quality.as_ref().expect("quality present");
assert!(quality.quality_score >= 0.0 && quality.quality_score <= 1.0);
let velocity = data.velocity.as_ref().expect("velocity present");
assert_eq!(velocity.pr_count, 0);
}
#[test]
fn aggregator_produces_developer_activity_with_score_ordering() {
let db = Database::open_in_memory().expect("open db");
let conn = db.connection();
for i in 0..5 {
conn.execute(
"INSERT INTO commits (sha, author_name, author_email, timestamp, message, repository, \
files_changed, insertions, deletions, is_merge) \
VALUES (?1, 'Alice', 'alice@example.com', '2024-01-15T10:00:00+00:00', \
'feat: change', 'repo-a', 1, 10, 1, 0)",
[format!("a{i}")],
)
.expect("seed alice");
}
conn.execute(
"INSERT INTO commits (sha, author_name, author_email, timestamp, message, repository, \
files_changed, insertions, deletions, is_merge) \
VALUES ('b1', 'Bob', 'bob@example.com', '2024-01-22T10:00:00+00:00', \
'feat: y', 'repo-a', 1, 1, 1, 0)",
[],
)
.expect("seed bob");
let data = Aggregator::build(&db, &baseline_config()).expect("aggregate");
let alice = data
.developer_activity
.iter()
.find(|d| d.developer_id == "alice@example.com")
.expect("alice present");
let bob = data
.developer_activity
.iter()
.find(|d| d.developer_id == "bob@example.com")
.expect("bob present");
assert_eq!(alice.total_commits, 5);
assert_eq!(bob.total_commits, 1);
assert!(
alice.activity_score > bob.activity_score,
"alice ({:.4}) should outrank bob ({:.4})",
alice.activity_score,
bob.activity_score
);
}
#[test]
fn csv_formatter_writes_new_report_files() {
let db = seed_db();
let cfg = baseline_config();
let data = Aggregator::build(&db, &cfg).expect("aggregate");
let dir = tmp_dir("csv-new");
let summary = csv_fmt::write_summary_csv(&data, &dir).expect("write summary");
let weekly_metrics =
csv_fmt::write_weekly_metrics_csv(&data, &dir).expect("write weekly metrics");
let dev_activity =
csv_fmt::write_developer_activity_csv(&data, &dir).expect("write dev activity");
let untracked = csv_fmt::write_untracked_csv(&data, &dir).expect("write untracked");
let weekly_cat = csv_fmt::write_weekly_categorization_csv(&data, &dir)
.expect("write weekly categorization");
let weekly_vel =
csv_fmt::write_weekly_velocity_csv(&data, &dir).expect("write weekly velocity");
let dora_csv = csv_fmt::write_weekly_dora_csv(&data, &dir).expect("write dora csv");
for p in [
&summary,
&weekly_metrics,
&dev_activity,
&untracked,
&weekly_cat,
&weekly_vel,
&dora_csv,
] {
assert!(p.exists(), "{} should exist", p.display());
}
let summary_text = std::fs::read_to_string(&summary).expect("read summary");
assert!(summary_text.starts_with("date_range,total_commits"));
let dev_text = std::fs::read_to_string(&dev_activity).expect("read dev activity");
assert!(dev_text.contains("activity_score"));
assert!(dev_text.contains("Alice"));
std::fs::remove_dir_all(&dir).ok();
}
#[test]
fn json_formatter_writes_velocity_quality_dora() {
let db = seed_db();
let cfg = baseline_config();
let data = Aggregator::build(&db, &cfg).expect("aggregate");
let dir = tmp_dir("json-new");
let velocity = json_fmt::write_velocity_json(&data, &dir).expect("write velocity");
let quality = json_fmt::write_quality_json(&data, &dir).expect("write quality");
let dora = json_fmt::write_dora_json(&data, &dir).expect("write dora");
let velocity_v: serde_json::Value =
serde_json::from_str(&std::fs::read_to_string(&velocity).expect("read"))
.expect("velocity json");
assert!(velocity_v.is_object());
assert!(velocity_v["pr_count"].is_number());
let quality_v: serde_json::Value =
serde_json::from_str(&std::fs::read_to_string(&quality).expect("read"))
.expect("quality json");
assert!(quality_v["quality_score"].as_f64().unwrap() >= 0.0);
let dora_v: serde_json::Value =
serde_json::from_str(&std::fs::read_to_string(&dora).expect("read"))
.expect("dora json");
assert!(dora_v["performance_level"].is_string());
std::fs::remove_dir_all(&dir).ok();
}
#[test]
fn pipeline_runs_all_formats_when_unspecified() {
let db = seed_db();
let dir = tmp_dir("pipeline");
let cfg = Config {
repositories: vec![RepositoryConfig {
path: PathBuf::from("/tmp/repo-a"),
name: Some("repo-a".into()),
..Default::default()
}],
output: Some(OutputConfig {
directory: Some(dir.clone()),
..Default::default()
}),
..Default::default()
};
let pipeline = ReportPipeline::new(cfg);
let stats = pipeline.run(&db).expect("run");
assert_eq!(stats.total_commits, 2);
assert_eq!(stats.total_authors, 2);
assert_eq!(stats.files_written.len(), 14);
for f in &stats.files_written {
assert!(f.exists(), "{} should exist", f.display());
}
std::fs::remove_dir_all(&dir).ok();
}
}