jumpserver/apps/ops/ansible/inventory.py

168 lines
5.2 KiB
Python
Raw Normal View History

2017-03-05 12:53:24 +00:00
# ~*~ coding: utf-8 ~*~
2017-12-06 10:31:51 +00:00
from ansible.inventory.host import Host
from ansible.vars.manager import VariableManager
from ansible.inventory.manager import InventoryManager
2017-03-05 12:53:24 +00:00
from ansible.parsing.dataloader import DataLoader
2017-12-10 16:29:25 +00:00
__all__ = [
'BaseHost', 'BaseInventory'
]
class BaseHost(Host):
2017-12-06 10:31:51 +00:00
def __init__(self, host_data):
"""
初始化
:param host_data: {
2022-08-11 07:45:03 +00:00
"name": "",
2017-12-06 10:31:51 +00:00
"ip": "",
"port": "",
2017-12-10 16:29:25 +00:00
# behind is not must be required
2017-12-06 10:31:51 +00:00
"username": "",
"password": "",
"private_key": "",
"become": {
"method": "",
"user": "",
"pass": "",
}
"groups": [],
"vars": {},
}
"""
self.host_data = host_data
2022-08-11 07:45:03 +00:00
hostname = host_data.get('name') or host_data.get('ip')
2017-12-06 10:31:51 +00:00
port = host_data.get('port') or 22
2017-12-10 16:29:25 +00:00
super().__init__(hostname, port)
2017-12-06 10:31:51 +00:00
self.__set_required_variables()
self.__set_extra_variables()
2017-03-05 12:53:24 +00:00
2017-12-06 10:31:51 +00:00
def __set_required_variables(self):
host_data = self.host_data
2022-09-21 03:17:14 +00:00
self.set_variable('ansible_host', host_data['address'])
2017-12-06 10:31:51 +00:00
self.set_variable('ansible_port', host_data['port'])
2017-12-10 16:29:25 +00:00
if host_data.get('username'):
self.set_variable('ansible_user', host_data['username'])
2017-03-05 12:53:24 +00:00
# 添加密码和密钥
2017-12-06 10:31:51 +00:00
if host_data.get('password'):
self.set_variable('ansible_ssh_pass', host_data['password'])
if host_data.get('private_key'):
self.set_variable('ansible_ssh_private_key_file', host_data['private_key'])
2017-03-05 12:53:24 +00:00
# 添加become支持
2017-12-06 10:31:51 +00:00
become = host_data.get("become", False)
2017-03-23 16:27:33 +00:00
if become:
2017-03-05 12:53:24 +00:00
self.set_variable("ansible_become", True)
2017-03-23 16:27:33 +00:00
self.set_variable("ansible_become_method", become.get('method', 'sudo'))
self.set_variable("ansible_become_user", become.get('user', 'root'))
self.set_variable("ansible_become_pass", become.get('pass', ''))
2017-03-05 12:53:24 +00:00
else:
self.set_variable("ansible_become", False)
2017-12-06 10:31:51 +00:00
def __set_extra_variables(self):
for k, v in self.host_data.get('vars', {}).items():
self.set_variable(k, v)
def __repr__(self):
return self.name
2017-03-05 12:53:24 +00:00
2017-12-06 10:31:51 +00:00
2017-12-10 16:29:25 +00:00
class BaseInventory(InventoryManager):
2017-03-05 12:53:24 +00:00
"""
提供生成Ansible inventory对象的方法
"""
2017-12-06 10:31:51 +00:00
loader_class = DataLoader
variable_manager_class = VariableManager
2017-12-10 16:29:25 +00:00
host_manager_class = BaseHost
2017-03-05 12:53:24 +00:00
def __init__(self, host_list=None, group_list=None):
2017-03-05 12:53:24 +00:00
"""
2017-12-06 10:31:51 +00:00
用于生成动态构建Ansible Inventory. super().__init__ 会自动调用
host_list: [{
2022-08-11 07:45:03 +00:00
"name": "",
"address": "",
2017-12-06 10:31:51 +00:00
"port": "",
"username": "",
"password": "",
"private_key": "",
"become": {
"method": "",
"user": "",
"pass": "",
},
"groups": [],
"vars": {},
},
]
group_list: [
{"name: "", children: [""]},
]
2017-12-10 16:29:25 +00:00
:param host_list:
:param group_list
2017-12-06 10:31:51 +00:00
"""
self.host_list = host_list or []
self.group_list = group_list or []
2017-12-10 16:29:25 +00:00
assert isinstance(host_list, list)
self.loader = self.loader_class()
self.variable_manager = self.variable_manager_class()
super().__init__(self.loader)
def get_groups(self):
return self._inventory.groups
def get_group(self, name):
return self._inventory.groups.get(name, None)
def get_or_create_group(self, name):
group = self.get_group(name)
if not group:
self.add_group(name)
return self.get_or_create_group(name)
else:
return group
2017-03-05 12:53:24 +00:00
def parse_groups(self):
for g in self.group_list:
parent = self.get_or_create_group(g.get("name"))
children = [self.get_or_create_group(n) for n in g.get('children', [])]
for child in children:
parent.add_child_group(child)
def parse_hosts(self):
2018-03-30 14:03:43 +00:00
group_all = self.get_or_create_group('all')
ungrouped = self.get_or_create_group('ungrouped')
2017-12-06 10:31:51 +00:00
for host_data in self.host_list:
host = self.host_manager_class(host_data=host_data)
2022-08-11 07:45:03 +00:00
self.hosts[host_data['name']] = host
2017-12-06 10:31:51 +00:00
groups_data = host_data.get('groups')
if groups_data:
for group_name in groups_data:
group = self.get_or_create_group(group_name)
2017-03-05 12:53:24 +00:00
group.add_host(host)
2018-03-30 14:03:43 +00:00
else:
ungrouped.add_host(host)
group_all.add_host(host)
def parse_sources(self, cache=False):
self.parse_groups()
self.parse_hosts()
2017-12-10 16:29:25 +00:00
def get_matched_hosts(self, pattern):
return self.get_hosts(pattern)
class JMSInventory:
def __init__(self, assets, account=None, ansible_connection='ssh',
account_policy='smart', host_var_callback=None):
"""
:param assets:
:param account: account username name if not set use account_policy
:param ansible_connection: ssh, local,
:param account_policy:
:param host_var_callback:
"""
pass