process_management.py
133 lines
| 4.6 KiB
| text/x-python
|
PythonLexer
r1885 | # -*- coding: utf-8 -*- | |||
r2487 | # Copyright (C) 2016-2018 RhodeCode GmbH | |||
r1885 | # | |||
# This program is free software: you can redistribute it and/or modify | ||||
# it under the terms of the GNU Affero General Public License, version 3 | ||||
# (only), as published by the Free Software Foundation. | ||||
# | ||||
# This program is distributed in the hope that it will be useful, | ||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||
# GNU General Public License for more details. | ||||
# | ||||
# You should have received a copy of the GNU Affero General Public License | ||||
# along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||
# | ||||
# This program is dual-licensed. If you wish to learn more about the | ||||
# RhodeCode Enterprise Edition, including its added features, Support services, | ||||
# and proprietary license terms, please see https://rhodecode.com/licenses/ | ||||
import logging | ||||
import psutil | ||||
r2503 | import signal | |||
r1885 | from pyramid.view import view_config | |||
from rhodecode.apps._base import BaseAppView | ||||
from rhodecode.apps.admin.navigation import navigation_list | ||||
from rhodecode.lib.auth import ( | ||||
LoginRequired, HasPermissionAllDecorator, CSRFRequired) | ||||
r1982 | from rhodecode.lib.utils2 import safe_int | |||
r1885 | ||||
log = logging.getLogger(__name__) | ||||
class AdminProcessManagementView(BaseAppView): | ||||
def load_default_context(self): | ||||
c = self._get_local_tmpl_context() | ||||
r2351 | ||||
r1885 | return c | |||
@LoginRequired() | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
@view_config( | ||||
route_name='admin_settings_process_management', request_method='GET', | ||||
renderer='rhodecode:templates/admin/settings/settings.mako') | ||||
def process_management(self): | ||||
_ = self.request.translate | ||||
c = self.load_default_context() | ||||
c.active = 'process_management' | ||||
c.navlist = navigation_list(self.request) | ||||
c.gunicorn_processes = ( | ||||
p for p in psutil.process_iter() if 'gunicorn' in p.name()) | ||||
return self._get_template_context(c) | ||||
@LoginRequired() | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
r2500 | @view_config( | |||
route_name='admin_settings_process_management_data', request_method='GET', | ||||
renderer='rhodecode:templates/admin/settings/settings_process_management_data.mako') | ||||
def process_management_data(self): | ||||
_ = self.request.translate | ||||
c = self.load_default_context() | ||||
c.gunicorn_processes = ( | ||||
p for p in psutil.process_iter() if 'gunicorn' in p.name()) | ||||
return self._get_template_context(c) | ||||
@LoginRequired() | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
r1885 | @CSRFRequired() | |||
@view_config( | ||||
route_name='admin_settings_process_management_signal', | ||||
request_method='POST', renderer='json_ext') | ||||
def process_management_signal(self): | ||||
pids = self.request.json.get('pids', []) | ||||
result = [] | ||||
def on_terminate(proc): | ||||
msg = "process `PID:{}` terminated with exit code {}".format( | ||||
proc.pid, proc.returncode) | ||||
result.append(msg) | ||||
procs = [] | ||||
for pid in pids: | ||||
pid = safe_int(pid) | ||||
if pid: | ||||
try: | ||||
proc = psutil.Process(pid) | ||||
except psutil.NoSuchProcess: | ||||
continue | ||||
children = proc.children(recursive=True) | ||||
if children: | ||||
print('Wont kill Master Process') | ||||
else: | ||||
procs.append(proc) | ||||
for p in procs: | ||||
p.terminate() | ||||
gone, alive = psutil.wait_procs(procs, timeout=10, callback=on_terminate) | ||||
for p in alive: | ||||
p.kill() | ||||
return {'result': result} | ||||
r2503 | ||||
@LoginRequired() | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
@CSRFRequired() | ||||
@view_config( | ||||
route_name='admin_settings_process_management_master_signal', | ||||
request_method='POST', renderer='json_ext') | ||||
def process_management_master_signal(self): | ||||
pid_data = self.request.json.get('pid_data', {}) | ||||
pid = safe_int(pid_data['pid']) | ||||
action = pid_data['action'] | ||||
if pid: | ||||
try: | ||||
proc = psutil.Process(pid) | ||||
except psutil.NoSuchProcess: | ||||
return {'result': 'failure_no_such_process'} | ||||
children = proc.children(recursive=True) | ||||
if children: | ||||
# master process | ||||
if action == '+' and len(children) <= 20: | ||||
proc.send_signal(signal.SIGTTIN) | ||||
elif action == '-' and len(children) >= 2: | ||||
proc.send_signal(signal.SIGTTOU) | ||||
else: | ||||
return {'result': 'failure_wrong_action'} | ||||
return {'result': 'success'} | ||||
return {'result': 'failure_not_master'} | ||||