This is a simple implementation of Raft built with Go and gRPC, aimed to help me learn through implementation. A GET and PUT interface is exposed externally.
Most of Raft functionality is implemented and tested, but small parts may be missing. Many improvements can be made, such as changing the backend storage (currently the key-value store is in memory), or having a more sophisticated way for clients to find which replica to contact. The code could also be cleaned up and more tests could be written (concurrency tests could be more reliable, currently I am not completely sure they are correct). Given time, I would try to accomplish these improvements.
A sample config is given (config.txt) using localhost. Each line in the config is a new replica, and each line has a internal address and an external address, respectively.
The server takes in two arguments, config path and replica id. Initialization is complete when all servers are started.
To start three localhost servers, use the config.txt provided and run ./bin/server
in three different terminals.
$ ./bin/server -config=config.txt -id=0
$ ./bin/server -config=config.txt -id=1
$ ./bin/server -config=config.txt -id=2
The client can be used by creating a client object using client.NewClient(...)
. The client object requires two arguments, config path and preferred replica id. The preferred replica id is the replica that the client will contact first on all requests. This is usually the closest replica.
An example usage is given here.
$ go build -o bin/client cmd/client/example_client.go
$ ./bin/client -config=config.txt -id=0