4 releases

0.1.3 Aug 13, 2024
0.1.2 Aug 13, 2024
0.1.1 Aug 12, 2024
0.1.0 Aug 12, 2024

#28 in #self-referential

Download history 109/week @ 2024-08-06 229/week @ 2024-08-13 2/week @ 2024-08-20 3/week @ 2024-08-27 8/week @ 2024-09-10 5/week @ 2024-09-17 12/week @ 2024-09-24 4/week @ 2024-10-01 2/week @ 2024-10-08

92 downloads per month
Used in incrstruct

MIT license

17KB
322 lines

Incremental Structure

crates.io docs.rs GitHub last commit

A Rust crate for building self-referencing structs using two-phase initialization.

Example

use std::cell::{Ref, RefCell};
use incrstruct::IncrStruct;

#[derive(IncrStruct)]
struct AStruct<'a> {
    #[borrows(b)]             // Borrowing from a tail field
    c: &'a Ref<'a, i32>,      // is possible.

    #[borrows(a)]             // You can only borrow from fields that
    b: Ref<'a, i32>,          // come after the current field.

    a: RefCell<i32>,          // A head field. Since you can only borrow
                              // immutable references, RefCell is useful.

    #[header]                 // The required header field.
    hdr: incrstruct::Header,  // The name is arbitrary.
}

// The AStructInit trait is generated by the derive macro and
// ensures the contract between the incrstruct library code and
// the user provided code matches. The functions are invoked in
// reverse field declaration order.
impl<'a> AStructInit<'a> for AStruct<'a> {
    fn init_field_c(b: &'a Ref<'a, i32>) -> &'a Ref<'a, i32> {
        b
    }

    fn init_field_b(a: &'a RefCell<i32>) -> Ref<'a, i32> {
        a.borrow()
    }
}

// Only head fields are provided to the generated `new_X` functions.
let my_a = AStruct::new_box(RefCell::new(42));

assert_eq!(*my_a.a.borrow(), *my_a.b);

See the documentation for more information.

Dependencies

~250–700KB
~17K SLoC