condtype

docs.rs crates.io github

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. 💖

Conditional Typing

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.

Examples

In the following example, [CondType] aliases either [&str] or [i32], depending on the boolean generic constant:

```rust use condtype::CondType;

let str: CondType = "hello"; let int: CondType = 42;

// Unsized types are also supported: let str: &CondType

[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.

```rust use condtype::condval;

const COND: bool = true;

let val = condval!(if COND { "hello" } else { 42 });

assert_eq!(val, "hello"); ```

Platform-Specific Types

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 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.

```rust use condtype::{condval, CondType}; use libc::{rlimt, RLIMINFINITY};

const RLIMINFINITYISMAX: bool = RLIMINFINITY == rlim_t::MAX;

type RlimOption = CondType, rlim_t>;

const RLIMNONE: RlimOption = condval!(if RLIMINFINITYISMAX { None:: } else { rlim_t::MAX });

// Convert from either RlimOption type to Option via the Into trait: let rlimnone: Option = RLIMNONE.into(); ```

Without this library, one could otherwise use [cfg_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.

```rust use cfgif::cfgif; use libc::rlim_t;

cfgif! { // Platforms where 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 = Option; const RLIMNONE: RlimOption = None; } } ```

Limitations

It is currently not possible to use [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

This library is available on crates.io and can be used by running the following cargo command in your project directory:

sh cargo add condtype

or by manually adding the following to your project's Cargo.toml:

toml [dependencies] condtype = "1.2.0"

License

Like the Rust project, this library may be used under either the MIT License or Apache License (Version 2.0).