Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/redbadger/crux

Cross-platform app development in Rust
https://github.com/redbadger/crux

mobile-development rust

Last synced: 4 days ago
JSON representation

Cross-platform app development in Rust

Awesome Lists containing this project

README

        

# [Crux](https://red-badger.com/crux) · [![GitHub license](https://img.shields.io/github/license/redbadger/crux?color=blue)](https://github.com/redbadger/crux/blob/master/LICENSE) [![Crate version](https://img.shields.io/crates/v/crux_core.svg)](https://crates.io/crates/crux_core) [![Docs](https://img.shields.io/badge/docs.rs-crux_core-green)](https://docs.rs/crux_core/) [![Build status](https://img.shields.io/github/actions/workflow/status/redbadger/crux/build.yaml)](https://github.com/redbadger/crux/actions)

## Cross-platform app development in Rust

- **Shared Core for Behavior** - Crux helps you share your app's business logic
and behavior across mobile (iOS/Android) and web — as a single reusable core
built with Rust.
- **Thin Shell for UI** - Crux recognizes that the best experiences are built
with modern declarative frameworks such as
[SwiftUI](https://developer.apple.com/xcode/swiftui/),
[Jetpack Compose](https://developer.android.com/jetpack/compose),
[React](https://reactjs.org/)/[Vue](https://vuejs.org/), or a WebAssembly
based framework (like [Yew](https://yew.rs/)) — however, it aims to keep this
UI layer as thin as it can be, with all other work done by the shared core.
- **Type Generation** - the interface with the core has static type checking
across languages — types and serialization code are generated for Swift,
Kotlin and TypeScript. Rust shells can import the core directly.
- **Capabilities** - capabilities express the intent for side effects such as
calling an API. Because all side effects (including UI) are performed by the
shell, the core becomes trivial to test comprehensively — test suites run in
milliseconds (not in minutes or hours).

## Getting Started

[Learn how to use Crux in your project](https://redbadger.github.io/crux).

Follow the
[readme in the project's repository on Github](https://github.com/redbadger/crux).

Read the [API documentation](https://docs.rs/crux_core/latest/crux_core/)

[Watch the introductory talk](https://www.youtube.com/watch?v=cWCZms92-1g&t=5s)
at the recent [Rust Nation 2023](https://www.rustnationuk.com/) conference in
London.

You can also join the friendly conversation on our
[Zulip channel](https://crux-community.zulipchat.com).

> Note, that Crux is experimental and currently under active development
> (probably not ready for use in production apps just yet). However, the master
> branch should always be working well, and we will try to keep the examples and
> documentation up to date as we go. We _do_ think that the API has now settled,
> so have a play! :-)

## Architectural Overview

![Logical architecture](./crux_core/architecture.svg)

The fundamental architectural concept is the strict separation of pure
computational tasks from tasks that cause side effects. This is similar to the
way [Elm](https://guide.elm-lang.org/architecture/) works.

### Side-effect-free core

In the above diagram, the inner "Core" is compiled and linked to the outer
"Shell" on each platform as a library:

- On iOS as a native static library
- On Android as a dynamic library using
[Java Native Access](https://github.com/java-native-access/jna)
- In a browser as a WebAssembly module

In fact, because WebAssembly (Wasm) is one of the compilation targets, the core
_must_ remain side-effect free, due to the sandboxed nature of the Wasm runtime
environment.

As such, the core is completely isolated and secure against software
supply-chain attacks, as it has no access to any external APIs. All it can do is
perform pure calculations and keep internal state.

Following the Elm architecture, the core defines the key component types within
the application:

- `Event` — an `enum` describing the events which the core can handle
- `Model` — describes the internal state of the application
- `ViewModel` — represents information that should be displayed to the user

The former two are tied together by the `update` function, familiar from Elm,
Redux or other event sourcing architectures, which currently has this type
signature:

```rust,ignore
fn update(
&self,
event: Event,
model: &mut Model,
capabilities: &Capabilities,
)
```

The job of the `update` function is to process an `Event`, update the model
accordingly, and potentially request some side-effects using capabilities.

### Application Shell

The enclosing platform native "Shell" is written using the language appropriate
for the platform, and acts as the runtime environment within which all the
non-pure tasks are performed. From the perspective of the core, the shell is the
platform on which the core runs.

## Communication Between the Application Shell and the Core

Following the Elm architecture, the interface with the core is message based.
This means that the core is unable to perform anything other than pure
calculations. To perform any task that creates a side-effect (such as an HTTP
call or random number generation), the core must request it from the shell.

The core has a concept of Capabilities — reusable interfaces for common
side-effects — supporting fire-and-forget, request/response, and streaming
semantics.

The only built-in capability is `Render`. But this repository contains a few
capabilities at various stages of maturity, and you can easily write your own if
you want to:

1. `Render` (ask UI to render the ViewModel) —
[source](./crux_core/src/capabilities/render.rs), built-in to `crux_core`,
request only
2. `Http` (full HTTP implementation based on the
[Surf](https://crates.io/crates/surf) API) — [source](./crux_http/README.md),
[crate](https://crates.io/crates/crux_http), request/response
3. `KeyValue` (basic key-value store API) — [source](./crux_kv/README.md),
[crate](https://crates.io/crates/crux_kv), request/response
4. `Time` (get current time, notify after duration, notify at instant) —
[source](./crux_time/README.md), [crate](https://crates.io/crates/crux_time),
request/response
5. `Platform` (get the current platform) — [source](./crux_platform/README.md),
[crate](https://crates.io/crates/crux_platform), request/response
6. `SSE` (basic Server-Sent Events) —
[source](./examples/counter/shared/src/capabilities/sse.rs),
request/streaming
7. `PubSub` (pub sub with streaming) —
[source](./examples/notes/shared/src/capabilities/pub_sub.rs),
request/response/streaming
8. `Timer` (timer start, finish, cancel) —
[source](./examples/notes/shared/src/capabilities/timer.rs),
request/response/streaming
9. `Delay` — part of
[tutorial](https://redbadger.github.io/crux/guide/capability_apis.html#basic-delay-capability)
in the [book](https://redbadger.github.io/crux)

![crux](./docs/src/crux.png)

This means the core interface is simple:

- `process_event: Event -> Vec` - processes a user interaction event
and potentially responds with capability requests. This is the API for the
_driving_ side in the above diagram.
- `handle_response: (uuid, SomeResponse) -> Vec` - handles the response
from the capability and potentially follows up with further requests. This is
the API for the _driven_ side in the above diagram.
- `view: () -> ViewModel` - provides the shell with the current data for
displaying user interface

Updating the user interface is considered a side-effect and is provided by the
built-in `Render` capability.

This design means the core can be tested very easily, without any mocking and
stubbing, by simply checking the Input/Output behaviour of the three functions.

### Foreign Function Interface

The Foreign Function Interface allowing the shell to call the above functions is
provided by Mozilla's [UniFFI](https://mozilla.github.io/uniffi-rs/) on a mobile
device, or in the browser, by
[wasm-pack](https://rustwasm.github.io/wasm-pack/).

In order to both send more complex data than UniFFI currently supports, and
enforce the message passing semantics, all messages are serialized, sent across
the boundary, then deserialized using
[serde-generate](https://docs.rs/serde-generate/latest/serde_generate/) which
also provides type generation for the foreign (non-Rust) languages.

This means that changes to types in the core, especially the `Event` and
`Request` types, propagate out into the shell implementations and cause type
errors where appropriate (such as an exhaustive match on an enum check).

### Message Types

Three types of message are exchanged between the application and the core.

- Messages of type `Event` are sent from the Shell to the Core in response to an
event happening in the user interface (the _driving_ side). They start a
potential sequence of further message exchanges between the shell and the
core. Messages are passed on unchanged.
- Messages of type `Request` are sent from the Core to the Shell to request the
execution of some side-effect-inducing task. The Core responds with zero or
more `Request` messages after receiving an `Event` message (the _driven_
side).
- Response messages are sent from the Shell to the Core carrying the result of
an earlier request.

`Request` messages contain the inputs for the requested side-effect, along with
a `uuid` used by the core to pair requests and their responses together. The
exact mechanics are not important, but it is important for the request's `uuid`
to be passed on to the corresponding response.

## Typical Message Exchange Cycle

A typical message exchange cycle may look like this:

1. User interaction occurs in the Shell, which results in an event
1. The Shell handles this event by constructing an `Event`
1. The Shell calls the Core's `process_event` function passing the `Event` as an
argument
1. The Core performs the required processing, updating both its inner state and
the view model
1. The Core returns one or more `Request` messages to the Shell

In the simplest case, the Core will respond to an `Event` by returning the
single `Request` - render.

This requests that the Shell re-renders the user interface. When `Render` is the
only response from the Core, the message cycle has completed and the Core has
now "settled".

In more complex cases however, the Core may well return multiple `Request`s;
each of which instructs the Shell to perform a side-effect-inducing task such
as:

- Make a network call, or
- Fetch the current date/time stamp, or
- Perform biometric authentication, or
- Obtain an image from the camera, or
- Whatever else you can think of...

Many of these side-effecting-inducing tasks are asynchronous. The Shell is
responsible for passing responses back to the core (to the `handle_response`
function), which may respond with further requests.

This exchange continues until the core stops requesting further side-effects
(typically the last side-effect requested would be `Render`).

## Run the Counter Example locally

Refer to [examples/counter](./examples/counter/README.md) README

## How to Start Your Own New Project

Refer to the
[Getting Started](https://redbadger.github.io/crux/getting_started/core.html)
section of the tutorials.

---

## Sponsors

Crux is kindly sponsored by the following organizations. Your help is very much
appreciated.

---

### Red Badger Consulting Limited

Red Badger logo

[Red Badger](https://red-badger.com/) is the digital product consultancy trusted
by blue chips and global brands. Our product design and technical pedigree allow
us to craft high-impact digital products customers want. We use modern
engineering approaches to deliver sustainable change. And embed digital
capabilities to power continuous innovation.

---

### Zulip

Zulip round icon

[Zulip](https://zulip.com/) is an open-source modern team chat app designed to
keep both live and asynchronous conversations organized.

Zulip sponsor Crux by providing our
[Zulip server](https://crux-community.zulipchat.com) — thank you Zulip!

---