pub struct Message {
pub id: String,
pub topic: String,
pub payload: Vec<u8>,
pub sender: String,
pub metadata: HashMap<String, String>,
pub created_at: i64,
}Expand description
A message for inter-agent communication.
Messages carry binary payloads between agents in a mesh.
§Example
use ceylon_runtime::Message;
// Create a message
let msg = Message::new("greeting", b"Hello!".to_vec(), "sender-agent");
// Access message properties
println!("Topic: {}", msg.topic);
println!("From: {}", msg.sender);Fields§
§id: StringUnique message identifier (auto-generated UUID).
topic: StringMessage topic/type for routing or categorization.
payload: Vec<u8>Binary payload data.
sender: StringName of the sending agent.
metadata: HashMap<String, String>Arbitrary metadata key-value pairs.
created_at: i64Creation timestamp in microseconds since epoch.
Implementations§
Source§impl Message
impl Message
Sourcepub fn new(
topic: impl Into<String>,
payload: Vec<u8>,
sender: impl Into<String>,
) -> Self
pub fn new( topic: impl Into<String>, payload: Vec<u8>, sender: impl Into<String>, ) -> Self
Create a new message.
§Arguments
topic- Message topic for routingpayload- Binary payloadsender- Name of the sending agent
Examples found in repository?
examples/mesh_architecture.rs (line 38)
27 async fn on_message(&mut self, msg: Message, _ctx: &mut AgentContext) -> Result<()> {
28 let content = String::from_utf8_lossy(&msg.payload);
29 println!("[Worker] Received task from {}: {}", msg.sender, content);
30
31 // Simulate processing
32 tokio::time::sleep(Duration::from_millis(500)).await;
33
34 // Send response
35 let response = format!("Processed: {}", content);
36 println!("[Worker] Sending response: {}", response);
37
38 let reply_msg = Message::new("result", response.into_bytes(), self.name());
39 self.mesh.send(reply_msg, &msg.sender).await?;
40
41 Ok(())
42 }
43}
44
45// --- Manager Agent ---
46struct ManagerAgent {
47 mesh: Arc<LocalMesh>,
48 completion_notify: Arc<Notify>,
49}
50
51#[async_trait]
52impl Agent for ManagerAgent {
53 fn name(&self) -> String {
54 "manager".to_string()
55 }
56
57 async fn on_start(&mut self, _ctx: &mut AgentContext) -> Result<()> {
58 println!("[Manager] Started. Sending task to worker...");
59
60 // Send a task to the worker
61 let task_msg = Message::new("task", b"Analyze data".to_vec(), self.name());
62 self.mesh.send(task_msg, "worker").await?;
63
64 Ok(())
65 }More examples
examples/llm_multi_agent_mesh.rs (line 91)
71 async fn on_message(&mut self, msg: Message, ctx: &mut AgentContext) -> Result<()> {
72 let topic = String::from_utf8_lossy(&msg.payload);
73 println!("\n[Researcher] Received research request: {}", topic);
74
75 // Use LLM to research the topic
76 let research_prompt = format!(
77 "Research the following topic and provide key information: {}",
78 topic
79 );
80
81 match self
82 .llm_agent
83 .send_message_and_get_response(&research_prompt, ctx)
84 .await
85 {
86 Ok(research_result) => {
87 println!("[Researcher] Research complete. Sending to summarizer...");
88
89 // Send research results to the summarizer
90 let research_msg =
91 Message::new("research_result", research_result.into_bytes(), self.name());
92 self.mesh.send(research_msg, "summarizer").await?;
93 }
94 Err(e) => {
95 eprintln!("[Researcher] Error during research: {}", e);
96 // Send error message to summarizer
97 let error_msg = Message::new(
98 "error",
99 format!("Research failed: {}", e).into_bytes(),
100 self.name(),
101 );
102 self.mesh.send(error_msg, "summarizer").await?;
103 }
104 }
105
106 Ok(())
107 }
108}
109
110// --- Summarizer Agent Wrapper ---
111// Wraps an LlmAgent to summarize research findings
112struct SummarizerAgent {
113 llm_agent: LlmAgent,
114 completion_notify: Arc<Notify>,
115}
116
117impl SummarizerAgent {
118 fn new(completion_notify: Arc<Notify>) -> Result<Self> {
119 let llm_agent = LlmAgent::builder("summarizer", "ollama::gemma3:latest")
120 .with_system_prompt(
121 "You are a summarization expert. When given research content, \
122 create a clear and concise summary with bullet points highlighting \
123 the most important facts. Keep the summary to 5-7 bullet points.",
124 )
125 .with_temperature(0.5)
126 .with_max_tokens(512)
127 .build()?;
128
129 Ok(Self {
130 llm_agent,
131 completion_notify,
132 })
133 }
134}
135
136#[async_trait]
137impl Agent for SummarizerAgent {
138 fn name(&self) -> String {
139 "summarizer".to_string()
140 }
141
142 async fn on_start(&mut self, _ctx: &mut AgentContext) -> Result<()> {
143 println!("[Summarizer] Started and ready to summarize.");
144 Ok(())
145 }
146
147 async fn on_message(&mut self, msg: Message, ctx: &mut AgentContext) -> Result<()> {
148 let content = String::from_utf8_lossy(&msg.payload);
149 println!("\n[Summarizer] Received content from {}", msg.sender);
150
151 if msg.topic == "error" {
152 println!("[Summarizer] Received error: {}", content);
153 self.completion_notify.notify_one();
154 return Ok(());
155 }
156
157 // Use LLM to summarize the research
158 let summary_prompt = format!(
159 "Please summarize the following research content into clear bullet points:\n\n{}",
160 content
161 );
162
163 match self
164 .llm_agent
165 .send_message_and_get_response(&summary_prompt, ctx)
166 .await
167 {
168 Ok(summary) => {
169 println!("\n========================================");
170 println!(" FINAL SUMMARY");
171 println!("========================================\n");
172 println!("{}", summary);
173 println!("\n========================================\n");
174 }
175 Err(e) => {
176 eprintln!("[Summarizer] Error during summarization: {}", e);
177 }
178 }
179
180 // Signal completion
181 self.completion_notify.notify_one();
182 Ok(())
183 }
184}
185
186// --- Coordinator Agent ---
187// Initiates the workflow by sending a research topic
188struct CoordinatorAgent {
189 mesh: Arc<LocalMesh>,
190 topic: String,
191}
192
193#[async_trait]
194impl Agent for CoordinatorAgent {
195 fn name(&self) -> String {
196 "coordinator".to_string()
197 }
198
199 async fn on_start(&mut self, _ctx: &mut AgentContext) -> Result<()> {
200 println!("[Coordinator] Starting multi-agent workflow...");
201 println!("[Coordinator] Topic: {}\n", self.topic);
202
203 // Small delay to ensure all agents are ready
204 tokio::time::sleep(Duration::from_millis(100)).await;
205
206 // Send topic to researcher
207 let task_msg = Message::new(
208 "research_topic",
209 self.topic.as_bytes().to_vec(),
210 self.name(),
211 );
212 self.mesh.send(task_msg, "researcher").await?;
213
214 Ok(())
215 }Sourcepub fn correlation_id(&self) -> Option<String>
pub fn correlation_id(&self) -> Option<String>
Get the correlation ID for request/response tracking.
Sourcepub fn set_correlation_id(&mut self, id: &str)
pub fn set_correlation_id(&mut self, id: &str)
Set a correlation ID for request/response tracking.
Trait Implementations§
Source§impl<'de> Deserialize<'de> for Message
impl<'de> Deserialize<'de> for Message
Source§fn deserialize<__D>(__deserializer: __D) -> Result<Self, __D::Error>where
__D: Deserializer<'de>,
fn deserialize<__D>(__deserializer: __D) -> Result<Self, __D::Error>where
__D: Deserializer<'de>,
Deserialize this value from the given Serde deserializer. Read more
Auto Trait Implementations§
impl Freeze for Message
impl RefUnwindSafe for Message
impl Send for Message
impl Sync for Message
impl Unpin for Message
impl UnwindSafe for Message
Blanket Implementations§
Source§impl<T> BorrowMut<T> for Twhere
T: ?Sized,
impl<T> BorrowMut<T> for Twhere
T: ?Sized,
Source§fn borrow_mut(&mut self) -> &mut T
fn borrow_mut(&mut self) -> &mut T
Mutably borrows from an owned value. Read more
Source§impl<T> CloneToUninit for Twhere
T: Clone,
impl<T> CloneToUninit for Twhere
T: Clone,
Source§impl<T> Instrument for T
impl<T> Instrument for T
Source§fn instrument(self, span: Span) -> Instrumented<Self>
fn instrument(self, span: Span) -> Instrumented<Self>
Source§fn in_current_span(self) -> Instrumented<Self>
fn in_current_span(self) -> Instrumented<Self>
Source§impl<T> IntoEither for T
impl<T> IntoEither for T
Source§fn into_either(self, into_left: bool) -> Either<Self, Self>
fn into_either(self, into_left: bool) -> Either<Self, Self>
Converts
self into a Left variant of Either<Self, Self>
if into_left is true.
Converts self into a Right variant of Either<Self, Self>
otherwise. Read moreSource§fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
Converts
self into a Left variant of Either<Self, Self>
if into_left(&self) returns true.
Converts self into a Right variant of Either<Self, Self>
otherwise. Read more