condtype
Choose Rust types at compile-time via boolean constants, brought to you by Nikolai Vazquez.
If you find this library 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 the [Either
] type, the type chosen by [CondType
]/[condval!
] is
directly used, rather than wrapped 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 either [&str
] or [i32
],
depending on the boolean generic constant:
```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
}); assert_eq!(val, "hello");
``` This library can make 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 library, one could otherwise use [ ```rust
use cfgif::cfgif;
use libc::rlim_t; cfgif! {
// Platforms where It is currently not possible to use [ This library 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
.Platform-Specific Types
RlimOption
type can be either
[Option]\<rlim_t>
or rlim_t
itself,
where [rlim_t::MAX
] can be treated as a sentinel value for
Option::None
if it is not equal to RLIM_INFINITY
.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 library 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.2.0"
License