# Copyright: (c) OpenSpug Organization. https://github.com/openspug/spug # Copyright: (c) # Released under the AGPL-3.0 License. from django.views.generic import View from django_redis import get_redis_connection from django.conf import settings from libs import json_response, JsonParser, Argument, human_datetime, auth from apps.exec.models import ExecTemplate, ExecHistory from apps.host.models import Host from apps.account.utils import has_host_perm import uuid import json import os class TemplateView(View): @auth('exec.template.view|exec.task.do|schedule.schedule.add|schedule.schedule.edit|\ monitor.monitor.add|monitor.monitor.edit') def get(self, request): templates = ExecTemplate.objects.all() types = [x['type'] for x in templates.order_by('type').values('type').distinct()] return json_response({'types': types, 'templates': [x.to_view() for x in templates]}) @auth('exec.template.add|exec.template.edit') def post(self, request): form, error = JsonParser( Argument('id', type=int, required=False), Argument('name', help='请输入模版名称'), Argument('type', help='请选择模版类型'), Argument('body', help='请输入模版内容'), Argument('interpreter', default='sh'), Argument('host_ids', type=list, handler=json.dumps, default=[]), Argument('parameters', type=list, handler=json.dumps, default=[]), Argument('desc', required=False) ).parse(request.body) if error is None: if form.id: form.updated_at = human_datetime() form.updated_by = request.user ExecTemplate.objects.filter(pk=form.pop('id')).update(**form) else: form.created_by = request.user ExecTemplate.objects.create(**form) return json_response(error=error) @auth('exec.template.del') def delete(self, request): form, error = JsonParser( Argument('id', type=int, help='请指定操作对象') ).parse(request.GET) if error is None: ExecTemplate.objects.filter(pk=form.id).delete() return json_response(error=error) class TaskView(View): @auth('exec.task.do') def get(self, request): records = ExecHistory.objects.filter(user=request.user).select_related('template') return json_response([x.to_view() for x in records]) @auth('exec.task.do') def post(self, request): form, error = JsonParser( Argument('host_ids', type=list, filter=lambda x: len(x), help='请选择执行主机'), Argument('command', help='请输入执行命令内容'), Argument('interpreter', default='sh'), Argument('template_id', type=int, required=False), Argument('params', type=dict, handler=json.dumps, default={}) ).parse(request.body) if error is None: if not has_host_perm(request.user, form.host_ids): return json_response(error='无权访问主机,请联系管理员') token, rds = uuid.uuid4().hex, get_redis_connection() form.host_ids.sort() if form.template_id: template = ExecTemplate.objects.filter(pk=form.template_id).first() if not template or template.body != form.command: form.template_id = None ExecHistory.objects.create( user=request.user, digest=token, interpreter=form.interpreter, template_id=form.template_id, command=form.command, host_ids=json.dumps(form.host_ids), params=form.params ) return json_response(token) return json_response(error=error) @auth('exec.task.do') def patch(self, request): form, error = JsonParser( Argument('token', help='参数错误'), Argument('cols', type=int, required=False), Argument('rows', type=int, required=False) ).parse(request.body) if error is None: term = None if form.cols and form.rows: term = {'width': form.cols, 'height': form.rows} rds = get_redis_connection() task = ExecHistory.objects.get(digest=form.token) for host in Host.objects.filter(id__in=json.loads(task.host_ids)): data = dict( key=host.id, name=host.name, token=task.digest, interpreter=task.interpreter, hostname=host.hostname, port=host.port, username=host.username, command=task.command, pkey=host.private_key, params=json.loads(task.params), term=term ) rds.rpush(settings.EXEC_WORKER_KEY, json.dumps(data)) return json_response(error=error) KILLER = ''' function kill_tree { local pid=$1 local and_self=${2:-0} local children=$(pgrep -P $pid) for child in $children; do kill_tree $child 1 done if [ $and_self -eq 1 ]; then kill $pid fi } kill_tree %s ''' @auth('exec.task.do|deploy.request.do') def handle_terminate(request): form, error = JsonParser( Argument('token', help='参数错误') ).parse(request.body) if error is None: rds = get_redis_connection() pid_str = rds.get(form.token) if not pid_str: return json_response(error='未找到关联进程') target, pid = pid_str.decode().split('.') if target.isdigit(): host = Host.objects.get(pk=target) with host.get_ssh() as ssh: command = KILLER % pid ssh.exec_command_raw(command) elif target == 'local': gid = os.getpgid(int(pid)) if gid: os.killpg(gid, 9) rds.delete(form.token) return json_response(error=error)