1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
//! [`DescriptorPlugin`] — blanket [`StatefulPlugin`] impl for descriptor-based plugins.
//!
//! Implement this trait instead of [`ElicitPlugin`] or [`StatefulPlugin`] directly when
//! your plugin exposes a static slice of [`ToolDescriptor`]s. The blanket impl routes
//! through [`StatefulPlugin`] (with [`NoContext`]) and provides `list_tools` and
//! `call_tool` for free.
use Tool;
use ;
use crateNoContext;
/// A plugin that exposes a static slice of [`ToolDescriptor`]s.
///
/// Implement this instead of [`ElicitPlugin`] to eliminate manual
/// `list_tools` and `call_tool` dispatch. A blanket impl provides
/// both for free.
///
/// # Example
///
/// ```rust,no_run
/// use elicitation::plugin::{DescriptorPlugin, ToolDescriptor, make_descriptor};
/// use rmcp::model::{CallToolResult, Content};
/// use schemars::JsonSchema;
/// use serde::Deserialize;
///
/// #[derive(Deserialize, JsonSchema)]
/// struct PingParams { message: String }
///
/// pub struct MyPlugin;
///
/// impl DescriptorPlugin for MyPlugin {
/// fn name(&self) -> &'static str { "my_plugin" }
/// fn descriptors(&self) -> &'static [ToolDescriptor] { &TOOLS }
/// }
///
/// static TOOLS: std::sync::LazyLock<Vec<ToolDescriptor>> = std::sync::LazyLock::new(|| {
/// vec![make_descriptor::<PingParams, _>(
/// "ping", "Echo a message",
/// |p| Box::pin(async move {
/// Ok(CallToolResult::success(vec![Content::text(p.message)]))
/// }),
/// )]
/// });
/// ```