/darkrmq

Lightweight library that handles RabbitMQ auto-reconnect and publishing retry routine for you.

Primary LanguageGoMIT LicenseMIT

PkgGoDev Build Status Go Report Card

Rabbitmq Failover Routine

Lightweight library that handles RabbitMQ auto-reconnect and publishing retry routine for you. The library is designed to save the developer from the headache when working with RabbitMQ.

darkmq solves your RabbitMQ reconnection problems:

Stop to do wrappers, do features!

Install

go get github.com/sagleft/darkrmq

Adding as dependency by "go dep"

$ dep ensure -add github.com/sagleft/darkrmq

Usage

Consuming

You need to implement Consumer and register it with StartConsumer or with StartMultipleConsumers. When connection is established (at first time or after reconnect) Declare method is called. It can be used to declare required RabbitMQ entities (consumer example).

Usage example:

// Consumer declares your own RabbitMQ consumer implementing darkmq.Consumer interface.
type Consumer struct {}
func (c *Consumer) Declare(ctx context.Context, ch *amqp.Channel) error {}
func (c *Consumer) Consume(ctx context.Context, ch *amqp.Channel) error {}

url := "amqp://guest:guest@127.0.0.1:5672/"

conn := darkmq.NewConnector(darkmq.Config{
    // How long to wait between reconnect
    Wait: 2 * time.Second,
})

ctx := context.Background()

go func() {
    err := conn.Dial(ctx, url)
    if err != nil {
    	log.Println(err)
    }
}()

consumer := &Consumer{}
go func() {
    err := conn.StartConsumer(ctx, consumer)
    if err != nil {
        log.Println(err)
    }
}()

Full example demonstrates messages consuming

Publishing

For publishing FireForgetPublisher and EnsurePublisher implemented. Both of them can be wrapped with RetryPublisher to repeat publishing on errors and mitigate short-term network problems.

Usage example:

ctx := context.Background()

url := "amqp://guest:guest@127.0.0.1:5672/"

conn := darkmq.NewConnector(darkmq.Config{
    // How long wait between reconnect
    Wait: 2 * time.Second,
})

pool := darkmq.NewPool(conn)
ensurePub := darkmq.NewEnsurePublisher(pool)
pub := darkmq.NewRetryPublisher(
    ensurePub,
    darkmq.PublishMaxAttemptsSetup(16),
    darkmq.PublishDelaySetup(darkmq.LinearDelay(10*time.Millisecond)),
)

go conn.Dial(ctx, url)

err := pub.Publish(ctx, "myexch", "myqueue", amqp.Publishing{Body: []byte("message")})
if err != nil {
    log.Println("publish error:", err)
}

Full example demonstrates messages publishing

Contributing

Pull requests are very much welcomed. Create your pull request, make sure a test or example is included that covers your change and your commits represent coherent changes that include a reason for the change.

To run the integration tests, make sure you have RabbitMQ running on any host (e.g with docker run --net=host -it --rm rabbitmq), then export the environment variable AMQP_URL=amqp://host/ and run go test -tags integration. As example:

AMQP_URL=amqp://guest:guest@127.0.0.1:5672/ go test -v -race -cpu=1,2 -tags integration -timeout 5s

Use golangci-lint to check code with linters:

golangci-lint run ./...

TravisCI will also run the integration tests and golangci-lint.