Skip to content

Commit 7500346

Browse files
committed
Added main languages documentation
1 parent c3f0b81 commit 7500346

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

42 files changed

+2011
-42
lines changed

docs/sdks/dart/actor_invocation.md

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
# Actor Invocation
2+
3+
TODO
4+
5+
[Next: Workflows](workflows.md)
6+
7+
[Previous: Actors](actors.md)

docs/sdks/dart/actors.md

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
# Dart 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+
TODO
20+
21+
## Stateful Actors
22+
23+
TODO
24+
25+
26+
## Considerations about Spawn actors
27+
28+
Another important feature of Spawn Actors is that the lifecycle of each Actor is managed by the platform itself.
29+
This means that an Actor will exist when it is invoked and that it will be deactivated after an idle time in its execution.
30+
This pattern is known as [Virtual Actors](#virtual-actors) but Spawn's implementation differs from some other known
31+
frameworks like [Orleans](https://www.microsoft.com/en-us/research/project/orleans-virtual-actors/) or
32+
[Dapr](https://docs.dapr.io/developing-applications/building-blocks/actors/actors-overview/)
33+
by defining a specific behavior depending on the type of Actor (named, unnamed, pooled, and etc...).
34+
35+
For example, named actors are instantiated the first time as soon as the host application registers them with the Spawn proxy.
36+
Whereas unnamed and pooled actors are instantiated the first time only when they receive their first invocation call.
37+
38+
[Next: Actor Invocation](actor_invocation.md)
39+
40+
[Previous: Getting Started](getting_started.md)

docs/sdks/dart/getting_started.md

+6-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,7 @@
11
# Dart Getting Started
2-
TODO
2+
3+
TODO
4+
5+
[Next: Actors](actors.md)
6+
7+
[Previous: SDKs](../../sdks.md)

docs/sdks/dart/projections.md

+6-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,7 @@
11
# Dart Projections
2-
TODO
2+
3+
TODO
4+
5+
[Next: SDKS](../../sdks.md)
6+
7+
[Previous: Workflows](workflows.md)

docs/sdks/dart/stateful.md

-2
This file was deleted.

docs/sdks/dart/stateless.md

-2
This file was deleted.

docs/sdks/dart/workflows.md

+25-1
Original file line numberDiff line numberDiff line change
@@ -1 +1,25 @@
1-
# Dart Workflows
1+
# Dart Workflows
2+
3+
Spawn has several mechanisms to facilitate integration between your actors or your application with the outside world. Below are some types of integration that Spawn provides:
4+
5+
## Broadcast
6+
7+
TODO
8+
9+
## Side Effects
10+
11+
TODO
12+
13+
## Forward
14+
15+
TODO
16+
17+
## Pipe
18+
19+
TODO
20+
21+
Forwards and pipes do not have an upper thread limit other than the request timeout.
22+
23+
[Next: Projections](projections.md)
24+
25+
[Previous: Actor Invocation](actor_invocation.md)

docs/sdks/elixir/actor_invocation.md

+70
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
# Actor Invocation
2+
3+
To invoke Actors, use:
4+
5+
```elixir
6+
iex> SpawnSdk.invoke("joe", system: "spawn-system", action: "Sum", payload: %Io.Eigr.Spawn.Example.MyBusinessMessage{value: 1})
7+
{:ok, %Io.Eigr.Spawn.Example.MyBusinessMessage{value: 12}}
8+
```
9+
10+
You can invoke actor default functions like "get" to get its current state
11+
12+
```elixir
13+
SpawnSdk.invoke("joe", system: "spawn-system", action: "get")
14+
```
15+
16+
Spawning Actors:
17+
18+
```elixir
19+
iex> SpawnSdk.spawn_actor("robert", system: "spawn-system", actor: "unnamed_actor")
20+
:ok
21+
```
22+
23+
You can also create Actors so that they are initialized from a certain revision number, that is, initialize actors from a specific point in time.
24+
25+
```elixir
26+
iex> SpawnSdk.spawn_actor("robert", system: "spawn-system", actor: "unnamed_actor", revision: 2)
27+
:ok
28+
```
29+
30+
In the above case the actor will be initialized with its state restored from the state as it was in revision 2 of its previous lifetime.
31+
32+
Invoke Spawned Actors:
33+
34+
```elixir
35+
iex> SpawnSdk.invoke("robert", system: "spawn-system", action: "sum", payload: %Io.Eigr.Spawn.Example.MyBusinessMessage{value: 1})
36+
{:ok, %Io.Eigr.Spawn.Example.MyBusinessMessage{value: 16}}
37+
```
38+
39+
Invoke Actors in a lazy way without having to spawn them before:
40+
41+
```elixir
42+
iex> SpawnSdk.invoke("robert_lazy", ref: SpawnSdkExample.Actors.UnnamedActor, system: "spawn-system", action: "sum", payload: %Io.Eigr.Spawn.Example.MyBusinessMessage{value: 1})
43+
{:ok, %Io.Eigr.Spawn.Example.MyBusinessMessage{value: 1}}
44+
```
45+
46+
Invoke Actors with a delay set in milliseconds:
47+
48+
```elixir
49+
iex> SpawnSdk.invoke("joe", system: "spawn-system", action: "ping", delay: 5_000)
50+
{:ok, :async}
51+
```
52+
53+
Invoke Actors scheduled to a specific DateTime:
54+
55+
```elixir
56+
iex> SpawnSdk.invoke("joe", system: "spawn-system", action: "ping", scheduled_to: ~U[2023-01-01 00:32:00.145Z])
57+
{:ok, :async}
58+
```
59+
60+
Invoke Pooled Actors:
61+
62+
```elixir
63+
iex> SpawnSdk.invoke("pooled_actor", system: "spawn-system", action: "ping", pooled: true)
64+
{:ok, nil}
65+
```
66+
67+
68+
[Next: Workflows](workflows.md)
69+
70+
[Previous: Actors](actors.md)

docs/sdks/elixir/actors.md

+120
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,120 @@
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

Comments
 (0)