Merge "Rewrite write_image.sh in Python"
This commit is contained in:
commit
2172122b87
@ -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.
|
||||||
|
@ -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!"
|
|
@ -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)
|
||||||
|
Loading…
Reference in New Issue
Block a user