Featture toggles for Rust, extensible and with background synchronization and administration UI.
```rust use rusotos3::S3Client; use rusotocore::Region; use feattle::*; use std::sync::Arc;
/// A struct with your feature toggles
feattles! {
struct MyFeattles {
/// Is this usage considered cool?
iscool: bool = true,
/// Limit the number of "blings" available.
/// This will not change the number of "blengs", though!
maxblings: i32,
/// List the actions that should not be available
blocked_actions: Vec
// Store their values and history in AWS' S3 let s3client = S3Client::new(Region::default()); let persistence = S3::new(s3client, "my-bucket".toowned(), "some/s3/prefix/".toowned());
// Create a new instance let my_feattles = Arc::new(MyFeattles::new(persistence));
// Poll the storage in the background BackgroundSync::new(&my_feattles).spawn();
// Start the admin UI with warp
let adminpanel = Arc::new(AdminPanel::new(myfeattles.clone(), "Project Panda - DEV".toowned()));
tokio::spawn(runwarpserver(adminpanel, ([127, 0, 0, 1], 3030)));
// Read values (note the use of *
)
asserteq!(*myfeattles.iscool(), true);
asserteq!(my_feattles.max_blings(), 0);
assert_eq!(myfeattles.blockedactions(), Vec::
The macro will generate a struct with the given name and visibility modifier (assuming private
by default). The generated struct implements [Feattles
] and also exposes one method for each
feattle.
The methods created for each feattle allow reading their current value. For example, for a
feattle is_cool: bool
, there will be a method like
pub fn is_cool(&self) -> MappedRwLockReadGuard<bool>
. Note the use of
[parking_lot::MappedRwLockReadGuard
] because the interior of the struct is stored behind a RwLock
to
control concurrent access.
A feattle is created with the syntax $key: $type [= $default]
. You can use doc coments (
starting with ///
) to describe nicely what they do in your system. You can use any type that
implements [FeattleValue
] and optionally provide a default. If not provided, the default
will be created with Default::default()
.
This crate only disposes of low-level methods to load current feattles with [Feattles::reload()
]
and update their values with [Feattles::update()
]. Please look for the crates
feattle-sync and
feattle-ui for higher-level functionalities.
Due to some restrictions on how the macro is written, you can only use [feattles!
] once per
module. For example, the following does not compile:
```compilefail use feattlecore::feattles;
feattles! { struct A { } } feattles! { struct B { } } ```
You can work around this limitation by creating a sub-module and then re-exporting the generated
struct. Note the use of pub struct
in the second case.
```rust
use feattle_core::feattles;
feattles! { struct A { } }
mod b { use feattle_core::feattles; feattles! { pub struct B { } } }
use b::B; ```
uuid::Uuid
].S3
] to integrate with AWS' S3run_warp_server
] for a read-to-use integration with [warp
]Licensed under either of
at your option.
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.
See CONTRIBUTING.md.