2017-03-05 12:53:24 +00:00
|
|
|
# ~*~ coding: utf-8 ~*~
|
2022-09-29 12:44:45 +00:00
|
|
|
import json
|
2022-10-08 11:12:04 +00:00
|
|
|
import os
|
2023-10-09 08:05:42 +00:00
|
|
|
import re
|
2022-10-09 12:54:11 +00:00
|
|
|
from collections import defaultdict
|
2022-09-29 12:44:45 +00:00
|
|
|
|
2022-10-08 11:51:29 +00:00
|
|
|
from django.utils.translation import gettext as _
|
2024-04-19 09:59:12 +00:00
|
|
|
from assets.const.category import Category
|
2022-10-08 11:51:29 +00:00
|
|
|
|
2022-10-08 08:55:14 +00:00
|
|
|
__all__ = ['JMSInventory']
|
2017-12-10 16:29:25 +00:00
|
|
|
|
|
|
|
|
2022-09-26 10:03:48 +00:00
|
|
|
class JMSInventory:
|
2023-04-26 10:50:30 +00:00
|
|
|
def __init__(
|
|
|
|
self, assets, account_policy='privileged_first',
|
|
|
|
account_prefer='root,Administrator', host_callback=None,
|
2023-12-20 08:02:13 +00:00
|
|
|
exclude_localhost=False, task_type=None, protocol=None
|
2023-04-26 10:50:30 +00:00
|
|
|
):
|
2022-09-26 10:03:48 +00:00
|
|
|
"""
|
|
|
|
:param assets:
|
2022-10-14 08:33:24 +00:00
|
|
|
:param account_prefer: account username name if not set use account_policy
|
2022-11-01 03:52:51 +00:00
|
|
|
:param account_policy: privileged_only, privileged_first, skip
|
2022-09-26 10:03:48 +00:00
|
|
|
"""
|
2022-09-29 12:44:45 +00:00
|
|
|
self.assets = self.clean_assets(assets)
|
2023-03-15 07:16:27 +00:00
|
|
|
self.account_prefer = self.get_account_prefer(account_prefer)
|
2022-09-29 12:44:45 +00:00
|
|
|
self.account_policy = account_policy
|
2022-10-28 08:25:16 +00:00
|
|
|
self.host_callback = host_callback
|
2022-12-07 12:13:26 +00:00
|
|
|
self.exclude_hosts = {}
|
2023-02-13 11:22:52 +00:00
|
|
|
self.exclude_localhost = exclude_localhost
|
2023-04-26 10:50:30 +00:00
|
|
|
self.task_type = task_type
|
2023-12-20 08:02:13 +00:00
|
|
|
self.protocol = protocol
|
2022-09-29 12:44:45 +00:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def clean_assets(assets):
|
|
|
|
from assets.models import Asset
|
|
|
|
asset_ids = [asset.id for asset in assets]
|
2022-10-20 12:34:15 +00:00
|
|
|
assets = Asset.objects.filter(id__in=asset_ids, is_active=True) \
|
2022-09-29 12:44:45 +00:00
|
|
|
.prefetch_related('platform', 'domain', 'accounts')
|
|
|
|
return assets
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def group_by_platform(assets):
|
|
|
|
groups = defaultdict(list)
|
|
|
|
for asset in assets:
|
|
|
|
groups[asset.platform].append(asset)
|
|
|
|
return groups
|
|
|
|
|
|
|
|
@staticmethod
|
2024-04-11 09:37:46 +00:00
|
|
|
def make_proxy_command(gateway, path_dir):
|
2022-09-29 12:44:45 +00:00
|
|
|
proxy_command_list = [
|
|
|
|
"ssh", "-o", "Port={}".format(gateway.port),
|
|
|
|
"-o", "StrictHostKeyChecking=no",
|
|
|
|
"{}@{}".format(gateway.username, gateway.address),
|
|
|
|
"-W", "%h:%p", "-q",
|
|
|
|
]
|
|
|
|
|
|
|
|
if gateway.password:
|
|
|
|
proxy_command_list.insert(
|
2023-03-17 08:57:40 +00:00
|
|
|
0, "sshpass -p {}".format(gateway.password)
|
2022-09-29 12:44:45 +00:00
|
|
|
)
|
|
|
|
if gateway.private_key:
|
2024-04-11 09:37:46 +00:00
|
|
|
proxy_command_list.append("-i {}".format(gateway.get_private_key_path(path_dir)))
|
2022-09-29 12:44:45 +00:00
|
|
|
|
2023-03-17 08:57:40 +00:00
|
|
|
proxy_command = "-o ProxyCommand='{}'".format(
|
2022-09-29 12:44:45 +00:00
|
|
|
" ".join(proxy_command_list)
|
|
|
|
)
|
|
|
|
return {"ansible_ssh_common_args": proxy_command}
|
|
|
|
|
2022-10-14 08:33:24 +00:00
|
|
|
@staticmethod
|
2024-04-11 09:37:46 +00:00
|
|
|
def make_account_ansible_vars(account, path_dir):
|
2022-10-14 08:33:24 +00:00
|
|
|
var = {
|
|
|
|
'ansible_user': account.username,
|
|
|
|
}
|
|
|
|
if not account.secret:
|
|
|
|
return var
|
2023-12-18 09:31:35 +00:00
|
|
|
|
2022-10-14 08:33:24 +00:00
|
|
|
if account.secret_type == 'password':
|
2023-12-18 09:31:35 +00:00
|
|
|
var['ansible_password'] = account.escape_jinja2_syntax(account.secret)
|
2022-10-14 08:33:24 +00:00
|
|
|
elif account.secret_type == 'ssh_key':
|
2024-04-11 09:37:46 +00:00
|
|
|
var['ansible_ssh_private_key_file'] = account.get_private_key_path(path_dir)
|
2022-10-14 08:33:24 +00:00
|
|
|
return var
|
|
|
|
|
2023-08-03 06:09:13 +00:00
|
|
|
@staticmethod
|
2024-04-11 09:37:46 +00:00
|
|
|
def make_custom_become_ansible_vars(account, su_from_auth, path_dir):
|
2023-10-09 06:35:21 +00:00
|
|
|
su_method = su_from_auth['ansible_become_method']
|
2023-08-03 06:09:13 +00:00
|
|
|
var = {
|
2023-10-09 06:35:21 +00:00
|
|
|
'custom_become': True,
|
|
|
|
'custom_become_method': su_method,
|
2023-08-03 06:09:13 +00:00
|
|
|
'custom_become_user': account.su_from.username,
|
2023-12-18 09:31:35 +00:00
|
|
|
'custom_become_password': account.escape_jinja2_syntax(account.su_from.secret),
|
2024-04-11 09:37:46 +00:00
|
|
|
'custom_become_private_key_path': account.su_from.get_private_key_path(path_dir)
|
2023-08-03 06:09:13 +00:00
|
|
|
}
|
|
|
|
return var
|
|
|
|
|
2024-04-01 09:28:18 +00:00
|
|
|
@staticmethod
|
|
|
|
def make_protocol_setting_vars(host, protocols):
|
|
|
|
# 针对 ssh 协议的特殊处理
|
|
|
|
for p in protocols:
|
|
|
|
if p.name == 'ssh':
|
|
|
|
if hasattr(p, 'setting'):
|
|
|
|
setting = getattr(p, 'setting')
|
|
|
|
host['old_ssh_version'] = setting.get('old_ssh_version', False)
|
|
|
|
|
2024-04-11 09:37:46 +00:00
|
|
|
def make_account_vars(self, host, asset, account, automation, protocol, platform, gateway, path_dir):
|
2023-05-12 07:37:13 +00:00
|
|
|
from accounts.const import AutomationTypes
|
2022-10-14 08:33:24 +00:00
|
|
|
if not account:
|
|
|
|
host['error'] = _("No account available")
|
|
|
|
return host
|
|
|
|
|
2023-03-30 06:58:20 +00:00
|
|
|
port = protocol.port if protocol else 22
|
2022-10-14 08:33:24 +00:00
|
|
|
host['ansible_host'] = asset.address
|
2023-03-30 06:58:20 +00:00
|
|
|
host['ansible_port'] = port
|
2022-10-14 08:33:24 +00:00
|
|
|
|
|
|
|
su_from = account.su_from
|
|
|
|
if platform.su_enabled and su_from:
|
2023-10-09 06:35:21 +00:00
|
|
|
su_from_auth = account.get_ansible_become_auth()
|
|
|
|
host.update(su_from_auth)
|
2024-04-11 09:37:46 +00:00
|
|
|
host.update(self.make_custom_become_ansible_vars(account, su_from_auth, path_dir))
|
2023-04-26 10:50:30 +00:00
|
|
|
elif platform.su_enabled and not su_from and \
|
|
|
|
self.task_type in (AutomationTypes.change_secret, AutomationTypes.push_account):
|
2024-04-11 09:37:46 +00:00
|
|
|
host.update(self.make_account_ansible_vars(account, path_dir))
|
2023-04-26 10:50:30 +00:00
|
|
|
host['ansible_become'] = True
|
|
|
|
host['ansible_become_user'] = 'root'
|
2023-12-18 09:31:35 +00:00
|
|
|
host['ansible_become_password'] = account.escape_jinja2_syntax(account.secret)
|
2022-10-14 08:33:24 +00:00
|
|
|
else:
|
2024-04-11 09:37:46 +00:00
|
|
|
host.update(self.make_account_ansible_vars(account, path_dir))
|
2022-10-14 08:33:24 +00:00
|
|
|
|
2024-04-22 03:27:48 +00:00
|
|
|
if platform.is_huawei():
|
2024-04-01 07:07:39 +00:00
|
|
|
host['ansible_connection'] = 'network_cli'
|
2024-04-26 10:57:02 +00:00
|
|
|
host['ansible_network_os'] = 'ce'
|
2024-04-01 07:07:39 +00:00
|
|
|
|
2022-10-14 08:33:24 +00:00
|
|
|
if gateway:
|
2023-03-30 06:58:20 +00:00
|
|
|
ansible_connection = host.get('ansible_connection', 'ssh')
|
2023-12-20 08:02:13 +00:00
|
|
|
if ansible_connection in ('local', 'winrm', 'rdp'):
|
2023-03-30 06:58:20 +00:00
|
|
|
host['gateway'] = {
|
|
|
|
'address': gateway.address, 'port': gateway.port,
|
2023-04-04 03:54:52 +00:00
|
|
|
'username': gateway.username, 'secret': gateway.password,
|
2024-04-11 09:37:46 +00:00
|
|
|
'private_key_path': gateway.get_private_key_path(path_dir)
|
2023-03-30 06:58:20 +00:00
|
|
|
}
|
|
|
|
host['jms_asset']['port'] = port
|
|
|
|
else:
|
2024-04-11 09:37:46 +00:00
|
|
|
ansible_ssh_common_args = self.make_proxy_command(gateway, path_dir)
|
2023-08-08 09:26:29 +00:00
|
|
|
host['jms_asset'].update(ansible_ssh_common_args)
|
|
|
|
host.update(ansible_ssh_common_args)
|
2023-03-22 07:26:23 +00:00
|
|
|
|
2023-12-20 08:02:13 +00:00
|
|
|
def get_primary_protocol(self, ansible_config, protocols):
|
2023-03-30 06:58:20 +00:00
|
|
|
invalid_protocol = type('protocol', (), {'name': 'null', 'port': 0})
|
2023-03-29 09:10:58 +00:00
|
|
|
ansible_connection = ansible_config.get('ansible_connection')
|
2023-03-30 06:58:20 +00:00
|
|
|
# 数值越小,优先级越高,若用户在 ansible_config 中配置了,则提高用户配置方式的优先级
|
|
|
|
protocol_priority = {'ssh': 10, 'winrm': 9, ansible_connection: 1}
|
2023-12-20 08:02:13 +00:00
|
|
|
if self.protocol:
|
|
|
|
protocol_priority.update({self.protocol: 0})
|
2023-03-30 06:58:20 +00:00
|
|
|
protocol_sorted = sorted(protocols, key=lambda x: protocol_priority.get(x.name, 999))
|
|
|
|
protocol = protocol_sorted[0] if protocol_sorted else invalid_protocol
|
2023-03-29 09:10:58 +00:00
|
|
|
return protocol
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def fill_ansible_config(ansible_config, protocol):
|
2023-12-20 08:02:13 +00:00
|
|
|
if protocol.name in ('ssh', 'winrm', 'rdp'):
|
2023-03-30 06:58:20 +00:00
|
|
|
ansible_config['ansible_connection'] = protocol.name
|
2023-04-03 01:57:40 +00:00
|
|
|
if protocol.name == 'winrm':
|
2023-03-29 09:10:58 +00:00
|
|
|
if protocol.setting.get('use_ssl', False):
|
|
|
|
ansible_config['ansible_winrm_scheme'] = 'https'
|
|
|
|
ansible_config['ansible_winrm_transport'] = 'ssl'
|
|
|
|
ansible_config['ansible_winrm_server_cert_validation'] = 'ignore'
|
|
|
|
else:
|
|
|
|
ansible_config['ansible_winrm_scheme'] = 'http'
|
2023-07-12 03:05:01 +00:00
|
|
|
ansible_config['ansible_winrm_transport'] = 'ntlm'
|
2024-03-01 08:17:18 +00:00
|
|
|
ansible_config['ansible_winrm_connection_timeout'] = 120
|
2023-03-29 09:10:58 +00:00
|
|
|
return ansible_config
|
|
|
|
|
2024-04-11 09:37:46 +00:00
|
|
|
def asset_to_host(self, asset, account, automation, protocols, platform, path_dir):
|
2023-03-30 06:58:20 +00:00
|
|
|
try:
|
|
|
|
ansible_config = dict(automation.ansible_config)
|
|
|
|
except (AttributeError, TypeError):
|
|
|
|
ansible_config = {}
|
|
|
|
|
2023-04-03 02:17:00 +00:00
|
|
|
protocol = self.get_primary_protocol(ansible_config, protocols)
|
2023-03-22 06:15:25 +00:00
|
|
|
|
2023-08-30 07:15:49 +00:00
|
|
|
tp, category = asset.type, asset.category
|
2023-10-09 08:05:42 +00:00
|
|
|
name = re.sub(r'[ \[\]/]', '_', asset.name)
|
2023-09-15 08:16:04 +00:00
|
|
|
secret_info = {k: v for k, v in asset.secret_info.items() if v}
|
2022-10-09 12:54:11 +00:00
|
|
|
host = {
|
2023-08-23 10:53:50 +00:00
|
|
|
'name': name,
|
2022-10-10 12:56:13 +00:00
|
|
|
'jms_asset': {
|
2022-10-10 05:56:42 +00:00
|
|
|
'id': str(asset.id), 'name': asset.name, 'address': asset.address,
|
2023-08-30 07:15:49 +00:00
|
|
|
'type': tp, 'category': category,
|
2023-03-30 06:58:20 +00:00
|
|
|
'protocol': protocol.name, 'port': protocol.port,
|
2023-09-15 08:16:04 +00:00
|
|
|
'spec_info': asset.spec_info, 'secret_info': secret_info,
|
2022-10-09 12:54:11 +00:00
|
|
|
'protocols': [{'name': p.name, 'port': p.port} for p in protocols],
|
|
|
|
},
|
2022-10-20 12:34:15 +00:00
|
|
|
'jms_account': {
|
2022-10-10 12:56:13 +00:00
|
|
|
'id': str(account.id), 'username': account.username,
|
2023-12-18 09:31:35 +00:00
|
|
|
'secret': account.escape_jinja2_syntax(account.secret),
|
2024-04-11 09:37:46 +00:00
|
|
|
'secret_type': account.secret_type, 'private_key_path': account.get_private_key_path(path_dir)
|
2022-10-10 12:56:13 +00:00
|
|
|
} if account else None
|
2022-10-09 12:54:11 +00:00
|
|
|
}
|
2022-12-07 12:13:26 +00:00
|
|
|
|
2024-04-01 09:28:18 +00:00
|
|
|
self.make_protocol_setting_vars(host, protocols)
|
|
|
|
|
2023-12-20 08:02:13 +00:00
|
|
|
protocols = host['jms_asset']['protocols']
|
|
|
|
host['jms_asset'].update({f"{p['name']}_port": p['port'] for p in protocols})
|
2023-08-30 07:15:49 +00:00
|
|
|
if host['jms_account'] and tp == 'oracle':
|
2022-11-09 10:23:00 +00:00
|
|
|
host['jms_account']['mode'] = 'sysdba' if account.privileged else None
|
|
|
|
|
2023-03-30 06:58:20 +00:00
|
|
|
ansible_config = self.fill_ansible_config(ansible_config, protocol)
|
2022-10-13 09:47:29 +00:00
|
|
|
host.update(ansible_config)
|
2022-10-27 10:53:10 +00:00
|
|
|
|
2022-09-29 12:44:45 +00:00
|
|
|
gateway = None
|
2023-01-18 09:14:02 +00:00
|
|
|
if not asset.is_gateway and asset.domain:
|
2022-09-29 12:44:45 +00:00
|
|
|
gateway = asset.domain.select_gateway()
|
|
|
|
|
2023-03-30 06:58:20 +00:00
|
|
|
self.make_account_vars(
|
2024-04-11 09:37:46 +00:00
|
|
|
host, asset, account, automation, protocol, platform, gateway, path_dir
|
2023-03-30 06:58:20 +00:00
|
|
|
)
|
2022-09-29 12:44:45 +00:00
|
|
|
return host
|
|
|
|
|
2024-04-17 11:59:23 +00:00
|
|
|
@staticmethod
|
|
|
|
def sorted_accounts(accounts):
|
2023-03-16 11:06:50 +00:00
|
|
|
connectivity_score = {'ok': 2, '-': 1, 'err': 0}
|
|
|
|
sort_key = lambda x: (x.privileged, connectivity_score.get(x.connectivity, 0), x.date_updated)
|
|
|
|
accounts_sorted = sorted(accounts, key=sort_key, reverse=True)
|
|
|
|
return accounts_sorted
|
2023-02-22 07:35:59 +00:00
|
|
|
|
2024-04-17 11:59:23 +00:00
|
|
|
def get_asset_sorted_accounts(self, asset):
|
|
|
|
accounts = list(asset.accounts.filter(is_active=True))
|
|
|
|
accounts_sorted = self.sorted_accounts(accounts)
|
|
|
|
return accounts_sorted
|
|
|
|
|
2023-03-15 07:16:27 +00:00
|
|
|
@staticmethod
|
|
|
|
def get_account_prefer(account_prefer):
|
|
|
|
account_usernames = []
|
|
|
|
if isinstance(account_prefer, str) and account_prefer:
|
|
|
|
account_usernames = list(map(lambda x: x.lower(), account_prefer.split(',')))
|
|
|
|
return account_usernames
|
|
|
|
|
|
|
|
def get_refer_account(self, accounts):
|
|
|
|
account = None
|
|
|
|
if accounts:
|
|
|
|
account = list(filter(
|
|
|
|
lambda a: a.username.lower() in self.account_prefer, accounts
|
|
|
|
))
|
|
|
|
account = account[0] if account else None
|
|
|
|
return account
|
|
|
|
|
2022-09-29 12:44:45 +00:00
|
|
|
def select_account(self, asset):
|
2023-03-16 11:06:50 +00:00
|
|
|
accounts = self.get_asset_sorted_accounts(asset)
|
|
|
|
if not accounts:
|
2023-02-08 12:39:24 +00:00
|
|
|
return None
|
2022-10-08 11:51:29 +00:00
|
|
|
|
2023-03-16 11:06:50 +00:00
|
|
|
refer_account = self.get_refer_account(accounts)
|
|
|
|
if refer_account:
|
|
|
|
return refer_account
|
2022-09-29 12:44:45 +00:00
|
|
|
|
2023-03-16 11:06:50 +00:00
|
|
|
account_selected = accounts[0]
|
|
|
|
if self.account_policy == 'skip':
|
|
|
|
return None
|
|
|
|
elif self.account_policy == 'privileged_first':
|
2022-11-01 03:52:51 +00:00
|
|
|
return account_selected
|
2023-03-16 11:06:50 +00:00
|
|
|
elif self.account_policy == 'privileged_only' and account_selected.privileged:
|
|
|
|
return account_selected
|
|
|
|
else:
|
|
|
|
return None
|
2022-09-29 12:44:45 +00:00
|
|
|
|
2023-03-29 09:10:58 +00:00
|
|
|
@staticmethod
|
2023-04-03 01:57:40 +00:00
|
|
|
def set_platform_protocol_setting_to_asset(asset, platform_protocols):
|
2023-03-29 09:10:58 +00:00
|
|
|
asset_protocols = asset.protocols.all()
|
|
|
|
for p in asset_protocols:
|
|
|
|
setattr(p, 'setting', platform_protocols.get(p.name, {}))
|
|
|
|
return asset_protocols
|
|
|
|
|
2022-10-20 12:34:15 +00:00
|
|
|
def generate(self, path_dir):
|
2022-09-29 12:44:45 +00:00
|
|
|
hosts = []
|
|
|
|
platform_assets = self.group_by_platform(self.assets)
|
|
|
|
for platform, assets in platform_assets.items():
|
|
|
|
automation = platform.automation
|
2023-03-29 09:10:58 +00:00
|
|
|
platform_protocols = {
|
|
|
|
p['name']: p['setting'] for p in platform.protocols.values('name', 'setting')
|
|
|
|
}
|
2022-10-13 09:47:29 +00:00
|
|
|
for asset in assets:
|
2023-04-03 01:57:40 +00:00
|
|
|
protocols = self.set_platform_protocol_setting_to_asset(asset, platform_protocols)
|
2022-09-29 12:44:45 +00:00
|
|
|
account = self.select_account(asset)
|
2024-04-11 09:37:46 +00:00
|
|
|
host = self.asset_to_host(asset, account, automation, protocols, platform, path_dir)
|
2022-10-12 10:08:57 +00:00
|
|
|
|
2022-10-08 11:51:29 +00:00
|
|
|
if not automation.ansible_enabled:
|
2022-10-12 10:08:57 +00:00
|
|
|
host['error'] = _('Ansible disabled')
|
|
|
|
|
2022-10-28 08:25:16 +00:00
|
|
|
if self.host_callback is not None:
|
|
|
|
host = self.host_callback(
|
2022-10-12 10:08:57 +00:00
|
|
|
host, asset=asset, account=account,
|
2022-10-20 12:34:15 +00:00
|
|
|
platform=platform, automation=automation,
|
|
|
|
path_dir=path_dir
|
2022-10-12 10:08:57 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
if isinstance(host, list):
|
|
|
|
hosts.extend(host)
|
2022-10-09 12:54:11 +00:00
|
|
|
else:
|
|
|
|
hosts.append(host)
|
2022-09-29 12:44:45 +00:00
|
|
|
|
2022-10-17 03:22:21 +00:00
|
|
|
exclude_hosts = list(filter(lambda x: x.get('error'), hosts))
|
2022-10-08 11:51:29 +00:00
|
|
|
if exclude_hosts:
|
|
|
|
print(_("Skip hosts below:"))
|
2022-10-10 05:56:42 +00:00
|
|
|
for i, host in enumerate(exclude_hosts, start=1):
|
2022-10-12 10:08:57 +00:00
|
|
|
print("{}: [{}] \t{}".format(i, host['name'], host['error']))
|
2022-12-07 12:13:26 +00:00
|
|
|
self.exclude_hosts[host['name']] = host['error']
|
2022-10-17 03:22:21 +00:00
|
|
|
hosts = list(filter(lambda x: not x.get('error'), hosts))
|
2022-09-29 12:44:45 +00:00
|
|
|
data = {'all': {'hosts': {}}}
|
|
|
|
for host in hosts:
|
|
|
|
name = host.pop('name')
|
|
|
|
data['all']['hosts'][name] = host
|
2023-08-01 02:39:34 +00:00
|
|
|
if not self.exclude_localhost:
|
|
|
|
data['all']['hosts'].update({
|
|
|
|
'localhost': {
|
|
|
|
'ansible_host': '127.0.0.1',
|
|
|
|
'ansible_connection': 'local'
|
|
|
|
}
|
|
|
|
})
|
2022-10-08 11:12:04 +00:00
|
|
|
return data
|
|
|
|
|
|
|
|
def write_to_file(self, path):
|
|
|
|
path_dir = os.path.dirname(path)
|
|
|
|
if not os.path.exists(path_dir):
|
|
|
|
os.makedirs(path_dir, 0o700, True)
|
2022-10-24 12:24:56 +00:00
|
|
|
data = self.generate(path_dir)
|
2022-09-29 12:44:45 +00:00
|
|
|
with open(path, 'w') as f:
|
|
|
|
f.write(json.dumps(data, indent=4))
|