Skip to content

Simply write and use `enum`s like rust native enums, freely passing through ffi.

License

Apache-2.0, MIT licenses found

Licenses found

Apache-2.0
LICENSE-APACHE
MIT
LICENSE-MIT
Notifications You must be signed in to change notification settings

AlseinX/ffi-enum

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

9 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

ffi-enum

Simply write and use enums like rust native enums, freely passing through ffi.

Crates.io Docs.rs MIT Or Apache-2.0 licensed Build Status

Why not using #[repr(C)] and #[non_exhaustive] ?

  • Rust's #[repr(C)] is not fully equal to a C-abi enum, and it is still an undefined behavior when an enum defined with #[repr(C)] recieves a value that is not listed in the definition of enum, from ffi.
  • #[non_exhaustive] is only designed for inter-crate compatibility, while the compiler might still optimize based on the assumption that the binary pattern of an enum value is always in its defined range.

Why not existing alternatives?

The current alternatives mostly define complex DSL with function macros, which could be hard to read/maintain and not supported by rustfmt or rust-analyzer, and they do not support derive macros.

While this crate offers a nearly native experience by trying the best to fully mimic the behaviors of native enums, despite it requires non-exhaustive matching even inside the defining crate. FFi enums are defined with native rust enum item syntax with full support of formatting, code hint, and auto completion.

use ffi_enum::prelude::*;
use serde::{Deserialize, Serialize};

#[ffi_enum]
#[derive(Debug, Serialize, Deserialize)]
#[serde(rename_all = "snake_case")]
pub enum Animal {
    Cat,
    Dog,
}

fn main() {
    let json = serde_json::to_string(&Animal::Cat).unwrap();
    assert_eq!(json, "\"cat\"");
    let value: Animal = serde_json::from_str(&json).unwrap();
    assert_eq!(value, Animal::Cat);
    let json = serde_json::to_string(&Animal::from(100u8)).unwrap();
    assert_eq!(json, "\"<Unknown>\"");
    let value: Animal = serde_json::from_str(&json).unwrap();
    assert_eq!(value, Animal::UNKNOWN);
}

About

Simply write and use `enum`s like rust native enums, freely passing through ffi.

Resources

License

Apache-2.0, MIT licenses found

Licenses found

Apache-2.0
LICENSE-APACHE
MIT
LICENSE-MIT

Stars

Watchers

Forks

Packages

No packages published

Languages