blob: 3c1a163aa82463d75b5bd6520ba3748b940fe572 [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
Robert Kroeger22fcb032015-04-30 07:40:11 -07005package utiltest
Bogdan Capritac87a9142014-07-21 10:38:13 -07006
7import (
Robin Thellend3480f8e2015-03-13 09:50:41 -07008 "errors"
9 "fmt"
Bogdan Capritad2b9f032014-10-10 17:43:29 -070010 "io/ioutil"
Bogdan Capritac87a9142014-07-21 10:38:13 -070011 "os"
Robert Kroegeref27d6e2015-05-01 16:58:59 -070012 "os/user"
Bogdan Capritad2b9f032014-10-10 17:43:29 -070013 "path/filepath"
Robert Kroeger1cb4a0d2014-10-20 11:55:38 -070014 "reflect"
Robert Kroeger95f4e682015-03-13 10:40:26 -070015 "regexp"
Robert Kroeger1cb4a0d2014-10-20 11:55:38 -070016 "sort"
Shyam Jayaramandbae76b2014-11-17 12:51:29 -080017 "strings"
Robert Kroeger4740f882015-06-11 16:26:13 -070018 "syscall"
Bogdan Capritac87a9142014-07-21 10:38:13 -070019 "testing"
Asim Shankar23dac322015-02-14 12:42:26 -080020 "time"
Bogdan Capritac87a9142014-07-21 10:38:13 -070021
Jiri Simsa6ac95222015-02-23 16:11:49 -080022 "v.io/v23"
23 "v.io/v23/context"
Jiri Simsa6ac95222015-02-23 16:11:49 -080024 "v.io/v23/naming"
25 "v.io/v23/options"
Matt Rosencrantz94502cf2015-03-18 09:43:44 -070026 "v.io/v23/rpc"
Jiri Simsa6ac95222015-02-23 16:11:49 -080027 "v.io/v23/security"
Todd Wang94c9d0b2015-04-01 14:27:00 -070028 "v.io/v23/services/application"
29 "v.io/v23/services/device"
30 "v.io/v23/services/logreader"
31 "v.io/v23/services/pprof"
32 "v.io/v23/services/stats"
Jiri Simsa6ac95222015-02-23 16:11:49 -080033 "v.io/v23/verror"
Cosmos Nicolaou18489732015-06-29 16:01:38 -070034 "v.io/x/ref/internal/logger"
Matt Rosencrantzbb6295d2015-06-19 15:13:58 -070035 "v.io/x/ref/lib/xrpc"
Suharsh Sivakumardcc11d72015-05-11 12:19:20 -070036 _ "v.io/x/ref/runtime/factories/roaming"
Bogdan Caprita1bba7362015-06-30 14:03:17 -070037 "v.io/x/ref/services/device/deviced/internal/impl"
Bogdan Caprita226d88e2015-06-30 17:34:57 -070038 "v.io/x/ref/services/device/deviced/internal/versioning"
Robert Kroeger60e03052015-05-04 17:35:16 -070039 "v.io/x/ref/services/internal/binarylib"
Todd Wang5fc36442015-04-07 15:15:27 -070040 "v.io/x/ref/services/internal/servicetest"
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -070041 "v.io/x/ref/test"
42 "v.io/x/ref/test/modules"
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -070043 "v.io/x/ref/test/testutil"
Bogdan Capritac87a9142014-07-21 10:38:13 -070044)
45
Bogdan Caprita916e99f2014-11-24 15:47:19 -080046const (
Bogdan Caprita916e99f2014-11-24 15:47:19 -080047 // TODO(caprita): Set the timeout in a more principled manner.
Bogdan Caprita2b050322015-04-17 09:04:03 -070048 killTimeout = 20 * time.Second
Bogdan Caprita916e99f2014-11-24 15:47:19 -080049)
Bogdan Capritad2b9f032014-10-10 17:43:29 -070050
Robert Kroegeref27d6e2015-05-01 16:58:59 -070051func init() {
52 impl.Describe = func() (descr device.Description, err error) {
53 return device.Description{Profiles: map[string]struct{}{"test-profile": struct{}{}}}, nil
54 }
Robert Kroeger3278a0c2015-05-05 14:29:24 -070055
Cosmos Nicolaou18489732015-06-29 16:01:38 -070056 impl.CleanupDir = func(ctx *context.T, dir, helper string) {
Robert Kroeger3278a0c2015-05-05 14:29:24 -070057 if dir == "" {
58 return
59 }
60 parentDir, base := filepath.Dir(dir), filepath.Base(dir)
61 var renamed string
62 if helper != "" {
63 renamed = filepath.Join(parentDir, "helper_deleted_"+base)
64 } else {
65 renamed = filepath.Join(parentDir, "deleted_"+base)
66 }
67 if err := os.Rename(dir, renamed); err != nil {
Cosmos Nicolaou18489732015-06-29 16:01:38 -070068 ctx.Errorf("Rename(%v, %v) failed: %v", dir, renamed, err)
Robert Kroeger3278a0c2015-05-05 14:29:24 -070069 }
70 }
Robert Kroeger742a6102015-06-15 10:41:28 -070071
Robert Kroegerafd85ff2015-06-17 17:40:24 -070072 // Return a sequence of times separated by 25 hours.
73 impl.MockableNow = func() time.Time {
74 now := time.Now()
75 impl.MockableNow = func() time.Time {
76 now = now.Add(time.Hour * 25)
77 return now
78 }
79 return now
80 }
81
Robert Kroegeref27d6e2015-05-01 16:58:59 -070082}
83
Todd Wang95873902015-05-22 14:21:30 -070084func EnvelopeFromShell(sh *modules.Shell, env []string, prog modules.Program, title string, retries int, window time.Duration, args ...string) application.Envelope {
85 args, nenv := sh.ProgramEnvelope(env, prog, args...)
Cosmos Nicolaouad2793f2014-10-27 16:24:15 -070086 return application.Envelope{
87 Title: title,
88 Args: args[1:],
89 // TODO(caprita): revisit how the environment is sanitized for arbirary
90 // apps.
Robert Kroeger73310c82015-05-14 16:51:10 -070091 Env: impl.VanadiumEnvironment(nenv),
92 Binary: application.SignedFile{File: MockBinaryRepoName},
93 Restarts: int32(retries),
94 RestartTimeWindow: window,
Bogdan Capritac87a9142014-07-21 10:38:13 -070095 }
96}
97
Arup Mukherjee13c00382015-06-04 15:54:38 -070098func SignedEnvelopeFromShell(ctx *context.T, sh *modules.Shell, env []string, prog modules.Program, title string, retries int, window time.Duration, args ...string) (application.Envelope, error) {
99 envelope := EnvelopeFromShell(sh, env, prog, title, retries, window, args...)
100 reader, cleanup, err := mockBinaryBytesReader()
101 defer cleanup()
102 sig, err := binarylib.Sign(ctx, reader)
103 if err != nil {
104 return application.Envelope{}, err
105 }
106 envelope.Binary.Signature = *sig
107
108 // Add a publisher blessing
109 p := v23.GetPrincipal(ctx)
110 publisher, err := p.Bless(p.PublicKey(), p.BlessingStore().Default(), "angryapp.v10", security.UnconstrainedUse())
111 if err != nil {
112 return application.Envelope{}, err
113 }
114 envelope.Publisher = publisher
115 return envelope, nil
116}
117
Robert Kroeger22fcb032015-04-30 07:40:11 -0700118// ResolveExpectNotFound verifies that the given name is not in the mounttable.
Cosmos Nicolaou33cf8982015-05-26 10:39:49 -0700119func ResolveExpectNotFound(t *testing.T, ctx *context.T, name string, retry bool) {
120 expectErr := naming.ErrNoSuchName.ID
121 for {
122 me, err := v23.GetNamespace(ctx).Resolve(ctx, name)
123 if err == nil || verror.ErrorID(err) != expectErr {
124 if retry {
125 time.Sleep(10 * time.Millisecond)
126 continue
127 }
128 if err == nil {
129 t.Fatalf(testutil.FormatLogLine(2, "Resolve(%v) succeeded with results %v when it was expected to fail", name, me.Names()))
130 } else {
131 t.Fatalf(testutil.FormatLogLine(2, "Resolve(%v) failed with error %v, expected error ID %v", name, err, expectErr))
132 }
133 } else {
134 return
135 }
Bogdan Capritac87a9142014-07-21 10:38:13 -0700136 }
137}
138
Robert Kroeger22fcb032015-04-30 07:40:11 -0700139// Resolve looks up the given name in the mounttable.
Cosmos Nicolaou33cf8982015-05-26 10:39:49 -0700140func Resolve(t *testing.T, ctx *context.T, name string, replicas int, retry bool) []string {
141 for {
142 me, err := v23.GetNamespace(ctx).Resolve(ctx, name)
143 if err != nil {
144 if retry {
145 time.Sleep(10 * time.Millisecond)
146 continue
147 } else {
148 t.Fatalf("Resolve(%v) failed: %v", name, err)
149 }
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800150 }
Cosmos Nicolaou33cf8982015-05-26 10:39:49 -0700151
152 filteredResults := []string{}
153 for _, r := range me.Names() {
154 if strings.Index(r, "@tcp") != -1 {
155 filteredResults = append(filteredResults, r)
156 }
157 }
158 // We are going to get a websocket and a tcp endpoint for each replica.
159 if want, got := replicas, len(filteredResults); want != got {
160 t.Fatalf("Resolve(%v) expected %d result(s), got %d instead", name, want, got)
161 }
162 return filteredResults
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800163 }
Bogdan Capritac87a9142014-07-21 10:38:13 -0700164}
165
166// The following set of functions are convenience wrappers around Update and
Bogdan Caprita2b219362014-12-09 17:03:33 -0800167// Revert for device manager.
Bogdan Capritac87a9142014-07-21 10:38:13 -0700168
Robert Kroeger22fcb032015-04-30 07:40:11 -0700169func DeviceStub(name string) device.DeviceClientMethods {
Bogdan Caprita9c4aa222014-12-10 14:46:30 -0800170 deviceName := naming.Join(name, "device")
Bogdan Capritaa456f472014-12-10 10:18:03 -0800171 return device.DeviceClient(deviceName)
Bogdan Capritac87a9142014-07-21 10:38:13 -0700172}
173
Robert Kroeger22fcb032015-04-30 07:40:11 -0700174func ClaimDevice(t *testing.T, ctx *context.T, claimableName, deviceName, extension, pairingToken string) {
Asim Shankar23dac322015-02-14 12:42:26 -0800175 // Setup blessings to be granted to the claimed device
Ankurdda16492015-04-07 12:35:42 -0700176 g := &granter{extension: extension}
Asim Shankar263c73b2015-03-19 18:31:26 -0700177 s := options.SkipServerEndpointAuthorization{}
Asim Shankarb547ea92015-02-17 18:49:45 -0800178 // Call the Claim RPC: Skip server authorization because the unclaimed
179 // device presents nothing that can be used to recognize it.
Bogdan Capritae8073682015-04-25 15:37:53 -0700180 if err := device.ClaimableClient(claimableName).Claim(ctx, pairingToken, g, s); err != nil {
181 t.Fatalf(testutil.FormatLogLine(2, "%q.Claim(%q) failed: %v [%v]", claimableName, pairingToken, verror.ErrorID(err), err))
Asim Shankar23dac322015-02-14 12:42:26 -0800182 }
183 // Wait for the device to remount itself with the device service after
184 // being claimed.
Asim Shankar23dac322015-02-14 12:42:26 -0800185 start := time.Now()
186 for {
Bogdan Capritae8073682015-04-25 15:37:53 -0700187 _, err := v23.GetNamespace(ctx).Resolve(ctx, deviceName)
188 if err == nil {
Asim Shankar23dac322015-02-14 12:42:26 -0800189 return
190 }
Cosmos Nicolaou18489732015-06-29 16:01:38 -0700191 ctx.VI(4).Infof("Resolve(%q) failed: %v", err)
Asim Shankar23dac322015-02-14 12:42:26 -0800192 time.Sleep(time.Millisecond)
193 if elapsed := time.Since(start); elapsed > time.Minute {
194 t.Fatalf("Device hasn't remounted itself in %v since it was claimed", elapsed)
195 }
196 }
197}
198
Robert Kroeger22fcb032015-04-30 07:40:11 -0700199func ClaimDeviceExpectError(t *testing.T, ctx *context.T, name, extension, pairingToken string, errID verror.ID) {
Asim Shankar23dac322015-02-14 12:42:26 -0800200 // Setup blessings to be granted to the claimed device
Ankurdda16492015-04-07 12:35:42 -0700201 g := &granter{extension: extension}
Asim Shankar263c73b2015-03-19 18:31:26 -0700202 s := options.SkipServerEndpointAuthorization{}
Asim Shankar23dac322015-02-14 12:42:26 -0800203 // Call the Claim RPC
Todd Wang8fa38762015-03-25 14:04:59 -0700204 if err := device.ClaimableClient(name).Claim(ctx, pairingToken, g, s); verror.ErrorID(err) != errID {
Asim Shankar23dac322015-02-14 12:42:26 -0800205 t.Fatalf(testutil.FormatLogLine(2, "%q.Claim(%q) expected to fail with %v, got %v [%v]", name, pairingToken, errID, verror.ErrorID(err), err))
206 }
207}
208
Robert Kroeger22fcb032015-04-30 07:40:11 -0700209func UpdateDeviceExpectError(t *testing.T, ctx *context.T, name string, errID verror.ID) {
210 if err := DeviceStub(name).Update(ctx); verror.ErrorID(err) != errID {
Asim Shankar23dac322015-02-14 12:42:26 -0800211 t.Fatalf(testutil.FormatLogLine(2, "%q.Update expected to fail with %v, got %v [%v]", name, errID, verror.ErrorID(err), err))
Bogdan Capritac87a9142014-07-21 10:38:13 -0700212 }
213}
214
Robert Kroeger22fcb032015-04-30 07:40:11 -0700215func UpdateDevice(t *testing.T, ctx *context.T, name string) {
216 if err := DeviceStub(name).Update(ctx); err != nil {
Asim Shankar23dac322015-02-14 12:42:26 -0800217 t.Fatalf(testutil.FormatLogLine(2, "%q.Update() failed: %v [%v]", name, verror.ErrorID(err), err))
Bogdan Capritac87a9142014-07-21 10:38:13 -0700218 }
219}
220
Robert Kroeger22fcb032015-04-30 07:40:11 -0700221func RevertDeviceExpectError(t *testing.T, ctx *context.T, name string, errID verror.ID) {
222 if err := DeviceStub(name).Revert(ctx); verror.ErrorID(err) != errID {
Asim Shankar23dac322015-02-14 12:42:26 -0800223 t.Fatalf(testutil.FormatLogLine(2, "%q.Revert() expected to fail with %v, got %v [%v]", name, errID, verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700224 }
225}
226
Robert Kroeger22fcb032015-04-30 07:40:11 -0700227func RevertDevice(t *testing.T, ctx *context.T, name string) {
228 if err := DeviceStub(name).Revert(ctx); err != nil {
Asim Shankar23dac322015-02-14 12:42:26 -0800229 t.Fatalf(testutil.FormatLogLine(2, "%q.Revert() failed: %v [%v]", name, verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700230 }
231}
232
Robert Kroeger22fcb032015-04-30 07:40:11 -0700233func KillDevice(t *testing.T, ctx *context.T, name string) {
234 if err := DeviceStub(name).Kill(ctx, killTimeout); err != nil {
Bogdan Caprita2b050322015-04-17 09:04:03 -0700235 t.Fatalf(testutil.FormatLogLine(2, "%q.Kill(%v) failed: %v [%v]", name, killTimeout, verror.ErrorID(err), err))
Bogdan Caprita4ea9b032014-12-27 14:56:51 -0800236 }
237}
238
Robert Kroeger22fcb032015-04-30 07:40:11 -0700239func ShutdownDevice(t *testing.T, ctx *context.T, name string) {
240 if err := DeviceStub(name).Delete(ctx); err != nil {
Bogdan Caprita2b050322015-04-17 09:04:03 -0700241 t.Fatalf(testutil.FormatLogLine(2, "%q.Delete() failed: %v [%v]", name, verror.ErrorID(err), err))
Bogdan Caprita4ea9b032014-12-27 14:56:51 -0800242 }
243}
244
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700245// The following set of functions are convenience wrappers around various app
246// management methods.
247
Robert Kroeger22fcb032015-04-30 07:40:11 -0700248func Ocfg(opt []interface{}) device.Config {
Bogdan Caprita17666dd2015-01-14 09:27:46 -0800249 for _, o := range opt {
250 if c, ok := o.(device.Config); ok {
251 return c
252 }
253 }
254 return device.Config{}
255}
256
Robert Kroeger22fcb032015-04-30 07:40:11 -0700257func Opkg(opt []interface{}) application.Packages {
Bogdan Capritac25a48c2015-02-12 13:45:51 -0800258 for _, o := range opt {
259 if c, ok := o.(application.Packages); ok {
260 return c
261 }
262 }
263 return application.Packages{}
264}
265
Robert Kroeger22fcb032015-04-30 07:40:11 -0700266func AppStub(nameComponents ...string) device.ApplicationClientMethods {
Robert Kroeger74645172015-02-11 14:22:21 -0800267 appsName := "dm/apps"
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700268 appName := naming.Join(append([]string{appsName}, nameComponents...)...)
Bogdan Capritaa456f472014-12-10 10:18:03 -0800269 return device.ApplicationClient(appName)
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700270}
271
Robert Kroeger22fcb032015-04-30 07:40:11 -0700272func StatsStub(nameComponents ...string) stats.StatsClientMethods {
Robert Kroegerbc226062015-03-16 17:24:07 -0700273 statsName := naming.Join(nameComponents...)
Robert Kroeger6e4ed122015-03-06 13:03:54 -0800274 return stats.StatsClient(statsName)
275}
276
Robert Kroeger22fcb032015-04-30 07:40:11 -0700277func InstallApp(t *testing.T, ctx *context.T, opt ...interface{}) string {
278 appID, err := AppStub().Install(ctx, MockApplicationRepoName, Ocfg(opt), Opkg(opt))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700279 if err != nil {
Asim Shankar23dac322015-02-14 12:42:26 -0800280 t.Fatalf(testutil.FormatLogLine(2, "Install failed: %v [%v]", verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700281 }
282 return appID
283}
284
Robert Kroeger22fcb032015-04-30 07:40:11 -0700285func InstallAppExpectError(t *testing.T, ctx *context.T, expectedError verror.ID, opt ...interface{}) {
286 if _, err := AppStub().Install(ctx, MockApplicationRepoName, Ocfg(opt), Opkg(opt)); err == nil || verror.ErrorID(err) != expectedError {
Asim Shankar23dac322015-02-14 12:42:26 -0800287 t.Fatalf(testutil.FormatLogLine(2, "Install expected to fail with %v, got %v [%v]", expectedError, verror.ErrorID(err), err))
Bogdan Caprita17666dd2015-01-14 09:27:46 -0800288 }
289}
290
Bogdan Caprita26929102014-11-07 11:56:56 -0800291type granter struct {
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700292 rpc.CallOpt
Bogdan Caprita26929102014-11-07 11:56:56 -0800293 p security.Principal
294 extension string
295}
296
Todd Wang4264e4b2015-04-16 22:43:40 -0700297func (g *granter) Grant(ctx *context.T, call security.Call) (security.Blessings, error) {
Ankurdda16492015-04-07 12:35:42 -0700298 p := call.LocalPrincipal()
299 return p.Bless(call.RemoteBlessings().PublicKey(), p.BlessingStore().Default(), g.extension, security.UnconstrainedUse())
Bogdan Caprita26929102014-11-07 11:56:56 -0800300}
301
Robert Kroeger22fcb032015-04-30 07:40:11 -0700302func LaunchAppImpl(t *testing.T, ctx *context.T, appID, grant string) (string, error) {
303 instanceID, err := NewInstanceImpl(t, ctx, appID, grant)
Robin Thellend3480f8e2015-03-13 09:50:41 -0700304 if err != nil {
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700305 return "", err
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700306 }
Robert Kroeger22fcb032015-04-30 07:40:11 -0700307 return instanceID, AppStub(appID, instanceID).Run(ctx)
Bogdan Caprita2b050322015-04-17 09:04:03 -0700308}
309
Robert Kroeger22fcb032015-04-30 07:40:11 -0700310func NewInstanceImpl(t *testing.T, ctx *context.T, appID, grant string) (string, error) {
Bogdan Caprita2b050322015-04-17 09:04:03 -0700311 ctx, delete := context.WithCancel(ctx)
312 defer delete()
313
Robert Kroeger22fcb032015-04-30 07:40:11 -0700314 call, err := AppStub(appID).Instantiate(ctx)
Bogdan Caprita2b050322015-04-17 09:04:03 -0700315 if err != nil {
316 return "", err
317 }
Robin Thellend3480f8e2015-03-13 09:50:41 -0700318 for call.RecvStream().Advance() {
319 switch msg := call.RecvStream().Value().(type) {
Bogdan Caprita2b050322015-04-17 09:04:03 -0700320 case device.BlessServerMessageInstancePublicKey:
Robin Thellend3480f8e2015-03-13 09:50:41 -0700321 p := v23.GetPrincipal(ctx)
322 pubKey, err := security.UnmarshalPublicKey(msg.Value)
323 if err != nil {
324 return "", err
325 }
326 blessings, err := p.Bless(pubKey, p.BlessingStore().Default(), grant, security.UnconstrainedUse())
327 if err != nil {
328 return "", errors.New("bless failed")
329 }
Jiri Simsad9a7b3c2015-08-12 16:38:27 -0700330 call.SendStream().Send(device.BlessClientMessageAppBlessings{Value: blessings})
Robin Thellend3480f8e2015-03-13 09:50:41 -0700331 default:
Bogdan Caprita2b050322015-04-17 09:04:03 -0700332 return "", fmt.Errorf("newInstanceImpl: received unexpected message: %#v", msg)
Robin Thellend3480f8e2015-03-13 09:50:41 -0700333 }
334 }
Bogdan Caprita2b050322015-04-17 09:04:03 -0700335 var instanceID string
336 if instanceID, err = call.Finish(); err != nil {
Robin Thellend3480f8e2015-03-13 09:50:41 -0700337 return "", err
338 }
Bogdan Caprita2b050322015-04-17 09:04:03 -0700339 return instanceID, nil
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700340}
341
Robert Kroeger22fcb032015-04-30 07:40:11 -0700342func LaunchApp(t *testing.T, ctx *context.T, appID string) string {
343 instanceID, err := LaunchAppImpl(t, ctx, appID, "forapp")
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700344 if err != nil {
Bogdan Caprita2b050322015-04-17 09:04:03 -0700345 t.Fatalf(testutil.FormatLogLine(2, "launching %v failed: %v [%v]", appID, verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700346 }
347 return instanceID
348}
349
Robert Kroeger22fcb032015-04-30 07:40:11 -0700350func LaunchAppExpectError(t *testing.T, ctx *context.T, appID string, expectedError verror.ID) {
351 if _, err := LaunchAppImpl(t, ctx, appID, "forapp"); err == nil || verror.ErrorID(err) != expectedError {
Bogdan Caprita2b050322015-04-17 09:04:03 -0700352 t.Fatalf(testutil.FormatLogLine(2, "launching %v expected to fail with %v, got %v [%v]", appID, expectedError, verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700353 }
354}
355
Robert Kroeger22fcb032015-04-30 07:40:11 -0700356func TerminateApp(t *testing.T, ctx *context.T, appID, instanceID string) {
357 if err := AppStub(appID, instanceID).Kill(ctx, killTimeout); err != nil {
Bogdan Caprita2b050322015-04-17 09:04:03 -0700358 t.Fatalf(testutil.FormatLogLine(2, "Kill(%v/%v) failed: %v [%v]", appID, instanceID, verror.ErrorID(err), err))
359 }
Robert Kroeger22fcb032015-04-30 07:40:11 -0700360 if err := AppStub(appID, instanceID).Delete(ctx); err != nil {
Bogdan Caprita2b050322015-04-17 09:04:03 -0700361 t.Fatalf(testutil.FormatLogLine(2, "Delete(%v/%v) failed: %v [%v]", appID, instanceID, verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700362 }
363}
364
Robert Kroeger22fcb032015-04-30 07:40:11 -0700365func KillApp(t *testing.T, ctx *context.T, appID, instanceID string) {
366 if err := AppStub(appID, instanceID).Kill(ctx, killTimeout); err != nil {
Bogdan Caprita2b050322015-04-17 09:04:03 -0700367 t.Fatalf(testutil.FormatLogLine(2, "Kill(%v/%v) failed: %v [%v]", appID, instanceID, verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700368 }
369}
370
Robert Kroeger22fcb032015-04-30 07:40:11 -0700371func DeleteApp(t *testing.T, ctx *context.T, appID, instanceID string) {
372 if err := AppStub(appID, instanceID).Delete(ctx); err != nil {
Bogdan Caprita2b050322015-04-17 09:04:03 -0700373 t.Fatalf(testutil.FormatLogLine(2, "Delete(%v/%v) failed: %v [%v]", appID, instanceID, verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700374 }
375}
376
Robert Kroeger22fcb032015-04-30 07:40:11 -0700377func RunApp(t *testing.T, ctx *context.T, appID, instanceID string) {
378 if err := AppStub(appID, instanceID).Run(ctx); err != nil {
Bogdan Caprita2b050322015-04-17 09:04:03 -0700379 t.Fatalf(testutil.FormatLogLine(2, "Run(%v/%v) failed: %v [%v]", appID, instanceID, verror.ErrorID(err), err))
380 }
381}
382
Robert Kroeger22fcb032015-04-30 07:40:11 -0700383func RunAppExpectError(t *testing.T, ctx *context.T, appID, instanceID string, expectedError verror.ID) {
384 if err := AppStub(appID, instanceID).Run(ctx); err == nil || verror.ErrorID(err) != expectedError {
Bogdan Caprita2b050322015-04-17 09:04:03 -0700385 t.Fatalf(testutil.FormatLogLine(2, "Run(%v/%v) expected to fail with %v, got %v [%v]", appID, instanceID, expectedError, verror.ErrorID(err), err))
Robert Kroeger1ce0bd72014-10-22 13:57:14 -0700386 }
387}
388
Robert Kroeger22fcb032015-04-30 07:40:11 -0700389func UpdateInstance(t *testing.T, ctx *context.T, appID, instanceID string) {
390 if err := AppStub(appID, instanceID).Update(ctx); err != nil {
Bogdan Capritab4d1dfd2015-02-26 11:18:34 -0800391 t.Fatalf(testutil.FormatLogLine(2, "Update(%v/%v) failed: %v [%v]", appID, instanceID, verror.ErrorID(err), err))
392 }
393}
394
Robert Kroeger22fcb032015-04-30 07:40:11 -0700395func UpdateInstanceExpectError(t *testing.T, ctx *context.T, appID, instanceID string, expectedError verror.ID) {
396 if err := AppStub(appID, instanceID).Update(ctx); err == nil || verror.ErrorID(err) != expectedError {
Bogdan Capritab4d1dfd2015-02-26 11:18:34 -0800397 t.Fatalf(testutil.FormatLogLine(2, "Update(%v/%v) expected to fail with %v, got %v [%v]", appID, instanceID, expectedError, verror.ErrorID(err), err))
398 }
399}
400
Robert Kroeger22fcb032015-04-30 07:40:11 -0700401func UpdateApp(t *testing.T, ctx *context.T, appID string) {
402 if err := AppStub(appID).Update(ctx); err != nil {
Asim Shankar23dac322015-02-14 12:42:26 -0800403 t.Fatalf(testutil.FormatLogLine(2, "Update(%v) failed: %v [%v]", appID, verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700404 }
405}
406
Robert Kroeger22fcb032015-04-30 07:40:11 -0700407func UpdateAppExpectError(t *testing.T, ctx *context.T, appID string, expectedError verror.ID) {
408 if err := AppStub(appID).Update(ctx); err == nil || verror.ErrorID(err) != expectedError {
Asim Shankar23dac322015-02-14 12:42:26 -0800409 t.Fatalf(testutil.FormatLogLine(2, "Update(%v) expected to fail with %v, got %v [%v]", appID, expectedError, verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700410 }
411}
412
Robert Kroeger22fcb032015-04-30 07:40:11 -0700413func RevertApp(t *testing.T, ctx *context.T, appID string) {
414 if err := AppStub(appID).Revert(ctx); err != nil {
Asim Shankar23dac322015-02-14 12:42:26 -0800415 t.Fatalf(testutil.FormatLogLine(2, "Revert(%v) failed: %v [%v]", appID, verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700416 }
417}
418
Robert Kroeger22fcb032015-04-30 07:40:11 -0700419func RevertAppExpectError(t *testing.T, ctx *context.T, appID string, expectedError verror.ID) {
420 if err := AppStub(appID).Revert(ctx); err == nil || verror.ErrorID(err) != expectedError {
Asim Shankar23dac322015-02-14 12:42:26 -0800421 t.Fatalf(testutil.FormatLogLine(2, "Revert(%v) expected to fail with %v, got %v [%v]", appID, expectedError, verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700422 }
423}
424
Robert Kroeger22fcb032015-04-30 07:40:11 -0700425func UninstallApp(t *testing.T, ctx *context.T, appID string) {
426 if err := AppStub(appID).Uninstall(ctx); err != nil {
Asim Shankar23dac322015-02-14 12:42:26 -0800427 t.Fatalf(testutil.FormatLogLine(2, "Uninstall(%v) failed: %v [%v]", appID, verror.ErrorID(err), err))
Bogdan Caprita48bbd142014-09-04 16:07:23 -0700428 }
Bogdan Capritac87a9142014-07-21 10:38:13 -0700429}
Robert Kroeger1cb4a0d2014-10-20 11:55:38 -0700430
Robert Kroeger22fcb032015-04-30 07:40:11 -0700431func Debug(t *testing.T, ctx *context.T, nameComponents ...string) string {
432 dbg, err := AppStub(nameComponents...).Debug(ctx)
Bogdan Capritad8373a12015-01-28 19:52:37 -0800433 if err != nil {
Asim Shankar23dac322015-02-14 12:42:26 -0800434 t.Fatalf(testutil.FormatLogLine(2, "Debug(%v) failed: %v [%v]", nameComponents, verror.ErrorID(err), err))
Bogdan Capritad8373a12015-01-28 19:52:37 -0800435 }
436 return dbg
437}
438
Bogdan Caprita019f0ad2015-05-21 10:45:10 -0700439func VerifyDeviceState(t *testing.T, ctx *context.T, want device.InstanceState, name string) string {
440 s, err := DeviceStub(name).Status(ctx)
441 if err != nil {
442 t.Fatalf(testutil.FormatLogLine(2, "Status(%v) failed: %v [%v]", name, verror.ErrorID(err), err))
443 }
Bogdan Capritad2188b72015-05-22 12:31:48 -0700444 status, ok := s.(device.StatusDevice)
Bogdan Caprita019f0ad2015-05-21 10:45:10 -0700445 if !ok {
446 t.Fatalf(testutil.FormatLogLine(2, "Status(%v) returned unknown type: %T", name, s))
447 }
448 if status.Value.State != want {
449 t.Fatalf(testutil.FormatLogLine(2, "Status(%v) state: wanted %v, got %v", name, want, status.Value.State))
450 }
451 return status.Value.Version
452}
453
Robert Kroeger22fcb032015-04-30 07:40:11 -0700454func Status(t *testing.T, ctx *context.T, nameComponents ...string) device.Status {
455 s, err := AppStub(nameComponents...).Status(ctx)
Bogdan Capritab7f5d772015-03-24 16:37:59 -0700456 if err != nil {
Robert Kroeger8c2be902015-06-19 15:33:45 -0700457 t.Errorf(testutil.FormatLogLine(3, "Status(%v) failed: %v [%v]", nameComponents, verror.ErrorID(err), err))
Bogdan Capritab7f5d772015-03-24 16:37:59 -0700458 }
459 return s
460}
461
Robert Kroeger22fcb032015-04-30 07:40:11 -0700462func VerifyState(t *testing.T, ctx *context.T, want interface{}, nameComponents ...string) string {
463 s := Status(t, ctx, nameComponents...)
Bogdan Capritab7f5d772015-03-24 16:37:59 -0700464 var (
465 state interface{}
466 version string
467 )
468 switch s := s.(type) {
469 case device.StatusInstance:
470 state = s.Value.State
471 version = s.Value.Version
472 case device.StatusInstallation:
473 state = s.Value.State
474 version = s.Value.Version
475 default:
Robert Kroeger8c2be902015-06-19 15:33:45 -0700476 t.Errorf(testutil.FormatLogLine(2, "Status(%v) returned unknown type: %T", nameComponents, s))
Bogdan Capritab7f5d772015-03-24 16:37:59 -0700477 }
478 if state != want {
Robert Kroeger8c2be902015-06-19 15:33:45 -0700479 t.Errorf(testutil.FormatLogLine(2, "Status(%v) state: wanted %v (%T), got %v (%T)", nameComponents, want, want, state, state))
Bogdan Capritab7f5d772015-03-24 16:37:59 -0700480 }
481 return version
482}
483
Robert Kroeger1cb4a0d2014-10-20 11:55:38 -0700484// Code to make Association lists sortable.
Bogdan Capritaa456f472014-12-10 10:18:03 -0800485type byIdentity []device.Association
Robert Kroeger1cb4a0d2014-10-20 11:55:38 -0700486
487func (a byIdentity) Len() int { return len(a) }
488func (a byIdentity) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
489func (a byIdentity) Less(i, j int) bool { return a[i].IdentityName < a[j].IdentityName }
490
Robert Kroeger22fcb032015-04-30 07:40:11 -0700491func CompareAssociations(t *testing.T, got, expected []device.Association) {
Robert Kroeger1cb4a0d2014-10-20 11:55:38 -0700492 sort.Sort(byIdentity(got))
493 sort.Sort(byIdentity(expected))
494 if !reflect.DeepEqual(got, expected) {
Robert Kroeger8c2be902015-06-19 15:33:45 -0700495 t.Errorf("ListAssociations() got %v, expected %v", got, expected)
Robert Kroeger1cb4a0d2014-10-20 11:55:38 -0700496 }
497}
Robert Kroeger94ec7562014-10-28 17:58:44 -0700498
Robert Kroeger22fcb032015-04-30 07:40:11 -0700499// GenerateSuidHelperScript builds a script to execute the test target as
Robert Kroeger94ec7562014-10-28 17:58:44 -0700500// a suidhelper instance and returns the path to the script.
Robert Kroeger22fcb032015-04-30 07:40:11 -0700501func GenerateSuidHelperScript(t *testing.T, root string) string {
Robert Kroeger94ec7562014-10-28 17:58:44 -0700502 output := "#!/bin/bash\n"
Asim Shankar3a6ca472015-03-31 00:35:37 -0700503 output += "V23_SUIDHELPER_TEST=1"
Robert Kroeger94ec7562014-10-28 17:58:44 -0700504 output += " "
Arup Mukherjee504a1e62015-02-25 11:09:56 -0800505 output += "exec " + os.Args[0] + " -minuid=1 -test.run=TestSuidHelper \"$@\""
Robert Kroeger94ec7562014-10-28 17:58:44 -0700506 output += "\n"
507
Cosmos Nicolaou18489732015-06-29 16:01:38 -0700508 logger.Global().VI(1).Infof("script\n%s", output)
Robert Kroeger94ec7562014-10-28 17:58:44 -0700509
510 if err := os.MkdirAll(root, 0755); err != nil {
511 t.Fatalf("MkdirAll failed: %v", err)
512 }
Robert Kroeger94ec7562014-10-28 17:58:44 -0700513 path := filepath.Join(root, "helper.sh")
514 if err := ioutil.WriteFile(path, []byte(output), 0755); err != nil {
515 t.Fatalf("WriteFile(%v) failed: %v", path, err)
516 }
517 return path
518}
Bogdan Capritac7e72b62015-01-07 19:22:23 -0800519
Robert Kroeger22fcb032015-04-30 07:40:11 -0700520// GenerateAgentScript creates a simple script that acts as the security agent
Bogdan Capritac7e72b62015-01-07 19:22:23 -0800521// for tests. It blackholes arguments meant for the agent.
Robert Kroeger22fcb032015-04-30 07:40:11 -0700522func GenerateAgentScript(t *testing.T, root string) string {
Bogdan Capritac7e72b62015-01-07 19:22:23 -0800523 output := `
524#!/bin/bash
525ARGS=$*
526for ARG in ${ARGS[@]}; do
527 if [[ ${ARG} = -- ]]; then
528 ARGS=(${ARGS[@]/$ARG})
529 break
530 elif [[ ${ARG} == --* ]]; then
531 ARGS=(${ARGS[@]/$ARG})
532 else
533 break
534 fi
535done
536
537exec ${ARGS[@]}
538`
539 if err := os.MkdirAll(root, 0755); err != nil {
540 t.Fatalf("MkdirAll failed: %v", err)
541 }
542 path := filepath.Join(root, "agenthelper.sh")
543 if err := ioutil.WriteFile(path, []byte(output), 0755); err != nil {
544 t.Fatalf("WriteFile(%v) failed: %v", path, err)
545 }
546 return path
547}
Asim Shankarb547ea92015-02-17 18:49:45 -0800548
Robert Kroeger22fcb032015-04-30 07:40:11 -0700549func CtxWithNewPrincipal(t *testing.T, ctx *context.T, idp *testutil.IDProvider, extension string) *context.T {
Todd Wangad492042015-04-17 15:58:40 -0700550 ret, err := v23.WithPrincipal(ctx, testutil.NewPrincipal())
Asim Shankarb547ea92015-02-17 18:49:45 -0800551 if err != nil {
Todd Wangad492042015-04-17 15:58:40 -0700552 t.Fatalf(testutil.FormatLogLine(2, "v23.WithPrincipal failed: %v", err))
Asim Shankarb547ea92015-02-17 18:49:45 -0800553 }
Jiri Simsa6ac95222015-02-23 16:11:49 -0800554 if err := idp.Bless(v23.GetPrincipal(ret), extension); err != nil {
Asim Shankarb547ea92015-02-17 18:49:45 -0800555 t.Fatalf(testutil.FormatLogLine(2, "idp.Bless(?, %q) failed: %v", extension, err))
556 }
557 return ret
558}
Robert Kroeger6e4ed122015-03-06 13:03:54 -0800559
560// TODO(rjkroege): This helper is generally useful. Use it to reduce
561// boiler plate across all device manager tests.
Cosmos Nicolaou18489732015-06-29 16:01:38 -0700562func StartupHelper(t *testing.T) (_ func(), ctx *context.T, _ *modules.Shell, _ *application.Envelope, _ string, _ string, _ *testutil.IDProvider) {
Todd Wang60052d82015-05-22 15:00:10 -0700563 ctx, shutdown := test.V23Init()
Robert Kroeger6e4ed122015-03-06 13:03:54 -0800564
565 // Make a new identity context.
Asim Shankar4a698282015-03-21 21:59:18 -0700566 idp := testutil.NewIDProvider("root")
Robert Kroeger22fcb032015-04-30 07:40:11 -0700567 ctx = CtxWithNewPrincipal(t, ctx, idp, "self")
Robert Kroeger6e4ed122015-03-06 13:03:54 -0800568
Todd Wang5fc36442015-04-07 15:15:27 -0700569 sh, deferFn := servicetest.CreateShellAndMountTable(t, ctx, nil)
Robert Kroeger6e4ed122015-03-06 13:03:54 -0800570
571 // Set up mock application and binary repositories.
Robert Kroeger22fcb032015-04-30 07:40:11 -0700572 envelope, envCleanup := StartMockRepos(t, ctx)
Robert Kroeger6e4ed122015-03-06 13:03:54 -0800573
Todd Wang5fc36442015-04-07 15:15:27 -0700574 root, rootCleanup := servicetest.SetupRootDir(t, "devicemanager")
Bogdan Caprita226d88e2015-06-30 17:34:57 -0700575 if err := versioning.SaveCreatorInfo(ctx, root); err != nil {
Robert Kroeger6e4ed122015-03-06 13:03:54 -0800576 t.Fatal(err)
577 }
578
579 // Create a script wrapping the test target that implements suidhelper.
Robert Kroeger22fcb032015-04-30 07:40:11 -0700580 helperPath := GenerateSuidHelperScript(t, root)
Robert Kroeger6e4ed122015-03-06 13:03:54 -0800581
582 return func() {
583 rootCleanup()
584 envCleanup()
585 deferFn()
586 shutdown()
587 }, ctx, sh, envelope, root, helperPath, idp
588}
Robert Kroeger95f4e682015-03-13 10:40:26 -0700589
Robert Kroeger22fcb032015-04-30 07:40:11 -0700590type GlobTestVector struct {
591 Name, Pattern string
592 Expected []string
Robert Kroeger95f4e682015-03-13 10:40:26 -0700593}
594
595type globTestRegexHelper struct {
596 logFileTimeStampRE *regexp.Regexp
597 logFileTrimInfoRE *regexp.Regexp
598 logFileRemoveErrorFatalWarningRE *regexp.Regexp
599 statsTrimRE *regexp.Regexp
600}
601
Robert Kroeger22fcb032015-04-30 07:40:11 -0700602func NewGlobTestRegexHelper(appName string) *globTestRegexHelper {
Robert Kroeger95f4e682015-03-13 10:40:26 -0700603 return &globTestRegexHelper{
604 logFileTimeStampRE: regexp.MustCompile("(STDOUT|STDERR)-[0-9]+$"),
605 logFileTrimInfoRE: regexp.MustCompile(appName + `\..*\.INFO\.[0-9.-]+$`),
606 logFileRemoveErrorFatalWarningRE: regexp.MustCompile("(ERROR|FATAL|WARNING)"),
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700607 statsTrimRE: regexp.MustCompile("/stats/(rpc|system(/start-time.*)?)$"),
Robert Kroeger95f4e682015-03-13 10:40:26 -0700608 }
609}
610
Robert Kroeger22fcb032015-04-30 07:40:11 -0700611// VerifyGlob verifies that for each GlobTestVector instance that the
Robert Kroeger95f4e682015-03-13 10:40:26 -0700612// pattern returns the expected matches.
Robert Kroeger22fcb032015-04-30 07:40:11 -0700613func VerifyGlob(t *testing.T, ctx *context.T, appName string, testcases []GlobTestVector, res *globTestRegexHelper) {
Robert Kroeger95f4e682015-03-13 10:40:26 -0700614 for _, tc := range testcases {
Robert Kroeger22fcb032015-04-30 07:40:11 -0700615 results, _, err := testutil.GlobName(ctx, tc.Name, tc.Pattern)
Robert Kroeger95f4e682015-03-13 10:40:26 -0700616 if err != nil {
Robert Kroeger22fcb032015-04-30 07:40:11 -0700617 t.Errorf(testutil.FormatLogLine(2, "unexpected glob error for (%q, %q): %v", tc.Name, tc.Pattern, err))
Robert Kroeger95f4e682015-03-13 10:40:26 -0700618 continue
619 }
620 filteredResults := []string{}
621 for _, name := range results {
622 // Keep only the stats object names that match this RE.
623 if strings.Contains(name, "/stats/") && !res.statsTrimRE.MatchString(name) {
624 continue
625 }
626 // Remove ERROR, WARNING, FATAL log files because
627 // they're not consistently there.
628 if res.logFileRemoveErrorFatalWarningRE.MatchString(name) {
629 continue
630 }
631 name = res.logFileTimeStampRE.ReplaceAllString(name, "$1-<timestamp>")
632 name = res.logFileTrimInfoRE.ReplaceAllString(name, appName+".<*>.INFO.<timestamp>")
633 filteredResults = append(filteredResults, name)
634 }
635 sort.Strings(filteredResults)
Robert Kroeger22fcb032015-04-30 07:40:11 -0700636 sort.Strings(tc.Expected)
637 if !reflect.DeepEqual(filteredResults, tc.Expected) {
638 t.Errorf(testutil.FormatLogLine(2, "unexpected result for (%q, %q). Got %q, want %q", tc.Name, tc.Pattern, filteredResults, tc.Expected))
Robert Kroeger95f4e682015-03-13 10:40:26 -0700639 }
640 }
641}
642
Robert Kroeger22fcb032015-04-30 07:40:11 -0700643// VerifyFailGlob verifies that for each GlobTestVector instance that the
Robert Kroeger8f914be2015-03-14 16:32:37 -0700644// pattern returns no matches.
Robert Kroeger22fcb032015-04-30 07:40:11 -0700645func VerifyFailGlob(t *testing.T, ctx *context.T, testcases []GlobTestVector) {
Robert Kroeger8f914be2015-03-14 16:32:37 -0700646 for _, tc := range testcases {
Robert Kroeger22fcb032015-04-30 07:40:11 -0700647 results, _, _ := testutil.GlobName(ctx, tc.Name, tc.Pattern)
Robert Kroeger8f914be2015-03-14 16:32:37 -0700648 if len(results) != 0 {
Robert Kroeger22fcb032015-04-30 07:40:11 -0700649 t.Errorf(testutil.FormatLogLine(2, "verifyFailGlob should have failed for %q, %q", tc.Name, tc.Pattern))
Robert Kroeger8f914be2015-03-14 16:32:37 -0700650 }
651 }
652}
653
Robert Kroeger22fcb032015-04-30 07:40:11 -0700654// VerifyLog calls Size() on a selection of log file objects to
Robert Kroeger95f4e682015-03-13 10:40:26 -0700655// demonstrate that the log files are accessible and have been written by
656// the application.
Robert Kroeger22fcb032015-04-30 07:40:11 -0700657func VerifyLog(t *testing.T, ctx *context.T, nameComponents ...string) {
Robert Kroeger8f914be2015-03-14 16:32:37 -0700658 a := nameComponents
659 pattern, prefix := a[len(a)-1], a[:len(a)-1]
660 path := naming.Join(prefix...)
661 files, _, err := testutil.GlobName(ctx, path, pattern)
Robert Kroeger95f4e682015-03-13 10:40:26 -0700662 if err != nil {
663 t.Errorf(testutil.FormatLogLine(2, "unexpected glob error: %v", err))
664 }
665 if want, got := 4, len(files); got < want {
666 t.Errorf(testutil.FormatLogLine(2, "Unexpected number of matches. Got %d, want at least %d", got, want))
667 }
668 for _, file := range files {
Robert Kroeger8f914be2015-03-14 16:32:37 -0700669 name := naming.Join(path, file)
Robert Kroeger95f4e682015-03-13 10:40:26 -0700670 c := logreader.LogFileClient(name)
671 if _, err := c.Size(ctx); err != nil {
672 t.Errorf(testutil.FormatLogLine(2, "Size(%q) failed: %v", name, err))
673 }
674 }
675}
676
Robert Kroeger22fcb032015-04-30 07:40:11 -0700677// VerifyStatsValues call Value() on some of the stats objects to prove
Robert Kroeger95f4e682015-03-13 10:40:26 -0700678// that they are correctly being proxied to the device manager.
Robert Kroeger22fcb032015-04-30 07:40:11 -0700679func VerifyStatsValues(t *testing.T, ctx *context.T, nameComponents ...string) {
Robert Kroeger8f914be2015-03-14 16:32:37 -0700680 a := nameComponents
681 pattern, prefix := a[len(a)-1], a[:len(a)-1]
682 path := naming.Join(prefix...)
683 objects, _, err := testutil.GlobName(ctx, path, pattern)
684
Robert Kroeger95f4e682015-03-13 10:40:26 -0700685 if err != nil {
686 t.Errorf(testutil.FormatLogLine(2, "unexpected glob error: %v", err))
687 }
688 if want, got := 2, len(objects); got != want {
689 t.Errorf(testutil.FormatLogLine(2, "Unexpected number of matches. Got %d, want %d", got, want))
690 }
691 for _, obj := range objects {
Robert Kroeger8f914be2015-03-14 16:32:37 -0700692 name := naming.Join(path, obj)
Robert Kroeger95f4e682015-03-13 10:40:26 -0700693 c := stats.StatsClient(name)
694 if _, err := c.Value(ctx); err != nil {
695 t.Errorf(testutil.FormatLogLine(2, "Value(%q) failed: %v", name, err))
696 }
697 }
698}
699
Robert Kroeger22fcb032015-04-30 07:40:11 -0700700// VerifyPProfCmdLine calls CmdLine() on the pprof object to validate
Robert Kroeger95f4e682015-03-13 10:40:26 -0700701// that it the proxy correctly accessess pprof names.
Robert Kroeger22fcb032015-04-30 07:40:11 -0700702func VerifyPProfCmdLine(t *testing.T, ctx *context.T, appName string, nameComponents ...string) {
Robert Kroeger95f4e682015-03-13 10:40:26 -0700703 name := naming.Join(nameComponents...)
704 c := pprof.PProfClient(name)
705 v, err := c.CmdLine(ctx)
706 if err != nil {
707 t.Errorf(testutil.FormatLogLine(2, "CmdLine(%q) failed: %v", name, err))
708 }
709 if len(v) == 0 {
Robert Kroeger8c2be902015-06-19 15:33:45 -0700710 t.Errorf("Unexpected empty cmdline: %v", v)
Robert Kroeger95f4e682015-03-13 10:40:26 -0700711 }
712 if got, want := filepath.Base(v[0]), appName; got != want {
713 t.Errorf(testutil.FormatLogLine(2, "Unexpected value for argv[0]. Got %v, want %v", got, want))
714 }
715
716}
Robert Kroeger5e273202015-04-06 16:56:38 -0700717
Robert Kroeger22fcb032015-04-30 07:40:11 -0700718func VerifyNoRunningProcesses(t *testing.T) {
Robert Kroeger5e273202015-04-06 16:56:38 -0700719 if impl.RunningChildrenProcesses() {
720 t.Errorf("device manager incorrectly terminating with child processes still running")
721 }
722}
Bogdan Capritae8073682015-04-25 15:37:53 -0700723
Robert Kroeger22fcb032015-04-30 07:40:11 -0700724func SetNamespaceRootsForUnclaimedDevice(ctx *context.T) (*context.T, error) {
Bogdan Capritae8073682015-04-25 15:37:53 -0700725 origroots := v23.GetNamespace(ctx).Roots()
726 roots := make([]string, len(origroots))
727 for i, orig := range origroots {
728 addr, suffix := naming.SplitAddressName(orig)
729 origep, err := v23.NewEndpoint(addr)
730 if err != nil {
731 return nil, err
732 }
733 ep := naming.FormatEndpoint(
734 origep.Addr().Network(),
735 origep.Addr().String(),
736 origep.RoutingID(),
737 naming.ServesMountTable(origep.ServesMountTable()))
738 roots[i] = naming.JoinAddressName(ep, suffix)
739 }
Cosmos Nicolaou18489732015-06-29 16:01:38 -0700740 ctx.Infof("Changing namespace roots from %v to %v", origroots, roots)
Bogdan Capritae8073682015-04-25 15:37:53 -0700741 ctx, _, err := v23.WithNewNamespace(ctx, roots...)
742 return ctx, err
743}
Robert Kroegeref27d6e2015-05-01 16:58:59 -0700744
745func UserName(t *testing.T) string {
746 u, err := user.Current()
747 if err != nil {
748 t.Fatalf("user.Current() failed: %v", err)
749 }
750 return u.Username
751}
Robert Kroeger6e2efd92015-05-01 18:47:28 -0700752
Robert Kroeger60e03052015-05-04 17:35:16 -0700753func StartRealBinaryRepository(t *testing.T, ctx *context.T, von string) func() {
754 rootDir, err := binarylib.SetupRootDir("")
755 if err != nil {
756 t.Fatalf("binarylib.SetupRootDir failed: %v", err)
757 }
758 state, err := binarylib.NewState(rootDir, "", 3)
759 if err != nil {
760 t.Fatalf("binarylib.NewState failed: %v", err)
761 }
Cosmos Nicolaou7a4221f2015-06-21 08:02:23 -0700762 d, err := binarylib.NewDispatcher(ctx, state)
Robert Kroeger60e03052015-05-04 17:35:16 -0700763 if err != nil {
764 t.Fatalf("server.NewDispatcher failed: %v", err)
765 }
Matt Rosencrantzbb6295d2015-06-19 15:13:58 -0700766 server, err := xrpc.NewDispatchingServer(ctx, von, d)
767 if err != nil {
Robert Kroeger60e03052015-05-04 17:35:16 -0700768 t.Fatalf("server.ServeDispatcher failed: %v", err)
769 }
770 return func() {
771 if err := server.Stop(); err != nil {
772 t.Fatalf("server.Stop failed: %v", err)
773 }
774 if err := os.RemoveAll(rootDir); err != nil {
775 t.Fatalf("os.RemoveAll(%q) failed: %v", rootDir, err)
776 }
777 }
778}
Robert Kroeger9e10d4f2015-05-07 16:33:04 -0700779
780func GetPid(t *testing.T, ctx *context.T, appID, instanceID string) int {
781 name := naming.Join("dm", "apps/"+appID+"/"+instanceID+"/stats/system/pid")
782 c := stats.StatsClient(name)
783 v, err := c.Value(ctx)
784 if err != nil {
785 t.Fatalf("Value() failed: %v", err)
786 }
787 return int(v.Int())
788}
Robert Kroeger4740f882015-06-11 16:26:13 -0700789
790// PollingWait polls a given process to make sure that it has exited
791// before continuing or fails with a time-out.
792func PollingWait(t *testing.T, pid int) {
793 timeOut := time.After(5 * time.Second)
794 for syscall.Kill(pid, 0) == nil {
795 select {
796 case <-timeOut:
Robert Kroeger8c2be902015-06-19 15:33:45 -0700797 syscall.Kill(pid, 9)
Robert Kroeger4740f882015-06-11 16:26:13 -0700798 t.Fatalf("Timed out waiting for PID %v to terminate", pid)
799 case <-time.After(time.Millisecond):
800 // Try again.
801 }
802 }
803}