condtype
Choose Rust types at compile-time via boolean constants, brought to you by Nikolai Vazquez.
If you find this crate useful, consider starring it as well as sponsoring or donating once. 💖
The [CondType
] type and [condval!
] macro choose types at compile-time using
[bool
] constants, just like std::conditional_t
in C++.
Unlike [Either
], the chosen type is directly used, rather than wrapping it
with an [enum
] type. This may be considered a form of dependent typing,
but it is limited in ability and is restricted to compile-time constants rather
than runtime values.
In the following example, [CondType
] aliases [&str
] or [i32
]:
```rust use condtype::CondType;
let str: CondType
// Unsized types are also supported:
let str: &CondType [ ```rust
use condtype::condval; const COND: bool = true; let val = condval!(if COND {
"hello"
} else {
42
});
``` This crate also enables making code for some platforms more efficient by using
smaller-sized types depending on platform-specific constants. In the following
example, the ```rust
use condtype::{condval, CondType};
use libc::{rlimt, RLIMINFINITY}; const RLIMINFINITYISMAX: bool = RLIMINFINITY == rlim_t::MAX; type RlimOption = CondType const RLIMNONE: RlimOption = condval!(if RLIMINFINITYISMAX {
None:: // Convert from either Without this crate, one could otherwise use [ ```rust
use cfgif::cfgif;
use libc::rlim_t; cfgif! {
// Platforms where It is currently not possible to use [ This crate is available on crates.io and can be
used by running the following or by manually adding the following to your project's Like the Rust project, this library may be used under either the
MIT License or
Apache License (Version 2.0).condval!
] enables choosing differently-typed values without specifying types.
In the following example, val
is inferred to be either [&str
] or [i32
],
depending on COND
.RlimOption
type can either be Option<rlim_t>
or rlim_t
itself, where rlim_t::MAX
can be treated as a sentinel value for
Option::None
.RlimOption
type to Option
via the Into
trait:
let rlimnone: Optioncfg_if!
] to achieve the same
goal. However, using #[cfg]
requires maintaining a list of platforms
and being more fine-grained if RLIM_INFINITY
is dependent on CPU architecture.RLIM_INFINITY != rlim_t::MAX
:
if #[cfg(any(
targetos = "macos",
targetos = "freebsd",
targetos = "solaris",
// ad nauseam...
))] {
type RlimOption = rlimt;
const RLIMNONE: RlimOption = rlimt::MAX;
} else {
type RlimOption = OptionLimitations
CondType
] or [condval!
] with a generic
constant because Rust does not yet consider trait implementations based on
booleans to be exhaustive.
Once that issue is resolved, all versions of this crate should just work with
generic constants.rust,ignore
fn generic<const B: bool>() {
let val: CondType<B, &str, i32> = condval!(if B {
"hello"
} else {
42
});
}
Install
cargo
command in your project directory:sh
cargo add condtype
Cargo.toml
:toml
[dependencies]
condtype = "1.1.0"
License