mirror of https://github.com/jumpserver/jumpserver
Merge branch 'v3' of github.com:jumpserver/jumpserver into v3
commit
26fa1f6f08
|
@ -10,7 +10,6 @@ from kubernetes.client.exceptions import ApiException
|
||||||
from rest_framework.generics import get_object_or_404
|
from rest_framework.generics import get_object_or_404
|
||||||
|
|
||||||
from common.utils import get_logger
|
from common.utils import get_logger
|
||||||
from common.tree import TreeNode
|
|
||||||
from assets.models import Account, Asset
|
from assets.models import Account, Asset
|
||||||
|
|
||||||
from ..const import CloudTypes, Category
|
from ..const import CloudTypes, Category
|
||||||
|
@ -19,13 +18,15 @@ logger = get_logger(__file__)
|
||||||
|
|
||||||
|
|
||||||
class KubernetesClient:
|
class KubernetesClient:
|
||||||
def __init__(self, url, token):
|
def __init__(self, url, token, proxy=None):
|
||||||
self.url = url
|
self.url = url
|
||||||
self.token = token
|
self.token = token
|
||||||
|
self.proxy = proxy
|
||||||
|
|
||||||
def get_api(self):
|
def get_api(self):
|
||||||
configuration = client.Configuration()
|
configuration = client.Configuration()
|
||||||
configuration.host = self.url
|
configuration.host = self.url
|
||||||
|
configuration.proxy = self.proxy
|
||||||
configuration.verify_ssl = False
|
configuration.verify_ssl = False
|
||||||
configuration.api_key = {"authorization": "Bearer " + self.token}
|
configuration.api_key = {"authorization": "Bearer " + self.token}
|
||||||
c = api_client.ApiClient(configuration=configuration)
|
c = api_client.ApiClient(configuration=configuration)
|
||||||
|
@ -82,11 +83,23 @@ class KubernetesClient:
|
||||||
data[namespace] = [pod_info, ]
|
data[namespace] = [pod_info, ]
|
||||||
return data
|
return data
|
||||||
|
|
||||||
@staticmethod
|
@classmethod
|
||||||
def get_kubernetes_data(app_id, username):
|
def get_proxy_url(cls, asset):
|
||||||
|
if not asset.domain:
|
||||||
|
return None
|
||||||
|
|
||||||
|
gateway = asset.domain.select_gateway()
|
||||||
|
if not gateway:
|
||||||
|
return None
|
||||||
|
return f'{gateway.address}:{gateway.port}'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def get_kubernetes_data(cls, app_id, username):
|
||||||
asset = get_object_or_404(Asset, id=app_id)
|
asset = get_object_or_404(Asset, id=app_id)
|
||||||
account = get_object_or_404(Account, asset=asset, username=username)
|
account = get_object_or_404(Account, asset=asset, username=username)
|
||||||
k8s = KubernetesClient(asset.address, account.secret)
|
k8s_url = f'{asset.address}:{asset.port}'
|
||||||
|
proxy_url = cls.get_proxy_url(asset)
|
||||||
|
k8s = cls(k8s_url, account.secret, proxy=proxy_url)
|
||||||
return k8s.get_pods()
|
return k8s.get_pods()
|
||||||
|
|
||||||
|
|
||||||
|
@ -112,7 +125,7 @@ class KubernetesTree:
|
||||||
def as_asset_tree_node(self, asset):
|
def as_asset_tree_node(self, asset):
|
||||||
i = urlencode({'asset_id': self.tree_id})
|
i = urlencode({'asset_id': self.tree_id})
|
||||||
node = self.create_tree_node(
|
node = self.create_tree_node(
|
||||||
i, str(asset.id), str(asset), 'asset',
|
i, str(asset.id), str(asset), 'asset', is_open=True,
|
||||||
)
|
)
|
||||||
return node
|
return node
|
||||||
|
|
||||||
|
@ -136,14 +149,14 @@ class KubernetesTree:
|
||||||
return node
|
return node
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def create_tree_node(id_, pid, name, identity, icon='', is_container=False):
|
def create_tree_node(id_, pid, name, identity, icon='', is_container=False, is_open=False):
|
||||||
node = {
|
node = {
|
||||||
'id': id_,
|
'id': id_,
|
||||||
'name': name,
|
'name': name,
|
||||||
'title': name,
|
'title': name,
|
||||||
'pId': pid,
|
'pId': pid,
|
||||||
'isParent': not is_container,
|
'isParent': not is_container,
|
||||||
'open': False,
|
'open': is_open,
|
||||||
'iconSkin': icon,
|
'iconSkin': icon,
|
||||||
'meta': {
|
'meta': {
|
||||||
'type': 'k8s',
|
'type': 'k8s',
|
||||||
|
|
|
@ -13,19 +13,25 @@ from common.drf.api import JMSReadOnlyModelViewSet
|
||||||
from common.plugins.es import QuerySet as ESQuerySet
|
from common.plugins.es import QuerySet as ESQuerySet
|
||||||
from common.drf.filters import DatetimeRangeFilter
|
from common.drf.filters import DatetimeRangeFilter
|
||||||
from common.api import CommonGenericViewSet
|
from common.api import CommonGenericViewSet
|
||||||
from orgs.mixins.api import OrgGenericViewSet, OrgBulkModelViewSet, OrgRelationMixin
|
from ops.models.job import JobAuditLog
|
||||||
|
from orgs.mixins.api import OrgGenericViewSet, OrgBulkModelViewSet
|
||||||
from orgs.utils import current_org
|
from orgs.utils import current_org
|
||||||
# from ops.models import CommandExecution
|
|
||||||
from . import filters
|
|
||||||
from .backends import TYPE_ENGINE_MAPPING
|
from .backends import TYPE_ENGINE_MAPPING
|
||||||
from .models import FTPLog, UserLoginLog, OperateLog, PasswordChangeLog
|
from .models import FTPLog, UserLoginLog, OperateLog, PasswordChangeLog
|
||||||
from .serializers import FTPLogSerializer, UserLoginLogSerializer
|
from .serializers import FTPLogSerializer, UserLoginLogSerializer, JobAuditLogSerializer
|
||||||
from .serializers import (
|
from .serializers import (
|
||||||
OperateLogSerializer, OperateLogActionDetailSerializer,
|
OperateLogSerializer, OperateLogActionDetailSerializer,
|
||||||
PasswordChangeLogSerializer
|
PasswordChangeLogSerializer
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class JobAuditViewSet(OrgBulkModelViewSet):
|
||||||
|
serializer_class = JobAuditLogSerializer
|
||||||
|
http_method_names = ('get', 'head', 'options',)
|
||||||
|
permission_classes = ()
|
||||||
|
model = JobAuditLog
|
||||||
|
|
||||||
|
|
||||||
class FTPLogViewSet(CreateModelMixin, ListModelMixin, OrgGenericViewSet):
|
class FTPLogViewSet(CreateModelMixin, ListModelMixin, OrgGenericViewSet):
|
||||||
model = FTPLog
|
model = FTPLog
|
||||||
serializer_class = FTPLogSerializer
|
serializer_class = FTPLogSerializer
|
||||||
|
|
|
@ -4,6 +4,8 @@ from django.utils.translation import ugettext_lazy as _
|
||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
|
|
||||||
from common.drf.fields import LabeledChoiceField
|
from common.drf.fields import LabeledChoiceField
|
||||||
|
from ops.models.job import JobAuditLog
|
||||||
|
from ops.serializers.job import JobExecutionSerializer
|
||||||
from terminal.models import Session
|
from terminal.models import Session
|
||||||
from . import models
|
from . import models
|
||||||
from .const import (
|
from .const import (
|
||||||
|
@ -15,6 +17,17 @@ from .const import (
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class JobAuditLogSerializer(JobExecutionSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = JobAuditLog
|
||||||
|
read_only_fields = ["timedelta", "time_cost", 'is_finished', 'date_start',
|
||||||
|
'date_finished',
|
||||||
|
'date_created',
|
||||||
|
'is_success',
|
||||||
|
'creator_name']
|
||||||
|
fields = read_only_fields + []
|
||||||
|
|
||||||
|
|
||||||
class FTPLogSerializer(serializers.ModelSerializer):
|
class FTPLogSerializer(serializers.ModelSerializer):
|
||||||
operate = LabeledChoiceField(choices=OperateChoices.choices, label=_("Operate"))
|
operate = LabeledChoiceField(choices=OperateChoices.choices, label=_("Operate"))
|
||||||
|
|
||||||
|
|
|
@ -7,7 +7,6 @@ from rest_framework.routers import DefaultRouter
|
||||||
from common import api as capi
|
from common import api as capi
|
||||||
from .. import api
|
from .. import api
|
||||||
|
|
||||||
|
|
||||||
app_name = "audits"
|
app_name = "audits"
|
||||||
|
|
||||||
router = DefaultRouter()
|
router = DefaultRouter()
|
||||||
|
@ -15,9 +14,7 @@ router.register(r'ftp-logs', api.FTPLogViewSet, 'ftp-log')
|
||||||
router.register(r'login-logs', api.UserLoginLogViewSet, 'login-log')
|
router.register(r'login-logs', api.UserLoginLogViewSet, 'login-log')
|
||||||
router.register(r'operate-logs', api.OperateLogViewSet, 'operate-log')
|
router.register(r'operate-logs', api.OperateLogViewSet, 'operate-log')
|
||||||
router.register(r'password-change-logs', api.PasswordChangeLogViewSet, 'password-change-log')
|
router.register(r'password-change-logs', api.PasswordChangeLogViewSet, 'password-change-log')
|
||||||
# router.register(r'command-execution-logs', api.CommandExecutionViewSet, 'command-execution-log')
|
router.register(r'job-logs', api.JobAuditViewSet, 'job-log')
|
||||||
# router.register(r'command-executions-hosts-relations', api.CommandExecutionHostRelationViewSet, 'command-executions-hosts-relation')
|
|
||||||
|
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
path('my-login-logs/', api.MyLoginLogAPIView.as_view(), name='my-login-log'),
|
path('my-login-logs/', api.MyLoginLogAPIView.as_view(), name='my-login-log'),
|
||||||
|
|
|
@ -44,9 +44,12 @@ class BaseService(object):
|
||||||
if self.is_running:
|
if self.is_running:
|
||||||
msg = f'{self.name} is running: {self.pid}.'
|
msg = f'{self.name} is running: {self.pid}.'
|
||||||
else:
|
else:
|
||||||
msg = '\033[31m{} is stopped.\033[0m\nYou can manual start it to find the error: \n' \
|
msg = f'{self.name} is stopped.'
|
||||||
' $ cd {}\n' \
|
if DEBUG:
|
||||||
' $ {}'.format(self.name, self.cwd, ' '.join(self.cmd))
|
msg = '\033[31m{} is stopped.\033[0m\nYou can manual start it to find the error: \n' \
|
||||||
|
' $ cd {}\n' \
|
||||||
|
' $ {}'.format(self.name, self.cwd, ' '.join(self.cmd))
|
||||||
|
|
||||||
print(msg)
|
print(msg)
|
||||||
|
|
||||||
# -- log --
|
# -- log --
|
||||||
|
@ -147,7 +150,6 @@ class BaseService(object):
|
||||||
self.remove_pid()
|
self.remove_pid()
|
||||||
break
|
break
|
||||||
else:
|
else:
|
||||||
time.sleep(1)
|
|
||||||
continue
|
continue
|
||||||
|
|
||||||
def watch(self):
|
def watch(self):
|
||||||
|
@ -203,4 +205,3 @@ class BaseService(object):
|
||||||
logging.info(f'Remove old log: {to_delete_dir}')
|
logging.info(f'Remove old log: {to_delete_dir}')
|
||||||
shutil.rmtree(to_delete_dir, ignore_errors=True)
|
shutil.rmtree(to_delete_dir, ignore_errors=True)
|
||||||
# -- end action --
|
# -- end action --
|
||||||
|
|
||||||
|
|
|
@ -40,7 +40,8 @@ class ServicesUtil(object):
|
||||||
service: BaseService
|
service: BaseService
|
||||||
service.start()
|
service.start()
|
||||||
self.files_preserve_map[service.name] = service.log_file
|
self.files_preserve_map[service.name] = service.log_file
|
||||||
time.sleep(1)
|
|
||||||
|
time.sleep(1)
|
||||||
|
|
||||||
def stop(self):
|
def stop(self):
|
||||||
for service in self._services:
|
for service in self._services:
|
||||||
|
|
|
@ -10,7 +10,7 @@ __all__ = ['JMSInventory']
|
||||||
|
|
||||||
class JMSInventory:
|
class JMSInventory:
|
||||||
def __init__(self, assets, account_policy='privileged_first',
|
def __init__(self, assets, account_policy='privileged_first',
|
||||||
account_prefer='root,Administrator', host_callback=None, unique_host_name=False):
|
account_prefer='root,Administrator', host_callback=None):
|
||||||
"""
|
"""
|
||||||
:param assets:
|
:param assets:
|
||||||
:param account_prefer: account username name if not set use account_policy
|
:param account_prefer: account username name if not set use account_policy
|
||||||
|
@ -21,7 +21,6 @@ class JMSInventory:
|
||||||
self.account_policy = account_policy
|
self.account_policy = account_policy
|
||||||
self.host_callback = host_callback
|
self.host_callback = host_callback
|
||||||
self.exclude_hosts = {}
|
self.exclude_hosts = {}
|
||||||
self.unique_host_name = unique_host_name
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def clean_assets(assets):
|
def clean_assets(assets):
|
||||||
|
@ -114,8 +113,6 @@ class JMSInventory:
|
||||||
'secret': account.secret, 'secret_type': account.secret_type
|
'secret': account.secret, 'secret_type': account.secret_type
|
||||||
} if account else None
|
} if account else None
|
||||||
}
|
}
|
||||||
if self.unique_host_name:
|
|
||||||
host['name'] += '({})'.format(asset.id)
|
|
||||||
|
|
||||||
if host['jms_account'] and asset.platform.type == 'oracle':
|
if host['jms_account'] and asset.platform.type == 'oracle':
|
||||||
host['jms_account']['mode'] = 'sysdba' if account.privileged else None
|
host['jms_account']['mode'] = 'sysdba' if account.privileged else None
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
from .base import SelfBulkModelViewSet
|
from orgs.mixins.api import OrgBulkModelViewSet
|
||||||
from ..models import AdHoc
|
from ..models import AdHoc
|
||||||
from ..serializers import (
|
from ..serializers import (
|
||||||
AdHocSerializer
|
AdHocSerializer
|
||||||
|
@ -10,7 +10,11 @@ __all__ = [
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
class AdHocViewSet(SelfBulkModelViewSet):
|
class AdHocViewSet(OrgBulkModelViewSet):
|
||||||
serializer_class = AdHocSerializer
|
serializer_class = AdHocSerializer
|
||||||
permission_classes = ()
|
permission_classes = ()
|
||||||
model = AdHoc
|
model = AdHoc
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
queryset = super().get_queryset()
|
||||||
|
return queryset.filter(creator=self.request.user)
|
||||||
|
|
|
@ -1,17 +0,0 @@
|
||||||
from rest_framework_bulk import BulkModelViewSet
|
|
||||||
|
|
||||||
from common.mixins import CommonApiMixin
|
|
||||||
|
|
||||||
__all__ = ['SelfBulkModelViewSet']
|
|
||||||
|
|
||||||
|
|
||||||
class SelfBulkModelViewSet(CommonApiMixin, BulkModelViewSet):
|
|
||||||
|
|
||||||
def get_queryset(self):
|
|
||||||
if hasattr(self, 'model'):
|
|
||||||
return self.model.objects.filter(creator=self.request.user)
|
|
||||||
else:
|
|
||||||
assert self.queryset is None, (
|
|
||||||
"'%s' should not include a `queryset` attribute"
|
|
||||||
% self.__class__.__name__
|
|
||||||
)
|
|
|
@ -2,14 +2,15 @@ from rest_framework.views import APIView
|
||||||
from django.shortcuts import get_object_or_404
|
from django.shortcuts import get_object_or_404
|
||||||
from rest_framework.response import Response
|
from rest_framework.response import Response
|
||||||
|
|
||||||
from ops.api.base import SelfBulkModelViewSet
|
|
||||||
from ops.models import Job, JobExecution
|
from ops.models import Job, JobExecution
|
||||||
|
from ops.models.job import JobAuditLog
|
||||||
from ops.serializers.job import JobSerializer, JobExecutionSerializer
|
from ops.serializers.job import JobSerializer, JobExecutionSerializer
|
||||||
|
|
||||||
__all__ = ['JobViewSet', 'JobExecutionViewSet', 'JobRunVariableHelpAPIView', 'JobAssetDetail']
|
__all__ = ['JobViewSet', 'JobExecutionViewSet', 'JobRunVariableHelpAPIView', 'JobAssetDetail', ]
|
||||||
|
|
||||||
from ops.tasks import run_ops_job_execution
|
from ops.tasks import run_ops_job_execution
|
||||||
from ops.variables import JMS_JOB_VARIABLE_HELP
|
from ops.variables import JMS_JOB_VARIABLE_HELP
|
||||||
|
from orgs.mixins.api import OrgBulkModelViewSet
|
||||||
|
|
||||||
|
|
||||||
def set_task_to_serializer_data(serializer, task):
|
def set_task_to_serializer_data(serializer, task):
|
||||||
|
@ -18,16 +19,17 @@ def set_task_to_serializer_data(serializer, task):
|
||||||
setattr(serializer, "_data", data)
|
setattr(serializer, "_data", data)
|
||||||
|
|
||||||
|
|
||||||
class JobViewSet(SelfBulkModelViewSet):
|
class JobViewSet(OrgBulkModelViewSet):
|
||||||
serializer_class = JobSerializer
|
serializer_class = JobSerializer
|
||||||
permission_classes = ()
|
permission_classes = ()
|
||||||
model = Job
|
model = Job
|
||||||
|
|
||||||
def get_queryset(self):
|
def get_queryset(self):
|
||||||
query_set = super().get_queryset()
|
queryset = super().get_queryset()
|
||||||
|
queryset = queryset.filter(creator=self.request.user)
|
||||||
if self.action != 'retrieve':
|
if self.action != 'retrieve':
|
||||||
return query_set.filter(instant=False)
|
return queryset.filter(instant=False)
|
||||||
return query_set
|
return queryset
|
||||||
|
|
||||||
def perform_create(self, serializer):
|
def perform_create(self, serializer):
|
||||||
instance = serializer.save()
|
instance = serializer.save()
|
||||||
|
@ -48,7 +50,7 @@ class JobViewSet(SelfBulkModelViewSet):
|
||||||
set_task_to_serializer_data(serializer, task)
|
set_task_to_serializer_data(serializer, task)
|
||||||
|
|
||||||
|
|
||||||
class JobExecutionViewSet(SelfBulkModelViewSet):
|
class JobExecutionViewSet(OrgBulkModelViewSet):
|
||||||
serializer_class = JobExecutionSerializer
|
serializer_class = JobExecutionSerializer
|
||||||
http_method_names = ('get', 'post', 'head', 'options',)
|
http_method_names = ('get', 'post', 'head', 'options',)
|
||||||
permission_classes = ()
|
permission_classes = ()
|
||||||
|
@ -60,11 +62,12 @@ class JobExecutionViewSet(SelfBulkModelViewSet):
|
||||||
set_task_to_serializer_data(serializer, task)
|
set_task_to_serializer_data(serializer, task)
|
||||||
|
|
||||||
def get_queryset(self):
|
def get_queryset(self):
|
||||||
query_set = super().get_queryset()
|
queryset = super().get_queryset()
|
||||||
|
queryset = queryset.filter(creator=self.request.user)
|
||||||
job_id = self.request.query_params.get('job_id')
|
job_id = self.request.query_params.get('job_id')
|
||||||
if job_id:
|
if job_id:
|
||||||
query_set = query_set.filter(job_id=job_id)
|
queryset = queryset.filter(job_id=job_id)
|
||||||
return query_set
|
return queryset
|
||||||
|
|
||||||
|
|
||||||
class JobRunVariableHelpAPIView(APIView):
|
class JobRunVariableHelpAPIView(APIView):
|
||||||
|
|
|
@ -2,11 +2,7 @@ import os
|
||||||
import zipfile
|
import zipfile
|
||||||
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from rest_framework_bulk import BulkModelViewSet
|
|
||||||
|
|
||||||
from common.mixins import CommonApiMixin
|
|
||||||
from orgs.mixins.api import OrgBulkModelViewSet
|
from orgs.mixins.api import OrgBulkModelViewSet
|
||||||
from .base import SelfBulkModelViewSet
|
|
||||||
from ..exception import PlaybookNoValidEntry
|
from ..exception import PlaybookNoValidEntry
|
||||||
from ..models import Playbook
|
from ..models import Playbook
|
||||||
from ..serializers.playbook import PlaybookSerializer
|
from ..serializers.playbook import PlaybookSerializer
|
||||||
|
@ -20,11 +16,16 @@ def unzip_playbook(src, dist):
|
||||||
fz.extract(file, dist)
|
fz.extract(file, dist)
|
||||||
|
|
||||||
|
|
||||||
class PlaybookViewSet(SelfBulkModelViewSet):
|
class PlaybookViewSet(OrgBulkModelViewSet):
|
||||||
serializer_class = PlaybookSerializer
|
serializer_class = PlaybookSerializer
|
||||||
permission_classes = ()
|
permission_classes = ()
|
||||||
model = Playbook
|
model = Playbook
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
queryset = super().get_queryset()
|
||||||
|
queryset = queryset.filter(creator=self.request.user)
|
||||||
|
return queryset
|
||||||
|
|
||||||
def perform_create(self, serializer):
|
def perform_create(self, serializer):
|
||||||
instance = serializer.save()
|
instance = serializer.save()
|
||||||
src_path = os.path.join(settings.MEDIA_ROOT, instance.path.name)
|
src_path = os.path.join(settings.MEDIA_ROOT, instance.path.name)
|
||||||
|
|
|
@ -0,0 +1,33 @@
|
||||||
|
# Generated by Django 3.2.14 on 2022-12-15 09:12
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('ops', '0028_auto_20221205_1627'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='adhoc',
|
||||||
|
name='org_id',
|
||||||
|
field=models.CharField(blank=True, db_index=True, default='', max_length=36, verbose_name='Organization'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='job',
|
||||||
|
name='org_id',
|
||||||
|
field=models.CharField(blank=True, db_index=True, default='', max_length=36, verbose_name='Organization'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='jobexecution',
|
||||||
|
name='org_id',
|
||||||
|
field=models.CharField(blank=True, db_index=True, default='', max_length=36, verbose_name='Organization'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='playbook',
|
||||||
|
name='org_id',
|
||||||
|
field=models.CharField(blank=True, db_index=True, default='', max_length=36, verbose_name='Organization'),
|
||||||
|
),
|
||||||
|
]
|
|
@ -9,10 +9,12 @@ from common.utils import get_logger
|
||||||
|
|
||||||
__all__ = ["AdHoc"]
|
__all__ = ["AdHoc"]
|
||||||
|
|
||||||
|
from orgs.mixins.models import JMSOrgBaseModel
|
||||||
|
|
||||||
logger = get_logger(__file__)
|
logger = get_logger(__file__)
|
||||||
|
|
||||||
|
|
||||||
class AdHoc(JMSBaseModel):
|
class AdHoc(JMSOrgBaseModel):
|
||||||
class Modules(models.TextChoices):
|
class Modules(models.TextChoices):
|
||||||
shell = 'shell', _('Shell')
|
shell = 'shell', _('Shell')
|
||||||
winshell = 'win_shell', _('Powershell')
|
winshell = 'win_shell', _('Powershell')
|
||||||
|
|
|
@ -9,15 +9,15 @@ from django.utils.translation import gettext_lazy as _
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from celery import current_task
|
from celery import current_task
|
||||||
|
|
||||||
__all__ = ["Job", "JobExecution"]
|
__all__ = ["Job", "JobExecution", "JobAuditLog"]
|
||||||
|
|
||||||
from common.db.models import JMSBaseModel
|
|
||||||
from ops.ansible import JMSInventory, AdHocRunner, PlaybookRunner
|
from ops.ansible import JMSInventory, AdHocRunner, PlaybookRunner
|
||||||
from ops.mixin import PeriodTaskModelMixin
|
from ops.mixin import PeriodTaskModelMixin
|
||||||
from ops.variables import *
|
from ops.variables import *
|
||||||
|
from orgs.mixins.models import JMSOrgBaseModel
|
||||||
|
|
||||||
|
|
||||||
class Job(JMSBaseModel, PeriodTaskModelMixin):
|
class Job(JMSOrgBaseModel, PeriodTaskModelMixin):
|
||||||
class Types(models.TextChoices):
|
class Types(models.TextChoices):
|
||||||
adhoc = 'adhoc', _('Adhoc')
|
adhoc = 'adhoc', _('Adhoc')
|
||||||
playbook = 'playbook', _('Playbook')
|
playbook = 'playbook', _('Playbook')
|
||||||
|
@ -88,7 +88,7 @@ class Job(JMSBaseModel, PeriodTaskModelMixin):
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def inventory(self):
|
def inventory(self):
|
||||||
return JMSInventory(self.assets.all(), self.runas_policy, self.runas, unique_host_name=True)
|
return JMSInventory(self.assets.all(), self.runas_policy, self.runas)
|
||||||
|
|
||||||
def create_execution(self):
|
def create_execution(self):
|
||||||
return self.executions.create()
|
return self.executions.create()
|
||||||
|
@ -97,7 +97,7 @@ class Job(JMSBaseModel, PeriodTaskModelMixin):
|
||||||
ordering = ['date_created']
|
ordering = ['date_created']
|
||||||
|
|
||||||
|
|
||||||
class JobExecution(JMSBaseModel):
|
class JobExecution(JMSOrgBaseModel):
|
||||||
id = models.UUIDField(default=uuid.uuid4, primary_key=True)
|
id = models.UUIDField(default=uuid.uuid4, primary_key=True)
|
||||||
task_id = models.UUIDField(null=True)
|
task_id = models.UUIDField(null=True)
|
||||||
status = models.CharField(max_length=16, verbose_name=_('Status'), default='running')
|
status = models.CharField(max_length=16, verbose_name=_('Status'), default='running')
|
||||||
|
@ -133,26 +133,25 @@ class JobExecution(JMSBaseModel):
|
||||||
"status": "ok",
|
"status": "ok",
|
||||||
"tasks": [],
|
"tasks": [],
|
||||||
}
|
}
|
||||||
host_name = "{}({})".format(asset.name, asset.id)
|
if self.summary["excludes"].get(asset.name, None):
|
||||||
if self.summary["excludes"].get(host_name, None):
|
|
||||||
asset_detail.update({"status": "excludes"})
|
asset_detail.update({"status": "excludes"})
|
||||||
result["detail"].append(asset_detail)
|
result["detail"].append(asset_detail)
|
||||||
break
|
break
|
||||||
if self.result["dark"].get(host_name, None):
|
if self.result["dark"].get(asset.name, None):
|
||||||
asset_detail.update({"status": "failed"})
|
asset_detail.update({"status": "failed"})
|
||||||
for key, task in self.result["dark"][host_name].items():
|
for key, task in self.result["dark"][asset.name].items():
|
||||||
task_detail = {"name": key,
|
task_detail = {"name": key,
|
||||||
"output": "{}{}".format(task.get("stdout", ""), task.get("stderr", ""))}
|
"output": "{}{}".format(task.get("stdout", ""), task.get("stderr", ""))}
|
||||||
asset_detail["tasks"].append(task_detail)
|
asset_detail["tasks"].append(task_detail)
|
||||||
if self.result["failures"].get(host_name, None):
|
if self.result["failures"].get(asset.name, None):
|
||||||
asset_detail.update({"status": "failed"})
|
asset_detail.update({"status": "failed"})
|
||||||
for key, task in self.result["failures"][host_name].items():
|
for key, task in self.result["failures"][asset.name].items():
|
||||||
task_detail = {"name": key,
|
task_detail = {"name": key,
|
||||||
"output": "{}{}".format(task.get("stdout", ""), task.get("stderr", ""))}
|
"output": "{}{}".format(task.get("stdout", ""), task.get("stderr", ""))}
|
||||||
asset_detail["tasks"].append(task_detail)
|
asset_detail["tasks"].append(task_detail)
|
||||||
|
|
||||||
if self.result["ok"].get(host_name, None):
|
if self.result["ok"].get(asset.name, None):
|
||||||
for key, task in self.result["ok"][host_name].items():
|
for key, task in self.result["ok"][asset.name].items():
|
||||||
task_detail = {"name": key,
|
task_detail = {"name": key,
|
||||||
"output": "{}{}".format(task.get("stdout", ""), task.get("stderr", ""))}
|
"output": "{}{}".format(task.get("stdout", ""), task.get("stderr", ""))}
|
||||||
asset_detail["tasks"].append(task_detail)
|
asset_detail["tasks"].append(task_detail)
|
||||||
|
@ -202,10 +201,11 @@ class JobExecution(JMSBaseModel):
|
||||||
|
|
||||||
def gather_static_variables(self):
|
def gather_static_variables(self):
|
||||||
default = {
|
default = {
|
||||||
JMS_USERNAME: self.creator.username,
|
JMS_JOB_ID: str(self.job.id),
|
||||||
JMS_JOB_ID: self.job.id,
|
|
||||||
JMS_JOB_NAME: self.job.name,
|
JMS_JOB_NAME: self.job.name,
|
||||||
}
|
}
|
||||||
|
if self.creator:
|
||||||
|
default.update({JMS_USERNAME: self.creator.username})
|
||||||
return default
|
return default
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -255,7 +255,10 @@ class JobExecution(JMSBaseModel):
|
||||||
this = self.__class__.objects.get(id=self.id)
|
this = self.__class__.objects.get(id=self.id)
|
||||||
this.status = status_mapper.get(cb.status, cb.status)
|
this.status = status_mapper.get(cb.status, cb.status)
|
||||||
this.summary.update(cb.summary)
|
this.summary.update(cb.summary)
|
||||||
this.result.update(cb.result)
|
if this.result:
|
||||||
|
this.result.update(cb.result)
|
||||||
|
else:
|
||||||
|
this.result = cb.result
|
||||||
this.finish_task()
|
this.finish_task()
|
||||||
|
|
||||||
def finish_task(self):
|
def finish_task(self):
|
||||||
|
@ -283,3 +286,12 @@ class JobExecution(JMSBaseModel):
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
ordering = ['-date_created']
|
ordering = ['-date_created']
|
||||||
|
|
||||||
|
|
||||||
|
class JobAuditLog(JobExecution):
|
||||||
|
@property
|
||||||
|
def creator_name(self):
|
||||||
|
return self.creator.name
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
proxy = True
|
||||||
|
|
|
@ -5,11 +5,11 @@ from django.conf import settings
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
from common.db.models import JMSBaseModel
|
|
||||||
from ops.exception import PlaybookNoValidEntry
|
from ops.exception import PlaybookNoValidEntry
|
||||||
|
from orgs.mixins.models import JMSOrgBaseModel
|
||||||
|
|
||||||
|
|
||||||
class Playbook(JMSBaseModel):
|
class Playbook(JMSOrgBaseModel):
|
||||||
id = models.UUIDField(default=uuid.uuid4, primary_key=True)
|
id = models.UUIDField(default=uuid.uuid4, primary_key=True)
|
||||||
name = models.CharField(max_length=128, verbose_name=_('Name'), null=True)
|
name = models.CharField(max_length=128, verbose_name=_('Name'), null=True)
|
||||||
path = models.FileField(upload_to='playbooks/')
|
path = models.FileField(upload_to='playbooks/')
|
||||||
|
|
|
@ -4,10 +4,11 @@ from __future__ import unicode_literals
|
||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
|
|
||||||
from common.drf.fields import ReadableHiddenField
|
from common.drf.fields import ReadableHiddenField
|
||||||
|
from orgs.mixins.serializers import BulkOrgResourceModelSerializer
|
||||||
from ..models import AdHoc
|
from ..models import AdHoc
|
||||||
|
|
||||||
|
|
||||||
class AdHocSerializer(serializers.ModelSerializer):
|
class AdHocSerializer(BulkOrgResourceModelSerializer):
|
||||||
creator = ReadableHiddenField(default=serializers.CurrentUserDefault())
|
creator = ReadableHiddenField(default=serializers.CurrentUserDefault())
|
||||||
row_count = serializers.IntegerField(read_only=True)
|
row_count = serializers.IntegerField(read_only=True)
|
||||||
size = serializers.IntegerField(read_only=True)
|
size = serializers.IntegerField(read_only=True)
|
||||||
|
|
|
@ -3,9 +3,11 @@ from rest_framework import serializers
|
||||||
from common.drf.fields import ReadableHiddenField
|
from common.drf.fields import ReadableHiddenField
|
||||||
from ops.mixin import PeriodTaskSerializerMixin
|
from ops.mixin import PeriodTaskSerializerMixin
|
||||||
from ops.models import Job, JobExecution
|
from ops.models import Job, JobExecution
|
||||||
|
from ops.models.job import JobAuditLog
|
||||||
|
from orgs.mixins.serializers import BulkOrgResourceModelSerializer
|
||||||
|
|
||||||
|
|
||||||
class JobSerializer(serializers.ModelSerializer, PeriodTaskSerializerMixin):
|
class JobSerializer(BulkOrgResourceModelSerializer, PeriodTaskSerializerMixin):
|
||||||
creator = ReadableHiddenField(default=serializers.CurrentUserDefault())
|
creator = ReadableHiddenField(default=serializers.CurrentUserDefault())
|
||||||
run_after_save = serializers.BooleanField(label=_("Run after save"), read_only=True, default=False, required=False)
|
run_after_save = serializers.BooleanField(label=_("Run after save"), read_only=True, default=False, required=False)
|
||||||
|
|
||||||
|
@ -25,7 +27,7 @@ class JobSerializer(serializers.ModelSerializer, PeriodTaskSerializerMixin):
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
class JobExecutionSerializer(serializers.ModelSerializer):
|
class JobExecutionSerializer(BulkOrgResourceModelSerializer):
|
||||||
creator = ReadableHiddenField(default=serializers.CurrentUserDefault())
|
creator = ReadableHiddenField(default=serializers.CurrentUserDefault())
|
||||||
job_type = serializers.ReadOnlyField(label=_("Job type"))
|
job_type = serializers.ReadOnlyField(label=_("Job type"))
|
||||||
count = serializers.ReadOnlyField(label=_("Count"))
|
count = serializers.ReadOnlyField(label=_("Count"))
|
||||||
|
@ -39,3 +41,6 @@ class JobExecutionSerializer(serializers.ModelSerializer):
|
||||||
fields = read_only_fields + [
|
fields = read_only_fields + [
|
||||||
"job", "parameters"
|
"job", "parameters"
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -12,7 +12,7 @@ def parse_playbook_name(path):
|
||||||
return file_name.split(".")[-2]
|
return file_name.split(".")[-2]
|
||||||
|
|
||||||
|
|
||||||
class PlaybookSerializer(serializers.ModelSerializer):
|
class PlaybookSerializer(BulkOrgResourceModelSerializer):
|
||||||
creator = ReadableHiddenField(default=serializers.CurrentUserDefault())
|
creator = ReadableHiddenField(default=serializers.CurrentUserDefault())
|
||||||
path = serializers.FileField(required=False)
|
path = serializers.FileField(required=False)
|
||||||
|
|
||||||
|
|
|
@ -36,7 +36,8 @@ def run_ops_job(job_id):
|
||||||
def run_ops_job_execution(execution_id, **kwargs):
|
def run_ops_job_execution(execution_id, **kwargs):
|
||||||
execution = get_object_or_none(JobExecution, id=execution_id)
|
execution = get_object_or_none(JobExecution, id=execution_id)
|
||||||
try:
|
try:
|
||||||
execution.start()
|
with tmp_to_org(execution.org):
|
||||||
|
execution.start()
|
||||||
except SoftTimeLimitExceeded:
|
except SoftTimeLimitExceeded:
|
||||||
execution.set_error('Run timeout')
|
execution.set_error('Run timeout')
|
||||||
logger.error("Run adhoc timeout")
|
logger.error("Run adhoc timeout")
|
||||||
|
|
|
@ -158,7 +158,7 @@ class UserGrantedK8sAsTreeApi(SelfOrPKUserMixin, ListAPIView):
|
||||||
asset_id = parent_info.get('asset_id')
|
asset_id = parent_info.get('asset_id')
|
||||||
asset_id = tree_id if not asset_id else asset_id
|
asset_id = tree_id if not asset_id else asset_id
|
||||||
|
|
||||||
if tree_id and not account_username:
|
if tree_id and not key and not account_username:
|
||||||
asset = self.asset(asset_id)
|
asset = self.asset(asset_id)
|
||||||
accounts = self.get_accounts(asset)
|
accounts = self.get_accounts(asset)
|
||||||
asset_node = KubernetesTree(tree_id).as_asset_tree_node(asset)
|
asset_node = KubernetesTree(tree_id).as_asset_tree_node(asset)
|
||||||
|
@ -168,6 +168,6 @@ class UserGrantedK8sAsTreeApi(SelfOrPKUserMixin, ListAPIView):
|
||||||
account, parent_info,
|
account, parent_info,
|
||||||
)
|
)
|
||||||
tree.append(account_node)
|
tree.append(account_node)
|
||||||
else:
|
elif key and account_username:
|
||||||
tree = KubernetesTree(key).async_tree_node(parent_info)
|
tree = KubernetesTree(key).async_tree_node(parent_info)
|
||||||
return Response(data=tree)
|
return Response(data=tree)
|
||||||
|
|
Loading…
Reference in New Issue