|
| 1 | +# Elixir Actors |
| 2 | + |
| 3 | +Spawn defines the following types of Actors: |
| 4 | + |
| 5 | +* **Named Actors**: Named actors are actors whose name is defined at compile time. They also behave slightly differently |
| 6 | +Then unnamed actors and pooled actors. Named actors when they are defined with the stateful parameter equal to True are |
| 7 | +immediately instantiated when they are registered at the beginning of the program, they can also only be referenced by |
| 8 | +the name given to them in their definition. |
| 9 | + |
| 10 | +* **Unnamed Actors**: Unlike named actors, unnamed actors are only created when they are named at runtime, that is, |
| 11 | +during program execution. Otherwise, they behave like named actors. |
| 12 | + |
| 13 | +* **Pooled Actors**: Pooled Actors, as the name suggests, are a collection of actors that are grouped under the same name |
| 14 | +assigned to them at compile time. Pooled actors are generally used when higher performance is needed and are also |
| 15 | +recommended for handling serverless loads. |
| 16 | + |
| 17 | +## Stateless Actors |
| 18 | + |
| 19 | +```elixir |
| 20 | +defmodule SpawnSdkExample.Actors.StatelessActor do |
| 21 | + use SpawnSdk.Actor, |
| 22 | + kind: :unnamed, |
| 23 | + stateful: false, # This is what defines this actor as stateless. |
| 24 | + state_type: Io.Eigr.Spawn.Example.MyState |
| 25 | + |
| 26 | +end |
| 27 | +``` |
| 28 | + |
| 29 | +## Stateful Actors |
| 30 | + |
| 31 | +Below are examples for stateful actors for both Named and Unnamed types. |
| 32 | + |
| 33 | +```elixir |
| 34 | +defmodule SpawnSdkExample.Actors.MyActor do |
| 35 | + use SpawnSdk.Actor, |
| 36 | + name: "jose", # Default is Full Qualified Module name a.k.a __MODULE__ |
| 37 | + kind: :named, # Default is already :named. Valid are :named | :unnamed |
| 38 | + stateful: true, # Default is already true |
| 39 | + state_type: Io.Eigr.Spawn.Example.MyState, # or :json if you don't care about protobuf types |
| 40 | + deactivate_timeout: 30_000, |
| 41 | + snapshot_timeout: 2_000 |
| 42 | + |
| 43 | + require Logger |
| 44 | + |
| 45 | + alias Io.Eigr.Spawn.Example.{MyState, MyBusinessMessage} |
| 46 | + |
| 47 | + # The callback could also be referenced to an existing function: |
| 48 | + # action "SomeAction", &some_defp_handler/0 |
| 49 | + # action "SomeAction", &SomeModule.handler/1 |
| 50 | + # action "SomeAction", &SomeModule.handler/2 |
| 51 | + |
| 52 | + init fn %Context{state: state} = ctx -> |
| 53 | + Logger.info("[joe] Received InitRequest. Context: #{inspect(ctx)}") |
| 54 | + |
| 55 | + Value.of() |
| 56 | + |> Value.state(state) |
| 57 | + end |
| 58 | + |
| 59 | + action "Sum", fn %Context{state: state} = ctx, %MyBusinessMessage{value: value} = data -> |
| 60 | + Logger.info("Received Request: #{inspect(data)}. Context: #{inspect(ctx)}") |
| 61 | + |
| 62 | + new_value = if is_nil(state), do: value, else: (state.value || 0) + value |
| 63 | + |
| 64 | + Value.of(%MyBusinessMessage{value: new_value}, %MyState{value: new_value}) |
| 65 | + end |
| 66 | +end |
| 67 | + |
| 68 | +``` |
| 69 | + |
| 70 | +We declare two actions that the Actor can do. An initialization action that will be called every time an Actor instance is created and an action that will be responsible for performing a simple sum. |
| 71 | + |
| 72 | +Note Keep in mind that any Action that has the names present in the list below will behave as an initialization Action and will be called when the Actor is started (if there is more than one Action with one of these names, only one will be called). |
| 73 | + |
| 74 | +Defaults inicialization Action names: "**init**", "**Init**", "**setup**", "**Setup**" |
| 75 | + |
| 76 | +### Stateful Unnamed Actors |
| 77 | + |
| 78 | +We can also create Unnamed Dynamic/Lazy actors, that is, despite having its unnamed behavior defined at compile time, a Lazy actor will only have a concrete instance when it is associated with an identifier/name at runtime. Below follows the same previous actor being defined as Unnamed. |
| 79 | + |
| 80 | +```elixir |
| 81 | +defmodule SpawnSdkExample.Actors.UnnamedActor do |
| 82 | + use SpawnSdk.Actor, |
| 83 | + name: "unnamed_actor", |
| 84 | + kind: :unnamed, |
| 85 | + state_type: Io.Eigr.Spawn.Example.MyState |
| 86 | + |
| 87 | + require Logger |
| 88 | + |
| 89 | + alias Io.Eigr.Spawn.Example.{MyState, MyBusinessMessage} |
| 90 | + |
| 91 | + action "Sum", fn %Context{state: state} = ctx, %MyBusinessMessage{value: value} = data -> |
| 92 | + Logger.info("Received Request: #{inspect(data)}. Context: #{inspect(ctx)}") |
| 93 | + |
| 94 | + new_value = if is_nil(state), do: value, else: (state.value || 0) + value |
| 95 | + |
| 96 | + Value.of(%MyBusinessMessage{value: new_value}, %MyState{value: new_value}) |
| 97 | + end |
| 98 | +end |
| 99 | +``` |
| 100 | + |
| 101 | +Notice that the only thing that has changed is the the kind of actor, in this case the kind is set to :unnamed. |
| 102 | + |
| 103 | +> **_NOTE:_** Can Elixir programmers think in terms of Named vs Unnamed actors as more or less known at startup vs dynamically supervised/registered? That is, defining your actors directly in the supervision tree or using a Dynamic Supervisor for that. |
| 104 | +
|
| 105 | + |
| 106 | +## Considerations about Spawn actors |
| 107 | + |
| 108 | +Another important feature of Spawn Actors is that the lifecycle of each Actor is managed by the platform itself. |
| 109 | +This means that an Actor will exist when it is invoked and that it will be deactivated after an idle time in its execution. |
| 110 | +This pattern is known as [Virtual Actors](#virtual-actors) but Spawn's implementation differs from some other known |
| 111 | +frameworks like [Orleans](https://www.microsoft.com/en-us/research/project/orleans-virtual-actors/) or |
| 112 | +[Dapr](https://docs.dapr.io/developing-applications/building-blocks/actors/actors-overview/) |
| 113 | +by defining a specific behavior depending on the type of Actor (named, unnamed, pooled, and etc...). |
| 114 | + |
| 115 | +For example, named actors are instantiated the first time as soon as the host application registers them with the Spawn proxy. |
| 116 | +Whereas unnamed and pooled actors are instantiated the first time only when they receive their first invocation call. |
| 117 | + |
| 118 | +[Next: Actor Invocation](actor_invocation.md) |
| 119 | + |
| 120 | +[Previous: Getting Started](getting_started.md) |
0 commit comments