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:
fn_stream
or try_fn_stream
, passing a closure (anonymous function).emitter
.
To return value from the stream, call .emit(value)
on emitter
and .await
on its result.
Once stream consumer has processed the value and called .next()
on stream, .await
will return.try_fn_stream
only) Return errors from closure via return Err(...)
or ?
(question mark) operator.Finite stream of numbers
```rust use asyncfnstream::fn_stream; use futures::Stream;
fn buildstream() -> impl Streamemitter
emitter.emit(i).await;
}
})
}
```
Read numbers from text file, with error handling
```rust use anyhow::Context; use asyncfnstream::tryfnstream; use futures::{pin_mut, Stream, StreamExt}; use tokio::{ fs::File, io::{AsyncBufReadExt, BufReader}, };
fn readnumbers(filename: String) -> impl Stream?
operator.
let file = BufReader::new(File::open(filename).await.context("Failed to open file")?);
pinmut!(file);
let mut line = String::new();
loop {
line.clear();
let bytecount = file
.readline(&mut line)
.await
.context("Failed to read line")?;
if byte_count == 0 {
break;
}
for token in line.split_ascii_whitespace() {
let number: i32 = token
.parse()
.with_context(|| format!("Failed to conver string \"{token}\" to number"))?;
// Return errors via `?` operator.
emitter.emit(number).await;
}
}
Ok(())
})
} ```
async-stream
?async-stream is great! It has a nice syntax, but it is based on macros which brings some flaws: * proc-macros sometimes interacts badly with IDEs such as rust-analyzer or IntelliJ Rust. see e.g. https://github.com/rust-lang/rust-analyzer/issues/11533 * proc-macros may increase build times