-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathnode_raft.go
94 lines (77 loc) · 2.55 KB
/
node_raft.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
package rafting
import (
"context"
"fmt"
"net"
"os"
"path/filepath"
transport "github.com/Jille/raft-grpc-transport"
adapters "github.com/danielgatis/go-logrus-adapters"
pb "github.com/danielgatis/go-rafting/protobuf"
"github.com/hashicorp/raft"
store "github.com/hashicorp/raft-boltdb/v2"
"google.golang.org/grpc"
"google.golang.org/grpc/credentials/insecure"
)
func initRaft(n *Node) error {
if err := os.MkdirAll(filepath.Join(n.dataDir, n.id), 0755); err != nil && !os.IsExist(err) {
return fmt.Errorf("os.MkdirAll(...): %w", err)
}
logStore, err := store.NewBoltStore(filepath.Join(n.dataDir, n.id, "log-store.db"))
if err != nil {
return fmt.Errorf("store.NewBoltStore(...): %w", err)
}
stableStore, err := store.NewBoltStore(filepath.Join(n.dataDir, n.id, "stable-store.db"))
if err != nil {
return fmt.Errorf("store.NewBoltStore(...): %w", err)
}
snapshotStore, err := raft.NewFileSnapshotStore(filepath.Join(n.dataDir, n.id), n.snapshotRetain, os.Stderr)
if err != nil {
return fmt.Errorf("raft.NewFileSnapshotStore(...): %w", err)
}
conf := raft.DefaultConfig()
conf.LocalID = raft.ServerID(n.id)
conf.Logger = adapters.NewHCLogAdapter(n.logger, "raft")
n.addr = fmt.Sprintf("%s:%d", "0.0.0.0", n.port)
n.grpcServer = grpc.NewServer()
n.raftTransport = transport.New(raft.ServerAddress(n.addr), []grpc.DialOption{grpc.WithTransportCredentials(insecure.NewCredentials())})
n.raftTransport.Register(n.grpcServer)
pb.RegisterRaftingServiceServer(n.grpcServer, newRaftingServerImpl(n))
n.raft, err = raft.NewRaft(conf, n.fsm, logStore, stableStore, snapshotStore, n.raftTransport.Transport())
if err != nil {
return fmt.Errorf(`raft.NewRaft(...): %w`, err)
}
return nil
}
func startRaft(ctx context.Context, n *Node) error {
go func() {
<-ctx.Done()
n.grpcServer.GracefulStop()
if err := n.raft.Snapshot().Error(); err != nil {
if err != raft.ErrNothingNewToSnapshot {
n.logger.Error(err)
}
}
if err := n.raft.Shutdown().Error(); err != nil {
n.logger.Error(err)
}
}()
if f := n.raft.BootstrapCluster(raft.Configuration{
Servers: []raft.Server{
{
ID: raft.ServerID(n.id),
Address: n.raftTransport.Transport().LocalAddr(),
},
},
}); f.Error() != nil && f.Error() != raft.ErrCantBootstrap {
return fmt.Errorf(`raft.BootstrapCluster(...): %w`, f.Error())
}
listen, err := net.Listen("tcp", n.addr)
if err != nil {
return fmt.Errorf(`net.Listen(...): %w`, err)
}
if err := n.grpcServer.Serve(listen); err != nil {
return fmt.Errorf(`grpcServer.Serve(...): %w`, err)
}
return nil
}