• Stars
    star
    169
  • Rank 224,453 (Top 5 %)
  • Language
    Rust
  • License
    Apache License 2.0
  • Created about 4 years ago
  • Updated over 2 years ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

Rusty bindings for Apple libraries

Fruity

github crates.io docs.rs

Rusty bindings for Apple libraries, brought to you by @NikolaiVazquez.

Index

  1. Donate
  2. Usage
    1. Feature Flags
  3. Goals
    1. Idiomatic Rust
    2. Zero Cost
  4. License

Donate

If this project is useful to you, consider sponsoring me or donating directly!

Doing so enables me to create high-quality open source software like this. ❀️

Usage

This library is available on crates.io and can be used in your project by adding the following to your project's Cargo.toml:

[dependencies.fruity]
version = "0.3.0"

Feature Flags

Each module for a library or framework has its own feature flag with the same name.

For example, this is how you enable the foundation module:

[dependencies.fruity]
version = "0.3.0"
features = ["foundation"]

This feature transitively enables the objc feature/module.

Goals

Idiomatic Rust

Fruity makes interfacing with these C and Objective-C APIs feel natural in Rust.

  • Automatic Reference Counting.

    Fruity takes advantage of Rust's ownership model to handle object reference counting for you.

    NSObject is a smart pointer that calls retain on Clone and release on Drop. This is exactly how Rust's Arc<T> works.

  • Option<NSObject>.

    In Objective-C, all objects are nullable unless marked with _Nonnull. This often leads to either very defensive checks or careless ignoring of null objects.

    Fruity reverses that and instead makes all objects (such as NSObject) non-null by default. An object can be made nullable by wrapping it with Option<T>.

    To make FFI safe and easy, the following Objective-C and Rust types are ABI-compatible:

    • NSObject * _Nonnull and NSObject

    • NSObject * _Nullable and Option<NSObject>

    This is because NSObject is a #[repr(transparent)] wrapper around a NonNull<T> pointer.

  • Result<T, NSError>.

    In Objective-C, methods take a pointer to where an NSError is placed upon failure. This makes it easy to avoid error handling and assume the happy path, which can lead to bugs when errors occur.

    Fruity instead returns a Result, which is the canonical way to handle errors in Rust. This ensures that errors must be acknowledged in some way.

  • Natural inheritance.

    Most of these types are classes that inherit from each other. Because true inheritance is not possible in Rust, Fruity uses Deref to model Objective-C subclassing.

  • Builder Pattern.

    Types like DispatchQueue have many configurable inputs to create an instance. Many of these inputs have standard default values, so it is cumbersome to specify them all each time. Swift solves this by having default parameters in init. However, Rust does not have default function parameters.

    Fruity instead solves this using the builder pattern. See DispatchQueueBuilder as an example. This reduces and simplifies code for creating dispatch queues.

Zero Cost

Using Fruity to interface with Objective-C libraries should have as little runtime cost as writing the same code directly in Objective-C.

This is true for the following:

  • Calling object methods.

    Method dispatch is always direct and does not need the error checking overhead of other wrappers that use the objc::msg_send! macro. This also reduces the size of your program by not emitting panics that would otherwise never get called.

    This library is carefully written to ensure that calls to objc_msgSend are always done with the correct object type, method selector, and arguments.

  • Getting a static class.

    Getters like NSString::class retrieve the class directly through its symbol. This is instantaneous, especially when compared to calling into the Objective-C runtime via objc_getClass.

  • Creating an NSString from a Rust string literal.

    The nsstring! macro creates an NSString literal (i.e. @"string") at compile time. There is no runtime dispatch/allocation/initialization cost.

Some parts of this library still aren't zero cost. Your help would be much appreciated here!

These are:

  • The selector! macro. See issue #2 for details.

License

This project is released under either the MIT License or Apache License (Version 2.0), at your choosing.

More Repositories

1

FileKit

Simple and expressive file management in Swift
Swift
2,289
star
2

RandomKit

Random data generation in Swift
Swift
1,473
star
3

divan

Fast and simple benchmarking for Rust projects
Rust
895
star
4

static-assertions

Ensure correct assumptions about constants, types, and more in Rust
Rust
566
star
5

Sage

A cross-platform chess library for Swift
Swift
376
star
6

impls

A Rust macro to determine if a type implements a logical trait expression
Rust
246
star
7

Menubar-Colors

A macOS app for convenient access to the system color panel
Swift
175
star
8

swift-bindgen

Bridging the gap between Swift and Rust
Rust
149
star
9

Unreachable

Unreachable code path optimization hint for Swift
Swift
103
star
10

Threadly

Type-safe thread-local storage in Swift
Swift
74
star
11

condtype

Choose Rust types at compile-time via constants
Rust
59
star
12

embed-plist-rs

Embed property list files like Info.plist directly in your Rust executable binary
Rust
39
star
13

Roman

Seamless Roman numeral conversion in Swift
Swift
36
star
14

rosy

Rust and Ruby, sitting in a tree 🌹
Rust
23
star
15

byte-set-rs

Efficient sets of bytes for Rust
Rust
19
star
16

cargo-emit

Talk to Cargo easily at build time
Rust
16
star
17

Clockmoji

Simple and intuitive time formatting using clock emoji.
Swift
13
star
18

XO

A cross-platform tic-tac-toe library for Swift
Swift
13
star
19

Lazy

Save the hard work for later, lazily evaluate values anywhere
Swift
12
star
20

Weak

Weak and Unowned as native Swift types
Swift
10
star
21

typebool

Type-level booleans in Rust for compile-time hackery
Rust
8
star
22

c-utf8-rs

UTF-8 encoded C strings for Rust
Rust
6
star
23

malloced

A malloc-ed box pointer type for Rust
Rust
5
star
24

dotfiles

Personal dotfiles
Shell
5
star
25

aloxide

Compile Rubies on Linux, macOS, and Windows
Rust
5
star
26

mods

Simpler Rust module declaration
Rust
5
star
27

bit-collection-rs

Iterate over bits in Rust
Rust
5
star
28

unsafe-unwrap-rs

Unsafely unwrap Result and Option types without checking in Rust
Rust
5
star
29

uncon-rs

Unchecked and unsafe type conversions in Rust
Rust
4
star
30

head-rs

Common Rust types with inline headers, such as HeaderVec for Vec
Rust
4
star
31

bad-rs

Unlicensed bad ideas
Rust
4
star
32

rust-workshop

A workshop on the Rust programming language
3
star
33

OneOrMore

A Swift collection of one or more elements
Swift
3
star
34

fmty

Rust library of composable `core::fmt` utilities
Rust
3
star
35

mem-cmp-rs

Safe memory comparison between types in Rust
Rust
2
star
36

ShiftOperations

A Β΅framework for a needed ShiftOperations protocol that does not appear in the Swift standard library
Swift
1
star
37

rust-selector-example

My attempt at creating Objective-C selectors in Rust at compile time
Shell
1
star
38

asygnal

Handle signals (e.g. ctrl-c) efficiently and asynchronously in Rust
Rust
1
star
39

chance-rs

Random number generation in Rust
Rust
1
star