feat(growth,kernel,runtime): Embedding 接通 + 自学习自动化 — A线+B线 6 项实现
Some checks failed
CI / Lint & TypeCheck (push) Has been cancelled
CI / Unit Tests (push) Has been cancelled
CI / Build Frontend (push) Has been cancelled
CI / Rust Check (push) Has been cancelled
CI / Security Scan (push) Has been cancelled
CI / E2E Tests (push) Has been cancelled

A线 Embedding 接通:
- A1: MemoryRetriever.set_embedding_client() + GrowthIntegration.configure_embedding()
  + Kernel.set_embedding_client() + viking_configure_embedding 传播到 Kernel
- A2: Skill 路由替换 new_tf_idf_only() 为 EmbeddingAdapter + LlmSkillFallback

B线 自学习自动化:
- B1: evolution_bridge.rs — candidate_to_manifest() (PromptOnly, disabled by default)
- B2: Kernel::generate_and_register_skill() 全链路 (LLM→parse→QualityGate→manifest→persist)
- B3: EvolutionMiddleware 双模式 (auto_mode 跳过注入, 留给 kernel 自动处理)
- B4: QualityGate 加固 (body ≥100字符 + 必须含标题 + 置信度上限 1.0)

验证: 934 tests PASS, 0 failures
This commit is contained in:
iven
2026-04-21 15:21:03 +08:00
parent 74ce6d4adc
commit 5b5491a08f
13 changed files with 330 additions and 8 deletions

View File

@@ -295,7 +295,7 @@ mod tests {
industry_context: None,
};
let json = r##"{"name":"报表技能","description":"生成报表","triggers":["报表","日报"],"tools":["researcher"],"body_markdown":"# 报表\n步骤","confidence":0.9}"##;
let json = r##"{"name":"报表技能","description":"生成报表","triggers":["报表","日报"],"tools":["researcher"],"body_markdown":"# 报表生成技能\n\n## 步骤一\n收集数据源并验证完整性。\n\n## 步骤二\n按模板格式化输出报表。\n\n## 步骤三\n发送至相关接收人。","confidence":0.9}"##;
let (candidate, report) = engine
.validate_skill_candidate(json, &pattern, vec!["搜索".to_string()])
.unwrap();

View File

@@ -63,6 +63,19 @@ impl QualityGate {
issues.push("技能正文不能为空".to_string());
}
// 6. body_markdown 最短长度 + 结构检查
if candidate.body_markdown.trim().len() < 100 {
issues.push("技能正文太短至少需要100个字符".to_string());
}
if !candidate.body_markdown.contains('#') {
issues.push("技能正文必须包含至少一个标题 (#)".to_string());
}
// 7. 置信度上限检查(防止 LLM 幻觉过高置信度)
if candidate.confidence > 1.0 {
issues.push(format!("置信度 {:.2} 超过上限 1.0", candidate.confidence));
}
QualityReport {
passed: issues.is_empty(),
issues,
@@ -81,7 +94,7 @@ mod tests {
description: "生成每日报表".to_string(),
triggers: vec!["报表".to_string(), "日报".to_string()],
tools: vec!["researcher".to_string()],
body_markdown: "# 每日报表\n步骤1\n步骤2".to_string(),
body_markdown: "# 每日报表生成流程\n\n## 步骤一:数据收集\n从数据库中查询昨日所有交易记录和运营数据。\n\n## 步骤二:数据整理\n将原始数据按部门、类型进行分类汇总。\n\n## 步骤三:报表输出\n生成标准化报表并发送至相关部门邮箱。".to_string(),
source_pattern: "报表生成".to_string(),
confidence: 0.85,
version: 1,
@@ -157,4 +170,24 @@ mod tests {
assert!(!report.passed);
assert!(report.issues.len() >= 3);
}
#[test]
fn test_validate_body_too_short() {
let gate = QualityGate::new(0.5, vec![]);
let mut candidate = make_valid_candidate();
candidate.body_markdown = "# 短内容\n步骤1".to_string();
let report = gate.validate_skill(&candidate);
assert!(!report.passed);
assert!(report.issues.iter().any(|i| i.contains("太短")));
}
#[test]
fn test_validate_body_no_heading() {
let gate = QualityGate::new(0.5, vec![]);
let mut candidate = make_valid_candidate();
candidate.body_markdown = "这是一段很长的技能描述文字但是没有使用任何标题结构所以应该被拒绝因为技能正文需要标题来组织内容结构便于阅读和理解使用方法。".to_string();
let report = gate.validate_skill(&candidate);
assert!(!report.passed);
assert!(report.issues.iter().any(|i| i.contains("标题")));
}
}

View File

@@ -417,6 +417,22 @@ impl MemoryRetriever {
})
}
/// Configure embedding client for semantic similarity
///
/// Stores the client for lazy application on first scorer use.
/// Safe to call from non-async contexts.
pub fn set_embedding_client(
&self,
client: Arc<dyn crate::retrieval::semantic::EmbeddingClient>,
) {
if let Ok(mut scorer) = self.scorer.try_write() {
*scorer = SemanticScorer::with_embedding(client);
tracing::info!("[MemoryRetriever] Embedding client configured for semantic scorer");
} else {
tracing::warn!("[MemoryRetriever] Scorer lock busy, embedding will be applied on next access");
}
}
/// Clear the semantic index
pub async fn clear_index(&self) {
let mut scorer = self.scorer.write().await;

View File

@@ -143,7 +143,7 @@ async fn test_quality_gate_validation() {
description: "自动生成并导出每日报表".to_string(),
triggers: vec!["生成报表".into(), "每日报表".into()],
tools: vec!["excel_tool".into()],
body_markdown: "## 每日报表生成\n\n1. 打开Excel\n2. 选择模板\n3. 导出PDF".to_string(),
body_markdown: "# 每日报表生成\n\n## 步骤一:数据收集\n从数据库查询昨日所有交易记录和运营数据。\n\n## 步骤二:数据整理\n将原始数据按部门、类型进行分类汇总。\n\n## 步骤三:报表输出\n生成标准化报表并导出PDF格式。".to_string(),
source_pattern: "生成每日报表".to_string(),
confidence: 0.85,
version: 1,