Move inline #[cfg(test)] mod tests { ... } blocks from 22 source files
into dedicated _tests.rs companion files, wired via:
#[cfg(test)]
#[path = "module_tests.rs"]
mod tests;
This keeps implementation-focused source files leaner and more scannable
while preserving full access to private items through `use super::*;`.
Modules extracted:
core: db, note_parser, payloads, project, references, sync_run,
timeline_collect, timeline_expand, timeline_seed
cli: list (55 tests), who (75 tests)
documents: extractor (43 tests), regenerator
embedding: change_detector, chunking
gitlab: graphql (wiremock async tests), transformers/issue
ingestion: dirty_tracker, discussions, issues, mr_diffs
Also adds conflicts_with("explain_score") to the --detail flag in the
who command to prevent mutually exclusive flags from being combined.
All 629 unit tests pass. No behavior changes.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
157 lines
4.4 KiB
Rust
157 lines
4.4 KiB
Rust
use super::*;
|
|
|
|
fn setup_db() -> Connection {
|
|
let conn = Connection::open_in_memory().unwrap();
|
|
conn.execute_batch(
|
|
"
|
|
CREATE TABLE projects (
|
|
id INTEGER PRIMARY KEY,
|
|
gitlab_project_id INTEGER UNIQUE NOT NULL,
|
|
path_with_namespace TEXT NOT NULL,
|
|
default_branch TEXT,
|
|
web_url TEXT,
|
|
created_at INTEGER,
|
|
updated_at INTEGER,
|
|
raw_payload_id INTEGER
|
|
);
|
|
CREATE INDEX idx_projects_path ON projects(path_with_namespace);
|
|
",
|
|
)
|
|
.unwrap();
|
|
conn
|
|
}
|
|
|
|
fn insert_project(conn: &Connection, id: i64, path: &str) {
|
|
conn.execute(
|
|
"INSERT INTO projects (id, gitlab_project_id, path_with_namespace) VALUES (?1, ?2, ?3)",
|
|
rusqlite::params![id, id * 100, path],
|
|
)
|
|
.unwrap();
|
|
}
|
|
|
|
#[test]
|
|
fn test_exact_match() {
|
|
let conn = setup_db();
|
|
insert_project(&conn, 1, "backend/auth-service");
|
|
let id = resolve_project(&conn, "backend/auth-service").unwrap();
|
|
assert_eq!(id, 1);
|
|
}
|
|
|
|
#[test]
|
|
fn test_case_insensitive() {
|
|
let conn = setup_db();
|
|
insert_project(&conn, 1, "backend/auth-service");
|
|
let id = resolve_project(&conn, "Backend/Auth-Service").unwrap();
|
|
assert_eq!(id, 1);
|
|
}
|
|
|
|
#[test]
|
|
fn test_suffix_unambiguous() {
|
|
let conn = setup_db();
|
|
insert_project(&conn, 1, "backend/auth-service");
|
|
insert_project(&conn, 2, "frontend/web-ui");
|
|
let id = resolve_project(&conn, "auth-service").unwrap();
|
|
assert_eq!(id, 1);
|
|
}
|
|
|
|
#[test]
|
|
fn test_suffix_ambiguous() {
|
|
let conn = setup_db();
|
|
insert_project(&conn, 1, "backend/auth-service");
|
|
insert_project(&conn, 2, "frontend/auth-service");
|
|
let err = resolve_project(&conn, "auth-service").unwrap_err();
|
|
let msg = err.to_string();
|
|
assert!(
|
|
msg.contains("ambiguous"),
|
|
"Expected ambiguous error, got: {}",
|
|
msg
|
|
);
|
|
assert!(msg.contains("backend/auth-service"));
|
|
assert!(msg.contains("frontend/auth-service"));
|
|
}
|
|
|
|
#[test]
|
|
fn test_substring_unambiguous() {
|
|
let conn = setup_db();
|
|
insert_project(&conn, 1, "vs/python-code");
|
|
insert_project(&conn, 2, "vs/typescript-code");
|
|
let id = resolve_project(&conn, "typescript").unwrap();
|
|
assert_eq!(id, 2);
|
|
}
|
|
|
|
#[test]
|
|
fn test_substring_case_insensitive() {
|
|
let conn = setup_db();
|
|
insert_project(&conn, 1, "vs/python-code");
|
|
insert_project(&conn, 2, "vs/typescript-code");
|
|
let id = resolve_project(&conn, "TypeScript").unwrap();
|
|
assert_eq!(id, 2);
|
|
}
|
|
|
|
#[test]
|
|
fn test_substring_ambiguous() {
|
|
let conn = setup_db();
|
|
insert_project(&conn, 1, "vs/python-code");
|
|
insert_project(&conn, 2, "vs/typescript-code");
|
|
let err = resolve_project(&conn, "code").unwrap_err();
|
|
let msg = err.to_string();
|
|
assert!(
|
|
msg.contains("ambiguous"),
|
|
"Expected ambiguous error, got: {}",
|
|
msg
|
|
);
|
|
assert!(msg.contains("vs/python-code"));
|
|
assert!(msg.contains("vs/typescript-code"));
|
|
}
|
|
|
|
#[test]
|
|
fn test_suffix_preferred_over_substring() {
|
|
let conn = setup_db();
|
|
insert_project(&conn, 1, "backend/auth-service");
|
|
insert_project(&conn, 2, "backend/auth-service-v2");
|
|
let id = resolve_project(&conn, "auth-service").unwrap();
|
|
assert_eq!(id, 1);
|
|
}
|
|
|
|
#[test]
|
|
fn test_no_match() {
|
|
let conn = setup_db();
|
|
insert_project(&conn, 1, "backend/auth-service");
|
|
let err = resolve_project(&conn, "nonexistent").unwrap_err();
|
|
let msg = err.to_string();
|
|
assert!(
|
|
msg.contains("not found"),
|
|
"Expected not found error, got: {}",
|
|
msg
|
|
);
|
|
assert!(msg.contains("backend/auth-service"));
|
|
}
|
|
|
|
#[test]
|
|
fn test_empty_projects() {
|
|
let conn = setup_db();
|
|
let err = resolve_project(&conn, "anything").unwrap_err();
|
|
let msg = err.to_string();
|
|
assert!(msg.contains("No projects have been synced"));
|
|
}
|
|
|
|
#[test]
|
|
fn test_underscore_not_wildcard() {
|
|
let conn = setup_db();
|
|
insert_project(&conn, 1, "backend/my_project");
|
|
insert_project(&conn, 2, "backend/my-project");
|
|
// `_` in user input must not match `-` (LIKE wildcard behavior)
|
|
let id = resolve_project(&conn, "my_project").unwrap();
|
|
assert_eq!(id, 1);
|
|
}
|
|
|
|
#[test]
|
|
fn test_percent_not_wildcard() {
|
|
let conn = setup_db();
|
|
insert_project(&conn, 1, "backend/a%b");
|
|
insert_project(&conn, 2, "backend/axyzb");
|
|
// `%` in user input must not match arbitrary strings
|
|
let id = resolve_project(&conn, "a%b").unwrap();
|
|
assert_eq!(id, 1);
|
|
}
|