Crates.io Crates.io

ghost_actor

A simple, ergonomic, idiomatic, macro for generating the boilerplate to use rust futures tasks in a concurrent actor style.

Hello World Example

```rust // Most of the GhostActor magic happens in this macro. // Sender and Handler traits will be generated here. ghostactor! { pub actor HelloWorldActor { fn helloworld() -> String; } }

// ... We'll skip implementing a handler for now ...

[tokio::main]

async fn main() { // spawn our actor, getting the actor sender. let sender = HelloWorldImpl::spawn().await;

// we can make async calls on the sender
assert_eq!("hello world!", &sender.hello_world().await.unwrap());

} ```

What's going on Here?

The ghost_actor! Macro

rust ghost_actor! { pub actor HelloWorldActor<GhostError> { fn hello_world() -> String; } }

The ghost_actor! macro takes care of writing the boilerplate for using async functions to communicate with an "actor" running as a futures task. The tests/examples here use tokio for the task executor, but the GhostActorBuilder returns a driver future for the actor task that you can manage any way you'd like.

The ghost_actor! macro generates some important types, many of which are derived by pasting words on to the end of your actor name. We'll use the actor name HelloWorldActor from above as an example:

Implementing an Actor Handler

```rust // We need a struct to implement our handler upon. struct HelloWorldImpl;

// All handlers must implement GhostControlHandler. // This provides a default no-op handleghostactor_shutdown impl. impl GhostControlHandler for HelloWorldImpl {}

// Implement GhostHandler for your specific GhostEvent type. // Don't worry, the compiler will let you know if you forget this : ) impl GhostHandler for HelloWorldImpl {}

// Now implement your actual handler - // auto generated by the ghost_event! macro. impl HelloWorldActorHandler for HelloWorldImpl { fn handlehelloworld( &mut self, ) -> HelloWorldActorHandlerResult { Ok(mustfuture::MustBoxFuture::new(async move { // return our results Ok("hello world!".tostring()) })) } } ```

Pretty straight forward. We implement a couple required traits, then our "Handler" trait that actually defines the logic of our actor. Then, we're ready to spawn it!

Spawning an actor

```rust impl HelloWorldImpl { // Use the GhostActorBuilder to construct the actor task. pub async fn spawn() -> GhostSender { // first we need a builder let builder = actor_builder::GhostActorBuilder::new();

    // now let's register an event channel with this actor.
    let sender = builder
        .channel_factory()
        .create_channel::<HelloWorldActor>()
        .await
        .unwrap();

    // actually spawn the actor driver task
    // providing our implementation
    tokio::task::spawn(builder.spawn(HelloWorldImpl));

    // return the sender that controls the actor
    sender
}

} ```

Note how we actually get access to the cheaply-clonable "Sender" before we have to construct our actor "Handler" item. This means you can create channels that will be able to message the actor, and include those senders in your handler struct. More on this later.

The Complete Hello World Example

```rust // Most of the GhostActor magic happens in this macro. // Sender and Handler traits will be generated here. ghostactor! { pub actor HelloWorldActor { fn helloworld() -> String; } }

// We need a struct to implement our handler upon. struct HelloWorldImpl;

// All handlers must implement GhostControlHandler. // This provides a default no-op handleghostactor_shutdown impl. impl GhostControlHandler for HelloWorldImpl {}

// Implement GhostHandler for your specific GhostEvent type. // Don't worry, the compiler will let you know if you forget this : ) impl GhostHandler for HelloWorldImpl {}

// Now implement your actual handler - // auto generated by the ghost_event! macro. impl HelloWorldActorHandler for HelloWorldImpl { fn handlehelloworld( &mut self, ) -> HelloWorldActorHandlerResult { Ok(mustfuture::MustBoxFuture::new(async move { // return our results Ok("hello world!".tostring()) })) } }

impl HelloWorldImpl { // Use the GhostActorBuilder to construct the actor task. pub async fn spawn() -> GhostSender { // first we need a builder let builder = actor_builder::GhostActorBuilder::new();

    // now let's register an event channel with this actor.
    let sender = builder
        .channel_factory()
        .create_channel::<HelloWorldActor>()
        .await
        .unwrap();

    // actually spawn the actor driver task
    // providing our implementation
    tokio::task::spawn(builder.spawn(HelloWorldImpl));

    // return the sender that controls the actor
    sender
}

}

[tokio::main]

async fn main() { // spawn our actor, getting the actor sender. let sender = HelloWorldImpl::spawn().await;

// we can make async calls on the sender
assert_eq!("hello world!", &sender.hello_world().await.unwrap());

} ```