添加 Plugin
当你需要通过 state key、phase hook、scheduled action 或 effect handler 扩展 agent 生命周期时,使用本页。
- 已在
Cargo.toml中添加awaken - 已了解
Phase与StateKey
- 定义一个状态键:
use awaken::{StateKey, KeyScope, MergeStrategy, StateError, JsonValue};use serde::{Serialize, Deserialize};
#[derive(Debug, Clone, Serialize, Deserialize)]pub struct AuditLog { pub entries: Vec<String>,}
pub struct AuditLogKey;
impl StateKey for AuditLogKey { type Value = AuditLog; const KEY: &'static str = "audit_log"; const MERGE: MergeStrategy = MergeStrategy::Exclusive;
type Update = AuditLog;
fn apply(value: &mut Self::Value, update: Self::Update) { *value = update; }
fn encode(value: &Self::Value) -> Result<JsonValue, StateError> { serde_json::to_value(value).map_err(|e| StateError::KeyEncode { key: Self::KEY.into(), message: e.to_string() }) }
fn decode(json: JsonValue) -> Result<Self::Value, StateError> { serde_json::from_value(json).map_err(|e| StateError::KeyDecode { key: Self::KEY.into(), message: e.to_string() }) }}- 实现一个 phase hook:
use async_trait::async_trait;use awaken::{PhaseHook, PhaseContext, StateCommand, StateError};
pub struct AuditHook;
#[async_trait]impl PhaseHook for AuditHook { async fn run(&self, ctx: &PhaseContext) -> Result<StateCommand, StateError> { let mut log = ctx.state::<AuditLogKey>().cloned().unwrap_or(AuditLog { entries: Vec::new(), }); log.entries.push(format!("Phase executed at {:?}", ctx.phase)); let mut cmd = StateCommand::new(); cmd.update::<AuditLogKey>(log); Ok(cmd) }}- 实现
Plugintrait:
use awaken::{Plugin, PluginDescriptor, PluginRegistrar, Phase, StateError, StateKeyOptions, KeyScope};
pub struct AuditPlugin;
impl Plugin for AuditPlugin { fn descriptor(&self) -> PluginDescriptor { PluginDescriptor { name: "audit" } }
fn register(&self, registrar: &mut PluginRegistrar) -> Result<(), StateError> { registrar.register_key::<AuditLogKey>(StateKeyOptions { scope: KeyScope::Run, ..Default::default() })?;
registrar.register_phase_hook( "audit", Phase::AfterInference, AuditHook, )?;
Ok(()) }}- 在 runtime 上注册插件,并在 agent 上激活它:
use std::sync::Arc;use awaken::engine::GenaiExecutor;use awaken::registry_spec::ModelSpec;use awaken::{AgentSpec, AgentRuntimeBuilder};
let mut spec = AgentSpec::new("assistant") .with_model_id("claude-sonnet") .with_system_prompt("You are a helpful assistant.") .with_hook_filter("audit");spec.plugin_ids.push("audit".into());
let runtime = AgentRuntimeBuilder::new() .with_plugin("audit", Arc::new(AuditPlugin)) .with_agent_spec(spec) .with_provider("anthropic", Arc::new(GenaiExecutor::new())) .with_model(ModelSpec::new("claude-sonnet", "anthropic", "claude-sonnet-4-20250514")) .build()?;plugin_ids 负责加载插件;with_hook_filter 只过滤已经加载的插件所提供的
hook、tool 和 request transform。
运行 agent 后查看状态快照,确认 audit_log 中出现了 hook 写入的条目。
| 错误 | 原因 | 修复 |
|---|---|---|
StateError::KeyAlreadyRegistered | 多个插件注册了同一个 key | 保证每个 StateKey::KEY 全局唯一 |
StateError::UnknownKey | 读取了未注册的状态键 | 确保注册该 key 的插件已激活 |
| hook 没有执行 | 插件未加载或 hook 被过滤 | 把插件 ID 加到 plugin_ids;使用 hook filter 时也加入 with_hook_filter |
crates/awaken-ext-observability/
crates/awaken-runtime/src/plugins/lifecycle.rscrates/awaken-runtime/src/plugins/registry.rscrates/awaken-runtime/src/hooks/phase_hook.rs