Don't run os-prober from grub2-mkconfig

By default, grub2-mkconfig scans everything to look for other
environments and then load those into the grub configuration.

It makes sense, but on newer versions of grub2 in distribution
images, os-prober is taking an exceptionally long time in some
cases where more than one storage device exists with other
filesystems.

As a result, of the os-prober execution by grub2-mkconfig, the
bootloader installation can completely time out and fail the
deployment. This is presently experienced with metalsmith on
centos8.

There are numerous sporatic reports of issues like this issue
where grub2-mkconfig hangs for some period of time, and this is
observable on Centos8.2 in our CI. While one report[0] mentions
this issue, Another bug [1] has the dialog that actually helps us
frame the context as to what we likely should do.

Also, fixes the unit testing so we actually test if we're running
with grub2. :\

[0]: https://bugzilla.redhat.com/show_bug.cgi?id=1744693
[1]: https://bugzilla.redhat.com/show_bug.cgi?id=1709682

Depends-On: https://review.opendev.org/#/c/748315
Change-Id: I14bf299afef3a1ddb2006fe5f182d7f0d249e734
This commit is contained in:
Julia Kreger 2020-09-16 13:10:13 -07:00
parent 81ba64a5ed
commit 6542a9cb04
5 changed files with 49 additions and 11 deletions

View File

@ -658,13 +658,13 @@ def _install_grub2(device, root_uuid, efi_system_part_uuid=None,
r'GRUB_CMDLINE_LINUX="\1 %s"' r'GRUB_CMDLINE_LINUX="\1 %s"'
% " ".join(rd_md_uuids), % " ".join(rd_md_uuids),
contents)) contents))
# Generate the grub configuration file
utils.execute('chroot %(path)s /bin/sh -c ' utils.execute('chroot %(path)s /bin/sh -c '
'"%(bin)s-mkconfig -o ' '"%(bin)s-mkconfig -o '
'/boot/%(bin)s/grub.cfg"' % '/boot/%(bin)s/grub.cfg"' %
{'path': path, 'bin': binary_name}, shell=True, {'path': path, 'bin': binary_name}, shell=True,
env_variables={'PATH': path_variable}) env_variables={'PATH': path_variable,
'GRUB_DISABLE_OS_PROBER': 'true'},
use_standard_locale=True)
LOG.info("GRUB2 successfully installed on %s", device) LOG.info("GRUB2 successfully installed on %s", device)

View File

@ -381,6 +381,7 @@ efibootmgr: ** Warning ** : Boot0005 has same label ironic1\n
expected = [mock.call('efibootmgr', '--version')] expected = [mock.call('efibootmgr', '--version')]
mock_execute.assert_has_calls(expected) mock_execute.assert_has_calls(expected)
@mock.patch.object(os.path, 'exists', lambda *_: True)
@mock.patch.object(image, '_is_bootloader_loaded', lambda *_: False) @mock.patch.object(image, '_is_bootloader_loaded', lambda *_: False)
@mock.patch.object(hardware, 'is_md_device', autospec=True) @mock.patch.object(hardware, 'is_md_device', autospec=True)
@mock.patch.object(hardware, 'md_get_raid_devices', autospec=True) @mock.patch.object(hardware, 'md_get_raid_devices', autospec=True)
@ -409,16 +410,18 @@ efibootmgr: ** Warning ** : Boot0005 has same label ironic1\n
env_variables={ env_variables={
'PATH': '/sbin:/bin:/usr/sbin:/sbin'}), 'PATH': '/sbin:/bin:/usr/sbin:/sbin'}),
mock.call(('chroot %s /bin/sh -c ' mock.call(('chroot %s /bin/sh -c '
'"grub-install %s"' % '"grub2-install %s"' %
(self.fake_dir, self.fake_dev)), shell=True, (self.fake_dir, self.fake_dev)), shell=True,
env_variables={ env_variables={
'PATH': '/sbin:/bin:/usr/sbin:/sbin'}), 'PATH': '/sbin:/bin:/usr/sbin:/sbin'}),
mock.call(('chroot %s /bin/sh -c ' mock.call(('chroot %s /bin/sh -c '
'"grub-mkconfig -o ' '"grub2-mkconfig -o '
'/boot/grub/grub.cfg"' % self.fake_dir), '/boot/grub2/grub.cfg"' % self.fake_dir),
shell=True, shell=True,
env_variables={ env_variables={
'PATH': '/sbin:/bin:/usr/sbin:/sbin'}), 'PATH': '/sbin:/bin:/usr/sbin:/sbin',
'GRUB_DISABLE_OS_PROBER': 'true'},
use_standard_locale=True),
mock.call(('chroot %s /bin/sh -c "umount -a -t vfat"' % mock.call(('chroot %s /bin/sh -c "umount -a -t vfat"' %
(self.fake_dir)), shell=True, (self.fake_dir)), shell=True,
env_variables={ env_variables={
@ -478,7 +481,9 @@ efibootmgr: ** Warning ** : Boot0005 has same label ironic1\n
'/boot/grub/grub.cfg"' % self.fake_dir), '/boot/grub/grub.cfg"' % self.fake_dir),
shell=True, shell=True,
env_variables={ env_variables={
'PATH': '/sbin:/bin:/usr/sbin:/sbin'}), 'PATH': '/sbin:/bin:/usr/sbin:/sbin',
'GRUB_DISABLE_OS_PROBER': 'true'},
use_standard_locale=True),
mock.call(('chroot %s /bin/sh -c "umount -a -t vfat"' % mock.call(('chroot %s /bin/sh -c "umount -a -t vfat"' %
(self.fake_dir)), shell=True, (self.fake_dir)), shell=True,
env_variables={ env_variables={
@ -556,7 +561,9 @@ efibootmgr: ** Warning ** : Boot0005 has same label ironic1\n
'/boot/grub/grub.cfg"' % self.fake_dir), '/boot/grub/grub.cfg"' % self.fake_dir),
shell=True, shell=True,
env_variables={ env_variables={
'PATH': '/sbin:/bin:/usr/sbin:/sbin'}), 'PATH': '/sbin:/bin:/usr/sbin:/sbin',
'GRUB_DISABLE_OS_PROBER': 'true'},
use_standard_locale=True),
mock.call('umount', self.fake_dir + '/boot/efi', mock.call('umount', self.fake_dir + '/boot/efi',
attempts=3, delay_on_retry=True), attempts=3, delay_on_retry=True),
mock.call(('chroot %s /bin/sh -c "umount -a -t vfat"' % mock.call(('chroot %s /bin/sh -c "umount -a -t vfat"' %
@ -963,7 +970,9 @@ efibootmgr: ** Warning ** : Boot0005 has same label ironic1\n
'/boot/grub/grub.cfg"' % self.fake_dir), '/boot/grub/grub.cfg"' % self.fake_dir),
shell=True, shell=True,
env_variables={ env_variables={
'PATH': '/sbin:/bin:/usr/sbin:/sbin'}), 'PATH': '/sbin:/bin:/usr/sbin:/sbin',
'GRUB_DISABLE_OS_PROBER': 'true'},
use_standard_locale=True),
mock.call('umount', self.fake_dir + '/boot/efi', mock.call('umount', self.fake_dir + '/boot/efi',
attempts=3, delay_on_retry=True), attempts=3, delay_on_retry=True),
mock.call(('chroot %s /bin/sh -c "umount -a -t vfat"' % mock.call(('chroot %s /bin/sh -c "umount -a -t vfat"' %
@ -1055,7 +1064,9 @@ efibootmgr: ** Warning ** : Boot0005 has same label ironic1\n
'/boot/grub/grub.cfg"' % self.fake_dir), '/boot/grub/grub.cfg"' % self.fake_dir),
shell=True, shell=True,
env_variables={ env_variables={
'PATH': '/sbin:/bin:/usr/sbin:/sbin'}), 'PATH': '/sbin:/bin:/usr/sbin:/sbin',
'GRUB_DISABLE_OS_PROBER': 'true'},
use_standard_locale=True),
mock.call(('chroot %s /bin/sh -c "umount -a -t vfat"' % mock.call(('chroot %s /bin/sh -c "umount -a -t vfat"' %
(self.fake_dir)), shell=True, (self.fake_dir)), shell=True,
env_variables={ env_variables={

View File

@ -0,0 +1,11 @@
---
fixes:
- |
Fixes an issue where deployments of Fedora or Centos can hang when
using grub2 with the execution of the ``grub2-mkconfig`` command
not returning before the deployment process times out. This is
because ``grub2-mkconfig`` triggers ``os-prober`` which can
take an extended period of time to evaluate additional unrelated
devices for dual-boot scenarios. Since operators are not dual
booting their machines enrolled in ironic, it seems like an
un-necessary scan and has thus been disabled.

View File

@ -154,6 +154,20 @@
IRONIC_AUTOMATED_CLEAN_ENABLED: False IRONIC_AUTOMATED_CLEAN_ENABLED: False
IRONIC_BUILD_DEPLOY_RAMDISK: True IRONIC_BUILD_DEPLOY_RAMDISK: True
# This job will superceed the above centos7 metalsmith integration job
- job:
name: metalsmith-integration-ipa-src-uefi
parent: metalsmith-integration-glance-localboot-centos8-uefi
required-projects:
- openstack/ironic-python-agent
- openstack/ironic-python-agent-builder
- openstack/ironic-lib
vars:
devstack_localrc:
# Don't waste time on cleaning, it's checked everywhere else
IRONIC_AUTOMATED_CLEAN_ENABLED: False
IRONIC_BUILD_DEPLOY_RAMDISK: True
# used by ironic-python-agent-builder # used by ironic-python-agent-builder
# we can use focal here but we need to disable dstat because of # we can use focal here but we need to disable dstat because of
# https://bugs.launchpad.net/ubuntu/+source/dstat/+bug/1866619 # https://bugs.launchpad.net/ubuntu/+source/dstat/+bug/1866619

View File

@ -16,6 +16,8 @@
- ipa-tempest-bios-ipmi-iscsi-src - ipa-tempest-bios-ipmi-iscsi-src
- ironic-standalone-ipa-src - ironic-standalone-ipa-src
- metalsmith-integration-ipa-src - metalsmith-integration-ipa-src
- metalsmith-integration-ipa-src-uefi:
voting: false
# NOTE(dtantsur): non-voting because IPA source code is very unlikely # NOTE(dtantsur): non-voting because IPA source code is very unlikely
# to break them. They rather serve as a canary for broken POST jobs. # to break them. They rather serve as a canary for broken POST jobs.
- ironic-python-agent-check-image-tinyipa: - ironic-python-agent-check-image-tinyipa: