A version of async-stream without macros.
This crate provides generic implementations of Stream trait.
Stream is an asynchronous version of std::iter::Iterator.
Two functions are provided - fn_stream and try_fn_stream.
If you need to create a stream that may result in error, use try_fn_stream, otherwise use fn_stream.
To create a stream:
- Invoke
fn_streamortry_fn_stream, passing a closure (anonymous function). - Closure will accept an
emitter. To return value from the stream, call.emit(value)onemitterand.awaiton its result. Once stream consumer has processed the value and called.next()on stream,.awaitwill return.
try_fn_stream provides some conveniences for returning errors:
- Errors can be return from closure via
return Err(...)or the question mark (?) operator. This will end the stream. - An
emitteralso has anemit_err()method to return errors without ending the stream.
fn_stream does not support cross-task streams, that is all stream values must be produced in the same task as the stream.
Specifically, it is not supported to move StreamEmitter to another thread or tokio task.
If your use case necessitates this, consider using async chanells for that.
Internal concurrency is supported within fn_stream (see examples/join.rs).
Finite stream of numbers
use async_fn_stream::fn_stream;
use futures_util::Stream;
fn build_stream() -> impl Stream<Item = i32> {
fn_stream(|emitter| async move {
for i in 0..3 {
// yield elements from stream via `emitter`
emitter.emit(i).await;
}
})
}Read numbers from text file, with error handling
use anyhow::Context;
use async_fn_stream::try_fn_stream;
use futures_util::{pin_mut, Stream, StreamExt};
use tokio::{
fs::File,
io::{AsyncBufReadExt, BufReader},
};
fn read_numbers(file_name: String) -> impl Stream<Item = Result<i32, anyhow::Error>> {
try_fn_stream(|emitter| async move {
// Return errors via `?` operator.
let file = BufReader::new(File::open(file_name).await.context("Failed to open file")?);
pin_mut!(file);
let mut line = String::new();
loop {
line.clear();
let byte_count = file
.read_line(&mut line)
.await
.context("Failed to read line")?;
if byte_count == 0 {
break;
}
for token in line.split_ascii_whitespace() {
let Ok(number) = token.parse::<i32>() else {
// Return errors via the `emit_err` method.
emitter.emit_err(
anyhow::anyhow!("Failed to convert string \"{token}\" to number")
).await;
continue;
};
emitter.emit(number).await;
}
}
Ok(())
})
}async-stream is great! It has a nice syntax, but it is based on macros which brings some flaws:
- proc-macros sometimes interact badly with IDEs such as
rust-analyzerorRustRover. see e.g. rust-lang/rust-analyzer#11533 - proc-macros may increase build times