5 releases
0.3.5 | Dec 23, 2024 |
---|---|
0.3.4 | Dec 19, 2024 |
0.3.3 | Dec 18, 2024 |
0.3.2 | Dec 16, 2024 |
0.3.1 | Dec 14, 2024 |
#431 in Rust patterns
355 downloads per month
35KB
369 lines
trait-cast
Requirements
This crate requires a nightly compiler.
What can this crate do?
This crate adds the TraitcastableAny
replacement trait for Any
.
It closely resembles the Any
trait for downcasting to a concrete type.
Additionally the TraitcastableAny
trait allows you to directly downcast to other &dyn Trait
s.
To make this work you must specify all target traits you want to be able to downcast to in the make_trait_castable(Trait1, Trait2, ...)
attribute macro.
This macro can be applied to structs, enums and unions.
It implements the TraitcastableAny
trait for your struct, enum or union.
Note: No modifications on the target traits are necessary. Which allows you to downcast to traits of other libraries you don't control.
Usage
-
Add the
trait_cast_rs
crate to yourCargo.toml
and switch to a nightly compiler. -
Add the
#[make_trait_castable(Trait1, Trait2, ...)]
macro to your struct/enum/union. List all traits you eventually want to be able todowncast
to. You must implement all listed traits. -
Use references to
dyn TraitcastableAny
throughout your code instead ofdyn Any
. -
Enjoy downcasting to trait objects.
Example
# #![cfg_attr(feature = "min_specialization", feature(min_specialization))]
# #![feature(ptr_metadata)]
use trait_cast::{
make_trait_castable, TraitcastableAny, TraitcastableAnyInfra, TraitcastableAnyInfraExt,
};
#[make_trait_castable(Print)]
struct Source(i32);
trait Print {
fn print(&self);
}
impl Print for Source {
fn print(&self) {
println!("{}", self.0);
}
}
let source = Box::new(Source(5));
let castable: Box<dyn TraitcastableAny> = source;
let x: &dyn Print = castable.downcast_ref().unwrap();
x.print();
EVEN MORE Examples 🔥
Check out the examples.
If you want to do something the make_trait_castable
attribute macro can't handle (like implementing for generic structs - pull requests are welcome)
check out the manual*.rs
examples.
There is also a decl marco available - check out the with_decl_macro*.rs
examples.
Features
-
alloc
- Adds special implementations forBox
,Rc
andArc
. Default feature. -
min_specialization
- ImplementsTraitcastableAny
for'static
types. Even types you don't control. However these default implementations ofTraitcastableAny
have no downcast targets.It additionally requires the following feature flags in the user code:
#![feature(min_specialization)]
-
downcast_unchecked
- Adds*_unchecked
variants to the downcast functions.
Upcasting to the real Any
With the trait_upcasting
rust feature you can even cast any &dyn TraitcastableAny
to &dyn Any
.
Alternatively you can list the Any
trait as a traitcast target.
However it is not possible to cast back to TraitcastableAny
(pull requests are welcome).
Authors
raldone01 and onestacked are the primary authors and maintainers of this library.
License
This project is released under either:
at your choosing.
Contribution
Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.
How it works
I will give you a quick rundown of our internal operations: 💦
Compile time:
-
Add a
casting
function for every downcast path to the concrete type. This function gets adyn TraitcastableAny
, which it then downcasts to a concrete type usingAny
in the background. In the last step it casts the concrete type to the wanted trait object and returns it. -
Add a
traitcast_targets
function that returns a const slice of (typeid
, transmuted casting function ptr).
Runtime:
- Get targets array
- Find the target
typeid
- Transmute function pointer back to original type
- Call the function pointer to get the wanted trait object
- Return it
- 💲 Profit 💲
SAFETY 🏰
- The unchecked variants of the
downcast
function all use unsafe - expectedly. - The only other use of unsafe is the transmutation of function pointers.
However when they are called they are transmuted back to their original type.
So this should be
105%
save.As long asTypeId
s don't collide.
Alternatives (and why our crate is the best)
This alternatives section is not exhaustive for a more objective/detailed comparison see the alternatives section of cast_trait_object.
- mopa: Had its last update 6 years ago. Has some unresolved unsoundness issues. Also requires modifications on traits themselves while we just modify the struct/enum/union (see note above).
- mopa-maintained: Might have fixed some issues but still has an old code base with just a version bump.
- traitcast:
Has no readme on crates.io.
Uses a GLOBAL REGISTRY with
lazy_static
. To be fair it allows you to use the defaultAny
and doesn't require nightly.
TODO: Remove this section once our last update is 6 years old.
Links
Dependencies
~4.5MB
~78K SLoC