jumpserver/apps/ops/ansible/runner.py

242 lines
7.8 KiB
Python
Raw Normal View History

2017-03-05 12:53:24 +00:00
# ~*~ coding: utf-8 ~*~
2017-03-05 15:30:14 +00:00
import os
import shutil
2017-12-06 10:31:51 +00:00
from collections import namedtuple
2017-03-05 15:30:14 +00:00
from ansible import context
from ansible.module_utils.common.collections import ImmutableDict
2017-03-05 15:30:14 +00:00
from ansible.executor.task_queue_manager import TaskQueueManager
2017-12-06 10:31:51 +00:00
from ansible.vars.manager import VariableManager
2017-03-05 15:30:14 +00:00
from ansible.parsing.dataloader import DataLoader
from ansible.executor.playbook_executor import PlaybookExecutor
from ansible.playbook.play import Play
import ansible.constants as C
from .callback import (
AdHocResultCallback, PlaybookResultCallBack, CommandResultCallback
)
2017-03-05 15:30:14 +00:00
from common.utils import get_logger
2017-12-06 10:31:51 +00:00
from .exceptions import AnsibleError
2017-03-05 15:30:14 +00:00
__all__ = ["AdHocRunner", "PlayBookRunner", "CommandRunner"]
2017-03-05 15:30:14 +00:00
C.HOST_KEY_CHECKING = False
logger = get_logger(__name__)
2017-12-06 10:31:51 +00:00
Options = namedtuple('Options', [
'listtags', 'listtasks', 'listhosts', 'syntax', 'connection',
'module_path', 'forks', 'remote_user', 'private_key_file', 'timeout',
'ssh_common_args', 'ssh_extra_args', 'sftp_extra_args',
'scp_extra_args', 'become', 'become_method', 'become_user',
'verbosity', 'check', 'extra_vars', 'playbook_path', 'passwords',
2017-12-28 04:03:02 +00:00
'diff', 'gathering', 'remote_tmp',
2017-12-06 10:31:51 +00:00
])
def get_default_options():
options = dict(
2017-12-06 10:31:51 +00:00
syntax=False,
timeout=30,
2017-12-06 10:31:51 +00:00
connection='ssh',
forks=10,
remote_user='root',
private_key_file=None,
become=None,
become_method=None,
become_user=None,
verbosity=1,
2017-12-06 10:31:51 +00:00
check=False,
diff=False,
2017-12-07 08:25:50 +00:00
gathering='implicit',
2017-12-28 04:03:02 +00:00
remote_tmp='/tmp/.ansible'
2017-12-06 10:31:51 +00:00
)
return options
2017-03-13 16:58:25 +00:00
# Jumpserver not use playbook
2017-12-06 10:31:51 +00:00
class PlayBookRunner:
2017-03-05 15:30:14 +00:00
"""
用于执行AnsiblePlaybook的接口.简化Playbook对象的使用.
"""
2017-12-06 10:31:51 +00:00
# Default results callback
results_callback_class = PlaybookResultCallBack
loader_class = DataLoader
variable_manager_class = VariableManager
options = get_default_options()
2017-12-10 16:29:25 +00:00
def __init__(self, inventory=None, options=None):
2017-12-06 10:31:51 +00:00
"""
:param options: Ansible options like ansible.cfg
2017-12-10 16:29:25 +00:00
:param inventory: Ansible inventory
2017-12-06 10:31:51 +00:00
"""
if options:
self.options = options
2017-03-05 15:30:14 +00:00
C.RETRY_FILES_ENABLED = False
2017-12-10 16:29:25 +00:00
self.inventory = inventory
2017-12-06 10:31:51 +00:00
self.loader = self.loader_class()
self.results_callback = self.results_callback_class()
self.playbook_path = options.playbook_path
self.variable_manager = self.variable_manager_class(
loader=self.loader, inventory=self.inventory
2017-03-05 15:30:14 +00:00
)
2017-12-06 10:31:51 +00:00
self.passwords = options.passwords
self.__check()
2017-03-05 15:30:14 +00:00
2017-12-06 10:31:51 +00:00
def __check(self):
if self.options.playbook_path is None or \
not os.path.exists(self.options.playbook_path):
raise AnsibleError(
"Not Found the playbook file: {}.".format(self.options.playbook_path)
)
if not self.inventory.list_hosts('all'):
raise AnsibleError('Inventory is empty')
2017-03-05 15:30:14 +00:00
2017-12-06 10:31:51 +00:00
def run(self):
executor = PlaybookExecutor(
2017-03-05 15:30:14 +00:00
playbooks=[self.playbook_path],
inventory=self.inventory,
variable_manager=self.variable_manager,
loader=self.loader,
passwords={"conn_pass": self.passwords}
2017-12-06 10:31:51 +00:00
)
context.CLIARGS = ImmutableDict(self.options)
2017-03-05 15:30:14 +00:00
2017-12-06 10:31:51 +00:00
if executor._tqm:
executor._tqm._stdout_callback = self.results_callback
executor.run()
executor._tqm.cleanup()
return self.results_callback.output
2017-03-05 15:30:14 +00:00
2017-12-06 10:31:51 +00:00
class AdHocRunner:
2017-03-05 15:30:14 +00:00
"""
2017-12-06 10:31:51 +00:00
ADHoc Runner接口
2017-03-05 15:30:14 +00:00
"""
2017-03-13 16:58:25 +00:00
results_callback_class = AdHocResultCallback
2018-03-30 14:03:43 +00:00
results_callback = None
2017-12-06 10:31:51 +00:00
loader_class = DataLoader
variable_manager_class = VariableManager
default_options = get_default_options()
2018-12-10 11:51:35 +00:00
command_modules_choices = ('shell', 'raw', 'command', 'script', 'win_shell')
2017-03-13 16:58:25 +00:00
2017-12-10 16:29:25 +00:00
def __init__(self, inventory, options=None):
2018-03-30 14:03:43 +00:00
self.options = self.update_options(options)
2017-12-10 16:29:25 +00:00
self.inventory = inventory
2017-03-05 15:30:14 +00:00
self.loader = DataLoader()
2017-12-10 16:29:25 +00:00
self.variable_manager = VariableManager(
loader=self.loader, inventory=self.inventory
)
2017-03-06 15:34:54 +00:00
2018-03-30 14:03:43 +00:00
def get_result_callback(self, file_obj=None):
return self.__class__.results_callback_class()
2018-03-30 14:03:43 +00:00
2017-03-06 15:34:54 +00:00
@staticmethod
def check_module_args(module_name, module_args=''):
if module_name in C.MODULE_REQUIRE_ARGS and not module_args:
err = "No argument passed to '%s' module." % module_name
2017-12-06 10:31:51 +00:00
raise AnsibleError(err)
def check_pattern(self, pattern):
2017-12-10 16:29:25 +00:00
if not pattern:
raise AnsibleError("Pattern `{}` is not valid!".format(pattern))
2017-12-06 10:31:51 +00:00
if not self.inventory.list_hosts("all"):
raise AnsibleError("Inventory is empty.")
if not self.inventory.list_hosts(pattern):
raise AnsibleError(
"pattern: %s dose not match any hosts." % pattern
)
2017-03-06 15:34:54 +00:00
2018-12-10 11:51:35 +00:00
def clean_args(self, module, args):
if not args:
return ''
2018-12-10 11:51:35 +00:00
if module not in self.command_modules_choices:
return args
if isinstance(args, str):
if args.startswith('executable='):
_args = args.split(' ')
executable, command = _args[0].split('=')[1], ' '.join(_args[1:])
args = {'executable': executable, '_raw_params': command}
else:
args = {'_raw_params': args}
return args
else:
return args
2017-12-10 16:29:25 +00:00
def clean_tasks(self, tasks):
cleaned_tasks = []
for task in tasks:
2018-12-10 11:51:35 +00:00
module = task['action']['module']
args = task['action'].get('args')
cleaned_args = self.clean_args(module, args)
task['action']['args'] = cleaned_args
self.check_module_args(module, cleaned_args)
2017-12-10 16:29:25 +00:00
cleaned_tasks.append(task)
return cleaned_tasks
2018-03-30 14:03:43 +00:00
def update_options(self, options):
_options = {k: v for k, v in self.default_options.items()}
2018-03-30 14:03:43 +00:00
if options and isinstance(options, dict):
_options.update(options)
return _options
2017-12-07 05:01:33 +00:00
def run(self, tasks, pattern, play_name='Ansible Ad-hoc', gather_facts='no'):
2017-03-06 15:34:54 +00:00
"""
2017-12-06 10:31:51 +00:00
:param tasks: [{'action': {'module': 'shell', 'args': 'ls'}, ...}, ]
:param pattern: all, *, or others
:param play_name: The play name
2018-03-30 14:03:43 +00:00
:param gather_facts:
2017-03-06 15:34:54 +00:00
:return:
"""
2017-12-10 16:29:25 +00:00
self.check_pattern(pattern)
self.results_callback = self.get_result_callback()
2017-12-10 16:29:25 +00:00
cleaned_tasks = self.clean_tasks(tasks)
context.CLIARGS = ImmutableDict(self.options)
2017-12-06 10:31:51 +00:00
play_source = dict(
name=play_name,
2017-03-06 15:34:54 +00:00
hosts=pattern,
2017-12-06 10:31:51 +00:00
gather_facts=gather_facts,
2017-12-10 16:29:25 +00:00
tasks=cleaned_tasks
2017-03-05 15:30:14 +00:00
)
2017-12-06 10:31:51 +00:00
play = Play().load(
play_source,
2017-03-05 15:30:14 +00:00
variable_manager=self.variable_manager,
loader=self.loader,
)
2017-12-06 10:31:51 +00:00
tqm = TaskQueueManager(
2017-03-05 15:30:14 +00:00
inventory=self.inventory,
variable_manager=self.variable_manager,
loader=self.loader,
2018-03-30 14:03:43 +00:00
stdout_callback=self.results_callback,
passwords={"conn_pass": self.options.get("password", "")}
2017-03-05 15:30:14 +00:00
)
try:
2017-12-06 10:31:51 +00:00
tqm.run(play)
2018-03-30 14:03:43 +00:00
return self.results_callback
2017-03-05 15:30:14 +00:00
except Exception as e:
2017-12-06 10:31:51 +00:00
raise AnsibleError(e)
2017-03-05 15:30:14 +00:00
finally:
if tqm is not None:
tqm.cleanup()
shutil.rmtree(C.DEFAULT_LOCAL_TMP, True)
2017-03-05 15:30:14 +00:00
2017-12-06 10:31:51 +00:00
class CommandRunner(AdHocRunner):
results_callback_class = CommandResultCallback
modules_choices = ('shell', 'raw', 'command', 'script')
def execute(self, cmd, pattern, module='shell'):
2017-12-06 10:31:51 +00:00
if module and module not in self.modules_choices:
raise AnsibleError("Module should in {}".format(self.modules_choices))
tasks = [
{"action": {"module": module, "args": cmd}}
]
return self.run(tasks, pattern, play_name=cmd)
2017-03-05 15:30:14 +00:00