2022-11-11 11:20:17 +00:00
|
|
|
import os
|
2023-01-18 10:03:33 +00:00
|
|
|
import shutil
|
2022-11-11 11:20:17 +00:00
|
|
|
import zipfile
|
|
|
|
|
|
|
|
from django.conf import settings
|
2023-09-19 10:04:24 +00:00
|
|
|
from django.core.exceptions import SuspiciousFileOperation
|
2023-01-18 10:03:33 +00:00
|
|
|
from django.shortcuts import get_object_or_404
|
2023-07-24 03:52:25 +00:00
|
|
|
from django.utils.translation import gettext_lazy as _
|
2023-02-13 07:39:29 +00:00
|
|
|
from rest_framework import status
|
2023-01-18 10:03:33 +00:00
|
|
|
|
2023-02-21 14:52:52 +00:00
|
|
|
from common.exceptions import JMSException
|
2022-11-29 11:44:12 +00:00
|
|
|
from orgs.mixins.api import OrgBulkModelViewSet
|
2023-02-22 12:04:00 +00:00
|
|
|
from rbac.permissions import RBACPermission
|
2022-12-02 04:21:56 +00:00
|
|
|
from ..exception import PlaybookNoValidEntry
|
2022-11-11 11:20:17 +00:00
|
|
|
from ..models import Playbook
|
|
|
|
from ..serializers.playbook import PlaybookSerializer
|
|
|
|
|
2023-01-18 10:03:33 +00:00
|
|
|
__all__ = ["PlaybookViewSet", "PlaybookFileBrowserAPIView"]
|
|
|
|
|
|
|
|
from rest_framework.views import APIView
|
|
|
|
from rest_framework.response import Response
|
2023-09-19 10:04:24 +00:00
|
|
|
from django.utils._os import safe_join
|
2022-11-11 11:20:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
def unzip_playbook(src, dist):
|
|
|
|
fz = zipfile.ZipFile(src, 'r')
|
|
|
|
for file in fz.namelist():
|
|
|
|
fz.extract(file, dist)
|
|
|
|
|
|
|
|
|
2022-12-15 09:25:21 +00:00
|
|
|
class PlaybookViewSet(OrgBulkModelViewSet):
|
2022-11-11 11:20:17 +00:00
|
|
|
serializer_class = PlaybookSerializer
|
2023-02-22 12:04:00 +00:00
|
|
|
permission_classes = (RBACPermission,)
|
2022-11-29 11:44:12 +00:00
|
|
|
model = Playbook
|
2023-02-09 09:44:35 +00:00
|
|
|
search_fields = ('name', 'comment')
|
2022-11-11 11:20:17 +00:00
|
|
|
|
2023-03-14 09:37:02 +00:00
|
|
|
def perform_destroy(self, instance):
|
|
|
|
if instance.job_set.exists():
|
|
|
|
raise JMSException(code='playbook_has_job', detail={"msg": _("Currently playbook is being used in a job")})
|
|
|
|
instance_id = instance.id
|
|
|
|
super().perform_destroy(instance)
|
2023-09-19 10:04:24 +00:00
|
|
|
dest_path = safe_join(settings.DATA_DIR, "ops", "playbook", instance_id.__str__())
|
2023-03-14 09:37:02 +00:00
|
|
|
shutil.rmtree(dest_path)
|
|
|
|
|
2022-12-15 09:25:21 +00:00
|
|
|
def get_queryset(self):
|
|
|
|
queryset = super().get_queryset()
|
|
|
|
queryset = queryset.filter(creator=self.request.user)
|
|
|
|
return queryset
|
|
|
|
|
2022-11-11 11:20:17 +00:00
|
|
|
def perform_create(self, serializer):
|
|
|
|
instance = serializer.save()
|
2023-02-09 08:16:20 +00:00
|
|
|
if 'multipart/form-data' in self.request.headers['Content-Type']:
|
2023-09-19 10:04:24 +00:00
|
|
|
src_path = safe_join(settings.MEDIA_ROOT, instance.path.name)
|
|
|
|
dest_path = safe_join(settings.DATA_DIR, "ops", "playbook", instance.id.__str__())
|
2024-03-18 06:19:15 +00:00
|
|
|
|
2023-02-21 14:52:52 +00:00
|
|
|
try:
|
|
|
|
unzip_playbook(src_path, dest_path)
|
2024-03-18 06:19:15 +00:00
|
|
|
except RuntimeError:
|
2023-02-21 14:52:52 +00:00
|
|
|
raise JMSException(code='invalid_playbook_file', detail={"msg": "Unzip failed"})
|
|
|
|
|
2023-02-09 08:16:20 +00:00
|
|
|
if 'main.yml' not in os.listdir(dest_path):
|
|
|
|
raise PlaybookNoValidEntry
|
|
|
|
|
2024-03-18 06:19:15 +00:00
|
|
|
elif instance.create_method == 'blank':
|
|
|
|
dest_path = safe_join(settings.DATA_DIR, "ops", "playbook", instance.id.__str__())
|
|
|
|
os.makedirs(dest_path)
|
|
|
|
with open(safe_join(dest_path, 'main.yml'), 'w') as f:
|
|
|
|
f.write('## write your playbook here')
|
2023-01-18 10:03:33 +00:00
|
|
|
|
|
|
|
|
|
|
|
class PlaybookFileBrowserAPIView(APIView):
|
2023-02-22 12:04:00 +00:00
|
|
|
permission_classes = (RBACPermission,)
|
|
|
|
rbac_perms = {
|
2024-03-18 06:19:15 +00:00
|
|
|
'GET': 'ops.view_playbook',
|
2023-03-01 10:27:29 +00:00
|
|
|
'POST': 'ops.change_playbook',
|
|
|
|
'DELETE': 'ops.change_playbook',
|
|
|
|
'PATCH': 'ops.change_playbook',
|
2023-02-22 12:04:00 +00:00
|
|
|
}
|
2023-02-09 07:48:42 +00:00
|
|
|
protected_files = ['root', 'main.yml']
|
|
|
|
|
2024-03-14 02:22:09 +00:00
|
|
|
def get_playbook(self, playbook_id):
|
|
|
|
playbook = get_object_or_404(Playbook, id=playbook_id, creator=self.request.user)
|
|
|
|
return playbook
|
|
|
|
|
2023-01-18 10:03:33 +00:00
|
|
|
def get(self, request, **kwargs):
|
|
|
|
playbook_id = kwargs.get('pk')
|
2024-03-14 02:22:09 +00:00
|
|
|
playbook = self.get_playbook(playbook_id)
|
2023-01-18 10:03:33 +00:00
|
|
|
work_path = playbook.work_dir
|
|
|
|
file_key = request.query_params.get('key', '')
|
|
|
|
if file_key:
|
2023-09-19 10:04:24 +00:00
|
|
|
try:
|
|
|
|
file_path = safe_join(work_path, file_key)
|
|
|
|
with open(file_path, 'r') as f:
|
2023-02-22 13:27:01 +00:00
|
|
|
content = f.read()
|
2023-09-19 10:04:24 +00:00
|
|
|
except UnicodeDecodeError:
|
|
|
|
content = _('Unsupported file content')
|
|
|
|
except SuspiciousFileOperation:
|
|
|
|
raise JMSException(code='invalid_file_path', detail={"msg": _("Invalid file path")})
|
|
|
|
return Response({'content': content})
|
2023-01-18 10:03:33 +00:00
|
|
|
else:
|
|
|
|
expand_key = request.query_params.get('expand', '')
|
|
|
|
nodes = self.generate_tree(playbook, work_path, expand_key)
|
|
|
|
return Response(nodes)
|
|
|
|
|
|
|
|
def post(self, request, **kwargs):
|
|
|
|
playbook_id = kwargs.get('pk')
|
2024-03-14 02:22:09 +00:00
|
|
|
playbook = self.get_playbook(playbook_id)
|
2023-01-18 10:03:33 +00:00
|
|
|
work_path = playbook.work_dir
|
|
|
|
|
|
|
|
parent_key = request.data.get('key', '')
|
|
|
|
if parent_key == 'root':
|
|
|
|
parent_key = ''
|
|
|
|
if os.path.dirname(parent_key) == 'root':
|
|
|
|
parent_key = os.path.basename(parent_key)
|
2023-09-19 10:04:24 +00:00
|
|
|
|
|
|
|
full_path = safe_join(work_path, parent_key)
|
2023-01-18 10:03:33 +00:00
|
|
|
|
|
|
|
is_directory = request.data.get('is_directory', False)
|
|
|
|
content = request.data.get('content', '')
|
2023-02-02 03:34:47 +00:00
|
|
|
name = request.data.get('name', '')
|
|
|
|
|
|
|
|
def find_new_name(p, is_file=False):
|
|
|
|
if not p:
|
|
|
|
if is_file:
|
|
|
|
p = 'new_file.yml'
|
|
|
|
else:
|
|
|
|
p = 'new_dir'
|
2023-09-19 10:04:24 +00:00
|
|
|
np = safe_join(full_path, p)
|
2023-02-02 03:34:47 +00:00
|
|
|
n = 0
|
|
|
|
while os.path.exists(np):
|
|
|
|
n += 1
|
2023-09-19 10:04:24 +00:00
|
|
|
np = safe_join(full_path, '{}({})'.format(p, n))
|
2023-02-02 03:34:47 +00:00
|
|
|
return np
|
2023-01-18 10:03:33 +00:00
|
|
|
|
2023-09-19 10:04:24 +00:00
|
|
|
try:
|
|
|
|
if is_directory:
|
|
|
|
new_file_path = find_new_name(name)
|
|
|
|
os.makedirs(new_file_path)
|
|
|
|
else:
|
|
|
|
new_file_path = find_new_name(name, True)
|
|
|
|
with open(new_file_path, 'w') as f:
|
|
|
|
f.write(content)
|
|
|
|
except SuspiciousFileOperation:
|
|
|
|
raise JMSException(code='invalid_file_path', detail={"msg": _("Invalid file path")})
|
2023-01-18 10:03:33 +00:00
|
|
|
|
|
|
|
relative_path = os.path.relpath(os.path.dirname(new_file_path), work_path)
|
|
|
|
new_node = {
|
|
|
|
"name": os.path.basename(new_file_path),
|
|
|
|
"title": os.path.basename(new_file_path),
|
2023-09-19 10:04:24 +00:00
|
|
|
"id": safe_join(relative_path, os.path.basename(new_file_path))
|
|
|
|
if not safe_join(relative_path, os.path.basename(new_file_path)).startswith('.')
|
2023-01-18 10:03:33 +00:00
|
|
|
else os.path.basename(new_file_path),
|
|
|
|
"isParent": is_directory,
|
|
|
|
"pId": relative_path if not relative_path.startswith('.') else 'root',
|
|
|
|
"open": True,
|
|
|
|
}
|
|
|
|
if not is_directory:
|
|
|
|
new_node['iconSkin'] = 'file'
|
|
|
|
return Response(new_node)
|
|
|
|
|
|
|
|
def patch(self, request, **kwargs):
|
|
|
|
playbook_id = kwargs.get('pk')
|
2024-03-14 02:22:09 +00:00
|
|
|
playbook = self.get_playbook(playbook_id)
|
2023-01-18 10:03:33 +00:00
|
|
|
work_path = playbook.work_dir
|
|
|
|
|
|
|
|
file_key = request.data.get('key', '')
|
2023-02-13 07:39:29 +00:00
|
|
|
new_name = request.data.get('new_name', '')
|
2023-02-09 07:48:42 +00:00
|
|
|
|
2023-02-13 07:39:29 +00:00
|
|
|
if file_key in self.protected_files and new_name:
|
2023-09-19 10:04:24 +00:00
|
|
|
raise JMSException(code='this_file_can_not_be_rename', detail={"msg": _("This file can not be rename")})
|
2023-02-09 07:48:42 +00:00
|
|
|
|
2023-01-18 10:03:33 +00:00
|
|
|
if os.path.dirname(file_key) == 'root':
|
|
|
|
file_key = os.path.basename(file_key)
|
|
|
|
|
|
|
|
content = request.data.get('content', '')
|
|
|
|
is_directory = request.data.get('is_directory', False)
|
|
|
|
|
|
|
|
if not file_key or file_key == 'root':
|
2023-02-13 07:39:29 +00:00
|
|
|
return Response(status=status.HTTP_400_BAD_REQUEST)
|
2023-09-19 10:04:24 +00:00
|
|
|
file_path = safe_join(work_path, file_key)
|
2023-01-18 10:03:33 +00:00
|
|
|
|
2023-02-21 07:22:08 +00:00
|
|
|
# rename
|
2023-01-18 10:03:33 +00:00
|
|
|
if new_name:
|
2023-09-19 10:04:24 +00:00
|
|
|
try:
|
|
|
|
new_file_path = safe_join(os.path.dirname(file_path), new_name)
|
|
|
|
if new_file_path == file_path:
|
|
|
|
return Response(status=status.HTTP_200_OK)
|
|
|
|
if os.path.exists(new_file_path):
|
|
|
|
raise JMSException(code='file_already exists', detail={"msg": _("File already exists")})
|
|
|
|
os.rename(file_path, new_file_path)
|
|
|
|
except SuspiciousFileOperation:
|
|
|
|
raise JMSException(code='invalid_file_path', detail={"msg": _("Invalid file path")})
|
|
|
|
|
2023-02-21 07:22:08 +00:00
|
|
|
# edit content
|
|
|
|
else:
|
|
|
|
if not is_directory:
|
|
|
|
with open(file_path, 'w') as f:
|
|
|
|
f.write(content)
|
2023-02-22 13:27:01 +00:00
|
|
|
return Response(status=status.HTTP_200_OK)
|
2023-01-18 10:03:33 +00:00
|
|
|
|
|
|
|
def delete(self, request, **kwargs):
|
|
|
|
playbook_id = kwargs.get('pk')
|
2024-03-14 02:22:09 +00:00
|
|
|
playbook = self.get_playbook(playbook_id)
|
2023-01-18 10:03:33 +00:00
|
|
|
work_path = playbook.work_dir
|
|
|
|
file_key = request.query_params.get('key', '')
|
|
|
|
if not file_key:
|
2023-09-19 10:04:24 +00:00
|
|
|
raise JMSException(code='file_key_is_required', detail={"msg": _("File key is required")})
|
|
|
|
|
2023-02-09 07:48:42 +00:00
|
|
|
if file_key in self.protected_files:
|
2023-09-19 10:04:24 +00:00
|
|
|
raise JMSException(code='This file can not be delete', detail={"msg": _("This file can not be delete")})
|
|
|
|
file_path = safe_join(work_path, file_key)
|
2023-01-18 10:03:33 +00:00
|
|
|
if os.path.isdir(file_path):
|
|
|
|
shutil.rmtree(file_path)
|
|
|
|
else:
|
|
|
|
os.remove(file_path)
|
|
|
|
return Response({'msg': 'ok'})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def generate_tree(playbook, root_path, expand_key=None):
|
|
|
|
nodes = [{
|
|
|
|
"name": playbook.name,
|
|
|
|
"title": playbook.name,
|
|
|
|
"id": 'root',
|
|
|
|
"isParent": True,
|
|
|
|
"open": True,
|
|
|
|
"pId": '',
|
|
|
|
"temp": False
|
|
|
|
}]
|
|
|
|
for path, dirs, files in os.walk(root_path):
|
|
|
|
dirs.sort()
|
|
|
|
files.sort()
|
|
|
|
|
|
|
|
relative_path = os.path.relpath(path, root_path)
|
|
|
|
for d in dirs:
|
2023-09-19 10:04:24 +00:00
|
|
|
dir_id = os.path.relpath(safe_join(path, d), root_path)
|
|
|
|
|
2023-01-18 10:03:33 +00:00
|
|
|
node = {
|
|
|
|
"name": d,
|
|
|
|
"title": d,
|
2023-09-19 10:04:24 +00:00
|
|
|
"id": dir_id,
|
2023-01-18 10:03:33 +00:00
|
|
|
"isParent": True,
|
2023-02-02 07:55:37 +00:00
|
|
|
"open": True,
|
2023-01-18 10:03:33 +00:00
|
|
|
"pId": relative_path if not relative_path.startswith('.') else 'root',
|
|
|
|
"temp": False
|
|
|
|
}
|
|
|
|
if expand_key == node['id']:
|
|
|
|
node['open'] = True
|
|
|
|
nodes.append(node)
|
|
|
|
for f in files:
|
2023-09-19 10:04:24 +00:00
|
|
|
file_id = os.path.relpath(safe_join(path, f), root_path)
|
2023-01-18 10:03:33 +00:00
|
|
|
node = {
|
|
|
|
"name": f,
|
|
|
|
"title": f,
|
|
|
|
"iconSkin": 'file',
|
2023-09-19 10:04:24 +00:00
|
|
|
"id": file_id,
|
2023-01-18 10:03:33 +00:00
|
|
|
"isParent": False,
|
|
|
|
"open": False,
|
|
|
|
"pId": relative_path if not relative_path.startswith('.') else 'root',
|
|
|
|
"temp": False
|
|
|
|
}
|
|
|
|
nodes.append(node)
|
|
|
|
return nodes
|