2017-12-04 08:41:00 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
2020-03-12 08:24:38 +00:00
|
|
|
import os
|
|
|
|
|
2020-09-27 06:34:47 +00:00
|
|
|
from django.conf import settings
|
2020-03-12 08:24:38 +00:00
|
|
|
from django.core.files.storage import default_storage
|
2020-10-27 03:35:31 +00:00
|
|
|
from django.utils.translation import ugettext as _
|
|
|
|
|
2020-03-12 08:24:38 +00:00
|
|
|
import jms_storage
|
2018-12-19 02:49:30 +00:00
|
|
|
|
2020-10-27 03:35:31 +00:00
|
|
|
from common.tasks import send_mail_async
|
|
|
|
from common.utils import get_logger, reverse
|
|
|
|
from settings.models import Setting
|
2020-12-10 12:50:22 +00:00
|
|
|
from . import const
|
2020-09-27 06:34:47 +00:00
|
|
|
|
2020-10-27 03:35:31 +00:00
|
|
|
from .models import ReplayStorage, Session, Command
|
2020-03-12 08:24:38 +00:00
|
|
|
|
|
|
|
logger = get_logger(__name__)
|
2017-12-12 04:19:45 +00:00
|
|
|
|
|
|
|
|
2020-03-12 08:24:38 +00:00
|
|
|
def find_session_replay_local(session):
|
|
|
|
# 新版本和老版本的文件后缀不同
|
|
|
|
session_path = session.get_rel_replay_path() # 存在外部存储上的路径
|
|
|
|
local_path = session.get_local_path()
|
|
|
|
local_path_v1 = session.get_local_path(version=1)
|
|
|
|
|
|
|
|
# 去default storage中查找
|
|
|
|
for _local_path in (local_path, local_path_v1, session_path):
|
|
|
|
if default_storage.exists(_local_path):
|
|
|
|
url = default_storage.url(_local_path)
|
|
|
|
return _local_path, url
|
|
|
|
return None, None
|
|
|
|
|
|
|
|
|
|
|
|
def download_session_replay(session):
|
|
|
|
session_path = session.get_rel_replay_path() # 存在外部存储上的路径
|
|
|
|
local_path = session.get_local_path()
|
|
|
|
replay_storages = ReplayStorage.objects.all()
|
|
|
|
configs = {
|
|
|
|
storage.name: storage.config
|
|
|
|
for storage in replay_storages
|
2021-01-12 10:06:42 +00:00
|
|
|
if not storage.type_null_or_server
|
2020-03-12 08:24:38 +00:00
|
|
|
}
|
2020-09-27 06:34:47 +00:00
|
|
|
if settings.SERVER_REPLAY_STORAGE:
|
|
|
|
configs['SERVER_REPLAY_STORAGE'] = settings.SERVER_REPLAY_STORAGE
|
2020-03-12 08:24:38 +00:00
|
|
|
if not configs:
|
|
|
|
msg = "Not found replay file, and not remote storage set"
|
|
|
|
return None, msg
|
|
|
|
|
|
|
|
# 保存到storage的路径
|
|
|
|
target_path = os.path.join(default_storage.base_location, local_path)
|
|
|
|
target_dir = os.path.dirname(target_path)
|
|
|
|
if not os.path.isdir(target_dir):
|
|
|
|
os.makedirs(target_dir, exist_ok=True)
|
|
|
|
storage = jms_storage.get_multi_object_storage(configs)
|
|
|
|
ok, err = storage.download(session_path, target_path)
|
|
|
|
if not ok:
|
|
|
|
msg = "Failed download replay file: {}".format(err)
|
|
|
|
logger.error(msg)
|
|
|
|
return None, msg
|
|
|
|
url = default_storage.url(local_path)
|
|
|
|
return local_path, url
|
|
|
|
|
|
|
|
|
|
|
|
def get_session_replay_url(session):
|
|
|
|
local_path, url = find_session_replay_local(session)
|
|
|
|
if local_path is None:
|
|
|
|
local_path, url = download_session_replay(session)
|
|
|
|
return local_path, url
|
2020-10-27 03:35:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
def send_command_alert_mail(command):
|
|
|
|
session_obj = Session.objects.get(id=command['session'])
|
2021-01-18 09:22:19 +00:00
|
|
|
|
|
|
|
input = command['input']
|
|
|
|
if isinstance(input, str):
|
|
|
|
input = input.replace('\r\n', ' ').replace('\r', ' ').replace('\n', ' ')
|
|
|
|
|
2020-10-27 03:35:31 +00:00
|
|
|
subject = _("Insecure Command Alert: [%(name)s->%(login_from)s@%(remote_addr)s] $%(command)s") % {
|
|
|
|
'name': command['user'],
|
|
|
|
'login_from': session_obj.get_login_from_display(),
|
|
|
|
'remote_addr': session_obj.remote_addr,
|
2021-01-18 09:22:19 +00:00
|
|
|
'command': input
|
2020-10-27 03:35:31 +00:00
|
|
|
}
|
2021-01-18 09:22:19 +00:00
|
|
|
|
2020-10-27 03:35:31 +00:00
|
|
|
recipient_list = settings.SECURITY_INSECURE_COMMAND_EMAIL_RECEIVER.split(',')
|
|
|
|
message = _("""
|
|
|
|
Command: %(command)s
|
|
|
|
<br>
|
|
|
|
Asset: %(host_name)s (%(host_ip)s)
|
|
|
|
<br>
|
|
|
|
User: %(user)s
|
|
|
|
<br>
|
|
|
|
Level: %(risk_level)s
|
|
|
|
<br>
|
|
|
|
Session: <a href="%(session_detail_url)s">session detail</a>
|
|
|
|
<br>
|
|
|
|
""") % {
|
|
|
|
'command': command['input'],
|
|
|
|
'host_name': command['asset'],
|
|
|
|
'host_ip': session_obj.asset_obj.ip,
|
|
|
|
'user': command['user'],
|
|
|
|
'risk_level': Command.get_risk_level_str(command['risk_level']),
|
|
|
|
'session_detail_url': reverse('api-terminal:session-detail',
|
|
|
|
kwargs={'pk': command['session']},
|
|
|
|
external=True, api_to_ui=True),
|
|
|
|
}
|
|
|
|
logger.debug(message)
|
|
|
|
|
|
|
|
send_mail_async.delay(subject, message, recipient_list, html_message=message)
|
2020-12-10 12:50:22 +00:00
|
|
|
|
|
|
|
|
2021-01-21 07:18:54 +00:00
|
|
|
def send_command_execution_alert_mail(command):
|
|
|
|
subject = _("Insecure Web Command Execution Alert: [%(name)s]") % {
|
|
|
|
'name': command['user'],
|
|
|
|
}
|
|
|
|
input = command['input']
|
|
|
|
input = input.replace('\n', '<br>')
|
|
|
|
recipient_list = settings.SECURITY_INSECURE_COMMAND_EMAIL_RECEIVER.split(',')
|
|
|
|
|
|
|
|
assets = ', '.join([str(asset) for asset in command['assets']])
|
|
|
|
message = _("""
|
|
|
|
<br>
|
|
|
|
Assets: %(assets)s
|
|
|
|
<br>
|
|
|
|
User: %(user)s
|
|
|
|
<br>
|
|
|
|
Level: %(risk_level)s
|
|
|
|
<br>
|
|
|
|
|
|
|
|
----------------- Commands ---------------- <br>
|
|
|
|
%(command)s <br>
|
|
|
|
----------------- Commands ---------------- <br>
|
|
|
|
""") % {
|
|
|
|
'command': input,
|
|
|
|
'assets': assets,
|
|
|
|
'user': command['user'],
|
|
|
|
'risk_level': Command.get_risk_level_str(command['risk_level']),
|
|
|
|
}
|
|
|
|
|
|
|
|
send_mail_async.delay(subject, message, recipient_list, html_message=message)
|
|
|
|
|
|
|
|
|
2020-12-10 12:50:22 +00:00
|
|
|
class ComponentsMetricsUtil(object):
|
|
|
|
|
2020-12-15 05:00:59 +00:00
|
|
|
@staticmethod
|
|
|
|
def get_components(tp=None):
|
2020-12-10 12:50:22 +00:00
|
|
|
from .models import Terminal
|
2020-12-17 09:31:44 +00:00
|
|
|
components = Terminal.objects.filter(is_deleted=False).order_by('type')
|
2020-12-15 05:00:59 +00:00
|
|
|
if tp:
|
|
|
|
components = components.filter(type=tp)
|
|
|
|
return components
|
2020-12-10 12:50:22 +00:00
|
|
|
|
2020-12-15 05:00:59 +00:00
|
|
|
def get_metrics(self, tp=None):
|
|
|
|
components = self.get_components(tp)
|
2020-12-15 07:46:02 +00:00
|
|
|
total_count = normal_count = high_count = critical_count = offline_count = \
|
|
|
|
session_active_total = 0
|
2020-12-15 05:00:59 +00:00
|
|
|
for component in components:
|
2020-12-10 12:50:22 +00:00
|
|
|
total_count += 1
|
2020-12-15 05:00:59 +00:00
|
|
|
if component.is_alive:
|
|
|
|
if component.is_normal:
|
|
|
|
normal_count += 1
|
|
|
|
elif component.is_high:
|
|
|
|
high_count += 1
|
|
|
|
else:
|
|
|
|
# critical
|
|
|
|
critical_count += 1
|
|
|
|
session_active_total += component.state.get('session_active_count', 0)
|
2020-12-10 12:50:22 +00:00
|
|
|
else:
|
2020-12-15 07:46:02 +00:00
|
|
|
offline_count += 1
|
2020-12-15 05:00:59 +00:00
|
|
|
return {
|
2020-12-10 12:50:22 +00:00
|
|
|
'total': total_count,
|
|
|
|
'normal': normal_count,
|
|
|
|
'high': high_count,
|
|
|
|
'critical': critical_count,
|
2020-12-15 07:46:02 +00:00
|
|
|
'offline': offline_count,
|
2020-12-10 12:50:22 +00:00
|
|
|
'session_active': session_active_total
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class ComponentsPrometheusMetricsUtil(ComponentsMetricsUtil):
|
|
|
|
|
|
|
|
@staticmethod
|
2020-12-15 05:00:59 +00:00
|
|
|
def convert_status_metrics(metrics):
|
2020-12-10 12:50:22 +00:00
|
|
|
return {
|
|
|
|
'any': metrics['total'],
|
|
|
|
'normal': metrics['normal'],
|
|
|
|
'high': metrics['high'],
|
2020-12-15 07:46:02 +00:00
|
|
|
'critical': metrics['critical'],
|
|
|
|
'offline': metrics['offline']
|
2020-12-10 12:50:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def get_prometheus_metrics_text(self):
|
2020-12-15 05:00:59 +00:00
|
|
|
prometheus_metrics = list()
|
2020-12-10 12:50:22 +00:00
|
|
|
|
2020-12-15 05:00:59 +00:00
|
|
|
# 各组件状态个数汇总
|
|
|
|
prometheus_metrics.append('# JumpServer 各组件状态个数汇总')
|
|
|
|
status_metric_text = 'jumpserver_components_status_total{component_type="%s", status="%s"} %s'
|
|
|
|
for tp in const.TerminalTypeChoices.types():
|
|
|
|
prometheus_metrics.append(f'## 组件: {tp}')
|
|
|
|
metrics_tp = self.get_metrics(tp)
|
|
|
|
status_metrics = self.convert_status_metrics(metrics_tp)
|
2020-12-10 12:50:22 +00:00
|
|
|
for status, value in status_metrics.items():
|
2020-12-15 05:00:59 +00:00
|
|
|
metric_text = status_metric_text % (tp, status, value)
|
2020-12-10 12:50:22 +00:00
|
|
|
prometheus_metrics.append(metric_text)
|
|
|
|
|
|
|
|
prometheus_metrics.append('\n')
|
2020-12-15 05:00:59 +00:00
|
|
|
|
|
|
|
# 各组件在线会话数汇总
|
2020-12-10 12:50:22 +00:00
|
|
|
prometheus_metrics.append('# JumpServer 各组件在线会话数汇总')
|
2020-12-15 05:00:59 +00:00
|
|
|
session_active_metric_text = 'jumpserver_components_session_active_total{component_type="%s"} %s'
|
|
|
|
for tp in const.TerminalTypeChoices.types():
|
|
|
|
prometheus_metrics.append(f'## 组件: {tp}')
|
|
|
|
metrics_tp = self.get_metrics(tp)
|
|
|
|
metric_text = session_active_metric_text % (tp, metrics_tp['session_active'])
|
2020-12-10 12:50:22 +00:00
|
|
|
prometheus_metrics.append(metric_text)
|
|
|
|
|
|
|
|
prometheus_metrics.append('\n')
|
2020-12-15 05:00:59 +00:00
|
|
|
|
|
|
|
# 各组件节点指标
|
|
|
|
prometheus_metrics.append('# JumpServer 各组件一些指标')
|
|
|
|
state_metric_text = 'jumpserver_components_%s{component_type="%s", component="%s"} %s'
|
|
|
|
states = [
|
2020-12-10 12:50:22 +00:00
|
|
|
'system_cpu_load_1', 'system_memory_used_percent',
|
|
|
|
'system_disk_used_percent', 'session_active_count'
|
|
|
|
]
|
2020-12-15 05:00:59 +00:00
|
|
|
for state in states:
|
|
|
|
prometheus_metrics.append(f'## 指标: {state}')
|
|
|
|
components = self.get_components()
|
|
|
|
for component in components:
|
2020-12-10 12:50:22 +00:00
|
|
|
if not component.is_alive:
|
|
|
|
continue
|
2020-12-15 05:00:59 +00:00
|
|
|
metric_text = state_metric_text % (
|
|
|
|
state, component.type, component.name, component.state.get(state)
|
2020-12-10 12:50:22 +00:00
|
|
|
)
|
|
|
|
prometheus_metrics.append(metric_text)
|
|
|
|
|
|
|
|
prometheus_metrics_text = '\n'.join(prometheus_metrics)
|
|
|
|
return prometheus_metrics_text
|