Files
kebab/crates/kebab-search/tests/hybrid.rs
altair823 911fb49550 refactor(rename): kb crates → kebab — Cargo packages, folders, Rust modules
프로젝트 이름 `kb` → `kebab` rename 의 첫 단계.

- workspace `Cargo.toml`: members `crates/kb-*` → `crates/kebab-*`,
  repository URL `altair823/kb` → `altair823/kebab`.
- 18 crate 폴더 rename via `git mv` (history 보존).
- 각 crate `Cargo.toml`: `name = "kb-*"` → `"kebab-*"`, path deps
  `../kb-*` → `../kebab-*`.
- 모든 `.rs`: `kb_<id>` snake-case 모듈 path 18 개 (`kb_core`,
  `kb_config`, `kb_app`, `kb_cli`, `kb_eval`, `kb_search`, `kb_chunk`,
  `kb_normalize`, `kb_source_fs`, `kb_parse_md`, `kb_parse_types`,
  `kb_store_sqlite`, `kb_store_vector`, `kb_embed`, `kb_embed_local`,
  `kb_llm`, `kb_llm_local`, `kb_rag`) → `kebab_<id>` 일괄 sed (단어
  경계 \\b 사용해 영어 문장 안의 "kb" 약어 미오염).

CLI binary 이름 (`[[bin]] name = "kb"`), 환경변수 `KB_*`, XDG paths,
tracing target, 그리고 docs sweep 은 다음 commit 에서.

## 검증

- `cargo check --workspace` clean — 모든 crate 빌드 통과 후 commit.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-02 03:28:08 +00:00

214 lines
8.0 KiB
Rust

//! Hybrid integration tests — touch a real `LanceVectorStore` +
//! `SqliteStore` + `MockEmbedder`. These tests are `#[ignore]`-d and
//! AVX-gated; see `tests/common/mod.rs` for the policy rationale.
//!
//! Mock-retriever unit tests live alongside the implementation in
//! `crates/kb-search/src/hybrid.rs` (no Lance, no AVX needed) — the
//! tests here exercise the full plumbing with the real Lance store.
mod common;
use std::path::PathBuf;
use std::sync::Arc;
use common::{
HybridEnv, id32, require_avx_or_panic, TEST_LEX_INDEX_VERSION, TEST_VEC_INDEX_VERSION,
};
use kebab_core::{
Retriever, SearchFilters, SearchHit, SearchMode, SearchQuery,
};
use kebab_search::{FusionPolicy, HybridRetriever};
use serde_json::json;
fn build_hybrid(env: &HybridEnv) -> HybridRetriever {
let lex: Arc<dyn Retriever> = Arc::new(env.lexical_retriever());
let vec: Arc<dyn Retriever> = Arc::new(env.vector_retriever());
HybridRetriever::with_policy(lex, vec, FusionPolicy::Rrf { k_rrf: 60 }, 5)
}
/// Seed a tiny corpus that lets us prove hybrid recall ≥ each side
/// independently. Two chunks are lexical-only matches ("rust cargo");
/// two chunks are vector-only matches (their text doesn't contain
/// the query token but their embedding still scores nearby because
/// MockEmbedder's hash distributes over all chunks).
fn seed_disjoint_corpus(env: &HybridEnv) -> Vec<String> {
// The lexical side will only match chunks that contain the query
// tokens. The vector side will rank ALL chunks by embedding
// similarity to the query — even ones whose text doesn't share
// a token with the query.
let chunks = [
// (chunk_id, doc_id, path, text, headings)
(id32("c1"), id32("d1"), "notes/rust1.md", "rust cargo macros", &["A"][..]),
(id32("c2"), id32("d2"), "notes/rust2.md", "rust traits and lifetimes", &["B"][..]),
(id32("c3"), id32("d3"), "notes/python.md", "python dataclasses tutorial", &["C"][..]),
(id32("c4"), id32("d4"), "notes/go.md", "go interfaces and channels", &["D"][..]),
];
let mut ids = Vec::new();
for (cid, did, path, text, headings) in &chunks {
env.seed_chunk(cid, did, path, text, headings, &[]);
env.embed_and_upsert(cid, did, text, headings);
ids.push(cid.clone());
}
ids
}
#[test]
#[ignore = "requires AVX-capable hardware (LanceDB)"]
fn hybrid_recall_disjoint_returns_union() {
require_avx_or_panic();
let env = HybridEnv::new();
let _ids = seed_disjoint_corpus(&env);
let h = build_hybrid(&env);
let q = SearchQuery {
text: "rust".to_string(),
mode: SearchMode::Hybrid,
k: 4,
filters: SearchFilters::default(),
};
let hits = h.search(&q).unwrap();
// The vector side will return up to 4 candidates regardless of
// text overlap; the lexical side will return only the rust* ones.
// Together the union must cover at least the lexical hits AND
// include at least one non-lexical chunk if vector found one.
assert!(!hits.is_empty(), "hybrid must return at least one hit");
// Every hit's RetrievalDetail.method must be Hybrid.
for h in &hits {
assert_eq!(h.retrieval.method, SearchMode::Hybrid);
// At least one of lex/vec_score must be Some.
assert!(
h.retrieval.lexical_score.is_some() || h.retrieval.vector_score.is_some(),
"hybrid hit must carry at least one mode's score"
);
}
// index_version composite token.
let iv = h.index_version();
assert!(iv.0.starts_with("hybrid:"));
assert!(iv.0.contains(TEST_LEX_INDEX_VERSION));
assert!(iv.0.contains(TEST_VEC_INDEX_VERSION));
// Lexical-only chunks (c1, c2) MUST appear: they're the only ones
// matching the FTS5 query, and the vector side over-fetches enough
// to include them too.
let ids: Vec<&str> = hits.iter().map(|h| h.chunk_id.0.as_str()).collect();
assert!(ids.contains(&id32("c1").as_str()));
assert!(ids.contains(&id32("c2").as_str()));
}
#[test]
#[ignore = "requires AVX-capable hardware (LanceDB)"]
fn hybrid_determinism_same_query_twice() {
require_avx_or_panic();
let env = HybridEnv::new();
let _ = seed_disjoint_corpus(&env);
let h = build_hybrid(&env);
let q = SearchQuery {
text: "rust".to_string(),
mode: SearchMode::Hybrid,
k: 4,
filters: SearchFilters::default(),
};
let a = h.search(&q).unwrap();
let b = h.search(&q).unwrap();
assert_eq!(a, b, "identical query must yield byte-identical Vec<SearchHit>");
}
#[test]
#[ignore = "requires AVX-capable hardware (LanceDB)"]
fn hybrid_snapshot_run_1() {
require_avx_or_panic();
let env = HybridEnv::new();
let _ = seed_disjoint_corpus(&env);
let h = build_hybrid(&env);
let q = SearchQuery {
text: "rust".to_string(),
mode: SearchMode::Hybrid,
k: 4,
filters: SearchFilters::default(),
};
let hits = h.search(&q).unwrap();
// Snapshot pins the structural shape:
// - chunk_id ordering
// - which side contributed (lexical_rank / vector_rank
// populated as Some/None)
// - that fusion_score is non-increasing
// - method = Hybrid for every hit
let actual = json!(
hits.iter().map(|h: &SearchHit| json!({
"chunk_id": h.chunk_id.0,
"rank": h.rank,
"method": h.retrieval.method,
"lexical_rank": h.retrieval.lexical_rank,
"vector_rank": h.retrieval.vector_rank,
"lex_some": h.retrieval.lexical_score.is_some(),
"vec_some": h.retrieval.vector_score.is_some(),
"fusion_score_positive": h.retrieval.fusion_score > 0.0,
})).collect::<Vec<_>>()
);
let fixture = PathBuf::from(env!("CARGO_MANIFEST_DIR"))
.join("tests")
.join("fixtures")
.join("search")
.join("hybrid")
.join("run-1.json");
if std::env::var_os("KB_UPDATE_SNAPSHOTS").is_some() {
std::fs::create_dir_all(fixture.parent().unwrap()).unwrap();
std::fs::write(&fixture, serde_json::to_string_pretty(&actual).unwrap()).unwrap();
eprintln!("[snapshot] regenerated {}", fixture.display());
// Fail loudly so that accidentally setting KB_UPDATE_SNAPSHOTS
// in CI surfaces as a test failure rather than a silent
// overwrite + green run. Same fail-loud-instead-of-silent-pass
// philosophy as P3-2's `SNAPSHOT_HASH_BASELINE = 0` and P3-3's
// placeholder fixture guards.
panic!(
"[snapshot] regenerated {}, re-run without KB_UPDATE_SNAPSHOTS to verify pin",
fixture.display()
);
}
let expected: serde_json::Value =
serde_json::from_str(&std::fs::read_to_string(&fixture).unwrap_or_else(|_| {
panic!(
"missing snapshot fixture at {}; run with \
KB_UPDATE_SNAPSHOTS=1 to create",
fixture.display()
)
}))
.unwrap();
// Refuse to silently "pass" against the committed placeholder. The
// placeholder JSON carries a `_comment` field with regeneration
// instructions; production fixtures (a captured list) do not.
if expected.get("_comment").is_some() {
panic!(
"snapshot fixture is a placeholder — regenerate on AVX hardware then commit. \
Path: {}. To regenerate: \
`KB_UPDATE_SNAPSHOTS=1 cargo test -p kb-search -- --ignored hybrid_snapshot`.",
fixture.display()
);
}
assert_eq!(
actual, expected,
"hybrid snapshot drift; rerun with KB_UPDATE_SNAPSHOTS=1 to regenerate"
);
// Independent guard: fusion scores must be non-increasing across
// the result list (rrf is rank-biased, so this is the
// semantically-correct ordering invariant).
for w in hits.windows(2) {
assert!(
w[0].retrieval.fusion_score >= w[1].retrieval.fusion_score,
"fusion scores not in descending order: {} then {}",
w[0].retrieval.fusion_score,
w[1].retrieval.fusion_score
);
}
}