blob: 32ead79ecbc9a9f9f1b2ff8dffc9ab0e10d7cd60 [file] [log] [blame]
Cosmos Nicolaoua429eff2014-11-19 14:02:34 -08001package rt_test
2
3import (
4 "bufio"
5 "fmt"
6 "io"
7 "os"
8 "os/signal"
9 "sync"
10 "syscall"
11
12 "veyron.io/veyron/veyron2/ipc"
13 "veyron.io/veyron/veyron2/rt"
14 "veyron.io/veyron/veyron2/vlog"
15
16 "veyron.io/veyron/veyron/lib/modules"
17 "veyron.io/veyron/veyron/lib/signals"
18 "veyron.io/veyron/veyron/profiles"
19)
20
21func init() {
22 modules.RegisterChild("simpleServerProgram", "", simpleServerProgram)
23 modules.RegisterChild("complexServerProgram", "", complexServerProgram)
24}
25
26type dummy struct{}
27
28func (*dummy) Echo(ipc.ServerContext) error { return nil }
29
30// makeServer sets up a simple dummy server.
31func makeServer() ipc.Server {
32 server, err := rt.R().NewServer()
33 if err != nil {
34 vlog.Fatalf("r.NewServer error: %s", err)
35 }
36 if _, err := server.Listen(profiles.LocalListenSpec); err != nil {
37 vlog.Fatalf("server.Listen error: %s", err)
38 }
39 if err := server.Serve("", &dummy{}, nil); err != nil {
40 vlog.Fatalf("server.Serve error: %s", err)
41 }
42 return server
43}
44
45// remoteCmdLoop listens on stdin and interprets commands sent over stdin (from
46// the parent process).
47func remoteCmdLoop(stdin io.Reader) func() {
48 done := make(chan struct{})
49 go func() {
Cosmos Nicolaoua429eff2014-11-19 14:02:34 -080050 scanner := bufio.NewScanner(stdin)
51 for scanner.Scan() {
52 switch scanner.Text() {
53 case "stop":
54 rt.R().AppCycle().Stop()
55 case "forcestop":
56 fmt.Println("straight exit")
57 rt.R().AppCycle().ForceStop()
58 case "close":
Bogdan Capritae06d7152014-11-22 09:32:05 -080059 close(done)
Cosmos Nicolaoua429eff2014-11-19 14:02:34 -080060 return
61 }
62 }
63 }()
64 return func() { <-done }
65}
66
67// complexServerProgram demonstrates the recommended way to write a more
68// complex server application (with several servers, a mix of interruptible
69// and blocking cleanup, and parallel and sequential cleanup execution).
70// For a more typical server, see simpleServerProgram.
71func complexServerProgram(stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error {
72 // Initialize the runtime. This is boilerplate.
73 r := rt.Init()
74
75 // This is part of the test setup -- we need a way to accept
76 // commands from the parent process to simulate Stop and
77 // RemoteStop commands that would normally be issued from
78 // application code.
Bogdan Capritaf30346f2014-11-21 15:41:21 -080079 defer remoteCmdLoop(stdin)()
Cosmos Nicolaoua429eff2014-11-19 14:02:34 -080080
81 // r.Cleanup is optional, but it's a good idea to clean up, especially
82 // since it takes care of flushing the logs before exiting.
83 defer r.Cleanup()
84
85 // Create a couple servers, and start serving.
86 server1 := makeServer()
87 server2 := makeServer()
88
89 // This is how to wait for a shutdown. In this example, a shutdown
90 // comes from a signal or a stop command.
91 var done sync.WaitGroup
92 done.Add(1)
93
94 // This is how to configure signal handling to allow clean shutdown.
95 sigChan := make(chan os.Signal, 2)
96 signal.Notify(sigChan, syscall.SIGTERM, syscall.SIGINT)
97
98 // This is how to configure handling of stop commands to allow clean
99 // shutdown.
100 stopChan := make(chan string, 2)
101 r.AppCycle().WaitForStop(stopChan)
102
103 // Blocking is used to prevent the process from exiting upon receiving a
104 // second signal or stop command while critical cleanup code is
105 // executing.
106 var blocking sync.WaitGroup
107 blockingCh := make(chan struct{})
108
109 // This is how to wait for a signal or stop command and initiate the
110 // clean shutdown steps.
111 go func() {
112 // First signal received.
113 select {
114 case sig := <-sigChan:
115 // If the developer wants to take different actions
116 // depending on the type of signal, they can do it here.
117 fmt.Fprintln(stdout, "Received signal", sig)
118 case stop := <-stopChan:
119 fmt.Fprintln(stdout, "Stop", stop)
120 }
121 // This commences the cleanup stage.
122 done.Done()
123 // Wait for a second signal or stop command, and force an exit,
124 // but only once all blocking cleanup code (if any) has
125 // completed.
126 select {
127 case <-sigChan:
128 case <-stopChan:
129 }
130 <-blockingCh
Bogdan Capritae06d7152014-11-22 09:32:05 -0800131 os.Exit(signals.DoubleStopExitCode)
Cosmos Nicolaoua429eff2014-11-19 14:02:34 -0800132 }()
133
134 // This communicates to the parent test driver process in our unit test
135 // that this server is ready and waiting on signals or stop commands.
136 // It's purely an artifact of our test setup.
137 fmt.Fprintln(stdout, "Ready")
138
139 // Wait for shutdown.
140 done.Wait()
141
142 // Stop the servers. In this example we stop them in goroutines to
143 // parallelize the wait, but if there was a dependency between the
144 // servers, the developer can simply stop them sequentially.
145 var waitServerStop sync.WaitGroup
146 waitServerStop.Add(2)
147 go func() {
148 server1.Stop()
149 waitServerStop.Done()
150 }()
151 go func() {
152 server2.Stop()
153 waitServerStop.Done()
154 }()
155 waitServerStop.Wait()
156
157 // This is where all cleanup code should go. By placing it at the end,
158 // we make its purpose and order of execution clear.
159
160 // This is an example of how to mix parallel and sequential cleanup
161 // steps. Most real-world servers will likely be simpler, with either
162 // just sequential or just parallel cleanup stages.
163
164 // parallelCleanup is used to wait for all goroutines executing cleanup
165 // code in parallel to finish.
166 var parallelCleanup sync.WaitGroup
167
168 // Simulate four parallel cleanup steps, two blocking and two
169 // interruptible.
170 parallelCleanup.Add(1)
171 blocking.Add(1)
172 go func() {
173 fmt.Fprintln(stdout, "Parallel blocking cleanup1")
174 blocking.Done()
175 parallelCleanup.Done()
176 }()
177
178 parallelCleanup.Add(1)
179 blocking.Add(1)
180 go func() {
181 fmt.Fprintln(stdout, "Parallel blocking cleanup2")
182 blocking.Done()
183 parallelCleanup.Done()
184 }()
185
186 parallelCleanup.Add(1)
187 go func() {
188 fmt.Fprintln(stdout, "Parallel interruptible cleanup1")
189 parallelCleanup.Done()
190 }()
191
192 parallelCleanup.Add(1)
193 go func() {
194 fmt.Fprintln(stdout, "Parallel interruptible cleanup2")
195 parallelCleanup.Done()
196 }()
197
198 // Simulate two sequential cleanup steps, one blocking and one
199 // interruptible.
200 fmt.Fprintln(stdout, "Sequential blocking cleanup")
201 blocking.Wait()
202 close(blockingCh)
203
204 fmt.Fprintln(stdout, "Sequential interruptible cleanup")
205
206 parallelCleanup.Wait()
207 return nil
208}
209
210// simpleServerProgram demonstrates the recommended way to write a typical
211// simple server application (with one server and a clean shutdown triggered by
212// a signal or a stop command). For an example of something more involved, see
213// complexServerProgram.
214func simpleServerProgram(stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error {
215 // Initialize the runtime. This is boilerplate.
216 r := rt.Init()
217
218 // This is part of the test setup -- we need a way to accept
219 // commands from the parent process to simulate Stop and
220 // RemoteStop commands that would normally be issued from
221 // application code.
Bogdan Capritaf30346f2014-11-21 15:41:21 -0800222 defer remoteCmdLoop(stdin)()
Cosmos Nicolaoua429eff2014-11-19 14:02:34 -0800223
224 // r.Cleanup is optional, but it's a good idea to clean up, especially
225 // since it takes care of flushing the logs before exiting.
226 //
227 // We use defer to ensure this is the last thing in the program (to
228 // avoid shutting down the runtime while it may still be in use), and to
229 // allow it to execute even if a panic occurs down the road.
230 defer r.Cleanup()
231
232 // Create a server, and start serving.
233 server := makeServer()
234
235 // This is how to wait for a shutdown. In this example, a shutdown
236 // comes from a signal or a stop command.
237 //
238 // Note, if the developer wants to exit immediately upon receiving a
239 // signal or stop command, they can skip this, in which case the default
240 // behavior is for the process to exit.
241 waiter := signals.ShutdownOnSignals()
242
243 // This communicates to the parent test driver process in our unit test
244 // that this server is ready and waiting on signals or stop commands.
245 // It's purely an artifact of our test setup.
246 fmt.Fprintln(stdout, "Ready")
247
248 // Use defer for anything that should still execute even if a panic
249 // occurs.
250 defer fmt.Fprintln(stdout, "Deferred cleanup")
251
252 // Wait for shutdown.
253 sig := <-waiter
254 // The developer could take different actions depending on the type of
255 // signal.
256 fmt.Fprintln(stdout, "Received signal", sig)
257
258 // Cleanup code starts here. Alternatively, these steps could be
259 // invoked through defer, but we list them here to make the order of
260 // operations obvious.
261
262 // Stop the server.
263 server.Stop()
264
265 // Note, this will not execute in cases of forced shutdown
266 // (e.g. SIGSTOP), when the process calls os.Exit (e.g. via log.Fatal),
267 // or when a panic occurs.
268 fmt.Fprintln(stdout, "Interruptible cleanup")
269
270 return nil
271}