blob: e8799b20d8b93137b66a96837d7cb2da55a53a32 [file] [log] [blame]
Jiri Simsad7616c92015-03-24 23:44:30 -07001// Copyright 2015 The Vanadium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
Jiri Simsa5293dcb2014-05-10 09:56:38 -07005package signals
6
7import (
Cosmos Nicolaoucc581722014-10-07 12:45:39 -07008 "bufio"
Jiri Simsa5293dcb2014-05-10 09:56:38 -07009 "fmt"
Cosmos Nicolaoucc581722014-10-07 12:45:39 -070010 "io"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070011 "os"
Cosmos Nicolaoucc581722014-10-07 12:45:39 -070012 "path/filepath"
13 "runtime"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070014 "syscall"
15 "testing"
16
Jiri Simsa6ac95222015-02-23 16:11:49 -080017 "v.io/v23"
18 "v.io/v23/context"
Jiri Simsa6ac95222015-02-23 16:11:49 -080019 "v.io/v23/naming"
Matt Rosencrantz94502cf2015-03-18 09:43:44 -070020 "v.io/v23/rpc"
Todd Wang94c9d0b2015-04-01 14:27:00 -070021 "v.io/v23/services/appcycle"
Jiri Simsa6ac95222015-02-23 16:11:49 -080022 "v.io/v23/vtrace"
Todd Wang712eeef2015-04-03 17:13:50 -070023 "v.io/x/ref/lib/mgmt"
Todd Wangb3511492015-04-07 23:32:34 -070024 "v.io/x/ref/lib/security/securityflag"
Todd Wangcd4b3cc2015-04-06 16:42:02 -070025 "v.io/x/ref/services/device"
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -070026 "v.io/x/ref/test"
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -070027 "v.io/x/ref/test/modules"
Todd Wangb3511492015-04-07 23:32:34 -070028
29 _ "v.io/x/ref/profiles"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070030)
31
Suharsh Sivakumard19c95d2015-02-19 14:44:50 -080032//go:generate v23 test generate
Jiri Simsa5293dcb2014-05-10 09:56:38 -070033
Suharsh Sivakumard68949c2015-01-26 10:32:23 -080034func stopLoop(stop func(), stdin io.Reader, ch chan<- struct{}) {
Cosmos Nicolaoucc581722014-10-07 12:45:39 -070035 scanner := bufio.NewScanner(stdin)
36 for scanner.Scan() {
37 switch scanner.Text() {
Jiri Simsa5293dcb2014-05-10 09:56:38 -070038 case "close":
39 close(ch)
40 return
41 case "stop":
Suharsh Sivakumard68949c2015-01-26 10:32:23 -080042 stop()
Jiri Simsa5293dcb2014-05-10 09:56:38 -070043 }
44 }
45}
46
Cosmos Nicolaoucc581722014-10-07 12:45:39 -070047func program(stdin io.Reader, stdout io.Writer, signals ...os.Signal) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -070048 ctx, shutdown := test.InitForTest()
Jiri Simsa5293dcb2014-05-10 09:56:38 -070049 closeStopLoop := make(chan struct{})
Jiri Simsa6ac95222015-02-23 16:11:49 -080050 go stopLoop(v23.GetAppCycle(ctx).Stop, stdin, closeStopLoop)
Suharsh Sivakumar946f64d2015-01-08 10:48:13 -080051 wait := ShutdownOnSignals(ctx, signals...)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -070052 fmt.Fprintf(stdout, "ready\n")
53 fmt.Fprintf(stdout, "received signal %s\n", <-wait)
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -080054 shutdown()
Jiri Simsa5293dcb2014-05-10 09:56:38 -070055 <-closeStopLoop
56}
57
Cosmos Nicolaoucc581722014-10-07 12:45:39 -070058func handleDefaults(stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error {
59 program(stdin, stdout)
60 return nil
Jiri Simsa5293dcb2014-05-10 09:56:38 -070061}
62
Cosmos Nicolaoucc581722014-10-07 12:45:39 -070063func handleCustom(stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error {
64 program(stdin, stdout, syscall.SIGABRT)
65 return nil
Jiri Simsa5293dcb2014-05-10 09:56:38 -070066}
67
Cosmos Nicolaoucc581722014-10-07 12:45:39 -070068func handleCustomWithStop(stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error {
69 program(stdin, stdout, STOP, syscall.SIGABRT, syscall.SIGHUP)
70 return nil
Bogdan Caprita1002ba42014-06-06 19:24:40 -070071}
72
Cosmos Nicolaoucc581722014-10-07 12:45:39 -070073func handleDefaultsIgnoreChan(stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -070074 ctx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -080075 defer shutdown()
Suharsh Sivakumar946f64d2015-01-08 10:48:13 -080076
Jiri Simsa5293dcb2014-05-10 09:56:38 -070077 closeStopLoop := make(chan struct{})
Jiri Simsa6ac95222015-02-23 16:11:49 -080078 go stopLoop(v23.GetAppCycle(ctx).Stop, stdin, closeStopLoop)
Suharsh Sivakumar946f64d2015-01-08 10:48:13 -080079 ShutdownOnSignals(ctx)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -070080 fmt.Fprintf(stdout, "ready\n")
Jiri Simsa5293dcb2014-05-10 09:56:38 -070081 <-closeStopLoop
Cosmos Nicolaoucc581722014-10-07 12:45:39 -070082 return nil
Jiri Simsa5293dcb2014-05-10 09:56:38 -070083}
84
85func isSignalInSet(sig os.Signal, set []os.Signal) bool {
86 for _, s := range set {
87 if sig == s {
88 return true
89 }
90 }
91 return false
92}
93
94func checkSignalIsDefault(t *testing.T, sig os.Signal) {
95 if !isSignalInSet(sig, defaultSignals()) {
96 t.Errorf("signal %s not in default signal set, as expected", sig)
97 }
98}
99
100func checkSignalIsNotDefault(t *testing.T, sig os.Signal) {
101 if isSignalInSet(sig, defaultSignals()) {
102 t.Errorf("signal %s unexpectedly in default signal set", sig)
103 }
104}
105
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700106func newShell(t *testing.T, ctx *context.T, command string) (*modules.Shell, modules.Handle) {
107 sh, err := modules.NewShell(ctx, nil, testing.Verbose(), t)
Cosmos Nicolaou344cc4a2014-11-26 15:38:43 -0800108 if err != nil {
109 t.Fatalf("unexpected error: %s", err)
110 }
Cosmos Nicolaou612ad382014-10-29 19:41:35 -0700111 handle, err := sh.Start(command, nil)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700112 if err != nil {
113 sh.Cleanup(os.Stderr, os.Stderr)
114 t.Fatalf("unexpected error: %s", err)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700115 return nil, nil
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700116 }
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700117 return sh, handle
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700118}
119
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700120// TestCleanShutdownSignal verifies that sending a signal to a child that
121// handles it by default causes the child to shut down cleanly.
122func TestCleanShutdownSignal(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700123 ctx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800124 defer shutdown()
125
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700126 sh, h := newShell(t, ctx, "handleDefaults")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700127 defer sh.Cleanup(os.Stderr, os.Stderr)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700128 h.Expect("ready")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700129 checkSignalIsDefault(t, syscall.SIGINT)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700130 syscall.Kill(h.Pid(), syscall.SIGINT)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700131 h.Expectf("received signal %s", syscall.SIGINT)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700132 fmt.Fprintf(h.Stdin(), "close\n")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700133 h.ExpectEOF()
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700134}
135
136// TestCleanShutdownStop verifies that sending a stop comamnd to a child that
137// handles stop commands by default causes the child to shut down cleanly.
138func TestCleanShutdownStop(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700139 ctx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800140 defer shutdown()
141
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700142 sh, h := newShell(t, ctx, "handleDefaults")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700143 defer sh.Cleanup(os.Stderr, os.Stderr)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700144 h.Expect("ready")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700145 fmt.Fprintf(h.Stdin(), "stop\n")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700146 h.Expectf("received signal %s", v23.LocalStop)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700147 fmt.Fprintf(h.Stdin(), "close\n")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700148 h.ExpectEOF()
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700149
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700150}
151
Bogdan Caprita1002ba42014-06-06 19:24:40 -0700152// TestCleanShutdownStopCustom verifies that sending a stop comamnd to a child
153// that handles stop command as part of a custom set of signals handled, causes
154// the child to shut down cleanly.
155func TestCleanShutdownStopCustom(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700156 ctx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800157 defer shutdown()
158
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700159 sh, h := newShell(t, ctx, "handleCustomWithStop")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700160 defer sh.Cleanup(os.Stderr, os.Stderr)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700161 h.Expect("ready")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700162 fmt.Fprintf(h.Stdin(), "stop\n")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700163 h.Expectf("received signal %s", v23.LocalStop)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700164 fmt.Fprintf(h.Stdin(), "close\n")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700165 h.ExpectEOF()
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700166}
167
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700168func testExitStatus(t *testing.T, h modules.Handle, code int) {
169 h.ExpectEOF()
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700170 _, file, line, _ := runtime.Caller(1)
171 file = filepath.Base(file)
172 if got, want := h.Shutdown(os.Stdout, os.Stderr), fmt.Errorf("exit status %d", code); got.Error() != want.Error() {
173 t.Errorf("%s:%d: got %q, want %q", file, line, got, want)
174 }
Bogdan Caprita1002ba42014-06-06 19:24:40 -0700175}
176
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700177// TestStopNoHandler verifies that sending a stop command to a child that does
178// not handle stop commands causes the child to exit immediately.
179func TestStopNoHandler(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700180 ctx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800181 defer shutdown()
182
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700183 sh, h := newShell(t, ctx, "handleCustom")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700184 defer sh.Cleanup(os.Stderr, os.Stderr)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700185 h.Expect("ready")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700186 fmt.Fprintf(h.Stdin(), "stop\n")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700187 testExitStatus(t, h, v23.UnhandledStopExitCode)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700188}
189
190// TestDoubleSignal verifies that sending a succession of two signals to a child
191// that handles these signals by default causes the child to exit immediately
192// upon receiving the second signal.
193func TestDoubleSignal(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700194 ctx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800195 defer shutdown()
196
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700197 sh, h := newShell(t, ctx, "handleDefaults")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700198 defer sh.Cleanup(os.Stderr, os.Stderr)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700199 h.Expect("ready")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700200 checkSignalIsDefault(t, syscall.SIGTERM)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700201 syscall.Kill(h.Pid(), syscall.SIGTERM)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700202 h.Expectf("received signal %s", syscall.SIGTERM)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700203 checkSignalIsDefault(t, syscall.SIGINT)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700204 syscall.Kill(h.Pid(), syscall.SIGINT)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700205 testExitStatus(t, h, DoubleStopExitCode)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700206}
207
208// TestSignalAndStop verifies that sending a signal followed by a stop command
209// to a child that handles these by default causes the child to exit immediately
210// upon receiving the stop command.
211func TestSignalAndStop(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700212 ctx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800213 defer shutdown()
214
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700215 sh, h := newShell(t, ctx, "handleDefaults")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700216 defer sh.Cleanup(os.Stderr, os.Stderr)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700217 h.Expect("ready")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700218 checkSignalIsDefault(t, syscall.SIGTERM)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700219 syscall.Kill(h.Pid(), syscall.SIGTERM)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700220 h.Expectf("received signal %s", syscall.SIGTERM)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700221 fmt.Fprintf(h.Stdin(), "stop\n")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700222 testExitStatus(t, h, DoubleStopExitCode)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700223}
224
225// TestDoubleStop verifies that sending a succession of stop commands to a child
226// that handles stop commands by default causes the child to exit immediately
227// upon receiving the second stop command.
228func TestDoubleStop(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700229 ctx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800230 defer shutdown()
231
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700232 sh, h := newShell(t, ctx, "handleDefaults")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700233 defer sh.Cleanup(os.Stderr, os.Stderr)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700234 h.Expect("ready")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700235 fmt.Fprintf(h.Stdin(), "stop\n")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700236 h.Expectf("received signal %s", v23.LocalStop)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700237 fmt.Fprintf(h.Stdin(), "stop\n")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700238 testExitStatus(t, h, DoubleStopExitCode)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700239}
240
241// TestSendUnhandledSignal verifies that sending a signal that the child does
242// not handle causes the child to exit as per the signal being sent.
243func TestSendUnhandledSignal(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700244 ctx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800245 defer shutdown()
246
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700247 sh, h := newShell(t, ctx, "handleDefaults")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700248 defer sh.Cleanup(os.Stderr, os.Stderr)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700249 h.Expect("ready")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700250 checkSignalIsNotDefault(t, syscall.SIGABRT)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700251 syscall.Kill(h.Pid(), syscall.SIGABRT)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700252 testExitStatus(t, h, 2)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700253}
254
255// TestDoubleSignalIgnoreChan verifies that, even if we ignore the channel that
256// ShutdownOnSignals returns, sending two signals should still cause the
257// process to exit (ensures that there is no dependency in ShutdownOnSignals
258// on having a goroutine read from the returned channel).
259func TestDoubleSignalIgnoreChan(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700260 ctx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800261 defer shutdown()
262
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700263 sh, h := newShell(t, ctx, "handleDefaultsIgnoreChan")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700264 defer sh.Cleanup(os.Stderr, os.Stderr)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700265 h.Expect("ready")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700266 // Even if we ignore the channel that ShutdownOnSignals returns,
267 // sending two signals should still cause the process to exit.
268 checkSignalIsDefault(t, syscall.SIGTERM)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700269 syscall.Kill(h.Pid(), syscall.SIGTERM)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700270 checkSignalIsDefault(t, syscall.SIGINT)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700271 syscall.Kill(h.Pid(), syscall.SIGINT)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700272 testExitStatus(t, h, DoubleStopExitCode)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700273}
274
275// TestHandlerCustomSignal verifies that sending a non-default signal to a
276// server that listens for that signal causes the server to shut down cleanly.
277func TestHandlerCustomSignal(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700278 ctx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800279 defer shutdown()
280
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700281 sh, h := newShell(t, ctx, "handleCustom")
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700282 defer sh.Cleanup(os.Stderr, os.Stderr)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700283 h.Expect("ready")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700284 checkSignalIsNotDefault(t, syscall.SIGABRT)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700285 syscall.Kill(h.Pid(), syscall.SIGABRT)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700286 h.Expectf("received signal %s", syscall.SIGABRT)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700287 fmt.Fprintf(h.Stdin(), "stop\n")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700288 h.ExpectEOF()
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700289}
Bogdan Caprita1002ba42014-06-06 19:24:40 -0700290
291// TestHandlerCustomSignalWithStop verifies that sending a custom stop signal
292// to a server that listens for that signal causes the server to shut down
293// cleanly, even when a STOP signal is also among the handled signals.
294func TestHandlerCustomSignalWithStop(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700295 rootCtx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800296 defer shutdown()
297
Jiri Simsa7fc6e872014-06-09 09:22:53 -0700298 for _, signal := range []syscall.Signal{syscall.SIGABRT, syscall.SIGHUP} {
Todd Wangad492042015-04-17 15:58:40 -0700299 ctx, _ := vtrace.WithNewTrace(rootCtx)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700300 sh, h := newShell(t, ctx, "handleCustomWithStop")
301 h.Expect("ready")
Bogdan Caprita1002ba42014-06-06 19:24:40 -0700302 checkSignalIsNotDefault(t, signal)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700303 syscall.Kill(h.Pid(), signal)
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700304 h.Expectf("received signal %s", signal)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700305 fmt.Fprintf(h.Stdin(), "close\n")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700306 h.ExpectEOF()
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700307 sh.Cleanup(os.Stderr, os.Stderr)
Bogdan Caprita1002ba42014-06-06 19:24:40 -0700308 }
309}
310
311// TestParseSignalsList verifies that ShutdownOnSignals correctly interprets
312// the input list of signals.
313func TestParseSignalsList(t *testing.T) {
314 list := []os.Signal{STOP, syscall.SIGTERM}
Matt Rosencrantzc7fecf12014-11-27 19:58:43 -0800315 ShutdownOnSignals(nil, list...)
Bogdan Caprita1002ba42014-06-06 19:24:40 -0700316 if !isSignalInSet(syscall.SIGTERM, list) {
317 t.Errorf("signal %s not in signal set, as expected: %v", syscall.SIGTERM, list)
318 }
319 if !isSignalInSet(STOP, list) {
320 t.Errorf("signal %s not in signal set, as expected: %v", STOP, list)
321 }
322}
Bogdan Capritaa4d9ee42014-06-20 16:42:53 -0700323
324type configServer struct {
325 ch chan<- string
326}
327
Todd Wang54feabe2015-04-15 23:38:26 -0700328func (c *configServer) Set(_ *context.T, _ rpc.ServerCall, key, value string) error {
Bogdan Capritaa4d9ee42014-06-20 16:42:53 -0700329 if key != mgmt.AppCycleManagerConfigKey {
330 return fmt.Errorf("Unexpected key: %v", key)
331 }
332 c.ch <- value
333 return nil
334
335}
336
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700337func createConfigServer(t *testing.T, ctx *context.T) (rpc.Server, string, <-chan string) {
Jiri Simsa6ac95222015-02-23 16:11:49 -0800338 server, err := v23.NewServer(ctx)
Bogdan Capritaa4d9ee42014-06-20 16:42:53 -0700339 if err != nil {
340 t.Fatalf("Got error: %v", err)
341 }
Bogdan Capritaa4d9ee42014-06-20 16:42:53 -0700342 ch := make(chan string)
Cosmos Nicolaou28dabfc2014-12-15 22:51:07 -0800343 var ep []naming.Endpoint
Jiri Simsa6ac95222015-02-23 16:11:49 -0800344 if ep, err = server.Listen(v23.GetListenSpec(ctx)); err != nil {
Bogdan Capritaa4d9ee42014-06-20 16:42:53 -0700345 t.Fatalf("Got error: %v", err)
346 }
Todd Wangb3511492015-04-07 23:32:34 -0700347 if err := server.Serve("", device.ConfigServer(&configServer{ch}), securityflag.NewAuthorizerOrDie()); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700348 t.Fatalf("Got error: %v", err)
349 }
Cosmos Nicolaou8bd8e102015-01-13 21:52:53 -0800350 return server, ep[0].Name(), ch
Bogdan Capritaa4d9ee42014-06-20 16:42:53 -0700351
352}
353
354// TestCleanRemoteShutdown verifies that remote shutdown works correctly.
355func TestCleanRemoteShutdown(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700356 ctx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800357 defer shutdown()
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700358
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700359 sh, err := modules.NewShell(ctx, nil, testing.Verbose(), t)
Cosmos Nicolaou344cc4a2014-11-26 15:38:43 -0800360 if err != nil {
361 t.Fatalf("unexpected error: %s", err)
362 }
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700363 defer sh.Cleanup(os.Stderr, os.Stderr)
364
Matt Rosencrantzf541b772015-01-13 07:58:59 -0800365 configServer, configServiceName, ch := createConfigServer(t, ctx)
Bogdan Capritaa4d9ee42014-06-20 16:42:53 -0700366 defer configServer.Stop()
Jiri Simsa37893392014-11-07 10:55:45 -0800367 sh.SetConfigKey(mgmt.ParentNameConfigKey, configServiceName)
368 sh.SetConfigKey(mgmt.ProtocolConfigKey, "tcp")
369 sh.SetConfigKey(mgmt.AddressConfigKey, "127.0.0.1:0")
Cosmos Nicolaou612ad382014-10-29 19:41:35 -0700370 h, err := sh.Start("handleDefaults", nil)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700371 if err != nil {
372 t.Fatalf("unexpected error: %s", err)
373 }
Bogdan Capritaa4d9ee42014-06-20 16:42:53 -0700374 appCycleName := <-ch
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700375 h.Expect("ready")
Todd Wang702385a2014-11-07 01:54:08 -0800376 appCycle := appcycle.AppCycleClient(appCycleName)
Matt Rosencrantzf541b772015-01-13 07:58:59 -0800377 stream, err := appCycle.Stop(ctx)
Bogdan Capritaa4d9ee42014-06-20 16:42:53 -0700378 if err != nil {
379 t.Fatalf("Got error: %v", err)
380 }
Shyam Jayaraman97b9dca2014-07-31 13:30:46 -0700381 rStream := stream.RecvStream()
382 if rStream.Advance() || rStream.Err() != nil {
383 t.Errorf("Expected EOF, got (%v, %v) instead: ", rStream.Value(), rStream.Err())
Bogdan Capritaa4d9ee42014-06-20 16:42:53 -0700384 }
385 if err := stream.Finish(); err != nil {
386 t.Fatalf("Got error: %v", err)
387 }
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700388 h.Expectf("received signal %s", v23.RemoteStop)
Cosmos Nicolaoucc581722014-10-07 12:45:39 -0700389 fmt.Fprintf(h.Stdin(), "close\n")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700390 h.ExpectEOF()
Bogdan Capritaa4d9ee42014-06-20 16:42:53 -0700391}