1#!/usr/bin/python 2# Copyright 2017 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 6import os 7import random 8import shutil 9import tempfile 10import unittest 11from contextlib import contextmanager 12 13import common 14from autotest_lib.client.bin import utils 15from autotest_lib.client.common_lib import error 16from autotest_lib.site_utils import lxc 17from autotest_lib.site_utils.lxc import constants 18from autotest_lib.site_utils.lxc import container as container_module 19from autotest_lib.site_utils.lxc import unittest_http 20from autotest_lib.site_utils.lxc import unittest_setup 21from autotest_lib.site_utils.lxc import utils as lxc_utils 22 23 24class ContainerTests(lxc_utils.LXCTests): 25 """Unit tests for the Container class.""" 26 27 @classmethod 28 def setUpClass(cls): 29 super(ContainerTests, cls).setUpClass() 30 cls.test_dir = tempfile.mkdtemp(dir=lxc.DEFAULT_CONTAINER_PATH, 31 prefix='container_unittest_') 32 33 # Check if a base container exists on this machine and download one if 34 # necessary. 35 image = lxc.BaseImage() 36 try: 37 cls.base_container = image.get() 38 cls.cleanup_base_container = False 39 except error.ContainerError: 40 image.setup() 41 cls.base_container = image.get() 42 cls.cleanup_base_container = True 43 assert(cls.base_container is not None) 44 45 46 @classmethod 47 def tearDownClass(cls): 48 cls.base_container = None 49 if not unittest_setup.config.skip_cleanup: 50 if cls.cleanup_base_container: 51 lxc.BaseImage().cleanup() 52 utils.run('sudo rm -r %s' % cls.test_dir) 53 54 55 def testInit(self): 56 """Verifies that containers initialize correctly.""" 57 # Make a container that just points to the base container. 58 container = lxc.Container.create_from_existing_dir( 59 self.base_container.container_path, 60 self.base_container.name) 61 # Calling is_running triggers an lxc-ls call, which should verify that 62 # the on-disk container is valid. 63 self.assertFalse(container.is_running()) 64 65 66 def testInitInvalid(self): 67 """Verifies that invalid containers can still be instantiated, 68 if not used. 69 """ 70 with tempfile.NamedTemporaryFile(dir=self.test_dir) as tmpfile: 71 name = os.path.basename(tmpfile.name) 72 container = lxc.Container.create_from_existing_dir(self.test_dir, 73 name) 74 with self.assertRaises(error.ContainerError): 75 container.refresh_status() 76 77 78 def testInvalidId(self): 79 """Verifies that corrupted ID files do not raise exceptions.""" 80 with self.createContainer() as container: 81 # Create a container with an empty ID file. 82 id_path = os.path.join(container.container_path, 83 container.name, 84 container_module._CONTAINER_ID_FILENAME) 85 utils.run('sudo touch %s' % id_path) 86 try: 87 # Verify that container creation doesn't raise exceptions. 88 test_container = lxc.Container.create_from_existing_dir( 89 self.test_dir, container.name) 90 self.assertIsNone(test_container.id) 91 except Exception: 92 self.fail('Unexpected exception:\n%s' % error.format_error()) 93 94 95 def testDefaultHostname(self): 96 """Verifies that the zygote starts up with a default hostname that is 97 the lxc container name.""" 98 test_name = 'testHostname' 99 with self.createContainer(name=test_name) as container: 100 container.start(wait_for_network=True) 101 hostname = container.attach_run('hostname').stdout.strip() 102 self.assertEqual(test_name, hostname) 103 104 105 def testSetHostnameRunning(self): 106 """Verifies that the hostname can be set on a running container.""" 107 with self.createContainer() as container: 108 expected_hostname = 'my-new-hostname' 109 container.start(wait_for_network=True) 110 container.set_hostname(expected_hostname) 111 hostname = container.attach_run('hostname -f').stdout.strip() 112 self.assertEqual(expected_hostname, hostname) 113 114 115 def testSetHostnameNotRunningRaisesException(self): 116 """Verifies that set_hostname on a stopped container raises an error. 117 118 The lxc.utsname config setting is unreliable (it only works if the 119 original container name is not a valid RFC-952 hostname, e.g. if it has 120 underscores). 121 122 A more reliable method exists for setting the hostname but it requires 123 the container to be running. To avoid confusion, setting the hostname 124 on a stopped container is disallowed. 125 126 This test verifies that the operation raises a ContainerError. 127 """ 128 with self.createContainer() as container: 129 with self.assertRaises(error.ContainerError): 130 # Ensure the container is not running 131 if container.is_running(): 132 raise RuntimeError('Container should not be running.') 133 container.set_hostname('foobar') 134 135 136 def testClone(self): 137 """Verifies that cloning a container works as expected.""" 138 clone = lxc.Container.clone(src=self.base_container, 139 new_name="testClone", 140 new_path=self.test_dir, 141 snapshot=True) 142 try: 143 # Throws an exception if the container is not valid. 144 clone.refresh_status() 145 finally: 146 clone.destroy() 147 148 149 def testCloneWithoutCleanup(self): 150 """Verifies that cloning a container to an existing name will fail as 151 expected. 152 """ 153 lxc.Container.clone(src=self.base_container, 154 new_name="testCloneWithoutCleanup", 155 new_path=self.test_dir, 156 snapshot=True) 157 with self.assertRaises(error.ContainerError): 158 lxc.Container.clone(src=self.base_container, 159 new_name="testCloneWithoutCleanup", 160 new_path=self.test_dir, 161 snapshot=True) 162 163 164 def testCloneWithCleanup(self): 165 """Verifies that cloning a container with cleanup works properly.""" 166 clone0 = lxc.Container.clone(src=self.base_container, 167 new_name="testClone", 168 new_path=self.test_dir, 169 snapshot=True) 170 clone0.start(wait_for_network=False) 171 tmpfile = clone0.attach_run('mktemp').stdout 172 # Verify that our tmpfile exists 173 clone0.attach_run('test -f %s' % tmpfile) 174 175 # Clone another container in place of the existing container. 176 clone1 = lxc.Container.clone(src=self.base_container, 177 new_name="testClone", 178 new_path=self.test_dir, 179 snapshot=True, 180 cleanup=True) 181 with self.assertRaises(error.CmdError): 182 clone1.attach_run('test -f %s' % tmpfile) 183 184 185 def testInstallSsp(self): 186 """Verifies that installing the ssp in the container works.""" 187 # Hard-coded path to some golden data for this test. 188 test_ssp = os.path.join( 189 common.autotest_dir, 190 'site_utils', 'lxc', 'test', 'test_ssp.tar.bz2') 191 # Create a container, install the self-served ssp, then check that it is 192 # installed into the container correctly. 193 with self.createContainer() as container: 194 with unittest_http.serve_locally(test_ssp) as url: 195 container.install_ssp(url) 196 container.start(wait_for_network=False) 197 198 # The test ssp just contains a couple of text files, in known 199 # locations. Verify the location and content of those files in the 200 # container. 201 cat = lambda path: container.attach_run('cat %s' % path).stdout 202 test0 = cat(os.path.join(constants.CONTAINER_AUTOTEST_DIR, 203 'test.0')) 204 test1 = cat(os.path.join(constants.CONTAINER_AUTOTEST_DIR, 205 'dir0', 'test.1')) 206 self.assertEquals('the five boxing wizards jumped quickly', 207 test0) 208 self.assertEquals('the quick brown fox jumps over the lazy dog', 209 test1) 210 211 212 def testInstallControlFile(self): 213 """Verifies that installing a control file in the container works.""" 214 _unused, tmpfile = tempfile.mkstemp() 215 with self.createContainer() as container: 216 container.install_control_file(tmpfile) 217 container.start(wait_for_network=False) 218 # Verify that the file is found in the container. 219 container.attach_run( 220 'test -f %s' % os.path.join(lxc.CONTROL_TEMP_PATH, 221 os.path.basename(tmpfile))) 222 223 224 def testCopyFile(self): 225 """Verifies that files are correctly copied into the container.""" 226 control_string = 'amazingly few discotheques provide jukeboxes' 227 with tempfile.NamedTemporaryFile() as tmpfile: 228 tmpfile.write(control_string) 229 tmpfile.flush() 230 231 with self.createContainer() as container: 232 dst = os.path.join(constants.CONTAINER_AUTOTEST_DIR, 233 os.path.basename(tmpfile.name)) 234 container.copy(tmpfile.name, dst) 235 container.start(wait_for_network=False) 236 # Verify the file content. 237 test_string = container.attach_run('cat %s' % dst).stdout 238 self.assertEquals(control_string, test_string) 239 240 241 def testCopyDirectory(self): 242 """Verifies that directories are correctly copied into the container.""" 243 control_string = 'pack my box with five dozen liquor jugs' 244 with lxc_utils.TempDir() as tmpdir: 245 fd, tmpfile = tempfile.mkstemp(dir=tmpdir) 246 f = os.fdopen(fd, 'w') 247 f.write(control_string) 248 f.close() 249 250 with self.createContainer() as container: 251 dst = os.path.join(constants.CONTAINER_AUTOTEST_DIR, 252 os.path.basename(tmpdir)) 253 container.copy(tmpdir, dst) 254 container.start(wait_for_network=False) 255 # Verify the file content. 256 test_file = os.path.join(dst, os.path.basename(tmpfile)) 257 test_string = container.attach_run('cat %s' % test_file).stdout 258 self.assertEquals(control_string, test_string) 259 260 261 def testMountDirectory(self): 262 """Verifies that read-write mounts work.""" 263 with lxc_utils.TempDir() as tmpdir, self.createContainer() as container: 264 dst = '/testMountDirectory/testMount' 265 container.mount_dir(tmpdir, dst, readonly=False) 266 container.start(wait_for_network=False) 267 268 # Verify that the mount point is correctly bound, and is read-write. 269 self.verifyBindMount(container, dst, tmpdir) 270 container.attach_run('test -r %s -a -w %s' % (dst, dst)) 271 272 273 def testMountDirectoryReadOnly(self): 274 """Verifies that read-only mounts work.""" 275 with lxc_utils.TempDir() as tmpdir, self.createContainer() as container: 276 dst = '/testMountDirectoryReadOnly/testMount' 277 container.mount_dir(tmpdir, dst, readonly=True) 278 container.start(wait_for_network=False) 279 280 # Verify that the mount point is correctly bound, and is read-only. 281 self.verifyBindMount(container, dst, tmpdir) 282 container.attach_run('test -r %s -a ! -w %s' % (dst, dst)) 283 284 285 def testMountDirectoryRelativePath(self): 286 """Verifies that relative-path mounts work.""" 287 with lxc_utils.TempDir() as tmpdir, self.createContainer() as container: 288 dst = 'testMountDirectoryRelativePath/testMount' 289 container.mount_dir(tmpdir, dst, readonly=True) 290 container.start(wait_for_network=False) 291 292 # Verify that the mount points is correctly bound.. 293 self.verifyBindMount(container, dst, tmpdir) 294 295 296 def testContainerIdPersistence(self): 297 """Verifies that container IDs correctly persist. 298 299 When a Container is instantiated on top of an existing container dir, 300 check that it picks up the correct ID. 301 """ 302 with self.createContainer() as container: 303 test_id = random_container_id() 304 container.id = test_id 305 306 # Set up another container and verify that its ID matches. 307 test_container = lxc.Container.create_from_existing_dir( 308 container.container_path, container.name) 309 310 self.assertEqual(test_id, test_container.id) 311 312 313 def testContainerIdIsNone_newContainer(self): 314 """Verifies that newly created/cloned containers have no ID.""" 315 with self.createContainer() as container: 316 self.assertIsNone(container.id) 317 # Set an ID, clone the container, and verify the clone has no ID. 318 container.id = random_container_id() 319 clone = lxc.Container.clone(src=container, 320 new_name=container.name + '_clone', 321 snapshot=True) 322 self.assertIsNotNone(container.id) 323 self.assertIsNone(clone.id) 324 325 326 @contextmanager 327 def createContainer(self, name=None): 328 """Creates a container from the base container, for testing. 329 Use this to ensure that containers get properly cleaned up after each 330 test. 331 332 @param name: An optional name for the new container. 333 """ 334 if name is None: 335 name = self.id().split('.')[-1] 336 container = lxc.Container.clone(src=self.base_container, 337 new_name=name, 338 new_path=self.test_dir, 339 snapshot=True) 340 try: 341 yield container 342 finally: 343 if not unittest_setup.config.skip_cleanup: 344 container.destroy() 345 346 347 def verifyBindMount(self, container, container_path, host_path): 348 """Verifies that a given path in a container is bind-mounted to a given 349 path in the host system. 350 351 @param container: The Container instance to be tested. 352 @param container_path: The path in the container to compare. 353 @param host_path: The path in the host system to compare. 354 """ 355 container_inode = (container.attach_run('ls -id %s' % container_path) 356 .stdout.split()[0]) 357 host_inode = utils.run('ls -id %s' % host_path).stdout.split()[0] 358 # Compare the container and host inodes - they should match. 359 self.assertEqual(container_inode, host_inode) 360 361 362class ContainerIdTests(lxc_utils.LXCTests): 363 """Unit tests for the ContainerId class.""" 364 365 def setUp(self): 366 self.test_dir = tempfile.mkdtemp() 367 368 369 def tearDown(self): 370 shutil.rmtree(self.test_dir) 371 372 373 def testPickle(self): 374 """Verifies the ContainerId persistence code.""" 375 # Create a random ID, then save and load it and compare them. 376 control = random_container_id() 377 control.save(self.test_dir) 378 379 test_data = lxc.ContainerId.load(self.test_dir) 380 self.assertEqual(control, test_data) 381 382 383def random_container_id(): 384 """Generate a random container ID for testing.""" 385 return lxc.ContainerId.create(random.randint(0, 1000)) 386 387 388if __name__ == '__main__': 389 unittest.main() 390