Message

Struct Message 

Source
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: String

Unique message identifier (auto-generated UUID).

§topic: String

Message topic/type for routing or categorization.

§payload: Vec<u8>

Binary payload data.

§sender: String

Name of the sending agent.

§metadata: HashMap<String, String>

Arbitrary metadata key-value pairs.

§created_at: i64

Creation timestamp in microseconds since epoch.

Implementations§

Source§

impl Message

Source

pub fn new( topic: impl Into<String>, payload: Vec<u8>, sender: impl Into<String>, ) -> Self

Create a new message.

§Arguments
  • topic - Message topic for routing
  • payload - Binary payload
  • sender - 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
Hide additional 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    }
Source

pub fn correlation_id(&self) -> Option<String>

Get the correlation ID for request/response tracking.

Source

pub fn set_correlation_id(&mut self, id: &str)

Set a correlation ID for request/response tracking.

Trait Implementations§

Source§

impl Clone for Message

Source§

fn clone(&self) -> Message

Returns a duplicate of the value. Read more
1.0.0 · Source§

fn clone_from(&mut self, source: &Self)

Performs copy-assignment from source. Read more
Source§

impl Debug for Message

Source§

fn fmt(&self, f: &mut Formatter<'_>) -> Result

Formats the value using the given formatter. Read more
Source§

impl<'de> Deserialize<'de> for Message

Source§

fn deserialize<__D>(__deserializer: __D) -> Result<Self, __D::Error>
where __D: Deserializer<'de>,

Deserialize this value from the given Serde deserializer. Read more
Source§

impl Serialize for Message

Source§

fn serialize<__S>(&self, __serializer: __S) -> Result<__S::Ok, __S::Error>
where __S: Serializer,

Serialize this value into the given Serde serializer. Read more

Auto Trait Implementations§

Blanket Implementations§

Source§

impl<T> Any for T
where T: 'static + ?Sized,

Source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
Source§

impl<T> Borrow<T> for T
where T: ?Sized,

Source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
Source§

impl<T> BorrowMut<T> for T
where T: ?Sized,

Source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
Source§

impl<T> CloneToUninit for T
where T: Clone,

Source§

unsafe fn clone_to_uninit(&self, dest: *mut u8)

🔬This is a nightly-only experimental API. (clone_to_uninit)
Performs copy-assignment from self to dest. Read more
Source§

impl<T> From<T> for T

Source§

fn from(t: T) -> T

Returns the argument unchanged.

Source§

impl<T> FromRef<T> for T
where T: Clone,

Source§

fn from_ref(input: &T) -> T

Converts to this type from a reference to the input type.
Source§

impl<T> Instrument for T

Source§

fn instrument(self, span: Span) -> Instrumented<Self>

Instruments this type with the provided Span, returning an Instrumented wrapper. Read more
Source§

fn in_current_span(self) -> Instrumented<Self>

Instruments this type with the current Span, returning an Instrumented wrapper. Read more
Source§

impl<T, U> Into<U> for T
where U: From<T>,

Source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

Source§

impl<T> IntoEither for T

Source§

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 more
Source§

fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
where F: FnOnce(&Self) -> bool,

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
Source§

impl<T> PolicyExt for T
where T: ?Sized,

Source§

fn and<P, B, E>(self, other: P) -> And<T, P>
where T: Policy<B, E>, P: Policy<B, E>,

Create a new Policy that returns Action::Follow only if self and other return Action::Follow. Read more
Source§

fn or<P, B, E>(self, other: P) -> Or<T, P>
where T: Policy<B, E>, P: Policy<B, E>,

Create a new Policy that returns Action::Follow if either self or other returns Action::Follow. Read more
Source§

impl<T> ToOwned for T
where T: Clone,

Source§

type Owned = T

The resulting type after obtaining ownership.
Source§

fn to_owned(&self) -> T

Creates owned data from borrowed data, usually by cloning. Read more
Source§

fn clone_into(&self, target: &mut T)

Uses borrowed data to replace owned data, usually by cloning. Read more
Source§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

Source§

type Error = Infallible

The type returned in the event of a conversion error.
Source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
Source§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

Source§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
Source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.
Source§

impl<T> WithSubscriber for T

Source§

fn with_subscriber<S>(self, subscriber: S) -> WithDispatch<Self>
where S: Into<Dispatch>,

Attaches the provided Subscriber to this type, returning a WithDispatch wrapper. Read more
Source§

fn with_current_subscriber(self) -> WithDispatch<Self>

Attaches the current default Subscriber to this type, returning a WithDispatch wrapper. Read more
Source§

impl<T> DeserializeOwned for T
where T: for<'de> Deserialize<'de>,

Source§

impl<A, B, T> HttpServerConnExec<A, B> for T
where B: Body,