Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1 | # |
| 2 | # Copyright (c) 2015, Intel Corporation. |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 3 | # |
Brad Bishop | c342db3 | 2019-05-15 21:57:59 -0400 | [diff] [blame] | 4 | # SPDX-License-Identifier: GPL-2.0-only |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 5 | # |
| 6 | # AUTHORS |
| 7 | # Ed Bartosh <ed.bartosh@linux.intel.com> |
| 8 | |
| 9 | """Test cases for wic.""" |
| 10 | |
| 11 | import os |
| 12 | import sys |
| 13 | import unittest |
Andrew Geissler | d159c7f | 2021-09-02 21:05:58 -0500 | [diff] [blame] | 14 | import hashlib |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 15 | |
| 16 | from glob import glob |
| 17 | from shutil import rmtree, copy |
| 18 | from functools import wraps, lru_cache |
| 19 | from tempfile import NamedTemporaryFile |
| 20 | |
| 21 | from oeqa.selftest.case import OESelftestTestCase |
| 22 | from oeqa.utils.commands import runCmd, bitbake, get_bb_var, get_bb_vars, runqemu |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 23 | |
| 24 | |
| 25 | @lru_cache(maxsize=32) |
| 26 | def get_host_arch(recipe): |
| 27 | """A cached call to get_bb_var('HOST_ARCH', <recipe>)""" |
| 28 | return get_bb_var('HOST_ARCH', recipe) |
| 29 | |
| 30 | |
| 31 | def only_for_arch(archs, image='core-image-minimal'): |
| 32 | """Decorator for wrapping test cases that can be run only for specific target |
| 33 | architectures. A list of compatible architectures is passed in `archs`. |
| 34 | Current architecture will be determined by parsing bitbake output for |
| 35 | `image` recipe. |
| 36 | """ |
| 37 | def wrapper(func): |
| 38 | @wraps(func) |
| 39 | def wrapped_f(*args, **kwargs): |
| 40 | arch = get_host_arch(image) |
| 41 | if archs and arch not in archs: |
| 42 | raise unittest.SkipTest("Testcase arch dependency not met: %s" % arch) |
| 43 | return func(*args, **kwargs) |
| 44 | wrapped_f.__name__ = func.__name__ |
| 45 | return wrapped_f |
| 46 | return wrapper |
| 47 | |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 48 | def extract_files(debugfs_output): |
| 49 | """ |
| 50 | extract file names from the output of debugfs -R 'ls -p', |
| 51 | which looks like this: |
| 52 | |
| 53 | /2/040755/0/0/.//\n |
| 54 | /2/040755/0/0/..//\n |
| 55 | /11/040700/0/0/lost+found^M//\n |
| 56 | /12/040755/1002/1002/run//\n |
| 57 | /13/040755/1002/1002/sys//\n |
| 58 | /14/040755/1002/1002/bin//\n |
| 59 | /80/040755/1002/1002/var//\n |
| 60 | /92/040755/1002/1002/tmp//\n |
| 61 | """ |
| 62 | # NOTE the occasional ^M in file names |
| 63 | return [line.split('/')[5].strip() for line in \ |
| 64 | debugfs_output.strip().split('/\n')] |
| 65 | |
| 66 | def files_own_by_root(debugfs_output): |
| 67 | for line in debugfs_output.strip().split('/\n'): |
| 68 | if line.split('/')[3:5] != ['0', '0']: |
| 69 | print(debugfs_output) |
| 70 | return False |
| 71 | return True |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 72 | |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 73 | class WicTestCase(OESelftestTestCase): |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 74 | """Wic test class.""" |
| 75 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 76 | image_is_ready = False |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 77 | wicenv_cache = {} |
| 78 | |
| 79 | def setUpLocal(self): |
| 80 | """This code is executed before each test method.""" |
Brad Bishop | c4ea075 | 2018-11-15 14:30:15 -0800 | [diff] [blame] | 81 | self.resultdir = self.builddir + "/wic-tmp/" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 82 | super(WicTestCase, self).setUpLocal() |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 83 | |
| 84 | # Do this here instead of in setUpClass as the base setUp does some |
| 85 | # clean up which can result in the native tools built earlier in |
| 86 | # setUpClass being unavailable. |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 87 | if not WicTestCase.image_is_ready: |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 88 | if get_bb_var('USE_NLS') == 'yes': |
| 89 | bitbake('wic-tools') |
| 90 | else: |
| 91 | self.skipTest('wic-tools cannot be built due its (intltool|gettext)-native dependency and NLS disable') |
| 92 | |
| 93 | bitbake('core-image-minimal') |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 94 | bitbake('core-image-minimal-mtdutils') |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 95 | WicTestCase.image_is_ready = True |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 96 | |
| 97 | rmtree(self.resultdir, ignore_errors=True) |
| 98 | |
| 99 | def tearDownLocal(self): |
| 100 | """Remove resultdir as it may contain images.""" |
| 101 | rmtree(self.resultdir, ignore_errors=True) |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 102 | super(WicTestCase, self).tearDownLocal() |
| 103 | |
| 104 | def _get_image_env_path(self, image): |
| 105 | """Generate and obtain the path to <image>.env""" |
| 106 | if image not in WicTestCase.wicenv_cache: |
| 107 | self.assertEqual(0, bitbake('%s -c do_rootfs_wicenv' % image).status) |
| 108 | bb_vars = get_bb_vars(['STAGING_DIR', 'MACHINE'], image) |
| 109 | stdir = bb_vars['STAGING_DIR'] |
| 110 | machine = bb_vars['MACHINE'] |
| 111 | WicTestCase.wicenv_cache[image] = os.path.join(stdir, machine, 'imgdata') |
| 112 | return WicTestCase.wicenv_cache[image] |
| 113 | |
| 114 | class Wic(WicTestCase): |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 115 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 116 | def test_version(self): |
| 117 | """Test wic --version""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 118 | runCmd('wic --version') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 119 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 120 | def test_help(self): |
| 121 | """Test wic --help and wic -h""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 122 | runCmd('wic --help') |
| 123 | runCmd('wic -h') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 124 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 125 | def test_createhelp(self): |
| 126 | """Test wic create --help""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 127 | runCmd('wic create --help') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 128 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 129 | def test_listhelp(self): |
| 130 | """Test wic list --help""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 131 | runCmd('wic list --help') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 132 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 133 | def test_help_create(self): |
| 134 | """Test wic help create""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 135 | runCmd('wic help create') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 136 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 137 | def test_help_list(self): |
| 138 | """Test wic help list""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 139 | runCmd('wic help list') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 140 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 141 | def test_help_overview(self): |
| 142 | """Test wic help overview""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 143 | runCmd('wic help overview') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 144 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 145 | def test_help_plugins(self): |
| 146 | """Test wic help plugins""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 147 | runCmd('wic help plugins') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 148 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 149 | def test_help_kickstart(self): |
| 150 | """Test wic help kickstart""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 151 | runCmd('wic help kickstart') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 152 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 153 | def test_list_images(self): |
| 154 | """Test wic list images""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 155 | runCmd('wic list images') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 156 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 157 | def test_list_source_plugins(self): |
| 158 | """Test wic list source-plugins""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 159 | runCmd('wic list source-plugins') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 160 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 161 | def test_listed_images_help(self): |
| 162 | """Test wic listed images help""" |
| 163 | output = runCmd('wic list images').output |
| 164 | imagelist = [line.split()[0] for line in output.splitlines()] |
| 165 | for image in imagelist: |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 166 | runCmd('wic list %s help' % image) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 167 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 168 | def test_unsupported_subcommand(self): |
| 169 | """Test unsupported subcommand""" |
| 170 | self.assertNotEqual(0, runCmd('wic unsupported', ignore_status=True).status) |
| 171 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 172 | def test_no_command(self): |
| 173 | """Test wic without command""" |
| 174 | self.assertEqual(1, runCmd('wic', ignore_status=True).status) |
| 175 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 176 | def test_build_image_name(self): |
| 177 | """Test wic create wictestdisk --image-name=core-image-minimal""" |
| 178 | cmd = "wic create wictestdisk --image-name=core-image-minimal -o %s" % self.resultdir |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 179 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 180 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*.direct"))) |
| 181 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 182 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 183 | def test_gpt_image(self): |
| 184 | """Test creation of core-image-minimal with gpt table and UUID boot""" |
| 185 | cmd = "wic create directdisk-gpt --image-name core-image-minimal -o %s" % self.resultdir |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 186 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 187 | self.assertEqual(1, len(glob(self.resultdir + "directdisk-*.direct"))) |
| 188 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 189 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 190 | def test_iso_image(self): |
| 191 | """Test creation of hybrid iso image with legacy and EFI boot""" |
| 192 | config = 'INITRAMFS_IMAGE = "core-image-minimal-initramfs"\n'\ |
Patrick Williams | 213cb26 | 2021-08-07 19:21:33 -0500 | [diff] [blame] | 193 | 'MACHINE_FEATURES:append = " efi"\n'\ |
| 194 | 'DEPENDS:pn-core-image-minimal += "syslinux"\n' |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 195 | self.append_config(config) |
Brad Bishop | c4ea075 | 2018-11-15 14:30:15 -0800 | [diff] [blame] | 196 | bitbake('core-image-minimal core-image-minimal-initramfs') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 197 | self.remove_config(config) |
| 198 | cmd = "wic create mkhybridiso --image-name core-image-minimal -o %s" % self.resultdir |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 199 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 200 | self.assertEqual(1, len(glob(self.resultdir + "HYBRID_ISO_IMG-*.direct"))) |
| 201 | self.assertEqual(1, len(glob(self.resultdir + "HYBRID_ISO_IMG-*.iso"))) |
| 202 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 203 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 204 | def test_qemux86_directdisk(self): |
| 205 | """Test creation of qemux-86-directdisk image""" |
| 206 | cmd = "wic create qemux86-directdisk -e core-image-minimal -o %s" % self.resultdir |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 207 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 208 | self.assertEqual(1, len(glob(self.resultdir + "qemux86-directdisk-*direct"))) |
| 209 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 210 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 211 | def test_mkefidisk(self): |
| 212 | """Test creation of mkefidisk image""" |
| 213 | cmd = "wic create mkefidisk -e core-image-minimal -o %s" % self.resultdir |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 214 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 215 | self.assertEqual(1, len(glob(self.resultdir + "mkefidisk-*direct"))) |
| 216 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 217 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 218 | def test_bootloader_config(self): |
| 219 | """Test creation of directdisk-bootloader-config image""" |
Patrick Williams | 213cb26 | 2021-08-07 19:21:33 -0500 | [diff] [blame] | 220 | config = 'DEPENDS:pn-core-image-minimal += "syslinux"\n' |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 221 | self.append_config(config) |
| 222 | bitbake('core-image-minimal') |
| 223 | self.remove_config(config) |
| 224 | cmd = "wic create directdisk-bootloader-config -e core-image-minimal -o %s" % self.resultdir |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 225 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 226 | self.assertEqual(1, len(glob(self.resultdir + "directdisk-bootloader-config-*direct"))) |
| 227 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 228 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 229 | def test_systemd_bootdisk(self): |
| 230 | """Test creation of systemd-bootdisk image""" |
Patrick Williams | 213cb26 | 2021-08-07 19:21:33 -0500 | [diff] [blame] | 231 | config = 'MACHINE_FEATURES:append = " efi"\n' |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 232 | self.append_config(config) |
| 233 | bitbake('core-image-minimal') |
| 234 | self.remove_config(config) |
| 235 | cmd = "wic create systemd-bootdisk -e core-image-minimal -o %s" % self.resultdir |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 236 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 237 | self.assertEqual(1, len(glob(self.resultdir + "systemd-bootdisk-*direct"))) |
| 238 | |
Andrew Geissler | c9f7865 | 2020-09-18 14:11:35 -0500 | [diff] [blame] | 239 | def test_efi_bootpart(self): |
| 240 | """Test creation of efi-bootpart image""" |
| 241 | cmd = "wic create mkefidisk -e core-image-minimal -o %s" % self.resultdir |
| 242 | kimgtype = get_bb_var('KERNEL_IMAGETYPE', 'core-image-minimal') |
| 243 | self.append_config('IMAGE_EFI_BOOT_FILES = "%s;kernel"\n' % kimgtype) |
| 244 | runCmd(cmd) |
| 245 | sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools') |
| 246 | images = glob(self.resultdir + "mkefidisk-*.direct") |
| 247 | result = runCmd("wic ls %s:1/ -n %s" % (images[0], sysroot)) |
| 248 | self.assertIn("kernel",result.output) |
| 249 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 250 | def test_sdimage_bootpart(self): |
| 251 | """Test creation of sdimage-bootpart image""" |
| 252 | cmd = "wic create sdimage-bootpart -e core-image-minimal -o %s" % self.resultdir |
| 253 | kimgtype = get_bb_var('KERNEL_IMAGETYPE', 'core-image-minimal') |
| 254 | self.write_config('IMAGE_BOOT_FILES = "%s"\n' % kimgtype) |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 255 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 256 | self.assertEqual(1, len(glob(self.resultdir + "sdimage-bootpart-*direct"))) |
| 257 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 258 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 259 | def test_default_output_dir(self): |
| 260 | """Test default output location""" |
| 261 | for fname in glob("directdisk-*.direct"): |
| 262 | os.remove(fname) |
Patrick Williams | 213cb26 | 2021-08-07 19:21:33 -0500 | [diff] [blame] | 263 | config = 'DEPENDS:pn-core-image-minimal += "syslinux"\n' |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 264 | self.append_config(config) |
| 265 | bitbake('core-image-minimal') |
| 266 | self.remove_config(config) |
| 267 | cmd = "wic create directdisk -e core-image-minimal" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 268 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 269 | self.assertEqual(1, len(glob("directdisk-*.direct"))) |
| 270 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 271 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 272 | def test_build_artifacts(self): |
| 273 | """Test wic create directdisk providing all artifacts.""" |
| 274 | bb_vars = get_bb_vars(['STAGING_DATADIR', 'RECIPE_SYSROOT_NATIVE'], |
| 275 | 'wic-tools') |
| 276 | bb_vars.update(get_bb_vars(['DEPLOY_DIR_IMAGE', 'IMAGE_ROOTFS'], |
| 277 | 'core-image-minimal')) |
| 278 | bbvars = {key.lower(): value for key, value in bb_vars.items()} |
| 279 | bbvars['resultdir'] = self.resultdir |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 280 | runCmd("wic create directdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 281 | "-b %(staging_datadir)s " |
| 282 | "-k %(deploy_dir_image)s " |
| 283 | "-n %(recipe_sysroot_native)s " |
| 284 | "-r %(image_rootfs)s " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 285 | "-o %(resultdir)s" % bbvars) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 286 | self.assertEqual(1, len(glob(self.resultdir + "directdisk-*.direct"))) |
| 287 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 288 | def test_compress_gzip(self): |
| 289 | """Test compressing an image with gzip""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 290 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 291 | "--image-name core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 292 | "-c gzip -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 293 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*.direct.gz"))) |
| 294 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 295 | def test_compress_bzip2(self): |
| 296 | """Test compressing an image with bzip2""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 297 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 298 | "--image-name=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 299 | "-c bzip2 -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 300 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*.direct.bz2"))) |
| 301 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 302 | def test_compress_xz(self): |
| 303 | """Test compressing an image with xz""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 304 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 305 | "--image-name=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 306 | "--compress-with=xz -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 307 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*.direct.xz"))) |
| 308 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 309 | def test_wrong_compressor(self): |
| 310 | """Test how wic breaks if wrong compressor is provided""" |
| 311 | self.assertEqual(2, runCmd("wic create wictestdisk " |
| 312 | "--image-name=core-image-minimal " |
| 313 | "-c wrong -o %s" % self.resultdir, |
| 314 | ignore_status=True).status) |
| 315 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 316 | def test_debug_short(self): |
| 317 | """Test -D option""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 318 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 319 | "--image-name=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 320 | "-D -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 321 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*.direct"))) |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 322 | self.assertEqual(1, len(glob(self.resultdir + "tmp.wic*"))) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 323 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 324 | def test_debug_long(self): |
| 325 | """Test --debug option""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 326 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 327 | "--image-name=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 328 | "--debug -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 329 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*.direct"))) |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 330 | self.assertEqual(1, len(glob(self.resultdir + "tmp.wic*"))) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 331 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 332 | def test_skip_build_check_short(self): |
| 333 | """Test -s option""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 334 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 335 | "--image-name=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 336 | "-s -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 337 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*.direct"))) |
| 338 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 339 | def test_skip_build_check_long(self): |
| 340 | """Test --skip-build-check option""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 341 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 342 | "--image-name=core-image-minimal " |
| 343 | "--skip-build-check " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 344 | "--outdir %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 345 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*.direct"))) |
| 346 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 347 | def test_build_rootfs_short(self): |
| 348 | """Test -f option""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 349 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 350 | "--image-name=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 351 | "-f -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 352 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*.direct"))) |
| 353 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 354 | def test_build_rootfs_long(self): |
| 355 | """Test --build-rootfs option""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 356 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 357 | "--image-name=core-image-minimal " |
| 358 | "--build-rootfs " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 359 | "--outdir %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 360 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*.direct"))) |
| 361 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 362 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 363 | def test_rootfs_indirect_recipes(self): |
| 364 | """Test usage of rootfs plugin with rootfs recipes""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 365 | runCmd("wic create directdisk-multi-rootfs " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 366 | "--image-name=core-image-minimal " |
| 367 | "--rootfs rootfs1=core-image-minimal " |
| 368 | "--rootfs rootfs2=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 369 | "--outdir %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 370 | self.assertEqual(1, len(glob(self.resultdir + "directdisk-multi-rootfs*.direct"))) |
| 371 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 372 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 373 | def test_rootfs_artifacts(self): |
| 374 | """Test usage of rootfs plugin with rootfs paths""" |
| 375 | bb_vars = get_bb_vars(['STAGING_DATADIR', 'RECIPE_SYSROOT_NATIVE'], |
| 376 | 'wic-tools') |
| 377 | bb_vars.update(get_bb_vars(['DEPLOY_DIR_IMAGE', 'IMAGE_ROOTFS'], |
| 378 | 'core-image-minimal')) |
| 379 | bbvars = {key.lower(): value for key, value in bb_vars.items()} |
| 380 | bbvars['wks'] = "directdisk-multi-rootfs" |
| 381 | bbvars['resultdir'] = self.resultdir |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 382 | runCmd("wic create %(wks)s " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 383 | "--bootimg-dir=%(staging_datadir)s " |
| 384 | "--kernel-dir=%(deploy_dir_image)s " |
| 385 | "--native-sysroot=%(recipe_sysroot_native)s " |
| 386 | "--rootfs-dir rootfs1=%(image_rootfs)s " |
| 387 | "--rootfs-dir rootfs2=%(image_rootfs)s " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 388 | "--outdir %(resultdir)s" % bbvars) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 389 | self.assertEqual(1, len(glob(self.resultdir + "%(wks)s-*.direct" % bbvars))) |
| 390 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 391 | def test_exclude_path(self): |
| 392 | """Test --exclude-path wks option.""" |
| 393 | |
| 394 | oldpath = os.environ['PATH'] |
| 395 | os.environ['PATH'] = get_bb_var("PATH", "wic-tools") |
| 396 | |
| 397 | try: |
| 398 | wks_file = 'temp.wks' |
| 399 | with open(wks_file, 'w') as wks: |
| 400 | rootfs_dir = get_bb_var('IMAGE_ROOTFS', 'core-image-minimal') |
| 401 | wks.write(""" |
| 402 | part / --source rootfs --ondisk mmcblk0 --fstype=ext4 --exclude-path usr |
| 403 | part /usr --source rootfs --ondisk mmcblk0 --fstype=ext4 --rootfs-dir %s/usr |
| 404 | part /etc --source rootfs --ondisk mmcblk0 --fstype=ext4 --exclude-path bin/ --rootfs-dir %s/usr""" |
| 405 | % (rootfs_dir, rootfs_dir)) |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 406 | runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 407 | % (wks_file, self.resultdir)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 408 | |
| 409 | os.remove(wks_file) |
| 410 | wicout = glob(self.resultdir + "%s-*direct" % 'temp') |
| 411 | self.assertEqual(1, len(wicout)) |
| 412 | |
| 413 | wicimg = wicout[0] |
| 414 | |
| 415 | # verify partition size with wic |
| 416 | res = runCmd("parted -m %s unit b p 2>/dev/null" % wicimg) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 417 | |
| 418 | # parse parted output which looks like this: |
| 419 | # BYT;\n |
| 420 | # /var/tmp/wic/build/tmpfwvjjkf_-201611101222-hda.direct:200MiB:file:512:512:msdos::;\n |
| 421 | # 1:0.00MiB:200MiB:200MiB:ext4::;\n |
| 422 | partlns = res.output.splitlines()[2:] |
| 423 | |
| 424 | self.assertEqual(3, len(partlns)) |
| 425 | |
| 426 | for part in [1, 2, 3]: |
| 427 | part_file = os.path.join(self.resultdir, "selftest_img.part%d" % part) |
| 428 | partln = partlns[part-1].split(":") |
| 429 | self.assertEqual(7, len(partln)) |
| 430 | start = int(partln[1].rstrip("B")) / 512 |
| 431 | length = int(partln[3].rstrip("B")) / 512 |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 432 | runCmd("dd if=%s of=%s skip=%d count=%d" % |
| 433 | (wicimg, part_file, start, length)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 434 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 435 | # Test partition 1, should contain the normal root directories, except |
| 436 | # /usr. |
| 437 | res = runCmd("debugfs -R 'ls -p' %s 2>/dev/null" % \ |
| 438 | os.path.join(self.resultdir, "selftest_img.part1")) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 439 | files = extract_files(res.output) |
| 440 | self.assertIn("etc", files) |
| 441 | self.assertNotIn("usr", files) |
| 442 | |
| 443 | # Partition 2, should contain common directories for /usr, not root |
| 444 | # directories. |
| 445 | res = runCmd("debugfs -R 'ls -p' %s 2>/dev/null" % \ |
| 446 | os.path.join(self.resultdir, "selftest_img.part2")) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 447 | files = extract_files(res.output) |
| 448 | self.assertNotIn("etc", files) |
| 449 | self.assertNotIn("usr", files) |
| 450 | self.assertIn("share", files) |
| 451 | |
| 452 | # Partition 3, should contain the same as partition 2, including the bin |
| 453 | # directory, but not the files inside it. |
| 454 | res = runCmd("debugfs -R 'ls -p' %s 2>/dev/null" % \ |
| 455 | os.path.join(self.resultdir, "selftest_img.part3")) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 456 | files = extract_files(res.output) |
| 457 | self.assertNotIn("etc", files) |
| 458 | self.assertNotIn("usr", files) |
| 459 | self.assertIn("share", files) |
| 460 | self.assertIn("bin", files) |
| 461 | res = runCmd("debugfs -R 'ls -p bin' %s 2>/dev/null" % \ |
| 462 | os.path.join(self.resultdir, "selftest_img.part3")) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 463 | files = extract_files(res.output) |
| 464 | self.assertIn(".", files) |
| 465 | self.assertIn("..", files) |
| 466 | self.assertEqual(2, len(files)) |
| 467 | |
| 468 | for part in [1, 2, 3]: |
| 469 | part_file = os.path.join(self.resultdir, "selftest_img.part%d" % part) |
| 470 | os.remove(part_file) |
| 471 | |
| 472 | finally: |
| 473 | os.environ['PATH'] = oldpath |
| 474 | |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 475 | def test_include_path(self): |
| 476 | """Test --include-path wks option.""" |
| 477 | |
| 478 | oldpath = os.environ['PATH'] |
| 479 | os.environ['PATH'] = get_bb_var("PATH", "wic-tools") |
| 480 | |
| 481 | try: |
| 482 | include_path = os.path.join(self.resultdir, 'test-include') |
| 483 | os.makedirs(include_path) |
| 484 | with open(os.path.join(include_path, 'test-file'), 'w') as t: |
| 485 | t.write("test\n") |
| 486 | wks_file = os.path.join(include_path, 'temp.wks') |
| 487 | with open(wks_file, 'w') as wks: |
| 488 | rootfs_dir = get_bb_var('IMAGE_ROOTFS', 'core-image-minimal') |
| 489 | wks.write(""" |
| 490 | part /part1 --source rootfs --ondisk mmcblk0 --fstype=ext4 |
| 491 | part /part2 --source rootfs --ondisk mmcblk0 --fstype=ext4 --include-path %s""" |
| 492 | % (include_path)) |
| 493 | runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 494 | % (wks_file, self.resultdir)) |
| 495 | |
| 496 | part1 = glob(os.path.join(self.resultdir, 'temp-*.direct.p1'))[0] |
| 497 | part2 = glob(os.path.join(self.resultdir, 'temp-*.direct.p2'))[0] |
| 498 | |
| 499 | # Test partition 1, should not contain 'test-file' |
| 500 | res = runCmd("debugfs -R 'ls -p' %s 2>/dev/null" % (part1)) |
| 501 | files = extract_files(res.output) |
| 502 | self.assertNotIn('test-file', files) |
| 503 | self.assertEqual(True, files_own_by_root(res.output)) |
| 504 | |
| 505 | # Test partition 2, should contain 'test-file' |
| 506 | res = runCmd("debugfs -R 'ls -p' %s 2>/dev/null" % (part2)) |
| 507 | files = extract_files(res.output) |
| 508 | self.assertIn('test-file', files) |
| 509 | self.assertEqual(True, files_own_by_root(res.output)) |
| 510 | |
| 511 | finally: |
| 512 | os.environ['PATH'] = oldpath |
| 513 | |
| 514 | def test_include_path_embeded(self): |
| 515 | """Test --include-path wks option.""" |
| 516 | |
| 517 | oldpath = os.environ['PATH'] |
| 518 | os.environ['PATH'] = get_bb_var("PATH", "wic-tools") |
| 519 | |
| 520 | try: |
| 521 | include_path = os.path.join(self.resultdir, 'test-include') |
| 522 | os.makedirs(include_path) |
| 523 | with open(os.path.join(include_path, 'test-file'), 'w') as t: |
| 524 | t.write("test\n") |
| 525 | wks_file = os.path.join(include_path, 'temp.wks') |
| 526 | with open(wks_file, 'w') as wks: |
| 527 | wks.write(""" |
| 528 | part / --source rootfs --fstype=ext4 --include-path %s --include-path core-image-minimal-mtdutils export/""" |
| 529 | % (include_path)) |
| 530 | runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 531 | % (wks_file, self.resultdir)) |
| 532 | |
| 533 | part1 = glob(os.path.join(self.resultdir, 'temp-*.direct.p1'))[0] |
| 534 | |
| 535 | res = runCmd("debugfs -R 'ls -p' %s 2>/dev/null" % (part1)) |
| 536 | files = extract_files(res.output) |
| 537 | self.assertIn('test-file', files) |
| 538 | self.assertEqual(True, files_own_by_root(res.output)) |
| 539 | |
| 540 | res = runCmd("debugfs -R 'ls -p /export/etc/' %s 2>/dev/null" % (part1)) |
| 541 | files = extract_files(res.output) |
| 542 | self.assertIn('passwd', files) |
| 543 | self.assertEqual(True, files_own_by_root(res.output)) |
| 544 | |
| 545 | finally: |
| 546 | os.environ['PATH'] = oldpath |
| 547 | |
| 548 | def test_include_path_errors(self): |
| 549 | """Test --include-path wks option error handling.""" |
| 550 | wks_file = 'temp.wks' |
| 551 | |
| 552 | # Absolute argument. |
| 553 | with open(wks_file, 'w') as wks: |
| 554 | wks.write("part / --source rootfs --fstype=ext4 --include-path core-image-minimal-mtdutils /export") |
| 555 | self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 556 | % (wks_file, self.resultdir), ignore_status=True).status) |
| 557 | os.remove(wks_file) |
| 558 | |
| 559 | # Argument pointing to parent directory. |
| 560 | with open(wks_file, 'w') as wks: |
| 561 | wks.write("part / --source rootfs --fstype=ext4 --include-path core-image-minimal-mtdutils ././..") |
| 562 | self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 563 | % (wks_file, self.resultdir), ignore_status=True).status) |
| 564 | os.remove(wks_file) |
| 565 | |
| 566 | # 3 Argument pointing to parent directory. |
| 567 | with open(wks_file, 'w') as wks: |
| 568 | wks.write("part / --source rootfs --fstype=ext4 --include-path core-image-minimal-mtdutils export/ dummy") |
| 569 | self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 570 | % (wks_file, self.resultdir), ignore_status=True).status) |
| 571 | os.remove(wks_file) |
| 572 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 573 | def test_exclude_path_errors(self): |
| 574 | """Test --exclude-path wks option error handling.""" |
| 575 | wks_file = 'temp.wks' |
| 576 | |
| 577 | # Absolute argument. |
| 578 | with open(wks_file, 'w') as wks: |
| 579 | wks.write("part / --source rootfs --ondisk mmcblk0 --fstype=ext4 --exclude-path /usr") |
| 580 | self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 581 | % (wks_file, self.resultdir), ignore_status=True).status) |
| 582 | os.remove(wks_file) |
| 583 | |
| 584 | # Argument pointing to parent directory. |
| 585 | with open(wks_file, 'w') as wks: |
| 586 | wks.write("part / --source rootfs --ondisk mmcblk0 --fstype=ext4 --exclude-path ././..") |
| 587 | self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 588 | % (wks_file, self.resultdir), ignore_status=True).status) |
| 589 | os.remove(wks_file) |
| 590 | |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 591 | def test_permissions(self): |
| 592 | """Test permissions are respected""" |
| 593 | |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 594 | # prepare wicenv and rootfs |
| 595 | bitbake('core-image-minimal core-image-minimal-mtdutils -c do_rootfs_wicenv') |
| 596 | |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 597 | oldpath = os.environ['PATH'] |
| 598 | os.environ['PATH'] = get_bb_var("PATH", "wic-tools") |
| 599 | |
| 600 | t_normal = """ |
| 601 | part / --source rootfs --fstype=ext4 |
| 602 | """ |
| 603 | t_exclude = """ |
| 604 | part / --source rootfs --fstype=ext4 --exclude-path=home |
| 605 | """ |
| 606 | t_multi = """ |
| 607 | part / --source rootfs --ondisk sda --fstype=ext4 |
| 608 | part /export --source rootfs --rootfs=core-image-minimal-mtdutils --fstype=ext4 |
| 609 | """ |
| 610 | t_change = """ |
| 611 | part / --source rootfs --ondisk sda --fstype=ext4 --exclude-path=etc/Â Â |
| 612 | part /etc --source rootfs --fstype=ext4 --change-directory=etc |
| 613 | """ |
| 614 | tests = [t_normal, t_exclude, t_multi, t_change] |
| 615 | |
| 616 | try: |
| 617 | for test in tests: |
| 618 | include_path = os.path.join(self.resultdir, 'test-include') |
| 619 | os.makedirs(include_path) |
| 620 | wks_file = os.path.join(include_path, 'temp.wks') |
| 621 | with open(wks_file, 'w') as wks: |
| 622 | wks.write(test) |
| 623 | runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 624 | % (wks_file, self.resultdir)) |
| 625 | |
| 626 | for part in glob(os.path.join(self.resultdir, 'temp-*.direct.p*')): |
| 627 | res = runCmd("debugfs -R 'ls -p' %s 2>/dev/null" % (part)) |
| 628 | self.assertEqual(True, files_own_by_root(res.output)) |
| 629 | |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 630 | config = 'IMAGE_FSTYPES += "wic"\nWKS_FILE = "%s"\n' % wks_file |
| 631 | self.append_config(config) |
| 632 | bitbake('core-image-minimal') |
| 633 | tmpdir = os.path.join(get_bb_var('WORKDIR', 'core-image-minimal'),'build-wic') |
| 634 | |
| 635 | # check each partition for permission |
| 636 | for part in glob(os.path.join(tmpdir, 'temp-*.direct.p*')): |
| 637 | res = runCmd("debugfs -R 'ls -p' %s 2>/dev/null" % (part)) |
| 638 | self.assertTrue(files_own_by_root(res.output) |
| 639 | ,msg='Files permission incorrect using wks set "%s"' % test) |
| 640 | |
| 641 | # clean config and result directory for next cases |
| 642 | self.remove_config(config) |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 643 | rmtree(self.resultdir, ignore_errors=True) |
| 644 | |
| 645 | finally: |
| 646 | os.environ['PATH'] = oldpath |
| 647 | |
| 648 | def test_change_directory(self): |
| 649 | """Test --change-directory wks option.""" |
| 650 | |
| 651 | oldpath = os.environ['PATH'] |
| 652 | os.environ['PATH'] = get_bb_var("PATH", "wic-tools") |
| 653 | |
| 654 | try: |
| 655 | include_path = os.path.join(self.resultdir, 'test-include') |
| 656 | os.makedirs(include_path) |
| 657 | wks_file = os.path.join(include_path, 'temp.wks') |
| 658 | with open(wks_file, 'w') as wks: |
| 659 | wks.write("part /etc --source rootfs --fstype=ext4 --change-directory=etc") |
| 660 | runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 661 | % (wks_file, self.resultdir)) |
| 662 | |
| 663 | part1 = glob(os.path.join(self.resultdir, 'temp-*.direct.p1'))[0] |
| 664 | |
| 665 | res = runCmd("debugfs -R 'ls -p' %s 2>/dev/null" % (part1)) |
| 666 | files = extract_files(res.output) |
| 667 | self.assertIn('passwd', files) |
| 668 | |
| 669 | finally: |
| 670 | os.environ['PATH'] = oldpath |
| 671 | |
| 672 | def test_change_directory_errors(self): |
| 673 | """Test --change-directory wks option error handling.""" |
| 674 | wks_file = 'temp.wks' |
| 675 | |
| 676 | # Absolute argument. |
| 677 | with open(wks_file, 'w') as wks: |
| 678 | wks.write("part / --source rootfs --fstype=ext4 --change-directory /usr") |
| 679 | self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 680 | % (wks_file, self.resultdir), ignore_status=True).status) |
| 681 | os.remove(wks_file) |
| 682 | |
| 683 | # Argument pointing to parent directory. |
| 684 | with open(wks_file, 'w') as wks: |
| 685 | wks.write("part / --source rootfs --fstype=ext4 --change-directory ././..") |
| 686 | self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 687 | % (wks_file, self.resultdir), ignore_status=True).status) |
| 688 | os.remove(wks_file) |
| 689 | |
Andrew Geissler | d159c7f | 2021-09-02 21:05:58 -0500 | [diff] [blame] | 690 | def test_no_fstab_update(self): |
| 691 | """Test --no-fstab-update wks option.""" |
| 692 | |
| 693 | oldpath = os.environ['PATH'] |
| 694 | os.environ['PATH'] = get_bb_var("PATH", "wic-tools") |
| 695 | |
| 696 | # Get stock fstab from base-files recipe |
| 697 | self.assertEqual(0, bitbake('base-files -c do_install').status) |
| 698 | bf_fstab = os.path.join(get_bb_var('D', 'base-files'), 'etc/fstab') |
| 699 | self.assertEqual(True, os.path.exists(bf_fstab)) |
| 700 | bf_fstab_md5sum = runCmd('md5sum %s 2>/dev/null' % bf_fstab).output.split(" ")[0] |
| 701 | |
| 702 | try: |
| 703 | no_fstab_update_path = os.path.join(self.resultdir, 'test-no-fstab-update') |
| 704 | os.makedirs(no_fstab_update_path) |
| 705 | wks_file = os.path.join(no_fstab_update_path, 'temp.wks') |
| 706 | with open(wks_file, 'w') as wks: |
| 707 | wks.writelines(['part / --source rootfs --fstype=ext4 --label rootfs\n', |
| 708 | 'part /mnt/p2 --source rootfs --rootfs-dir=core-image-minimal ', |
| 709 | '--fstype=ext4 --label p2 --no-fstab-update\n']) |
| 710 | runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 711 | % (wks_file, self.resultdir)) |
| 712 | |
| 713 | part_fstab_md5sum = [] |
| 714 | for i in range(1, 3): |
| 715 | part = glob(os.path.join(self.resultdir, 'temp-*.direct.p') + str(i))[0] |
| 716 | part_fstab = runCmd("debugfs -R 'cat etc/fstab' %s 2>/dev/null" % (part)) |
| 717 | part_fstab_md5sum.append(hashlib.md5((part_fstab.output + "\n\n").encode('utf-8')).hexdigest()) |
| 718 | |
| 719 | # '/etc/fstab' in partition 2 should contain the same stock fstab file |
| 720 | # as the one installed by the base-file recipe. |
| 721 | self.assertEqual(bf_fstab_md5sum, part_fstab_md5sum[1]) |
| 722 | |
| 723 | # '/etc/fstab' in partition 1 should contain an updated fstab file. |
| 724 | self.assertNotEqual(bf_fstab_md5sum, part_fstab_md5sum[0]) |
| 725 | |
| 726 | finally: |
| 727 | os.environ['PATH'] = oldpath |
| 728 | |
| 729 | def test_no_fstab_update_errors(self): |
| 730 | """Test --no-fstab-update wks option error handling.""" |
| 731 | wks_file = 'temp.wks' |
| 732 | |
| 733 | # Absolute argument. |
| 734 | with open(wks_file, 'w') as wks: |
| 735 | wks.write("part / --source rootfs --fstype=ext4 --no-fstab-update /etc") |
| 736 | self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 737 | % (wks_file, self.resultdir), ignore_status=True).status) |
| 738 | os.remove(wks_file) |
| 739 | |
| 740 | # Argument pointing to parent directory. |
| 741 | with open(wks_file, 'w') as wks: |
| 742 | wks.write("part / --source rootfs --fstype=ext4 --no-fstab-update ././..") |
| 743 | self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \ |
| 744 | % (wks_file, self.resultdir), ignore_status=True).status) |
| 745 | os.remove(wks_file) |
| 746 | |
Andrew Geissler | 5199d83 | 2021-09-24 16:47:35 -0500 | [diff] [blame] | 747 | def test_extra_space(self): |
| 748 | """Test --extra-space wks option.""" |
| 749 | extraspace = 1024**3 |
| 750 | runCmd("wic create wictestdisk " |
| 751 | "--image-name core-image-minimal " |
| 752 | "--extra-space %i -o %s" % (extraspace ,self.resultdir)) |
| 753 | wicout = glob(self.resultdir + "wictestdisk-*.direct") |
| 754 | self.assertEqual(1, len(wicout)) |
| 755 | size = os.path.getsize(wicout[0]) |
| 756 | self.assertTrue(size > extraspace) |
| 757 | |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 758 | class Wic2(WicTestCase): |
| 759 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 760 | def test_bmap_short(self): |
| 761 | """Test generation of .bmap file -m option""" |
| 762 | cmd = "wic create wictestdisk -e core-image-minimal -m -o %s" % self.resultdir |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 763 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 764 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*direct"))) |
| 765 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*direct.bmap"))) |
| 766 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 767 | def test_bmap_long(self): |
| 768 | """Test generation of .bmap file --bmap option""" |
| 769 | cmd = "wic create wictestdisk -e core-image-minimal --bmap -o %s" % self.resultdir |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 770 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 771 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*direct"))) |
| 772 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*direct.bmap"))) |
| 773 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 774 | def test_image_env(self): |
| 775 | """Test generation of <image>.env files.""" |
| 776 | image = 'core-image-minimal' |
| 777 | imgdatadir = self._get_image_env_path(image) |
| 778 | |
| 779 | bb_vars = get_bb_vars(['IMAGE_BASENAME', 'WICVARS'], image) |
| 780 | basename = bb_vars['IMAGE_BASENAME'] |
| 781 | self.assertEqual(basename, image) |
| 782 | path = os.path.join(imgdatadir, basename) + '.env' |
| 783 | self.assertTrue(os.path.isfile(path)) |
| 784 | |
| 785 | wicvars = set(bb_vars['WICVARS'].split()) |
| 786 | # filter out optional variables |
| 787 | wicvars = wicvars.difference(('DEPLOY_DIR_IMAGE', 'IMAGE_BOOT_FILES', |
Brad Bishop | 96ff198 | 2019-08-19 13:50:42 -0400 | [diff] [blame] | 788 | 'INITRD', 'INITRD_LIVE', 'ISODIR','INITRAMFS_IMAGE', |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 789 | 'INITRAMFS_IMAGE_BUNDLE', 'INITRAMFS_LINK_NAME', |
Andrew Geissler | c9f7865 | 2020-09-18 14:11:35 -0500 | [diff] [blame] | 790 | 'APPEND', 'IMAGE_EFI_BOOT_FILES')) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 791 | with open(path) as envfile: |
| 792 | content = dict(line.split("=", 1) for line in envfile) |
| 793 | # test if variables used by wic present in the .env file |
| 794 | for var in wicvars: |
| 795 | self.assertTrue(var in content, "%s is not in .env file" % var) |
| 796 | self.assertTrue(content[var]) |
| 797 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 798 | def test_image_vars_dir_short(self): |
| 799 | """Test image vars directory selection -v option""" |
| 800 | image = 'core-image-minimal' |
| 801 | imgenvdir = self._get_image_env_path(image) |
| 802 | native_sysroot = get_bb_var("RECIPE_SYSROOT_NATIVE", "wic-tools") |
| 803 | |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 804 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 805 | "--image-name=%s -v %s -n %s -o %s" |
| 806 | % (image, imgenvdir, native_sysroot, |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 807 | self.resultdir)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 808 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*direct"))) |
| 809 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 810 | def test_image_vars_dir_long(self): |
| 811 | """Test image vars directory selection --vars option""" |
| 812 | image = 'core-image-minimal' |
| 813 | imgenvdir = self._get_image_env_path(image) |
| 814 | native_sysroot = get_bb_var("RECIPE_SYSROOT_NATIVE", "wic-tools") |
| 815 | |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 816 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 817 | "--image-name=%s " |
| 818 | "--vars %s " |
| 819 | "--native-sysroot %s " |
| 820 | "--outdir %s" |
| 821 | % (image, imgenvdir, native_sysroot, |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 822 | self.resultdir)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 823 | self.assertEqual(1, len(glob(self.resultdir + "wictestdisk-*direct"))) |
| 824 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 825 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 826 | def test_wic_image_type(self): |
| 827 | """Test building wic images by bitbake""" |
| 828 | config = 'IMAGE_FSTYPES += "wic"\nWKS_FILE = "wic-image-minimal"\n'\ |
Patrick Williams | 213cb26 | 2021-08-07 19:21:33 -0500 | [diff] [blame] | 829 | 'MACHINE_FEATURES:append = " efi"\n' |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 830 | self.append_config(config) |
| 831 | self.assertEqual(0, bitbake('wic-image-minimal').status) |
| 832 | self.remove_config(config) |
| 833 | |
| 834 | bb_vars = get_bb_vars(['DEPLOY_DIR_IMAGE', 'MACHINE']) |
| 835 | deploy_dir = bb_vars['DEPLOY_DIR_IMAGE'] |
| 836 | machine = bb_vars['MACHINE'] |
| 837 | prefix = os.path.join(deploy_dir, 'wic-image-minimal-%s.' % machine) |
| 838 | # check if we have result image and manifests symlinks |
| 839 | # pointing to existing files |
| 840 | for suffix in ('wic', 'manifest'): |
| 841 | path = prefix + suffix |
| 842 | self.assertTrue(os.path.islink(path)) |
| 843 | self.assertTrue(os.path.isfile(os.path.realpath(path))) |
| 844 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 845 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 846 | def test_qemu(self): |
| 847 | """Test wic-image-minimal under qemu""" |
| 848 | config = 'IMAGE_FSTYPES += "wic"\nWKS_FILE = "wic-image-minimal"\n'\ |
Patrick Williams | 213cb26 | 2021-08-07 19:21:33 -0500 | [diff] [blame] | 849 | 'MACHINE_FEATURES:append = " efi"\n' |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 850 | self.append_config(config) |
| 851 | self.assertEqual(0, bitbake('wic-image-minimal').status) |
| 852 | self.remove_config(config) |
| 853 | |
| 854 | with runqemu('wic-image-minimal', ssh=False) as qemu: |
Andrew Geissler | 99467da | 2019-02-25 18:54:23 -0600 | [diff] [blame] | 855 | cmd = "mount | grep '^/dev/' | cut -f1,3 -d ' ' | egrep -c -e '/dev/sda1 /boot' " \ |
| 856 | "-e '/dev/root /|/dev/sda2 /' -e '/dev/sda3 /media' -e '/dev/sda4 /mnt'" |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 857 | status, output = qemu.run_serial(cmd) |
Andrew Geissler | 99467da | 2019-02-25 18:54:23 -0600 | [diff] [blame] | 858 | self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output)) |
| 859 | self.assertEqual(output, '4') |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 860 | cmd = "grep UUID= /etc/fstab" |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 861 | status, output = qemu.run_serial(cmd) |
| 862 | self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output)) |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 863 | self.assertEqual(output, 'UUID=2c71ef06-a81d-4735-9d3a-379b69c6bdba\t/media\text4\tdefaults\t0\t0') |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 864 | |
| 865 | @only_for_arch(['i586', 'i686', 'x86_64']) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 866 | def test_qemu_efi(self): |
| 867 | """Test core-image-minimal efi image under qemu""" |
| 868 | config = 'IMAGE_FSTYPES = "wic"\nWKS_FILE = "mkefidisk.wks"\n' |
| 869 | self.append_config(config) |
| 870 | self.assertEqual(0, bitbake('core-image-minimal ovmf').status) |
| 871 | self.remove_config(config) |
| 872 | |
| 873 | with runqemu('core-image-minimal', ssh=False, |
| 874 | runqemuparams='ovmf', image_fstype='wic') as qemu: |
| 875 | cmd = "grep sda. /proc/partitions |wc -l" |
| 876 | status, output = qemu.run_serial(cmd) |
| 877 | self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output)) |
| 878 | self.assertEqual(output, '3') |
| 879 | |
| 880 | @staticmethod |
| 881 | def _make_fixed_size_wks(size): |
| 882 | """ |
| 883 | Create a wks of an image with a single partition. Size of the partition is set |
| 884 | using --fixed-size flag. Returns a tuple: (path to wks file, wks image name) |
| 885 | """ |
| 886 | with NamedTemporaryFile("w", suffix=".wks", delete=False) as tempf: |
| 887 | wkspath = tempf.name |
| 888 | tempf.write("part " \ |
| 889 | "--source rootfs --ondisk hda --align 4 --fixed-size %d " |
| 890 | "--fstype=ext4\n" % size) |
Andrew Geissler | 4ed12e1 | 2020-06-05 18:00:41 -0500 | [diff] [blame] | 891 | |
| 892 | return wkspath |
| 893 | |
| 894 | def _get_wic_partitions(self, wkspath, native_sysroot=None, ignore_status=False): |
| 895 | p = runCmd("wic create %s -e core-image-minimal -o %s" % (wkspath, self.resultdir), |
| 896 | ignore_status=ignore_status) |
| 897 | |
| 898 | if p.status: |
| 899 | return (p, None) |
| 900 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 901 | wksname = os.path.splitext(os.path.basename(wkspath))[0] |
| 902 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 903 | wicout = glob(self.resultdir + "%s-*direct" % wksname) |
Andrew Geissler | 4ed12e1 | 2020-06-05 18:00:41 -0500 | [diff] [blame] | 904 | |
| 905 | if not wicout: |
| 906 | return (p, None) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 907 | |
| 908 | wicimg = wicout[0] |
| 909 | |
Andrew Geissler | 4ed12e1 | 2020-06-05 18:00:41 -0500 | [diff] [blame] | 910 | if not native_sysroot: |
| 911 | native_sysroot = get_bb_var("RECIPE_SYSROOT_NATIVE", "wic-tools") |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 912 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 913 | # verify partition size with wic |
Andrew Geissler | 4ed12e1 | 2020-06-05 18:00:41 -0500 | [diff] [blame] | 914 | res = runCmd("parted -m %s unit kib p 2>/dev/null" % wicimg, |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 915 | native_sysroot=native_sysroot) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 916 | |
| 917 | # parse parted output which looks like this: |
| 918 | # BYT;\n |
| 919 | # /var/tmp/wic/build/tmpfwvjjkf_-201611101222-hda.direct:200MiB:file:512:512:msdos::;\n |
| 920 | # 1:0.00MiB:200MiB:200MiB:ext4::;\n |
Andrew Geissler | 4ed12e1 | 2020-06-05 18:00:41 -0500 | [diff] [blame] | 921 | return (p, res.output.splitlines()[2:]) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 922 | |
Andrew Geissler | 4ed12e1 | 2020-06-05 18:00:41 -0500 | [diff] [blame] | 923 | def test_fixed_size(self): |
| 924 | """ |
| 925 | Test creation of a simple image with partition size controlled through |
| 926 | --fixed-size flag |
| 927 | """ |
| 928 | wkspath = Wic2._make_fixed_size_wks(200) |
| 929 | _, partlns = self._get_wic_partitions(wkspath) |
| 930 | os.remove(wkspath) |
| 931 | |
| 932 | self.assertEqual(partlns, [ |
| 933 | "1:4.00kiB:204804kiB:204800kiB:ext4::;", |
| 934 | ]) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 935 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 936 | def test_fixed_size_error(self): |
| 937 | """ |
| 938 | Test creation of a simple image with partition size controlled through |
| 939 | --fixed-size flag. The size of partition is intentionally set to 1MiB |
| 940 | in order to trigger an error in wic. |
| 941 | """ |
Andrew Geissler | 4ed12e1 | 2020-06-05 18:00:41 -0500 | [diff] [blame] | 942 | wkspath = Wic2._make_fixed_size_wks(1) |
| 943 | p, _ = self._get_wic_partitions(wkspath, ignore_status=True) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 944 | os.remove(wkspath) |
Andrew Geissler | 4ed12e1 | 2020-06-05 18:00:41 -0500 | [diff] [blame] | 945 | |
| 946 | self.assertNotEqual(p.status, 0, "wic exited successfully when an error was expected:\n%s" % p.output) |
| 947 | |
| 948 | def test_offset(self): |
| 949 | native_sysroot = get_bb_var("RECIPE_SYSROOT_NATIVE", "wic-tools") |
| 950 | |
| 951 | with NamedTemporaryFile("w", suffix=".wks") as tempf: |
| 952 | # Test that partitions are placed at the correct offsets, default KB |
| 953 | tempf.write("bootloader --ptable gpt\n" \ |
| 954 | "part / --source rootfs --ondisk hda --offset 32 --fixed-size 100M --fstype=ext4\n" \ |
| 955 | "part /bar --ondisk hda --offset 102432 --fixed-size 100M --fstype=ext4\n") |
| 956 | tempf.flush() |
| 957 | |
| 958 | _, partlns = self._get_wic_partitions(tempf.name, native_sysroot) |
| 959 | self.assertEqual(partlns, [ |
| 960 | "1:32.0kiB:102432kiB:102400kiB:ext4:primary:;", |
| 961 | "2:102432kiB:204832kiB:102400kiB:ext4:primary:;", |
| 962 | ]) |
| 963 | |
| 964 | with NamedTemporaryFile("w", suffix=".wks") as tempf: |
| 965 | # Test that partitions are placed at the correct offsets, same with explicit KB |
| 966 | tempf.write("bootloader --ptable gpt\n" \ |
| 967 | "part / --source rootfs --ondisk hda --offset 32K --fixed-size 100M --fstype=ext4\n" \ |
| 968 | "part /bar --ondisk hda --offset 102432K --fixed-size 100M --fstype=ext4\n") |
| 969 | tempf.flush() |
| 970 | |
| 971 | _, partlns = self._get_wic_partitions(tempf.name, native_sysroot) |
| 972 | self.assertEqual(partlns, [ |
| 973 | "1:32.0kiB:102432kiB:102400kiB:ext4:primary:;", |
| 974 | "2:102432kiB:204832kiB:102400kiB:ext4:primary:;", |
| 975 | ]) |
| 976 | |
| 977 | with NamedTemporaryFile("w", suffix=".wks") as tempf: |
| 978 | # Test that partitions are placed at the correct offsets using MB |
| 979 | tempf.write("bootloader --ptable gpt\n" \ |
| 980 | "part / --source rootfs --ondisk hda --offset 32K --fixed-size 100M --fstype=ext4\n" \ |
| 981 | "part /bar --ondisk hda --offset 101M --fixed-size 100M --fstype=ext4\n") |
| 982 | tempf.flush() |
| 983 | |
| 984 | _, partlns = self._get_wic_partitions(tempf.name, native_sysroot) |
| 985 | self.assertEqual(partlns, [ |
| 986 | "1:32.0kiB:102432kiB:102400kiB:ext4:primary:;", |
| 987 | "2:103424kiB:205824kiB:102400kiB:ext4:primary:;", |
| 988 | ]) |
| 989 | |
| 990 | with NamedTemporaryFile("w", suffix=".wks") as tempf: |
Andrew Geissler | c9f7865 | 2020-09-18 14:11:35 -0500 | [diff] [blame] | 991 | # Test that partitions can be placed on a 512 byte sector boundary |
| 992 | tempf.write("bootloader --ptable gpt\n" \ |
| 993 | "part / --source rootfs --ondisk hda --offset 65s --fixed-size 99M --fstype=ext4\n" \ |
| 994 | "part /bar --ondisk hda --offset 102432 --fixed-size 100M --fstype=ext4\n") |
| 995 | tempf.flush() |
| 996 | |
| 997 | _, partlns = self._get_wic_partitions(tempf.name, native_sysroot) |
| 998 | self.assertEqual(partlns, [ |
| 999 | "1:32.5kiB:101408kiB:101376kiB:ext4:primary:;", |
| 1000 | "2:102432kiB:204832kiB:102400kiB:ext4:primary:;", |
| 1001 | ]) |
| 1002 | |
| 1003 | with NamedTemporaryFile("w", suffix=".wks") as tempf: |
| 1004 | # Test that a partition can be placed immediately after a MSDOS partition table |
| 1005 | tempf.write("bootloader --ptable msdos\n" \ |
| 1006 | "part / --source rootfs --ondisk hda --offset 1s --fixed-size 100M --fstype=ext4\n") |
| 1007 | tempf.flush() |
| 1008 | |
| 1009 | _, partlns = self._get_wic_partitions(tempf.name, native_sysroot) |
| 1010 | self.assertEqual(partlns, [ |
| 1011 | "1:0.50kiB:102400kiB:102400kiB:ext4::;", |
| 1012 | ]) |
| 1013 | |
| 1014 | with NamedTemporaryFile("w", suffix=".wks") as tempf: |
Andrew Geissler | 4ed12e1 | 2020-06-05 18:00:41 -0500 | [diff] [blame] | 1015 | # Test that image creation fails if the partitions would overlap |
| 1016 | tempf.write("bootloader --ptable gpt\n" \ |
| 1017 | "part / --source rootfs --ondisk hda --offset 32 --fixed-size 100M --fstype=ext4\n" \ |
| 1018 | "part /bar --ondisk hda --offset 102431 --fixed-size 100M --fstype=ext4\n") |
| 1019 | tempf.flush() |
| 1020 | |
| 1021 | p, _ = self._get_wic_partitions(tempf.name, ignore_status=True) |
| 1022 | self.assertNotEqual(p.status, 0, "wic exited successfully when an error was expected:\n%s" % p.output) |
| 1023 | |
| 1024 | with NamedTemporaryFile("w", suffix=".wks") as tempf: |
| 1025 | # Test that partitions are not allowed to overlap with the booloader |
| 1026 | tempf.write("bootloader --ptable gpt\n" \ |
| 1027 | "part / --source rootfs --ondisk hda --offset 8 --fixed-size 100M --fstype=ext4\n") |
| 1028 | tempf.flush() |
| 1029 | |
| 1030 | p, _ = self._get_wic_partitions(tempf.name, ignore_status=True) |
| 1031 | self.assertNotEqual(p.status, 0, "wic exited successfully when an error was expected:\n%s" % p.output) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1032 | |
Andrew Geissler | 5a43b43 | 2020-06-13 10:46:56 -0500 | [diff] [blame] | 1033 | def test_extra_space(self): |
| 1034 | native_sysroot = get_bb_var("RECIPE_SYSROOT_NATIVE", "wic-tools") |
| 1035 | |
| 1036 | with NamedTemporaryFile("w", suffix=".wks") as tempf: |
| 1037 | tempf.write("bootloader --ptable gpt\n" \ |
| 1038 | "part / --source rootfs --ondisk hda --extra-space 200M --fstype=ext4\n") |
| 1039 | tempf.flush() |
| 1040 | |
| 1041 | _, partlns = self._get_wic_partitions(tempf.name, native_sysroot) |
| 1042 | self.assertEqual(len(partlns), 1) |
| 1043 | size = partlns[0].split(':')[3] |
| 1044 | self.assertRegex(size, r'^[0-9]+kiB$') |
| 1045 | size = int(size[:-3]) |
| 1046 | self.assertGreaterEqual(size, 204800) |
| 1047 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1048 | @only_for_arch(['i586', 'i686', 'x86_64']) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1049 | def test_rawcopy_plugin_qemu(self): |
| 1050 | """Test rawcopy plugin in qemu""" |
Andrew Geissler | 95ac1b8 | 2021-03-31 14:34:31 -0500 | [diff] [blame] | 1051 | # build ext4 and then use it for a wic image |
| 1052 | config = 'IMAGE_FSTYPES = "ext4"\n' |
| 1053 | self.append_config(config) |
| 1054 | self.assertEqual(0, bitbake('core-image-minimal').status) |
| 1055 | self.remove_config(config) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1056 | |
Andrew Geissler | 95ac1b8 | 2021-03-31 14:34:31 -0500 | [diff] [blame] | 1057 | config = 'IMAGE_FSTYPES = "wic"\nWKS_FILE = "test_rawcopy_plugin.wks.in"\n' |
| 1058 | self.append_config(config) |
| 1059 | self.assertEqual(0, bitbake('core-image-minimal-mtdutils').status) |
| 1060 | self.remove_config(config) |
| 1061 | |
| 1062 | with runqemu('core-image-minimal-mtdutils', ssh=False, image_fstype='wic') as qemu: |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1063 | cmd = "grep sda. /proc/partitions |wc -l" |
| 1064 | status, output = qemu.run_serial(cmd) |
| 1065 | self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output)) |
| 1066 | self.assertEqual(output, '2') |
| 1067 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1068 | def test_rawcopy_plugin(self): |
| 1069 | """Test rawcopy plugin""" |
| 1070 | img = 'core-image-minimal' |
| 1071 | machine = get_bb_var('MACHINE', img) |
| 1072 | with NamedTemporaryFile("w", suffix=".wks") as wks: |
| 1073 | wks.writelines(['part /boot --active --source bootimg-pcbios\n', |
| 1074 | 'part / --source rawcopy --sourceparams="file=%s-%s.ext4" --use-uuid\n'\ |
| 1075 | % (img, machine), |
| 1076 | 'bootloader --timeout=0 --append="console=ttyS0,115200n8"\n']) |
| 1077 | wks.flush() |
| 1078 | cmd = "wic create %s -e %s -o %s" % (wks.name, img, self.resultdir) |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1079 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1080 | wksname = os.path.splitext(os.path.basename(wks.name))[0] |
| 1081 | out = glob(self.resultdir + "%s-*direct" % wksname) |
| 1082 | self.assertEqual(1, len(out)) |
| 1083 | |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 1084 | def test_empty_plugin(self): |
| 1085 | """Test empty plugin""" |
| 1086 | config = 'IMAGE_FSTYPES = "wic"\nWKS_FILE = "test_empty_plugin.wks"\n' |
| 1087 | self.append_config(config) |
| 1088 | self.assertEqual(0, bitbake('core-image-minimal').status) |
| 1089 | self.remove_config(config) |
| 1090 | |
| 1091 | bb_vars = get_bb_vars(['DEPLOY_DIR_IMAGE', 'MACHINE']) |
| 1092 | deploy_dir = bb_vars['DEPLOY_DIR_IMAGE'] |
| 1093 | machine = bb_vars['MACHINE'] |
| 1094 | image_path = os.path.join(deploy_dir, 'core-image-minimal-%s.wic' % machine) |
| 1095 | self.assertEqual(True, os.path.exists(image_path)) |
| 1096 | |
| 1097 | sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools') |
| 1098 | |
| 1099 | # Fstype column from 'wic ls' should be empty for the second partition |
| 1100 | # as listed in test_empty_plugin.wks |
| 1101 | result = runCmd("wic ls %s -n %s | awk -F ' ' '{print $1 \" \" $5}' | grep '^2' | wc -w" % (image_path, sysroot)) |
| 1102 | self.assertEqual('1', result.output) |
| 1103 | |
Brad Bishop | 96ff198 | 2019-08-19 13:50:42 -0400 | [diff] [blame] | 1104 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 1105 | def test_biosplusefi_plugin_qemu(self): |
| 1106 | """Test biosplusefi plugin in qemu""" |
Patrick Williams | 213cb26 | 2021-08-07 19:21:33 -0500 | [diff] [blame] | 1107 | config = 'IMAGE_FSTYPES = "wic"\nWKS_FILE = "test_biosplusefi_plugin.wks"\nMACHINE_FEATURES:append = " efi"\n' |
Brad Bishop | 08902b0 | 2019-08-20 09:16:51 -0400 | [diff] [blame] | 1108 | self.append_config(config) |
| 1109 | self.assertEqual(0, bitbake('core-image-minimal').status) |
| 1110 | self.remove_config(config) |
Brad Bishop | 96ff198 | 2019-08-19 13:50:42 -0400 | [diff] [blame] | 1111 | |
| 1112 | with runqemu('core-image-minimal', ssh=False, image_fstype='wic') as qemu: |
| 1113 | # Check that we have ONLY two /dev/sda* partitions (/boot and /) |
| 1114 | cmd = "grep sda. /proc/partitions | wc -l" |
| 1115 | status, output = qemu.run_serial(cmd) |
| 1116 | self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output)) |
| 1117 | self.assertEqual(output, '2') |
| 1118 | # Check that /dev/sda1 is /boot and that either /dev/root OR /dev/sda2 is / |
| 1119 | cmd = "mount | grep '^/dev/' | cut -f1,3 -d ' ' | egrep -c -e '/dev/sda1 /boot' -e '/dev/root /|/dev/sda2 /'" |
| 1120 | status, output = qemu.run_serial(cmd) |
| 1121 | self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output)) |
| 1122 | self.assertEqual(output, '2') |
| 1123 | # Check that /boot has EFI bootx64.efi (required for EFI) |
| 1124 | cmd = "ls /boot/EFI/BOOT/bootx64.efi | wc -l" |
| 1125 | status, output = qemu.run_serial(cmd) |
| 1126 | self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output)) |
| 1127 | self.assertEqual(output, '1') |
| 1128 | # Check that "BOOTABLE" flag is set on boot partition (required for PC-Bios) |
| 1129 | # Trailing "cat" seems to be required; otherwise run_serial() sends back echo of the input command |
| 1130 | cmd = "fdisk -l /dev/sda | grep /dev/sda1 | awk {print'$2'} | cat" |
| 1131 | status, output = qemu.run_serial(cmd) |
| 1132 | self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output)) |
| 1133 | self.assertEqual(output, '*') |
| 1134 | |
| 1135 | @only_for_arch(['i586', 'i686', 'x86_64']) |
| 1136 | def test_biosplusefi_plugin(self): |
| 1137 | """Test biosplusefi plugin""" |
| 1138 | # Wic generation below may fail depending on the order of the unittests |
| 1139 | # This is because bootimg-pcbios (that bootimg-biosplusefi uses) generate its MBR inside STAGING_DATADIR directory |
| 1140 | # which may or may not exists depending on what was built already |
| 1141 | # If an image hasn't been built yet, directory ${STAGING_DATADIR}/syslinux won't exists and _get_bootimg_dir() |
| 1142 | # will raise with "Couldn't find correct bootimg_dir" |
| 1143 | # The easiest way to work-around this issue is to make sure we already built an image here, hence the bitbake call |
Patrick Williams | 213cb26 | 2021-08-07 19:21:33 -0500 | [diff] [blame] | 1144 | config = 'IMAGE_FSTYPES = "wic"\nWKS_FILE = "test_biosplusefi_plugin.wks"\nMACHINE_FEATURES:append = " efi"\n' |
Brad Bishop | 08902b0 | 2019-08-20 09:16:51 -0400 | [diff] [blame] | 1145 | self.append_config(config) |
| 1146 | self.assertEqual(0, bitbake('core-image-minimal').status) |
| 1147 | self.remove_config(config) |
Brad Bishop | 96ff198 | 2019-08-19 13:50:42 -0400 | [diff] [blame] | 1148 | |
| 1149 | img = 'core-image-minimal' |
| 1150 | with NamedTemporaryFile("w", suffix=".wks") as wks: |
| 1151 | wks.writelines(['part /boot --active --source bootimg-biosplusefi --sourceparams="loader=grub-efi"\n', |
| 1152 | 'part / --source rootfs --fstype=ext4 --align 1024 --use-uuid\n'\ |
| 1153 | 'bootloader --timeout=0 --append="console=ttyS0,115200n8"\n']) |
| 1154 | wks.flush() |
| 1155 | cmd = "wic create %s -e %s -o %s" % (wks.name, img, self.resultdir) |
| 1156 | runCmd(cmd) |
| 1157 | wksname = os.path.splitext(os.path.basename(wks.name))[0] |
| 1158 | out = glob(self.resultdir + "%s-*.direct" % wksname) |
| 1159 | self.assertEqual(1, len(out)) |
| 1160 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1161 | def test_fs_types(self): |
| 1162 | """Test filesystem types for empty and not empty partitions""" |
| 1163 | img = 'core-image-minimal' |
| 1164 | with NamedTemporaryFile("w", suffix=".wks") as wks: |
| 1165 | wks.writelines(['part ext2 --fstype ext2 --source rootfs\n', |
| 1166 | 'part btrfs --fstype btrfs --source rootfs --size 40M\n', |
| 1167 | 'part squash --fstype squashfs --source rootfs\n', |
| 1168 | 'part swap --fstype swap --size 1M\n', |
| 1169 | 'part emptyvfat --fstype vfat --size 1M\n', |
| 1170 | 'part emptymsdos --fstype msdos --size 1M\n', |
| 1171 | 'part emptyext2 --fstype ext2 --size 1M\n', |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1172 | 'part emptybtrfs --fstype btrfs --size 150M\n']) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1173 | wks.flush() |
| 1174 | cmd = "wic create %s -e %s -o %s" % (wks.name, img, self.resultdir) |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1175 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1176 | wksname = os.path.splitext(os.path.basename(wks.name))[0] |
| 1177 | out = glob(self.resultdir + "%s-*direct" % wksname) |
| 1178 | self.assertEqual(1, len(out)) |
| 1179 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1180 | def test_kickstart_parser(self): |
| 1181 | """Test wks parser options""" |
| 1182 | with NamedTemporaryFile("w", suffix=".wks") as wks: |
| 1183 | wks.writelines(['part / --fstype ext3 --source rootfs --system-id 0xFF '\ |
| 1184 | '--overhead-factor 1.2 --size 100k\n']) |
| 1185 | wks.flush() |
| 1186 | cmd = "wic create %s -e core-image-minimal -o %s" % (wks.name, self.resultdir) |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1187 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1188 | wksname = os.path.splitext(os.path.basename(wks.name))[0] |
| 1189 | out = glob(self.resultdir + "%s-*direct" % wksname) |
| 1190 | self.assertEqual(1, len(out)) |
| 1191 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1192 | def test_image_bootpart_globbed(self): |
| 1193 | """Test globbed sources with image-bootpart plugin""" |
| 1194 | img = "core-image-minimal" |
| 1195 | cmd = "wic create sdimage-bootpart -e %s -o %s" % (img, self.resultdir) |
| 1196 | config = 'IMAGE_BOOT_FILES = "%s*"' % get_bb_var('KERNEL_IMAGETYPE', img) |
| 1197 | self.append_config(config) |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1198 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1199 | self.remove_config(config) |
| 1200 | self.assertEqual(1, len(glob(self.resultdir + "sdimage-bootpart-*direct"))) |
| 1201 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1202 | def test_sparse_copy(self): |
| 1203 | """Test sparse_copy with FIEMAP and SEEK_HOLE filemap APIs""" |
| 1204 | libpath = os.path.join(get_bb_var('COREBASE'), 'scripts', 'lib', 'wic') |
| 1205 | sys.path.insert(0, libpath) |
| 1206 | from filemap import FilemapFiemap, FilemapSeek, sparse_copy, ErrorNotSupp |
| 1207 | with NamedTemporaryFile("w", suffix=".wic-sparse") as sparse: |
| 1208 | src_name = sparse.name |
| 1209 | src_size = 1024 * 10 |
| 1210 | sparse.truncate(src_size) |
| 1211 | # write one byte to the file |
| 1212 | with open(src_name, 'r+b') as sfile: |
| 1213 | sfile.seek(1024 * 4) |
| 1214 | sfile.write(b'\x00') |
| 1215 | dest = sparse.name + '.out' |
| 1216 | # copy src file to dest using different filemap APIs |
| 1217 | for api in (FilemapFiemap, FilemapSeek, None): |
| 1218 | if os.path.exists(dest): |
| 1219 | os.unlink(dest) |
| 1220 | try: |
| 1221 | sparse_copy(sparse.name, dest, api=api) |
| 1222 | except ErrorNotSupp: |
| 1223 | continue # skip unsupported API |
| 1224 | dest_stat = os.stat(dest) |
| 1225 | self.assertEqual(dest_stat.st_size, src_size) |
| 1226 | # 8 blocks is 4K (physical sector size) |
| 1227 | self.assertEqual(dest_stat.st_blocks, 8) |
| 1228 | os.unlink(dest) |
| 1229 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1230 | def test_wic_ls(self): |
| 1231 | """Test listing image content using 'wic ls'""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1232 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1233 | "--image-name=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1234 | "-D -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1235 | images = glob(self.resultdir + "wictestdisk-*.direct") |
| 1236 | self.assertEqual(1, len(images)) |
| 1237 | |
| 1238 | sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools') |
| 1239 | |
| 1240 | # list partitions |
| 1241 | result = runCmd("wic ls %s -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1242 | self.assertEqual(3, len(result.output.split('\n'))) |
| 1243 | |
| 1244 | # list directory content of the first partition |
| 1245 | result = runCmd("wic ls %s:1/ -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1246 | self.assertEqual(6, len(result.output.split('\n'))) |
| 1247 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1248 | def test_wic_cp(self): |
| 1249 | """Test copy files and directories to the the wic image.""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1250 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1251 | "--image-name=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1252 | "-D -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1253 | images = glob(self.resultdir + "wictestdisk-*.direct") |
| 1254 | self.assertEqual(1, len(images)) |
| 1255 | |
| 1256 | sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools') |
| 1257 | |
| 1258 | # list directory content of the first partition |
| 1259 | result = runCmd("wic ls %s:1/ -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1260 | self.assertEqual(6, len(result.output.split('\n'))) |
| 1261 | |
| 1262 | with NamedTemporaryFile("w", suffix=".wic-cp") as testfile: |
| 1263 | testfile.write("test") |
| 1264 | |
| 1265 | # copy file to the partition |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1266 | runCmd("wic cp %s %s:1/ -n %s" % (testfile.name, images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1267 | |
| 1268 | # check if file is there |
| 1269 | result = runCmd("wic ls %s:1/ -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1270 | self.assertEqual(7, len(result.output.split('\n'))) |
| 1271 | self.assertTrue(os.path.basename(testfile.name) in result.output) |
| 1272 | |
| 1273 | # prepare directory |
| 1274 | testdir = os.path.join(self.resultdir, 'wic-test-cp-dir') |
| 1275 | testsubdir = os.path.join(testdir, 'subdir') |
| 1276 | os.makedirs(os.path.join(testsubdir)) |
| 1277 | copy(testfile.name, testdir) |
| 1278 | |
| 1279 | # copy directory to the partition |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1280 | runCmd("wic cp %s %s:1/ -n %s" % (testdir, images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1281 | |
| 1282 | # check if directory is there |
| 1283 | result = runCmd("wic ls %s:1/ -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1284 | self.assertEqual(8, len(result.output.split('\n'))) |
| 1285 | self.assertTrue(os.path.basename(testdir) in result.output) |
| 1286 | |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 1287 | # copy the file from the partition and check if it success |
| 1288 | dest = '%s-cp' % testfile.name |
| 1289 | runCmd("wic cp %s:1/%s %s -n %s" % (images[0], |
| 1290 | os.path.basename(testfile.name), dest, sysroot)) |
| 1291 | self.assertTrue(os.path.exists(dest)) |
| 1292 | |
| 1293 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1294 | def test_wic_rm(self): |
| 1295 | """Test removing files and directories from the the wic image.""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1296 | runCmd("wic create mkefidisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1297 | "--image-name=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1298 | "-D -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1299 | images = glob(self.resultdir + "mkefidisk-*.direct") |
| 1300 | self.assertEqual(1, len(images)) |
| 1301 | |
| 1302 | sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools') |
| 1303 | |
| 1304 | # list directory content of the first partition |
| 1305 | result = runCmd("wic ls %s:1 -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1306 | self.assertIn('\nBZIMAGE ', result.output) |
| 1307 | self.assertIn('\nEFI <DIR> ', result.output) |
| 1308 | |
| 1309 | # remove file |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1310 | runCmd("wic rm %s:1/bzimage -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1311 | |
| 1312 | # remove directory |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1313 | runCmd("wic rm %s:1/efi -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1314 | |
| 1315 | # check if they're removed |
| 1316 | result = runCmd("wic ls %s:1 -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1317 | self.assertNotIn('\nBZIMAGE ', result.output) |
| 1318 | self.assertNotIn('\nEFI <DIR> ', result.output) |
| 1319 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1320 | def test_mkfs_extraopts(self): |
| 1321 | """Test wks option --mkfs-extraopts for empty and not empty partitions""" |
| 1322 | img = 'core-image-minimal' |
| 1323 | with NamedTemporaryFile("w", suffix=".wks") as wks: |
| 1324 | wks.writelines( |
| 1325 | ['part ext2 --fstype ext2 --source rootfs --mkfs-extraopts "-D -F -i 8192"\n', |
| 1326 | "part btrfs --fstype btrfs --source rootfs --size 40M --mkfs-extraopts='--quiet'\n", |
| 1327 | 'part squash --fstype squashfs --source rootfs --mkfs-extraopts "-no-sparse -b 4096"\n', |
| 1328 | 'part emptyvfat --fstype vfat --size 1M --mkfs-extraopts "-S 1024 -s 64"\n', |
| 1329 | 'part emptymsdos --fstype msdos --size 1M --mkfs-extraopts "-S 1024 -s 64"\n', |
| 1330 | 'part emptyext2 --fstype ext2 --size 1M --mkfs-extraopts "-D -F -i 8192"\n', |
| 1331 | 'part emptybtrfs --fstype btrfs --size 100M --mkfs-extraopts "--mixed -K"\n']) |
| 1332 | wks.flush() |
| 1333 | cmd = "wic create %s -e %s -o %s" % (wks.name, img, self.resultdir) |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1334 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1335 | wksname = os.path.splitext(os.path.basename(wks.name))[0] |
| 1336 | out = glob(self.resultdir + "%s-*direct" % wksname) |
| 1337 | self.assertEqual(1, len(out)) |
| 1338 | |
| 1339 | def test_expand_mbr_image(self): |
| 1340 | """Test wic write --expand command for mbr image""" |
| 1341 | # build an image |
| 1342 | config = 'IMAGE_FSTYPES = "wic"\nWKS_FILE = "directdisk.wks"\n' |
| 1343 | self.append_config(config) |
| 1344 | self.assertEqual(0, bitbake('core-image-minimal').status) |
| 1345 | |
| 1346 | # get path to the image |
| 1347 | bb_vars = get_bb_vars(['DEPLOY_DIR_IMAGE', 'MACHINE']) |
| 1348 | deploy_dir = bb_vars['DEPLOY_DIR_IMAGE'] |
| 1349 | machine = bb_vars['MACHINE'] |
| 1350 | image_path = os.path.join(deploy_dir, 'core-image-minimal-%s.wic' % machine) |
| 1351 | |
| 1352 | self.remove_config(config) |
| 1353 | |
| 1354 | try: |
| 1355 | # expand image to 1G |
| 1356 | new_image_path = None |
| 1357 | with NamedTemporaryFile(mode='wb', suffix='.wic.exp', |
| 1358 | dir=deploy_dir, delete=False) as sparse: |
| 1359 | sparse.truncate(1024 ** 3) |
| 1360 | new_image_path = sparse.name |
| 1361 | |
| 1362 | sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools') |
| 1363 | cmd = "wic write -n %s --expand 1:0 %s %s" % (sysroot, image_path, new_image_path) |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1364 | runCmd(cmd) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1365 | |
| 1366 | # check if partitions are expanded |
| 1367 | orig = runCmd("wic ls %s -n %s" % (image_path, sysroot)) |
| 1368 | exp = runCmd("wic ls %s -n %s" % (new_image_path, sysroot)) |
| 1369 | orig_sizes = [int(line.split()[3]) for line in orig.output.split('\n')[1:]] |
| 1370 | exp_sizes = [int(line.split()[3]) for line in exp.output.split('\n')[1:]] |
| 1371 | self.assertEqual(orig_sizes[0], exp_sizes[0]) # first partition is not resized |
| 1372 | self.assertTrue(orig_sizes[1] < exp_sizes[1]) |
| 1373 | |
| 1374 | # Check if all free space is partitioned |
| 1375 | result = runCmd("%s/usr/sbin/sfdisk -F %s" % (sysroot, new_image_path)) |
| 1376 | self.assertTrue("0 B, 0 bytes, 0 sectors" in result.output) |
| 1377 | |
Andrew Geissler | c926e17 | 2021-05-07 16:11:35 -0500 | [diff] [blame] | 1378 | bb.utils.rename(image_path, image_path + '.bak') |
| 1379 | bb.utils.rename(new_image_path, image_path) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1380 | |
| 1381 | # Check if it boots in qemu |
| 1382 | with runqemu('core-image-minimal', ssh=False) as qemu: |
| 1383 | cmd = "ls /etc/" |
| 1384 | status, output = qemu.run_serial('true') |
| 1385 | self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output)) |
| 1386 | finally: |
| 1387 | if os.path.exists(new_image_path): |
| 1388 | os.unlink(new_image_path) |
| 1389 | if os.path.exists(image_path + '.bak'): |
Andrew Geissler | c926e17 | 2021-05-07 16:11:35 -0500 | [diff] [blame] | 1390 | bb.utils.rename(image_path + '.bak', image_path) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1391 | |
| 1392 | def test_wic_ls_ext(self): |
| 1393 | """Test listing content of the ext partition using 'wic ls'""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1394 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1395 | "--image-name=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1396 | "-D -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1397 | images = glob(self.resultdir + "wictestdisk-*.direct") |
| 1398 | self.assertEqual(1, len(images)) |
| 1399 | |
| 1400 | sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools') |
| 1401 | |
| 1402 | # list directory content of the second ext4 partition |
| 1403 | result = runCmd("wic ls %s:2/ -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1404 | self.assertTrue(set(['bin', 'home', 'proc', 'usr', 'var', 'dev', 'lib', 'sbin']).issubset( |
| 1405 | set(line.split()[-1] for line in result.output.split('\n') if line))) |
| 1406 | |
| 1407 | def test_wic_cp_ext(self): |
| 1408 | """Test copy files and directories to the ext partition.""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1409 | runCmd("wic create wictestdisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1410 | "--image-name=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1411 | "-D -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1412 | images = glob(self.resultdir + "wictestdisk-*.direct") |
| 1413 | self.assertEqual(1, len(images)) |
| 1414 | |
| 1415 | sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools') |
| 1416 | |
| 1417 | # list directory content of the ext4 partition |
| 1418 | result = runCmd("wic ls %s:2/ -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1419 | dirs = set(line.split()[-1] for line in result.output.split('\n') if line) |
| 1420 | self.assertTrue(set(['bin', 'home', 'proc', 'usr', 'var', 'dev', 'lib', 'sbin']).issubset(dirs)) |
| 1421 | |
| 1422 | with NamedTemporaryFile("w", suffix=".wic-cp") as testfile: |
| 1423 | testfile.write("test") |
| 1424 | |
| 1425 | # copy file to the partition |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1426 | runCmd("wic cp %s %s:2/ -n %s" % (testfile.name, images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1427 | |
| 1428 | # check if file is there |
| 1429 | result = runCmd("wic ls %s:2/ -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1430 | newdirs = set(line.split()[-1] for line in result.output.split('\n') if line) |
| 1431 | self.assertEqual(newdirs.difference(dirs), set([os.path.basename(testfile.name)])) |
| 1432 | |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 1433 | # check if the file to copy is in the partition |
| 1434 | result = runCmd("wic ls %s:2/etc/ -n %s" % (images[0], sysroot)) |
| 1435 | self.assertTrue('fstab' in [line.split()[-1] for line in result.output.split('\n') if line]) |
| 1436 | |
| 1437 | # copy file from the partition, replace the temporary file content with it and |
| 1438 | # check for the file size to validate the copy |
| 1439 | runCmd("wic cp %s:2/etc/fstab %s -n %s" % (images[0], testfile.name, sysroot)) |
| 1440 | self.assertTrue(os.stat(testfile.name).st_size > 0) |
| 1441 | |
| 1442 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1443 | def test_wic_rm_ext(self): |
| 1444 | """Test removing files from the ext partition.""" |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1445 | runCmd("wic create mkefidisk " |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1446 | "--image-name=core-image-minimal " |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1447 | "-D -o %s" % self.resultdir) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1448 | images = glob(self.resultdir + "mkefidisk-*.direct") |
| 1449 | self.assertEqual(1, len(images)) |
| 1450 | |
| 1451 | sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools') |
| 1452 | |
| 1453 | # list directory content of the /etc directory on ext4 partition |
| 1454 | result = runCmd("wic ls %s:2/etc/ -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1455 | self.assertTrue('fstab' in [line.split()[-1] for line in result.output.split('\n') if line]) |
| 1456 | |
| 1457 | # remove file |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 1458 | runCmd("wic rm %s:2/etc/fstab -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1459 | |
| 1460 | # check if it's removed |
| 1461 | result = runCmd("wic ls %s:2/etc/ -n %s" % (images[0], sysroot)) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 1462 | self.assertTrue('fstab' not in [line.split()[-1] for line in result.output.split('\n') if line]) |
Brad Bishop | 6dbb316 | 2019-11-25 09:41:34 -0500 | [diff] [blame] | 1463 | |
| 1464 | # remove non-empty directory |
| 1465 | runCmd("wic rm -r %s:2/etc/ -n %s" % (images[0], sysroot)) |
| 1466 | |
| 1467 | # check if it's removed |
| 1468 | result = runCmd("wic ls %s:2/ -n %s" % (images[0], sysroot)) |
| 1469 | self.assertTrue('etc' not in [line.split()[-1] for line in result.output.split('\n') if line]) |