mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
[PR #5803/f38bfadd backport][stable-6] Bugfix: proxmox_disk - read time out on import (#5881)
Bugfix: proxmox_disk - read time out on import (#5803)
* Use async calls and fix docs
* Add changelog fragment
(cherry picked from commit f38bfaddf0
)
Co-authored-by: castorsky <csky57@gmail.com>
This commit is contained in:
parent
ee07d8320a
commit
54099d77ff
2 changed files with 41 additions and 22 deletions
2
changelogs/fragments/5803-proxmox-read-timeout.yml
Normal file
2
changelogs/fragments/5803-proxmox-read-timeout.yml
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
bugfixes:
|
||||||
|
- proxmox_disk - fixed issue with read timeout on import action (https://github.com/ansible-collections/community.general/pull/5803).
|
|
@ -104,6 +104,7 @@ options:
|
||||||
- Move the disk to this storage when I(state=moved).
|
- Move the disk to this storage when I(state=moved).
|
||||||
- You can move between storages only in scope of one VM.
|
- You can move between storages only in scope of one VM.
|
||||||
- Mutually exclusive with I(target_vmid).
|
- Mutually exclusive with I(target_vmid).
|
||||||
|
- Consider increasing I(timeout) in case of large disk images or slow storage backend.
|
||||||
type: str
|
type: str
|
||||||
target_vmid:
|
target_vmid:
|
||||||
description:
|
description:
|
||||||
|
@ -113,8 +114,8 @@ options:
|
||||||
type: int
|
type: int
|
||||||
timeout:
|
timeout:
|
||||||
description:
|
description:
|
||||||
- Timeout in seconds to wait when moving disk.
|
- Timeout in seconds to wait for slow operations such as importing disk or moving disk between storages.
|
||||||
- Used only when I(state=moved).
|
- Used only when I(state) is C(present) or C(moved).
|
||||||
type: int
|
type: int
|
||||||
default: 600
|
default: 600
|
||||||
aio:
|
aio:
|
||||||
|
@ -172,6 +173,7 @@ options:
|
||||||
- C(<STORAGE>:<VMID>/<FULL_NAME>) or C(<ABSOLUTE_PATH>/<FULL_NAME>)
|
- C(<STORAGE>:<VMID>/<FULL_NAME>) or C(<ABSOLUTE_PATH>/<FULL_NAME>)
|
||||||
- Attention! Only root can use absolute paths.
|
- Attention! Only root can use absolute paths.
|
||||||
- This parameter is mutually exclusive with I(size).
|
- This parameter is mutually exclusive with I(size).
|
||||||
|
- Increase I(timeout) parameter when importing large disk images or using slow storage.
|
||||||
type: str
|
type: str
|
||||||
iops:
|
iops:
|
||||||
description:
|
description:
|
||||||
|
@ -471,6 +473,16 @@ class ProxmoxDiskAnsible(ProxmoxAnsible):
|
||||||
params.update(dict((k, int(v)) for k, v in params.items() if isinstance(v, bool)))
|
params.update(dict((k, int(v)) for k, v in params.items() if isinstance(v, bool)))
|
||||||
return params
|
return params
|
||||||
|
|
||||||
|
def wait_till_complete_or_timeout(self, node_name, task_id):
|
||||||
|
timeout = self.module.params['timeout']
|
||||||
|
while timeout:
|
||||||
|
if self.api_task_ok(node_name, task_id):
|
||||||
|
return True
|
||||||
|
timeout -= 1
|
||||||
|
if timeout <= 0:
|
||||||
|
return False
|
||||||
|
sleep(1)
|
||||||
|
|
||||||
def create_disk(self, disk, vmid, vm, vm_config):
|
def create_disk(self, disk, vmid, vm, vm_config):
|
||||||
create = self.module.params['create']
|
create = self.module.params['create']
|
||||||
if create == 'disabled' and disk not in vm_config:
|
if create == 'disabled' and disk not in vm_config:
|
||||||
|
@ -484,20 +496,23 @@ class ProxmoxDiskAnsible(ProxmoxAnsible):
|
||||||
|
|
||||||
if import_string:
|
if import_string:
|
||||||
config_str = "%s:%s,import-from=%s" % (self.module.params["storage"], "0", import_string)
|
config_str = "%s:%s,import-from=%s" % (self.module.params["storage"], "0", import_string)
|
||||||
|
timeout_str = "Reached timeout while importing VM disk. Last line in task before timeout: %s"
|
||||||
|
ok_str = "Disk %s imported into VM %s"
|
||||||
else:
|
else:
|
||||||
config_str = "%s:%s" % (self.module.params["storage"], self.module.params["size"])
|
config_str = "%s:%s" % (self.module.params["storage"], self.module.params["size"])
|
||||||
|
ok_str = "Disk %s created in VM %s"
|
||||||
|
timeout_str = "Reached timeout while creating VM disk. Last line in task before timeout: %s"
|
||||||
|
|
||||||
for k, v in attributes.items():
|
for k, v in attributes.items():
|
||||||
config_str += ',%s=%s' % (k, v)
|
config_str += ',%s=%s' % (k, v)
|
||||||
|
|
||||||
create_disk = {self.module.params["disk"]: config_str}
|
disk_config_to_apply = {self.module.params["disk"]: config_str}
|
||||||
self.proxmox_api.nodes(vm['node']).qemu(vmid).config.set(**create_disk)
|
|
||||||
return True, "Disk %s created in VM %s" % (disk, vmid)
|
|
||||||
|
|
||||||
if create in ['disabled', 'regular'] and disk in vm_config:
|
if create in ['disabled', 'regular'] and disk in vm_config:
|
||||||
# UPDATE
|
# UPDATE
|
||||||
disk_config = disk_conf_str_to_dict(vm_config[disk])
|
disk_config = disk_conf_str_to_dict(vm_config[disk])
|
||||||
config_str = disk_config["volume"]
|
config_str = disk_config["volume"]
|
||||||
|
ok_str = "Disk %s updated in VM %s"
|
||||||
attributes = self.get_create_attributes()
|
attributes = self.get_create_attributes()
|
||||||
# 'import_from' fails on disk updates
|
# 'import_from' fails on disk updates
|
||||||
attributes.pop('import_from', None)
|
attributes.pop('import_from', None)
|
||||||
|
@ -513,9 +528,16 @@ class ProxmoxDiskAnsible(ProxmoxAnsible):
|
||||||
if disk_config == attributes:
|
if disk_config == attributes:
|
||||||
return False, "Disk %s is up to date in VM %s" % (disk, vmid)
|
return False, "Disk %s is up to date in VM %s" % (disk, vmid)
|
||||||
|
|
||||||
update_disk = {self.module.params["disk"]: config_str}
|
disk_config_to_apply = {self.module.params["disk"]: config_str}
|
||||||
self.proxmox_api.nodes(vm['node']).qemu(vmid).config.set(**update_disk)
|
|
||||||
return True, "Disk %s updated in VM %s" % (disk, vmid)
|
current_task_id = self.proxmox_api.nodes(vm['node']).qemu(vmid).config.post(**disk_config_to_apply)
|
||||||
|
task_success = self.wait_till_complete_or_timeout(vm['node'], current_task_id)
|
||||||
|
if task_success:
|
||||||
|
return True, ok_str % (disk, vmid)
|
||||||
|
else:
|
||||||
|
self.module.fail_json(
|
||||||
|
msg=timeout_str % self.proxmox_api.nodes(vm['node']).tasks(current_task_id).log.get()[:1]
|
||||||
|
)
|
||||||
|
|
||||||
def move_disk(self, disk, vmid, vm, vm_config):
|
def move_disk(self, disk, vmid, vm, vm_config):
|
||||||
params = dict()
|
params = dict()
|
||||||
|
@ -535,20 +557,15 @@ class ProxmoxDiskAnsible(ProxmoxAnsible):
|
||||||
if params['storage'] == disk_config['storage_name']:
|
if params['storage'] == disk_config['storage_name']:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
taskid = self.proxmox_api.nodes(vm['node']).qemu(vmid).move_disk.post(**params)
|
task_id = self.proxmox_api.nodes(vm['node']).qemu(vmid).move_disk.post(**params)
|
||||||
timeout = self.module.params['timeout']
|
task_success = self.wait_till_complete_or_timeout(vm['node'], task_id)
|
||||||
while timeout:
|
if task_success:
|
||||||
status_data = self.proxmox_api.nodes(vm['node']).tasks(taskid).status.get()
|
|
||||||
if status_data['status'] == 'stopped' and status_data['exitstatus'] == 'OK':
|
|
||||||
return True
|
return True
|
||||||
if timeout <= 0:
|
else:
|
||||||
self.module.fail_json(
|
self.module.fail_json(
|
||||||
msg='Reached timeout while waiting for moving VM disk. Last line in task before timeout: %s' %
|
msg='Reached timeout while waiting for moving VM disk. Last line in task before timeout: %s' %
|
||||||
self.proxmox_api.nodes(vm['node']).tasks(taskid).log.get()[:1])
|
self.proxmox_api.nodes(vm['node']).tasks(task_id).log.get()[:1]
|
||||||
|
)
|
||||||
sleep(1)
|
|
||||||
timeout -= 1
|
|
||||||
return True
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
|
|
Loading…
Reference in a new issue