use calloop::EventLoop;
// futures = "0.3"use futures::sink::SinkExt;
use futures::stream::StreamExt;
fnmain() -> std::io::Result<()> {
let (exec, sched) = calloop::futures::executor()?;
letmut event_loop = EventLoop::try_new()?;
let handle = event_loop.handle();
handle
.insert_source(exec, |evt, _metadata, _shared| {
// Print the value of the async block ie. the return value.println!("Async block ended with: {}", evt);
})
.map_err(|e| e.error)?;
// Let's create two channels for our async blocks below. The blocks will// exchange messages via these channels.let (mut sender_friendly, mut receiver_friendly) = futures::channel::mpsc::unbounded();
let (mut sender_aloof, mut receiver_aloof) = futures::channel::mpsc::unbounded();
// Our toy async code.let async_friendly_task = asyncmove {
sender_friendly.send("Hello,").await.ok();
ifletSome(msg) = receiver_aloof.next().await {
println!("Aloof said: {}", msg);
}
sender_friendly.send("world!").await.ok();
ifletSome(msg) = receiver_aloof.next().await {
println!("Aloof said: {}", msg);
}
"Bye!" };
let async_aloof_task = asyncmove {
ifletSome(msg) = receiver_friendly.next().await {
println!("Friendly said: {}", msg);
}
sender_aloof.send("Oh,").await.ok();
ifletSome(msg) = receiver_friendly.next().await {
println!("Friendly said: {}", msg);
}
sender_aloof.send("it's you.").await.ok();
"Regards." };
// Schedule the async block to be run in the event loop. sched.schedule(async_friendly_task).unwrap();
sched.schedule(async_aloof_task).unwrap();
// Run the event loop.println!("Starting event loop. Use Ctrl-C to exit.");
event_loop.run(None, &mut (), |_| {})?;
println!("Event loop ended.");
Ok(())
}
The executor, the part that executes the future, goes in the event loop:
use calloop::EventLoop;
// futures = "0.3"use futures::sink::SinkExt;
use futures::stream::StreamExt;
fnmain() -> std::io::Result<()> {
let (exec, sched) = calloop::futures::executor()?;
letmut event_loop = EventLoop::try_new()?;
let handle = event_loop.handle();
handle
.insert_source(exec, |evt, _metadata, _shared| {
// Print the value of the async block ie. the return value.println!("Async block ended with: {}", evt);
})
.map_err(|e| e.error)?;
// Let's create two channels for our async blocks below. The blocks will// exchange messages via these channels.let (mut sender_friendly, mut receiver_friendly) = futures::channel::mpsc::unbounded();
let (mut sender_aloof, mut receiver_aloof) = futures::channel::mpsc::unbounded();
// Our toy async code.let async_friendly_task = asyncmove {
sender_friendly.send("Hello,").await.ok();
ifletSome(msg) = receiver_aloof.next().await {
println!("Aloof said: {}", msg);
}
sender_friendly.send("world!").await.ok();
ifletSome(msg) = receiver_aloof.next().await {
println!("Aloof said: {}", msg);
}
"Bye!" };
let async_aloof_task = asyncmove {
ifletSome(msg) = receiver_friendly.next().await {
println!("Friendly said: {}", msg);
}
sender_aloof.send("Oh,").await.ok();
ifletSome(msg) = receiver_friendly.next().await {
println!("Friendly said: {}", msg);
}
sender_aloof.send("it's you.").await.ok();
"Regards." };
// Schedule the async block to be run in the event loop. sched.schedule(async_friendly_task).unwrap();
sched.schedule(async_aloof_task).unwrap();
// Run the event loop.println!("Starting event loop. Use Ctrl-C to exit.");
event_loop.run(None, &mut (), |_| {})?;
println!("Event loop ended.");
Ok(())
}
Now let's write our async code in full:
use calloop::EventLoop;
// futures = "0.3"use futures::sink::SinkExt;
use futures::stream::StreamExt;
fnmain() -> std::io::Result<()> {
let (exec, sched) = calloop::futures::executor()?;
letmut event_loop = EventLoop::try_new()?;
let handle = event_loop.handle();
handle
.insert_source(exec, |evt, _metadata, _shared| {
// Print the value of the async block ie. the return value.println!("Async block ended with: {}", evt);
})
.map_err(|e| e.error)?;
// Let's create two channels for our async blocks below. The blocks will// exchange messages via these channels.let (mut sender_friendly, mut receiver_friendly) = futures::channel::mpsc::unbounded();
let (mut sender_aloof, mut receiver_aloof) = futures::channel::mpsc::unbounded();
// Our toy async code.let async_friendly_task = asyncmove {
sender_friendly.send("Hello,").await.ok();
ifletSome(msg) = receiver_aloof.next().await {
println!("Aloof said: {}", msg);
}
sender_friendly.send("world!").await.ok();
ifletSome(msg) = receiver_aloof.next().await {
println!("Aloof said: {}", msg);
}
"Bye!"
};
let async_aloof_task = asyncmove {
ifletSome(msg) = receiver_friendly.next().await {
println!("Friendly said: {}", msg);
}
sender_aloof.send("Oh,").await.ok();
ifletSome(msg) = receiver_friendly.next().await {
println!("Friendly said: {}", msg);
}
sender_aloof.send("it's you.").await.ok();
"Regards."
};
// Schedule the async block to be run in the event loop. sched.schedule(async_friendly_task).unwrap();
sched.schedule(async_aloof_task).unwrap();
// Run the event loop.println!("Starting event loop. Use Ctrl-C to exit.");
event_loop.run(None, &mut (), |_| {})?;
println!("Event loop ended.");
Ok(())
}
Like any block in Rust, the value of your async block is the last expression ie. it is effectively "returned" from the block, which means it will be provided to your executor's callback as the first argument (the "event"). You'll see this in the output with the Async block ended with: ... lines.
Finally, we run the loop:
use calloop::EventLoop;
// futures = "0.3"use futures::sink::SinkExt;
use futures::stream::StreamExt;
fnmain() -> std::io::Result<()> {
let (exec, sched) = calloop::futures::executor()?;
letmut event_loop = EventLoop::try_new()?;
let handle = event_loop.handle();
handle
.insert_source(exec, |evt, _metadata, _shared| {
// Print the value of the async block ie. the return value.println!("Async block ended with: {}", evt);
})
.map_err(|e| e.error)?;
// Let's create two channels for our async blocks below. The blocks will// exchange messages via these channels.let (mut sender_friendly, mut receiver_friendly) = futures::channel::mpsc::unbounded();
let (mut sender_aloof, mut receiver_aloof) = futures::channel::mpsc::unbounded();
// Our toy async code.let async_friendly_task = asyncmove {
sender_friendly.send("Hello,").await.ok();
ifletSome(msg) = receiver_aloof.next().await {
println!("Aloof said: {}", msg);
}
sender_friendly.send("world!").await.ok();
ifletSome(msg) = receiver_aloof.next().await {
println!("Aloof said: {}", msg);
}
"Bye!" };
let async_aloof_task = asyncmove {
ifletSome(msg) = receiver_friendly.next().await {
println!("Friendly said: {}", msg);
}
sender_aloof.send("Oh,").await.ok();
ifletSome(msg) = receiver_friendly.next().await {
println!("Friendly said: {}", msg);
}
sender_aloof.send("it's you.").await.ok();
"Regards." };
// Schedule the async block to be run in the event loop.
sched.schedule(async_friendly_task).unwrap();
sched.schedule(async_aloof_task).unwrap();
// Run the event loop.println!("Starting event loop. Use Ctrl-C to exit.");
event_loop.run(None, &mut (), |_| {})?;
println!("Event loop ended.");
Ok(())
}
Note that for the sake of keeping this example short, I've written the async code before running the loop. But async code can be scheduled from callbacks, or other sources within the loop too.
One of Calloop's strengths is that it is completely single threaded as written. However, many async crates are implemented using threads eg. async-std and async-process. This is not an inherent problem! Calloop will work perfectly well with such implementations in general. However, if you have selected Calloop because of your own constraints around threading, be aware of this.