Merge "Rewrite write_image.sh in Python"

This commit is contained in:
Zuul 2021-05-26 17:17:02 +00:00 committed by Gerrit Code Review
commit 2172122b87
3 changed files with 42 additions and 78 deletions

View File

@ -23,6 +23,7 @@ from ironic_lib import exception
from oslo_concurrency import processutils from oslo_concurrency import processutils
from oslo_config import cfg from oslo_config import cfg
from oslo_log import log from oslo_log import log
from oslo_utils import units
import requests import requests
from ironic_python_agent import errors from ironic_python_agent import errors
@ -46,16 +47,6 @@ def _image_location(image_info):
return os.path.join(tempfile.gettempdir(), image_info['id']) return os.path.join(tempfile.gettempdir(), image_info['id'])
def _path_to_script(script):
"""Get the location of a script which ships with ironic-python-agent.
:param script: The script name as a string.
:returns: The relative path to the script.
"""
cwd = os.path.dirname(os.path.realpath(__file__))
return os.path.join(cwd, '..', script)
def _download_with_proxy(image_info, url, image_id): def _download_with_proxy(image_info, url, image_id):
"""Opens a download stream for the given URL. """Opens a download stream for the given URL.
@ -203,14 +194,24 @@ def _write_whole_disk_image(image, image_info, device):
:raises: ImageWriteError if the command to write the image encounters an :raises: ImageWriteError if the command to write the image encounters an
error. error.
""" """
script = _path_to_script('shell/write_image.sh') # FIXME(dtantsur): pass the real node UUID for logging
command = ['/bin/bash', script, image, device] disk_utils.destroy_disk_metadata(device, '')
disk_utils.udev_settle()
command = ['qemu-img', 'convert',
'-t', 'directsync', '-O', 'host_device', '-W',
image, device]
LOG.info('Writing image with command: {}'.format(' '.join(command))) LOG.info('Writing image with command: {}'.format(' '.join(command)))
try: try:
stdout, stderr = utils.execute(*command) # TODO(dtantsur): switch to disk_utils.convert_image when it supports
# -t and -W flags and defaults to 2 GiB memory limit.
limits = processutils.ProcessLimits(address_space=2048 * units.Mi)
utils.execute(*command, prlimit=limits)
except processutils.ProcessExecutionError as e: except processutils.ProcessExecutionError as e:
raise errors.ImageWriteError(device, e.exit_code, e.stdout, e.stderr) raise errors.ImageWriteError(device, e.exit_code, e.stdout, e.stderr)
disk_utils.trigger_device_rescan(device)
def _write_image(image_info, device): def _write_image(image_info, device):
"""Writes an image to the specified device. """Writes an image to the specified device.

View File

@ -1,55 +0,0 @@
#!/bin/bash
# Copyright 2013 Rackspace, Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
set -e
log() {
echo "`basename $0`: $@"
}
usage() {
[[ -z "$1" ]] || echo -e "USAGE ERROR: $@\n"
echo "`basename $0`: IMAGEFILE DEVICE"
echo " - This script images DEVICE with IMAGEFILE"
exit 1
}
IMAGEFILE="$1"
DEVICE="$2"
[[ -f $IMAGEFILE ]] || usage "$IMAGEFILE (IMAGEFILE) is not a file"
[[ -b $DEVICE ]] || usage "$DEVICE (DEVICE) is not a block device"
# In production this will be replaced with secure erasing the drives
# For now we need to ensure there aren't any old (GPT) partitions on the drive
log "Erasing existing GPT and MBR data structures from ${DEVICE}"
# NOTE(gfidente): GPT uses 33*512 sectors, this is an attempt to avoid bug:
# https://bugs.launchpad.net/ironic-python-agent/+bug/1737556
DEVICE_SECTORS_COUNT=`blockdev --getsz $DEVICE`
dd bs=512 if=/dev/zero of=$DEVICE count=33
dd bs=512 if=/dev/zero of=$DEVICE count=33 seek=$((${DEVICE_SECTORS_COUNT} - 33))
sgdisk -Z $DEVICE
udevadm settle
log "Imaging $IMAGEFILE to $DEVICE"
# limit the memory usage for qemu-img to 2 GiB
ulimit -v 2097152
qemu-img convert -t directsync -O host_device -W $IMAGEFILE $DEVICE
sync
log "${DEVICE} imaged successfully!"

View File

@ -168,25 +168,45 @@ class TestStandbyExtension(base.IronicAgentTest):
self.assertEqual(expected_loc, location) self.assertEqual(expected_loc, location)
@mock.patch('ironic_lib.disk_utils.fix_gpt_partition', autospec=True) @mock.patch('ironic_lib.disk_utils.fix_gpt_partition', autospec=True)
@mock.patch('builtins.open', autospec=True) @mock.patch('ironic_lib.disk_utils.trigger_device_rescan', autospec=True)
@mock.patch('ironic_python_agent.utils.execute', autospec=True) @mock.patch('ironic_python_agent.utils.execute', autospec=True)
def test_write_image(self, execute_mock, open_mock, fix_gpt_mock): @mock.patch('ironic_lib.disk_utils.udev_settle', autospec=True)
@mock.patch('ironic_lib.disk_utils.destroy_disk_metadata', autospec=True)
def test_write_image(self, wipe_mock, udev_mock, execute_mock,
rescan_mock, fix_gpt_mock):
image_info = _build_fake_image_info() image_info = _build_fake_image_info()
device = '/dev/sda' device = '/dev/sda'
location = standby._image_location(image_info) location = standby._image_location(image_info)
script = standby._path_to_script('shell/write_image.sh') command = ['qemu-img', 'convert', '-t', 'directsync',
command = ['/bin/bash', script, location, device] '-O', 'host_device', '-W', location, device]
execute_mock.return_value = ('', '')
standby._write_image(image_info, device) standby._write_image(image_info, device)
execute_mock.assert_called_once_with(*command)
execute_mock.assert_called_once_with(*command, prlimit=mock.ANY)
wipe_mock.assert_called_once_with(device, '')
udev_mock.assert_called_once_with()
rescan_mock.assert_called_once_with(device)
fix_gpt_mock.assert_called_once_with(device, node_uuid=None) fix_gpt_mock.assert_called_once_with(device, node_uuid=None)
@mock.patch('ironic_lib.disk_utils.fix_gpt_partition', autospec=True)
@mock.patch('ironic_lib.disk_utils.trigger_device_rescan', autospec=True)
@mock.patch('ironic_python_agent.utils.execute', autospec=True)
@mock.patch('ironic_lib.disk_utils.udev_settle', autospec=True)
@mock.patch('ironic_lib.disk_utils.destroy_disk_metadata', autospec=True)
def test_write_image_gpt_fails(self, wipe_mock, udev_mock, execute_mock,
rescan_mock, fix_gpt_mock):
image_info = _build_fake_image_info()
device = '/dev/sda'
fix_gpt_mock.side_effect = exception.InstanceDeployFailure fix_gpt_mock.side_effect = exception.InstanceDeployFailure
standby._write_image(image_info, device) standby._write_image(image_info, device)
execute_mock.reset_mock() @mock.patch('ironic_python_agent.utils.execute', autospec=True)
execute_mock.return_value = ('', '') @mock.patch('ironic_lib.disk_utils.udev_settle', autospec=True)
@mock.patch('ironic_lib.disk_utils.destroy_disk_metadata', autospec=True)
def test_write_image_fails(self, wipe_mock, udev_mock, execute_mock):
image_info = _build_fake_image_info()
device = '/dev/sda'
execute_mock.side_effect = processutils.ProcessExecutionError execute_mock.side_effect = processutils.ProcessExecutionError
self.assertRaises(errors.ImageWriteError, self.assertRaises(errors.ImageWriteError,
@ -194,8 +214,6 @@ class TestStandbyExtension(base.IronicAgentTest):
image_info, image_info,
device) device)
execute_mock.assert_called_once_with(*command)
@mock.patch.object(utils, 'get_node_boot_mode', lambda self: 'bios') @mock.patch.object(utils, 'get_node_boot_mode', lambda self: 'bios')
@mock.patch.object(hardware, 'dispatch_to_managers', autospec=True) @mock.patch.object(hardware, 'dispatch_to_managers', autospec=True)
@mock.patch('builtins.open', autospec=True) @mock.patch('builtins.open', autospec=True)