Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/mikea/tractor

Actor System for Golang
https://github.com/mikea/tractor

actor-framework actor-model golang golang-library

Last synced: about 1 month ago
JSON representation

Actor System for Golang

Awesome Lists containing this project

README

        

![build](https://github.com/mikea/tractor/workflows/Go/badge.svg)

# Tractor

*[WORK IN PROGRESS]*

Actor System for Golang.

The goal of the project at this stage is to design an idiomatic go api for actor programming model together with common
services provided by an actor system.

## Documentation

### Defining Actors

#### Setup Handler

To define an actor the setup handler needs to be provided:

```go
type SetupHandler func(ctx ActorContext) MessageHandler
```

This callback is executed by the system every time new instance of the actor is created. The job of the callback is to
initialize all resources, send all initial requests, setup and return the message handler responsible for processing
actor mailbox messages.

#### Message Handler

Message handler callback processes the incoming message and returns a new handler to be used after the message.

```go
type MessageHandler func(message interface{}) MessageHandler
```

Two special values are recognized:

- `nil` - signals to use the *same* handler that was used to process the message
- `Stopped()` - signals to the system to terminate the current actor.

#### Setup Parameters

To parametrize an actor create a closure binding initialization parameters:

```go
func Countdown(start int) SetupHandler {
return func(ctx ActorContext) MessageHandler {
count := start
return func(msg interface{}) MessageHandler {
count = count - 1
if count == 0 {
return Stopped()
}
return nil
}
}
}
```

### Spawning Actors

Any actor can spawn a child by invoking `ctx.Spawn()` method with a setup handler:

```go
ctx.Spawn(Countdown(10))
```

### Actor Lifecycle

#### Signals

Receive messages related to the actor lifecycle by enabling signal delivery in the context:

```go
func(ctx ActorContext) MessageHandler {
ctx.DeliverSignals(true)
return func(msg interface{}) MessageHandler {
switch msg.(type) {
case PostInitSignal:
// first message delivered after the initialization
case PreStopSignal:
// delivered before terminating children
case PostStopSignal:
// delivered after terminating all children
}
return nil
}
}
```

#### Watching Actors

Any actor can be notified when any other actor is terminated:

```go
func(ctx ActorContext) MessageHandler {
ctx.Watch(ctx.Spawn(child))
return func(msg interface{}) MessageHandler {
switch msg.(type) {
case Terminated:
// child was terminated
}
return nil
}
}
```

or with a custom message:

```go
ctx.WatchWith(ctx.Spawn(child), "childTerminated")
```

### Actor Communication

#### Tell

Sending messages to a different actor is potentially a difficult enterprise and might involve other actor creation. Thus
current actor context is required to send messsages:

```go
child := ctx.Spawn(setupChild)
child.Tell(ctx, "ping")
```

#### Sender

During the event processing the sender of the message is available:

```
return func(msg interface{}) MessageHandler {
if msg == "ping" {
ctx.Sender().Tell(ctx, "pong")
}
return nil
}
```

#### Ask

Asking an actor means sending it a message and expecting a reply back.
`Ask()` method in context provides a bridge between go channels and actor communcation, enabling you to write blocking
code:

```go
reply := <-ctx.Ask(ref, "ping")
```

### Actor System

#### Starting System

To start the system you neeed to provide setup handler for the root actor:

```go
system := tractor.Start(Root())
```

#### Shutting Down

The system shuts down when root actor stops. You can wait for actor system to finish:

```go
system.Wait()
```

#### Communicating With System

Communication with the running system is done through the root actor reference and system context:

```go
system.Root().Tell(system.Context(), "request")
```

### Patterns

#### Typed Reference

Dealing with untyped messages might be challenges to scale. A simple typed wrappers can be created that will define
public api:

```go
type getAndIncrement struct{ }

func Counter() SetupHandler {
return func(ctx ActorContext) MessageHandler {
count := 0
return func(m interface{}) MessageHandler {
switch m.(type) {
case getAndIncrement:
ctx.Sender().Tell(ctx, count)
count++
}
return nil
}
}
}

type CounterRef struct {
Ref ActorRef
}

func (ref CounterRef) GetAndIncrement(ctx ActorContext) chan interface{} {
return ctx.Ask(ref.Ref, getAndIncrement{})
}
```

## Development

Use nix to set up development environment:

```bash
nix-shell
go test ./...
```