Ansible module: fix deployment for private and/or shared images
The ansible module deployment was broken for private/shared images because the original request context was not available anymore at the time the image informations were fetched Change-Id: Id35dcf8ddc209bb8318433d6bf26aba848e074ee Story: #2006178 Task: #35702
This commit is contained in:
parent
bc8959765b
commit
d0c31d748d
@ -277,7 +277,6 @@ def _prepare_variables(task):
|
|||||||
for i_key, i_value in i_info.items():
|
for i_key, i_value in i_info.items():
|
||||||
if i_key.startswith('image_'):
|
if i_key.startswith('image_'):
|
||||||
image[i_key[6:]] = i_value
|
image[i_key[6:]] = i_value
|
||||||
image['mem_req'] = _calculate_memory_req(task)
|
|
||||||
|
|
||||||
checksum = image.get('checksum')
|
checksum = image.get('checksum')
|
||||||
if checksum:
|
if checksum:
|
||||||
@ -443,9 +442,24 @@ class AnsibleDeploy(agent_base.HeartbeatMixin, base.DeployInterface):
|
|||||||
@task_manager.require_exclusive_lock
|
@task_manager.require_exclusive_lock
|
||||||
def deploy(self, task):
|
def deploy(self, task):
|
||||||
"""Perform a deployment to a node."""
|
"""Perform a deployment to a node."""
|
||||||
|
self._required_image_info(task)
|
||||||
manager_utils.node_power_action(task, states.REBOOT)
|
manager_utils.node_power_action(task, states.REBOOT)
|
||||||
return states.DEPLOYWAIT
|
return states.DEPLOYWAIT
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _required_image_info(task):
|
||||||
|
"""Gather and save needed image info while the context is good.
|
||||||
|
|
||||||
|
Gather image info that will be needed later, during the
|
||||||
|
continue_deploy execution, where the context won't be the same
|
||||||
|
anymore, since coming from the server's heartbeat.
|
||||||
|
"""
|
||||||
|
node = task.node
|
||||||
|
i_info = node.instance_info
|
||||||
|
i_info['image_mem_req'] = _calculate_memory_req(task)
|
||||||
|
node.instance_info = i_info
|
||||||
|
node.save()
|
||||||
|
|
||||||
@METRICS.timer('AnsibleDeploy.tear_down')
|
@METRICS.timer('AnsibleDeploy.tear_down')
|
||||||
@task_manager.require_exclusive_lock
|
@task_manager.require_exclusive_lock
|
||||||
def tear_down(self, task):
|
def tear_down(self, task):
|
||||||
|
@ -375,22 +375,25 @@ class TestAnsibleMethods(AnsibleDeployTestCaseBase):
|
|||||||
self.assertIn(six.text_type(key), six.text_type(exc))
|
self.assertIn(six.text_type(key), six.text_type(exc))
|
||||||
self.assertIn(six.text_type(value), six.text_type(exc))
|
self.assertIn(six.text_type(value), six.text_type(exc))
|
||||||
|
|
||||||
@mock.patch.object(ansible_deploy, '_calculate_memory_req', autospec=True,
|
def test__prepare_variables(self):
|
||||||
return_value=2000)
|
i_info = self.node.instance_info
|
||||||
def test__prepare_variables(self, mem_req_mock):
|
i_info['image_mem_req'] = 3000
|
||||||
|
i_info['image_whatever'] = 'hello'
|
||||||
|
self.node.instance_info = i_info
|
||||||
|
self.node.save()
|
||||||
|
|
||||||
expected = {"image": {"url": "http://image",
|
expected = {"image": {"url": "http://image",
|
||||||
"validate_certs": "yes",
|
"validate_certs": "yes",
|
||||||
"source": "fake-image",
|
"source": "fake-image",
|
||||||
"mem_req": 2000,
|
"mem_req": 3000,
|
||||||
"disk_format": "qcow2",
|
"disk_format": "qcow2",
|
||||||
"checksum": "md5:checksum"}}
|
"checksum": "md5:checksum",
|
||||||
|
"whatever": "hello"}}
|
||||||
with task_manager.acquire(self.context, self.node.uuid) as task:
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
||||||
self.assertEqual(expected,
|
self.assertEqual(expected,
|
||||||
ansible_deploy._prepare_variables(task))
|
ansible_deploy._prepare_variables(task))
|
||||||
|
|
||||||
@mock.patch.object(ansible_deploy, '_calculate_memory_req', autospec=True,
|
def test__prepare_variables_root_device_hints(self):
|
||||||
return_value=2000)
|
|
||||||
def test__prepare_variables_root_device_hints(self, mem_req_mock):
|
|
||||||
props = self.node.properties
|
props = self.node.properties
|
||||||
props['root_device'] = {"wwn": "fake-wwn"}
|
props['root_device'] = {"wwn": "fake-wwn"}
|
||||||
self.node.properties = props
|
self.node.properties = props
|
||||||
@ -398,7 +401,6 @@ class TestAnsibleMethods(AnsibleDeployTestCaseBase):
|
|||||||
expected = {"image": {"url": "http://image",
|
expected = {"image": {"url": "http://image",
|
||||||
"validate_certs": "yes",
|
"validate_certs": "yes",
|
||||||
"source": "fake-image",
|
"source": "fake-image",
|
||||||
"mem_req": 2000,
|
|
||||||
"disk_format": "qcow2",
|
"disk_format": "qcow2",
|
||||||
"checksum": "md5:checksum"},
|
"checksum": "md5:checksum"},
|
||||||
"root_device_hints": {"wwn": "fake-wwn"}}
|
"root_device_hints": {"wwn": "fake-wwn"}}
|
||||||
@ -406,9 +408,7 @@ class TestAnsibleMethods(AnsibleDeployTestCaseBase):
|
|||||||
self.assertEqual(expected,
|
self.assertEqual(expected,
|
||||||
ansible_deploy._prepare_variables(task))
|
ansible_deploy._prepare_variables(task))
|
||||||
|
|
||||||
@mock.patch.object(ansible_deploy, '_calculate_memory_req', autospec=True,
|
def test__prepare_variables_insecure_activated(self):
|
||||||
return_value=2000)
|
|
||||||
def test__prepare_variables_noglance(self, mem_req_mock):
|
|
||||||
self.config(image_store_insecure=True, group='ansible')
|
self.config(image_store_insecure=True, group='ansible')
|
||||||
i_info = self.node.instance_info
|
i_info = self.node.instance_info
|
||||||
i_info['image_checksum'] = 'sha256:checksum'
|
i_info['image_checksum'] = 'sha256:checksum'
|
||||||
@ -417,16 +417,13 @@ class TestAnsibleMethods(AnsibleDeployTestCaseBase):
|
|||||||
expected = {"image": {"url": "http://image",
|
expected = {"image": {"url": "http://image",
|
||||||
"validate_certs": "no",
|
"validate_certs": "no",
|
||||||
"source": "fake-image",
|
"source": "fake-image",
|
||||||
"mem_req": 2000,
|
|
||||||
"disk_format": "qcow2",
|
"disk_format": "qcow2",
|
||||||
"checksum": "sha256:checksum"}}
|
"checksum": "sha256:checksum"}}
|
||||||
with task_manager.acquire(self.context, self.node.uuid) as task:
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
||||||
self.assertEqual(expected,
|
self.assertEqual(expected,
|
||||||
ansible_deploy._prepare_variables(task))
|
ansible_deploy._prepare_variables(task))
|
||||||
|
|
||||||
@mock.patch.object(ansible_deploy, '_calculate_memory_req', autospec=True,
|
def test__prepare_variables_configdrive_url(self):
|
||||||
return_value=2000)
|
|
||||||
def test__prepare_variables_configdrive_url(self, mem_req_mock):
|
|
||||||
i_info = self.node.instance_info
|
i_info = self.node.instance_info
|
||||||
i_info['configdrive'] = 'http://configdrive_url'
|
i_info['configdrive'] = 'http://configdrive_url'
|
||||||
self.node.instance_info = i_info
|
self.node.instance_info = i_info
|
||||||
@ -434,7 +431,6 @@ class TestAnsibleMethods(AnsibleDeployTestCaseBase):
|
|||||||
expected = {"image": {"url": "http://image",
|
expected = {"image": {"url": "http://image",
|
||||||
"validate_certs": "yes",
|
"validate_certs": "yes",
|
||||||
"source": "fake-image",
|
"source": "fake-image",
|
||||||
"mem_req": 2000,
|
|
||||||
"disk_format": "qcow2",
|
"disk_format": "qcow2",
|
||||||
"checksum": "md5:checksum"},
|
"checksum": "md5:checksum"},
|
||||||
'configdrive': {'type': 'url',
|
'configdrive': {'type': 'url',
|
||||||
@ -443,9 +439,7 @@ class TestAnsibleMethods(AnsibleDeployTestCaseBase):
|
|||||||
self.assertEqual(expected,
|
self.assertEqual(expected,
|
||||||
ansible_deploy._prepare_variables(task))
|
ansible_deploy._prepare_variables(task))
|
||||||
|
|
||||||
@mock.patch.object(ansible_deploy, '_calculate_memory_req', autospec=True,
|
def test__prepare_variables_configdrive_file(self):
|
||||||
return_value=2000)
|
|
||||||
def test__prepare_variables_configdrive_file(self, mem_req_mock):
|
|
||||||
i_info = self.node.instance_info
|
i_info = self.node.instance_info
|
||||||
i_info['configdrive'] = 'fake-content'
|
i_info['configdrive'] = 'fake-content'
|
||||||
self.node.instance_info = i_info
|
self.node.instance_info = i_info
|
||||||
@ -456,7 +450,6 @@ class TestAnsibleMethods(AnsibleDeployTestCaseBase):
|
|||||||
expected = {"image": {"url": "http://image",
|
expected = {"image": {"url": "http://image",
|
||||||
"validate_certs": "yes",
|
"validate_certs": "yes",
|
||||||
"source": "fake-image",
|
"source": "fake-image",
|
||||||
"mem_req": 2000,
|
|
||||||
"disk_format": "qcow2",
|
"disk_format": "qcow2",
|
||||||
"checksum": "md5:checksum"},
|
"checksum": "md5:checksum"},
|
||||||
'configdrive': {'type': 'file',
|
'configdrive': {'type': 'file',
|
||||||
@ -587,13 +580,18 @@ class TestAnsibleDeploy(AnsibleDeployTestCaseBase):
|
|||||||
task.driver.boot, task)
|
task.driver.boot, task)
|
||||||
get_boot_mock.assert_called_once_with(task.node)
|
get_boot_mock.assert_called_once_with(task.node)
|
||||||
|
|
||||||
|
@mock.patch.object(ansible_deploy, '_calculate_memory_req', autospec=True,
|
||||||
|
return_value=2000)
|
||||||
@mock.patch.object(utils, 'node_power_action', autospec=True)
|
@mock.patch.object(utils, 'node_power_action', autospec=True)
|
||||||
def test_deploy(self, power_mock):
|
def test_deploy(self, power_mock, mem_req_mock):
|
||||||
with task_manager.acquire(
|
with task_manager.acquire(
|
||||||
self.context, self.node['uuid'], shared=False) as task:
|
self.context, self.node['uuid'], shared=False) as task:
|
||||||
driver_return = self.driver.deploy(task)
|
driver_return = self.driver.deploy(task)
|
||||||
self.assertEqual(driver_return, states.DEPLOYWAIT)
|
self.assertEqual(driver_return, states.DEPLOYWAIT)
|
||||||
power_mock.assert_called_once_with(task, states.REBOOT)
|
power_mock.assert_called_once_with(task, states.REBOOT)
|
||||||
|
mem_req_mock.assert_called_once_with(task)
|
||||||
|
i_info = task.node.instance_info
|
||||||
|
self.assertEqual(i_info['image_mem_req'], 2000)
|
||||||
|
|
||||||
@mock.patch.object(utils, 'node_power_action', autospec=True)
|
@mock.patch.object(utils, 'node_power_action', autospec=True)
|
||||||
def test_tear_down(self, power_mock):
|
def test_tear_down(self, power_mock):
|
||||||
|
7
releasenotes/notes/bug-35702-25da234580ca0c31.yaml
Normal file
7
releasenotes/notes/bug-35702-25da234580ca0c31.yaml
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
---
|
||||||
|
fixes:
|
||||||
|
- |
|
||||||
|
Fixes an issue regarding the ``ansible`` deploy interface. Node
|
||||||
|
deployment was broken for any image that was not public because
|
||||||
|
the original request context was not available anymore at the time
|
||||||
|
some image information was fetched.
|
Loading…
x
Reference in New Issue
Block a user