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
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
|
package app
import (
"context"
"fmt"
"log/slog"
"net"
"os"
"os/signal"
"sync"
"syscall"
"time"
"github.com/ChausseBenjamin/rafta/internal/db"
"github.com/ChausseBenjamin/rafta/internal/logging"
"github.com/ChausseBenjamin/rafta/internal/pb"
"github.com/urfave/cli/v3"
"google.golang.org/grpc"
)
func action(ctx context.Context, cmd *cli.Command) error {
err := logging.Setup(
cmd.String(FlagLogLevel),
cmd.String(FlagLogFormat),
cmd.String(FlagLogOutput),
)
if err != nil {
slog.WarnContext(ctx, "Error(s) occurred during logger initialization", logging.ErrKey, err)
}
slog.InfoContext(ctx, "Starting rafta server")
errAppChan := make(chan error)
readyChan := make(chan bool)
shutdownDone := make(chan struct{}) // Signals when graceful shutdown is complete
var once sync.Once
gracefulShutdown := func() {}
brutalShutdown := func() {}
application := func() {
server, store, err := initApp(ctx, cmd)
if err != nil {
errAppChan <- err
return
}
gracefulShutdown = func() {
once.Do(func() { // Ensure brutal shutdown isn't triggered later
server.GracefulStop()
store.Close()
slog.InfoContext(ctx, "Application shutdown")
close(shutdownDone) // Signal that graceful shutdown is complete
})
}
brutalShutdown = func() {
once.Do(func() { // Ensure graceful shutdown isn't re-executed
slog.WarnContext(ctx, "Graceful shutdown delay exceeded, shutting down NOW!")
server.Stop()
store.Close()
})
}
port := fmt.Sprintf(":%d", cmd.Int(FlagListenPort))
listener, err := net.Listen("tcp", port)
if err != nil {
errAppChan <- err
return
}
slog.InfoContext(ctx, "Server listening", "port", cmd.Int(FlagListenPort))
readyChan <- true
if err := server.Serve(listener); err != nil {
errAppChan <- err
}
}
go application()
stopChan := waitForTermChan()
running := true
for running {
select {
case errApp := <-errAppChan:
if errApp != nil {
slog.ErrorContext(ctx, "Application error", logging.ErrKey, errApp)
}
return errApp
case <-stopChan:
slog.InfoContext(ctx, "Shutdown requested")
go gracefulShutdown()
select {
case <-shutdownDone: // If graceful shutdown completes in time, exit normally
case <-time.After(cmd.Duration(FlagGraceTimeout)): // Timeout exceeded
brutalShutdown()
}
running = false
}
}
return nil
}
func waitForTermChan() chan os.Signal {
stopChan := make(chan os.Signal, 1)
signal.Notify(stopChan, syscall.SIGINT, syscall.SIGTERM)
return stopChan
}
func initApp(ctx context.Context, cmd *cli.Command) (*grpc.Server, *db.Store, error) {
store, err := db.Setup(ctx, cmd.String(FlagDBPath))
if err != nil {
slog.ErrorContext(ctx, "Unable to setup database", logging.ErrKey, err)
return nil, nil, err
}
server, err := pb.Setup(ctx, store)
if err != nil {
slog.ErrorContext(ctx, "Unable to setup gRPC server", logging.ErrKey, err)
return nil, nil, err
}
return server, store, nil
}
|