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

https://github.com/jarvanstack/evbus

CQRS eventbus golang
https://github.com/jarvanstack/evbus

Last synced: 6 months ago
JSON representation

CQRS eventbus golang

Awesome Lists containing this project

README

          

# evbus

Package evbus is the little and lightweight eventbus with async compatibility for GoLang.

## Installation

Make sure that Go is installed on your computer.
Type the following command in your terminal:

go get github.com/jarvanstack/evbus

After it the package is ready to use.

## Import package in your project

Add following line in your `*.go` file:

```go
import "github.com/jarvanstack/evbus"
```

If you unhappy to use long `evbus`, you can do something like this:

```go
import (
evbus "github.com/jarvanstack/evbus"
)
```

## Example

```go
func calculator(a int, b int) {
fmt.Printf("%d\n", a + b)
}

func main() {
bus := evbus.New();
bus.Subscribe("main:calculator", calculator);
bus.Publish("main:calculator", 20, 40);
bus.Unsubscribe("main:calculator", calculator);
}
```

## Implemented methods

* **New()**
* **Subscribe()**
* **SubscribeOnce()**
* **HasCallback()**
* **Unsubscribe()**
* **Publish()**
* **SubscribeAsync()**
* **SubscribeOnceAsync()**
* **WaitAsync()**

## New()

New returns new evbus with empty handlers.

```go
bus := evbus.New();
```

## Subscribe(topic string, fn interface{}) error

Subscribe to a topic. Returns error if `fn` is not a function.

```go
func Handler() { ... }
...
bus.Subscribe("topic:handler", Handler)
```

## SubscribeOnce(topic string, fn interface{}) error

Subscribe to a topic once. Handler will be removed after executing. Returns error if `fn` is not a function.

```go
func HelloWorld() { ... }
...
bus.SubscribeOnce("topic:handler", HelloWorld)
```

## Unsubscribe(topic string, fn interface{}) error

Remove callback defined for a topic. Returns error if there are no callbacks subscribed to the topic.

```go
bus.Unsubscribe("topic:handler", HelloWord);
```

## HasCallback(topic string) bool

Returns true if exists any callback subscribed to the topic.

## Publish(topic string, args ...interface{})

Publish executes callback defined for a topic. Any additional argument will be transferred to the callback.

```go
func Handler(str string) { ... }
...
bus.Subscribe("topic:handler", Handler)
...
bus.Publish("topic:handler", "Hello, World!");
```

## SubscribeAsync(topic string, fn interface{}, transactional bool)

Subscribe to a topic with an asynchronous callback. Returns error if `fn` is not a function.

```go
func slowCalculator(a, b int) {
time.Sleep(3 * time.Second)
fmt.Printf("%d\n", a + b)
}

bus := evbus.New()
bus.SubscribeAsync("main:slow_calculator", slowCalculator, false)

bus.Publish("main:slow_calculator", 20, 60)

fmt.Println("start: do some stuff while waiting for a result")
fmt.Println("end: do some stuff while waiting for a result")

bus.WaitAsync() // wait for all async callbacks to complete

fmt.Println("do some stuff after waiting for result")
```

Transactional determines whether subsequent callbacks for a topic are run serially (true) or concurrently(false)

## SubscribeOnceAsync(topic string, args ...interface{})

SubscribeOnceAsync works like SubscribeOnce except the callback to executed asynchronously

## WaitAsync()

WaitAsync waits for all async callbacks to complete.

## Cross Process Events

Works with two rpc services:

* a client service to listen to remotely published events from a server
* a server service to listen to client subscriptions

server.go

```go
var (
netAAddr = ":2050"
netAPath = "/_net_bus_A"
netBAddr = ":2055"
netBPath = "/_net_bus_B"
)

fmt.Printf("Start\n")

netA := NewNetworkBus(netAAddr, netAPath)
netA.Start()
netB := NewNetworkBus(netBAddr, netBPath)
netB.Start()

// netA 推送 netB 消费
netB.Subscribe("event1", func(a int) {
fmt.Printf("[netA 推送 netB 消费] %d \n", a)
}, netAAddr, netAPath)
netA.EventBus().Publish("event1", 10)

// netB 推送 netA 消费
netA.Subscribe("event2", func(a int) {
fmt.Printf("[netB 推送 netA 消费] %d \n", a)
}, netBAddr, netBPath)
netB.EventBus().Publish("event2", 10)

// netA 推送 netA 本地消费
netA.EventBus().Subscribe("event3", func(a int) {
fmt.Printf("[netA 推送 netA 本地消费] %d \n", a)
})
netA.EventBus().Publish("event3", 10)

// netA 推送 netA 本地查询
netA.EventBus().Subscribe("query1", func(result *int) {
*result = 10
return
})
var result int
netA.EventBus().Publish("query1", &result)
fmt.Printf("[netA 推送 netA 本地查询] %d \n", result)

// netA 推送 netB 查询 (panic)
// netB.Subscribe("query2", func(result *int) {
// *result = 10
// return
// }, netAAddr, netAPath)
// var result2 int
// netA.EventBus().Publish("query2", &result2)
// fmt.Printf("[netA 推送 netB 查询] %d \n", result2)

netA.Stop()
netB.Stop()
```

client.go

```go
func main() {
client := NewClient(":2015", "/_client_bus_", New())
client.Start()
client.Subscribe("main:calculator", calculator, ":2010", "/_server_bus_")
// ...
client.Stop()
}
```

## Notes

Documentation is available here: [godoc.org](https://godoc.org/github.com/jarvanstack/evbus).
Full information about code coverage is also available here: [evbus on gocover.io](http://gocover.io/github.com/jarvanstack/evbus).

## Support

If you do have a contribution for the package feel free to put up a Pull Request or open Issue.

## Special thanks to [contributors](https://github.com/jarvanstack/evbus/graphs/contributors)

* [Brian Downs](https://github.com/briandowns)
* [Dominik Schulz](https://github.com/gittex)
* [bennAH](https://github.com/bennAH)
* [John Noble] ()
* [Evan Borgstrom] ()