2017-12-12 04:19:45 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
2022-11-17 12:48:50 +00:00
|
|
|
from collections import defaultdict
|
2017-12-12 04:19:45 +00:00
|
|
|
|
2021-01-12 10:06:42 +00:00
|
|
|
from django.db.models import TextChoices
|
|
|
|
from django.utils.translation import ugettext_lazy as _
|
2019-12-05 07:09:25 +00:00
|
|
|
|
2022-11-16 13:05:15 +00:00
|
|
|
from assets.const import Protocol
|
|
|
|
|
|
|
|
|
2021-01-12 10:06:42 +00:00
|
|
|
# Replay & Command Storage Choices
|
|
|
|
# --------------------------------
|
2019-12-05 07:09:25 +00:00
|
|
|
|
|
|
|
|
2022-11-04 03:40:16 +00:00
|
|
|
class ReplayStorageType(TextChoices):
|
2021-01-12 10:06:42 +00:00
|
|
|
null = 'null', 'Null',
|
|
|
|
server = 'server', 'Server'
|
|
|
|
s3 = 's3', 'S3'
|
|
|
|
ceph = 'ceph', 'Ceph'
|
|
|
|
swift = 'swift', 'Swift'
|
|
|
|
oss = 'oss', 'OSS'
|
|
|
|
azure = 'azure', 'Azure'
|
2021-04-12 11:37:04 +00:00
|
|
|
obs = 'obs', 'OBS'
|
2022-02-28 11:28:58 +00:00
|
|
|
cos = 'cos', 'COS'
|
2019-12-05 07:09:25 +00:00
|
|
|
|
|
|
|
|
2022-11-04 03:40:16 +00:00
|
|
|
class CommandStorageType(TextChoices):
|
2021-01-12 10:06:42 +00:00
|
|
|
null = 'null', 'Null',
|
|
|
|
server = 'server', 'Server'
|
|
|
|
es = 'es', 'Elasticsearch'
|
2020-12-10 12:50:22 +00:00
|
|
|
|
|
|
|
|
2021-01-12 10:06:42 +00:00
|
|
|
# Component Status Choices
|
|
|
|
# ------------------------
|
2020-12-10 12:50:22 +00:00
|
|
|
|
2022-11-04 03:40:16 +00:00
|
|
|
class ComponentLoad(TextChoices):
|
2020-12-10 12:50:22 +00:00
|
|
|
critical = 'critical', _('Critical')
|
|
|
|
high = 'high', _('High')
|
|
|
|
normal = 'normal', _('Normal')
|
2021-03-26 11:09:34 +00:00
|
|
|
offline = 'offline', _('Offline')
|
2020-12-10 12:50:22 +00:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def status(cls):
|
|
|
|
return set(dict(cls.choices).keys())
|
|
|
|
|
|
|
|
|
2022-11-17 12:48:50 +00:00
|
|
|
class HttpMethod(TextChoices):
|
|
|
|
web_gui = 'web_gui', 'Web GUI'
|
|
|
|
web_cli = 'web_cli', 'Web CLI'
|
2020-12-10 12:50:22 +00:00
|
|
|
|
2022-11-16 13:05:15 +00:00
|
|
|
|
2022-11-17 12:48:50 +00:00
|
|
|
class NativeClient(TextChoices):
|
2022-11-16 13:05:15 +00:00
|
|
|
# Koko
|
2022-11-18 11:29:19 +00:00
|
|
|
ssh = 'ssh', 'SSH'
|
2022-11-16 13:05:15 +00:00
|
|
|
putty = 'putty', 'PuTTY'
|
|
|
|
xshell = 'xshell', 'Xshell'
|
|
|
|
|
|
|
|
# Magnus
|
2022-11-29 06:42:04 +00:00
|
|
|
mysql = 'db_client_mysql', _('DB Client')
|
|
|
|
psql = 'db_client_psql', _('DB Client')
|
|
|
|
sqlplus = 'db_client_sqlplus', _('DB Client')
|
|
|
|
redis = 'db_client_redis', _('DB Client')
|
|
|
|
mongodb = 'db_client_mongodb', _('DB Client')
|
2022-11-16 13:05:15 +00:00
|
|
|
|
|
|
|
# Razor
|
|
|
|
mstsc = 'mstsc', 'Remote Desktop'
|
|
|
|
|
|
|
|
@classmethod
|
2022-11-17 12:48:50 +00:00
|
|
|
def get_native_clients(cls):
|
2022-11-17 03:46:35 +00:00
|
|
|
clients = {
|
2022-11-17 12:48:50 +00:00
|
|
|
Protocol.ssh: {
|
|
|
|
'default': [cls.ssh],
|
|
|
|
'windows': [cls.putty],
|
|
|
|
},
|
|
|
|
Protocol.rdp: [cls.mstsc],
|
2022-11-29 06:42:04 +00:00
|
|
|
Protocol.mysql: [cls.mysql],
|
|
|
|
Protocol.oracle: [cls.sqlplus],
|
|
|
|
Protocol.postgresql: [cls.psql],
|
|
|
|
Protocol.redis: [cls.redis],
|
|
|
|
Protocol.mongodb: [cls.mongodb],
|
2022-11-17 03:46:35 +00:00
|
|
|
}
|
2022-11-17 12:48:50 +00:00
|
|
|
return clients
|
|
|
|
|
2022-11-29 06:42:04 +00:00
|
|
|
@classmethod
|
|
|
|
def get_target_protocol(cls, name, os):
|
|
|
|
for protocol, clients in cls.get_native_clients().items():
|
|
|
|
if isinstance(clients, dict):
|
|
|
|
clients = clients.get(os) or clients.get('default')
|
|
|
|
if name in clients:
|
|
|
|
return protocol
|
|
|
|
return None
|
|
|
|
|
2022-11-17 12:48:50 +00:00
|
|
|
@classmethod
|
2022-11-18 11:29:19 +00:00
|
|
|
def get_methods(cls, os='windows'):
|
2022-11-17 12:48:50 +00:00
|
|
|
clients_map = cls.get_native_clients()
|
|
|
|
methods = defaultdict(list)
|
|
|
|
|
|
|
|
for protocol, _clients in clients_map.items():
|
|
|
|
if isinstance(_clients, dict):
|
|
|
|
_clients = _clients.get(os, _clients['default'])
|
|
|
|
for client in _clients:
|
|
|
|
methods[protocol].append({
|
|
|
|
'value': client.value,
|
|
|
|
'label': client.label,
|
|
|
|
'type': 'native',
|
|
|
|
})
|
|
|
|
return methods
|
2022-11-17 03:46:35 +00:00
|
|
|
|
|
|
|
@classmethod
|
2022-11-29 06:42:04 +00:00
|
|
|
def get_launch_command(cls, name, token, endpoint, os='windows'):
|
2022-11-17 03:46:35 +00:00
|
|
|
commands = {
|
2022-11-29 06:42:04 +00:00
|
|
|
cls.ssh: f'ssh {token.id}@{endpoint.host} -p {endpoint.ssh_port}',
|
|
|
|
cls.putty: f'putty -ssh {token.id}@{endpoint.host} -P {endpoint.ssh_port}',
|
|
|
|
cls.xshell: f'xshell -url ssh://{token.id}:{token.value}@{endpoint.host}:{endpoint.ssh_port}',
|
|
|
|
# cls.mysql: 'mysql -h {hostname} -P {port} -u {username} -p',
|
|
|
|
# cls.psql: {
|
2022-11-28 14:58:43 +00:00
|
|
|
# 'default': 'psql -h {hostname} -p {port} -U {username} -W',
|
|
|
|
# 'windows': 'psql /h {hostname} /p {port} /U {username} -W',
|
|
|
|
# },
|
2022-11-29 06:42:04 +00:00
|
|
|
# cls.sqlplus: 'sqlplus {username}/{password}@{hostname}:{port}',
|
|
|
|
# cls.redis: 'redis-cli -h {hostname} -p {port} -a {password}',
|
2022-11-16 13:05:15 +00:00
|
|
|
}
|
2022-11-17 03:46:35 +00:00
|
|
|
command = commands.get(name)
|
|
|
|
if isinstance(command, dict):
|
|
|
|
command = command.get(os, command.get('default'))
|
|
|
|
return command
|
|
|
|
|
|
|
|
|
2022-11-18 11:29:19 +00:00
|
|
|
class AppletMethod:
|
2022-11-17 03:46:35 +00:00
|
|
|
@classmethod
|
2022-11-18 11:29:19 +00:00
|
|
|
def get_methods(cls):
|
2022-11-17 03:46:35 +00:00
|
|
|
from .models import Applet
|
2022-11-17 12:48:50 +00:00
|
|
|
applets = Applet.objects.all()
|
|
|
|
methods = defaultdict(list)
|
|
|
|
for applet in applets:
|
|
|
|
for protocol in applet.protocols:
|
|
|
|
methods[protocol].append({
|
|
|
|
'value': applet.name,
|
|
|
|
'label': applet.display_name,
|
|
|
|
'icon': applet.icon,
|
|
|
|
})
|
|
|
|
return methods
|
2022-11-16 13:05:15 +00:00
|
|
|
|
|
|
|
|
2022-11-17 12:48:50 +00:00
|
|
|
class TerminalType(TextChoices):
|
|
|
|
koko = 'koko', 'KoKo'
|
|
|
|
guacamole = 'guacamole', 'Guacamole'
|
|
|
|
omnidb = 'omnidb', 'OmniDB'
|
|
|
|
xrdp = 'xrdp', 'Xrdp'
|
|
|
|
lion = 'lion', 'Lion'
|
|
|
|
core = 'core', 'Core'
|
|
|
|
celery = 'celery', 'Celery'
|
|
|
|
magnus = 'magnus', 'Magnus'
|
|
|
|
razor = 'razor', 'Razor'
|
|
|
|
tinker = 'tinker', 'Tinker'
|
2022-11-16 13:05:15 +00:00
|
|
|
|
|
|
|
@classmethod
|
2022-11-17 12:48:50 +00:00
|
|
|
def types(cls):
|
|
|
|
return set(dict(cls.choices).keys())
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def protocols(cls):
|
2022-11-18 11:29:19 +00:00
|
|
|
protocols = {
|
2022-11-17 12:48:50 +00:00
|
|
|
cls.koko: {
|
2022-11-18 11:29:19 +00:00
|
|
|
'web_method': HttpMethod.web_cli,
|
2022-11-17 12:48:50 +00:00
|
|
|
'listen': [Protocol.ssh, Protocol.http],
|
|
|
|
'support': [
|
|
|
|
Protocol.ssh, Protocol.telnet,
|
|
|
|
Protocol.mysql, Protocol.postgresql,
|
|
|
|
Protocol.oracle, Protocol.sqlserver,
|
|
|
|
Protocol.mariadb, Protocol.redis,
|
2022-11-18 11:29:19 +00:00
|
|
|
Protocol.mongodb, Protocol.k8s,
|
2022-11-17 12:48:50 +00:00
|
|
|
],
|
|
|
|
'match': 'm2m'
|
|
|
|
},
|
|
|
|
cls.omnidb: {
|
2022-11-18 11:29:19 +00:00
|
|
|
'web_method': HttpMethod.web_gui,
|
2022-11-17 12:48:50 +00:00
|
|
|
'listen': [Protocol.http],
|
|
|
|
'support': [
|
|
|
|
Protocol.mysql, Protocol.postgresql, Protocol.oracle,
|
|
|
|
Protocol.sqlserver, Protocol.mariadb
|
|
|
|
],
|
|
|
|
'match': 'm2m'
|
|
|
|
},
|
|
|
|
cls.lion: {
|
2022-11-18 11:29:19 +00:00
|
|
|
'web_method': HttpMethod.web_gui,
|
2022-11-17 12:48:50 +00:00
|
|
|
'listen': [Protocol.http],
|
|
|
|
'support': [Protocol.rdp, Protocol.vnc],
|
|
|
|
'match': 'm2m'
|
|
|
|
},
|
|
|
|
cls.magnus: {
|
|
|
|
'listen': [],
|
|
|
|
'support': [
|
2022-11-28 09:57:33 +00:00
|
|
|
Protocol.mysql, Protocol.postgresql,
|
|
|
|
Protocol.oracle, Protocol.mariadb
|
2022-11-17 12:48:50 +00:00
|
|
|
],
|
|
|
|
'match': 'map'
|
|
|
|
},
|
|
|
|
cls.razor: {
|
|
|
|
'listen': [Protocol.rdp],
|
|
|
|
'support': [Protocol.rdp],
|
|
|
|
'match': 'map'
|
2022-11-18 11:29:19 +00:00
|
|
|
},
|
2022-11-16 13:05:15 +00:00
|
|
|
}
|
2022-11-18 11:29:19 +00:00
|
|
|
return protocols
|
2022-11-17 12:48:50 +00:00
|
|
|
|
2022-11-29 13:41:33 +00:00
|
|
|
@classmethod
|
|
|
|
def get_connect_method(cls, name, protocol, os):
|
|
|
|
methods = cls.get_protocols_connect_methods(os)
|
|
|
|
protocol_methods = methods.get(protocol, [])
|
|
|
|
for method in protocol_methods:
|
|
|
|
if method['value'] == name:
|
|
|
|
return method
|
|
|
|
return None
|
|
|
|
|
2022-11-17 12:48:50 +00:00
|
|
|
@classmethod
|
|
|
|
def get_protocols_connect_methods(cls, os):
|
|
|
|
methods = defaultdict(list)
|
2022-11-18 11:29:19 +00:00
|
|
|
native_methods = NativeClient.get_methods(os)
|
|
|
|
applet_methods = AppletMethod.get_methods()
|
2022-11-17 12:48:50 +00:00
|
|
|
|
|
|
|
for component, component_protocol in cls.protocols().items():
|
|
|
|
support = component_protocol['support']
|
|
|
|
|
|
|
|
for protocol in support:
|
|
|
|
if component_protocol['match'] == 'map':
|
|
|
|
listen = [protocol]
|
|
|
|
else:
|
|
|
|
listen = component_protocol['listen']
|
|
|
|
|
|
|
|
for listen_protocol in listen:
|
|
|
|
if listen_protocol == Protocol.http:
|
2022-11-18 11:29:19 +00:00
|
|
|
web_protocol = component_protocol['web_method']
|
|
|
|
methods[protocol.value].append({
|
2022-11-17 12:48:50 +00:00
|
|
|
'value': web_protocol.value,
|
|
|
|
'label': web_protocol.label,
|
2022-11-29 06:42:04 +00:00
|
|
|
'endpoint_protocol': 'http',
|
2022-11-17 12:48:50 +00:00
|
|
|
'type': 'web',
|
2022-11-18 11:29:19 +00:00
|
|
|
'component': component.value,
|
2022-11-17 12:48:50 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
# Native method
|
2022-11-18 11:29:19 +00:00
|
|
|
methods[protocol.value].extend([
|
2022-11-29 06:42:04 +00:00
|
|
|
{
|
|
|
|
'component': component.value,
|
|
|
|
'type': 'native',
|
|
|
|
'endpoint_protocol': listen_protocol,
|
|
|
|
**method
|
|
|
|
}
|
2022-11-18 11:29:19 +00:00
|
|
|
for method in native_methods[listen_protocol]
|
|
|
|
])
|
|
|
|
|
|
|
|
for protocol, applet_methods in applet_methods.items():
|
|
|
|
for method in applet_methods:
|
|
|
|
method['type'] = 'applet'
|
2022-11-29 06:42:04 +00:00
|
|
|
method['listen'] = 'rdp'
|
2022-11-18 11:29:19 +00:00
|
|
|
method['component'] = cls.tinker.value
|
|
|
|
methods[protocol].extend(applet_methods)
|
2022-11-17 12:48:50 +00:00
|
|
|
return methods
|