nxos_file_copy enhancement (#42424)

* add file_pull option

* typo correction

* fix shippable error

* review comments

* fix for review comments by mwiebe

* possible shippable fix

* remove file_copy from ignore

* more review comments addressed

* review comments

* add localhost for remote scp server in tests

* timeout fix
This commit is contained in:
saichint 2018-07-25 05:16:55 -07:00 committed by Trishna Guha
parent a488b3a8ed
commit fa5f396a4b
3 changed files with 218 additions and 46 deletions

View file

@ -25,52 +25,112 @@ DOCUMENTATION = '''
module: nxos_file_copy module: nxos_file_copy
extends_documentation_fragment: nxos extends_documentation_fragment: nxos
version_added: "2.2" version_added: "2.2"
short_description: Copy a file to a remote NXOS device over SCP. short_description: Copy a file to a remote NXOS device.
description: description:
- Copy a file to the flash (or bootflash) remote network device - This module supports two different workflows for copying a file
on NXOS devices. This module only supports the use of connection to flash (or bootflash) on NXOS devices. Files can either be (1) pushed
C(network_cli) or C(Cli) transport with connection C(local). from the Ansible controller to the device or (2) pulled from a remote SCP
file server to the device. File copies are initiated from the NXOS
device to the remote SCP server. This module only supports the
use of connection C(network_cli) or C(Cli) transport with connection C(local).
author: author:
- Jason Edelman (@jedelman8) - Jason Edelman (@jedelman8)
- Gabriele Gerbino (@GGabriele) - Gabriele Gerbino (@GGabriele)
notes: notes:
- Tested against NXOSv 7.3.(0)D1(1) on VIRL - Tested against NXOS 7.0(3)I2(5), 7.0(3)I4(6), 7.0(3)I5(3),
- The feature must be enabled with feature scp-server. 7.0(3)I6(1), 7.0(3)I7(3), 6.0(2)A8(8), 7.0(3)F3(4), 7.3(0)D1(1),
- If the file is already present (md5 sums match), no transfer will 8.3(0)
take place. - When pushing files (file_pull is False) to the NXOS device,
feature scp-server must be enabled.
- When pulling files (file_pull is True) to the NXOS device,
feature scp-server is not required.
- When pulling files (file_pull is True) to the NXOS device,
no transfer will take place if the file is already present.
- Check mode will tell you if the file would be copied. - Check mode will tell you if the file would be copied.
requirements: requirements:
- paramiko - paramiko (required when file_pull is False)
- SCPClient (required when file_pull is False)
- pexpect (required when file_pull is True)
options: options:
local_file: local_file:
description: description:
- Path to local file. Local directory must exist. - When (file_pull is False) this is the path to the local file on the Ansible controller.
required: true The local directory must exist.
- When (file_pull is True) this is the file name used on the NXOS device.
remote_file: remote_file:
description: description:
- Remote file path of the copy. Remote directories must exist. - When (file_pull is False) this is the remote file path on the NXOS device.
If omitted, the name of the local file will be used. If omitted, the name of the local file will be used.
The remote directory must exist.
- When (file_pull is True) this is the full path to the file on the remote SCP
server to be copied to the NXOS device.
file_system: file_system:
description: description:
- The remote file system of the device. If omitted, - The remote file system of the device. If omitted,
devices that support a I(file_system) parameter will use devices that support a I(file_system) parameter will use
their default values. their default values.
default: "bootflash:"
connect_ssh_port: connect_ssh_port:
description: description:
- SSH port to connect to server during transfer of file - SSH port to connect to server during transfer of file
default: 22 default: 22
version_added: "2.5" version_added: "2.5"
file_pull:
description:
- When (False) file is copied from the Ansible controller to the NXOS device.
- When (True) file is copied from a remote SCP server to the NXOS device.
In this mode, the file copy is initiated from the NXOS device.
- If the file is already present on the device it will be overwritten and
therefore the operation is NOT idempotent.
type: bool
default: False
version_added: "2.7"
file_pull_timeout:
description:
- Use this parameter to set timeout in seconds, when transferring
large files or when the network is slow.
default: 300
version_added: "2.7"
remote_scp_server:
description:
- The remote scp server address which is used to pull the file.
This is required if file_pull is True.
version_added: "2.7"
remote_scp_server_user:
description:
- The remote scp server username which is used to pull the file.
This is required if file_pull is True.
version_added: "2.7"
remote_scp_server_password:
description:
- The remote scp server password which is used to pull the file.
This is required if file_pull is True.
version_added: "2.7"
''' '''
EXAMPLES = ''' EXAMPLES = '''
- nxos_file_copy: # File copy from ansible controller to nxos device
local_file: "./test_file.txt" - name: "copy from server to device"
remote_file: "test_file.txt" nxos_file_copy:
local_file: "./test_file.txt"
remote_file: "test_file.txt"
# Initiate file copy from the nxos device to transfer file from an SCP server back to the nxos device
- name: "initiate file copy from device"
nxos_file_copy:
nxos_file_copy:
file_pull: True
local_file: "xyz"
remote_file: "/mydir/abc"
remote_scp_server: "192.168.0.1"
remote_scp_server_user: "myUser"
remote_scp_server_password: "myPassword"
''' '''
RETURN = ''' RETURN = '''
transfer_status: transfer_status:
description: Whether a file was transferred. "No Transfer" or "Sent". description: Whether a file was transferred. "No Transfer" or "Sent".
If file_pull is successful, it is set to "Received".
returned: success returned: success
type: string type: string
sample: 'Sent' sample: 'Sent'
@ -89,10 +149,12 @@ remote_file:
import os import os
import re import re
import time import time
import traceback
from ansible.module_utils.network.nxos.nxos import run_commands from ansible.module_utils.network.nxos.nxos import run_commands
from ansible.module_utils.network.nxos.nxos import nxos_argument_spec, check_args from ansible.module_utils.network.nxos.nxos import nxos_argument_spec, check_args
from ansible.module_utils.basic import AnsibleModule from ansible.module_utils.basic import AnsibleModule
from ansible.module_utils._text import to_native, to_text
try: try:
import paramiko import paramiko
@ -106,6 +168,12 @@ try:
except ImportError: except ImportError:
HAS_SCP = False HAS_SCP = False
try:
import pexpect
HAS_PEXPECT = True
except ImportError:
HAS_PEXPECT = False
def remote_file_exists(module, dst, file_system='bootflash:'): def remote_file_exists(module, dst, file_system='bootflash:'):
command = 'dir {0}/{1}'.format(file_system, dst) command = 'dir {0}/{1}'.format(file_system, dst)
@ -146,7 +214,7 @@ def enough_space(module):
return True return True
def transfer_file(module, dest): def transfer_file_to_device(module, dest):
file_size = os.path.getsize(module.params['local_file']) file_size = os.path.getsize(module.params['local_file'])
if not enough_space(module): if not enough_space(module):
@ -185,30 +253,107 @@ def transfer_file(module, dest):
return True return True
def copy_file_from_remote(module, local, file_system='bootflash:'):
hostname = module.params['host']
username = module.params['username']
password = module.params['password']
port = module.params['connect_ssh_port']
try:
child = pexpect.spawn('ssh ' + username + '@' + hostname + ' -p' + str(port))
# response could be unknown host addition or Password
index = child.expect(['yes', '(?i)Password'])
if index == 0:
child.sendline('yes')
child.expect('(?i)Password')
child.sendline(password)
child.expect('#')
command = ('copy scp://' + module.params['remote_scp_server_user'] +
'@' + module.params['remote_scp_server'] + module.params['remote_file'] +
' ' + file_system + local + ' vrf management')
child.sendline(command)
# response could be remote host connection time out,
# there is already an existing file with the same name,
# unknown host addition or password
index = child.expect(['timed out', 'existing', 'yes', '(?i)password'], timeout=180)
if index == 0:
module.fail_json(msg='Timeout occured due to remote scp server not responding')
elif index == 1:
child.sendline('y')
# response could be unknown host addition or Password
sub_index = child.expect(['yes', '(?i)password'])
if sub_index == 0:
child.sendline('yes')
child.expect('(?i)password')
elif index == 2:
child.sendline('yes')
child.expect('(?i)password')
child.sendline(module.params['remote_scp_server_password'])
fpt = module.params['file_pull_timeout']
# response could be that there is no space left on device,
# permission denied due to wrong user/password,
# remote file non-existent or success
index = child.expect(['No space', 'Permission denied', 'No such file', '#'], timeout=fpt)
if index == 0:
module.fail_json(msg='File copy failed due to no space left on the device')
elif index == 1:
module.fail_json(msg='Username/Password for remote scp server is wrong')
elif index == 2:
module.fail_json(msg='File copy failed due to remote file not present')
except pexpect.ExceptionPexpect as e:
module.fail_json(msg='%s' % to_native(e), exception=traceback.format_exc())
child.close()
def main(): def main():
argument_spec = dict( argument_spec = dict(
local_file=dict(required=True), local_file=dict(type='str'),
remote_file=dict(required=False), remote_file=dict(type='str'),
file_system=dict(required=False, default='bootflash:'), file_system=dict(required=False, default='bootflash:'),
connect_ssh_port=dict(required=False, type='int', default=22), connect_ssh_port=dict(required=False, type='int', default=22),
file_pull=dict(type='bool', default=False),
file_pull_timeout=dict(type='int', default=300),
remote_scp_server=dict(type='str'),
remote_scp_server_user=dict(type='str'),
remote_scp_server_password=dict(no_log=True),
) )
argument_spec.update(nxos_argument_spec) argument_spec.update(nxos_argument_spec)
module = AnsibleModule(argument_spec=argument_spec, supports_check_mode=True) required_if = [("file_pull", True, ["remote_file", "remote_scp_server"]),
("file_pull", False, ["local_file"])]
if not HAS_PARAMIKO: required_together = [['remote_scp_server',
module.fail_json( 'remote_scp_server_user',
msg='library paramiko is required but does not appear to be ' 'remote_scp_server_password']]
'installed. It can be installed using `pip install paramiko`'
)
if not HAS_SCP: module = AnsibleModule(argument_spec=argument_spec,
module.fail_json( required_if=required_if,
msg='library scp is required but does not appear to be ' required_together=required_together,
'installed. It can be installed using `pip install scp`' supports_check_mode=True)
)
file_pull = module.params['file_pull']
if file_pull:
if not HAS_PEXPECT:
module.fail_json(
msg='library pexpect is required when file_pull is True but does not appear to be '
'installed. It can be installed using `pip install pexpect`'
)
else:
if not HAS_PARAMIKO:
module.fail_json(
msg='library paramiko is required when file_pull is False but does not appear to be '
'installed. It can be installed using `pip install paramiko`'
)
if not HAS_SCP:
module.fail_json(
msg='library scp is required when file_pull is False but does not appear to be '
'installed. It can be installed using `pip install scp`'
)
warnings = list() warnings = list()
check_args(module, warnings) check_args(module, warnings)
results = dict(changed=False, warnings=warnings) results = dict(changed=False, warnings=warnings)
@ -218,28 +363,40 @@ def main():
file_system = module.params['file_system'] file_system = module.params['file_system']
results['transfer_status'] = 'No Transfer' results['transfer_status'] = 'No Transfer'
results['local_file'] = local_file
results['file_system'] = file_system results['file_system'] = file_system
if not local_file_exists(module): if file_pull:
module.fail_json(msg="Local file {0} not found".format(local_file)) src = remote_file.split('/')[-1]
local = local_file or src
dest = remote_file or os.path.basename(local_file) if not module.check_mode:
remote_exists = remote_file_exists(module, dest, file_system=file_system) copy_file_from_remote(module, local, file_system=file_system)
results['transfer_status'] = 'Received'
if not remote_exists:
results['changed'] = True results['changed'] = True
file_exists = False results['remote_file'] = src
results['local_file'] = local
else: else:
file_exists = True if not local_file_exists(module):
module.fail_json(msg="Local file {0} not found".format(local_file))
if not module.check_mode and not file_exists: dest = remote_file or os.path.basename(local_file)
transfer_file(module, dest) remote_exists = remote_file_exists(module, dest, file_system=file_system)
results['transfer_status'] = 'Sent'
if remote_file is None: if not remote_exists:
remote_file = os.path.basename(local_file) results['changed'] = True
results['remote_file'] = remote_file file_exists = False
else:
file_exists = True
if not module.check_mode and not file_exists:
transfer_file_to_device(module, dest)
results['transfer_status'] = 'Sent'
results['local_file'] = local_file
if remote_file is None:
remote_file = os.path.basename(local_file)
results['remote_file'] = remote_file
module.exit_json(**results) module.exit_json(**results)

View file

@ -6,6 +6,7 @@
commands: commands:
- terminal dont-ask - terminal dont-ask
- delete network-integration.cfg - delete network-integration.cfg
- delete network-integration_copy.cfg
ignore_errors: yes ignore_errors: yes
- name: "Setup - Turn on feature scp-server" - name: "Setup - Turn on feature scp-server"
@ -53,10 +54,25 @@
- assert: *false - assert: *false
- name: "Setup - Remove existing file" - name: "Copy file using file_pull"
nxos_command: *remove_file nxos_file_copy: &copy_pull
file_pull: True
file_pull_timeout: 1200
local_file: "network-integration_copy.cfg"
remote_file: "/network-integration.cfg"
remote_scp_server: "{{ inventory_hostname_short }}"
remote_scp_server_user: "{{ ansible_ssh_user }}"
remote_scp_server_password: "{{ ansible_ssh_pass }}"
register: result register: result
- assert: *true
- name: "Overwrite the file"
nxos_file_copy: *copy_pull
register: result
- assert: *true
rescue: rescue:
- debug: msg="TRANSPORT:CLI nxos_file_copy failure detected" - debug: msg="TRANSPORT:CLI nxos_file_copy failure detected"

View file

@ -896,7 +896,6 @@ lib/ansible/modules/network/nxos/nxos_bgp_neighbor_af.py E325
lib/ansible/modules/network/nxos/nxos_bgp_neighbor_af.py E326 lib/ansible/modules/network/nxos/nxos_bgp_neighbor_af.py E326
lib/ansible/modules/network/nxos/nxos_command.py E326 lib/ansible/modules/network/nxos/nxos_command.py E326
lib/ansible/modules/network/nxos/nxos_config.py E324 lib/ansible/modules/network/nxos/nxos_config.py E324
lib/ansible/modules/network/nxos/nxos_file_copy.py E324
lib/ansible/modules/network/nxos/nxos_gir.py E326 lib/ansible/modules/network/nxos/nxos_gir.py E326
lib/ansible/modules/network/nxos/nxos_igmp_interface.py E326 lib/ansible/modules/network/nxos/nxos_igmp_interface.py E326
lib/ansible/modules/network/nxos/nxos_interface.py E324 lib/ansible/modules/network/nxos/nxos_interface.py E324