2016-09-03 11:05:50 +00:00
|
|
|
|
# ~*~ coding: utf-8 ~*~
|
2016-11-24 09:10:43 +00:00
|
|
|
|
from __future__ import unicode_literals, print_function
|
2016-09-03 11:05:50 +00:00
|
|
|
|
|
|
|
|
|
import os
|
|
|
|
|
import json
|
2016-10-28 05:41:11 +00:00
|
|
|
|
import logging
|
2016-10-31 09:41:26 +00:00
|
|
|
|
import traceback
|
2016-10-28 05:41:11 +00:00
|
|
|
|
import ansible.constants as default_config
|
|
|
|
|
|
2016-10-31 09:41:26 +00:00
|
|
|
|
from uuid import uuid4
|
|
|
|
|
from django.utils import timezone
|
2016-09-03 11:05:50 +00:00
|
|
|
|
from ansible.executor.task_queue_manager import TaskQueueManager
|
|
|
|
|
from ansible.inventory import Inventory, Host, Group
|
|
|
|
|
from ansible.vars import VariableManager
|
|
|
|
|
from ansible.parsing.dataloader import DataLoader
|
|
|
|
|
from ansible.executor import playbook_executor
|
|
|
|
|
from ansible.utils.display import Display
|
|
|
|
|
from ansible.playbook.play import Play
|
|
|
|
|
from ansible.plugins.callback import CallbackBase
|
|
|
|
|
|
2016-12-19 06:07:21 +00:00
|
|
|
|
from ops.models import TaskRecord, AnsiblePlay, AnsibleTask, AnsibleHostResult
|
2016-10-31 09:41:26 +00:00
|
|
|
|
|
2016-12-19 15:46:03 +00:00
|
|
|
|
__all__ = ["ADHocRunner", "Config"]
|
|
|
|
|
|
2016-10-28 05:41:11 +00:00
|
|
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
2016-09-03 11:05:50 +00:00
|
|
|
|
|
|
|
|
|
class AnsibleError(StandardError):
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Config(object):
|
2016-11-01 02:37:03 +00:00
|
|
|
|
"""Ansible运行时配置类, 用于初始化Ansible的一些默认配置.
|
2016-09-03 11:05:50 +00:00
|
|
|
|
"""
|
|
|
|
|
def __init__(self, verbosity=None, inventory=None, listhosts=None, subset=None, module_paths=None, extra_vars=None,
|
2016-11-01 02:37:03 +00:00
|
|
|
|
forks=10, ask_vault_pass=False, vault_password_files=None, new_vault_password_file=None,
|
2016-10-27 07:20:16 +00:00
|
|
|
|
output_file=None, tags=None, skip_tags=None, one_line=None, tree=None, ask_sudo_pass=False, ask_su_pass=False,
|
|
|
|
|
sudo=None, sudo_user=None, become=None, become_method=None, become_user=None, become_ask_pass=False,
|
2016-11-01 02:37:03 +00:00
|
|
|
|
ask_pass=False, private_key_file=None, remote_user=None, connection="smart", timeout=10, ssh_common_args=None,
|
2016-10-27 07:20:16 +00:00
|
|
|
|
sftp_extra_args=None, scp_extra_args=None, ssh_extra_args=None, poll_interval=None, seconds=None, check=False,
|
2016-09-03 11:05:50 +00:00
|
|
|
|
syntax=None, diff=None, force_handlers=None, flush_cache=None, listtasks=None, listtags=None, module_path=None):
|
|
|
|
|
self.verbosity = verbosity
|
|
|
|
|
self.inventory = inventory
|
|
|
|
|
self.listhosts = listhosts
|
|
|
|
|
self.subset = subset
|
|
|
|
|
self.module_paths = module_paths
|
|
|
|
|
self.extra_vars = extra_vars
|
|
|
|
|
self.forks = forks
|
|
|
|
|
self.ask_vault_pass = ask_vault_pass
|
|
|
|
|
self.vault_password_files = vault_password_files
|
|
|
|
|
self.new_vault_password_file = new_vault_password_file
|
|
|
|
|
self.output_file = output_file
|
|
|
|
|
self.tags = tags
|
|
|
|
|
self.skip_tags = skip_tags
|
|
|
|
|
self.one_line = one_line
|
|
|
|
|
self.tree = tree
|
|
|
|
|
self.ask_sudo_pass = ask_sudo_pass
|
|
|
|
|
self.ask_su_pass = ask_su_pass
|
|
|
|
|
self.sudo = sudo
|
|
|
|
|
self.sudo_user = sudo_user
|
|
|
|
|
self.become = become
|
|
|
|
|
self.become_method = become_method
|
|
|
|
|
self.become_user = become_user
|
|
|
|
|
self.become_ask_pass = become_ask_pass
|
|
|
|
|
self.ask_pass = ask_pass
|
|
|
|
|
self.private_key_file = private_key_file
|
|
|
|
|
self.remote_user = remote_user
|
|
|
|
|
self.connection = connection
|
|
|
|
|
self.timeout = timeout
|
|
|
|
|
self.ssh_common_args = ssh_common_args
|
|
|
|
|
self.sftp_extra_args = sftp_extra_args
|
|
|
|
|
self.scp_extra_args = scp_extra_args
|
|
|
|
|
self.ssh_extra_args = ssh_extra_args
|
|
|
|
|
self.poll_interval = poll_interval
|
|
|
|
|
self.seconds = seconds
|
|
|
|
|
self.check = check
|
|
|
|
|
self.syntax = syntax
|
|
|
|
|
self.diff = diff
|
|
|
|
|
self.force_handlers = force_handlers
|
|
|
|
|
self.flush_cache = flush_cache
|
|
|
|
|
self.listtasks = listtasks
|
|
|
|
|
self.listtags = listtags
|
|
|
|
|
self.module_path = module_path
|
|
|
|
|
self.__overwrite_default()
|
|
|
|
|
|
|
|
|
|
def __overwrite_default(self):
|
|
|
|
|
"""上面并不能包含Ansible所有的配置, 如果有其他的配置,
|
|
|
|
|
可以通过替换default_config模块里面的变量进行重载,
|
|
|
|
|
比如 default_config.DEFAULT_ASK_PASS = False.
|
|
|
|
|
"""
|
|
|
|
|
default_config.HOST_KEY_CHECKING = False
|
|
|
|
|
|
|
|
|
|
|
2016-10-25 10:15:02 +00:00
|
|
|
|
class InventoryMixin(object):
|
|
|
|
|
"""提供生成Ansible inventory对象的方法
|
2016-09-03 11:05:50 +00:00
|
|
|
|
"""
|
|
|
|
|
|
2016-10-25 10:15:02 +00:00
|
|
|
|
def gen_inventory(self):
|
|
|
|
|
"""用于生成动态构建Ansible Inventory.
|
2016-10-27 07:20:16 +00:00
|
|
|
|
self.hosts: [
|
|
|
|
|
{"host": <ip>,
|
|
|
|
|
"port": <port>,
|
|
|
|
|
"user": <user>,
|
|
|
|
|
"pass": <pass>,
|
|
|
|
|
"key": <sshKey>,
|
|
|
|
|
"group": <default>
|
|
|
|
|
"other_host_var": <other>},
|
|
|
|
|
{...},
|
|
|
|
|
]
|
|
|
|
|
self.group_vars: {
|
|
|
|
|
"groupName1": {"var1": <value>, "var2": <value>, ...},
|
|
|
|
|
"groupName2": {"var1": <value>, "var2": <value>, ...},
|
|
|
|
|
}
|
2016-09-03 11:05:50 +00:00
|
|
|
|
|
2016-10-25 10:15:02 +00:00
|
|
|
|
:return: 返回一个Ansible的inventory对象
|
2016-09-03 11:05:50 +00:00
|
|
|
|
"""
|
2016-10-25 10:15:02 +00:00
|
|
|
|
|
|
|
|
|
# TODO: 验证输入
|
|
|
|
|
|
2016-10-27 07:20:16 +00:00
|
|
|
|
# 创建Ansible Group,如果没有则创建default组
|
2016-10-25 10:15:02 +00:00
|
|
|
|
for asset in self.hosts:
|
2016-09-03 11:05:50 +00:00
|
|
|
|
g_name = asset.get('group', 'default')
|
|
|
|
|
if g_name not in [g.name for g in self.groups]:
|
2016-10-27 07:20:16 +00:00
|
|
|
|
group = Group(name=g_name)
|
2016-09-03 11:05:50 +00:00
|
|
|
|
self.groups.append(group)
|
|
|
|
|
|
2016-10-27 07:20:16 +00:00
|
|
|
|
# 添加组变量到相应的组上
|
2016-10-25 10:15:02 +00:00
|
|
|
|
for group_name, variables in self.group_vars.iteritems():
|
2016-09-03 11:05:50 +00:00
|
|
|
|
for g in self.groups:
|
|
|
|
|
if g.name == group_name:
|
2016-10-27 07:20:16 +00:00
|
|
|
|
for v_name, v_value in variables.iteritems():
|
2016-09-03 11:05:50 +00:00
|
|
|
|
g.set_variable(v_name, v_value)
|
|
|
|
|
|
|
|
|
|
# 往组里面添加Host
|
2016-10-25 10:15:02 +00:00
|
|
|
|
for asset in self.hosts:
|
2016-10-27 07:20:16 +00:00
|
|
|
|
# 添加Host链接的常用变量(host,port,user,pass,key)
|
2016-09-03 11:05:50 +00:00
|
|
|
|
host = Host(name=asset['name'], port=asset['port'])
|
2016-10-27 07:20:16 +00:00
|
|
|
|
host.set_variable('ansible_host', asset['ip'])
|
|
|
|
|
host.set_variable('ansible_port', asset['port'])
|
|
|
|
|
host.set_variable('ansible_user', asset['username'])
|
2016-09-03 11:05:50 +00:00
|
|
|
|
|
2016-10-27 07:20:16 +00:00
|
|
|
|
# 添加密码和秘钥
|
2016-09-03 11:05:50 +00:00
|
|
|
|
if asset.get('password'):
|
|
|
|
|
host.set_variable('ansible_ssh_pass', asset['password'])
|
|
|
|
|
if asset.get('key'):
|
|
|
|
|
host.set_variable('ansible_ssh_private_key_file', asset['key'])
|
|
|
|
|
|
2016-10-27 07:20:16 +00:00
|
|
|
|
# 添加become支持
|
|
|
|
|
become = asset.get("become", None)
|
|
|
|
|
if become is not None:
|
|
|
|
|
host.set_variable("ansible_become", True)
|
|
|
|
|
host.set_variable("ansible_become_method", become.get('method'))
|
|
|
|
|
host.set_variable("ansible_become_user", become.get('user'))
|
|
|
|
|
host.set_variable("ansible_become_pass", become.get('pass'))
|
|
|
|
|
else:
|
|
|
|
|
host.set_variable("ansible_become", False)
|
|
|
|
|
|
|
|
|
|
# 添加其他Host的额外变量
|
2016-09-03 11:05:50 +00:00
|
|
|
|
for key, value in asset.iteritems():
|
|
|
|
|
if key not in ["name", "port", "ip", "username", "password", "key"]:
|
|
|
|
|
host.set_variable(key, value)
|
2016-10-27 07:20:16 +00:00
|
|
|
|
|
|
|
|
|
# 将host添加到组里面
|
2016-09-03 11:05:50 +00:00
|
|
|
|
for g in self.groups:
|
|
|
|
|
if g.name == asset.get('group', 'default'):
|
|
|
|
|
g.add_host(host)
|
|
|
|
|
|
2016-10-27 07:20:16 +00:00
|
|
|
|
# 将组添加到Inventory里面,生成真正的inventory对象
|
2016-10-25 10:15:02 +00:00
|
|
|
|
inventory = Inventory(loader=self.loader, variable_manager=self.variable_manager, host_list=[])
|
2016-09-03 11:05:50 +00:00
|
|
|
|
for g in self.groups:
|
2016-10-25 10:15:02 +00:00
|
|
|
|
inventory.add_group(g)
|
|
|
|
|
self.variable_manager.set_inventory(inventory)
|
|
|
|
|
return inventory
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class CallbackModule(CallbackBase):
|
|
|
|
|
"""处理和分析Ansible运行结果,并保存数据.
|
|
|
|
|
"""
|
|
|
|
|
CALLBACK_VERSION = 2.0
|
|
|
|
|
CALLBACK_TYPE = 'stdout'
|
|
|
|
|
CALLBACK_NAME = 'json'
|
|
|
|
|
|
2016-10-31 09:41:26 +00:00
|
|
|
|
def __init__(self, tasker_id, display=None):
|
2016-10-25 10:15:02 +00:00
|
|
|
|
super(CallbackModule, self).__init__(display)
|
|
|
|
|
self.results = []
|
2016-10-27 07:20:16 +00:00
|
|
|
|
self.output = {}
|
2016-10-31 09:41:26 +00:00
|
|
|
|
self.tasker_id = tasker_id
|
2016-10-25 10:15:02 +00:00
|
|
|
|
|
|
|
|
|
def _new_play(self, play):
|
2016-10-28 05:41:11 +00:00
|
|
|
|
"""将Play保持到数据里面
|
|
|
|
|
"""
|
|
|
|
|
ret = {
|
2016-10-31 09:41:26 +00:00
|
|
|
|
'tasker': self.tasker_id,
|
2016-10-28 05:41:11 +00:00
|
|
|
|
'name': play.name,
|
|
|
|
|
'uuid': str(play._uuid),
|
2016-10-25 10:15:02 +00:00
|
|
|
|
'tasks': []
|
|
|
|
|
}
|
|
|
|
|
|
2016-10-28 05:41:11 +00:00
|
|
|
|
try:
|
2016-12-19 06:07:21 +00:00
|
|
|
|
tasker = TaskRecord.objects.get(uuid=self.tasker_id)
|
2016-10-31 09:41:26 +00:00
|
|
|
|
play = AnsiblePlay(tasker, name=ret['name'], uuid=ret['uuid'])
|
2016-10-28 05:41:11 +00:00
|
|
|
|
play.save()
|
|
|
|
|
except Exception as e:
|
2016-10-31 09:41:26 +00:00
|
|
|
|
traceback.print_exc()
|
2016-10-28 05:41:11 +00:00
|
|
|
|
logger.error("Save ansible play uuid to database error!, %s" % e.message)
|
|
|
|
|
|
|
|
|
|
return ret
|
|
|
|
|
|
2016-10-25 10:15:02 +00:00
|
|
|
|
def _new_task(self, task):
|
2016-10-28 05:41:11 +00:00
|
|
|
|
"""将Task保持到数据库里,需要和Play进行关联
|
|
|
|
|
"""
|
|
|
|
|
ret = {
|
|
|
|
|
'name': task.name,
|
|
|
|
|
'uuid': str(task._uuid),
|
2016-10-27 07:20:16 +00:00
|
|
|
|
'failed': {},
|
|
|
|
|
'unreachable': {},
|
|
|
|
|
'skipped': {},
|
|
|
|
|
'no_hosts': {},
|
|
|
|
|
'success': {}
|
2016-10-25 10:15:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-10-28 05:41:11 +00:00
|
|
|
|
try:
|
|
|
|
|
play = AnsiblePlay.objects.get(uuid=self.__play_uuid)
|
|
|
|
|
task = AnsibleTask(play=play, uuid=ret['uuid'], name=ret['name'])
|
|
|
|
|
task.save()
|
|
|
|
|
except Exception as e:
|
|
|
|
|
logger.error("Save ansible task uuid to database error!, %s" % e.message)
|
|
|
|
|
|
|
|
|
|
return ret
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def __task_uuid(self):
|
|
|
|
|
return self.results[-1]['tasks'][-1]['uuid']
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def __play_uuid(self):
|
|
|
|
|
return self.results[-1]['uuid']
|
|
|
|
|
|
2016-10-28 06:58:09 +00:00
|
|
|
|
def save_task_result(self, result, status):
|
2016-10-28 05:41:11 +00:00
|
|
|
|
try:
|
|
|
|
|
task = AnsibleTask.objects.get(uuid=self.__task_uuid)
|
|
|
|
|
host_result = AnsibleHostResult(task=task, name=result._host)
|
2016-10-28 06:58:09 +00:00
|
|
|
|
if status == "failed":
|
|
|
|
|
host_result.failed = json.dumps(result._result)
|
|
|
|
|
elif status == "unreachable":
|
|
|
|
|
host_result.unreachable = json.dumps(result._result)
|
|
|
|
|
elif status == "skipped":
|
|
|
|
|
host_result.skipped = json.dumps(result._result)
|
|
|
|
|
elif status == "success":
|
|
|
|
|
host_result.success = json.dumps(result._result)
|
|
|
|
|
else:
|
|
|
|
|
logger.error("No such status(failed|unreachable|skipped|success), please check!")
|
2016-10-28 05:41:11 +00:00
|
|
|
|
host_result.save()
|
|
|
|
|
except Exception as e:
|
|
|
|
|
logger.error("Save Ansible host result to database error!, %s" % e.message)
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
|
def save_no_host_result(task):
|
|
|
|
|
try:
|
|
|
|
|
task = AnsibleTask.objects.get(uuid=task._uuid)
|
|
|
|
|
host_result = AnsibleHostResult(task=task, no_host="no host to run this task")
|
|
|
|
|
host_result.save()
|
|
|
|
|
except Exception as e:
|
|
|
|
|
logger.error("Save Ansible host result to database error!, %s" % e.message)
|
|
|
|
|
|
2016-10-27 07:20:16 +00:00
|
|
|
|
def v2_runner_on_failed(self, result, ignore_errors=False):
|
2016-10-28 06:58:09 +00:00
|
|
|
|
self.save_task_result(result, "failed")
|
2016-10-27 07:20:16 +00:00
|
|
|
|
host = result._host
|
|
|
|
|
self.results[-1]['tasks'][-1]['failed'][host.name] = result._result
|
|
|
|
|
|
|
|
|
|
def v2_runner_on_unreachable(self, result):
|
2016-10-28 06:58:09 +00:00
|
|
|
|
self.save_task_result(result, "unreachable")
|
2016-10-27 07:20:16 +00:00
|
|
|
|
host = result._host
|
|
|
|
|
self.results[-1]['tasks'][-1]['unreachable'][host.name] = result._result
|
|
|
|
|
|
|
|
|
|
def v2_runner_on_skipped(self, result):
|
2016-10-28 06:58:09 +00:00
|
|
|
|
self.save_task_result(result, "skipped")
|
2016-10-27 07:20:16 +00:00
|
|
|
|
host = result._host
|
|
|
|
|
self.results[-1]['tasks'][-1]['skipped'][host.name] = result._result
|
|
|
|
|
|
|
|
|
|
def v2_runner_on_no_hosts(self, task):
|
2016-10-28 05:41:11 +00:00
|
|
|
|
self.save_no_host_result(task)
|
|
|
|
|
self.results[-1]['tasks'][-1]['no_hosts']['msg'] = "no host to run this task"
|
2016-10-27 07:20:16 +00:00
|
|
|
|
|
|
|
|
|
def v2_runner_on_ok(self, result):
|
2016-10-28 06:58:09 +00:00
|
|
|
|
self.save_task_result(result, "success")
|
2016-10-27 07:20:16 +00:00
|
|
|
|
host = result._host
|
|
|
|
|
self.results[-1]['tasks'][-1]['success'][host.name] = result._result
|
|
|
|
|
|
2016-10-25 10:15:02 +00:00
|
|
|
|
def v2_playbook_on_play_start(self, play):
|
|
|
|
|
self.results.append(self._new_play(play))
|
|
|
|
|
|
|
|
|
|
def v2_playbook_on_task_start(self, task, is_conditional):
|
|
|
|
|
self.results[-1]['tasks'].append(self._new_task(task))
|
|
|
|
|
|
|
|
|
|
def v2_playbook_on_stats(self, stats):
|
2016-10-28 05:41:11 +00:00
|
|
|
|
"""AdHoc模式下这个钩子不会执行
|
|
|
|
|
"""
|
2016-10-25 10:15:02 +00:00
|
|
|
|
hosts = sorted(stats.processed.keys())
|
|
|
|
|
|
|
|
|
|
summary = {}
|
|
|
|
|
for h in hosts:
|
|
|
|
|
s = stats.summarize(h)
|
|
|
|
|
summary[h] = s
|
|
|
|
|
|
2016-10-27 07:20:16 +00:00
|
|
|
|
self.output['plays'] = self.results
|
|
|
|
|
self.output['stats'] = summary
|
2016-11-24 09:10:43 +00:00
|
|
|
|
print("summary: %s", summary)
|
2016-09-03 11:05:50 +00:00
|
|
|
|
|
2016-10-25 10:15:02 +00:00
|
|
|
|
|
|
|
|
|
class PlayBookRunner(InventoryMixin):
|
|
|
|
|
"""用于执行AnsiblePlaybook的接口.简化Playbook对象的使用.
|
2016-09-03 11:05:50 +00:00
|
|
|
|
"""
|
|
|
|
|
|
2016-10-25 10:15:02 +00:00
|
|
|
|
def __init__(self, config, palybook_path, playbook_var, become_pass, *hosts, **group_vars):
|
2016-09-03 11:05:50 +00:00
|
|
|
|
"""
|
2016-10-25 10:15:02 +00:00
|
|
|
|
|
2016-09-03 11:05:50 +00:00
|
|
|
|
:param config: Config实例
|
|
|
|
|
:param palybook_path: playbook的路径
|
|
|
|
|
:param playbook_var: 执行Playbook时的变量
|
|
|
|
|
:param become_pass: sudo passsword
|
2016-10-25 10:15:02 +00:00
|
|
|
|
:param hosts: 可变位置参数, 为一个资产列表, 每一个资产用dict表示, 以下是这个dict必须包含的key
|
|
|
|
|
[{
|
|
|
|
|
"name": "asset_name",
|
|
|
|
|
"ip": "asset_ip",
|
|
|
|
|
"port": "asset_port",
|
|
|
|
|
"username": "asset_user",
|
|
|
|
|
"password": "asset_pass",
|
|
|
|
|
"key": "asset_private_key",
|
|
|
|
|
"group": "asset_group_name",
|
|
|
|
|
...
|
|
|
|
|
}]
|
|
|
|
|
:param group_vars: 可变关键字参数, 是资产组变量, 记录对应的资产组变量
|
|
|
|
|
"groupName1": {"group_variable1": "value1",...}
|
|
|
|
|
"groupName2": {"group_variable1": "value1",...}
|
2016-09-03 11:05:50 +00:00
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
self.options = config
|
|
|
|
|
|
|
|
|
|
# 设置verbosity级别, 及命令行的--verbose选项
|
|
|
|
|
self.display = Display()
|
|
|
|
|
self.display.verbosity = self.options.verbosity
|
|
|
|
|
playbook_executor.verbosity = self.options.verbosity
|
|
|
|
|
|
|
|
|
|
# sudo成其他用户的配置
|
|
|
|
|
self.options.become = True
|
|
|
|
|
self.options.become_method = 'sudo'
|
|
|
|
|
self.options.become_user = 'root'
|
|
|
|
|
passwords = {'become_pass': become_pass}
|
|
|
|
|
|
|
|
|
|
# 传入playbook的路径,以及执行需要的变量
|
|
|
|
|
pb_dir = os.path.dirname(__file__)
|
|
|
|
|
playbook = "%s/%s" % (pb_dir, palybook_path)
|
|
|
|
|
|
2016-10-25 10:15:02 +00:00
|
|
|
|
# 生成Ansible inventory, 这些变量Mixin都会用到
|
|
|
|
|
self.hosts = hosts
|
|
|
|
|
self.group_vars = group_vars
|
|
|
|
|
self.loader = DataLoader()
|
|
|
|
|
self.variable_manager = VariableManager()
|
|
|
|
|
self.groups = []
|
|
|
|
|
self.variable_manager.extra_vars = playbook_var
|
|
|
|
|
self.inventory = self.gen_inventory()
|
|
|
|
|
|
2016-09-03 11:05:50 +00:00
|
|
|
|
# 初始化playbook的executor
|
|
|
|
|
self.pbex = playbook_executor.PlaybookExecutor(
|
|
|
|
|
playbooks=[playbook],
|
2016-10-25 10:15:02 +00:00
|
|
|
|
inventory=self.inventory,
|
|
|
|
|
variable_manager=self.variable_manager,
|
|
|
|
|
loader=self.loader,
|
2016-09-03 11:05:50 +00:00
|
|
|
|
options=self.options,
|
|
|
|
|
passwords=passwords)
|
|
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
|
"""执行Playbook, 记录执行日志, 处理执行结果.
|
|
|
|
|
:return: <AnsibleResult>对象
|
|
|
|
|
"""
|
|
|
|
|
self.pbex.run()
|
|
|
|
|
stats = self.pbex._tqm._stats
|
|
|
|
|
|
|
|
|
|
# 测试执行是否成功
|
|
|
|
|
run_success = True
|
|
|
|
|
hosts = sorted(stats.processed.keys())
|
|
|
|
|
for h in hosts:
|
|
|
|
|
t = stats.summarize(h)
|
|
|
|
|
if t['unreachable'] > 0 or t['failures'] > 0:
|
|
|
|
|
run_success = False
|
|
|
|
|
|
|
|
|
|
# TODO: 记录执行日志, 处理执行结果.
|
|
|
|
|
|
|
|
|
|
return stats
|
|
|
|
|
|
|
|
|
|
|
2016-10-25 10:15:02 +00:00
|
|
|
|
class ADHocRunner(InventoryMixin):
|
2016-09-03 11:05:50 +00:00
|
|
|
|
"""ADHoc接口
|
|
|
|
|
"""
|
2016-12-19 15:46:03 +00:00
|
|
|
|
def __init__(self, play_data, config=None, *hosts, **group_vars):
|
2016-09-03 11:05:50 +00:00
|
|
|
|
"""
|
2016-10-25 10:15:02 +00:00
|
|
|
|
:param hosts: 见PlaybookRunner参数
|
|
|
|
|
:param group_vars: 见PlaybookRunner参数
|
2016-09-03 11:05:50 +00:00
|
|
|
|
:param config: Config实例
|
2016-10-25 10:15:02 +00:00
|
|
|
|
|
2016-09-03 11:05:50 +00:00
|
|
|
|
:param play_data:
|
|
|
|
|
play_data = dict(
|
|
|
|
|
name="Ansible Ad-Hoc",
|
|
|
|
|
hosts=pattern,
|
|
|
|
|
gather_facts=True,
|
|
|
|
|
tasks=[dict(action=dict(module='service', args={'name': 'vsftpd', 'state': 'restarted'}), async=async, poll=poll)]
|
|
|
|
|
)
|
|
|
|
|
"""
|
2016-12-19 15:46:03 +00:00
|
|
|
|
self.options = config if config != None else Config()
|
2016-09-03 11:05:50 +00:00
|
|
|
|
|
|
|
|
|
# 设置verbosity级别, 及命令行的--verbose选项
|
|
|
|
|
self.display = Display()
|
|
|
|
|
self.display.verbosity = self.options.verbosity
|
|
|
|
|
|
2016-10-27 07:20:16 +00:00
|
|
|
|
# sudo的配置移到了Host级别去了,因此这里不再需要处理
|
|
|
|
|
self.passwords = None
|
2016-09-03 11:05:50 +00:00
|
|
|
|
|
2016-10-25 10:15:02 +00:00
|
|
|
|
# 生成Ansible inventory, 这些变量Mixin都会用到
|
|
|
|
|
self.hosts = hosts
|
|
|
|
|
self.group_vars = group_vars
|
|
|
|
|
self.loader = DataLoader()
|
|
|
|
|
self.variable_manager = VariableManager()
|
|
|
|
|
self.groups = []
|
|
|
|
|
self.inventory = self.gen_inventory()
|
2016-09-03 11:05:50 +00:00
|
|
|
|
|
2016-10-25 10:15:02 +00:00
|
|
|
|
self.play = Play().load(play_data, variable_manager=self.variable_manager, loader=self.loader)
|
2016-09-03 11:05:50 +00:00
|
|
|
|
|
2016-10-28 05:41:11 +00:00
|
|
|
|
@staticmethod
|
2016-10-31 09:41:26 +00:00
|
|
|
|
def update_db_tasker(tasker_id, ext_code):
|
2016-10-28 05:41:11 +00:00
|
|
|
|
try:
|
2016-12-19 06:07:21 +00:00
|
|
|
|
tasker = TaskRecord.objects.get(uuid=tasker_id)
|
2016-10-31 09:41:26 +00:00
|
|
|
|
tasker.end = timezone.now()
|
|
|
|
|
tasker.completed = True
|
|
|
|
|
tasker.exit_code = ext_code
|
|
|
|
|
tasker.save()
|
2016-10-28 05:41:11 +00:00
|
|
|
|
except Exception as e:
|
2016-10-31 09:41:26 +00:00
|
|
|
|
logger.error("Update Tasker Status into database error!, %s" % e.message)
|
2016-10-28 05:41:11 +00:00
|
|
|
|
|
2016-10-31 09:41:26 +00:00
|
|
|
|
def create_db_tasker(self, name, uuid):
|
|
|
|
|
try:
|
|
|
|
|
hosts = [host.get('name') for host in self.hosts]
|
2016-12-19 06:07:21 +00:00
|
|
|
|
tasker = TaskRecord(name=name, uuid=uuid, hosts=','.join(hosts), start=timezone.now())
|
2016-10-31 09:41:26 +00:00
|
|
|
|
tasker.save()
|
|
|
|
|
except Exception as e:
|
|
|
|
|
logger.error("Save Tasker to database error!, %s" % e.message)
|
|
|
|
|
|
|
|
|
|
def run(self, tasker_name, tasker_uuid):
|
2016-10-28 06:58:09 +00:00
|
|
|
|
"""执行ADHoc, 执行完后, 修改AnsiblePlay的状态为完成状态.
|
2016-10-31 09:41:26 +00:00
|
|
|
|
|
|
|
|
|
:param tasker_uuid <str> 用于标示此次task
|
2016-09-03 11:05:50 +00:00
|
|
|
|
"""
|
2016-10-31 09:41:26 +00:00
|
|
|
|
# 初始化callback插件,以及Tasker
|
|
|
|
|
|
|
|
|
|
self.create_db_tasker(tasker_name, tasker_uuid)
|
|
|
|
|
self.results_callback = CallbackModule(tasker_uuid)
|
|
|
|
|
|
2016-09-03 11:05:50 +00:00
|
|
|
|
tqm = None
|
|
|
|
|
# TODO:日志和结果分析
|
|
|
|
|
try:
|
|
|
|
|
tqm = TaskQueueManager(
|
2016-10-25 10:15:02 +00:00
|
|
|
|
inventory=self.inventory,
|
|
|
|
|
variable_manager=self.variable_manager,
|
|
|
|
|
loader=self.loader,
|
|
|
|
|
stdout_callback=self.results_callback,
|
2016-09-03 11:05:50 +00:00
|
|
|
|
options=self.options,
|
|
|
|
|
passwords=self.passwords
|
|
|
|
|
)
|
2016-10-25 10:15:02 +00:00
|
|
|
|
ext_code = tqm.run(self.play)
|
2016-10-28 06:58:09 +00:00
|
|
|
|
result = self.results_callback.results
|
2016-10-28 05:41:11 +00:00
|
|
|
|
|
2016-10-31 09:41:26 +00:00
|
|
|
|
# 任务运行结束, 标示任务完成
|
|
|
|
|
self.update_db_tasker(tasker_uuid, ext_code)
|
2016-10-28 05:41:11 +00:00
|
|
|
|
|
2016-10-28 06:58:09 +00:00
|
|
|
|
ret = json.dumps(result)
|
|
|
|
|
return ext_code, ret
|
2016-10-28 05:41:11 +00:00
|
|
|
|
|
2016-09-03 11:05:50 +00:00
|
|
|
|
finally:
|
|
|
|
|
if tqm:
|
|
|
|
|
tqm.cleanup()
|
|
|
|
|
|
|
|
|
|
|
2016-10-28 05:41:11 +00:00
|
|
|
|
def test_run():
|
2016-09-03 11:05:50 +00:00
|
|
|
|
conf = Config()
|
2016-10-27 07:20:16 +00:00
|
|
|
|
assets = [
|
|
|
|
|
{
|
|
|
|
|
"name": "192.168.1.119",
|
|
|
|
|
"ip": "192.168.1.119",
|
|
|
|
|
"port": "22",
|
|
|
|
|
"username": "root",
|
2016-10-31 09:41:26 +00:00
|
|
|
|
"password": "tongfang_test",
|
2016-10-27 07:20:16 +00:00
|
|
|
|
"key": "asset_private_key",
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"name": "192.168.232.135",
|
|
|
|
|
"ip": "192.168.232.135",
|
2016-09-03 11:05:50 +00:00
|
|
|
|
"port": "22",
|
|
|
|
|
"username": "yumaojun",
|
2016-10-27 07:20:16 +00:00
|
|
|
|
"password": "xxx",
|
2016-09-03 11:05:50 +00:00
|
|
|
|
"key": "asset_private_key",
|
2016-10-28 09:43:56 +00:00
|
|
|
|
"become": {"method": "sudo", "user": "root", "pass": "xxx"}
|
2016-10-27 07:20:16 +00:00
|
|
|
|
},
|
|
|
|
|
]
|
2016-10-25 10:15:02 +00:00
|
|
|
|
# 初始化Play
|
|
|
|
|
play_source = {
|
|
|
|
|
"name": "Ansible Play",
|
2016-10-27 07:20:16 +00:00
|
|
|
|
"hosts": "default",
|
2016-10-25 10:15:02 +00:00
|
|
|
|
"gather_facts": "no",
|
|
|
|
|
"tasks": [
|
2016-10-31 09:41:26 +00:00
|
|
|
|
dict(action=dict(module='ping')),
|
2016-10-25 10:15:02 +00:00
|
|
|
|
]
|
|
|
|
|
}
|
2016-10-27 07:20:16 +00:00
|
|
|
|
hoc = ADHocRunner(conf, play_source, *assets)
|
2016-10-31 09:41:26 +00:00
|
|
|
|
uuid = "tasker-" + uuid4().hex
|
|
|
|
|
ext_code, result = hoc.run("test_task", uuid)
|
2016-11-24 09:10:43 +00:00
|
|
|
|
print(ext_code)
|
|
|
|
|
print(result)
|
2016-10-28 05:41:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
|
test_run()
|