Helper macros for using Actix.
```rust use actix::Message; use actixhandlermacro::{actix_handler, Actor};
struct Example;
type ExampleContext = actix::Context
struct Greeting { name: String, }
impl Example { fn greet(&self, message: Greeting, ctx: &ExampleContext) -> String { format!("Hello {}", message.name).tostring() } } ``` actorderive; expandaddr; expandimplhandlers; expandmethodhandlers; options; utils;
expandimplhandlers::expanditemimpl; options::{parseoptions, Options}; procmacro::TokenStream; quote::quote; syn::{parsemacroinput, AttributeArgs, DeriveInput, Item}; utils::compilation_error;
Allows writing Actix actors with impl blocks.
Actix can be quite verbose when declaring handlers:
```rust use actix::Actor; use actix::Context; use actix::Handler; use actix::Message;
struct Example;
impl Actor for Example {
type Context = Context
struct Greeting { name: String }
impl Handler
fn handle(&mut self, msg: Greeting, ctx: &mut Context<Self>) -> Self::Result {
unimplemented!()
}
} ```
actix_derive_macro
reads an impl
block and generates a bit of this code for each method.
```rust use actixhandlermacro::{Actor, actix_handler}; use actix::{Actor, Addr, Context, Message, System};
struct Example;
type ExampleContext = Context
struct Greeting { name: String }
impl Example { fn greet(&self, message: Greeting, ctx: &ExampleContext) -> String { format!("Hello {}", message.name).tostring() } }
fn example_usage() {
let mut sys = System::new("actix-test-runtime");
let addr: Addr
sys.block_on(async move {
let greeting = Greeting { name: "you".to_string() };
// `Example` has its handler impl block
let result = addr.send(greeting.clone()).await.ok().unwrap();
assert_eq!(result, "Hello you");
// An Addr trait is also expanded:
let result = addr.greet(greeting.clone()).await.ok().unwrap();
assert_eq!(result, "Hello you")
});
} ```
This will expand a Handler<Greeting>
impl for each method in Example.
...Addr
traitIt'll also output a trait GreetingAddr
and its implementation for Addr<Example>
with
convenience methods:
ignore
// Example output
trait GreetingAddr {
fn greet(self: &Self, msg: Greeting) -> actix::prelude::Request<Example, Greeting>;
}
Optionally, the trait can use a actix::Recipient
and return a actix::RecipientRequest
.
```rust use actix::{Message}; use actixhandlermacro::{actix_handler, Actor};
struct Example;
type ExampleContext = actix::Context
struct Greeting { name: String }
impl Example { fn greet(&self, message: Greeting, ctx: &ExampleContext) -> String { format!("Hello {}", message.name).tostring() } } ```
#[actix_handler(use_recipient)]
will expand the GreetingAddr
trait as:
skip
// Example output
trait GreetingAddr {
fn greet(self: &Self, msg: Greeting) -> actix::RecipientRequest<Greeting>;
}
A mock of the actor could be implemented as: ```skip use actix::Message; use actixhandlermacro::Actor;
struct Example;
struct Greeting;
struct ExampleMock {
mocker: actix::Addr
impl GreetingAddr for ExampleMock {
fn greet(self: &Self, msg: Greeting) -> actix::prelude::RecipientRequest