fixed-buffer
This is a Rust library with fixed-size buffers, useful for network protocol parsers.
Features
- Write bytes to the buffer and read them back
- Lives on the stack
- Does not allocate memory
- Supports tokio's AsyncRead and AsyncWrite
- Use it to read a stream, search for a delimiter, and save leftover bytes for the next read.
- Easy to learn & use. Easy to maintain code that uses it.
- Works with Rust
latest
, beta
, and nightly
- No macros
Limitations
- Not a circular buffer.
You can call
shift()
periodically to move unread bytes to the front of the buffer.
- There is no
iterate_delimited(AsyncRead)
.
Because of borrowing rules, such a function would need to return non-borrowed (allocated and copied) data.
Examples
Read and handle requests from a remote client:
```rust
use fixed_buffer::FixedBuf;
use std::io::Error;
use tokio::io::AsyncWriteExt;
use tokio::net::TcpStream;
async fn handleconn(mut tcpstream: TcpStream) -> Result<(), Error> {
let (mut input, mut output) = tcpstream.split();
let mut buf: FixedBuf<[u8; 4096]> = FixedBuf::new([0; 4096]);
loop {
// Read a line and leave leftover bytes in buf
.
let linebytes = match buf.readdelimited(&mut input, b"\n").await? {
Some(linebytes) => linebytes,
None => return Ok(()),
};
let request = Request::parse(linebytes)?;
// Read any request payload from buf
+ TcpStream
.
let payloadreader = tokio::io::AsyncReadExt::chain(&mut buf, &mut input);
handlerequest(&mut output, payload_reader, request).await?;
}
}
```
For a runnable example, see examples/server.rs.
Read and process records:
```rust
fn tryprocessrecord(b: &[u8]) -> Result {
if b.len() < 2 {
return Ok(0)
}
if b.startswith("ab".asbytes()) {
println!("found record");
Ok(2)
} else {
Err(Error::new(ErrorKind::InvalidData, "bad record"))
}
}
async fn readandprocess(mut input: R)
-> Result<(), Error> {
let mut buf: FixedBuf<[u8; 1024]> = FixedBuf::new([0; 1024]);
loop {
// Read a chunk into the buffer.
let mut writable = buf.writable()
.okor(Error::new(ErrorKind::InvalidData, "record too long, buffer full"))?;
let byteswritten = AsyncReadExt::read(&mut input, &mut writable).await?;
if byteswritten == 0 {
return if buf.len() == 0 {
Ok(()) // EOF at record boundary
} else {
// EOF in the middle of a record
Err(Error::from(ErrorKind::UnexpectedEof))
};
}
buf.wrote(byteswritten);
// Process records in the buffer.
loop {
let bytes_read = try_process_record(buf.readable())?;
if bytes_read == 0 {
break;
}
buf.read_bytes(bytes_read);
}
// Shift data in the buffer to free up space at the end for writing.
buf.shift();
}
}
```
Documentation
https://docs.rs/fixed-buffer
The filled
constructor is useful in tests.
Alternatives
Release Process
- Edit
Cargo.toml
and bump version number.
- Run
./release.sh
Changelog
- v0.1.7 - Add
FixedBuf::escape_ascii
.
- v0.1.6 - Add
filled
constructor.
- v0.1.5 - Change
read_delimited
to return Option<&[u8]>
, for clean EOF handling.
- v0.1.4 - Add
clear()
.
- v0.1.3
- Thanks to freax13 for these changes:
- Support any buffer size. Now you can make
FixedBuf<[u8; 42]>
.
- Support any
AsRef<[u8]> + AsMut<[u8]>
value for internal memory:
[u8; N]
Box<[u8; N]>
&mut [u8]
Vec<u8>
- Renamed
new_with_mem
to new
.
Use FixedBuf::default()
to construct any FixedBuf<T: Default>
, which includes
arrays of sizes up to 32.
- v0.1.2 - Updated documentation.
- v0.1.1 - First published version
TODO
- DONE - Try to make this crate comply with the Rust API Guidelines.
- DONE - Find out how to include Readme.md info in the crate's docs.
- DONE - Make the repo public
- DONE - Set up continuous integration tests and banner.
- https://github.com/actions-rs/example
- https://alican.codes/rust-github-actions/
- DONE - Add some documentation tests
- https://doc.rust-lang.org/rustdoc/documentation-tests.html
- https://doc.rust-lang.org/stable/rust-by-example/testing/doc_testing.html
- DONE - Set up public repository on Gitlab.com
- https://gitlab.com/mattdark/firebase-example/blob/master/.gitlab-ci.yml
- https://medium.com/astraol/optimizing-ci-cd-pipeline-for-rust-projects-gitlab-docker-98df64ae3bc4
- https://hub.docker.com/_/rust
- DONE - Custom buffer length.
- https://crates.io/crates/generic-array
- https://crates.io/crates/block-buffer
- https://crates.io/crates/string-wrapper
- DONE - Publish to creates.io
- DONE - Read through https://crate-ci.github.io/index.html
- DONE - Get a code review from an experienced rustacean
- DONE - Add and update a changelog
- Update it manually
- https://crate-ci.github.io/release/changelog.html
- Add features: std, tokio, async-std
- Simplify
read_delimited()
- Make a more generic read_frame that takes a frame detector function.
Make
read_delimited
use that.
- Implement FixedBuf::chain(AsyncRead) which buffers reads like tokio::io::ReadBuf.
- Fix FixedBuf rustdoc link to box_benchmark.
- Switch to const generics once they are stable:
- https://github.com/rust-lang/rust/issues/44580
- https://stackoverflow.com/a/56543462
- Set up CI on:
- DONE - Linux x86 64-bit
- macOS
- Windows
- https://crate-ci.github.io/pr/testing.html#travisci
- Linux ARM 64-bit (Raspberry Pi 3 and newer)
- Linux ARM 32-bit (Raspberry Pi 2)
- RISCV & ESP32 firmware?