blob: 1d94d6786202a2b9e3e926326b0f38634d45d7ec [file] [log] [blame]
Mike Frysingere58c0e22017-10-04 15:43:30 -04001# -*- coding: utf-8 -*-
Ryan Cuiafd6c5c2012-07-30 17:48:22 -07002# Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
3# Use of this source code is governed by a BSD-style license that can be
4# found in the LICENSE file.
5
Steve Funge984a532013-11-25 17:09:25 -08006"""Unit tests for the deploy_chrome script."""
Ryan Cuiafd6c5c2012-07-30 17:48:22 -07007
Mike Frysinger383367e2014-09-16 15:06:17 -04008from __future__ import print_function
9
Mike Frysingerea838d12014-12-08 11:55:32 -050010import mock
Ryan Cuiafd6c5c2012-07-30 17:48:22 -070011import os
David James88e6f032013-03-02 08:13:20 -080012import time
Ryan Cuiafd6c5c2012-07-30 17:48:22 -070013
David Pursellcfd58872015-03-19 09:15:48 -070014from chromite.cli.cros import cros_chrome_sdk_unittest
Ryan Cuief91e702013-02-04 12:06:36 -080015from chromite.lib import chrome_util
Ryan Cuiafd6c5c2012-07-30 17:48:22 -070016from chromite.lib import cros_build_lib
17from chromite.lib import cros_test_lib
Ryan Cui686ec052013-02-12 16:39:41 -080018from chromite.lib import osutils
Ryan Cuiafd6c5c2012-07-30 17:48:22 -070019from chromite.lib import partial_mock
Robert Flack1dc7ea82014-11-26 13:50:24 -050020from chromite.lib import remote_access
Ryan Cuiafd6c5c2012-07-30 17:48:22 -070021from chromite.lib import remote_access_unittest
22from chromite.scripts import deploy_chrome
23
Ryan Cuief91e702013-02-04 12:06:36 -080024
Ryan Cuiafd6c5c2012-07-30 17:48:22 -070025# pylint: disable=W0212
26
27_REGULAR_TO = ('--to', 'monkey')
28_GS_PATH = 'gs://foon'
29
30
31def _ParseCommandLine(argv):
32 return deploy_chrome._ParseCommandLine(['--log-level', 'debug'] + argv)
33
34
35class InterfaceTest(cros_test_lib.OutputTestCase):
36 """Tests the commandline interface of the script."""
37
Bernie Thompson93b9ee62018-02-21 14:56:16 -080038 BOARD = 'eve'
Ryan Cui686ec052013-02-12 16:39:41 -080039
Ryan Cuiafd6c5c2012-07-30 17:48:22 -070040 def testGsLocalPathUnSpecified(self):
41 """Test no chrome path specified."""
42 with self.OutputCapturer():
43 self.assertRaises2(SystemExit, _ParseCommandLine, list(_REGULAR_TO),
44 check_attrs={'code': 2})
45
46 def testGsPathSpecified(self):
47 """Test case of GS path specified."""
48 argv = list(_REGULAR_TO) + ['--gs-path', _GS_PATH]
49 _ParseCommandLine(argv)
50
51 def testLocalPathSpecified(self):
52 """Test case of local path specified."""
Mike Frysingerd6e2df02014-11-26 02:55:04 -050053 argv = list(_REGULAR_TO) + ['--local-pkg-path', '/path/to/chrome']
Ryan Cuiafd6c5c2012-07-30 17:48:22 -070054 _ParseCommandLine(argv)
55
56 def testNoTarget(self):
57 """Test no target specified."""
58 argv = ['--gs-path', _GS_PATH]
Ryan Cuief91e702013-02-04 12:06:36 -080059 self.assertParseError(argv)
60
61 def assertParseError(self, argv):
Ryan Cuiafd6c5c2012-07-30 17:48:22 -070062 with self.OutputCapturer():
63 self.assertRaises2(SystemExit, _ParseCommandLine, argv,
64 check_attrs={'code': 2})
65
Achuith Bhandarkar31a3eb02018-03-22 16:33:48 -070066 def testNoBoard(self):
67 """Test cases where --board is not specified."""
Ryan Cui686ec052013-02-12 16:39:41 -080068 argv = ['--staging-only', '--build-dir=/path/to/nowhere']
69 self.assertParseError(argv)
70
Achuith Bhandarkar31a3eb02018-03-22 16:33:48 -070071 # Don't need --board if no stripping is necessary.
72 argv_nostrip = argv + ['--nostrip']
73 _ParseCommandLine(argv_nostrip)
74
75 # Don't need --board if strip binary is provided.
76 argv_strip_bin = argv + ['--strip-bin', 'strip.bin']
77 _ParseCommandLine(argv_strip_bin)
78
Thiago Goncales12793312013-05-23 11:26:17 -070079 def testMountOptionSetsTargetDir(self):
80 argv = list(_REGULAR_TO) + ['--gs-path', _GS_PATH, '--mount']
Mike Frysingerc3061a62015-06-04 04:16:18 -040081 options = _ParseCommandLine(argv)
Thiago Goncales12793312013-05-23 11:26:17 -070082 self.assertIsNot(options.target_dir, None)
83
84 def testMountOptionSetsMountDir(self):
85 argv = list(_REGULAR_TO) + ['--gs-path', _GS_PATH, '--mount']
Mike Frysingerc3061a62015-06-04 04:16:18 -040086 options = _ParseCommandLine(argv)
Thiago Goncales12793312013-05-23 11:26:17 -070087 self.assertIsNot(options.mount_dir, None)
88
89 def testMountOptionDoesNotOverrideTargetDir(self):
90 argv = list(_REGULAR_TO) + ['--gs-path', _GS_PATH, '--mount',
91 '--target-dir', '/foo/bar/cow']
Mike Frysingerc3061a62015-06-04 04:16:18 -040092 options = _ParseCommandLine(argv)
Thiago Goncales12793312013-05-23 11:26:17 -070093 self.assertEqual(options.target_dir, '/foo/bar/cow')
94
95 def testMountOptionDoesNotOverrideMountDir(self):
96 argv = list(_REGULAR_TO) + ['--gs-path', _GS_PATH, '--mount',
97 '--mount-dir', '/foo/bar/cow']
Mike Frysingerc3061a62015-06-04 04:16:18 -040098 options = _ParseCommandLine(argv)
Thiago Goncales12793312013-05-23 11:26:17 -070099 self.assertEqual(options.mount_dir, '/foo/bar/cow')
100
Adrian Eldera2c548a2017-11-07 19:01:29 -0500101 def testSshIdentityOptionSetsOption(self):
102 argv = list(_REGULAR_TO) + ['--private-key', '/foo/bar/key',
103 '--board', 'cedar',
Bernie Thompson93b9ee62018-02-21 14:56:16 -0800104 '--build-dir', '/path/to/nowhere']
Adrian Eldera2c548a2017-11-07 19:01:29 -0500105 options = _ParseCommandLine(argv)
106 self.assertEqual(options.private_key, '/foo/bar/key')
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700107
108class DeployChromeMock(partial_mock.PartialMock):
Steve Funge984a532013-11-25 17:09:25 -0800109 """Deploy Chrome Mock Class."""
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700110
111 TARGET = 'chromite.scripts.deploy_chrome.DeployChrome'
David James88e6f032013-03-02 08:13:20 -0800112 ATTRS = ('_KillProcsIfNeeded', '_DisableRootfsVerification')
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700113
David James88e6f032013-03-02 08:13:20 -0800114 def __init__(self):
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700115 partial_mock.PartialMock.__init__(self)
Robert Flack1dc7ea82014-11-26 13:50:24 -0500116 self.remote_device_mock = remote_access_unittest.RemoteDeviceMock()
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700117 # Target starts off as having rootfs verification enabled.
Ryan Cuie18f24f2012-12-03 18:39:55 -0800118 self.rsh_mock = remote_access_unittest.RemoteShMock()
David James88e6f032013-03-02 08:13:20 -0800119 self.rsh_mock.SetDefaultCmdResult(0)
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700120 self.MockMountCmd(1)
David Haddock3151d912017-10-24 03:50:32 +0000121 self.rsh_mock.AddCmdResult(
122 deploy_chrome.LSOF_COMMAND % (deploy_chrome._CHROME_DIR,), 1)
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700123
124 def MockMountCmd(self, returnvalue):
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700125 self.rsh_mock.AddCmdResult(deploy_chrome.MOUNT_RW_COMMAND,
David James88e6f032013-03-02 08:13:20 -0800126 returnvalue)
127
128 def _DisableRootfsVerification(self, inst):
129 with mock.patch.object(time, 'sleep'):
130 self.backup['_DisableRootfsVerification'](inst)
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700131
Ryan Cui4d6fca92012-12-13 16:41:56 -0800132 def PreStart(self):
Robert Flack1dc7ea82014-11-26 13:50:24 -0500133 self.remote_device_mock.start()
Ryan Cui4d6fca92012-12-13 16:41:56 -0800134 self.rsh_mock.start()
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700135
Ryan Cui4d6fca92012-12-13 16:41:56 -0800136 def PreStop(self):
137 self.rsh_mock.stop()
Robert Flack1dc7ea82014-11-26 13:50:24 -0500138 self.remote_device_mock.stop()
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700139
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700140 def _KillProcsIfNeeded(self, _inst):
141 # Fully stub out for now.
142 pass
143
144
Ryan Cuief91e702013-02-04 12:06:36 -0800145class DeployTest(cros_test_lib.MockTempDirTestCase):
Steve Funge984a532013-11-25 17:09:25 -0800146 """Setup a deploy object with a GS-path for use in tests."""
147
Ryan Cuief91e702013-02-04 12:06:36 -0800148 def _GetDeployChrome(self, args):
Mike Frysingerc3061a62015-06-04 04:16:18 -0400149 options = _ParseCommandLine(args)
Ryan Cuia56a71e2012-10-18 18:40:35 -0700150 return deploy_chrome.DeployChrome(
151 options, self.tempdir, os.path.join(self.tempdir, 'staging'))
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700152
153 def setUp(self):
Ryan Cuif1416f32013-01-22 18:43:41 -0800154 self.deploy_mock = self.StartPatcher(DeployChromeMock())
Ryan Cuief91e702013-02-04 12:06:36 -0800155 self.deploy = self._GetDeployChrome(
David James88e6f032013-03-02 08:13:20 -0800156 list(_REGULAR_TO) + ['--gs-path', _GS_PATH, '--force'])
Luigi Semenzato1bc79b22016-11-22 16:32:17 -0800157 self.remote_reboot_mock = \
158 self.PatchObject(remote_access.RemoteAccess, 'RemoteReboot',
159 return_value=True)
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700160
David James88e6f032013-03-02 08:13:20 -0800161class TestDisableRootfsVerification(DeployTest):
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700162 """Testing disabling of rootfs verification and RO mode."""
163
David James88e6f032013-03-02 08:13:20 -0800164 def testDisableRootfsVerificationSuccess(self):
165 """Test the working case, disabling rootfs verification."""
166 self.deploy_mock.MockMountCmd(0)
167 self.deploy._DisableRootfsVerification()
Robert Flack1dc7ea82014-11-26 13:50:24 -0500168 self.assertFalse(self.deploy._target_dir_is_still_readonly.is_set())
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700169
170 def testDisableRootfsVerificationFailure(self):
171 """Test failure to disable rootfs verification."""
Luigi Semenzato1bc79b22016-11-22 16:32:17 -0800172 #pylint: disable=unused-argument
Shuqian Zhao14e61092017-11-17 00:02:16 +0000173 def RaiseRunCommandError(timeout_sec=None):
Luigi Semenzato1bc79b22016-11-22 16:32:17 -0800174 raise cros_build_lib.RunCommandError('Mock RunCommandError', 0)
175 self.remote_reboot_mock.side_effect = RaiseRunCommandError
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700176 self.assertRaises(cros_build_lib.RunCommandError,
David James88e6f032013-03-02 08:13:20 -0800177 self.deploy._DisableRootfsVerification)
Luigi Semenzato1bc79b22016-11-22 16:32:17 -0800178 self.remote_reboot_mock.side_effect = None
Robert Flack1dc7ea82014-11-26 13:50:24 -0500179 self.assertFalse(self.deploy._target_dir_is_still_readonly.is_set())
David James88e6f032013-03-02 08:13:20 -0800180
181
182class TestMount(DeployTest):
183 """Testing mount success and failure."""
184
185 def testSuccess(self):
186 """Test case where we are able to mount as writable."""
Robert Flack1dc7ea82014-11-26 13:50:24 -0500187 self.assertFalse(self.deploy._target_dir_is_still_readonly.is_set())
David James88e6f032013-03-02 08:13:20 -0800188 self.deploy_mock.MockMountCmd(0)
189 self.deploy._MountRootfsAsWritable()
Robert Flack1dc7ea82014-11-26 13:50:24 -0500190 self.assertFalse(self.deploy._target_dir_is_still_readonly.is_set())
David James88e6f032013-03-02 08:13:20 -0800191
192 def testMountError(self):
193 """Test that mount failure doesn't raise an exception by default."""
Robert Flack1dc7ea82014-11-26 13:50:24 -0500194 self.assertFalse(self.deploy._target_dir_is_still_readonly.is_set())
Mike Frysinger74ccd572015-05-21 21:18:20 -0400195 self.PatchObject(remote_access.RemoteDevice, 'IsDirWritable',
Robert Flack1dc7ea82014-11-26 13:50:24 -0500196 return_value=False, autospec=True)
David James88e6f032013-03-02 08:13:20 -0800197 self.deploy._MountRootfsAsWritable()
Robert Flack1dc7ea82014-11-26 13:50:24 -0500198 self.assertTrue(self.deploy._target_dir_is_still_readonly.is_set())
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700199
200 def testMountRwFailure(self):
David James88e6f032013-03-02 08:13:20 -0800201 """Test that mount failure raises an exception if error_code_ok=False."""
202 self.assertRaises(cros_build_lib.RunCommandError,
203 self.deploy._MountRootfsAsWritable, error_code_ok=False)
Robert Flack1dc7ea82014-11-26 13:50:24 -0500204 self.assertFalse(self.deploy._target_dir_is_still_readonly.is_set())
205
206 def testMountTempDir(self):
207 """Test that mount succeeds if target dir is writable."""
208 self.assertFalse(self.deploy._target_dir_is_still_readonly.is_set())
Mike Frysinger74ccd572015-05-21 21:18:20 -0400209 self.PatchObject(remote_access.RemoteDevice, 'IsDirWritable',
Robert Flack1dc7ea82014-11-26 13:50:24 -0500210 return_value=True, autospec=True)
211 self.deploy._MountRootfsAsWritable()
212 self.assertFalse(self.deploy._target_dir_is_still_readonly.is_set())
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700213
214
Ryan Cuief91e702013-02-04 12:06:36 -0800215class TestUiJobStarted(DeployTest):
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700216 """Test detection of a running 'ui' job."""
217
Ryan Cuif2d1a582013-02-19 14:08:13 -0800218 def MockStatusUiCmd(self, **kwargs):
David Haddock3151d912017-10-24 03:50:32 +0000219 self.deploy_mock.rsh_mock.AddCmdResult('status ui', **kwargs)
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700220
221 def testUiJobStartedFalse(self):
222 """Correct results with a stopped job."""
Ryan Cuif2d1a582013-02-19 14:08:13 -0800223 self.MockStatusUiCmd(output='ui stop/waiting')
224 self.assertFalse(self.deploy._CheckUiJobStarted())
225
226 def testNoUiJob(self):
227 """Correct results when the job doesn't exist."""
228 self.MockStatusUiCmd(error='start: Unknown job: ui', returncode=1)
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700229 self.assertFalse(self.deploy._CheckUiJobStarted())
230
231 def testCheckRootfsWriteableTrue(self):
232 """Correct results with a running job."""
Ryan Cuif2d1a582013-02-19 14:08:13 -0800233 self.MockStatusUiCmd(output='ui start/running, process 297')
Ryan Cuiafd6c5c2012-07-30 17:48:22 -0700234 self.assertTrue(self.deploy._CheckUiJobStarted())
235
236
Ryan Cuief91e702013-02-04 12:06:36 -0800237class StagingTest(cros_test_lib.MockTempDirTestCase):
238 """Test user-mode and ebuild-mode staging functionality."""
239
240 def setUp(self):
Ryan Cuief91e702013-02-04 12:06:36 -0800241 self.staging_dir = os.path.join(self.tempdir, 'staging')
242 self.build_dir = os.path.join(self.tempdir, 'build_dir')
Ryan Cui686ec052013-02-12 16:39:41 -0800243 self.common_flags = ['--build-dir', self.build_dir,
Bernie Thompson93b9ee62018-02-21 14:56:16 -0800244 '--board=eve', '--staging-only', '--cache-dir',
Ryan Cui686ec052013-02-12 16:39:41 -0800245 self.tempdir]
Ryan Cuia0215a72013-02-14 16:20:45 -0800246 self.sdk_mock = self.StartPatcher(cros_chrome_sdk_unittest.SDKFetcherMock())
Ryan Cui686ec052013-02-12 16:39:41 -0800247 self.PatchObject(
248 osutils, 'SourceEnvironment', autospec=True,
249 return_value={'STRIP': 'x86_64-cros-linux-gnu-strip'})
Ryan Cuief91e702013-02-04 12:06:36 -0800250
David Jamesa6e08892013-03-01 13:34:11 -0800251 def testSingleFileDeployFailure(self):
252 """Default staging enforces that mandatory files are copied"""
Mike Frysingerc3061a62015-06-04 04:16:18 -0400253 options = _ParseCommandLine(self.common_flags)
David Jamesa6e08892013-03-01 13:34:11 -0800254 osutils.Touch(os.path.join(self.build_dir, 'chrome'), makedirs=True)
255 self.assertRaises(
256 chrome_util.MissingPathError, deploy_chrome._PrepareStagingDir,
Daniel Eratc89829c2014-05-12 17:24:21 -0700257 options, self.tempdir, self.staging_dir, chrome_util._COPY_PATHS_CHROME)
Ryan Cuief91e702013-02-04 12:06:36 -0800258
David Jamesa6e08892013-03-01 13:34:11 -0800259 def testSloppyDeployFailure(self):
260 """Sloppy staging enforces that at least one file is copied."""
Mike Frysingerc3061a62015-06-04 04:16:18 -0400261 options = _ParseCommandLine(self.common_flags + ['--sloppy'])
David Jamesa6e08892013-03-01 13:34:11 -0800262 self.assertRaises(
263 chrome_util.MissingPathError, deploy_chrome._PrepareStagingDir,
Daniel Eratc89829c2014-05-12 17:24:21 -0700264 options, self.tempdir, self.staging_dir, chrome_util._COPY_PATHS_CHROME)
David Jamesa6e08892013-03-01 13:34:11 -0800265
266 def testSloppyDeploySuccess(self):
267 """Sloppy staging - stage one file."""
Mike Frysingerc3061a62015-06-04 04:16:18 -0400268 options = _ParseCommandLine(self.common_flags + ['--sloppy'])
David Jamesa6e08892013-03-01 13:34:11 -0800269 osutils.Touch(os.path.join(self.build_dir, 'chrome'), makedirs=True)
Steve Funge984a532013-11-25 17:09:25 -0800270 deploy_chrome._PrepareStagingDir(options, self.tempdir, self.staging_dir,
Daniel Eratc89829c2014-05-12 17:24:21 -0700271 chrome_util._COPY_PATHS_CHROME)
David Jamesa6e08892013-03-01 13:34:11 -0800272
Steve Funge984a532013-11-25 17:09:25 -0800273
274class DeployTestBuildDir(cros_test_lib.MockTempDirTestCase):
Daniel Erat1ae46382014-08-14 10:23:39 -0700275 """Set up a deploy object with a build-dir for use in deployment type tests"""
Steve Funge984a532013-11-25 17:09:25 -0800276
277 def _GetDeployChrome(self, args):
Mike Frysingerc3061a62015-06-04 04:16:18 -0400278 options = _ParseCommandLine(args)
Steve Funge984a532013-11-25 17:09:25 -0800279 return deploy_chrome.DeployChrome(
280 options, self.tempdir, os.path.join(self.tempdir, 'staging'))
281
282 def setUp(self):
283 self.staging_dir = os.path.join(self.tempdir, 'staging')
284 self.build_dir = os.path.join(self.tempdir, 'build_dir')
285 self.deploy_mock = self.StartPatcher(DeployChromeMock())
286 self.deploy = self._GetDeployChrome(
287 list(_REGULAR_TO) + ['--build-dir', self.build_dir,
Bernie Thompson93b9ee62018-02-21 14:56:16 -0800288 '--board=eve', '--staging-only', '--cache-dir',
Steve Funge984a532013-11-25 17:09:25 -0800289 self.tempdir, '--sloppy'])
290
Daniel Erat1ae46382014-08-14 10:23:39 -0700291 def getCopyPath(self, source_path):
292 """Return a chrome_util.Path or None if not present."""
293 paths = [p for p in self.deploy.copy_paths if p.src == source_path]
294 return paths[0] if paths else None
Steve Funge984a532013-11-25 17:09:25 -0800295
Daniel Erat1ae46382014-08-14 10:23:39 -0700296class TestDeploymentType(DeployTestBuildDir):
Steve Funge984a532013-11-25 17:09:25 -0800297 """Test detection of deployment type using build dir."""
298
Daniel Erat1ae46382014-08-14 10:23:39 -0700299 def testAppShellDetection(self):
300 """Check for an app_shell deployment"""
301 osutils.Touch(os.path.join(self.deploy.options.build_dir, 'app_shell'),
Steve Funge984a532013-11-25 17:09:25 -0800302 makedirs=True)
303 self.deploy._CheckDeployType()
Daniel Erat1ae46382014-08-14 10:23:39 -0700304 self.assertTrue(self.getCopyPath('app_shell'))
305 self.assertFalse(self.getCopyPath('chrome'))
Steve Funge984a532013-11-25 17:09:25 -0800306
Daniel Erat1ae46382014-08-14 10:23:39 -0700307 def testChromeAndAppShellDetection(self):
Daniel Eratf53bd3a2016-12-02 11:28:36 -0700308 """Check for a chrome deployment when app_shell also exists."""
Steve Fung63d705d2014-03-16 03:14:03 -0700309 osutils.Touch(os.path.join(self.deploy.options.build_dir, 'chrome'),
310 makedirs=True)
Daniel Erat1ae46382014-08-14 10:23:39 -0700311 osutils.Touch(os.path.join(self.deploy.options.build_dir, 'app_shell'),
Steve Fung63d705d2014-03-16 03:14:03 -0700312 makedirs=True)
313 self.deploy._CheckDeployType()
Daniel Erat1ae46382014-08-14 10:23:39 -0700314 self.assertTrue(self.getCopyPath('chrome'))
Daniel Erat9813f0e2014-11-12 11:00:28 -0700315 self.assertFalse(self.getCopyPath('app_shell'))
Steve Fung63d705d2014-03-16 03:14:03 -0700316
Steve Funge984a532013-11-25 17:09:25 -0800317 def testChromeDetection(self):
318 """Check for a regular chrome deployment"""
319 osutils.Touch(os.path.join(self.deploy.options.build_dir, 'chrome'),
320 makedirs=True)
321 self.deploy._CheckDeployType()
Daniel Erat1ae46382014-08-14 10:23:39 -0700322 self.assertTrue(self.getCopyPath('chrome'))
Daniel Erat9813f0e2014-11-12 11:00:28 -0700323 self.assertFalse(self.getCopyPath('app_shell'))