2016-03-08 20:49:07 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
2016-03-08 07:07:00 +01:00
|
|
|
# (c) 2015, Toshio Kuratomi <tkuratomi@ansible.com>
|
|
|
|
#
|
|
|
|
# This file is part of Ansible
|
|
|
|
#
|
|
|
|
# Ansible is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# Ansible is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
# Make coding more python3-ish
|
|
|
|
from __future__ import (absolute_import, division, print_function)
|
|
|
|
__metaclass__ = type
|
|
|
|
|
|
|
|
from io import StringIO
|
2017-02-01 17:48:23 +01:00
|
|
|
import pytest
|
|
|
|
|
2016-03-08 07:07:00 +01:00
|
|
|
|
2016-03-08 20:49:07 +01:00
|
|
|
from ansible import constants as C
|
2019-01-23 17:32:25 +01:00
|
|
|
from ansible.errors import AnsibleAuthenticationFailure
|
2017-02-01 17:48:23 +01:00
|
|
|
from ansible.compat.selectors import SelectorKey, EVENT_READ
|
2018-10-13 05:01:14 +02:00
|
|
|
from units.compat import unittest
|
|
|
|
from units.compat.mock import patch, MagicMock, PropertyMock
|
2016-03-08 20:49:07 +01:00
|
|
|
from ansible.errors import AnsibleError, AnsibleConnectionFailure, AnsibleFileNotFound
|
2017-03-23 21:35:05 +01:00
|
|
|
from ansible.module_utils.six.moves import shlex_quote
|
2017-05-30 19:05:19 +02:00
|
|
|
from ansible.module_utils._text import to_bytes
|
2016-03-08 07:07:00 +01:00
|
|
|
from ansible.playbook.play_context import PlayContext
|
|
|
|
from ansible.plugins.connection import ssh
|
Become plugins (#50991)
* [WIP] become plugins
Move from hardcoded method to plugins for ease of use, expansion and overrides
- load into connection as it is going to be the main consumer
- play_context will also use to keep backwards compat API
- ensure shell is used to construct commands when needed
- migrate settings remove from base config in favor of plugin specific configs
- cleanup ansible-doc
- add become plugin docs
- remove deprecated sudo/su code and keywords
- adjust become options for cli
- set plugin options from context
- ensure config defs are avaialbe before instance
- refactored getting the shell plugin, fixed tests
- changed into regex as they were string matching, which does not work with random string generation
- explicitly set flags for play context tests
- moved plugin loading up front
- now loads for basedir also
- allow pyc/o for non m modules
- fixes to tests and some plugins
- migrate to play objects fro play_context
- simiplify gathering
- added utf8 headers
- moved option setting
- add fail msg to dzdo
- use tuple for multiple options on fail/missing
- fix relative plugin paths
- shift from play context to play
- all tasks already inherit this from play directly
- remove obsolete 'set play'
- correct environment handling
- add wrap_exe option to pfexec
- fix runas to noop
- fixed setting play context
- added password configs
- removed required false
- remove from doc building till they are ready
future development:
- deal with 'enable' and 'runas' which are not 'command wrappers' but 'state flags' and currently hardcoded in diff subsystems
* cleanup
remove callers to removed func
removed --sudo cli doc refs
remove runas become_exe
ensure keyerorr on plugin
also fix backwards compat, missing method is attributeerror, not ansible error
get remote_user consistently
ignore missing system_tmpdirs on plugin load
correct config precedence
add deprecation
fix networking imports
backwards compat for plugins using BECOME_METHODS
* Port become_plugins to context.CLIARGS
This is a work in progress:
* Stop passing options around everywhere as we can use context.CLIARGS
instead
* Refactor make_become_commands as asked for by alikins
* Typo in comment fix
* Stop loading values from the cli in more than one place
Both play and play_context were saving default values from the cli
arguments directly. This changes things so that the default values are
loaded into the play and then play_context takes them from there.
* Rename BECOME_PLUGIN_PATH to DEFAULT_BECOME_PLUGIN_PATH
As alikins said, all other plugin paths are named
DEFAULT_plugintype_PLUGIN_PATH. If we're going to rename these, that
should be done all at one time rather than piecemeal.
* One to throw away
This is a set of hacks to get setting FieldAttribute defaults to command
line args to work. It's not fully done yet.
After talking it over with sivel and jimi-c this should be done by
fixing FieldAttributeBase and _get_parent_attribute() calls to do the
right thing when there is a non-None default.
What we want to be able to do ideally is something like this:
class Base(FieldAttributeBase):
_check_mode = FieldAttribute([..] default=lambda: context.CLIARGS['check'])
class Play(Base):
# lambda so that we have a chance to parse the command line args
# before we get here. In the future we might be able to restructure
# this so that the cli parsing code runs before these classes are
# defined.
class Task(Base):
pass
And still have a playbook like this function:
---
- hosts:
tasks:
- command: whoami
check_mode: True
(The check_mode test that is added as a separate commit in this PR will
let you test variations on this case).
There's a few separate reasons that the code doesn't let us do this or
a non-ugly workaround for this as written right now. The fix that
jimi-c, sivel, and I talked about may let us do this or it may still
require a workaround (but less ugly) (having one class that has the
FieldAttributes with default values and one class that inherits from
that but just overrides the FieldAttributes which now have defaults)
* Revert "One to throw away"
This reverts commit 23aa883cbed11429ef1be2a2d0ed18f83a3b8064.
* Set FieldAttr defaults directly from CLIARGS
* Remove dead code
* Move timeout directly to PlayContext, it's never needed on Play
* just for backwards compat, add a static version of BECOME_METHODS to constants
* Make the become attr on the connection public, since it's used outside of the connection
* Logic fix
* Nuke connection testing if it supports specific become methods
* Remove unused vars
* Address rebase issues
* Fix path encoding issue
* Remove unused import
* Various cleanups
* Restore network_cli check in _low_level_execute_command
* type improvements for cliargs_deferred_get and swap shallowcopy to default to False
* minor cleanups
* Allow the su plugin to work, since it doesn't define a prompt the same way
* Fix up ksu become plugin
* Only set prompt if build_become_command was called
* Add helper to assist connection plugins in knowing they need to wait for a prompt
* Fix tests and code expectations
* Doc updates
* Various additional minor cleanups
* Make doas functional
* Don't change connection signature, load become plugin from TaskExecutor
* Remove unused imports
* Add comment about setting the become plugin on the playcontext
* Fix up tests for recent changes
* Support 'Password:' natively for the doas plugin
* Make default prompts raw
* wording cleanups. ci_complete
* Remove unrelated changes
* Address spelling mistake
* Restore removed test, and udpate to use new functionality
* Add changelog fragment
* Don't hard fail in set_attributes_from_cli on missing CLI keys
* Remove unrelated change to loader
* Remove internal deprecated FieldAttributes now
* Emit deprecation warnings now
2019-02-11 18:27:44 +01:00
|
|
|
from ansible.plugins.loader import connection_loader, become_loader
|
2016-09-07 07:54:17 +02:00
|
|
|
|
2016-03-08 07:07:00 +01:00
|
|
|
|
|
|
|
class TestConnectionBaseClass(unittest.TestCase):
|
|
|
|
|
|
|
|
def test_plugins_connection_ssh_basic(self):
|
|
|
|
pc = PlayContext()
|
|
|
|
new_stdin = StringIO()
|
|
|
|
conn = ssh.Connection(pc, new_stdin)
|
|
|
|
|
|
|
|
# connect just returns self, so assert that
|
|
|
|
res = conn._connect()
|
|
|
|
self.assertEqual(conn, res)
|
|
|
|
|
|
|
|
ssh.SSHPASS_AVAILABLE = False
|
|
|
|
self.assertFalse(conn._sshpass_available())
|
|
|
|
|
|
|
|
ssh.SSHPASS_AVAILABLE = True
|
|
|
|
self.assertTrue(conn._sshpass_available())
|
|
|
|
|
|
|
|
with patch('subprocess.Popen') as p:
|
|
|
|
ssh.SSHPASS_AVAILABLE = None
|
|
|
|
p.return_value = MagicMock()
|
|
|
|
self.assertTrue(conn._sshpass_available())
|
|
|
|
|
|
|
|
ssh.SSHPASS_AVAILABLE = None
|
|
|
|
p.return_value = None
|
|
|
|
p.side_effect = OSError()
|
|
|
|
self.assertFalse(conn._sshpass_available())
|
|
|
|
|
|
|
|
conn.close()
|
|
|
|
self.assertFalse(conn._connected)
|
|
|
|
|
|
|
|
def test_plugins_connection_ssh__build_command(self):
|
|
|
|
pc = PlayContext()
|
|
|
|
new_stdin = StringIO()
|
2018-04-24 00:36:35 +02:00
|
|
|
conn = connection_loader.get('ssh', pc, new_stdin)
|
2016-03-08 07:07:00 +01:00
|
|
|
conn._build_command('ssh')
|
|
|
|
|
2017-03-02 23:56:29 +01:00
|
|
|
def test_plugins_connection_ssh_exec_command(self):
|
2016-03-08 07:07:00 +01:00
|
|
|
pc = PlayContext()
|
|
|
|
new_stdin = StringIO()
|
2018-04-24 00:36:35 +02:00
|
|
|
conn = connection_loader.get('ssh', pc, new_stdin)
|
2016-03-08 07:07:00 +01:00
|
|
|
|
|
|
|
conn._build_command = MagicMock()
|
|
|
|
conn._build_command.return_value = 'ssh something something'
|
|
|
|
conn._run = MagicMock()
|
|
|
|
conn._run.return_value = (0, 'stdout', 'stderr')
|
2017-11-22 17:19:43 +01:00
|
|
|
conn.get_option = MagicMock()
|
|
|
|
conn.get_option.return_value = True
|
2016-03-08 07:07:00 +01:00
|
|
|
|
2017-03-02 23:56:29 +01:00
|
|
|
res, stdout, stderr = conn.exec_command('ssh')
|
|
|
|
res, stdout, stderr = conn.exec_command('ssh', 'this is some data')
|
2016-03-08 07:07:00 +01:00
|
|
|
|
|
|
|
def test_plugins_connection_ssh__examine_output(self):
|
|
|
|
pc = PlayContext()
|
|
|
|
new_stdin = StringIO()
|
|
|
|
|
2018-04-24 00:36:35 +02:00
|
|
|
conn = connection_loader.get('ssh', pc, new_stdin)
|
Become plugins (#50991)
* [WIP] become plugins
Move from hardcoded method to plugins for ease of use, expansion and overrides
- load into connection as it is going to be the main consumer
- play_context will also use to keep backwards compat API
- ensure shell is used to construct commands when needed
- migrate settings remove from base config in favor of plugin specific configs
- cleanup ansible-doc
- add become plugin docs
- remove deprecated sudo/su code and keywords
- adjust become options for cli
- set plugin options from context
- ensure config defs are avaialbe before instance
- refactored getting the shell plugin, fixed tests
- changed into regex as they were string matching, which does not work with random string generation
- explicitly set flags for play context tests
- moved plugin loading up front
- now loads for basedir also
- allow pyc/o for non m modules
- fixes to tests and some plugins
- migrate to play objects fro play_context
- simiplify gathering
- added utf8 headers
- moved option setting
- add fail msg to dzdo
- use tuple for multiple options on fail/missing
- fix relative plugin paths
- shift from play context to play
- all tasks already inherit this from play directly
- remove obsolete 'set play'
- correct environment handling
- add wrap_exe option to pfexec
- fix runas to noop
- fixed setting play context
- added password configs
- removed required false
- remove from doc building till they are ready
future development:
- deal with 'enable' and 'runas' which are not 'command wrappers' but 'state flags' and currently hardcoded in diff subsystems
* cleanup
remove callers to removed func
removed --sudo cli doc refs
remove runas become_exe
ensure keyerorr on plugin
also fix backwards compat, missing method is attributeerror, not ansible error
get remote_user consistently
ignore missing system_tmpdirs on plugin load
correct config precedence
add deprecation
fix networking imports
backwards compat for plugins using BECOME_METHODS
* Port become_plugins to context.CLIARGS
This is a work in progress:
* Stop passing options around everywhere as we can use context.CLIARGS
instead
* Refactor make_become_commands as asked for by alikins
* Typo in comment fix
* Stop loading values from the cli in more than one place
Both play and play_context were saving default values from the cli
arguments directly. This changes things so that the default values are
loaded into the play and then play_context takes them from there.
* Rename BECOME_PLUGIN_PATH to DEFAULT_BECOME_PLUGIN_PATH
As alikins said, all other plugin paths are named
DEFAULT_plugintype_PLUGIN_PATH. If we're going to rename these, that
should be done all at one time rather than piecemeal.
* One to throw away
This is a set of hacks to get setting FieldAttribute defaults to command
line args to work. It's not fully done yet.
After talking it over with sivel and jimi-c this should be done by
fixing FieldAttributeBase and _get_parent_attribute() calls to do the
right thing when there is a non-None default.
What we want to be able to do ideally is something like this:
class Base(FieldAttributeBase):
_check_mode = FieldAttribute([..] default=lambda: context.CLIARGS['check'])
class Play(Base):
# lambda so that we have a chance to parse the command line args
# before we get here. In the future we might be able to restructure
# this so that the cli parsing code runs before these classes are
# defined.
class Task(Base):
pass
And still have a playbook like this function:
---
- hosts:
tasks:
- command: whoami
check_mode: True
(The check_mode test that is added as a separate commit in this PR will
let you test variations on this case).
There's a few separate reasons that the code doesn't let us do this or
a non-ugly workaround for this as written right now. The fix that
jimi-c, sivel, and I talked about may let us do this or it may still
require a workaround (but less ugly) (having one class that has the
FieldAttributes with default values and one class that inherits from
that but just overrides the FieldAttributes which now have defaults)
* Revert "One to throw away"
This reverts commit 23aa883cbed11429ef1be2a2d0ed18f83a3b8064.
* Set FieldAttr defaults directly from CLIARGS
* Remove dead code
* Move timeout directly to PlayContext, it's never needed on Play
* just for backwards compat, add a static version of BECOME_METHODS to constants
* Make the become attr on the connection public, since it's used outside of the connection
* Logic fix
* Nuke connection testing if it supports specific become methods
* Remove unused vars
* Address rebase issues
* Fix path encoding issue
* Remove unused import
* Various cleanups
* Restore network_cli check in _low_level_execute_command
* type improvements for cliargs_deferred_get and swap shallowcopy to default to False
* minor cleanups
* Allow the su plugin to work, since it doesn't define a prompt the same way
* Fix up ksu become plugin
* Only set prompt if build_become_command was called
* Add helper to assist connection plugins in knowing they need to wait for a prompt
* Fix tests and code expectations
* Doc updates
* Various additional minor cleanups
* Make doas functional
* Don't change connection signature, load become plugin from TaskExecutor
* Remove unused imports
* Add comment about setting the become plugin on the playcontext
* Fix up tests for recent changes
* Support 'Password:' natively for the doas plugin
* Make default prompts raw
* wording cleanups. ci_complete
* Remove unrelated changes
* Address spelling mistake
* Restore removed test, and udpate to use new functionality
* Add changelog fragment
* Don't hard fail in set_attributes_from_cli on missing CLI keys
* Remove unrelated change to loader
* Remove internal deprecated FieldAttributes now
* Emit deprecation warnings now
2019-02-11 18:27:44 +01:00
|
|
|
conn.set_become_plugin(become_loader.get('sudo'))
|
2016-03-08 07:07:00 +01:00
|
|
|
|
2017-05-30 19:05:19 +02:00
|
|
|
conn.check_password_prompt = MagicMock()
|
|
|
|
conn.check_become_success = MagicMock()
|
2016-03-08 07:07:00 +01:00
|
|
|
conn.check_incorrect_password = MagicMock()
|
2017-05-30 19:05:19 +02:00
|
|
|
conn.check_missing_password = MagicMock()
|
2016-03-08 07:07:00 +01:00
|
|
|
|
|
|
|
def _check_password_prompt(line):
|
2016-08-25 19:57:55 +02:00
|
|
|
if b'foo' in line:
|
2016-03-08 07:07:00 +01:00
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
def _check_become_success(line):
|
2016-08-25 19:57:55 +02:00
|
|
|
if b'BECOME-SUCCESS-abcdefghijklmnopqrstuvxyz' in line:
|
2016-03-08 07:07:00 +01:00
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
def _check_incorrect_password(line):
|
2016-08-25 19:57:55 +02:00
|
|
|
if b'incorrect password' in line:
|
2016-03-08 07:07:00 +01:00
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
def _check_missing_password(line):
|
2016-08-25 19:57:55 +02:00
|
|
|
if b'bad password' in line:
|
2016-03-08 07:07:00 +01:00
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
2017-05-30 19:05:19 +02:00
|
|
|
conn.check_password_prompt.side_effect = _check_password_prompt
|
|
|
|
conn.check_become_success.side_effect = _check_become_success
|
2016-03-08 07:07:00 +01:00
|
|
|
conn.check_incorrect_password.side_effect = _check_incorrect_password
|
2017-05-30 19:05:19 +02:00
|
|
|
conn.check_missing_password.side_effect = _check_missing_password
|
2016-03-08 07:07:00 +01:00
|
|
|
|
|
|
|
# test examining output for prompt
|
|
|
|
conn._flags = dict(
|
2017-05-30 19:05:19 +02:00
|
|
|
become_prompt=False,
|
|
|
|
become_success=False,
|
|
|
|
become_error=False,
|
|
|
|
become_nopasswd_error=False,
|
2016-03-08 07:07:00 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
pc.prompt = True
|
Become plugins (#50991)
* [WIP] become plugins
Move from hardcoded method to plugins for ease of use, expansion and overrides
- load into connection as it is going to be the main consumer
- play_context will also use to keep backwards compat API
- ensure shell is used to construct commands when needed
- migrate settings remove from base config in favor of plugin specific configs
- cleanup ansible-doc
- add become plugin docs
- remove deprecated sudo/su code and keywords
- adjust become options for cli
- set plugin options from context
- ensure config defs are avaialbe before instance
- refactored getting the shell plugin, fixed tests
- changed into regex as they were string matching, which does not work with random string generation
- explicitly set flags for play context tests
- moved plugin loading up front
- now loads for basedir also
- allow pyc/o for non m modules
- fixes to tests and some plugins
- migrate to play objects fro play_context
- simiplify gathering
- added utf8 headers
- moved option setting
- add fail msg to dzdo
- use tuple for multiple options on fail/missing
- fix relative plugin paths
- shift from play context to play
- all tasks already inherit this from play directly
- remove obsolete 'set play'
- correct environment handling
- add wrap_exe option to pfexec
- fix runas to noop
- fixed setting play context
- added password configs
- removed required false
- remove from doc building till they are ready
future development:
- deal with 'enable' and 'runas' which are not 'command wrappers' but 'state flags' and currently hardcoded in diff subsystems
* cleanup
remove callers to removed func
removed --sudo cli doc refs
remove runas become_exe
ensure keyerorr on plugin
also fix backwards compat, missing method is attributeerror, not ansible error
get remote_user consistently
ignore missing system_tmpdirs on plugin load
correct config precedence
add deprecation
fix networking imports
backwards compat for plugins using BECOME_METHODS
* Port become_plugins to context.CLIARGS
This is a work in progress:
* Stop passing options around everywhere as we can use context.CLIARGS
instead
* Refactor make_become_commands as asked for by alikins
* Typo in comment fix
* Stop loading values from the cli in more than one place
Both play and play_context were saving default values from the cli
arguments directly. This changes things so that the default values are
loaded into the play and then play_context takes them from there.
* Rename BECOME_PLUGIN_PATH to DEFAULT_BECOME_PLUGIN_PATH
As alikins said, all other plugin paths are named
DEFAULT_plugintype_PLUGIN_PATH. If we're going to rename these, that
should be done all at one time rather than piecemeal.
* One to throw away
This is a set of hacks to get setting FieldAttribute defaults to command
line args to work. It's not fully done yet.
After talking it over with sivel and jimi-c this should be done by
fixing FieldAttributeBase and _get_parent_attribute() calls to do the
right thing when there is a non-None default.
What we want to be able to do ideally is something like this:
class Base(FieldAttributeBase):
_check_mode = FieldAttribute([..] default=lambda: context.CLIARGS['check'])
class Play(Base):
# lambda so that we have a chance to parse the command line args
# before we get here. In the future we might be able to restructure
# this so that the cli parsing code runs before these classes are
# defined.
class Task(Base):
pass
And still have a playbook like this function:
---
- hosts:
tasks:
- command: whoami
check_mode: True
(The check_mode test that is added as a separate commit in this PR will
let you test variations on this case).
There's a few separate reasons that the code doesn't let us do this or
a non-ugly workaround for this as written right now. The fix that
jimi-c, sivel, and I talked about may let us do this or it may still
require a workaround (but less ugly) (having one class that has the
FieldAttributes with default values and one class that inherits from
that but just overrides the FieldAttributes which now have defaults)
* Revert "One to throw away"
This reverts commit 23aa883cbed11429ef1be2a2d0ed18f83a3b8064.
* Set FieldAttr defaults directly from CLIARGS
* Remove dead code
* Move timeout directly to PlayContext, it's never needed on Play
* just for backwards compat, add a static version of BECOME_METHODS to constants
* Make the become attr on the connection public, since it's used outside of the connection
* Logic fix
* Nuke connection testing if it supports specific become methods
* Remove unused vars
* Address rebase issues
* Fix path encoding issue
* Remove unused import
* Various cleanups
* Restore network_cli check in _low_level_execute_command
* type improvements for cliargs_deferred_get and swap shallowcopy to default to False
* minor cleanups
* Allow the su plugin to work, since it doesn't define a prompt the same way
* Fix up ksu become plugin
* Only set prompt if build_become_command was called
* Add helper to assist connection plugins in knowing they need to wait for a prompt
* Fix tests and code expectations
* Doc updates
* Various additional minor cleanups
* Make doas functional
* Don't change connection signature, load become plugin from TaskExecutor
* Remove unused imports
* Add comment about setting the become plugin on the playcontext
* Fix up tests for recent changes
* Support 'Password:' natively for the doas plugin
* Make default prompts raw
* wording cleanups. ci_complete
* Remove unrelated changes
* Address spelling mistake
* Restore removed test, and udpate to use new functionality
* Add changelog fragment
* Don't hard fail in set_attributes_from_cli on missing CLI keys
* Remove unrelated change to loader
* Remove internal deprecated FieldAttributes now
* Emit deprecation warnings now
2019-02-11 18:27:44 +01:00
|
|
|
conn.become.prompt = True
|
|
|
|
|
|
|
|
def get_option(option):
|
|
|
|
if option == 'become_pass':
|
|
|
|
return 'password'
|
|
|
|
return None
|
|
|
|
|
|
|
|
conn.become.get_option = get_option
|
2016-08-25 19:57:55 +02:00
|
|
|
output, unprocessed = conn._examine_output(u'source', u'state', b'line 1\nline 2\nfoo\nline 3\nthis should be the remainder', False)
|
|
|
|
self.assertEqual(output, b'line 1\nline 2\nline 3\n')
|
|
|
|
self.assertEqual(unprocessed, b'this should be the remainder')
|
2016-03-08 07:07:00 +01:00
|
|
|
self.assertTrue(conn._flags['become_prompt'])
|
|
|
|
self.assertFalse(conn._flags['become_success'])
|
|
|
|
self.assertFalse(conn._flags['become_error'])
|
|
|
|
self.assertFalse(conn._flags['become_nopasswd_error'])
|
|
|
|
|
|
|
|
# test examining output for become prompt
|
|
|
|
conn._flags = dict(
|
2017-05-30 19:05:19 +02:00
|
|
|
become_prompt=False,
|
|
|
|
become_success=False,
|
|
|
|
become_error=False,
|
|
|
|
become_nopasswd_error=False,
|
2016-03-08 07:07:00 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
pc.prompt = False
|
Become plugins (#50991)
* [WIP] become plugins
Move from hardcoded method to plugins for ease of use, expansion and overrides
- load into connection as it is going to be the main consumer
- play_context will also use to keep backwards compat API
- ensure shell is used to construct commands when needed
- migrate settings remove from base config in favor of plugin specific configs
- cleanup ansible-doc
- add become plugin docs
- remove deprecated sudo/su code and keywords
- adjust become options for cli
- set plugin options from context
- ensure config defs are avaialbe before instance
- refactored getting the shell plugin, fixed tests
- changed into regex as they were string matching, which does not work with random string generation
- explicitly set flags for play context tests
- moved plugin loading up front
- now loads for basedir also
- allow pyc/o for non m modules
- fixes to tests and some plugins
- migrate to play objects fro play_context
- simiplify gathering
- added utf8 headers
- moved option setting
- add fail msg to dzdo
- use tuple for multiple options on fail/missing
- fix relative plugin paths
- shift from play context to play
- all tasks already inherit this from play directly
- remove obsolete 'set play'
- correct environment handling
- add wrap_exe option to pfexec
- fix runas to noop
- fixed setting play context
- added password configs
- removed required false
- remove from doc building till they are ready
future development:
- deal with 'enable' and 'runas' which are not 'command wrappers' but 'state flags' and currently hardcoded in diff subsystems
* cleanup
remove callers to removed func
removed --sudo cli doc refs
remove runas become_exe
ensure keyerorr on plugin
also fix backwards compat, missing method is attributeerror, not ansible error
get remote_user consistently
ignore missing system_tmpdirs on plugin load
correct config precedence
add deprecation
fix networking imports
backwards compat for plugins using BECOME_METHODS
* Port become_plugins to context.CLIARGS
This is a work in progress:
* Stop passing options around everywhere as we can use context.CLIARGS
instead
* Refactor make_become_commands as asked for by alikins
* Typo in comment fix
* Stop loading values from the cli in more than one place
Both play and play_context were saving default values from the cli
arguments directly. This changes things so that the default values are
loaded into the play and then play_context takes them from there.
* Rename BECOME_PLUGIN_PATH to DEFAULT_BECOME_PLUGIN_PATH
As alikins said, all other plugin paths are named
DEFAULT_plugintype_PLUGIN_PATH. If we're going to rename these, that
should be done all at one time rather than piecemeal.
* One to throw away
This is a set of hacks to get setting FieldAttribute defaults to command
line args to work. It's not fully done yet.
After talking it over with sivel and jimi-c this should be done by
fixing FieldAttributeBase and _get_parent_attribute() calls to do the
right thing when there is a non-None default.
What we want to be able to do ideally is something like this:
class Base(FieldAttributeBase):
_check_mode = FieldAttribute([..] default=lambda: context.CLIARGS['check'])
class Play(Base):
# lambda so that we have a chance to parse the command line args
# before we get here. In the future we might be able to restructure
# this so that the cli parsing code runs before these classes are
# defined.
class Task(Base):
pass
And still have a playbook like this function:
---
- hosts:
tasks:
- command: whoami
check_mode: True
(The check_mode test that is added as a separate commit in this PR will
let you test variations on this case).
There's a few separate reasons that the code doesn't let us do this or
a non-ugly workaround for this as written right now. The fix that
jimi-c, sivel, and I talked about may let us do this or it may still
require a workaround (but less ugly) (having one class that has the
FieldAttributes with default values and one class that inherits from
that but just overrides the FieldAttributes which now have defaults)
* Revert "One to throw away"
This reverts commit 23aa883cbed11429ef1be2a2d0ed18f83a3b8064.
* Set FieldAttr defaults directly from CLIARGS
* Remove dead code
* Move timeout directly to PlayContext, it's never needed on Play
* just for backwards compat, add a static version of BECOME_METHODS to constants
* Make the become attr on the connection public, since it's used outside of the connection
* Logic fix
* Nuke connection testing if it supports specific become methods
* Remove unused vars
* Address rebase issues
* Fix path encoding issue
* Remove unused import
* Various cleanups
* Restore network_cli check in _low_level_execute_command
* type improvements for cliargs_deferred_get and swap shallowcopy to default to False
* minor cleanups
* Allow the su plugin to work, since it doesn't define a prompt the same way
* Fix up ksu become plugin
* Only set prompt if build_become_command was called
* Add helper to assist connection plugins in knowing they need to wait for a prompt
* Fix tests and code expectations
* Doc updates
* Various additional minor cleanups
* Make doas functional
* Don't change connection signature, load become plugin from TaskExecutor
* Remove unused imports
* Add comment about setting the become plugin on the playcontext
* Fix up tests for recent changes
* Support 'Password:' natively for the doas plugin
* Make default prompts raw
* wording cleanups. ci_complete
* Remove unrelated changes
* Address spelling mistake
* Restore removed test, and udpate to use new functionality
* Add changelog fragment
* Don't hard fail in set_attributes_from_cli on missing CLI keys
* Remove unrelated change to loader
* Remove internal deprecated FieldAttributes now
* Emit deprecation warnings now
2019-02-11 18:27:44 +01:00
|
|
|
conn.become.prompt = False
|
2016-08-25 19:57:55 +02:00
|
|
|
pc.success_key = u'BECOME-SUCCESS-abcdefghijklmnopqrstuvxyz'
|
Become plugins (#50991)
* [WIP] become plugins
Move from hardcoded method to plugins for ease of use, expansion and overrides
- load into connection as it is going to be the main consumer
- play_context will also use to keep backwards compat API
- ensure shell is used to construct commands when needed
- migrate settings remove from base config in favor of plugin specific configs
- cleanup ansible-doc
- add become plugin docs
- remove deprecated sudo/su code and keywords
- adjust become options for cli
- set plugin options from context
- ensure config defs are avaialbe before instance
- refactored getting the shell plugin, fixed tests
- changed into regex as they were string matching, which does not work with random string generation
- explicitly set flags for play context tests
- moved plugin loading up front
- now loads for basedir also
- allow pyc/o for non m modules
- fixes to tests and some plugins
- migrate to play objects fro play_context
- simiplify gathering
- added utf8 headers
- moved option setting
- add fail msg to dzdo
- use tuple for multiple options on fail/missing
- fix relative plugin paths
- shift from play context to play
- all tasks already inherit this from play directly
- remove obsolete 'set play'
- correct environment handling
- add wrap_exe option to pfexec
- fix runas to noop
- fixed setting play context
- added password configs
- removed required false
- remove from doc building till they are ready
future development:
- deal with 'enable' and 'runas' which are not 'command wrappers' but 'state flags' and currently hardcoded in diff subsystems
* cleanup
remove callers to removed func
removed --sudo cli doc refs
remove runas become_exe
ensure keyerorr on plugin
also fix backwards compat, missing method is attributeerror, not ansible error
get remote_user consistently
ignore missing system_tmpdirs on plugin load
correct config precedence
add deprecation
fix networking imports
backwards compat for plugins using BECOME_METHODS
* Port become_plugins to context.CLIARGS
This is a work in progress:
* Stop passing options around everywhere as we can use context.CLIARGS
instead
* Refactor make_become_commands as asked for by alikins
* Typo in comment fix
* Stop loading values from the cli in more than one place
Both play and play_context were saving default values from the cli
arguments directly. This changes things so that the default values are
loaded into the play and then play_context takes them from there.
* Rename BECOME_PLUGIN_PATH to DEFAULT_BECOME_PLUGIN_PATH
As alikins said, all other plugin paths are named
DEFAULT_plugintype_PLUGIN_PATH. If we're going to rename these, that
should be done all at one time rather than piecemeal.
* One to throw away
This is a set of hacks to get setting FieldAttribute defaults to command
line args to work. It's not fully done yet.
After talking it over with sivel and jimi-c this should be done by
fixing FieldAttributeBase and _get_parent_attribute() calls to do the
right thing when there is a non-None default.
What we want to be able to do ideally is something like this:
class Base(FieldAttributeBase):
_check_mode = FieldAttribute([..] default=lambda: context.CLIARGS['check'])
class Play(Base):
# lambda so that we have a chance to parse the command line args
# before we get here. In the future we might be able to restructure
# this so that the cli parsing code runs before these classes are
# defined.
class Task(Base):
pass
And still have a playbook like this function:
---
- hosts:
tasks:
- command: whoami
check_mode: True
(The check_mode test that is added as a separate commit in this PR will
let you test variations on this case).
There's a few separate reasons that the code doesn't let us do this or
a non-ugly workaround for this as written right now. The fix that
jimi-c, sivel, and I talked about may let us do this or it may still
require a workaround (but less ugly) (having one class that has the
FieldAttributes with default values and one class that inherits from
that but just overrides the FieldAttributes which now have defaults)
* Revert "One to throw away"
This reverts commit 23aa883cbed11429ef1be2a2d0ed18f83a3b8064.
* Set FieldAttr defaults directly from CLIARGS
* Remove dead code
* Move timeout directly to PlayContext, it's never needed on Play
* just for backwards compat, add a static version of BECOME_METHODS to constants
* Make the become attr on the connection public, since it's used outside of the connection
* Logic fix
* Nuke connection testing if it supports specific become methods
* Remove unused vars
* Address rebase issues
* Fix path encoding issue
* Remove unused import
* Various cleanups
* Restore network_cli check in _low_level_execute_command
* type improvements for cliargs_deferred_get and swap shallowcopy to default to False
* minor cleanups
* Allow the su plugin to work, since it doesn't define a prompt the same way
* Fix up ksu become plugin
* Only set prompt if build_become_command was called
* Add helper to assist connection plugins in knowing they need to wait for a prompt
* Fix tests and code expectations
* Doc updates
* Various additional minor cleanups
* Make doas functional
* Don't change connection signature, load become plugin from TaskExecutor
* Remove unused imports
* Add comment about setting the become plugin on the playcontext
* Fix up tests for recent changes
* Support 'Password:' natively for the doas plugin
* Make default prompts raw
* wording cleanups. ci_complete
* Remove unrelated changes
* Address spelling mistake
* Restore removed test, and udpate to use new functionality
* Add changelog fragment
* Don't hard fail in set_attributes_from_cli on missing CLI keys
* Remove unrelated change to loader
* Remove internal deprecated FieldAttributes now
* Emit deprecation warnings now
2019-02-11 18:27:44 +01:00
|
|
|
conn.become.success = u'BECOME-SUCCESS-abcdefghijklmnopqrstuvxyz'
|
2016-08-25 19:57:55 +02:00
|
|
|
output, unprocessed = conn._examine_output(u'source', u'state', b'line 1\nline 2\nBECOME-SUCCESS-abcdefghijklmnopqrstuvxyz\nline 3\n', False)
|
|
|
|
self.assertEqual(output, b'line 1\nline 2\nline 3\n')
|
|
|
|
self.assertEqual(unprocessed, b'')
|
2016-03-08 07:07:00 +01:00
|
|
|
self.assertFalse(conn._flags['become_prompt'])
|
|
|
|
self.assertTrue(conn._flags['become_success'])
|
|
|
|
self.assertFalse(conn._flags['become_error'])
|
|
|
|
self.assertFalse(conn._flags['become_nopasswd_error'])
|
|
|
|
|
|
|
|
# test examining output for become failure
|
|
|
|
conn._flags = dict(
|
2017-05-30 19:05:19 +02:00
|
|
|
become_prompt=False,
|
|
|
|
become_success=False,
|
|
|
|
become_error=False,
|
|
|
|
become_nopasswd_error=False,
|
2016-03-08 07:07:00 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
pc.prompt = False
|
Become plugins (#50991)
* [WIP] become plugins
Move from hardcoded method to plugins for ease of use, expansion and overrides
- load into connection as it is going to be the main consumer
- play_context will also use to keep backwards compat API
- ensure shell is used to construct commands when needed
- migrate settings remove from base config in favor of plugin specific configs
- cleanup ansible-doc
- add become plugin docs
- remove deprecated sudo/su code and keywords
- adjust become options for cli
- set plugin options from context
- ensure config defs are avaialbe before instance
- refactored getting the shell plugin, fixed tests
- changed into regex as they were string matching, which does not work with random string generation
- explicitly set flags for play context tests
- moved plugin loading up front
- now loads for basedir also
- allow pyc/o for non m modules
- fixes to tests and some plugins
- migrate to play objects fro play_context
- simiplify gathering
- added utf8 headers
- moved option setting
- add fail msg to dzdo
- use tuple for multiple options on fail/missing
- fix relative plugin paths
- shift from play context to play
- all tasks already inherit this from play directly
- remove obsolete 'set play'
- correct environment handling
- add wrap_exe option to pfexec
- fix runas to noop
- fixed setting play context
- added password configs
- removed required false
- remove from doc building till they are ready
future development:
- deal with 'enable' and 'runas' which are not 'command wrappers' but 'state flags' and currently hardcoded in diff subsystems
* cleanup
remove callers to removed func
removed --sudo cli doc refs
remove runas become_exe
ensure keyerorr on plugin
also fix backwards compat, missing method is attributeerror, not ansible error
get remote_user consistently
ignore missing system_tmpdirs on plugin load
correct config precedence
add deprecation
fix networking imports
backwards compat for plugins using BECOME_METHODS
* Port become_plugins to context.CLIARGS
This is a work in progress:
* Stop passing options around everywhere as we can use context.CLIARGS
instead
* Refactor make_become_commands as asked for by alikins
* Typo in comment fix
* Stop loading values from the cli in more than one place
Both play and play_context were saving default values from the cli
arguments directly. This changes things so that the default values are
loaded into the play and then play_context takes them from there.
* Rename BECOME_PLUGIN_PATH to DEFAULT_BECOME_PLUGIN_PATH
As alikins said, all other plugin paths are named
DEFAULT_plugintype_PLUGIN_PATH. If we're going to rename these, that
should be done all at one time rather than piecemeal.
* One to throw away
This is a set of hacks to get setting FieldAttribute defaults to command
line args to work. It's not fully done yet.
After talking it over with sivel and jimi-c this should be done by
fixing FieldAttributeBase and _get_parent_attribute() calls to do the
right thing when there is a non-None default.
What we want to be able to do ideally is something like this:
class Base(FieldAttributeBase):
_check_mode = FieldAttribute([..] default=lambda: context.CLIARGS['check'])
class Play(Base):
# lambda so that we have a chance to parse the command line args
# before we get here. In the future we might be able to restructure
# this so that the cli parsing code runs before these classes are
# defined.
class Task(Base):
pass
And still have a playbook like this function:
---
- hosts:
tasks:
- command: whoami
check_mode: True
(The check_mode test that is added as a separate commit in this PR will
let you test variations on this case).
There's a few separate reasons that the code doesn't let us do this or
a non-ugly workaround for this as written right now. The fix that
jimi-c, sivel, and I talked about may let us do this or it may still
require a workaround (but less ugly) (having one class that has the
FieldAttributes with default values and one class that inherits from
that but just overrides the FieldAttributes which now have defaults)
* Revert "One to throw away"
This reverts commit 23aa883cbed11429ef1be2a2d0ed18f83a3b8064.
* Set FieldAttr defaults directly from CLIARGS
* Remove dead code
* Move timeout directly to PlayContext, it's never needed on Play
* just for backwards compat, add a static version of BECOME_METHODS to constants
* Make the become attr on the connection public, since it's used outside of the connection
* Logic fix
* Nuke connection testing if it supports specific become methods
* Remove unused vars
* Address rebase issues
* Fix path encoding issue
* Remove unused import
* Various cleanups
* Restore network_cli check in _low_level_execute_command
* type improvements for cliargs_deferred_get and swap shallowcopy to default to False
* minor cleanups
* Allow the su plugin to work, since it doesn't define a prompt the same way
* Fix up ksu become plugin
* Only set prompt if build_become_command was called
* Add helper to assist connection plugins in knowing they need to wait for a prompt
* Fix tests and code expectations
* Doc updates
* Various additional minor cleanups
* Make doas functional
* Don't change connection signature, load become plugin from TaskExecutor
* Remove unused imports
* Add comment about setting the become plugin on the playcontext
* Fix up tests for recent changes
* Support 'Password:' natively for the doas plugin
* Make default prompts raw
* wording cleanups. ci_complete
* Remove unrelated changes
* Address spelling mistake
* Restore removed test, and udpate to use new functionality
* Add changelog fragment
* Don't hard fail in set_attributes_from_cli on missing CLI keys
* Remove unrelated change to loader
* Remove internal deprecated FieldAttributes now
* Emit deprecation warnings now
2019-02-11 18:27:44 +01:00
|
|
|
conn.become.prompt = False
|
2016-03-08 07:07:00 +01:00
|
|
|
pc.success_key = None
|
2016-08-25 19:57:55 +02:00
|
|
|
output, unprocessed = conn._examine_output(u'source', u'state', b'line 1\nline 2\nincorrect password\n', True)
|
|
|
|
self.assertEqual(output, b'line 1\nline 2\nincorrect password\n')
|
|
|
|
self.assertEqual(unprocessed, b'')
|
2016-03-08 07:07:00 +01:00
|
|
|
self.assertFalse(conn._flags['become_prompt'])
|
|
|
|
self.assertFalse(conn._flags['become_success'])
|
|
|
|
self.assertTrue(conn._flags['become_error'])
|
|
|
|
self.assertFalse(conn._flags['become_nopasswd_error'])
|
|
|
|
|
|
|
|
# test examining output for missing password
|
|
|
|
conn._flags = dict(
|
2017-05-30 19:05:19 +02:00
|
|
|
become_prompt=False,
|
|
|
|
become_success=False,
|
|
|
|
become_error=False,
|
|
|
|
become_nopasswd_error=False,
|
2016-03-08 07:07:00 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
pc.prompt = False
|
Become plugins (#50991)
* [WIP] become plugins
Move from hardcoded method to plugins for ease of use, expansion and overrides
- load into connection as it is going to be the main consumer
- play_context will also use to keep backwards compat API
- ensure shell is used to construct commands when needed
- migrate settings remove from base config in favor of plugin specific configs
- cleanup ansible-doc
- add become plugin docs
- remove deprecated sudo/su code and keywords
- adjust become options for cli
- set plugin options from context
- ensure config defs are avaialbe before instance
- refactored getting the shell plugin, fixed tests
- changed into regex as they were string matching, which does not work with random string generation
- explicitly set flags for play context tests
- moved plugin loading up front
- now loads for basedir also
- allow pyc/o for non m modules
- fixes to tests and some plugins
- migrate to play objects fro play_context
- simiplify gathering
- added utf8 headers
- moved option setting
- add fail msg to dzdo
- use tuple for multiple options on fail/missing
- fix relative plugin paths
- shift from play context to play
- all tasks already inherit this from play directly
- remove obsolete 'set play'
- correct environment handling
- add wrap_exe option to pfexec
- fix runas to noop
- fixed setting play context
- added password configs
- removed required false
- remove from doc building till they are ready
future development:
- deal with 'enable' and 'runas' which are not 'command wrappers' but 'state flags' and currently hardcoded in diff subsystems
* cleanup
remove callers to removed func
removed --sudo cli doc refs
remove runas become_exe
ensure keyerorr on plugin
also fix backwards compat, missing method is attributeerror, not ansible error
get remote_user consistently
ignore missing system_tmpdirs on plugin load
correct config precedence
add deprecation
fix networking imports
backwards compat for plugins using BECOME_METHODS
* Port become_plugins to context.CLIARGS
This is a work in progress:
* Stop passing options around everywhere as we can use context.CLIARGS
instead
* Refactor make_become_commands as asked for by alikins
* Typo in comment fix
* Stop loading values from the cli in more than one place
Both play and play_context were saving default values from the cli
arguments directly. This changes things so that the default values are
loaded into the play and then play_context takes them from there.
* Rename BECOME_PLUGIN_PATH to DEFAULT_BECOME_PLUGIN_PATH
As alikins said, all other plugin paths are named
DEFAULT_plugintype_PLUGIN_PATH. If we're going to rename these, that
should be done all at one time rather than piecemeal.
* One to throw away
This is a set of hacks to get setting FieldAttribute defaults to command
line args to work. It's not fully done yet.
After talking it over with sivel and jimi-c this should be done by
fixing FieldAttributeBase and _get_parent_attribute() calls to do the
right thing when there is a non-None default.
What we want to be able to do ideally is something like this:
class Base(FieldAttributeBase):
_check_mode = FieldAttribute([..] default=lambda: context.CLIARGS['check'])
class Play(Base):
# lambda so that we have a chance to parse the command line args
# before we get here. In the future we might be able to restructure
# this so that the cli parsing code runs before these classes are
# defined.
class Task(Base):
pass
And still have a playbook like this function:
---
- hosts:
tasks:
- command: whoami
check_mode: True
(The check_mode test that is added as a separate commit in this PR will
let you test variations on this case).
There's a few separate reasons that the code doesn't let us do this or
a non-ugly workaround for this as written right now. The fix that
jimi-c, sivel, and I talked about may let us do this or it may still
require a workaround (but less ugly) (having one class that has the
FieldAttributes with default values and one class that inherits from
that but just overrides the FieldAttributes which now have defaults)
* Revert "One to throw away"
This reverts commit 23aa883cbed11429ef1be2a2d0ed18f83a3b8064.
* Set FieldAttr defaults directly from CLIARGS
* Remove dead code
* Move timeout directly to PlayContext, it's never needed on Play
* just for backwards compat, add a static version of BECOME_METHODS to constants
* Make the become attr on the connection public, since it's used outside of the connection
* Logic fix
* Nuke connection testing if it supports specific become methods
* Remove unused vars
* Address rebase issues
* Fix path encoding issue
* Remove unused import
* Various cleanups
* Restore network_cli check in _low_level_execute_command
* type improvements for cliargs_deferred_get and swap shallowcopy to default to False
* minor cleanups
* Allow the su plugin to work, since it doesn't define a prompt the same way
* Fix up ksu become plugin
* Only set prompt if build_become_command was called
* Add helper to assist connection plugins in knowing they need to wait for a prompt
* Fix tests and code expectations
* Doc updates
* Various additional minor cleanups
* Make doas functional
* Don't change connection signature, load become plugin from TaskExecutor
* Remove unused imports
* Add comment about setting the become plugin on the playcontext
* Fix up tests for recent changes
* Support 'Password:' natively for the doas plugin
* Make default prompts raw
* wording cleanups. ci_complete
* Remove unrelated changes
* Address spelling mistake
* Restore removed test, and udpate to use new functionality
* Add changelog fragment
* Don't hard fail in set_attributes_from_cli on missing CLI keys
* Remove unrelated change to loader
* Remove internal deprecated FieldAttributes now
* Emit deprecation warnings now
2019-02-11 18:27:44 +01:00
|
|
|
conn.become.prompt = False
|
2016-03-08 07:07:00 +01:00
|
|
|
pc.success_key = None
|
2016-08-25 19:57:55 +02:00
|
|
|
output, unprocessed = conn._examine_output(u'source', u'state', b'line 1\nbad password\n', True)
|
|
|
|
self.assertEqual(output, b'line 1\nbad password\n')
|
|
|
|
self.assertEqual(unprocessed, b'')
|
2016-03-08 07:07:00 +01:00
|
|
|
self.assertFalse(conn._flags['become_prompt'])
|
|
|
|
self.assertFalse(conn._flags['become_success'])
|
|
|
|
self.assertFalse(conn._flags['become_error'])
|
|
|
|
self.assertTrue(conn._flags['become_nopasswd_error'])
|
|
|
|
|
2016-03-08 20:49:07 +01:00
|
|
|
@patch('time.sleep')
|
|
|
|
@patch('os.path.exists')
|
2017-03-02 23:56:29 +01:00
|
|
|
def test_plugins_connection_ssh_put_file(self, mock_ospe, mock_sleep):
|
2016-03-08 20:49:07 +01:00
|
|
|
pc = PlayContext()
|
|
|
|
new_stdin = StringIO()
|
2018-04-24 00:36:35 +02:00
|
|
|
conn = connection_loader.get('ssh', pc, new_stdin)
|
2016-03-08 20:49:07 +01:00
|
|
|
conn._build_command = MagicMock()
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run = MagicMock()
|
2016-03-08 20:49:07 +01:00
|
|
|
|
|
|
|
mock_ospe.return_value = True
|
|
|
|
conn._build_command.return_value = 'some command to run'
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.return_value = (0, '', '')
|
2016-03-08 20:49:07 +01:00
|
|
|
conn.host = "some_host"
|
|
|
|
|
2017-03-02 23:56:29 +01:00
|
|
|
C.ANSIBLE_SSH_RETRIES = 9
|
|
|
|
|
2016-10-21 15:59:56 +02:00
|
|
|
# Test with C.DEFAULT_SCP_IF_SSH set to smart
|
|
|
|
# Test when SFTP works
|
|
|
|
C.DEFAULT_SCP_IF_SSH = 'smart'
|
2016-11-17 22:18:29 +01:00
|
|
|
expected_in_data = b' '.join((b'put', to_bytes(shlex_quote('/path/to/in/file')), to_bytes(shlex_quote('/path/to/dest/file')))) + b'\n'
|
2016-10-21 15:59:56 +02:00
|
|
|
conn.put_file('/path/to/in/file', '/path/to/dest/file')
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.assert_called_with('some command to run', expected_in_data, checkrc=False)
|
2016-10-21 15:59:56 +02:00
|
|
|
|
|
|
|
# Test when SFTP doesn't work but SCP does
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.side_effect = [(1, 'stdout', 'some errors'), (0, '', '')]
|
2016-10-21 15:59:56 +02:00
|
|
|
conn.put_file('/path/to/in/file', '/path/to/dest/file')
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.assert_called_with('some command to run', None, checkrc=False)
|
|
|
|
conn._bare_run.side_effect = None
|
2016-10-21 15:59:56 +02:00
|
|
|
|
2016-03-08 20:49:07 +01:00
|
|
|
# test with C.DEFAULT_SCP_IF_SSH enabled
|
|
|
|
C.DEFAULT_SCP_IF_SSH = True
|
2016-09-07 07:54:17 +02:00
|
|
|
conn.put_file('/path/to/in/file', '/path/to/dest/file')
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.assert_called_with('some command to run', None, checkrc=False)
|
2016-03-08 20:49:07 +01:00
|
|
|
|
2016-09-07 07:54:17 +02:00
|
|
|
conn.put_file(u'/path/to/in/file/with/unicode-fö〩', u'/path/to/dest/file/with/unicode-fö〩')
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.assert_called_with('some command to run', None, checkrc=False)
|
2016-03-08 20:49:07 +01:00
|
|
|
|
|
|
|
# test with C.DEFAULT_SCP_IF_SSH disabled
|
|
|
|
C.DEFAULT_SCP_IF_SSH = False
|
2016-11-17 22:18:29 +01:00
|
|
|
expected_in_data = b' '.join((b'put', to_bytes(shlex_quote('/path/to/in/file')), to_bytes(shlex_quote('/path/to/dest/file')))) + b'\n'
|
2016-09-07 07:54:17 +02:00
|
|
|
conn.put_file('/path/to/in/file', '/path/to/dest/file')
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.assert_called_with('some command to run', expected_in_data, checkrc=False)
|
2016-03-08 20:49:07 +01:00
|
|
|
|
2016-09-07 07:54:17 +02:00
|
|
|
expected_in_data = b' '.join((b'put',
|
2017-05-30 19:05:19 +02:00
|
|
|
to_bytes(shlex_quote('/path/to/in/file/with/unicode-fö〩')),
|
|
|
|
to_bytes(shlex_quote('/path/to/dest/file/with/unicode-fö〩')))) + b'\n'
|
2016-09-07 07:54:17 +02:00
|
|
|
conn.put_file(u'/path/to/in/file/with/unicode-fö〩', u'/path/to/dest/file/with/unicode-fö〩')
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.assert_called_with('some command to run', expected_in_data, checkrc=False)
|
2016-03-08 20:49:07 +01:00
|
|
|
|
|
|
|
# test that a non-zero rc raises an error
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.return_value = (1, 'stdout', 'some errors')
|
2016-03-08 20:49:07 +01:00
|
|
|
self.assertRaises(AnsibleError, conn.put_file, '/path/to/bad/file', '/remote/path/to/file')
|
|
|
|
|
|
|
|
# test that a not-found path raises an error
|
|
|
|
mock_ospe.return_value = False
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.return_value = (0, 'stdout', '')
|
2016-03-08 20:49:07 +01:00
|
|
|
self.assertRaises(AnsibleFileNotFound, conn.put_file, '/path/to/bad/file', '/remote/path/to/file')
|
|
|
|
|
2017-03-02 23:56:29 +01:00
|
|
|
@patch('time.sleep')
|
|
|
|
def test_plugins_connection_ssh_fetch_file(self, mock_sleep):
|
2016-03-08 20:49:07 +01:00
|
|
|
pc = PlayContext()
|
|
|
|
new_stdin = StringIO()
|
2018-04-24 00:36:35 +02:00
|
|
|
conn = connection_loader.get('ssh', pc, new_stdin)
|
2016-03-08 20:49:07 +01:00
|
|
|
conn._build_command = MagicMock()
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run = MagicMock()
|
2018-04-24 00:36:35 +02:00
|
|
|
conn._load_name = 'ssh'
|
2016-03-08 20:49:07 +01:00
|
|
|
|
|
|
|
conn._build_command.return_value = 'some command to run'
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.return_value = (0, '', '')
|
2016-03-08 20:49:07 +01:00
|
|
|
conn.host = "some_host"
|
|
|
|
|
2017-03-02 23:56:29 +01:00
|
|
|
C.ANSIBLE_SSH_RETRIES = 9
|
|
|
|
|
2016-10-21 15:59:56 +02:00
|
|
|
# Test with C.DEFAULT_SCP_IF_SSH set to smart
|
|
|
|
# Test when SFTP works
|
|
|
|
C.DEFAULT_SCP_IF_SSH = 'smart'
|
2016-11-17 22:18:29 +01:00
|
|
|
expected_in_data = b' '.join((b'get', to_bytes(shlex_quote('/path/to/in/file')), to_bytes(shlex_quote('/path/to/dest/file')))) + b'\n'
|
2018-04-24 00:36:35 +02:00
|
|
|
conn.set_options({})
|
2016-10-21 15:59:56 +02:00
|
|
|
conn.fetch_file('/path/to/in/file', '/path/to/dest/file')
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.assert_called_with('some command to run', expected_in_data, checkrc=False)
|
2016-10-21 15:59:56 +02:00
|
|
|
|
|
|
|
# Test when SFTP doesn't work but SCP does
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.side_effect = [(1, 'stdout', 'some errors'), (0, '', '')]
|
2016-10-21 15:59:56 +02:00
|
|
|
conn.fetch_file('/path/to/in/file', '/path/to/dest/file')
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.assert_called_with('some command to run', None, checkrc=False)
|
|
|
|
conn._bare_run.side_effect = None
|
2016-10-21 15:59:56 +02:00
|
|
|
|
2016-03-08 20:49:07 +01:00
|
|
|
# test with C.DEFAULT_SCP_IF_SSH enabled
|
|
|
|
C.DEFAULT_SCP_IF_SSH = True
|
2016-09-07 07:54:17 +02:00
|
|
|
conn.fetch_file('/path/to/in/file', '/path/to/dest/file')
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.assert_called_with('some command to run', None, checkrc=False)
|
2016-03-08 20:49:07 +01:00
|
|
|
|
2016-09-07 07:54:17 +02:00
|
|
|
conn.fetch_file(u'/path/to/in/file/with/unicode-fö〩', u'/path/to/dest/file/with/unicode-fö〩')
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.assert_called_with('some command to run', None, checkrc=False)
|
2016-03-08 20:49:07 +01:00
|
|
|
|
|
|
|
# test with C.DEFAULT_SCP_IF_SSH disabled
|
|
|
|
C.DEFAULT_SCP_IF_SSH = False
|
2016-11-17 22:18:29 +01:00
|
|
|
expected_in_data = b' '.join((b'get', to_bytes(shlex_quote('/path/to/in/file')), to_bytes(shlex_quote('/path/to/dest/file')))) + b'\n'
|
2016-09-07 07:54:17 +02:00
|
|
|
conn.fetch_file('/path/to/in/file', '/path/to/dest/file')
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.assert_called_with('some command to run', expected_in_data, checkrc=False)
|
2016-03-08 20:49:07 +01:00
|
|
|
|
2016-09-07 07:54:17 +02:00
|
|
|
expected_in_data = b' '.join((b'get',
|
2017-05-30 19:05:19 +02:00
|
|
|
to_bytes(shlex_quote('/path/to/in/file/with/unicode-fö〩')),
|
|
|
|
to_bytes(shlex_quote('/path/to/dest/file/with/unicode-fö〩')))) + b'\n'
|
2016-09-07 07:54:17 +02:00
|
|
|
conn.fetch_file(u'/path/to/in/file/with/unicode-fö〩', u'/path/to/dest/file/with/unicode-fö〩')
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.assert_called_with('some command to run', expected_in_data, checkrc=False)
|
2016-03-08 20:49:07 +01:00
|
|
|
|
|
|
|
# test that a non-zero rc raises an error
|
2017-08-13 07:33:01 +02:00
|
|
|
conn._bare_run.return_value = (1, 'stdout', 'some errors')
|
2016-03-08 20:49:07 +01:00
|
|
|
self.assertRaises(AnsibleError, conn.fetch_file, '/path/to/bad/file', '/remote/path/to/file')
|
2017-02-01 17:48:23 +01:00
|
|
|
|
|
|
|
|
|
|
|
class MockSelector(object):
|
|
|
|
def __init__(self):
|
|
|
|
self.files_watched = 0
|
|
|
|
self.register = MagicMock(side_effect=self._register)
|
|
|
|
self.unregister = MagicMock(side_effect=self._unregister)
|
|
|
|
self.close = MagicMock()
|
|
|
|
self.get_map = MagicMock(side_effect=self._get_map)
|
|
|
|
self.select = MagicMock()
|
|
|
|
|
|
|
|
def _register(self, *args, **kwargs):
|
|
|
|
self.files_watched += 1
|
|
|
|
|
|
|
|
def _unregister(self, *args, **kwargs):
|
|
|
|
self.files_watched -= 1
|
|
|
|
|
|
|
|
def _get_map(self, *args, **kwargs):
|
|
|
|
return self.files_watched
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def mock_run_env(request, mocker):
|
|
|
|
pc = PlayContext()
|
|
|
|
new_stdin = StringIO()
|
|
|
|
|
2018-04-24 00:36:35 +02:00
|
|
|
conn = connection_loader.get('ssh', pc, new_stdin)
|
Become plugins (#50991)
* [WIP] become plugins
Move from hardcoded method to plugins for ease of use, expansion and overrides
- load into connection as it is going to be the main consumer
- play_context will also use to keep backwards compat API
- ensure shell is used to construct commands when needed
- migrate settings remove from base config in favor of plugin specific configs
- cleanup ansible-doc
- add become plugin docs
- remove deprecated sudo/su code and keywords
- adjust become options for cli
- set plugin options from context
- ensure config defs are avaialbe before instance
- refactored getting the shell plugin, fixed tests
- changed into regex as they were string matching, which does not work with random string generation
- explicitly set flags for play context tests
- moved plugin loading up front
- now loads for basedir also
- allow pyc/o for non m modules
- fixes to tests and some plugins
- migrate to play objects fro play_context
- simiplify gathering
- added utf8 headers
- moved option setting
- add fail msg to dzdo
- use tuple for multiple options on fail/missing
- fix relative plugin paths
- shift from play context to play
- all tasks already inherit this from play directly
- remove obsolete 'set play'
- correct environment handling
- add wrap_exe option to pfexec
- fix runas to noop
- fixed setting play context
- added password configs
- removed required false
- remove from doc building till they are ready
future development:
- deal with 'enable' and 'runas' which are not 'command wrappers' but 'state flags' and currently hardcoded in diff subsystems
* cleanup
remove callers to removed func
removed --sudo cli doc refs
remove runas become_exe
ensure keyerorr on plugin
also fix backwards compat, missing method is attributeerror, not ansible error
get remote_user consistently
ignore missing system_tmpdirs on plugin load
correct config precedence
add deprecation
fix networking imports
backwards compat for plugins using BECOME_METHODS
* Port become_plugins to context.CLIARGS
This is a work in progress:
* Stop passing options around everywhere as we can use context.CLIARGS
instead
* Refactor make_become_commands as asked for by alikins
* Typo in comment fix
* Stop loading values from the cli in more than one place
Both play and play_context were saving default values from the cli
arguments directly. This changes things so that the default values are
loaded into the play and then play_context takes them from there.
* Rename BECOME_PLUGIN_PATH to DEFAULT_BECOME_PLUGIN_PATH
As alikins said, all other plugin paths are named
DEFAULT_plugintype_PLUGIN_PATH. If we're going to rename these, that
should be done all at one time rather than piecemeal.
* One to throw away
This is a set of hacks to get setting FieldAttribute defaults to command
line args to work. It's not fully done yet.
After talking it over with sivel and jimi-c this should be done by
fixing FieldAttributeBase and _get_parent_attribute() calls to do the
right thing when there is a non-None default.
What we want to be able to do ideally is something like this:
class Base(FieldAttributeBase):
_check_mode = FieldAttribute([..] default=lambda: context.CLIARGS['check'])
class Play(Base):
# lambda so that we have a chance to parse the command line args
# before we get here. In the future we might be able to restructure
# this so that the cli parsing code runs before these classes are
# defined.
class Task(Base):
pass
And still have a playbook like this function:
---
- hosts:
tasks:
- command: whoami
check_mode: True
(The check_mode test that is added as a separate commit in this PR will
let you test variations on this case).
There's a few separate reasons that the code doesn't let us do this or
a non-ugly workaround for this as written right now. The fix that
jimi-c, sivel, and I talked about may let us do this or it may still
require a workaround (but less ugly) (having one class that has the
FieldAttributes with default values and one class that inherits from
that but just overrides the FieldAttributes which now have defaults)
* Revert "One to throw away"
This reverts commit 23aa883cbed11429ef1be2a2d0ed18f83a3b8064.
* Set FieldAttr defaults directly from CLIARGS
* Remove dead code
* Move timeout directly to PlayContext, it's never needed on Play
* just for backwards compat, add a static version of BECOME_METHODS to constants
* Make the become attr on the connection public, since it's used outside of the connection
* Logic fix
* Nuke connection testing if it supports specific become methods
* Remove unused vars
* Address rebase issues
* Fix path encoding issue
* Remove unused import
* Various cleanups
* Restore network_cli check in _low_level_execute_command
* type improvements for cliargs_deferred_get and swap shallowcopy to default to False
* minor cleanups
* Allow the su plugin to work, since it doesn't define a prompt the same way
* Fix up ksu become plugin
* Only set prompt if build_become_command was called
* Add helper to assist connection plugins in knowing they need to wait for a prompt
* Fix tests and code expectations
* Doc updates
* Various additional minor cleanups
* Make doas functional
* Don't change connection signature, load become plugin from TaskExecutor
* Remove unused imports
* Add comment about setting the become plugin on the playcontext
* Fix up tests for recent changes
* Support 'Password:' natively for the doas plugin
* Make default prompts raw
* wording cleanups. ci_complete
* Remove unrelated changes
* Address spelling mistake
* Restore removed test, and udpate to use new functionality
* Add changelog fragment
* Don't hard fail in set_attributes_from_cli on missing CLI keys
* Remove unrelated change to loader
* Remove internal deprecated FieldAttributes now
* Emit deprecation warnings now
2019-02-11 18:27:44 +01:00
|
|
|
conn.set_become_plugin(become_loader.get('sudo'))
|
2017-02-01 17:48:23 +01:00
|
|
|
conn._send_initial_data = MagicMock()
|
|
|
|
conn._examine_output = MagicMock()
|
|
|
|
conn._terminate_process = MagicMock()
|
2018-04-24 00:36:35 +02:00
|
|
|
conn._load_name = 'ssh'
|
2017-02-01 17:48:23 +01:00
|
|
|
conn.sshpass_pipe = [MagicMock(), MagicMock()]
|
|
|
|
|
|
|
|
request.cls.pc = pc
|
|
|
|
request.cls.conn = conn
|
|
|
|
|
|
|
|
mock_popen_res = MagicMock()
|
|
|
|
mock_popen_res.poll = MagicMock()
|
|
|
|
mock_popen_res.wait = MagicMock()
|
|
|
|
mock_popen_res.stdin = MagicMock()
|
|
|
|
mock_popen_res.stdin.fileno.return_value = 1000
|
|
|
|
mock_popen_res.stdout = MagicMock()
|
|
|
|
mock_popen_res.stdout.fileno.return_value = 1001
|
|
|
|
mock_popen_res.stderr = MagicMock()
|
|
|
|
mock_popen_res.stderr.fileno.return_value = 1002
|
|
|
|
mock_popen_res.returncode = 0
|
|
|
|
request.cls.mock_popen_res = mock_popen_res
|
|
|
|
|
|
|
|
mock_popen = mocker.patch('subprocess.Popen', return_value=mock_popen_res)
|
|
|
|
request.cls.mock_popen = mock_popen
|
|
|
|
|
|
|
|
request.cls.mock_selector = MockSelector()
|
|
|
|
mocker.patch('ansible.compat.selectors.DefaultSelector', lambda: request.cls.mock_selector)
|
|
|
|
|
|
|
|
request.cls.mock_openpty = mocker.patch('pty.openpty')
|
|
|
|
|
|
|
|
mocker.patch('fcntl.fcntl')
|
|
|
|
mocker.patch('os.write')
|
|
|
|
mocker.patch('os.close')
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.usefixtures('mock_run_env')
|
|
|
|
class TestSSHConnectionRun(object):
|
|
|
|
# FIXME:
|
|
|
|
# These tests are little more than a smoketest. Need to enhance them
|
|
|
|
# a bit to check that they're calling the relevant functions and making
|
|
|
|
# complete coverage of the code paths
|
|
|
|
def test_no_escalation(self):
|
|
|
|
self.mock_popen_res.stdout.read.side_effect = [b"my_stdout\n", b"second_line"]
|
|
|
|
self.mock_popen_res.stderr.read.side_effect = [b"my_stderr"]
|
|
|
|
self.mock_selector.select.side_effect = [
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[]]
|
|
|
|
self.mock_selector.get_map.side_effect = lambda: True
|
|
|
|
|
|
|
|
return_code, b_stdout, b_stderr = self.conn._run("ssh", "this is input data")
|
|
|
|
assert return_code == 0
|
|
|
|
assert b_stdout == b'my_stdout\nsecond_line'
|
|
|
|
assert b_stderr == b'my_stderr'
|
|
|
|
assert self.mock_selector.register.called is True
|
|
|
|
assert self.mock_selector.register.call_count == 2
|
|
|
|
assert self.conn._send_initial_data.called is True
|
|
|
|
assert self.conn._send_initial_data.call_count == 1
|
|
|
|
assert self.conn._send_initial_data.call_args[0][1] == 'this is input data'
|
|
|
|
|
|
|
|
def test_with_password(self):
|
|
|
|
# test with a password set to trigger the sshpass write
|
|
|
|
self.pc.password = '12345'
|
|
|
|
self.mock_popen_res.stdout.read.side_effect = [b"some data", b"", b""]
|
|
|
|
self.mock_popen_res.stderr.read.side_effect = [b""]
|
|
|
|
self.mock_selector.select.side_effect = [
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[]]
|
|
|
|
self.mock_selector.get_map.side_effect = lambda: True
|
|
|
|
|
|
|
|
return_code, b_stdout, b_stderr = self.conn._run(["ssh", "is", "a", "cmd"], "this is more data")
|
|
|
|
assert return_code == 0
|
|
|
|
assert b_stdout == b'some data'
|
|
|
|
assert b_stderr == b''
|
|
|
|
assert self.mock_selector.register.called is True
|
|
|
|
assert self.mock_selector.register.call_count == 2
|
|
|
|
assert self.conn._send_initial_data.called is True
|
|
|
|
assert self.conn._send_initial_data.call_count == 1
|
|
|
|
assert self.conn._send_initial_data.call_args[0][1] == 'this is more data'
|
|
|
|
|
|
|
|
def _password_with_prompt_examine_output(self, sourice, state, b_chunk, sudoable):
|
|
|
|
if state == 'awaiting_prompt':
|
|
|
|
self.conn._flags['become_prompt'] = True
|
|
|
|
elif state == 'awaiting_escalation':
|
|
|
|
self.conn._flags['become_success'] = True
|
|
|
|
return (b'', b'')
|
|
|
|
|
2016-04-16 00:52:35 +02:00
|
|
|
def test_password_with_prompt(self):
|
2017-02-01 17:48:23 +01:00
|
|
|
# test with password prompting enabled
|
|
|
|
self.pc.password = None
|
Become plugins (#50991)
* [WIP] become plugins
Move from hardcoded method to plugins for ease of use, expansion and overrides
- load into connection as it is going to be the main consumer
- play_context will also use to keep backwards compat API
- ensure shell is used to construct commands when needed
- migrate settings remove from base config in favor of plugin specific configs
- cleanup ansible-doc
- add become plugin docs
- remove deprecated sudo/su code and keywords
- adjust become options for cli
- set plugin options from context
- ensure config defs are avaialbe before instance
- refactored getting the shell plugin, fixed tests
- changed into regex as they were string matching, which does not work with random string generation
- explicitly set flags for play context tests
- moved plugin loading up front
- now loads for basedir also
- allow pyc/o for non m modules
- fixes to tests and some plugins
- migrate to play objects fro play_context
- simiplify gathering
- added utf8 headers
- moved option setting
- add fail msg to dzdo
- use tuple for multiple options on fail/missing
- fix relative plugin paths
- shift from play context to play
- all tasks already inherit this from play directly
- remove obsolete 'set play'
- correct environment handling
- add wrap_exe option to pfexec
- fix runas to noop
- fixed setting play context
- added password configs
- removed required false
- remove from doc building till they are ready
future development:
- deal with 'enable' and 'runas' which are not 'command wrappers' but 'state flags' and currently hardcoded in diff subsystems
* cleanup
remove callers to removed func
removed --sudo cli doc refs
remove runas become_exe
ensure keyerorr on plugin
also fix backwards compat, missing method is attributeerror, not ansible error
get remote_user consistently
ignore missing system_tmpdirs on plugin load
correct config precedence
add deprecation
fix networking imports
backwards compat for plugins using BECOME_METHODS
* Port become_plugins to context.CLIARGS
This is a work in progress:
* Stop passing options around everywhere as we can use context.CLIARGS
instead
* Refactor make_become_commands as asked for by alikins
* Typo in comment fix
* Stop loading values from the cli in more than one place
Both play and play_context were saving default values from the cli
arguments directly. This changes things so that the default values are
loaded into the play and then play_context takes them from there.
* Rename BECOME_PLUGIN_PATH to DEFAULT_BECOME_PLUGIN_PATH
As alikins said, all other plugin paths are named
DEFAULT_plugintype_PLUGIN_PATH. If we're going to rename these, that
should be done all at one time rather than piecemeal.
* One to throw away
This is a set of hacks to get setting FieldAttribute defaults to command
line args to work. It's not fully done yet.
After talking it over with sivel and jimi-c this should be done by
fixing FieldAttributeBase and _get_parent_attribute() calls to do the
right thing when there is a non-None default.
What we want to be able to do ideally is something like this:
class Base(FieldAttributeBase):
_check_mode = FieldAttribute([..] default=lambda: context.CLIARGS['check'])
class Play(Base):
# lambda so that we have a chance to parse the command line args
# before we get here. In the future we might be able to restructure
# this so that the cli parsing code runs before these classes are
# defined.
class Task(Base):
pass
And still have a playbook like this function:
---
- hosts:
tasks:
- command: whoami
check_mode: True
(The check_mode test that is added as a separate commit in this PR will
let you test variations on this case).
There's a few separate reasons that the code doesn't let us do this or
a non-ugly workaround for this as written right now. The fix that
jimi-c, sivel, and I talked about may let us do this or it may still
require a workaround (but less ugly) (having one class that has the
FieldAttributes with default values and one class that inherits from
that but just overrides the FieldAttributes which now have defaults)
* Revert "One to throw away"
This reverts commit 23aa883cbed11429ef1be2a2d0ed18f83a3b8064.
* Set FieldAttr defaults directly from CLIARGS
* Remove dead code
* Move timeout directly to PlayContext, it's never needed on Play
* just for backwards compat, add a static version of BECOME_METHODS to constants
* Make the become attr on the connection public, since it's used outside of the connection
* Logic fix
* Nuke connection testing if it supports specific become methods
* Remove unused vars
* Address rebase issues
* Fix path encoding issue
* Remove unused import
* Various cleanups
* Restore network_cli check in _low_level_execute_command
* type improvements for cliargs_deferred_get and swap shallowcopy to default to False
* minor cleanups
* Allow the su plugin to work, since it doesn't define a prompt the same way
* Fix up ksu become plugin
* Only set prompt if build_become_command was called
* Add helper to assist connection plugins in knowing they need to wait for a prompt
* Fix tests and code expectations
* Doc updates
* Various additional minor cleanups
* Make doas functional
* Don't change connection signature, load become plugin from TaskExecutor
* Remove unused imports
* Add comment about setting the become plugin on the playcontext
* Fix up tests for recent changes
* Support 'Password:' natively for the doas plugin
* Make default prompts raw
* wording cleanups. ci_complete
* Remove unrelated changes
* Address spelling mistake
* Restore removed test, and udpate to use new functionality
* Add changelog fragment
* Don't hard fail in set_attributes_from_cli on missing CLI keys
* Remove unrelated change to loader
* Remove internal deprecated FieldAttributes now
* Emit deprecation warnings now
2019-02-11 18:27:44 +01:00
|
|
|
self.conn.become.prompt = b'Password:'
|
2017-02-01 17:48:23 +01:00
|
|
|
self.conn._examine_output.side_effect = self._password_with_prompt_examine_output
|
|
|
|
self.mock_popen_res.stdout.read.side_effect = [b"Password:", b"Success", b""]
|
|
|
|
self.mock_popen_res.stderr.read.side_effect = [b""]
|
|
|
|
self.mock_selector.select.side_effect = [
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ),
|
|
|
|
(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[]]
|
|
|
|
self.mock_selector.get_map.side_effect = lambda: True
|
|
|
|
|
|
|
|
return_code, b_stdout, b_stderr = self.conn._run("ssh", "this is input data")
|
|
|
|
assert return_code == 0
|
|
|
|
assert b_stdout == b''
|
|
|
|
assert b_stderr == b''
|
|
|
|
assert self.mock_selector.register.called is True
|
|
|
|
assert self.mock_selector.register.call_count == 2
|
|
|
|
assert self.conn._send_initial_data.called is True
|
|
|
|
assert self.conn._send_initial_data.call_count == 1
|
|
|
|
assert self.conn._send_initial_data.call_args[0][1] == 'this is input data'
|
|
|
|
|
2016-04-16 00:52:35 +02:00
|
|
|
def test_password_with_become(self):
|
2017-02-01 17:48:23 +01:00
|
|
|
# test with some become settings
|
|
|
|
self.pc.prompt = b'Password:'
|
Become plugins (#50991)
* [WIP] become plugins
Move from hardcoded method to plugins for ease of use, expansion and overrides
- load into connection as it is going to be the main consumer
- play_context will also use to keep backwards compat API
- ensure shell is used to construct commands when needed
- migrate settings remove from base config in favor of plugin specific configs
- cleanup ansible-doc
- add become plugin docs
- remove deprecated sudo/su code and keywords
- adjust become options for cli
- set plugin options from context
- ensure config defs are avaialbe before instance
- refactored getting the shell plugin, fixed tests
- changed into regex as they were string matching, which does not work with random string generation
- explicitly set flags for play context tests
- moved plugin loading up front
- now loads for basedir also
- allow pyc/o for non m modules
- fixes to tests and some plugins
- migrate to play objects fro play_context
- simiplify gathering
- added utf8 headers
- moved option setting
- add fail msg to dzdo
- use tuple for multiple options on fail/missing
- fix relative plugin paths
- shift from play context to play
- all tasks already inherit this from play directly
- remove obsolete 'set play'
- correct environment handling
- add wrap_exe option to pfexec
- fix runas to noop
- fixed setting play context
- added password configs
- removed required false
- remove from doc building till they are ready
future development:
- deal with 'enable' and 'runas' which are not 'command wrappers' but 'state flags' and currently hardcoded in diff subsystems
* cleanup
remove callers to removed func
removed --sudo cli doc refs
remove runas become_exe
ensure keyerorr on plugin
also fix backwards compat, missing method is attributeerror, not ansible error
get remote_user consistently
ignore missing system_tmpdirs on plugin load
correct config precedence
add deprecation
fix networking imports
backwards compat for plugins using BECOME_METHODS
* Port become_plugins to context.CLIARGS
This is a work in progress:
* Stop passing options around everywhere as we can use context.CLIARGS
instead
* Refactor make_become_commands as asked for by alikins
* Typo in comment fix
* Stop loading values from the cli in more than one place
Both play and play_context were saving default values from the cli
arguments directly. This changes things so that the default values are
loaded into the play and then play_context takes them from there.
* Rename BECOME_PLUGIN_PATH to DEFAULT_BECOME_PLUGIN_PATH
As alikins said, all other plugin paths are named
DEFAULT_plugintype_PLUGIN_PATH. If we're going to rename these, that
should be done all at one time rather than piecemeal.
* One to throw away
This is a set of hacks to get setting FieldAttribute defaults to command
line args to work. It's not fully done yet.
After talking it over with sivel and jimi-c this should be done by
fixing FieldAttributeBase and _get_parent_attribute() calls to do the
right thing when there is a non-None default.
What we want to be able to do ideally is something like this:
class Base(FieldAttributeBase):
_check_mode = FieldAttribute([..] default=lambda: context.CLIARGS['check'])
class Play(Base):
# lambda so that we have a chance to parse the command line args
# before we get here. In the future we might be able to restructure
# this so that the cli parsing code runs before these classes are
# defined.
class Task(Base):
pass
And still have a playbook like this function:
---
- hosts:
tasks:
- command: whoami
check_mode: True
(The check_mode test that is added as a separate commit in this PR will
let you test variations on this case).
There's a few separate reasons that the code doesn't let us do this or
a non-ugly workaround for this as written right now. The fix that
jimi-c, sivel, and I talked about may let us do this or it may still
require a workaround (but less ugly) (having one class that has the
FieldAttributes with default values and one class that inherits from
that but just overrides the FieldAttributes which now have defaults)
* Revert "One to throw away"
This reverts commit 23aa883cbed11429ef1be2a2d0ed18f83a3b8064.
* Set FieldAttr defaults directly from CLIARGS
* Remove dead code
* Move timeout directly to PlayContext, it's never needed on Play
* just for backwards compat, add a static version of BECOME_METHODS to constants
* Make the become attr on the connection public, since it's used outside of the connection
* Logic fix
* Nuke connection testing if it supports specific become methods
* Remove unused vars
* Address rebase issues
* Fix path encoding issue
* Remove unused import
* Various cleanups
* Restore network_cli check in _low_level_execute_command
* type improvements for cliargs_deferred_get and swap shallowcopy to default to False
* minor cleanups
* Allow the su plugin to work, since it doesn't define a prompt the same way
* Fix up ksu become plugin
* Only set prompt if build_become_command was called
* Add helper to assist connection plugins in knowing they need to wait for a prompt
* Fix tests and code expectations
* Doc updates
* Various additional minor cleanups
* Make doas functional
* Don't change connection signature, load become plugin from TaskExecutor
* Remove unused imports
* Add comment about setting the become plugin on the playcontext
* Fix up tests for recent changes
* Support 'Password:' natively for the doas plugin
* Make default prompts raw
* wording cleanups. ci_complete
* Remove unrelated changes
* Address spelling mistake
* Restore removed test, and udpate to use new functionality
* Add changelog fragment
* Don't hard fail in set_attributes_from_cli on missing CLI keys
* Remove unrelated change to loader
* Remove internal deprecated FieldAttributes now
* Emit deprecation warnings now
2019-02-11 18:27:44 +01:00
|
|
|
self.conn.become.prompt = b'Password:'
|
2017-02-01 17:48:23 +01:00
|
|
|
self.pc.become = True
|
|
|
|
self.pc.success_key = 'BECOME-SUCCESS-abcdefg'
|
Become plugins (#50991)
* [WIP] become plugins
Move from hardcoded method to plugins for ease of use, expansion and overrides
- load into connection as it is going to be the main consumer
- play_context will also use to keep backwards compat API
- ensure shell is used to construct commands when needed
- migrate settings remove from base config in favor of plugin specific configs
- cleanup ansible-doc
- add become plugin docs
- remove deprecated sudo/su code and keywords
- adjust become options for cli
- set plugin options from context
- ensure config defs are avaialbe before instance
- refactored getting the shell plugin, fixed tests
- changed into regex as they were string matching, which does not work with random string generation
- explicitly set flags for play context tests
- moved plugin loading up front
- now loads for basedir also
- allow pyc/o for non m modules
- fixes to tests and some plugins
- migrate to play objects fro play_context
- simiplify gathering
- added utf8 headers
- moved option setting
- add fail msg to dzdo
- use tuple for multiple options on fail/missing
- fix relative plugin paths
- shift from play context to play
- all tasks already inherit this from play directly
- remove obsolete 'set play'
- correct environment handling
- add wrap_exe option to pfexec
- fix runas to noop
- fixed setting play context
- added password configs
- removed required false
- remove from doc building till they are ready
future development:
- deal with 'enable' and 'runas' which are not 'command wrappers' but 'state flags' and currently hardcoded in diff subsystems
* cleanup
remove callers to removed func
removed --sudo cli doc refs
remove runas become_exe
ensure keyerorr on plugin
also fix backwards compat, missing method is attributeerror, not ansible error
get remote_user consistently
ignore missing system_tmpdirs on plugin load
correct config precedence
add deprecation
fix networking imports
backwards compat for plugins using BECOME_METHODS
* Port become_plugins to context.CLIARGS
This is a work in progress:
* Stop passing options around everywhere as we can use context.CLIARGS
instead
* Refactor make_become_commands as asked for by alikins
* Typo in comment fix
* Stop loading values from the cli in more than one place
Both play and play_context were saving default values from the cli
arguments directly. This changes things so that the default values are
loaded into the play and then play_context takes them from there.
* Rename BECOME_PLUGIN_PATH to DEFAULT_BECOME_PLUGIN_PATH
As alikins said, all other plugin paths are named
DEFAULT_plugintype_PLUGIN_PATH. If we're going to rename these, that
should be done all at one time rather than piecemeal.
* One to throw away
This is a set of hacks to get setting FieldAttribute defaults to command
line args to work. It's not fully done yet.
After talking it over with sivel and jimi-c this should be done by
fixing FieldAttributeBase and _get_parent_attribute() calls to do the
right thing when there is a non-None default.
What we want to be able to do ideally is something like this:
class Base(FieldAttributeBase):
_check_mode = FieldAttribute([..] default=lambda: context.CLIARGS['check'])
class Play(Base):
# lambda so that we have a chance to parse the command line args
# before we get here. In the future we might be able to restructure
# this so that the cli parsing code runs before these classes are
# defined.
class Task(Base):
pass
And still have a playbook like this function:
---
- hosts:
tasks:
- command: whoami
check_mode: True
(The check_mode test that is added as a separate commit in this PR will
let you test variations on this case).
There's a few separate reasons that the code doesn't let us do this or
a non-ugly workaround for this as written right now. The fix that
jimi-c, sivel, and I talked about may let us do this or it may still
require a workaround (but less ugly) (having one class that has the
FieldAttributes with default values and one class that inherits from
that but just overrides the FieldAttributes which now have defaults)
* Revert "One to throw away"
This reverts commit 23aa883cbed11429ef1be2a2d0ed18f83a3b8064.
* Set FieldAttr defaults directly from CLIARGS
* Remove dead code
* Move timeout directly to PlayContext, it's never needed on Play
* just for backwards compat, add a static version of BECOME_METHODS to constants
* Make the become attr on the connection public, since it's used outside of the connection
* Logic fix
* Nuke connection testing if it supports specific become methods
* Remove unused vars
* Address rebase issues
* Fix path encoding issue
* Remove unused import
* Various cleanups
* Restore network_cli check in _low_level_execute_command
* type improvements for cliargs_deferred_get and swap shallowcopy to default to False
* minor cleanups
* Allow the su plugin to work, since it doesn't define a prompt the same way
* Fix up ksu become plugin
* Only set prompt if build_become_command was called
* Add helper to assist connection plugins in knowing they need to wait for a prompt
* Fix tests and code expectations
* Doc updates
* Various additional minor cleanups
* Make doas functional
* Don't change connection signature, load become plugin from TaskExecutor
* Remove unused imports
* Add comment about setting the become plugin on the playcontext
* Fix up tests for recent changes
* Support 'Password:' natively for the doas plugin
* Make default prompts raw
* wording cleanups. ci_complete
* Remove unrelated changes
* Address spelling mistake
* Restore removed test, and udpate to use new functionality
* Add changelog fragment
* Don't hard fail in set_attributes_from_cli on missing CLI keys
* Remove unrelated change to loader
* Remove internal deprecated FieldAttributes now
* Emit deprecation warnings now
2019-02-11 18:27:44 +01:00
|
|
|
self.conn.become._id = 'abcdefg'
|
2017-02-01 17:48:23 +01:00
|
|
|
self.conn._examine_output.side_effect = self._password_with_prompt_examine_output
|
|
|
|
self.mock_popen_res.stdout.read.side_effect = [b"Password:", b"BECOME-SUCCESS-abcdefg", b"abc"]
|
|
|
|
self.mock_popen_res.stderr.read.side_effect = [b"123"]
|
|
|
|
self.mock_selector.select.side_effect = [
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[]]
|
|
|
|
self.mock_selector.get_map.side_effect = lambda: True
|
|
|
|
|
|
|
|
return_code, b_stdout, b_stderr = self.conn._run("ssh", "this is input data")
|
2018-01-18 16:28:25 +01:00
|
|
|
self.mock_popen_res.stdin.flush.assert_called_once_with()
|
2017-02-01 17:48:23 +01:00
|
|
|
assert return_code == 0
|
|
|
|
assert b_stdout == b'abc'
|
|
|
|
assert b_stderr == b'123'
|
|
|
|
assert self.mock_selector.register.called is True
|
|
|
|
assert self.mock_selector.register.call_count == 2
|
|
|
|
assert self.conn._send_initial_data.called is True
|
|
|
|
assert self.conn._send_initial_data.call_count == 1
|
|
|
|
assert self.conn._send_initial_data.call_args[0][1] == 'this is input data'
|
|
|
|
|
|
|
|
def test_pasword_without_data(self):
|
|
|
|
# simulate no data input but Popen using new pty's fails
|
|
|
|
self.mock_popen.return_value = None
|
|
|
|
self.mock_popen.side_effect = [OSError(), self.mock_popen_res]
|
|
|
|
|
|
|
|
# simulate no data input
|
|
|
|
self.mock_openpty.return_value = (98, 99)
|
|
|
|
self.mock_popen_res.stdout.read.side_effect = [b"some data", b"", b""]
|
|
|
|
self.mock_popen_res.stderr.read.side_effect = [b""]
|
|
|
|
self.mock_selector.select.side_effect = [
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[]]
|
|
|
|
self.mock_selector.get_map.side_effect = lambda: True
|
|
|
|
|
|
|
|
return_code, b_stdout, b_stderr = self.conn._run("ssh", "")
|
|
|
|
assert return_code == 0
|
|
|
|
assert b_stdout == b'some data'
|
|
|
|
assert b_stderr == b''
|
|
|
|
assert self.mock_selector.register.called is True
|
|
|
|
assert self.mock_selector.register.call_count == 2
|
|
|
|
assert self.conn._send_initial_data.called is False
|
2017-03-02 23:56:29 +01:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.usefixtures('mock_run_env')
|
|
|
|
class TestSSHConnectionRetries(object):
|
2019-01-23 17:32:25 +01:00
|
|
|
def test_incorrect_password(self, monkeypatch):
|
|
|
|
monkeypatch.setattr(C, 'HOST_KEY_CHECKING', False)
|
|
|
|
monkeypatch.setattr(C, 'ANSIBLE_SSH_RETRIES', 5)
|
|
|
|
monkeypatch.setattr('time.sleep', lambda x: None)
|
|
|
|
|
|
|
|
self.mock_popen_res.stdout.read.side_effect = [b'']
|
|
|
|
self.mock_popen_res.stderr.read.side_effect = [b'Permission denied, please try again.\r\n']
|
|
|
|
type(self.mock_popen_res).returncode = PropertyMock(side_effect=[5] * 4)
|
|
|
|
|
|
|
|
self.mock_selector.select.side_effect = [
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[],
|
|
|
|
]
|
|
|
|
|
|
|
|
self.mock_selector.get_map.side_effect = lambda: True
|
|
|
|
|
|
|
|
self.conn._build_command = MagicMock()
|
|
|
|
self.conn._build_command.return_value = [b'sshpass', b'-d41', b'ssh', b'-C']
|
|
|
|
self.conn.get_option = MagicMock()
|
|
|
|
self.conn.get_option.return_value = True
|
|
|
|
|
|
|
|
exception_info = pytest.raises(AnsibleAuthenticationFailure, self.conn.exec_command, 'sshpass', 'some data')
|
|
|
|
assert exception_info.value.message == ('Invalid/incorrect username/password. Skipping remaining 5 retries to prevent account lockout: '
|
|
|
|
'Permission denied, please try again.')
|
|
|
|
assert self.mock_popen.call_count == 1
|
|
|
|
|
2017-03-03 05:09:06 +01:00
|
|
|
def test_retry_then_success(self, monkeypatch):
|
|
|
|
monkeypatch.setattr(C, 'HOST_KEY_CHECKING', False)
|
|
|
|
monkeypatch.setattr(C, 'ANSIBLE_SSH_RETRIES', 3)
|
|
|
|
|
|
|
|
monkeypatch.setattr('time.sleep', lambda x: None)
|
|
|
|
|
2017-03-02 23:56:29 +01:00
|
|
|
self.mock_popen_res.stdout.read.side_effect = [b"", b"my_stdout\n", b"second_line"]
|
|
|
|
self.mock_popen_res.stderr.read.side_effect = [b"", b"my_stderr"]
|
|
|
|
type(self.mock_popen_res).returncode = PropertyMock(side_effect=[255] * 3 + [0] * 4)
|
|
|
|
|
|
|
|
self.mock_selector.select.side_effect = [
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[]
|
|
|
|
]
|
|
|
|
self.mock_selector.get_map.side_effect = lambda: True
|
|
|
|
|
|
|
|
self.conn._build_command = MagicMock()
|
|
|
|
self.conn._build_command.return_value = 'ssh'
|
2017-11-22 17:19:43 +01:00
|
|
|
self.conn.get_option = MagicMock()
|
|
|
|
self.conn.get_option.return_value = True
|
2017-03-02 23:56:29 +01:00
|
|
|
|
|
|
|
return_code, b_stdout, b_stderr = self.conn.exec_command('ssh', 'some data')
|
|
|
|
assert return_code == 0
|
|
|
|
assert b_stdout == b'my_stdout\nsecond_line'
|
|
|
|
assert b_stderr == b'my_stderr'
|
|
|
|
|
2017-03-03 05:09:06 +01:00
|
|
|
def test_multiple_failures(self, monkeypatch):
|
|
|
|
monkeypatch.setattr(C, 'HOST_KEY_CHECKING', False)
|
|
|
|
monkeypatch.setattr(C, 'ANSIBLE_SSH_RETRIES', 9)
|
|
|
|
|
|
|
|
monkeypatch.setattr('time.sleep', lambda x: None)
|
2017-03-02 23:56:29 +01:00
|
|
|
|
2016-04-16 00:52:35 +02:00
|
|
|
self.mock_popen_res.stdout.read.side_effect = [b""] * 10
|
|
|
|
self.mock_popen_res.stderr.read.side_effect = [b""] * 10
|
2017-03-02 23:56:29 +01:00
|
|
|
type(self.mock_popen_res).returncode = PropertyMock(side_effect=[255] * 30)
|
|
|
|
|
|
|
|
self.mock_selector.select.side_effect = [
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[],
|
|
|
|
] * 10
|
|
|
|
self.mock_selector.get_map.side_effect = lambda: True
|
|
|
|
|
|
|
|
self.conn._build_command = MagicMock()
|
|
|
|
self.conn._build_command.return_value = 'ssh'
|
2017-11-22 17:19:43 +01:00
|
|
|
self.conn.get_option = MagicMock()
|
|
|
|
self.conn.get_option.return_value = True
|
2017-03-02 23:56:29 +01:00
|
|
|
|
|
|
|
pytest.raises(AnsibleConnectionFailure, self.conn.exec_command, 'ssh', 'some data')
|
|
|
|
assert self.mock_popen.call_count == 10
|
|
|
|
|
2017-03-03 05:09:06 +01:00
|
|
|
def test_abitrary_exceptions(self, monkeypatch):
|
|
|
|
monkeypatch.setattr(C, 'HOST_KEY_CHECKING', False)
|
|
|
|
monkeypatch.setattr(C, 'ANSIBLE_SSH_RETRIES', 9)
|
|
|
|
|
|
|
|
monkeypatch.setattr('time.sleep', lambda x: None)
|
2017-03-02 23:56:29 +01:00
|
|
|
|
|
|
|
self.conn._build_command = MagicMock()
|
|
|
|
self.conn._build_command.return_value = 'ssh'
|
2017-11-22 17:19:43 +01:00
|
|
|
self.conn.get_option = MagicMock()
|
|
|
|
self.conn.get_option.return_value = True
|
2017-03-02 23:56:29 +01:00
|
|
|
|
|
|
|
self.mock_popen.side_effect = [Exception('bad')] * 10
|
|
|
|
pytest.raises(Exception, self.conn.exec_command, 'ssh', 'some data')
|
|
|
|
assert self.mock_popen.call_count == 10
|
|
|
|
|
2017-03-03 05:09:06 +01:00
|
|
|
def test_put_file_retries(self, monkeypatch):
|
|
|
|
monkeypatch.setattr(C, 'HOST_KEY_CHECKING', False)
|
|
|
|
monkeypatch.setattr(C, 'ANSIBLE_SSH_RETRIES', 3)
|
|
|
|
|
|
|
|
monkeypatch.setattr('time.sleep', lambda x: None)
|
|
|
|
monkeypatch.setattr('ansible.plugins.connection.ssh.os.path.exists', lambda x: True)
|
2017-03-02 23:56:29 +01:00
|
|
|
|
|
|
|
self.mock_popen_res.stdout.read.side_effect = [b"", b"my_stdout\n", b"second_line"]
|
|
|
|
self.mock_popen_res.stderr.read.side_effect = [b"", b"my_stderr"]
|
2017-04-07 12:06:29 +02:00
|
|
|
type(self.mock_popen_res).returncode = PropertyMock(side_effect=[255] * 4 + [0] * 4)
|
2017-03-02 23:56:29 +01:00
|
|
|
|
|
|
|
self.mock_selector.select.side_effect = [
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[]
|
|
|
|
]
|
|
|
|
self.mock_selector.get_map.side_effect = lambda: True
|
|
|
|
|
|
|
|
self.conn._build_command = MagicMock()
|
2017-03-03 05:09:06 +01:00
|
|
|
self.conn._build_command.return_value = 'sftp'
|
2017-03-02 23:56:29 +01:00
|
|
|
|
|
|
|
return_code, b_stdout, b_stderr = self.conn.put_file('/path/to/in/file', '/path/to/dest/file')
|
|
|
|
assert return_code == 0
|
|
|
|
assert b_stdout == b"my_stdout\nsecond_line"
|
|
|
|
assert b_stderr == b"my_stderr"
|
|
|
|
assert self.mock_popen.call_count == 2
|
|
|
|
|
2017-03-03 05:09:06 +01:00
|
|
|
def test_fetch_file_retries(self, monkeypatch):
|
|
|
|
monkeypatch.setattr(C, 'HOST_KEY_CHECKING', False)
|
|
|
|
monkeypatch.setattr(C, 'ANSIBLE_SSH_RETRIES', 3)
|
|
|
|
|
|
|
|
monkeypatch.setattr('time.sleep', lambda x: None)
|
|
|
|
monkeypatch.setattr('ansible.plugins.connection.ssh.os.path.exists', lambda x: True)
|
2017-03-02 23:56:29 +01:00
|
|
|
|
|
|
|
self.mock_popen_res.stdout.read.side_effect = [b"", b"my_stdout\n", b"second_line"]
|
|
|
|
self.mock_popen_res.stderr.read.side_effect = [b"", b"my_stderr"]
|
2017-04-07 12:06:29 +02:00
|
|
|
type(self.mock_popen_res).returncode = PropertyMock(side_effect=[255] * 4 + [0] * 4)
|
2017-03-02 23:56:29 +01:00
|
|
|
|
|
|
|
self.mock_selector.select.side_effect = [
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stdout, 1001, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[(SelectorKey(self.mock_popen_res.stderr, 1002, [EVENT_READ], None), EVENT_READ)],
|
|
|
|
[]
|
|
|
|
]
|
|
|
|
self.mock_selector.get_map.side_effect = lambda: True
|
|
|
|
|
|
|
|
self.conn._build_command = MagicMock()
|
2017-03-03 05:09:06 +01:00
|
|
|
self.conn._build_command.return_value = 'sftp'
|
2017-03-02 23:56:29 +01:00
|
|
|
|
|
|
|
return_code, b_stdout, b_stderr = self.conn.fetch_file('/path/to/in/file', '/path/to/dest/file')
|
|
|
|
assert return_code == 0
|
|
|
|
assert b_stdout == b"my_stdout\nsecond_line"
|
|
|
|
assert b_stderr == b"my_stderr"
|
|
|
|
assert self.mock_popen.call_count == 2
|