blob: 742439ce119b12fbc96803ec12c9993bda4e3caa [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
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -08005// Test the device manager and related services and tools.
6//
7// By default, this script tests the device manager in a fashion amenable
8// to automatic testing: the --single_user is passed to the device
9// manager so that all device manager components run as the same user and
10// no user input (such as an agent pass phrase) is needed.
11//
12// When this script is invoked with the --with_suid <user> flag, it
13// installs the device manager in its more secure multi-account
14// configuration where the device manager runs under the account of the
15// invoker and test apps will be executed as <user>. This mode will
16// require root permisisons to install and may require configuring an
17// agent passphrase.
18//
19// For exanple:
20//
21// v23 go test -v . --v23.tests --with_suid vanaguest
22//
23// to test a device manager with multi-account support enabled for app
24// account vanaguest.
25//
Todd Wangcd4b3cc2015-04-06 16:42:02 -070026package device_test
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -080027
28//go:generate v23 test generate .
29
30import (
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -080031 "errors"
32 "flag"
33 "fmt"
34 "io/ioutil"
35 "math/rand"
36 "os"
37 "path/filepath"
38 "strings"
39 "time"
40
Asim Shankar59b8b692015-03-30 01:23:36 -070041 "v.io/x/ref/envvar"
Jiri Simsaffceefa2015-02-28 11:03:34 -080042 _ "v.io/x/ref/profiles"
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -070043 "v.io/x/ref/test/v23tests"
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -080044)
45
46var (
47 suidUserFlag string
48 hostname string
49 errTimeout = errors.New("timeout")
50)
51
52func init() {
53 flag.StringVar(&suidUserFlag, "with_suid", "", "run the device manager as the specified user")
54 name, err := os.Hostname()
55 if err != nil {
56 panic(fmt.Sprintf("Hostname() failed: %v", err))
57 }
58 hostname = name
59}
60
Asim Shankar6cc759d2015-03-14 03:31:44 -070061func V23TestDeviceManager(i *v23tests.T) {
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -080062 defer fmt.Fprintf(os.Stderr, "--------------- SHUTDOWN ---------------\n")
63 userFlag := "--single_user"
64 withSuid := false
65 if len(suidUserFlag) > 0 {
66 userFlag = "--with_suid=" + suidUserFlag
67 withSuid = true
68 }
69 i.Logf("user flag: %q", userFlag)
70
Asim Shankar6cc759d2015-03-14 03:31:44 -070071 var (
72 workDir = i.NewTempDir()
73 binStagingDir = mkSubdir(i, workDir, "bin")
74 dmInstallDir = filepath.Join(workDir, "dm")
75
76 // All vanadium command-line utitilities will be run by a
77 // principal that has "root/alice" as its blessing.
78 // (Where "root" comes from i.Principal().BlessingStore().Default()).
79 // Create those credentials and options to use to setup the
80 // binaries with them.
81 aliceCreds, _ = i.Shell().NewChildCredentials("alice")
82 aliceOpts = i.Shell().DefaultStartOpts().ExternalCommand().WithCustomCredentials(aliceCreds)
83
84 // Build all the command-line tools and set them up to run as alice.
85 // applicationd/binaryd servers will be run by alice too.
86 namespaceBin = i.BuildV23Pkg("v.io/x/ref/cmd/namespace").WithStartOpts(aliceOpts)
Todd Wang4aaf8fa2015-04-03 18:14:26 -070087 debugBin = i.BuildV23Pkg("v.io/x/ref/services/debug/debug").WithStartOpts(aliceOpts)
Todd Wangcd4b3cc2015-04-06 16:42:02 -070088 deviceBin = i.BuildV23Pkg("v.io/x/ref/services/device/device").WithStartOpts(aliceOpts)
Todd Wang1f7a6c62015-04-03 17:05:09 -070089 binaryBin = i.BuildV23Pkg("v.io/x/ref/services/binary/binary").WithStartOpts(aliceOpts)
Todd Wang159f6ee2015-04-02 18:57:46 -070090 applicationBin = i.BuildV23Pkg("v.io/x/ref/services/application/application").WithStartOpts(aliceOpts)
Todd Wang1f7a6c62015-04-03 17:05:09 -070091 binarydBin = i.BuildV23Pkg("v.io/x/ref/services/binary/binaryd").WithStartOpts(aliceOpts)
Todd Wang159f6ee2015-04-02 18:57:46 -070092 applicationdBin = i.BuildV23Pkg("v.io/x/ref/services/application/applicationd").WithStartOpts(aliceOpts)
Asim Shankar6cc759d2015-03-14 03:31:44 -070093
94 // The devicex script is not provided with any credentials, it
95 // will generate its own. This means that on "devicex start"
96 // the device will have no useful credentials and until "device
97 // claim" is invoked (as alice), it will just sit around
98 // waiting to be claimed.
99 //
100 // Other binaries, like applicationd and binaryd will be run by alice.
Bogdan Capritacb055fa2015-04-08 10:21:23 -0700101 deviceScript = i.BinaryFromPath("./devicex").WithEnv("V23_DEVICE_DIR=" + dmInstallDir)
Asim Shankar6cc759d2015-03-14 03:31:44 -0700102
103 mtName = "devices/" + hostname // Name under which the device manager will publish itself.
104 )
105
Asim Shankarf32d24d2015-04-01 16:34:26 -0700106 v23tests.RunRootMT(i, "--v23.tcp.address=127.0.0.1:0")
Asim Shankar6cc759d2015-03-14 03:31:44 -0700107 buildAndCopyBinaries(
108 i,
109 binStagingDir,
Todd Wangcd4b3cc2015-04-06 16:42:02 -0700110 "v.io/x/ref/services/device/deviced",
Todd Wangb3511492015-04-07 23:32:34 -0700111 "v.io/x/ref/services/agent/agentd",
Todd Wang392a9cc2015-04-06 14:35:11 -0700112 "v.io/x/ref/services/device/suidhelper",
113 "v.io/x/ref/services/device/inithelper")
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800114
115 appDName := "applicationd"
116 devicedAppName := filepath.Join(appDName, "deviced", "test")
117
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800118 deviceScript.Start(
119 "install",
120 binStagingDir,
121 userFlag,
122 "--origin="+devicedAppName,
123 "--",
Asim Shankarf32d24d2015-04-01 16:34:26 -0700124 "--v23.tcp.address=127.0.0.1:0",
Suharsh Sivakumarf77f91f2015-04-01 19:17:16 -0700125 "--neighborhood-name="+fmt.Sprintf("%s-%d-%d", hostname, os.Getpid(), rand.Int())).
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800126 WaitOrDie(os.Stdout, os.Stderr)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800127 deviceScript.Start("start").WaitOrDie(os.Stdout, os.Stderr)
128
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800129 resolve := func(name string) string {
130 resolver := func() (interface{}, error) {
Asim Shankar6cc759d2015-03-14 03:31:44 -0700131 // Use Start, rather than Run, since it's ok for 'namespace resolve'
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800132 // to fail with 'name doesn't exist'
133 inv := namespaceBin.Start("resolve", name)
134 // Cleanup after ourselves to avoid leaving a ton of invocations
135 // lying around which obscure logging output.
136 defer inv.Wait(nil, os.Stderr)
137 if r := strings.TrimRight(inv.Output(), "\n"); len(r) > 0 {
138 return r, nil
139 }
140 return nil, nil
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800141 }
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800142 return i.WaitFor(resolver, 100*time.Millisecond, time.Minute).(string)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800143 }
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800144 mtEP := resolve(mtName)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800145
146 // Verify that device manager's mounttable is published under the expected
147 // name (hostname).
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800148 if got := namespaceBin.Run("glob", mtName); len(got) == 0 {
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800149 i.Fatalf("glob failed for %q", mtName)
150 }
151
Asim Shankar6cc759d2015-03-14 03:31:44 -0700152 // Claim the device as "root/alice/myworkstation".
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800153 deviceBin.Start("claim", mtName+"/devmgr/device", "myworkstation")
154
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800155 resolveChange := func(name, old string) string {
156 resolver := func() (interface{}, error) {
157 inv := namespaceBin.Start("resolve", name)
158 defer inv.Wait(nil, os.Stderr)
Suharsh Sivakumarb8958052015-02-27 16:50:52 -0800159 if r := strings.TrimRight(inv.Output(), "\n"); len(r) > 0 && r != old {
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800160 return r, nil
161 }
162 return nil, nil
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800163 }
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800164 return i.WaitFor(resolver, 100*time.Millisecond, time.Minute).(string)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800165 }
166
167 // Wait for the device manager to update its mount table entry.
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800168 mtEP = resolveChange(mtName, mtEP)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800169
170 if withSuid {
171 /*
172 "${DEVICE_BIN}" associate add "${MT_NAME}/devmgr/device" "${SUID_USER}" "alice"
173 shell_test::assert_eq "$("${DEVICE_BIN}" associate list "${MT_NAME}/devmgr/device")" \
174 "alice ${SUID_USER}" "${LINENO}"
175 */
176 }
177
178 // Verify the device's default blessing is as expected.
179 inv := debugBin.Start("stats", "read", mtName+"/devmgr/__debug/stats/security/principal/*/blessingstore")
Suharsh Sivakumar4bbe8ed2015-04-09 14:21:44 -0700180 inv.ExpectSetEventuallyRE(".*Default Blessings[ ]+root/alice/myworkstation$")
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800181
182 // Get the device's profile, which should be set to non-empty string
183 inv = deviceBin.Start("describe", mtName+"/devmgr/device")
184
185 parts := inv.ExpectRE(`{Profiles:map\[(.*):{}\]}`, 1)
186 expectOneMatch := func(parts [][]string) string {
187 if len(parts) != 1 || len(parts[0]) != 2 {
188 loc := v23tests.Caller(1)
189 i.Fatalf("%s: failed to match profile: %#v", loc, parts)
190 }
191 return parts[0][1]
192 }
193 deviceProfile := expectOneMatch(parts)
194 if len(deviceProfile) == 0 {
195 i.Fatalf("failed to get profile")
196 }
197
Asim Shankar6cc759d2015-03-14 03:31:44 -0700198 // Start a binaryd server that will serve the binary for the test
199 // application to be installed on the device.
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800200 binarydName := "binaryd"
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800201 binarydBin.Start(
202 "--name="+binarydName,
Suharsh Sivakumar65e38502015-04-01 18:33:18 -0700203 "--root-dir="+filepath.Join(workDir, "binstore"),
Asim Shankarf32d24d2015-04-01 16:34:26 -0700204 "--v23.tcp.address=127.0.0.1:0",
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800205 "--http=127.0.0.1:0")
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800206 sampleAppBinName := binarydName + "/testapp"
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800207 binaryBin.Run("upload", sampleAppBinName, binarydBin.Path())
Asim Shankar6cc759d2015-03-14 03:31:44 -0700208 if got := namespaceBin.Run("glob", sampleAppBinName); len(got) == 0 {
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800209 i.Fatalf("glob failed for %q", sampleAppBinName)
210 }
211
Asim Shankar6cc759d2015-03-14 03:31:44 -0700212 // Start an applicationd server that will serve the application
213 // envelope for the test application to be installed on the device.
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800214 applicationdBin.Start(
215 "--name="+appDName,
Asim Shankar6cc759d2015-03-14 03:31:44 -0700216 "--store="+mkSubdir(i, workDir, "appstore"),
Asim Shankarf32d24d2015-04-01 16:34:26 -0700217 "--v23.tcp.address=127.0.0.1:0",
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800218 )
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800219 sampleAppName := appDName + "/testapp/v0"
220 appPubName := "testbinaryd"
221 appEnvelopeFilename := filepath.Join(workDir, "app.envelope")
Suharsh Sivakumar65e38502015-04-01 18:33:18 -0700222 appEnvelope := fmt.Sprintf("{\"Title\":\"BINARYD\", \"Args\":[\"--name=%s\", \"--root-dir=./binstore\", \"--v23.tcp.address=127.0.0.1:0\", \"--http=127.0.0.1:0\"], \"Binary\":{\"File\":%q}, \"Env\":[]}", appPubName, sampleAppBinName)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800223 ioutil.WriteFile(appEnvelopeFilename, []byte(appEnvelope), 0666)
224 defer os.Remove(appEnvelopeFilename)
225
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800226 output := applicationBin.Run("put", sampleAppName, deviceProfile, appEnvelopeFilename)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800227 if got, want := output, "Application envelope added successfully."; got != want {
228 i.Fatalf("got %q, want %q", got, want)
229 }
230
231 // Verify that the envelope we uploaded shows up with glob.
232 inv = applicationBin.Start("match", sampleAppName, deviceProfile)
233 parts = inv.ExpectSetEventuallyRE(`"Title": "(.*)",`, `"File": "(.*)",`)
234 if got, want := len(parts), 2; got != want {
235 i.Fatalf("got %d, want %d", got, want)
236 }
237 for line, want := range []string{"BINARYD", sampleAppBinName} {
238 if got := parts[line][1]; got != want {
239 i.Fatalf("got %q, want %q", got, want)
240 }
241 }
242
243 // Install the app on the device.
244 inv = deviceBin.Start("install", mtName+"/devmgr/apps", sampleAppName)
Bogdan Caprita2b050322015-04-17 09:04:03 -0700245 installationName := inv.ReadLine()
246 if installationName == "" {
247 i.Fatalf("got empty installation name from install")
248 }
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800249
250 // Verify that the installation shows up when globbing the device manager.
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800251 output = namespaceBin.Run("glob", mtName+"/devmgr/apps/BINARYD/*")
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800252 if got, want := output, installationName; got != want {
253 i.Fatalf("got %q, want %q", got, want)
254 }
255
256 // Start an instance of the app, granting it blessing extension myapp.
Bogdan Caprita2b050322015-04-17 09:04:03 -0700257 inv = deviceBin.Start("instantiate", installationName, "myapp")
258 instanceName := inv.ReadLine()
259 if instanceName == "" {
260 i.Fatalf("got empty instance name from new")
261 }
262 deviceBin.Start("run", instanceName)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800263
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800264 resolve(mtName + "/" + appPubName)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800265
266 // Verify that the instance shows up when globbing the device manager.
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800267 output = namespaceBin.Run("glob", mtName+"/devmgr/apps/BINARYD/*/*")
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800268 if got, want := output, instanceName; got != want {
269 i.Fatalf("got %q, want %q", got, want)
270 }
271
272 // TODO(rjkroege): Verify that the app is actually running as ${SUID_USER}
273
274 // Verify the app's default blessing.
275 inv = debugBin.Start("stats", "read", instanceName+"/stats/security/principal/*/blessingstore")
Suharsh Sivakumar4bbe8ed2015-04-09 14:21:44 -0700276 inv.ExpectSetEventuallyRE(".*Default Blessings[ ]+root/alice/myapp$")
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800277
Bogdan Caprita2b050322015-04-17 09:04:03 -0700278 // Kill and delete the instance.
279 deviceBin.Run("kill", instanceName)
280 deviceBin.Run("delete", instanceName)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800281
282 // Verify that logs, but not stats, show up when globbing the
Bogdan Caprita2b050322015-04-17 09:04:03 -0700283 // not-running instance.
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800284 if output = namespaceBin.Run("glob", instanceName+"/stats/..."); len(output) > 0 {
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800285 i.Fatalf("no output expected for glob %s/stats/..., got %q", output, instanceName)
286 }
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800287 if output = namespaceBin.Run("glob", instanceName+"/logs/..."); len(output) == 0 {
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800288 i.Fatalf("output expected for glob %s/logs/..., but got none", instanceName)
289 }
290
291 // Upload a deviced binary
292 devicedAppBinName := binarydName + "/deviced"
Todd Wangcd4b3cc2015-04-06 16:42:02 -0700293 binaryBin.Run("upload", devicedAppBinName, i.BuildGoPkg("v.io/x/ref/services/device/deviced").Path())
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800294
Asim Shankar6cc759d2015-03-14 03:31:44 -0700295 // Upload a device manager envelope.
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800296 devicedEnvelopeFilename := filepath.Join(workDir, "deviced.envelope")
Asim Shankar6cc759d2015-03-14 03:31:44 -0700297 devicedEnvelope := fmt.Sprintf("{\"Title\":\"device manager\", \"Binary\":{\"File\":%q}}", devicedAppBinName)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800298 ioutil.WriteFile(devicedEnvelopeFilename, []byte(devicedEnvelope), 0666)
299 defer os.Remove(devicedEnvelopeFilename)
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800300 applicationBin.Run("put", devicedAppName, deviceProfile, devicedEnvelopeFilename)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800301
302 // Update the device manager.
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800303 deviceBin.Run("update", mtName+"/devmgr/device")
304 mtEP = resolveChange(mtName, mtEP)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800305
306 // Verify that device manager's mounttable is still published under the
307 // expected name (hostname).
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800308 if namespaceBin.Run("glob", mtName) == "" {
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800309 i.Fatalf("failed to glob %s", mtName)
310 }
311
312 // Revert the device manager
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800313 deviceBin.Run("revert", mtName+"/devmgr/device")
314 mtEP = resolveChange(mtName, mtEP)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800315
316 // Verify that device manager's mounttable is still published under the
317 // expected name (hostname).
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800318 if namespaceBin.Run("glob", mtName) == "" {
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800319 i.Fatalf("failed to glob %s", mtName)
320 }
321
322 // Verify that the local mounttable exists, and that the device manager,
323 // the global namespace, and the neighborhood are mounted on it.
324 n := mtEP + "/devmgr"
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800325 if namespaceBin.Run("resolve", n) == "" {
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800326 i.Fatalf("failed to resolve %s", n)
327 }
328 n = mtEP + "/nh"
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800329 if namespaceBin.Run("resolve", n) == "" {
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800330 i.Fatalf("failed to resolve %s", n)
331 }
Asim Shankar59b8b692015-03-30 01:23:36 -0700332 namespaceRoot, _ := i.GetVar(envvar.NamespacePrefix)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800333 n = mtEP + "/global"
Asim Shankar43d1f932015-03-24 20:57:56 -0700334 if got, want := namespaceBin.Run("resolve", n), namespaceRoot; got != want {
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800335 i.Fatalf("got %q, want %q", got, want)
336 }
337
Bogdan Caprita2b050322015-04-17 09:04:03 -0700338 // Kill the device manager (which causes it to be restarted), wait for
339 // the endpoint to change.
340 deviceBin.Run("kill", mtName+"/devmgr/device")
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800341 mtEP = resolveChange(mtName, mtEP)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800342
Bogdan Caprita2b050322015-04-17 09:04:03 -0700343 // Shut down the device manager.
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800344 deviceScript.Run("stop")
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800345
346 // Wait for the mounttable entry to go away.
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800347 resolveGone := func(name string) string {
348 resolver := func() (interface{}, error) {
349 inv := namespaceBin.Start("resolve", name)
350 defer inv.Wait(nil, os.Stderr)
351 if r := strings.TrimRight(inv.Output(), "\n"); len(r) == 0 {
352 return r, nil
353 }
354 return nil, nil
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800355 }
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800356 return i.WaitFor(resolver, 100*time.Millisecond, time.Minute).(string)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800357 }
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800358 resolveGone(mtName)
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800359
360 fi, err := ioutil.ReadDir(dmInstallDir)
361 if err != nil {
362 i.Fatalf("failed to readdir for %q: %v", dmInstallDir, err)
363 }
364
Cosmos Nicolaouabbcb202015-02-26 11:24:52 -0800365 deviceScript.Run("uninstall")
Cosmos Nicolaou93dd88b2015-02-19 15:10:53 -0800366
367 fi, err = ioutil.ReadDir(dmInstallDir)
368 if err == nil || len(fi) > 0 {
369 i.Fatalf("managed to read %d entries from %q", len(fi), dmInstallDir)
370 }
371 if err != nil && !strings.Contains(err.Error(), "no such file or directory") {
372 i.Fatalf("wrong error: %v", err)
373 }
374}
Asim Shankar6cc759d2015-03-14 03:31:44 -0700375
376func buildAndCopyBinaries(i *v23tests.T, destinationDir string, packages ...string) {
377 var args []string
378 for _, pkg := range packages {
379 args = append(args, i.BuildGoPkg(pkg).Path())
380 }
381 args = append(args, destinationDir)
382 i.BinaryFromPath("/bin/cp").Start(args...).WaitOrDie(os.Stdout, os.Stderr)
383}
384
385func mkSubdir(i *v23tests.T, parent, child string) string {
386 dir := filepath.Join(parent, child)
387 if err := os.Mkdir(dir, 0755); err != nil {
388 i.Fatalf("failed to create %q: %v", dir, err)
389 }
390 return dir
391}