This crate provides traits and macros that make your application's structs and functions interactive.
Annotating a struct with #[derive(Interactive)]
, a struct's methods with #[Methods]
and a free function with #[Function]
will implement a set of traits,
that will allow you to access them as if Rust had a REPL.
Use this crate as an alternative for "print debugging" or as an ergonomic testing API.
This crate is no_std
compatible so you can use it to interact with embedded devices
and blink those LEDs from a USB or UART connection.
Interactive
], [Methods
] and [Function
]InteractiveRoot
] for iteval_to_string
but others allow for more custom behaviour)Since this crate makes a lot of use of the [Debug
] trait the helper macro [PartialDebug
] is provided.
It implements Debug
for a struct replacing all fields that do not implement Debug
with a placeholder.
Functions like get_all_field_names
are provided.
This makes it possible to implement things like auto-completion.
Have a look at the autocomplete example for how this might be done using the rustyline crate.
```rust use oy::{Interactive, Methods, InteractiveRoot, Function, PartialDebug};
struct NoDebug;
struct ChildStruct { lastsum: f32, nodebug: NoDebug, }
impl ChildStruct { fn add(&mut self, a: f32, b: f32) -> f32 { self.lastsum = a + b; self.lastsum } }
struct ParentStruct { child: ChildStruct, }
struct Root { parent: ParentStruct, }
fn splitstrat(s: &str, mid: usize) -> (&str, &str) { s.split_at(mid) }
let mut root = Root::default(); asserteq!(root.evaltostring("parent.child.add(4.2, 6.9)"), "11.1"); asserteq!(root.evaltostring("parent.child"), "ChildStruct { lastsum: 11.1, nodebug: Unknown }"); // splitstrat("foobar", 3) => ("foo", "bar") asserteq!(root.evaltostring("splitstr_at(\"foobar\", 3)"), "(\"foo\", \"bar\")"); ```
This crate makes use of the unstable specialization
feature, so it is only available on nightly.
Methods like try_as_interactive
are implemented on all types.
The method normally returns an error but in the specialized case
a trait object (&dyn Interactive
in this case) is returned.
The macros then implement getters that look something like this:
rust
fn get_field<'a>(&'a self, field_name: &'a str) -> Result<'_, &dyn Interactive> {
match field_name {
"field1" => self.field1.try_as_interactive(),
"field2" => self.field2.try_as_interactive(),
_ => Err(InteractiveError::FieldNotFound {
type_name: "Struct",
field_name,
}),
}
}
See the macro's documentation for more details.
Licensed under either of Apache License, Version 2.0 or MIT license at your option.
Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in this crate by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.