Technologies and design patterns of Erlang/OTP have been proven over the years. Now in Golang.
Up to x5 times faster than original Erlang/OTP in terms of network messaging.
The easiest way to create an OTP-designed application in Golang.
https://ergo.services
Purpose
The goal of this project is to leverage Erlang/OTP experience with Golang performance. Ergo Framework implements DIST protocol, ETF data format and OTP design patterns gen.Server
, gen.Supervisor
, gen.Application
which makes you able to create distributed, high performance and reliable microservice solutions having native integration with Erlang infrastructure
Cloud
You can join your services made with Ergo Framework into a single cluster with transparent networking using our Cloud Overlay Network where they can connect to each other smoothly, no matter where they run - AWS, Azure or GCP, or anywhere else. All these connections are secured with end-to-end encryption. Read more in this article https://blog.ergo.services/cloud-overlay-network-3a133d47efe5.
Features
- Support Erlang 25 (including Alias and Remote Spawn features)
- Spawn Erlang-like processes
- Register/unregister processes with simple atom
gen.Server
behavior support (with atomic state)
gen.Supervisor
behavior support with all known restart strategies support
- One For One
- One For All
- Rest For One
- Simple One For One
gen.Application
behavior support with all known starting types support
- Permanent
- Temporary
- Transient
gen.Stage
behavior support (originated from Elixir's GenStage). This is abstraction built on top of gen.Server
to provide a simple way to create a distributed Producer/Consumer architecture, while automatically managing the concept of backpressure. This implementation is fully compatible with Elixir's GenStage. Example is here examples/genstage or just run go run ./examples/genstage
to see it in action
gen.Saga
behavior support. It implements Saga design pattern - a sequence of transactions that updates each service state and publishes the result (or cancels the transaction or triggers the next transaction step). gen.Saga
also provides a feature of interim results (can be used as transaction progress or as a part of pipeline processing), time deadline (to limit transaction lifespan), two-phase commit (to make distributed transaction atomic). Here is example examples/gensaga.
gen.Raft
behavior support. It's improved implementation of Raft consensus algorithm. The key improvement is using quorum under the hood to manage the leader election process and make the Raft cluster more reliable. This implementation supports quorums of 3, 5, 7, 9, or 11 quorum members. Here is an example of this feature examples/genraft.
- Connect to (accept connection from) any Erlang/Elixir node within a cluster
- Making sync request
ServerProcess.Call
, async - ServerProcess.Cast
or Process.Send
in fashion of gen_server:call
, gen_server:cast
, erlang:send
accordingly
- Monitor processes/nodes, local/remote
- Link processes local/remote
- RPC callbacks support
- embedded EPMD (in order to get rid of erlang' dependencies)
- Unmarshalling terms into the struct using
etf.TermIntoStruct
, etf.TermProplistIntoStruct
or to the string using etf.TermToString
- Custom marshaling/unmarshaling via
Marshal
and Unmarshal
interfaces
- Encryption (TLS 1.3) support (including autogenerating self-signed certificates)
- Compression support (with customization of compression level and threshold). It can be configured for the node or a particular process.
- Proxy support with end-to-end encryption, includeing compression/fragmentation/linking/monitoring features.
- Tested and confirmed support Windows, Darwin (MacOS), Linux, FreeBSD.
- Zero dependencies. All features are implemented using the standard Golang library.
Requirements
Versioning
Golang introduced v2 rule a while ago to solve complicated dependency issues. We found this solution very controversial and there is still a lot of discussion around it. So, we decided to keep the old way for the versioning, but have to use the git tag with v1 as a major version (due to "v2 rule" restrictions). Since now we use git tag pattern 1.999.XYZ where X - major number, Y - minor, Z - patch version.
Changelog
Here are the changes of latest release. For more details see the ChangeLog
v2.2.1 2023-02-01 [tag version v1.999.221]
- Now you can join your services made with Ergo Framework into a single cluster with transparent networking using our Cloud Overlay Network where they can connect to each other smoothly, no matter where they run - AWS, Azure or GCP, or anywhere else. All these connections are secured with end-to-end encryption. Read more in this article https://blog.ergo.services/cloud-overlay-network-3a133d47efe5. Here is an example of this feature in action examples/cloud
examples
moved to https://github.com/ergo-services/examples
- Added support Erlang OTP/25
- Improved handling
nil
values for the registered types using etf.RegisterType(...)
- Improved self-signed certificate generation
- Introduced
ergo.debug
option that enables extended debug information for lib.Log(...)
/lib.Warning(...)
- Fixed
gen.TCP
and gen.UDP
(missing callbacks)
- Fixed ETF registering type with
etf.Pid
, etf.Alias
or etf.Ref
value types
- Fixed Cloud client
- Fixed #117 (incorrect hanshake process finalization)
- Fixed #139 (panic of the gen.Stage partition dispatcher)
v2.2.0 2022-10-18 [tag version v1.999.220]
- Introduced
gen.Web
behavior. It implements Web API Gateway pattern is also sometimes known as the "Backend For Frontend" (BFF). See example examples/genweb
- Introduced
gen.TCP
behavior - socket acceptor pool for TCP protocols. It provides everything you need to accept TCP connections and process packets with a small code base and low latency. Here is simple example examples/gentcp
- Introduced
gen.UDP
- the same as gen.TCP
, but for UDP protocols. Example is here examples/genudp
- Introduced Events. This is a simple pub/sub feature within a node - any
gen.Process
can become a producer by registering a new event gen.Event
using method gen.Process.RegisterEvent
, while the others can subscribe to these events using gen.Process.MonitorEvent
. Subscriber process will also receive gen.MessageEventDown
if a producer process went down (terminated). This feature behaves in a monitor manner but only works within a node. You may also want to subscribe to a system event - node.EventNetwork
to receive event notification on connect/disconnect any peers. Here is simple example of this feature examples/events
- Introduced Cloud Client - allows connecting to the cloud platform https://ergo.sevices. You may want to register your email there, and we will inform you about the platform launch day
- Introduced type registration for the ETF encoding/decoding. This feature allows you to get rid of manually decoding with
etf.TermIntoStruct
for the receiving messages. Register your type using etf.RegisterType(...)
, and you will be receiving messages in a native type
- Predefined set of errors has moved to the
lib
package
- Updated
gen.ServerBehavior.HandleDirect
method (got extra argument etf.Ref
to distinguish the requests). This change allows you to handle these requests asynchronously using method gen.ServerProcess.Reply(...)
- Updated
node.Options
. Now it has field Listeners
(type node.Listener
). It allows you to start any number of listeners with custom options - Port
, TLS
settings, or custom Handshake
/Proto
interfaces
- Fixed build on 32-bit arch
- Fixed freezing on ARM arch #102
- Fixed problem with encoding negative int8
- Fixed #103 (there was an issue on interop with Elixir's GenStage)
- Fixed node stuck on start if it uses the name which is already taken in EPMD
- Fixed incorrect
gen.ProcessOptions.Context
handling
Benchmarks
Here is simple EndToEnd test demonstrates performance of messaging subsystem
Hardware: workstation with AMD Ryzen Threadripper 3970X (64) @ 3.700GHz
❯❯❯❯ go test -bench=NodeParallel -run=XXX -benchtime=10s
goos: linux
goarch: amd64
pkg: github.com/ergo-services/ergo/tests
cpu: AMD Ryzen Threadripper 3970X 32-Core Processor
BenchmarkNodeParallel-64 4738918 2532 ns/op
BenchmarkNodeParallelSingleNode-64 100000000 429.8 ns/op
PASS
ok github.com/ergo-services/ergo/tests 29.596s
these numbers show almost 500.000 sync requests per second for the network messaging via localhost and 10.000.000 sync requests per second for the local messaging (within a node).
Compression
This benchmark shows the performance of compression for sending 1MB message between two nodes (via a network).
❯❯❯❯ go test -bench=NodeCompression -run=XXX -benchtime=10s
goos: linux
goarch: amd64
pkg: github.com/ergo-services/ergo/tests
cpu: AMD Ryzen Threadripper 3970X 32-Core Processor
BenchmarkNodeCompressionDisabled1MBempty-64 2400 4957483 ns/op
BenchmarkNodeCompressionEnabled1MBempty-64 5769 2088051 ns/op
BenchmarkNodeCompressionEnabled1MBstring-64 5202 2077099 ns/op
PASS
ok github.com/ergo-services/ergo/tests 56.708s
It demonstrates more than 2 times improvement.
Proxy
This benchmark demonstrates how proxy feature and e2e encryption impact a messaging performance.
❯❯❯❯ go test -bench=NodeProxy -run=XXX -benchtime=10s
goos: linux
goarch: amd64
pkg: github.com/ergo-services/ergo/tests
cpu: AMD Ryzen Threadripper 3970X 32-Core Processor
BenchmarkNodeProxy_NodeA_to_NodeC_direct_Message_1KB-64 1908477 6337 ns/op
BenchmarkNodeProxy_NodeA_to_NodeC_via_NodeB_Message_1KB-64 1700984 7062 ns/op
BenchmarkNodeProxy_NodeA_to_NodeC_via_NodeB_Message_1KB_Encrypted-64 1271125 9410 ns/op
PASS
ok github.com/ergo-services/ergo/tests 45.649s
Ergo Framework vs original Erlang/OTP
Hardware: laptop with Intel(R) Core(TM) i5-8265U (4 cores. 8 with HT)
sources of these benchmarks are here
EPMD
Ergo Framework has embedded EPMD implementation in order to run your node without external epmd process needs. By default, it works as a client with erlang' epmd daemon or others ergo's nodes either.
The one thing that makes embedded EPMD different is the behavior of handling connection hangs - if ergo' node is running as an EPMD client and lost connection, it tries either to run its own embedded EPMD service or to restore the lost connection.
Examples
Code below is a simple implementation of gen.Server pattern examples/genserver
package main
import (
"fmt"
"time"
"github.com/ergo-services/ergo/etf"
"github.com/ergo-services/ergo/gen"
)
type simple struct {
gen.Server
}
func (s *simple) HandleInfo(process *gen.ServerProcess, message etf.Term) gen.ServerStatus {
value := message.(int)
fmt.Printf("HandleInfo: %#v \n", message)
if value > 104 {
return gen.ServerStatusStop
}
// sending message with delay 1 second
fmt.Println("increase this value by 1 and send it to itself again")
process.SendAfter(process.Self(), value+1, time.Second)
return gen.ServerStatusOK
}
here is output of this code
$ go run ./examples/simple
HandleInfo: 100
HandleInfo: 101
HandleInfo: 102
HandleInfo: 103
HandleInfo: 104
HandleInfo: 105
exited
See https://github.com/ergo-services/examples for more details
Elixir Phoenix Users
Users of the Elixir Phoenix framework might encounter timeouts when trying to connect a Phoenix node
to an ergo node. The reason is that, in addition to global_name_server and net_kernel,
Phoenix attempts to broadcast messages to the pg2 PubSub handler
To work with Phoenix nodes, you must create and register a dedicated pg2 GenServer, and
spawn it inside your node. The spawning process must have "pg2" as a process name:
type Pg2GenServer struct {
gen.Server
}
func main() {
// ...
pg2 := &Pg2GenServer{}
node1, _ := ergo.StartNode("node1@localhost", "cookies", node.Options{})
process, _ := node1.Spawn("pg2", gen.ProcessOptions{}, pg2, nil)
// ...
}
Development and debugging
There are options already defined that you might want to use
-ergo.trace
- enable debug info (logging via lib.Log(...)
)
-ergo.debug
- enable extended debug info (logging via lib.Log(...)
and lib.Warning(...)
)
-ergo.norecover
- disable panic catching
-ergo.warning
- enable/disable warnings (logging via lib.Warning(...)
. Default: enable)
To enable Golang profiler just add --tags debug
in your go run
or go build
like this:
go run --tags debug ./examples/genserver/demoGenServer.go
Now golang' profiler is available at http://localhost:9009/debug/pprof
To check test coverage:
go test -coverprofile=cover.out ./...
go tool cover -html=cover.out -o coverage.html
To run tests with cleaned test cache:
go vet
go clean -testcache
go test -v ./...
To run benchmarks:
go test -bench=Node -run=X -benchmem
Companies are using Ergo Framework
is your company using Ergo? add your company logo/name here
Commercial support
please, visit https://ergo.services for more information