mirror of
https://github.com/usmannasir/cyberpanel.git
synced 2026-05-09 14:35:52 +02:00
Manage Applications: Elasticsearch, Redis, RabbitMQ installers and UI hardening
Add modular application packages with backup-aware install/upgrade/downgrade, DNF-backed version lists with TTL caching, and HTML bootstrap for faster loads. Improve the version picker (labels, selection state, background meta refresh) and route applicationMeta through shared page meta cache. Update static assets and cache buster for manageServices.js. Repository also includes related updates to serviceManager, upgrade tooling, website functions, and user management from this development tree.
This commit is contained in:
@@ -2329,9 +2329,6 @@
|
||||
<a href="{% url 'aiScannerHome' %}" class="menu-item">
|
||||
<span>AI Scanner</span>
|
||||
</a>
|
||||
<a href="#" class="menu-item" onclick="loadSecurityManagement(); return false;">
|
||||
<span>Security Management</span>
|
||||
</a>
|
||||
</div>
|
||||
|
||||
<a href="#" class="menu-item" onclick="toggleSubmenu('mail-settings-submenu', this); return false;">
|
||||
@@ -2507,7 +2504,7 @@
|
||||
<script src="{% static 'serverStatus/serverStatus.js' %}?v={{ CP_VERSION }}" data-cfasync="false"></script>
|
||||
<script src="{% static 'firewall/firewall.js' %}?v={{ CP_VERSION }}&fw={{ FIREWALL_STATIC_VERSION|default:CP_VERSION }}&cb=4" data-cfasync="false"></script>
|
||||
<script src="{% static 'emailPremium/emailPremium.js' %}?v={{ CP_VERSION }}" data-cfasync="false"></script>
|
||||
<script src="{% static 'manageServices/manageServices.js' %}?v={{ CP_VERSION }}" data-cfasync="false"></script>
|
||||
<script src="{% static 'manageServices/manageServices.js' %}?v={{ CP_VERSION }}&msModal=20260401d" data-cfasync="false"></script>
|
||||
<script src="{% static 'CLManager/CLManager.js' %}?v={{ CP_VERSION }}" data-cfasync="false"></script>
|
||||
|
||||
<!-- Scripts -->
|
||||
@@ -2953,9 +2950,6 @@
|
||||
});
|
||||
}
|
||||
|
||||
function loadSecurityManagement() {
|
||||
window.open('{% url "securityManagementPage" %}', '_blank');
|
||||
}
|
||||
</script>
|
||||
|
||||
{% block footer_scripts %}{% endblock %}
|
||||
|
||||
193
manageServices/application_backup.py
Normal file
193
manageServices/application_backup.py
Normal file
@@ -0,0 +1,193 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
Full config + data backups for managed applications (pre version change).
|
||||
"""
|
||||
import json
|
||||
import os
|
||||
import shutil
|
||||
import subprocess
|
||||
import tarfile
|
||||
import time
|
||||
|
||||
CONFIG_PATHS = {
|
||||
'Elasticsearch': ['/etc/elasticsearch'],
|
||||
'Redis': ['/etc/redis', '/etc/redis.conf'],
|
||||
'RabbitMQ': ['/etc/rabbitmq'],
|
||||
}
|
||||
|
||||
DATA_PATHS = {
|
||||
'Elasticsearch': ['/var/lib/elasticsearch'],
|
||||
'Redis': ['/var/lib/redis'],
|
||||
'RabbitMQ': ['/var/lib/rabbitmq'],
|
||||
}
|
||||
|
||||
SERVICE_UNITS = {
|
||||
'Elasticsearch': 'elasticsearch',
|
||||
'Redis': 'redis',
|
||||
'RabbitMQ': 'rabbitmq-server',
|
||||
}
|
||||
|
||||
CHOWN_CMDS = {
|
||||
'Elasticsearch': 'chown -R elasticsearch:elasticsearch /var/lib/elasticsearch /etc/elasticsearch',
|
||||
'Redis': 'chown -R redis:redis /var/lib/redis /etc/redis /etc/redis.conf 2>/dev/null; true',
|
||||
'RabbitMQ': 'chown -R rabbitmq:rabbitmq /var/lib/rabbitmq /etc/rabbitmq',
|
||||
}
|
||||
|
||||
BACKUP_ROOT = '/home/cyberpanel/backups/manageApplications'
|
||||
|
||||
|
||||
def _existing_paths(app_name):
|
||||
out = []
|
||||
for p in CONFIG_PATHS.get(app_name, []) + DATA_PATHS.get(app_name, []):
|
||||
if os.path.exists(p):
|
||||
out.append(p)
|
||||
return out
|
||||
|
||||
|
||||
def create_managed_app_backup(app_name, status_file):
|
||||
"""
|
||||
Tar config + data paths into BACKUP_ROOT/<app>/<epoch>/bundle.tar.gz.
|
||||
Returns backup directory path, or '' on failure / nothing to back up.
|
||||
"""
|
||||
def log(msg):
|
||||
try:
|
||||
status_file.write(msg + '\n')
|
||||
status_file.flush()
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
paths = _existing_paths(app_name)
|
||||
if not paths:
|
||||
log('No paths on disk to back up for {0}; skipping archive.'.format(app_name))
|
||||
return ''
|
||||
|
||||
ts = int(time.time())
|
||||
safe = app_name.lower().replace(' ', '_')
|
||||
backup_dir = os.path.join(BACKUP_ROOT, safe, str(ts))
|
||||
os.makedirs(backup_dir, mode=0o750, exist_ok=True)
|
||||
archive = os.path.join(backup_dir, 'bundle.tar.gz')
|
||||
|
||||
try:
|
||||
with tarfile.open(archive, 'w:gz', compresslevel=6) as tf:
|
||||
for abs_path in paths:
|
||||
arc = abs_path.lstrip('/')
|
||||
tf.add(abs_path, arcname=arc, recursive=True)
|
||||
manifest = {
|
||||
'app': app_name,
|
||||
'created': ts,
|
||||
'paths': [p.lstrip('/') for p in paths],
|
||||
}
|
||||
with open(os.path.join(backup_dir, 'manifest.json'), 'w') as mh:
|
||||
json.dump(manifest, mh, indent=2)
|
||||
log('Backup created at {0}'.format(backup_dir))
|
||||
return backup_dir
|
||||
except Exception as err:
|
||||
log('Backup failed: {0}'.format(err))
|
||||
try:
|
||||
shutil.rmtree(backup_dir, ignore_errors=True)
|
||||
except Exception:
|
||||
pass
|
||||
return ''
|
||||
|
||||
|
||||
def _archive_path(backup_dir):
|
||||
return os.path.join(backup_dir, 'bundle.tar.gz')
|
||||
|
||||
|
||||
def merge_data_from_backup(app_name, backup_dir, status_file):
|
||||
"""Overlay saved data directories from backup onto live system (preserves package layout)."""
|
||||
def log(msg):
|
||||
try:
|
||||
status_file.write(msg + '\n')
|
||||
status_file.flush()
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
arc = _archive_path(backup_dir)
|
||||
if not os.path.isfile(arc):
|
||||
log('No bundle at {0}; skip data merge.'.format(arc))
|
||||
return False
|
||||
data_prefixes = [p.lstrip('/') for p in DATA_PATHS.get(app_name, [])]
|
||||
if not data_prefixes:
|
||||
return True
|
||||
try:
|
||||
with tarfile.open(arc, 'r:gz') as tf:
|
||||
for m in tf.getmembers():
|
||||
name = m.name
|
||||
if m.isfile() or m.isdir():
|
||||
for pref in data_prefixes:
|
||||
if name == pref or name.startswith(pref + '/'):
|
||||
tf.extract(m, path='/', set_attrs=False)
|
||||
break
|
||||
log('Merged data trees from backup for {0}.'.format(app_name))
|
||||
return True
|
||||
except Exception as err:
|
||||
log('Data merge failed: {0}'.format(err))
|
||||
return False
|
||||
|
||||
|
||||
def restore_full_backup(backup_dir, status_file):
|
||||
"""Extract full bundle to / (recovery)."""
|
||||
def log(msg):
|
||||
try:
|
||||
status_file.write(msg + '\n')
|
||||
status_file.flush()
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
arc = _archive_path(backup_dir)
|
||||
if not os.path.isfile(arc):
|
||||
log('Cannot restore: missing {0}'.format(arc))
|
||||
return False
|
||||
try:
|
||||
with tarfile.open(arc, 'r:gz') as tf:
|
||||
for m in tf.getmembers():
|
||||
tf.extract(m, path='/', set_attrs=False)
|
||||
log('Full restore from backup completed.')
|
||||
return True
|
||||
except Exception as err:
|
||||
log('Full restore failed: {0}'.format(err))
|
||||
return False
|
||||
|
||||
|
||||
def cleanup_managed_backup(backup_dir, status_file):
|
||||
def log(msg):
|
||||
try:
|
||||
status_file.write(msg + '\n')
|
||||
status_file.flush()
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
if not backup_dir or not os.path.isdir(backup_dir):
|
||||
return
|
||||
try:
|
||||
shutil.rmtree(backup_dir, ignore_errors=True)
|
||||
log('Removed backup directory after successful change: {0}'.format(backup_dir))
|
||||
except Exception as err:
|
||||
log('Could not remove backup dir: {0}'.format(err))
|
||||
|
||||
|
||||
def chown_app_paths(app_name, status_writer):
|
||||
cmd = CHOWN_CMDS.get(app_name)
|
||||
if not cmd:
|
||||
return
|
||||
try:
|
||||
subprocess.call(cmd, shell=True, stdout=status_writer, stderr=status_writer)
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
|
||||
def service_is_active(app_name):
|
||||
unit = SERVICE_UNITS.get(app_name)
|
||||
if not unit:
|
||||
return False
|
||||
try:
|
||||
r = subprocess.run(
|
||||
['systemctl', 'is-active', unit],
|
||||
capture_output=True,
|
||||
text=True,
|
||||
timeout=30,
|
||||
)
|
||||
return r.stdout.strip() == 'active'
|
||||
except Exception:
|
||||
return False
|
||||
141
manageServices/application_detection.py
Normal file
141
manageServices/application_detection.py
Normal file
@@ -0,0 +1,141 @@
|
||||
import os
|
||||
import re
|
||||
import subprocess
|
||||
|
||||
|
||||
APP_PACKAGE_MAP = {
|
||||
'Elasticsearch': {
|
||||
'rhel': 'elasticsearch',
|
||||
'debian': 'elasticsearch',
|
||||
'service': 'elasticsearch',
|
||||
'binary_paths': ['/usr/share/elasticsearch/bin/elasticsearch']
|
||||
},
|
||||
'Redis': {
|
||||
'rhel': 'redis',
|
||||
'debian': 'redis-server',
|
||||
'service': 'redis',
|
||||
'binary_paths': ['/usr/bin/redis-server']
|
||||
},
|
||||
'RabbitMQ': {
|
||||
'rhel': 'rabbitmq-server',
|
||||
'debian': 'rabbitmq-server',
|
||||
'service': 'rabbitmq-server',
|
||||
'binary_paths': ['/usr/sbin/rabbitmq-server', '/usr/lib/rabbitmq/bin/rabbitmq-server']
|
||||
}
|
||||
}
|
||||
|
||||
APP_MARKERS = {
|
||||
'Elasticsearch': '/home/cyberpanel/elasticsearch',
|
||||
'Redis': '/home/cyberpanel/redis',
|
||||
'RabbitMQ': '/home/cyberpanel/rabbitmq'
|
||||
}
|
||||
|
||||
|
||||
def _run(cmd):
|
||||
try:
|
||||
res = subprocess.run(cmd, capture_output=True, text=True, timeout=12)
|
||||
return res.returncode, (res.stdout or '').strip(), (res.stderr or '').strip()
|
||||
except Exception as err:
|
||||
return 1, '', str(err)
|
||||
|
||||
|
||||
def is_debian_family():
|
||||
return os.path.exists('/etc/debian_version') or os.path.exists('/etc/lsb-release')
|
||||
|
||||
|
||||
def is_centos7():
|
||||
release_paths = ['/etc/centos-release', '/etc/redhat-release', '/etc/os-release']
|
||||
text_blob = ''
|
||||
for path in release_paths:
|
||||
try:
|
||||
if os.path.exists(path):
|
||||
with open(path, 'r') as fh:
|
||||
text_blob += fh.read().lower() + '\n'
|
||||
except Exception:
|
||||
continue
|
||||
return ('centos' in text_blob and ('release 7' in text_blob or 'version_id="7' in text_blob))
|
||||
|
||||
|
||||
def managed_apps_os_support():
|
||||
if is_centos7():
|
||||
return {
|
||||
'supported': False,
|
||||
'reason': 'CentOS 7 is EOL and not supported for managed applications.'
|
||||
}
|
||||
return {
|
||||
'supported': True,
|
||||
'reason': ''
|
||||
}
|
||||
|
||||
|
||||
def package_name_for_app(app_name):
|
||||
app_map = APP_PACKAGE_MAP.get(app_name, {})
|
||||
if not app_map:
|
||||
return ''
|
||||
if is_debian_family():
|
||||
return app_map.get('debian', '')
|
||||
return app_map.get('rhel', '')
|
||||
|
||||
|
||||
def _rpm_installed(pkg_name):
|
||||
rc, out, _ = _run(['rpm', '-q', pkg_name])
|
||||
if rc == 0:
|
||||
return True, out
|
||||
return False, ''
|
||||
|
||||
|
||||
def _dpkg_installed(pkg_name):
|
||||
rc, out, _ = _run(['dpkg-query', '-W', '-f=${Version}', pkg_name])
|
||||
if rc == 0 and out:
|
||||
return True, out
|
||||
return False, ''
|
||||
|
||||
|
||||
def _systemd_active(service_name):
|
||||
rc, out, _ = _run(['systemctl', 'is-active', service_name])
|
||||
return rc == 0 and out.strip() == 'active'
|
||||
|
||||
|
||||
def detect_installed_version(app_name):
|
||||
pkg_name = package_name_for_app(app_name)
|
||||
if not pkg_name:
|
||||
return ''
|
||||
|
||||
if is_debian_family():
|
||||
ok, ver = _dpkg_installed(pkg_name)
|
||||
else:
|
||||
ok, ver = _rpm_installed(pkg_name)
|
||||
|
||||
if not ok:
|
||||
return ''
|
||||
|
||||
if app_name in ('Elasticsearch', 'Redis', 'RabbitMQ'):
|
||||
match = re.search(r'(\d+\.\d+\.\d+)', ver)
|
||||
return match.group(1) if match else ver
|
||||
|
||||
return ver
|
||||
|
||||
|
||||
def detect_app_state(app_name):
|
||||
marker_path = APP_MARKERS.get(app_name, '')
|
||||
package_name = package_name_for_app(app_name)
|
||||
service_name = APP_PACKAGE_MAP.get(app_name, {}).get('service', '')
|
||||
binary_paths = APP_PACKAGE_MAP.get(app_name, {}).get('binary_paths', [])
|
||||
|
||||
installed_version = detect_installed_version(app_name)
|
||||
marker_exists = bool(marker_path and os.path.exists(marker_path))
|
||||
service_active = _systemd_active(service_name) if service_name else False
|
||||
binary_exists = any(os.path.exists(path) for path in binary_paths)
|
||||
|
||||
installed = bool(installed_version or service_active or binary_exists)
|
||||
|
||||
return {
|
||||
'appName': app_name,
|
||||
'packageName': package_name,
|
||||
'markerPath': marker_path,
|
||||
'markerExists': marker_exists,
|
||||
'installed': installed,
|
||||
'installedVersion': installed_version,
|
||||
'serviceActive': service_active,
|
||||
'binaryExists': binary_exists
|
||||
}
|
||||
212
manageServices/application_elasticsearch.py
Normal file
212
manageServices/application_elasticsearch.py
Normal file
@@ -0,0 +1,212 @@
|
||||
import os
|
||||
import subprocess
|
||||
import time
|
||||
|
||||
from serverStatus.serverStatusUtil import ServerStatusUtil
|
||||
from plogical import CyberCPLogFileWriter as logging
|
||||
from manageServices.application_backup import (
|
||||
CHOWN_CMDS,
|
||||
cleanup_managed_backup,
|
||||
create_managed_app_backup,
|
||||
merge_data_from_backup,
|
||||
restore_full_backup,
|
||||
service_is_active,
|
||||
)
|
||||
from manageServices.application_detection import detect_app_state, is_debian_family
|
||||
|
||||
|
||||
def _es_major_normalized(es_major):
|
||||
m = str(es_major).strip()
|
||||
if m in ('7', '8', '9'):
|
||||
return m
|
||||
return '8'
|
||||
|
||||
|
||||
def _write_repo(es_major):
|
||||
major = _es_major_normalized(es_major)
|
||||
if is_debian_family():
|
||||
repo_file = '/etc/apt/sources.list.d/elastic-{0}.x.list'.format(major)
|
||||
cmd = 'echo "deb https://artifacts.elastic.co/packages/{0}.x/apt stable main" | sudo tee {1}'.format(major, repo_file)
|
||||
subprocess.call(cmd, shell=True)
|
||||
return repo_file
|
||||
|
||||
repo_file = '/etc/yum.repos.d/elasticsearch.repo'
|
||||
content = '''
|
||||
[elasticsearch]
|
||||
name=Elasticsearch repository for {0}.x packages
|
||||
baseurl=https://artifacts.elastic.co/packages/{0}.x/yum
|
||||
gpgcheck=1
|
||||
gpgkey=https://artifacts.elastic.co/GPG-KEY-elasticsearch
|
||||
enabled=0
|
||||
autorefresh=1
|
||||
type=rpm-md
|
||||
'''.format(major)
|
||||
with open(repo_file, 'w') as handle:
|
||||
handle.write(content)
|
||||
return repo_file
|
||||
|
||||
|
||||
def _ensure_tmpdir(status_file):
|
||||
ServerStatusUtil.executioner('mkdir -p /home/elasticsearch/tmp', status_file)
|
||||
ServerStatusUtil.executioner('chown elasticsearch:elasticsearch /home/elasticsearch/tmp', status_file)
|
||||
jvm_options = '/etc/elasticsearch/jvm.options'
|
||||
line = '-Djava.io.tmpdir=/home/elasticsearch/tmp\n'
|
||||
try:
|
||||
if os.path.exists(jvm_options):
|
||||
with open(jvm_options, 'r') as handle:
|
||||
body = handle.read()
|
||||
if line.strip() not in body:
|
||||
with open(jvm_options, 'a') as handle:
|
||||
handle.write(line)
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
|
||||
def adopt_or_reconcile(status_file):
|
||||
state = detect_app_state('Elasticsearch')
|
||||
if state['installed'] and not state['markerExists']:
|
||||
ServerStatusUtil.executioner('touch /home/cyberpanel/elasticsearch', status_file)
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath,
|
||||
'Elasticsearch detected and adopted by marker reconciliation.\n'
|
||||
)
|
||||
return state
|
||||
|
||||
|
||||
def _resolve_target_version(version, es_major):
|
||||
if version and str(version).strip() != 'latest':
|
||||
return str(version).strip()
|
||||
from manageServices.application_versions import get_latest_version
|
||||
return get_latest_version('Elasticsearch', es_major, '3') or ''
|
||||
|
||||
|
||||
def _run_elasticsearch_packages(version, es_major, status_file, allow_downgrade):
|
||||
if is_debian_family():
|
||||
subprocess.call(
|
||||
'wget -qO - https://artifacts.elastic.co/GPG-KEY-elasticsearch | sudo apt-key add -',
|
||||
shell=True,
|
||||
)
|
||||
ServerStatusUtil.executioner('apt-get install apt-transport-https -y', status_file)
|
||||
_write_repo(es_major)
|
||||
ServerStatusUtil.executioner('apt-get update -y', status_file)
|
||||
if version and version != 'latest':
|
||||
cmd = (
|
||||
'DEBIAN_FRONTEND=noninteractive apt-get install -y '
|
||||
'--allow-downgrades elasticsearch={0}'
|
||||
).format(version)
|
||||
else:
|
||||
cmd = 'DEBIAN_FRONTEND=noninteractive apt-get install elasticsearch -y'
|
||||
ServerStatusUtil.executioner(cmd, status_file)
|
||||
return
|
||||
|
||||
ServerStatusUtil.executioner(
|
||||
'rpm --import https://artifacts.elastic.co/GPG-KEY-elasticsearch', status_file
|
||||
)
|
||||
_write_repo(es_major)
|
||||
ad = ' --allow-downgrade' if allow_downgrade else ''
|
||||
if version and version != 'latest':
|
||||
cmd = 'dnf install{0} -y --enablerepo=elasticsearch elasticsearch-{1}'.format(
|
||||
ad, version
|
||||
)
|
||||
ServerStatusUtil.executioner(cmd, status_file)
|
||||
else:
|
||||
cmd = 'dnf install{0} -y --enablerepo=elasticsearch elasticsearch'.format(ad)
|
||||
ServerStatusUtil.executioner(cmd, status_file)
|
||||
|
||||
|
||||
def install(version='latest', es_major='8'):
|
||||
status_file = open(ServerStatusUtil.lswsInstallStatusPath, 'w')
|
||||
adopt_or_reconcile(status_file)
|
||||
|
||||
from manageServices.application_versions import version_compare
|
||||
|
||||
state = detect_app_state('Elasticsearch')
|
||||
backup_dir = ''
|
||||
allow_downgrade = False
|
||||
target = _resolve_target_version(version, es_major)
|
||||
|
||||
if state['installed'] and state.get('installedVersion'):
|
||||
status_file.write(
|
||||
'Pre-version-change backup and service stop (Elasticsearch)...\n'
|
||||
)
|
||||
status_file.flush()
|
||||
iv = state['installedVersion']
|
||||
if target and version_compare(iv, target) > 0:
|
||||
allow_downgrade = True
|
||||
status_file.write(
|
||||
'Downgrade path: allowing package manager downgrade where supported.\n'
|
||||
)
|
||||
status_file.flush()
|
||||
backup_dir = create_managed_app_backup('Elasticsearch', status_file)
|
||||
ServerStatusUtil.executioner('systemctl stop elasticsearch', status_file)
|
||||
|
||||
_run_elasticsearch_packages(version, es_major, status_file, allow_downgrade)
|
||||
if backup_dir:
|
||||
merge_data_from_backup('Elasticsearch', backup_dir, status_file)
|
||||
ServerStatusUtil.executioner(CHOWN_CMDS['Elasticsearch'], status_file)
|
||||
|
||||
_ensure_tmpdir(status_file)
|
||||
ServerStatusUtil.executioner('systemctl enable elasticsearch', status_file)
|
||||
ServerStatusUtil.executioner('systemctl start elasticsearch', status_file)
|
||||
time.sleep(3)
|
||||
|
||||
if backup_dir:
|
||||
if service_is_active('Elasticsearch'):
|
||||
cleanup_managed_backup(backup_dir, status_file)
|
||||
status_file.write(
|
||||
'Elasticsearch version change completed; backup removed after success.\n'
|
||||
)
|
||||
else:
|
||||
status_file.write(
|
||||
'Elasticsearch failed to start; restoring from backup...\n'
|
||||
)
|
||||
restore_full_backup(backup_dir, status_file)
|
||||
ServerStatusUtil.executioner(CHOWN_CMDS['Elasticsearch'], status_file)
|
||||
ServerStatusUtil.executioner('systemctl start elasticsearch', status_file)
|
||||
time.sleep(2)
|
||||
if not service_is_active('Elasticsearch'):
|
||||
status_file.write(
|
||||
'Recovery unclear — backup kept at {0}\n'.format(backup_dir)
|
||||
)
|
||||
else:
|
||||
status_file.write(
|
||||
'Prior state restored from backup. Backup retained for safety.\n'
|
||||
)
|
||||
status_file.flush()
|
||||
|
||||
ServerStatusUtil.executioner('touch /home/cyberpanel/elasticsearch', status_file)
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath, 'Elasticsearch installed.[200]\n', 1
|
||||
)
|
||||
return 0
|
||||
|
||||
|
||||
def upgrade(version='latest', es_major='8'):
|
||||
return install(version=version, es_major=es_major)
|
||||
|
||||
|
||||
def remove():
|
||||
status_file = open(ServerStatusUtil.lswsInstallStatusPath, 'w')
|
||||
if is_debian_family():
|
||||
for major in ('7', '8', '9'):
|
||||
path = '/etc/apt/sources.list.d/elastic-{0}.x.list'.format(major)
|
||||
try:
|
||||
os.remove(path)
|
||||
except Exception:
|
||||
pass
|
||||
ServerStatusUtil.executioner(
|
||||
'DEBIAN_FRONTEND=noninteractive apt-get remove elasticsearch -y', status_file
|
||||
)
|
||||
else:
|
||||
try:
|
||||
os.remove('/etc/yum.repos.d/elasticsearch.repo')
|
||||
except Exception:
|
||||
pass
|
||||
ServerStatusUtil.executioner('yum erase elasticsearch -y', status_file)
|
||||
|
||||
ServerStatusUtil.executioner('rm -rf /home/elasticsearch/tmp', status_file)
|
||||
ServerStatusUtil.executioner('rm -f /home/cyberpanel/elasticsearch', status_file)
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath, 'Elasticsearch removed.[200]\n', 1
|
||||
)
|
||||
return 0
|
||||
146
manageServices/application_page_meta.py
Normal file
146
manageServices/application_page_meta.py
Normal file
@@ -0,0 +1,146 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""Server-side metadata for Manage Applications page (version lists in HTML)."""
|
||||
import json
|
||||
import threading
|
||||
import time
|
||||
|
||||
from .application_detection import detect_app_state, managed_apps_os_support
|
||||
from .application_versions import get_available_versions, version_compare
|
||||
|
||||
_APP_IMAGES = {
|
||||
'Elasticsearch': '/static/manageServices/images/elastic-search.png',
|
||||
'Redis': '/static/manageServices/images/redis.png',
|
||||
'RabbitMQ': '/static/manageServices/images/rabbitmq-logo.svg',
|
||||
}
|
||||
|
||||
_PAGE_META_TTL_SECONDS = 20
|
||||
_PAGE_META_CACHE = {}
|
||||
_PAGE_META_LOCK = threading.Lock()
|
||||
|
||||
|
||||
def _page_meta_cache_get(cache_key):
|
||||
now = time.time()
|
||||
with _PAGE_META_LOCK:
|
||||
item = _PAGE_META_CACHE.get(cache_key)
|
||||
if not item:
|
||||
return None
|
||||
ts, payload = item
|
||||
if now - ts > _PAGE_META_TTL_SECONDS:
|
||||
try:
|
||||
del _PAGE_META_CACHE[cache_key]
|
||||
except Exception:
|
||||
pass
|
||||
return None
|
||||
services, meta_json = payload
|
||||
return list(services), str(meta_json)
|
||||
|
||||
|
||||
def _page_meta_cache_put(cache_key, services, meta_json):
|
||||
with _PAGE_META_LOCK:
|
||||
# Keep cache tiny (we only have a handful of key combos).
|
||||
if len(_PAGE_META_CACHE) > 12:
|
||||
_PAGE_META_CACHE.clear()
|
||||
_PAGE_META_CACHE[cache_key] = (
|
||||
time.time(),
|
||||
(list(services), str(meta_json)),
|
||||
)
|
||||
|
||||
|
||||
def build_manage_applications_page_data(es_major='8', rabbitmq_stream='3'):
|
||||
"""
|
||||
Build `services` for card HTML and a JSON-serializable bootstrap matching
|
||||
/manageservices/applicationMeta shape (default ES major 8, RMQ stream 3).
|
||||
"""
|
||||
services = []
|
||||
bootstrap_apps = []
|
||||
support = managed_apps_os_support()
|
||||
major = str(es_major).strip() if str(es_major).strip() in ('7', '8', '9') else '8'
|
||||
rmq = str(rabbitmq_stream).strip() if str(rabbitmq_stream).strip() in ('3', '4') else '3'
|
||||
cache_key = 'major:{0}|rmq:{1}|support:{2}'.format(
|
||||
major, rmq, 1 if support.get('supported') else 0
|
||||
)
|
||||
|
||||
cached = _page_meta_cache_get(cache_key)
|
||||
if cached is not None:
|
||||
return cached
|
||||
|
||||
for app_name in ('Elasticsearch', 'Redis', 'RabbitMQ'):
|
||||
state = detect_app_state(app_name)
|
||||
services.append({
|
||||
'image': _APP_IMAGES[app_name],
|
||||
'name': app_name,
|
||||
'installed': 'Installed' if state['installed'] else 'Not-Installed',
|
||||
'installedVersion': state.get('installedVersion', ''),
|
||||
})
|
||||
|
||||
versions = []
|
||||
latest_branch = ''
|
||||
latest_global = ''
|
||||
if support['supported']:
|
||||
try:
|
||||
versions = get_available_versions(app_name, major, rmq)
|
||||
except BaseException:
|
||||
versions = []
|
||||
if versions:
|
||||
latest_branch = versions[0]
|
||||
latest_global = latest_branch
|
||||
|
||||
installed_version = state['installedVersion']
|
||||
if installed_version and installed_version not in versions:
|
||||
versions = [installed_version] + versions
|
||||
|
||||
ref_latest = latest_global or latest_branch
|
||||
update_available = bool(
|
||||
state['installed']
|
||||
and installed_version
|
||||
and ref_latest
|
||||
and version_compare(installed_version, ref_latest) < 0
|
||||
)
|
||||
|
||||
bootstrap_apps.append({
|
||||
'name': app_name,
|
||||
'installed': state['installed'],
|
||||
'installedVersion': installed_version,
|
||||
'latestAvailable': latest_branch,
|
||||
'latestOverall': latest_global,
|
||||
'updateAvailable': update_available,
|
||||
'crossBranchUpdateSuggested': False,
|
||||
'versions': versions,
|
||||
'packageName': state['packageName'],
|
||||
'adopted': bool(state['installed'] and not state['markerExists']),
|
||||
'major': major if app_name == 'Elasticsearch' else '',
|
||||
'rabbitmqStream': rmq if app_name == 'RabbitMQ' else '',
|
||||
})
|
||||
|
||||
bootstrap = {'status': 1, 'apps': bootstrap_apps}
|
||||
meta_json = json.dumps(bootstrap, ensure_ascii=False)
|
||||
_page_meta_cache_put(cache_key, services, meta_json)
|
||||
return services, meta_json
|
||||
|
||||
|
||||
def get_application_meta_response_dict(es_major='8', rabbitmq_stream='3'):
|
||||
"""
|
||||
JSON payload for POST /manageservices/applicationMeta.
|
||||
Reuses the same TTL cache as the Manage Applications HTML bootstrap so
|
||||
modal refresh hits warm cache after a page load (or prior request).
|
||||
"""
|
||||
support = managed_apps_os_support()
|
||||
major = str(es_major).strip() if str(es_major).strip() in ('7', '8', '9') else '8'
|
||||
rmq = str(rabbitmq_stream).strip() if str(rabbitmq_stream).strip() in ('3', '4') else '3'
|
||||
cache_key = 'major:{0}|rmq:{1}|support:{2}'.format(
|
||||
major, rmq, 1 if support.get('supported') else 0
|
||||
)
|
||||
|
||||
cached = _page_meta_cache_get(cache_key)
|
||||
if cached is not None:
|
||||
_services, meta_json = cached
|
||||
else:
|
||||
_services, meta_json = build_manage_applications_page_data(major, rmq)
|
||||
|
||||
payload = json.loads(meta_json)
|
||||
return {
|
||||
'status': 1,
|
||||
'osSupportedForManagedApps': support['supported'],
|
||||
'unsupportedReason': support['reason'],
|
||||
'apps': payload.get('apps') or [],
|
||||
}
|
||||
146
manageServices/application_rabbitmq.py
Normal file
146
manageServices/application_rabbitmq.py
Normal file
@@ -0,0 +1,146 @@
|
||||
import time
|
||||
|
||||
from serverStatus.serverStatusUtil import ServerStatusUtil
|
||||
from plogical import CyberCPLogFileWriter as logging
|
||||
|
||||
from manageServices.application_backup import (
|
||||
CHOWN_CMDS,
|
||||
cleanup_managed_backup,
|
||||
create_managed_app_backup,
|
||||
merge_data_from_backup,
|
||||
restore_full_backup,
|
||||
service_is_active,
|
||||
)
|
||||
from manageServices.application_detection import detect_app_state, is_debian_family
|
||||
from manageServices.application_rabbitmq_repo import (
|
||||
normalize_rabbitmq_stream,
|
||||
ensure_rabbitmq_team_repos,
|
||||
ensure_erlang_meets_minimum,
|
||||
)
|
||||
|
||||
|
||||
def adopt_or_reconcile(status_file):
|
||||
state = detect_app_state('RabbitMQ')
|
||||
if state['installed'] and not state['markerExists']:
|
||||
ServerStatusUtil.executioner('touch /home/cyberpanel/rabbitmq', status_file)
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath,
|
||||
'RabbitMQ detected and adopted by marker reconciliation.\n'
|
||||
)
|
||||
return state
|
||||
|
||||
|
||||
def _resolve_target_version(version, stream):
|
||||
if version and str(version).strip() != 'latest':
|
||||
return str(version).strip()
|
||||
from manageServices.application_versions import get_latest_version
|
||||
return get_latest_version('RabbitMQ', '8', stream) or ''
|
||||
|
||||
|
||||
def _run_rabbitmq_packages(version, status_file, allow_downgrade):
|
||||
ad = ' --allow-downgrade' if allow_downgrade else ''
|
||||
if is_debian_family():
|
||||
if version and version != 'latest':
|
||||
cmd = (
|
||||
'DEBIAN_FRONTEND=noninteractive apt-get install -y '
|
||||
'--allow-downgrades rabbitmq-server={0}'
|
||||
).format(version)
|
||||
else:
|
||||
cmd = 'DEBIAN_FRONTEND=noninteractive apt-get install rabbitmq-server -y'
|
||||
ServerStatusUtil.executioner(cmd, status_file)
|
||||
return
|
||||
|
||||
if version and version != 'latest':
|
||||
cmd = 'dnf install{0} -y rabbitmq-server-{1}'.format(ad, version)
|
||||
ServerStatusUtil.executioner(cmd, status_file)
|
||||
else:
|
||||
cmd = 'dnf install{0} -y rabbitmq-server'.format(ad)
|
||||
ServerStatusUtil.executioner(cmd, status_file)
|
||||
|
||||
|
||||
def install(version='latest', stream='3'):
|
||||
stream = normalize_rabbitmq_stream(stream)
|
||||
status_file = open(ServerStatusUtil.lswsInstallStatusPath, 'w')
|
||||
adopt_or_reconcile(status_file)
|
||||
|
||||
from manageServices.application_versions import version_compare
|
||||
|
||||
ensure_rabbitmq_team_repos(stream, status_file=status_file)
|
||||
ensure_erlang_meets_minimum(stream, version, status_file=status_file)
|
||||
|
||||
state = detect_app_state('RabbitMQ')
|
||||
backup_dir = ''
|
||||
allow_downgrade = False
|
||||
target = _resolve_target_version(version, stream)
|
||||
|
||||
if state['installed'] and state.get('installedVersion'):
|
||||
status_file.write(
|
||||
'Pre-version-change backup and service stop (RabbitMQ)...\n'
|
||||
)
|
||||
status_file.flush()
|
||||
iv = state['installedVersion']
|
||||
if target and version_compare(iv, target) > 0:
|
||||
allow_downgrade = True
|
||||
status_file.write('Downgrade path enabled for RabbitMQ.\n')
|
||||
status_file.flush()
|
||||
backup_dir = create_managed_app_backup('RabbitMQ', status_file)
|
||||
ServerStatusUtil.executioner('systemctl stop rabbitmq-server', status_file)
|
||||
|
||||
_run_rabbitmq_packages(version, status_file, allow_downgrade)
|
||||
if backup_dir:
|
||||
merge_data_from_backup('RabbitMQ', backup_dir, status_file)
|
||||
ServerStatusUtil.executioner(CHOWN_CMDS['RabbitMQ'], status_file)
|
||||
|
||||
ServerStatusUtil.executioner('systemctl enable rabbitmq-server', status_file)
|
||||
ServerStatusUtil.executioner('systemctl start rabbitmq-server', status_file)
|
||||
time.sleep(4)
|
||||
|
||||
if backup_dir:
|
||||
if service_is_active('RabbitMQ'):
|
||||
cleanup_managed_backup(backup_dir, status_file)
|
||||
status_file.write(
|
||||
'RabbitMQ version change completed; backup removed after success.\n'
|
||||
)
|
||||
else:
|
||||
status_file.write('RabbitMQ failed to start; restoring from backup...\n')
|
||||
restore_full_backup(backup_dir, status_file)
|
||||
ServerStatusUtil.executioner(CHOWN_CMDS['RabbitMQ'], status_file)
|
||||
ServerStatusUtil.executioner('systemctl start rabbitmq-server', status_file)
|
||||
time.sleep(4)
|
||||
if not service_is_active('RabbitMQ'):
|
||||
status_file.write(
|
||||
'Recovery unclear — backup kept at {0}\n'.format(backup_dir)
|
||||
)
|
||||
else:
|
||||
status_file.write(
|
||||
'Prior state restored from backup. Backup retained for safety.\n'
|
||||
)
|
||||
status_file.flush()
|
||||
|
||||
ServerStatusUtil.executioner('touch /home/cyberpanel/rabbitmq', status_file)
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath, 'RabbitMQ installed.[200]\n', 1
|
||||
)
|
||||
return 0
|
||||
|
||||
|
||||
def upgrade(version='latest', stream='3'):
|
||||
return install(version=version, stream=stream)
|
||||
|
||||
|
||||
def remove():
|
||||
status_file = open(ServerStatusUtil.lswsInstallStatusPath, 'w')
|
||||
ServerStatusUtil.executioner('systemctl stop rabbitmq-server', status_file)
|
||||
ServerStatusUtil.executioner('systemctl disable rabbitmq-server', status_file)
|
||||
if is_debian_family():
|
||||
ServerStatusUtil.executioner(
|
||||
'DEBIAN_FRONTEND=noninteractive apt-get remove rabbitmq-server -y',
|
||||
status_file,
|
||||
)
|
||||
else:
|
||||
ServerStatusUtil.executioner('yum erase rabbitmq-server -y', status_file)
|
||||
ServerStatusUtil.executioner('rm -f /home/cyberpanel/rabbitmq', status_file)
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath, 'RabbitMQ removed.[200]\n', 1
|
||||
)
|
||||
return 0
|
||||
225
manageServices/application_rabbitmq_repo.py
Normal file
225
manageServices/application_rabbitmq_repo.py
Normal file
@@ -0,0 +1,225 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
Team RabbitMQ package repositories (Packagecloud) and Erlang compatibility
|
||||
for RabbitMQ 3.x vs 4.x installation streams.
|
||||
"""
|
||||
import re
|
||||
import subprocess
|
||||
|
||||
from manageServices.application_detection import is_debian_family
|
||||
|
||||
# Official Packagecloud install scripts (RabbitMQ team).
|
||||
_RPM_ERLANG_SCRIPT = (
|
||||
'https://packagecloud.io/install/repositories/rabbitmq/rabbitmq-erlang/script.rpm.sh'
|
||||
)
|
||||
_RPM_SERVER_SCRIPT = (
|
||||
'https://packagecloud.io/install/repositories/rabbitmq/rabbitmq-server/script.rpm.sh'
|
||||
)
|
||||
_DEB_ERLANG_SCRIPT = (
|
||||
'https://packagecloud.io/install/repositories/rabbitmq/rabbitmq-erlang/script.deb.sh'
|
||||
)
|
||||
_DEB_SERVER_SCRIPT = (
|
||||
'https://packagecloud.io/install/repositories/rabbitmq/rabbitmq-server/script.deb.sh'
|
||||
)
|
||||
|
||||
# Minimum OTP major for each product stream (see rabbitmq.com docs / compatibility).
|
||||
_MIN_OTP_STREAM_3 = 25
|
||||
_MIN_OTP_STREAM_4 = 26
|
||||
|
||||
|
||||
def _run(cmd, timeout=300):
|
||||
try:
|
||||
res = subprocess.run(
|
||||
cmd, capture_output=True, text=True, timeout=timeout, shell=False
|
||||
)
|
||||
return res.returncode, (res.stdout or ''), (res.stderr or '')
|
||||
except Exception as err:
|
||||
return 1, '', str(err)
|
||||
|
||||
|
||||
def _run_shell_trusted(script_url, timeout=300):
|
||||
"""Run packagecloud install script from fixed RabbitMQ-team URL only."""
|
||||
allowed = {_RPM_ERLANG_SCRIPT, _RPM_SERVER_SCRIPT, _DEB_ERLANG_SCRIPT, _DEB_SERVER_SCRIPT}
|
||||
if script_url not in allowed:
|
||||
return 1, '', 'Invalid repository script URL.'
|
||||
# curl -fsSL ... | bash (URLs are allowlisted above)
|
||||
cmd = 'curl -1fsSL {0} | bash'.format(script_url)
|
||||
return _run(['/bin/bash', '-lc', cmd], timeout=timeout)
|
||||
|
||||
|
||||
def normalize_rabbitmq_stream(value):
|
||||
s = str(value or '3').strip()
|
||||
if s in ('4', '4.x', '41', '4.1'):
|
||||
return '4'
|
||||
return '3'
|
||||
|
||||
|
||||
def _write_status(status_file, message):
|
||||
if status_file is None:
|
||||
return
|
||||
try:
|
||||
status_file.write(message + '\n')
|
||||
status_file.flush()
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
|
||||
def ensure_rabbitmq_team_repos(stream, status_file=None):
|
||||
"""
|
||||
Idempotently enable rabbitmq-erlang and rabbitmq-server Packagecloud repos.
|
||||
Required so 3.13.x, 4.x, and matching Erlang builds are visible to the
|
||||
package manager.
|
||||
"""
|
||||
stream = normalize_rabbitmq_stream(stream)
|
||||
_write_status(
|
||||
status_file,
|
||||
'Ensuring Team RabbitMQ repositories (stream {0})...'.format(stream)
|
||||
)
|
||||
if is_debian_family():
|
||||
rc, out, err = _run_shell_trusted(_DEB_ERLANG_SCRIPT)
|
||||
if rc != 0:
|
||||
_write_status(status_file, 'rabbitmq-erlang repo script: ' + (err or out or 'failed'))
|
||||
rc2, out2, err2 = _run_shell_trusted(_DEB_SERVER_SCRIPT)
|
||||
if rc2 != 0:
|
||||
_write_status(
|
||||
status_file, 'rabbitmq-server repo script: ' + (err2 or out2 or 'failed')
|
||||
)
|
||||
_run(['apt-get', 'update', '-y'], timeout=120)
|
||||
else:
|
||||
rc, out, err = _run_shell_trusted(_RPM_ERLANG_SCRIPT)
|
||||
if rc != 0:
|
||||
_write_status(status_file, 'rabbitmq-erlang repo script: ' + (err or out or 'failed'))
|
||||
rc2, out2, err2 = _run_shell_trusted(_RPM_SERVER_SCRIPT)
|
||||
if rc2 != 0:
|
||||
_write_status(
|
||||
status_file, 'rabbitmq-server repo script: ' + (err2 or out2 or 'failed')
|
||||
)
|
||||
# Prefer dnf; yum exists as symlink on EL8/9.
|
||||
for cache_cmd in (['dnf', 'makecache', '-y'], ['yum', 'makecache', '-y']):
|
||||
c_rc, _, _ = _run(cache_cmd, timeout=120)
|
||||
if c_rc == 0:
|
||||
break
|
||||
_write_status(status_file, 'Team RabbitMQ repositories ready.')
|
||||
|
||||
|
||||
def get_erlang_otp_major():
|
||||
"""Best-effort current Erlang/OTP major version (integer or 0 if unknown)."""
|
||||
rc, out, _ = _run(
|
||||
[
|
||||
'erl',
|
||||
'-noshell',
|
||||
'-eval',
|
||||
'io:format("~s~n", [erlang:system_info(otp_release)]), halt().',
|
||||
],
|
||||
timeout=15,
|
||||
)
|
||||
if rc == 0 and out:
|
||||
m = re.search(r'(\d+)', out.strip())
|
||||
if m:
|
||||
return int(m.group(1))
|
||||
# rpm: erlang from RabbitMQ repo may report R26 flavour
|
||||
rc2, out2, _ = _run(['rpm', '-q', '--qf', '%{VERSION}', 'erlang'], timeout=10)
|
||||
if rc2 == 0 and out2:
|
||||
m = re.search(r'^(\d+)', out2.strip())
|
||||
if m:
|
||||
return int(m.group(1))
|
||||
return 0
|
||||
|
||||
|
||||
def minimum_otp_for_stream(stream):
|
||||
stream = normalize_rabbitmq_stream(stream)
|
||||
if stream == '4':
|
||||
return _MIN_OTP_STREAM_4
|
||||
return _MIN_OTP_STREAM_3
|
||||
|
||||
|
||||
def minimum_otp_for_rabbitmq_version(version_str):
|
||||
"""Infer OTP floor from chosen RabbitMQ version when possible."""
|
||||
if not version_str or version_str == 'latest':
|
||||
return None
|
||||
m = re.match(r'^(\d+)', str(version_str).strip())
|
||||
if not m:
|
||||
return None
|
||||
major = int(m.group(1))
|
||||
if major >= 4:
|
||||
return _MIN_OTP_STREAM_4
|
||||
if major >= 3:
|
||||
return _MIN_OTP_STREAM_3
|
||||
return None
|
||||
|
||||
|
||||
def ensure_erlang_meets_minimum(stream, version, status_file=None):
|
||||
"""
|
||||
Upgrade/install Erlang from enabled repos if OTP is below the minimum
|
||||
for the selected RabbitMQ stream or explicit target version.
|
||||
"""
|
||||
stream = normalize_rabbitmq_stream(stream)
|
||||
need = minimum_otp_for_stream(stream)
|
||||
version_floor = minimum_otp_for_rabbitmq_version(version)
|
||||
if version_floor is not None:
|
||||
need = max(need, version_floor)
|
||||
|
||||
current = get_erlang_otp_major()
|
||||
if current >= need:
|
||||
_write_status(
|
||||
status_file,
|
||||
'Erlang/OTP {0} satisfies minimum {1} for this RabbitMQ target.'.format(
|
||||
current or 'unknown', need
|
||||
)
|
||||
)
|
||||
return
|
||||
|
||||
_write_status(
|
||||
status_file,
|
||||
'Erlang/OTP {0} is below required {1}; installing/upgrading erlang from Team RabbitMQ repo...'.format(
|
||||
current or 'unknown', need
|
||||
)
|
||||
)
|
||||
if is_debian_family():
|
||||
_run(
|
||||
[
|
||||
'/bin/bash',
|
||||
'-lc',
|
||||
'DEBIAN_FRONTEND=noninteractive apt-get install -y erlang',
|
||||
],
|
||||
timeout=600,
|
||||
)
|
||||
else:
|
||||
for inst in (
|
||||
['dnf', 'install', '-y', 'erlang'],
|
||||
['yum', 'install', '-y', 'erlang'],
|
||||
):
|
||||
rc, _, _ = _run(inst, timeout=600)
|
||||
if rc == 0:
|
||||
break
|
||||
|
||||
after = get_erlang_otp_major()
|
||||
if after < need:
|
||||
_write_status(
|
||||
status_file,
|
||||
'WARNING: Erlang may still be below OTP {0} (reported {1}). '
|
||||
'Check /root/cyberpanel or logs and install correct erlang package.'.format(
|
||||
need, after or 'unknown'
|
||||
)
|
||||
)
|
||||
else:
|
||||
_write_status(status_file, 'Erlang/OTP updated to {0}.'.format(after))
|
||||
|
||||
|
||||
def filter_versions_for_stream(versions, stream):
|
||||
"""Keep only versions whose major matches RabbitMQ stream (3 or 4)."""
|
||||
stream = normalize_rabbitmq_stream(stream)
|
||||
result = []
|
||||
seen = set()
|
||||
for raw in versions or []:
|
||||
v = (raw or '').strip()
|
||||
if not v or v in seen:
|
||||
continue
|
||||
if v == 'latest':
|
||||
continue
|
||||
m = re.search(r'(\d+)\.(\d+)', v)
|
||||
if m and m.group(1) == stream:
|
||||
seen.add(v)
|
||||
result.append(v)
|
||||
# Preserve descending-ish order (caller already sorted newest first)
|
||||
return result
|
||||
132
manageServices/application_redis.py
Normal file
132
manageServices/application_redis.py
Normal file
@@ -0,0 +1,132 @@
|
||||
import time
|
||||
|
||||
from serverStatus.serverStatusUtil import ServerStatusUtil
|
||||
from plogical import CyberCPLogFileWriter as logging
|
||||
|
||||
from manageServices.application_backup import (
|
||||
CHOWN_CMDS,
|
||||
cleanup_managed_backup,
|
||||
create_managed_app_backup,
|
||||
merge_data_from_backup,
|
||||
restore_full_backup,
|
||||
service_is_active,
|
||||
)
|
||||
from manageServices.application_detection import detect_app_state, is_debian_family
|
||||
|
||||
|
||||
def adopt_or_reconcile(status_file):
|
||||
state = detect_app_state('Redis')
|
||||
if state['installed'] and not state['markerExists']:
|
||||
ServerStatusUtil.executioner('touch /home/cyberpanel/redis', status_file)
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath,
|
||||
'Redis detected and adopted by marker reconciliation.\n'
|
||||
)
|
||||
return state
|
||||
|
||||
|
||||
def _resolve_target_version(version):
|
||||
if version and str(version).strip() != 'latest':
|
||||
return str(version).strip()
|
||||
from manageServices.application_versions import get_latest_version
|
||||
return get_latest_version('Redis', '8', '3') or ''
|
||||
|
||||
|
||||
def _run_redis_packages(version, status_file, allow_downgrade):
|
||||
ad = ' --allow-downgrade' if allow_downgrade else ''
|
||||
if is_debian_family():
|
||||
if version and version != 'latest':
|
||||
cmd = (
|
||||
'DEBIAN_FRONTEND=noninteractive apt-get install -y '
|
||||
'--allow-downgrades redis-server={0}'
|
||||
).format(version)
|
||||
else:
|
||||
cmd = 'DEBIAN_FRONTEND=noninteractive apt-get install redis-server -y'
|
||||
ServerStatusUtil.executioner(cmd, status_file)
|
||||
return
|
||||
|
||||
if version and version != 'latest':
|
||||
cmd = 'dnf install{0} -y redis-{1}'.format(ad, version)
|
||||
ServerStatusUtil.executioner(cmd, status_file)
|
||||
else:
|
||||
cmd = 'dnf install{0} -y redis'.format(ad)
|
||||
ServerStatusUtil.executioner(cmd, status_file)
|
||||
|
||||
|
||||
def install(version='latest'):
|
||||
status_file = open(ServerStatusUtil.lswsInstallStatusPath, 'w')
|
||||
adopt_or_reconcile(status_file)
|
||||
|
||||
from manageServices.application_versions import version_compare
|
||||
|
||||
state = detect_app_state('Redis')
|
||||
backup_dir = ''
|
||||
allow_downgrade = False
|
||||
target = _resolve_target_version(version)
|
||||
|
||||
if state['installed'] and state.get('installedVersion'):
|
||||
status_file.write('Pre-version-change backup and service stop (Redis)...\n')
|
||||
status_file.flush()
|
||||
iv = state['installedVersion']
|
||||
if target and version_compare(iv, target) > 0:
|
||||
allow_downgrade = True
|
||||
status_file.write('Downgrade path enabled for Redis.\n')
|
||||
status_file.flush()
|
||||
backup_dir = create_managed_app_backup('Redis', status_file)
|
||||
ServerStatusUtil.executioner('systemctl stop redis', status_file)
|
||||
|
||||
_run_redis_packages(version, status_file, allow_downgrade)
|
||||
if backup_dir:
|
||||
merge_data_from_backup('Redis', backup_dir, status_file)
|
||||
ServerStatusUtil.executioner(CHOWN_CMDS['Redis'], status_file)
|
||||
|
||||
ServerStatusUtil.executioner('systemctl enable redis', status_file)
|
||||
ServerStatusUtil.executioner('systemctl start redis', status_file)
|
||||
time.sleep(2)
|
||||
|
||||
if backup_dir:
|
||||
if service_is_active('Redis'):
|
||||
cleanup_managed_backup(backup_dir, status_file)
|
||||
status_file.write(
|
||||
'Redis version change completed; backup removed after success.\n'
|
||||
)
|
||||
else:
|
||||
status_file.write('Redis failed to start; restoring from backup...\n')
|
||||
restore_full_backup(backup_dir, status_file)
|
||||
ServerStatusUtil.executioner(CHOWN_CMDS['Redis'], status_file)
|
||||
ServerStatusUtil.executioner('systemctl start redis', status_file)
|
||||
time.sleep(2)
|
||||
if not service_is_active('Redis'):
|
||||
status_file.write(
|
||||
'Recovery unclear — backup kept at {0}\n'.format(backup_dir)
|
||||
)
|
||||
else:
|
||||
status_file.write(
|
||||
'Prior state restored from backup. Backup retained for safety.\n'
|
||||
)
|
||||
status_file.flush()
|
||||
|
||||
ServerStatusUtil.executioner('touch /home/cyberpanel/redis', status_file)
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath, 'Redis installed.[200]\n', 1
|
||||
)
|
||||
return 0
|
||||
|
||||
|
||||
def upgrade(version='latest'):
|
||||
return install(version=version)
|
||||
|
||||
|
||||
def remove():
|
||||
status_file = open(ServerStatusUtil.lswsInstallStatusPath, 'w')
|
||||
if is_debian_family():
|
||||
ServerStatusUtil.executioner(
|
||||
'DEBIAN_FRONTEND=noninteractive apt-get remove redis-server -y', status_file
|
||||
)
|
||||
else:
|
||||
ServerStatusUtil.executioner('yum erase redis -y', status_file)
|
||||
ServerStatusUtil.executioner('rm -f /home/cyberpanel/redis', status_file)
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath, 'Redis removed.[200]\n', 1
|
||||
)
|
||||
return 0
|
||||
380
manageServices/application_versions.py
Normal file
380
manageServices/application_versions.py
Normal file
@@ -0,0 +1,380 @@
|
||||
import os
|
||||
import re
|
||||
import subprocess
|
||||
import threading
|
||||
import time
|
||||
|
||||
from manageServices.application_detection import is_debian_family, package_name_for_app
|
||||
|
||||
# applicationMeta can call get_available_versions many times per request (ES 7/8/9, RMQ 3/4).
|
||||
# Concurrent DNF from every WSGI worker exhausts lscpd and returns HTTP 503. Cache + serialize cold fetches.
|
||||
_VERSION_CACHE = {}
|
||||
_VERSION_CACHE_LOCK = threading.Lock()
|
||||
_DNF_COLD_FETCH_LOCK = threading.Lock()
|
||||
|
||||
# Seconds; override with CYBERCP_MANAGED_APPS_VERSION_CACHE_TTL if needed
|
||||
_CACHE_TTL_SEC = int(os.environ.get('CYBERCP_MANAGED_APPS_VERSION_CACHE_TTL', '300'))
|
||||
|
||||
|
||||
def _version_cache_key(app_name, es_major, rabbitmq_stream):
|
||||
debian = is_debian_family()
|
||||
if app_name == 'Elasticsearch':
|
||||
em = normalize_elasticsearch_major(es_major)
|
||||
else:
|
||||
em = ''
|
||||
rs = ''
|
||||
if app_name == 'RabbitMQ':
|
||||
from manageServices.application_rabbitmq_repo import normalize_rabbitmq_stream
|
||||
rs = normalize_rabbitmq_stream(rabbitmq_stream)
|
||||
return (str(app_name), em, rs, debian)
|
||||
|
||||
|
||||
def _cache_get_versions(key):
|
||||
now = time.monotonic()
|
||||
with _VERSION_CACHE_LOCK:
|
||||
entry = _VERSION_CACHE.get(key)
|
||||
if not entry:
|
||||
return None
|
||||
ts, versions = entry
|
||||
if (now - ts) >= _CACHE_TTL_SEC:
|
||||
try:
|
||||
del _VERSION_CACHE[key]
|
||||
except KeyError:
|
||||
pass
|
||||
return None
|
||||
# Never use a poisoned empty cache (DNF timeout / lock) as a hit.
|
||||
if not versions:
|
||||
try:
|
||||
del _VERSION_CACHE[key]
|
||||
except KeyError:
|
||||
pass
|
||||
return None
|
||||
return list(versions)
|
||||
|
||||
|
||||
def _cache_put_versions(key, versions):
|
||||
snap = list(versions or [])
|
||||
if not snap:
|
||||
return
|
||||
with _VERSION_CACHE_LOCK:
|
||||
_VERSION_CACHE[key] = (time.monotonic(), snap)
|
||||
|
||||
# User-writable DNF snippet dir (panel runs as user `cyberpanel`; cannot rely on /etc).
|
||||
_CYBERPANEL_DNF_EXTRA = '/home/cyberpanel/.cyberpanel-dnf/repos.d'
|
||||
|
||||
|
||||
def _version_tuple(ver):
|
||||
"""Numeric tuple for semver-style compare; empty if not usable."""
|
||||
if ver is None:
|
||||
return ()
|
||||
s = str(ver).strip()
|
||||
if not s or s.lower() == 'latest':
|
||||
return ()
|
||||
parts = []
|
||||
for x in re.findall(r'\d+', s):
|
||||
try:
|
||||
parts.append(int(x))
|
||||
except ValueError:
|
||||
break
|
||||
return tuple(parts)
|
||||
|
||||
|
||||
def version_compare(a, b):
|
||||
"""
|
||||
Compare two version strings: return -1 if a < b, 0 if equal or incomparable, 1 if a > b.
|
||||
"""
|
||||
ta = _version_tuple(a)
|
||||
tb = _version_tuple(b)
|
||||
if not ta or not tb:
|
||||
return 0
|
||||
length = max(len(ta), len(tb))
|
||||
for i in range(length):
|
||||
x = ta[i] if i < len(ta) else 0
|
||||
y = tb[i] if i < len(tb) else 0
|
||||
if x < y:
|
||||
return -1
|
||||
if x > y:
|
||||
return 1
|
||||
return 0
|
||||
|
||||
|
||||
def _max_version_string(candidates):
|
||||
best = ''
|
||||
for v in candidates or []:
|
||||
if not v:
|
||||
continue
|
||||
if not best or version_compare(best, v) < 0:
|
||||
best = v
|
||||
return best
|
||||
|
||||
|
||||
def _run(cmd, timeout=120):
|
||||
try:
|
||||
res = subprocess.run(cmd, capture_output=True, text=True, timeout=timeout)
|
||||
return res.returncode, (res.stdout or ''), (res.stderr or '')
|
||||
except Exception as err:
|
||||
return 1, '', str(err)
|
||||
|
||||
|
||||
def normalize_elasticsearch_major(es_major):
|
||||
"""Supported Elasticsearch package streams (official artifacts.elastic.co)."""
|
||||
m = str(es_major).strip()
|
||||
if m in ('7', '8', '9'):
|
||||
return m
|
||||
return '8'
|
||||
|
||||
|
||||
def _ensure_cyberpanel_es_repo(es_major):
|
||||
"""Elasticsearch official repo for version discovery (no root; gpg off for repoquery-only)."""
|
||||
major = normalize_elasticsearch_major(es_major)
|
||||
try:
|
||||
os.makedirs(_CYBERPANEL_DNF_EXTRA, mode=0o755, exist_ok=True)
|
||||
except Exception:
|
||||
return
|
||||
path = os.path.join(
|
||||
_CYBERPANEL_DNF_EXTRA, 'cyberpanel-elasticsearch-{0}.repo'.format(major)
|
||||
)
|
||||
content = (
|
||||
'[cyberpanel-elasticsearch-{0}]\n'
|
||||
'name=Elasticsearch {0}.x metadata (CyberPanel)\n'
|
||||
'baseurl=https://artifacts.elastic.co/packages/{0}.x/yum\n'
|
||||
'gpgcheck=0\n'
|
||||
'repo_gpgcheck=0\n'
|
||||
'enabled=1\n'
|
||||
).format(major)
|
||||
try:
|
||||
with open(path, 'w') as handle:
|
||||
handle.write(content)
|
||||
os.chmod(path, 0o644)
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
|
||||
def _normalize_versions(raw_versions, max_items=25):
|
||||
versions = []
|
||||
seen = set()
|
||||
for item in raw_versions:
|
||||
value = (item or '').strip()
|
||||
if not value or value in seen:
|
||||
continue
|
||||
seen.add(value)
|
||||
versions.append(value)
|
||||
return versions[:max_items]
|
||||
|
||||
|
||||
def _sort_versions_desc(candidates):
|
||||
def key_fn(ver):
|
||||
nums = [int(x) for x in re.findall(r'\d+', ver) if x.isdigit()]
|
||||
return nums or [0]
|
||||
|
||||
try:
|
||||
return sorted(set(candidates), key=key_fn, reverse=True)
|
||||
except Exception:
|
||||
return sorted(set(candidates), reverse=True)
|
||||
|
||||
|
||||
def _dnf_reposdir_flag(use_cyberpanel_extra):
|
||||
if not use_cyberpanel_extra:
|
||||
return []
|
||||
if not os.path.isdir(_CYBERPANEL_DNF_EXTRA):
|
||||
try:
|
||||
os.makedirs(_CYBERPANEL_DNF_EXTRA, mode=0o755, exist_ok=True)
|
||||
except Exception:
|
||||
return []
|
||||
return ['--setopt=reposdir=/etc/yum.repos.d,{0}'.format(_CYBERPANEL_DNF_EXTRA)]
|
||||
|
||||
|
||||
def _rhel_repoquery_versions(pkg_name, use_cyberpanel_extra_repos=False, enablerepos=None):
|
||||
"""
|
||||
Resolve distinct %{version} strings from enabled repos.
|
||||
RPM NEVRA text parsing is brittle (el9_7 etc.); repoquery --qf is reliable.
|
||||
"""
|
||||
cmd = (
|
||||
['dnf']
|
||||
+ _dnf_reposdir_flag(use_cyberpanel_extra_repos)
|
||||
+ [
|
||||
'repoquery',
|
||||
'--show-duplicates',
|
||||
'--latest-limit=50',
|
||||
'--qf',
|
||||
'%{version}',
|
||||
pkg_name,
|
||||
]
|
||||
)
|
||||
if enablerepos:
|
||||
for repo_id in enablerepos:
|
||||
cmd.extend(['--enablerepo', repo_id])
|
||||
rc, out, err = _run(cmd, timeout=180)
|
||||
raw = []
|
||||
if rc == 0 and out.strip():
|
||||
for line in out.splitlines():
|
||||
v = (line or '').strip()
|
||||
if v and re.match(r'^[0-9]', v):
|
||||
raw.append(v)
|
||||
if raw:
|
||||
return _normalize_versions(_sort_versions_desc(raw))
|
||||
|
||||
# Legacy systems / fallback
|
||||
rc2, out2, _ = _run(
|
||||
['yum', 'repoquery', '--show-duplicates', '--qf', '%{version}', pkg_name],
|
||||
timeout=120,
|
||||
)
|
||||
raw2 = []
|
||||
if rc2 == 0 and out2.strip():
|
||||
for line in out2.splitlines():
|
||||
v = (line or '').strip()
|
||||
if v and re.match(r'^[0-9]', v):
|
||||
raw2.append(v)
|
||||
if raw2:
|
||||
return _normalize_versions(_sort_versions_desc(raw2))
|
||||
|
||||
# Oldest fallback: yum list
|
||||
rc3, out3, _ = _run(['yum', '--showduplicates', 'list', pkg_name], timeout=120)
|
||||
if rc3 == 0:
|
||||
for line in out3.splitlines():
|
||||
row = line.strip()
|
||||
if not row or row.startswith('Loaded plugins') or row.startswith('Available'):
|
||||
continue
|
||||
fields = row.split()
|
||||
if len(fields) >= 2 and pkg_name in fields[0]:
|
||||
raw2.append(fields[1])
|
||||
if raw2:
|
||||
return _normalize_versions(_sort_versions_desc(raw2))
|
||||
return []
|
||||
|
||||
|
||||
def _debian_versions(pkg_name):
|
||||
versions = []
|
||||
_run(['apt-get', 'update', '-y'], timeout=180)
|
||||
rc, out, _ = _run(['apt-cache', 'madison', pkg_name], timeout=60)
|
||||
if rc != 0:
|
||||
return []
|
||||
for line in out.splitlines():
|
||||
if '|' not in line:
|
||||
continue
|
||||
parts = [p.strip() for p in line.split('|')]
|
||||
if len(parts) >= 2 and parts[1]:
|
||||
versions.append(parts[1])
|
||||
collected = []
|
||||
for v in versions:
|
||||
m = re.search(r'(\d+\.\d+\.\d+)', v)
|
||||
collected.append(m.group(1) if m else v)
|
||||
return _normalize_versions(_sort_versions_desc(collected))
|
||||
|
||||
|
||||
def _filter_es_major(versions, es_major):
|
||||
major = normalize_elasticsearch_major(es_major)
|
||||
out = []
|
||||
for v in versions or []:
|
||||
head = (v.split('.') or [''])[0]
|
||||
if head == major:
|
||||
out.append(v)
|
||||
return out
|
||||
|
||||
|
||||
def _get_available_versions_uncached(app_name, es_major='8', rabbitmq_stream='3'):
|
||||
pkg_name = package_name_for_app(app_name)
|
||||
if app_name == 'Elasticsearch':
|
||||
pkg_name = 'elasticsearch'
|
||||
|
||||
if not pkg_name:
|
||||
return []
|
||||
|
||||
rmq_stream = '3'
|
||||
if app_name == 'RabbitMQ':
|
||||
from manageServices.application_rabbitmq_repo import (
|
||||
normalize_rabbitmq_stream,
|
||||
ensure_rabbitmq_team_repos,
|
||||
filter_versions_for_stream,
|
||||
)
|
||||
rmq_stream = normalize_rabbitmq_stream(rabbitmq_stream)
|
||||
ensure_rabbitmq_team_repos(rmq_stream)
|
||||
|
||||
if is_debian_family():
|
||||
versions = _debian_versions(pkg_name)
|
||||
if app_name == 'Elasticsearch':
|
||||
versions = _filter_es_major(versions, es_major)
|
||||
else:
|
||||
if app_name == 'Elasticsearch':
|
||||
_ensure_cyberpanel_es_repo(es_major)
|
||||
versions = _rhel_repoquery_versions(
|
||||
pkg_name, use_cyberpanel_extra_repos=True
|
||||
)
|
||||
versions = _filter_es_major(versions, es_major)
|
||||
else:
|
||||
versions = _rhel_repoquery_versions(pkg_name)
|
||||
|
||||
if app_name == 'RabbitMQ':
|
||||
from manageServices.application_rabbitmq_repo import filter_versions_for_stream
|
||||
versions = filter_versions_for_stream(versions, rmq_stream)
|
||||
return versions
|
||||
|
||||
|
||||
def get_available_versions(app_name, es_major='8', rabbitmq_stream='3'):
|
||||
"""
|
||||
Cached wrapper: avoids hammering DNF from many concurrent panel workers (503 on Manage Applications).
|
||||
"""
|
||||
key = _version_cache_key(app_name, es_major, rabbitmq_stream)
|
||||
hit = _cache_get_versions(key)
|
||||
if hit is not None:
|
||||
return hit
|
||||
|
||||
with _DNF_COLD_FETCH_LOCK:
|
||||
hit2 = _cache_get_versions(key)
|
||||
if hit2 is not None:
|
||||
return hit2
|
||||
versions = _get_available_versions_uncached(
|
||||
app_name, es_major, rabbitmq_stream
|
||||
)
|
||||
if versions:
|
||||
_cache_put_versions(key, versions)
|
||||
return list(versions)
|
||||
|
||||
|
||||
def get_latest_version(app_name, es_major='8', rabbitmq_stream='3'):
|
||||
versions = get_available_versions(app_name, es_major, rabbitmq_stream)
|
||||
if not versions:
|
||||
return ''
|
||||
return versions[0]
|
||||
|
||||
|
||||
def get_branch_and_global_latest(app_name, es_major='8', rabbitmq_stream='3'):
|
||||
"""
|
||||
Latest on the UI-selected branch/stream vs latest across all supported branches.
|
||||
|
||||
Returns (latest_on_branch, latest_global).
|
||||
"""
|
||||
latest_branch = get_latest_version(app_name, es_major, rabbitmq_stream)
|
||||
if app_name == 'Elasticsearch':
|
||||
candidates = []
|
||||
for m in ('7', '8', '9'):
|
||||
v = get_latest_version('Elasticsearch', m, rabbitmq_stream)
|
||||
if v:
|
||||
candidates.append(v)
|
||||
latest_global = _max_version_string(candidates) if candidates else ''
|
||||
elif app_name == 'RabbitMQ':
|
||||
candidates = []
|
||||
for s in ('3', '4'):
|
||||
v = get_latest_version('RabbitMQ', es_major, s)
|
||||
if v:
|
||||
candidates.append(v)
|
||||
latest_global = _max_version_string(candidates) if candidates else ''
|
||||
else:
|
||||
latest_global = latest_branch
|
||||
if not latest_global:
|
||||
latest_global = latest_branch
|
||||
return latest_branch, latest_global
|
||||
|
||||
|
||||
def cross_branch_newer_suggested(installed, latest_branch, latest_global):
|
||||
"""
|
||||
True when installed is current (or ahead of) the selected branch latest but
|
||||
a newer release exists on another line (e.g. 8.x latest installed, 9.x exists).
|
||||
"""
|
||||
if not installed or not latest_global:
|
||||
return False
|
||||
if version_compare(installed, latest_global) >= 0:
|
||||
return False
|
||||
if not latest_branch:
|
||||
return True
|
||||
return version_compare(installed, latest_branch) >= 0
|
||||
@@ -1,6 +1,10 @@
|
||||
import os
|
||||
import os.path
|
||||
import sys
|
||||
import django
|
||||
repo_root = os.path.abspath(os.path.join(os.path.dirname(__file__), '..'))
|
||||
if repo_root not in sys.path:
|
||||
sys.path.append(repo_root)
|
||||
sys.path.append('/usr/local/CyberCP')
|
||||
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "CyberCP.settings")
|
||||
django.setup()
|
||||
@@ -12,6 +16,10 @@ import argparse
|
||||
from serverStatus.serverStatusUtil import ServerStatusUtil
|
||||
from plogical import CyberCPLogFileWriter as logging
|
||||
import subprocess
|
||||
from manageServices.application_detection import managed_apps_os_support
|
||||
from manageServices import application_elasticsearch
|
||||
from manageServices import application_redis
|
||||
from manageServices import application_rabbitmq
|
||||
|
||||
class ServiceManager:
|
||||
|
||||
@@ -142,254 +150,114 @@ autosecondary=yes
|
||||
Supermasters(ip=self.extraArgs['masterServerIP'], nameserver=self.extraArgs['slaveServerNS'], account='').save()
|
||||
|
||||
@staticmethod
|
||||
def InstallElasticSearch():
|
||||
def InstallElasticSearch(version='latest', esMajor='8'):
|
||||
return application_elasticsearch.install(version=version, es_major=esMajor)
|
||||
|
||||
statusFile = open(ServerStatusUtil.lswsInstallStatusPath, 'w')
|
||||
|
||||
if ProcessUtilities.decideDistro() == ProcessUtilities.centos or ProcessUtilities.decideDistro() == ProcessUtilities.cent8:
|
||||
command = 'rpm --import https://artifacts.elastic.co/GPG-KEY-elasticsearch'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
repoPath = '/etc/yum.repos.d/elasticsearch.repo'
|
||||
|
||||
content = '''
|
||||
[elasticsearch]
|
||||
name=Elasticsearch repository for 7.x packages
|
||||
baseurl=https://artifacts.elastic.co/packages/7.x/yum
|
||||
gpgcheck=1
|
||||
gpgkey=https://artifacts.elastic.co/GPG-KEY-elasticsearch
|
||||
enabled=0
|
||||
autorefresh=1
|
||||
type=rpm-md
|
||||
'''
|
||||
|
||||
writeToFile = open(repoPath, 'w')
|
||||
writeToFile.write(content)
|
||||
writeToFile.close()
|
||||
|
||||
command = 'yum install --enablerepo=elasticsearch elasticsearch -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
else:
|
||||
command = 'wget -qO - https://artifacts.elastic.co/GPG-KEY-elasticsearch | sudo apt-key add -'
|
||||
subprocess.call(command, shell=True)
|
||||
|
||||
command = 'apt-get install apt-transport-https -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
command = 'echo "deb https://artifacts.elastic.co/packages/7.x/apt stable main" | sudo tee /etc/apt/sources.list.d/elastic-7.x.list'
|
||||
subprocess.call(command, shell=True)
|
||||
|
||||
command = 'apt-get update -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
command = 'apt-get install elasticsearch -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
### Tmp folder configurations
|
||||
|
||||
command = 'mkdir -p /home/elasticsearch/tmp'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
command = 'chown elasticsearch:elasticsearch /home/elasticsearch/tmp'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
jvmOptions = '/etc/elasticsearch/jvm.options'
|
||||
|
||||
writeToFile = open(jvmOptions, 'a')
|
||||
writeToFile.write('-Djava.io.tmpdir=/home/elasticsearch/tmp\n')
|
||||
writeToFile.close()
|
||||
|
||||
command = 'systemctl enable elasticsearch'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
command = 'systemctl start elasticsearch'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
command = 'touch /home/cyberpanel/elasticsearch'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
|
||||
|
||||
logging.CyberCPLogFileWriter.statusWriter(ServerStatusUtil.lswsInstallStatusPath,
|
||||
"Packages successfully installed.[200]\n", 1)
|
||||
return 0
|
||||
@staticmethod
|
||||
def UpgradeElasticSearch(version='latest', esMajor='8'):
|
||||
return application_elasticsearch.upgrade(version=version, es_major=esMajor)
|
||||
|
||||
@staticmethod
|
||||
def RemoveElasticSearch():
|
||||
|
||||
statusFile = open(ServerStatusUtil.lswsInstallStatusPath, 'w')
|
||||
|
||||
if ProcessUtilities.decideDistro() == ProcessUtilities.centos or ProcessUtilities.decideDistro() == ProcessUtilities.cent8:
|
||||
command = 'rpm --import https://artifacts.elastic.co/GPG-KEY-elasticsearch'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
repoPath = '/etc/yum.repos.d/elasticsearch.repo'
|
||||
|
||||
try:
|
||||
os.remove(repoPath)
|
||||
except:
|
||||
pass
|
||||
|
||||
command = 'yum erase elasticsearch -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
else:
|
||||
|
||||
try:
|
||||
os.remove('/etc/apt/sources.list.d/elastic-7.x.list')
|
||||
except:
|
||||
pass
|
||||
|
||||
|
||||
command = 'apt-get remove elasticsearch -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
### Tmp folder configurations
|
||||
|
||||
command = 'rm -rf /home/elasticsearch/tmp'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
|
||||
command = 'rm -f /home/cyberpanel/elasticsearch'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
logging.CyberCPLogFileWriter.statusWriter(ServerStatusUtil.lswsInstallStatusPath,
|
||||
"ElasticSearch successfully removed.[200]\n", 1)
|
||||
return 0
|
||||
return application_elasticsearch.remove()
|
||||
|
||||
@staticmethod
|
||||
def InstallRedis():
|
||||
def InstallRedis(version='latest'):
|
||||
return application_redis.install(version=version)
|
||||
|
||||
statusFile = open(ServerStatusUtil.lswsInstallStatusPath, 'w')
|
||||
|
||||
if ProcessUtilities.decideDistro() == ProcessUtilities.centos or ProcessUtilities.decideDistro() == ProcessUtilities.cent8:
|
||||
command = 'yum install redis -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
else:
|
||||
|
||||
command = 'DEBIAN_FRONTEND=noninteractive apt-get install redis-server -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
|
||||
command = 'systemctl enable redis'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
command = 'systemctl start redis'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
command = 'touch /home/cyberpanel/redis'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
logging.CyberCPLogFileWriter.statusWriter(ServerStatusUtil.lswsInstallStatusPath,
|
||||
"Redis successfully installed.[200]\n", 1)
|
||||
return 0
|
||||
@staticmethod
|
||||
def UpgradeRedis(version='latest'):
|
||||
return application_redis.upgrade(version=version)
|
||||
|
||||
@staticmethod
|
||||
def RemoveRedis():
|
||||
|
||||
statusFile = open(ServerStatusUtil.lswsInstallStatusPath, 'w')
|
||||
|
||||
if ProcessUtilities.decideDistro() == ProcessUtilities.centos or ProcessUtilities.decideDistro() == ProcessUtilities.cent8:
|
||||
command = 'yum erase redis -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
else:
|
||||
|
||||
command = 'apt-get remove redis-server -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
|
||||
command = 'rm -f /home/cyberpanel/redis'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
logging.CyberCPLogFileWriter.statusWriter(ServerStatusUtil.lswsInstallStatusPath,
|
||||
"Redis successfully removed.[200]\n", 1)
|
||||
return 0
|
||||
return application_redis.remove()
|
||||
|
||||
@staticmethod
|
||||
def InstallRabbitMQ():
|
||||
statusFile = open(ServerStatusUtil.lswsInstallStatusPath, 'w')
|
||||
def InstallRabbitMQ(version='latest', stream='3'):
|
||||
return application_rabbitmq.install(version=version, stream=stream)
|
||||
|
||||
try:
|
||||
if ProcessUtilities.decideDistro() == ProcessUtilities.centos or ProcessUtilities.decideDistro() == ProcessUtilities.cent8:
|
||||
command = 'yum install rabbitmq-server -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
else:
|
||||
command = 'DEBIAN_FRONTEND=noninteractive apt-get install rabbitmq-server -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
command = 'systemctl enable rabbitmq-server'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
command = 'systemctl start rabbitmq-server'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
command = 'touch /home/cyberpanel/rabbitmq'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath,
|
||||
"RabbitMQ successfully installed.[200]\n", 1
|
||||
)
|
||||
return 0
|
||||
except BaseException as msg:
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath,
|
||||
"RabbitMQ installation failed: %s.[500]\n" % (str(msg)), 0
|
||||
)
|
||||
return 1
|
||||
@staticmethod
|
||||
def UpgradeRabbitMQ(version='latest', stream='3'):
|
||||
return application_rabbitmq.upgrade(version=version, stream=stream)
|
||||
|
||||
@staticmethod
|
||||
def RemoveRabbitMQ():
|
||||
statusFile = open(ServerStatusUtil.lswsInstallStatusPath, 'w')
|
||||
|
||||
try:
|
||||
command = 'systemctl stop rabbitmq-server'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
command = 'systemctl disable rabbitmq-server'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
if ProcessUtilities.decideDistro() == ProcessUtilities.centos or ProcessUtilities.decideDistro() == ProcessUtilities.cent8:
|
||||
command = 'yum erase rabbitmq-server -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
else:
|
||||
command = 'DEBIAN_FRONTEND=noninteractive apt-get remove rabbitmq-server -y'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
command = 'rm -f /home/cyberpanel/rabbitmq'
|
||||
ServerStatusUtil.executioner(command, statusFile)
|
||||
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath,
|
||||
"RabbitMQ successfully removed.[200]\n", 1
|
||||
)
|
||||
return 0
|
||||
except BaseException as msg:
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath,
|
||||
"RabbitMQ removal failed: %s.[500]\n" % (str(msg)), 0
|
||||
)
|
||||
return 1
|
||||
return application_rabbitmq.remove()
|
||||
|
||||
def main():
|
||||
|
||||
parser = argparse.ArgumentParser(description='CyberPanel Application Manager')
|
||||
parser.add_argument('--function', help='Function')
|
||||
|
||||
parser.add_argument('--app', help='Application name')
|
||||
parser.add_argument('--action', help='Action to run: install|remove|upgrade')
|
||||
parser.add_argument('--version', default='latest', help='Target package version or latest')
|
||||
parser.add_argument('--esMajor', default='8', help='Elasticsearch major stream (7|8|9)')
|
||||
parser.add_argument('--rabbitmqStream', default='3', help='RabbitMQ major stream (3|4)')
|
||||
|
||||
args = vars(parser.parse_args())
|
||||
|
||||
support = managed_apps_os_support()
|
||||
if not support['supported']:
|
||||
logging.CyberCPLogFileWriter.statusWriter(
|
||||
ServerStatusUtil.lswsInstallStatusPath,
|
||||
support['reason'] + '\n',
|
||||
1
|
||||
)
|
||||
return
|
||||
|
||||
if args["function"] == "InstallElasticSearch":
|
||||
ServiceManager.InstallElasticSearch()
|
||||
ServiceManager.InstallElasticSearch(version=args.get('version', 'latest'), esMajor=args.get('esMajor', '8'))
|
||||
elif args["function"] == "UpgradeElasticSearch":
|
||||
ServiceManager.UpgradeElasticSearch(version=args.get('version', 'latest'), esMajor=args.get('esMajor', '8'))
|
||||
elif args["function"] == "RemoveElasticSearch":
|
||||
ServiceManager.RemoveElasticSearch()
|
||||
elif args["function"] == "InstallRedis":
|
||||
ServiceManager.InstallRedis()
|
||||
ServiceManager.InstallRedis(version=args.get('version', 'latest'))
|
||||
elif args["function"] == "UpgradeRedis":
|
||||
ServiceManager.UpgradeRedis(version=args.get('version', 'latest'))
|
||||
elif args["function"] == "RemoveRedis":
|
||||
ServiceManager.RemoveRedis()
|
||||
elif args["function"] == "InstallRabbitMQ":
|
||||
ServiceManager.InstallRabbitMQ()
|
||||
ServiceManager.InstallRabbitMQ(
|
||||
version=args.get('version', 'latest'),
|
||||
stream=args.get('rabbitmqStream', '3'),
|
||||
)
|
||||
elif args["function"] == "UpgradeRabbitMQ":
|
||||
ServiceManager.UpgradeRabbitMQ(
|
||||
version=args.get('version', 'latest'),
|
||||
stream=args.get('rabbitmqStream', '3'),
|
||||
)
|
||||
elif args["function"] == "RemoveRabbitMQ":
|
||||
ServiceManager.RemoveRabbitMQ()
|
||||
elif args.get("app") and args.get("action"):
|
||||
app_name = args.get("app")
|
||||
action = args.get("action").lower()
|
||||
version = args.get("version", "latest")
|
||||
es_major = args.get("esMajor", "8")
|
||||
rmq_stream = args.get("rabbitmqStream", "3")
|
||||
|
||||
if app_name == 'Elasticsearch':
|
||||
if action == 'install':
|
||||
ServiceManager.InstallElasticSearch(version=version, esMajor=es_major)
|
||||
elif action == 'upgrade':
|
||||
ServiceManager.UpgradeElasticSearch(version=version, esMajor=es_major)
|
||||
elif action == 'remove':
|
||||
ServiceManager.RemoveElasticSearch()
|
||||
elif app_name == 'Redis':
|
||||
if action == 'install':
|
||||
ServiceManager.InstallRedis(version=version)
|
||||
elif action == 'upgrade':
|
||||
ServiceManager.UpgradeRedis(version=version)
|
||||
elif action == 'remove':
|
||||
ServiceManager.RemoveRedis()
|
||||
elif app_name == 'RabbitMQ':
|
||||
if action == 'install':
|
||||
ServiceManager.InstallRabbitMQ(version=version, stream=rmq_stream)
|
||||
elif action == 'upgrade':
|
||||
ServiceManager.UpgradeRabbitMQ(version=version, stream=rmq_stream)
|
||||
elif action == 'remove':
|
||||
ServiceManager.RemoveRabbitMQ()
|
||||
|
||||
|
||||
|
||||
|
||||
BIN
manageServices/static/manageServices/images/rabbitmq.png
Normal file
BIN
manageServices/static/manageServices/images/rabbitmq.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 13 KiB |
@@ -428,10 +428,411 @@ app.controller('pureFTPD', function ($scope, $http, $timeout, $window) {
|
||||
|
||||
app.controller('manageApplications', function ($scope, $http, $timeout, $window) {
|
||||
|
||||
/**
|
||||
* Normalize entries from applicationMeta (strings, numbers, or rare object shapes)
|
||||
* so version pickers never show blank rows. CyberPanel uses {$ ... $} interpolation
|
||||
* in templates; list labels use versionLabel() for consistent display.
|
||||
*/
|
||||
function normalizeVersionToken(v) {
|
||||
if (v === null || v === undefined) {
|
||||
return '';
|
||||
}
|
||||
if (v === 'latest') {
|
||||
return 'latest';
|
||||
}
|
||||
if (typeof v === 'number' && isFinite(v)) {
|
||||
return String(v);
|
||||
}
|
||||
if (angular.isObject(v)) {
|
||||
var o = v;
|
||||
var cand = o.version || o.Version || o.value || o.name || o.ver || o.label;
|
||||
if (cand !== undefined && cand !== null) {
|
||||
return String(cand).trim();
|
||||
}
|
||||
try {
|
||||
return JSON.stringify(o);
|
||||
} catch (ignore) {
|
||||
return '';
|
||||
}
|
||||
}
|
||||
return String(v).trim();
|
||||
}
|
||||
|
||||
function sanitizeVersionsArray(vers) {
|
||||
if (!angular.isArray(vers)) {
|
||||
return [];
|
||||
}
|
||||
var out = [];
|
||||
var seen = {};
|
||||
vers.forEach(function (raw) {
|
||||
var t = normalizeVersionToken(raw);
|
||||
if (!t || t === 'latest') {
|
||||
return;
|
||||
}
|
||||
if (!seen[t]) {
|
||||
seen[t] = true;
|
||||
out.push(t);
|
||||
}
|
||||
});
|
||||
return out;
|
||||
}
|
||||
|
||||
$scope.versionLabel = function (v) {
|
||||
if (v === 'latest') {
|
||||
return 'latest';
|
||||
}
|
||||
var t = normalizeVersionToken(v);
|
||||
return t || '(unknown)';
|
||||
};
|
||||
|
||||
$scope.versionTrackId = function (idx, v) {
|
||||
return String(idx) + '|' + $scope.versionLabel(v);
|
||||
};
|
||||
|
||||
/* false = long-running install/remove/poll (show spinners); true = idle */
|
||||
$scope.cyberpanelLoading = true;
|
||||
/** Background applicationMeta refresh — separate from cyberpanelLoading so the page does not “freeze” on every modal open. */
|
||||
$scope.appsMetaRefreshing = false;
|
||||
$scope.apps = [
|
||||
{name: 'Elasticsearch', image: '/static/manageServices/images/elastic-search.png'},
|
||||
{name: 'Redis', image: '/static/manageServices/images/redis.png'},
|
||||
{name: 'RabbitMQ', image: '/static/manageServices/images/rabbitmq-logo.svg'}
|
||||
];
|
||||
|
||||
$scope.removeInstall = function (appName, status) {
|
||||
(function mergeMetaBootstrap() {
|
||||
var el = document.getElementById('manageApplicationsMetaBootstrap');
|
||||
if (!el || !el.textContent) {
|
||||
return;
|
||||
}
|
||||
var raw = el.textContent.trim();
|
||||
if (!raw) {
|
||||
return;
|
||||
}
|
||||
try {
|
||||
var boot = JSON.parse(raw);
|
||||
if (!boot || Number(boot.status) !== 1) {
|
||||
return;
|
||||
}
|
||||
var appMap = {};
|
||||
(boot.apps || []).forEach(function (a) {
|
||||
if (a && a.name) {
|
||||
appMap[a.name] = a;
|
||||
}
|
||||
});
|
||||
$scope.apps = $scope.apps.map(function (baseApp) {
|
||||
var meta = appMap[baseApp.name] || {};
|
||||
var vers = meta.versions;
|
||||
if (!angular.isArray(vers)) {
|
||||
vers = [];
|
||||
}
|
||||
vers = sanitizeVersionsArray(vers);
|
||||
return {
|
||||
name: baseApp.name,
|
||||
image: baseApp.image,
|
||||
installed: !!meta.installed,
|
||||
installedVersion: meta.installedVersion || '',
|
||||
updateAvailable: !!meta.updateAvailable,
|
||||
crossBranchUpdateSuggested: !!meta.crossBranchUpdateSuggested,
|
||||
versions: vers,
|
||||
latestAvailable: meta.latestAvailable || '',
|
||||
latestOverall: meta.latestOverall || ''
|
||||
};
|
||||
});
|
||||
} catch (ignore) {
|
||||
/* keep bare apps list */
|
||||
}
|
||||
})();
|
||||
|
||||
$scope.selectedVersion = 'latest';
|
||||
/** Row highlight uses $index so only one row looks selected (avoids sticky :focus / repeater quirks). */
|
||||
$scope.selectedVersionRowIndex = 0;
|
||||
$scope.selectedVersions = ['latest'];
|
||||
|
||||
$scope.recalcSelectedVersionRowIndex = function () {
|
||||
var list = $scope.selectedVersions || [];
|
||||
var sel = $scope.selectedVersion;
|
||||
var i = list.indexOf(sel);
|
||||
if (i < 0) {
|
||||
var normSel = normalizeVersionToken(sel);
|
||||
if (normSel) {
|
||||
for (var j = 0; j < list.length; j += 1) {
|
||||
if (normalizeVersionToken(list[j]) === normSel) {
|
||||
i = j;
|
||||
$scope.selectedVersion = list[j];
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
$scope.selectedVersionRowIndex = (i >= 0) ? i : 0;
|
||||
};
|
||||
|
||||
$scope.selectManagedAppVersion = function (idx, v, $event) {
|
||||
var n = (typeof idx === 'number') ? idx : parseInt(idx, 10);
|
||||
if (!isFinite(n) || n < 0) {
|
||||
n = 0;
|
||||
}
|
||||
$scope.selectedVersionRowIndex = n;
|
||||
$scope.selectedVersion = v;
|
||||
if ($event && $event.target && typeof $event.target.blur === 'function') {
|
||||
$event.target.blur();
|
||||
} else if (typeof document !== 'undefined' && document.activeElement && typeof document.activeElement.blur === 'function') {
|
||||
document.activeElement.blur();
|
||||
}
|
||||
};
|
||||
$scope.selectedEsMajor = '8';
|
||||
$scope.selectedRabbitmqStream = '3';
|
||||
$scope.confirmAction = false;
|
||||
$scope.selectedCurrentVersion = '';
|
||||
|
||||
/**
|
||||
* When the install/upgrade modal is open, re-apply version list from latest applicationMeta.
|
||||
* (Page-load meta can be empty for ES if dnf was slow; opening the modal must refetch.)
|
||||
*/
|
||||
$scope.syncModalVersionLists = function () {
|
||||
if (!$scope.appName || ($scope.status !== 'Installing' && $scope.status !== 'Upgrading')) {
|
||||
return;
|
||||
}
|
||||
if ($scope.appName !== 'Elasticsearch' && $scope.appName !== 'Redis' && $scope.appName !== 'RabbitMQ') {
|
||||
return;
|
||||
}
|
||||
var meta = $scope.findAppMeta($scope.appName);
|
||||
var vers = sanitizeVersionsArray((meta && meta.versions) ? meta.versions : []);
|
||||
$scope.selectedVersions = ['latest'].concat(vers);
|
||||
var curRaw = (meta && meta.installedVersion) ? meta.installedVersion : ($scope.selectedCurrentVersion || '');
|
||||
var cur = normalizeVersionToken(curRaw) || String(curRaw || '').trim();
|
||||
if (cur && $scope.selectedVersions.indexOf(cur) === -1) {
|
||||
$scope.selectedVersions.push(cur);
|
||||
}
|
||||
if (cur) {
|
||||
$scope.selectedCurrentVersion = cur;
|
||||
}
|
||||
var prevSel = $scope.selectedVersion;
|
||||
if (prevSel && $scope.selectedVersions.indexOf(prevSel) !== -1) {
|
||||
$scope.selectedVersion = prevSel;
|
||||
} else {
|
||||
$scope.selectedVersion = 'latest';
|
||||
}
|
||||
var realVers = ($scope.selectedVersions || []).filter(function (v) {
|
||||
return v && v !== 'latest';
|
||||
});
|
||||
$scope.repoShowsOnlyOneStream = ($scope.status === 'Upgrading' && realVers.length <= 1);
|
||||
$scope.recalcSelectedVersionRowIndex();
|
||||
};
|
||||
|
||||
$scope.refreshMeta = function () {
|
||||
$scope.appsMetaRefreshing = true;
|
||||
var url = "/manageservices/applicationMeta";
|
||||
var data = {
|
||||
esMajor: $scope.selectedEsMajor,
|
||||
rabbitmqStream: $scope.selectedRabbitmqStream
|
||||
};
|
||||
var config = {
|
||||
headers: {
|
||||
'Content-Type': 'application/json;charset=UTF-8',
|
||||
'X-CSRFToken': getCookie('csrftoken')
|
||||
},
|
||||
transformRequest: function (payload) {
|
||||
return angular.toJson(payload);
|
||||
}
|
||||
};
|
||||
|
||||
return $http.post(url, data, config).then(function (response) {
|
||||
$scope.appsMetaRefreshing = false;
|
||||
var payload = response.data;
|
||||
var ok = payload && (payload.status === 1 || payload.status === '1');
|
||||
if (ok) {
|
||||
var appMap = {};
|
||||
(payload.apps || []).forEach(function (app) {
|
||||
appMap[app.name] = app;
|
||||
});
|
||||
$scope.apps = $scope.apps.map(function (baseApp) {
|
||||
var meta = appMap[baseApp.name] || {};
|
||||
var vers = meta.versions;
|
||||
if (!angular.isArray(vers)) {
|
||||
vers = [];
|
||||
}
|
||||
vers = sanitizeVersionsArray(vers);
|
||||
return {
|
||||
name: baseApp.name,
|
||||
image: baseApp.image,
|
||||
installed: !!meta.installed,
|
||||
installedVersion: meta.installedVersion || '',
|
||||
updateAvailable: !!meta.updateAvailable,
|
||||
crossBranchUpdateSuggested: !!meta.crossBranchUpdateSuggested,
|
||||
versions: vers,
|
||||
latestAvailable: meta.latestAvailable || '',
|
||||
latestOverall: meta.latestOverall || ''
|
||||
};
|
||||
});
|
||||
$scope.syncModalVersionLists();
|
||||
} else {
|
||||
new PNotify({
|
||||
title: 'Operation Failed!',
|
||||
text: (payload && (payload.error_message || payload.errorMessage)) || 'Could not load application metadata.',
|
||||
type: 'error'
|
||||
});
|
||||
}
|
||||
}, function () {
|
||||
$scope.appsMetaRefreshing = false;
|
||||
new PNotify({
|
||||
title: 'Operation Failed!',
|
||||
text: 'Could not connect to server, please refresh this page',
|
||||
type: 'error'
|
||||
});
|
||||
});
|
||||
};
|
||||
|
||||
$scope.prepareAction = function (service, status, bootstrapInstalledVersion) {
|
||||
if (bootstrapInstalledVersion === undefined || bootstrapInstalledVersion === null) {
|
||||
bootstrapInstalledVersion = '';
|
||||
} else {
|
||||
bootstrapInstalledVersion = String(bootstrapInstalledVersion).trim();
|
||||
}
|
||||
$scope.status = status;
|
||||
$scope.appName = service.name;
|
||||
$scope.confirmAction = false;
|
||||
var effectiveInstalled = (service.installedVersion || bootstrapInstalledVersion || '').trim();
|
||||
$scope.selectedCurrentVersion = effectiveInstalled;
|
||||
|
||||
if (service.name === 'RabbitMQ') {
|
||||
if (effectiveInstalled && /^4\./.test(effectiveInstalled)) {
|
||||
$scope.selectedRabbitmqStream = '4';
|
||||
} else if (effectiveInstalled && /^3\./.test(effectiveInstalled)) {
|
||||
$scope.selectedRabbitmqStream = '3';
|
||||
}
|
||||
}
|
||||
|
||||
if (service.name === 'Elasticsearch' && effectiveInstalled) {
|
||||
var iv = effectiveInstalled;
|
||||
if (/^9\./.test(iv)) {
|
||||
$scope.selectedEsMajor = '9';
|
||||
} else if (/^8\./.test(iv)) {
|
||||
$scope.selectedEsMajor = '8';
|
||||
} else if (/^7\./.test(iv)) {
|
||||
$scope.selectedEsMajor = '7';
|
||||
}
|
||||
}
|
||||
|
||||
$scope.selectedVersions = ['latest'];
|
||||
var svcVers = sanitizeVersionsArray(service.versions || []);
|
||||
if (svcVers.length > 0) {
|
||||
$scope.selectedVersions = ['latest'].concat(svcVers);
|
||||
}
|
||||
var curPick = normalizeVersionToken(effectiveInstalled) || effectiveInstalled;
|
||||
if (curPick && $scope.selectedVersions.indexOf(curPick) === -1) {
|
||||
$scope.selectedVersions.push(curPick);
|
||||
}
|
||||
$scope.selectedVersion = 'latest';
|
||||
$scope.requestData = '';
|
||||
|
||||
var realVers = ($scope.selectedVersions || []).filter(function (v) {
|
||||
return v && v !== 'latest';
|
||||
});
|
||||
$scope.repoShowsOnlyOneStream = ($scope.status === 'Upgrading' && realVers.length <= 1);
|
||||
$scope.recalcSelectedVersionRowIndex();
|
||||
};
|
||||
|
||||
$scope.findAppMeta = function (appName) {
|
||||
var found = null;
|
||||
($scope.apps || []).forEach(function (item) {
|
||||
if (item.name === appName) {
|
||||
found = item;
|
||||
}
|
||||
});
|
||||
return found || {};
|
||||
};
|
||||
|
||||
$scope.prepareActionByName = function (appName, status, bootstrapInstalledVersion) {
|
||||
var meta = $scope.findAppMeta(appName);
|
||||
if (!meta.name) {
|
||||
meta = {name: appName, versions: []};
|
||||
}
|
||||
var mver = meta.versions;
|
||||
if (!angular.isArray(mver)) {
|
||||
mver = [];
|
||||
}
|
||||
var merged = {
|
||||
name: meta.name,
|
||||
image: meta.image,
|
||||
installed: meta.installed,
|
||||
installedVersion: meta.installedVersion || '',
|
||||
versions: mver
|
||||
};
|
||||
$scope.prepareAction(merged, status, bootstrapInstalledVersion);
|
||||
};
|
||||
|
||||
/**
|
||||
* Prepare scope then show modal (do not use data-toggle + ng-click — Bootstrap can
|
||||
* open the dialog before Angular runs ng-click, leaving status/appName unset).
|
||||
* For managed apps, wait for applicationMeta so version lists and installedVersion are correct (upgrade vs downgrade).
|
||||
*/
|
||||
$scope.openApplicationsModal = function (appName, status, bootstrapInstalledVersion) {
|
||||
var needMeta = (appName === 'RabbitMQ' || appName === 'Elasticsearch' || appName === 'Redis');
|
||||
var showModal = function () {
|
||||
if (typeof window.jQuery !== 'undefined' && jQuery('#settings').modal) {
|
||||
jQuery('#settings').modal('show');
|
||||
}
|
||||
};
|
||||
if (needMeta) {
|
||||
// applicationMeta uses selectedRabbitmqStream / selectedEsMajor. If the user left
|
||||
// 4.x selected earlier, the first POST would ask for 4.x while the node is on 3.x —
|
||||
// filtered versions come back empty and the Version dropdown shows only "latest".
|
||||
var bootVer = (bootstrapInstalledVersion !== undefined && bootstrapInstalledVersion !== null)
|
||||
? String(bootstrapInstalledVersion).trim()
|
||||
: '';
|
||||
if (!bootVer && appName) {
|
||||
var metaEarly = $scope.findAppMeta(appName);
|
||||
if (metaEarly && metaEarly.installedVersion) {
|
||||
bootVer = String(metaEarly.installedVersion).trim();
|
||||
}
|
||||
}
|
||||
if (appName === 'RabbitMQ' && bootVer) {
|
||||
if (/^4\./.test(bootVer)) {
|
||||
$scope.selectedRabbitmqStream = '4';
|
||||
} else if (/^3\./.test(bootVer)) {
|
||||
$scope.selectedRabbitmqStream = '3';
|
||||
}
|
||||
}
|
||||
if (appName === 'Elasticsearch' && bootVer) {
|
||||
var biv = bootVer;
|
||||
if (/^9\./.test(biv)) {
|
||||
$scope.selectedEsMajor = '9';
|
||||
} else if (/^8\./.test(biv)) {
|
||||
$scope.selectedEsMajor = '8';
|
||||
} else if (/^7\./.test(biv)) {
|
||||
$scope.selectedEsMajor = '7';
|
||||
}
|
||||
}
|
||||
// Open immediately with bootstrap / in-memory meta; refresh in background.
|
||||
// Blocking on applicationMeta made "Change version" feel frozen (dnf/repoquery).
|
||||
$scope.appName = appName;
|
||||
$scope.status = status;
|
||||
$scope.prepareActionByName(appName, status, bootstrapInstalledVersion);
|
||||
showModal();
|
||||
$timeout(function () {
|
||||
$scope.refreshMeta();
|
||||
}, 0);
|
||||
} else {
|
||||
$scope.prepareActionByName(appName, status, bootstrapInstalledVersion);
|
||||
showModal();
|
||||
}
|
||||
};
|
||||
|
||||
$scope.runAction = function () {
|
||||
var appName = $scope.appName;
|
||||
var status = $scope.status;
|
||||
if (!appName || !status) {
|
||||
return;
|
||||
}
|
||||
if ((status === 'Removing' || status === 'Upgrading') && !$scope.confirmAction) {
|
||||
new PNotify({
|
||||
title: 'Confirmation Required',
|
||||
text: 'Please confirm this action before proceeding.',
|
||||
type: 'warning'
|
||||
});
|
||||
return;
|
||||
}
|
||||
$scope.status = status;
|
||||
$scope.appName = appName;
|
||||
|
||||
@@ -441,7 +842,11 @@ app.controller('manageApplications', function ($scope, $http, $timeout, $window)
|
||||
|
||||
var data = {
|
||||
appName: appName,
|
||||
status: status
|
||||
status: status,
|
||||
version: $scope.selectedVersion || 'latest',
|
||||
esMajor: $scope.selectedEsMajor || '8',
|
||||
rabbitmqStream: $scope.selectedRabbitmqStream || '3',
|
||||
confirmAction: $scope.confirmAction === true
|
||||
};
|
||||
|
||||
var config = {
|
||||
@@ -524,6 +929,9 @@ app.controller('manageApplications', function ($scope, $http, $timeout, $window)
|
||||
|
||||
}
|
||||
|
||||
// Do not fetch package metadata on page load; it can block workers under DNF load.
|
||||
// Metadata is fetched on-demand when opening install/version-change modals.
|
||||
|
||||
});
|
||||
|
||||
/* Java script code */
|
||||
@@ -260,8 +260,113 @@
|
||||
|
||||
.modal-body {
|
||||
padding: 25px;
|
||||
/* Theme may set light text globally; native <select>/<option> must stay dark-on-light. */
|
||||
color: var(--text-primary, #2f3640);
|
||||
}
|
||||
|
||||
|
||||
.applications-container .modal-body select.form-control {
|
||||
color: #0f172a;
|
||||
background-color: #ffffff;
|
||||
}
|
||||
|
||||
.applications-container .modal-body select.form-control option {
|
||||
color: #0f172a;
|
||||
background-color: #ffffff;
|
||||
}
|
||||
|
||||
.applications-container #settings .modal-body .manage-apps-version-select {
|
||||
color: #0f172a;
|
||||
background-color: #ffffff;
|
||||
}
|
||||
|
||||
.applications-container #settings .modal-body .manage-apps-version-select option {
|
||||
color: #0f172a;
|
||||
background-color: #f8fafc;
|
||||
}
|
||||
|
||||
/* Native <select> open state is OS-drawn; use a custom list so version text is always readable. */
|
||||
.manage-apps-version-picker {
|
||||
border: 1px solid #94a3b8;
|
||||
border-radius: 8px;
|
||||
overflow: hidden;
|
||||
background: #ffffff;
|
||||
margin-top: 6px;
|
||||
}
|
||||
|
||||
.manage-apps-version-current {
|
||||
padding: 10px 12px;
|
||||
background: #e2e8f0;
|
||||
color: #0f172a;
|
||||
font-size: 14px;
|
||||
font-weight: 600;
|
||||
border-bottom: 1px solid #94a3b8;
|
||||
}
|
||||
|
||||
.manage-apps-version-rows {
|
||||
max-height: 240px;
|
||||
overflow-y: auto;
|
||||
background: #ffffff;
|
||||
}
|
||||
|
||||
.manage-apps-version-row {
|
||||
display: block;
|
||||
width: 100%;
|
||||
text-align: left;
|
||||
padding: 10px 14px;
|
||||
margin: 0;
|
||||
border: 0;
|
||||
border-bottom: 1px solid #e2e8f0;
|
||||
background: #ffffff !important;
|
||||
color: #0f172a !important;
|
||||
font-size: 14px;
|
||||
line-height: 1.35;
|
||||
cursor: pointer;
|
||||
-webkit-text-fill-color: #0f172a;
|
||||
}
|
||||
|
||||
/* Only one row should read as "selected"; non-active rows stay white unless hover/focus. */
|
||||
.manage-apps-version-rows button.manage-apps-version-row:not(.is-active) {
|
||||
background: #ffffff !important;
|
||||
}
|
||||
|
||||
.manage-apps-version-rows button.manage-apps-version-row:not(.is-active):hover,
|
||||
.manage-apps-version-rows button.manage-apps-version-row:not(.is-active):focus {
|
||||
background: #f1f5f9 !important;
|
||||
color: #0f172a !important;
|
||||
-webkit-text-fill-color: #0f172a;
|
||||
outline: none;
|
||||
}
|
||||
|
||||
.manage-apps-version-rows button.manage-apps-version-row.is-active,
|
||||
.manage-apps-version-rows button.manage-apps-version-row.is-active:hover,
|
||||
.manage-apps-version-rows button.manage-apps-version-row.is-active:focus {
|
||||
background: #c7d2fe !important;
|
||||
color: #1e1b4b !important;
|
||||
-webkit-text-fill-color: #1e1b4b;
|
||||
font-weight: 600;
|
||||
outline: none;
|
||||
}
|
||||
|
||||
.manage-apps-version-row:last-child {
|
||||
border-bottom: 0;
|
||||
}
|
||||
|
||||
/* Win over theme + .ng-binding readability rules (must stay dark on white row). */
|
||||
.applications-container #settings .modal-body .manage-apps-version-row.ng-binding {
|
||||
color: #0f172a !important;
|
||||
-webkit-text-fill-color: #0f172a !important;
|
||||
}
|
||||
|
||||
[data-theme="dark"] .applications-container #settings .modal-body .manage-apps-version-row.ng-binding {
|
||||
color: #0f172a !important;
|
||||
-webkit-text-fill-color: #0f172a !important;
|
||||
}
|
||||
|
||||
.applications-container #settings .modal-body .manage-apps-version-row.ng-binding.is-active {
|
||||
color: #1e1b4b !important;
|
||||
-webkit-text-fill-color: #1e1b4b !important;
|
||||
}
|
||||
|
||||
.install-log {
|
||||
background: var(--bg-secondary, #f8f9ff);
|
||||
border: 1px solid var(--border-color, #e8e9ff);
|
||||
@@ -358,6 +463,7 @@
|
||||
|
||||
<div class="applications-wrapper">
|
||||
<div class="applications-container" ng-controller="manageApplications">
|
||||
<script type="application/json" id="manageApplicationsMetaBootstrap">{{ application_meta_bootstrap_json|safe }}</script>
|
||||
<!-- Page Header -->
|
||||
<div class="page-header">
|
||||
<h1>
|
||||
@@ -373,7 +479,7 @@
|
||||
<div class="content-section">
|
||||
<h2 class="section-title">
|
||||
{% trans "Available Applications" %}
|
||||
<span ng-hide="cyberpanelLoading" class="loading-spinner"></span>
|
||||
<span ng-cloak ng-hide="cyberpanelLoading" class="loading-spinner"></span>
|
||||
</h2>
|
||||
|
||||
{% if services %}
|
||||
@@ -397,6 +503,21 @@
|
||||
{% trans "Not Installed" %}
|
||||
</span>
|
||||
{% endif %}
|
||||
<div ng-if="findAppMeta('{{ service.name }}').installed && findAppMeta('{{ service.name }}').updateAvailable" style="margin-top: 8px;">
|
||||
<span class="app-status" style="background:#dbeafe;color:#1e40af;font-size:12px;padding:4px 10px;border-radius:6px;display:inline-flex;align-items:center;gap:6px;">
|
||||
<i class="fas fa-arrow-circle-up"></i>
|
||||
{% trans "Update available" %}
|
||||
</span>
|
||||
</div>
|
||||
{% if service.installedVersion %}
|
||||
<div style="font-size: 12px; margin-top: 8px; color: #64748b;">
|
||||
{% trans "Installed Version" %}: {{ service.installedVersion }}
|
||||
</div>
|
||||
{% else %}
|
||||
<div style="font-size: 12px; margin-top: 8px; color: #64748b;" ng-if="findAppMeta('{{ service.name }}').installedVersion">
|
||||
{% trans "Installed Version" %}: {$ findAppMeta('{{ service.name }}').installedVersion $}
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -413,25 +534,29 @@
|
||||
</div>
|
||||
|
||||
<div class="app-actions">
|
||||
{% if service.installed == 'Installed' %}
|
||||
<button type="button"
|
||||
class="action-btn remove"
|
||||
data-toggle="modal"
|
||||
data-target="#settings"
|
||||
ng-click="removeInstall('{{ service.name }}', 'Removing')">
|
||||
<i class="fas fa-trash-alt"></i>
|
||||
{% trans "Remove" %}
|
||||
</button>
|
||||
{% else %}
|
||||
<button type="button"
|
||||
class="action-btn install"
|
||||
data-toggle="modal"
|
||||
data-target="#settings"
|
||||
ng-click="removeInstall('{{ service.name }}', 'Installing')">
|
||||
<button type="button"
|
||||
class="action-btn install"
|
||||
{% if service.installed == 'Installed' %}style="display:none;"{% endif %}
|
||||
ng-click="openApplicationsModal('{{ service.name }}', 'Installing')">
|
||||
<i class="fas fa-download"></i>
|
||||
{% trans "Install" %}
|
||||
</button>
|
||||
{% endif %}
|
||||
|
||||
<button type="button"
|
||||
class="action-btn remove"
|
||||
{% if service.installed != 'Installed' %}style="display:none;"{% endif %}
|
||||
ng-click="openApplicationsModal('{{ service.name }}', 'Removing', '{{ service.installedVersion|escapejs }}')">
|
||||
<i class="fas fa-trash-alt"></i>
|
||||
{% trans "Remove" %}
|
||||
</button>
|
||||
|
||||
<button type="button"
|
||||
class="action-btn install"
|
||||
{% if service.installed != 'Installed' %}style="display:none;"{% endif %}
|
||||
ng-click="openApplicationsModal('{{ service.name }}', 'Upgrading', '{{ service.installedVersion|escapejs }}')">
|
||||
<i class="fas fa-exchange-alt"></i>
|
||||
{% trans "Change version" %}
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
{% endfor %}
|
||||
@@ -453,12 +578,87 @@
|
||||
<div class="modal-content">
|
||||
<div class="modal-header">
|
||||
<h4 class="modal-title">
|
||||
{$ status $} {$ appName $}
|
||||
<span ng-hide="cyberpanelLoading" class="loading-spinner"></span>
|
||||
<span ng-if="status == 'Installing'">{% trans "Installing" %}</span>
|
||||
<span ng-if="status == 'Removing'">{% trans "Removing" %}</span>
|
||||
<span ng-if="status == 'Upgrading'">{% trans "Change version" %}</span>
|
||||
{$ appName $}
|
||||
<span ng-cloak ng-show="appsMetaRefreshing || !cyberpanelLoading" class="loading-spinner"></span>
|
||||
</h4>
|
||||
<button type="button" class="close" data-dismiss="modal">×</button>
|
||||
</div>
|
||||
<div class="modal-body">
|
||||
<div style="margin-bottom: 10px;" ng-if="status == 'Installing' || status == 'Upgrading'">
|
||||
<label style="color: #0f172a;">{% trans "Version" %}</label>
|
||||
<div class="manage-apps-version-picker">
|
||||
<div class="manage-apps-version-current">
|
||||
{% trans "Selected" %}: <span>{$ versionLabel(selectedVersion) $}</span>
|
||||
</div>
|
||||
<div class="manage-apps-version-rows">
|
||||
<button type="button"
|
||||
class="manage-apps-version-row"
|
||||
ng-repeat="v in selectedVersions track by versionTrackId($index, v)"
|
||||
ng-class="{'is-active': selectedVersionRowIndex === $index}"
|
||||
ng-click="selectManagedAppVersion($index, v, $event)">{$ versionLabel(v) $}</button>
|
||||
</div>
|
||||
</div>
|
||||
<p style="margin-top: 8px; font-size: 12px; color: #475569;" ng-if="status == 'Upgrading'">
|
||||
{% trans "There is no separate Downgrade button: pick any version lower than your current one in this list to downgrade (the installer allows package downgrades when needed)." %}
|
||||
</p>
|
||||
<p style="margin-top: 6px; font-size: 12px; color: #b45309;" ng-if="status == 'Upgrading' && repoShowsOnlyOneStream">
|
||||
{% trans "Only one package version is visible from your enabled repositories, so there may be nothing older to select. Older builds often require archive or vault repositories for your OS, or another distro major that still publishes them." %}
|
||||
</p>
|
||||
</div>
|
||||
<div style="margin-bottom: 10px; font-size: 13px; color: #64748b;" ng-if="status == 'Upgrading' && selectedCurrentVersion">
|
||||
{% trans "Current Version" %}: {$ selectedCurrentVersion $}
|
||||
</div>
|
||||
<div style="margin-bottom: 10px; padding: 8px 10px; background: #f8fafc; border-radius: 6px; font-size: 12px; color: #475569;"
|
||||
ng-if="status == 'Upgrading'">
|
||||
{% trans "A backup of application data is created automatically before upgrading or downgrading. Data is merged into the new version after packages install; the backup is removed only after a successful start." %}
|
||||
</div>
|
||||
<div style="margin-bottom: 10px;" ng-if="appName == 'Elasticsearch' && (status == 'Installing' || status == 'Upgrading')">
|
||||
<label style="color: #0f172a;">{% trans "Elasticsearch Major" %}</label>
|
||||
<div class="manage-apps-version-picker">
|
||||
<div class="manage-apps-version-current">{% trans "Major" %}: <span ng-bind="selectedEsMajor"></span></div>
|
||||
<div class="manage-apps-version-rows">
|
||||
<button type="button" class="manage-apps-version-row" ng-class="{'is-active': selectedEsMajor === '7'}" ng-click="selectedEsMajor = '7'; refreshMeta()">7</button>
|
||||
<button type="button" class="manage-apps-version-row" ng-class="{'is-active': selectedEsMajor === '8'}" ng-click="selectedEsMajor = '8'; refreshMeta()">8</button>
|
||||
<button type="button" class="manage-apps-version-row" ng-class="{'is-active': selectedEsMajor === '9'}" ng-click="selectedEsMajor = '9'; refreshMeta()">9</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div style="margin-bottom: 10px;" ng-if="appName == 'RabbitMQ' && (status == 'Installing' || status == 'Upgrading')">
|
||||
<label style="color: #0f172a;">{% trans "RabbitMQ major line" %}</label>
|
||||
<div class="manage-apps-version-picker">
|
||||
<div class="manage-apps-version-current">{% trans "Stream" %}: <span ng-bind="selectedRabbitmqStream === '4' ? '4.x' : '3.x'"></span></div>
|
||||
<div class="manage-apps-version-rows">
|
||||
<button type="button" class="manage-apps-version-row" ng-class="{'is-active': selectedRabbitmqStream === '3'}" ng-click="selectedRabbitmqStream = '3'; refreshMeta()">3.x ({% trans "maintenance" %})</button>
|
||||
<button type="button" class="manage-apps-version-row" ng-class="{'is-active': selectedRabbitmqStream === '4'}" ng-click="selectedRabbitmqStream = '4'; refreshMeta()">4.x</button>
|
||||
</div>
|
||||
</div>
|
||||
<p style="margin-top: 8px; font-size: 12px; color: #64748b;">
|
||||
{% trans "Uses Team RabbitMQ Packagecloud repos; 4.x requires a compatible Erlang (OTP 26+). The installer will try to align Erlang when needed." %}
|
||||
</p>
|
||||
</div>
|
||||
<div style="margin-bottom: 10px; padding: 10px 12px; background: #f0f9ff; border-radius: 8px; font-size: 13px; color: #1e3a5f; border: 1px solid #bae6fd;"
|
||||
ng-if="(status == 'Installing' || status == 'Upgrading') && findAppMeta(appName).crossBranchUpdateSuggested">
|
||||
<i class="fas fa-info-circle"></i>
|
||||
{% trans "A newer release is available on another version line. Change the major/stream above, pick a version, then run a version change (upgrade or downgrade)." %}
|
||||
<span ng-if="findAppMeta(appName).latestOverall"> {$ findAppMeta(appName).latestOverall $}</span>
|
||||
</div>
|
||||
<div style="margin-bottom: 10px;" ng-if="status == 'Removing' || status == 'Upgrading'">
|
||||
<label>
|
||||
<input type="checkbox" ng-model="confirmAction">
|
||||
<span ng-if="status == 'Removing'">{% trans "Confirm Remove" %}</span>
|
||||
<span ng-if="status == 'Upgrading'">{% trans "Confirm version change" %}</span>
|
||||
</label>
|
||||
</div>
|
||||
<div style="margin-bottom: 10px;">
|
||||
<button type="button" class="btn btn-primary" ng-click="runAction()">
|
||||
<span ng-if="status == 'Installing'">{% trans "Start Install" %}</span>
|
||||
<span ng-if="status == 'Removing'">{% trans "Start Remove" %}</span>
|
||||
<span ng-if="status == 'Upgrading'">{% trans "Start version change" %}</span>
|
||||
</button>
|
||||
</div>
|
||||
<div class="install-log">
|
||||
<textarea ng-model="requestData" rows="15" class="log-textarea" readonly></textarea>
|
||||
</div>
|
||||
@@ -469,6 +669,4 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Manage Applications JS -->
|
||||
<script src="{% static 'manageServices/manageServices.js' %}"></script>
|
||||
{% endblock %}
|
||||
@@ -10,5 +10,6 @@ urlpatterns = [
|
||||
path('saveStatus', views.saveStatus, name='saveStatus'),
|
||||
|
||||
path('manageApplications', views.manageApplications, name='manageApplications'),
|
||||
path('applicationMeta', views.applicationMeta, name='applicationMeta'),
|
||||
path('removeInstall', views.removeInstall, name='removeInstall'),
|
||||
]
|
||||
|
||||
@@ -5,6 +5,7 @@ import plogical.CyberCPLogFileWriter as logging
|
||||
from loginSystem.views import loadLoginPage
|
||||
import os
|
||||
import json
|
||||
import shlex
|
||||
|
||||
from plogical.httpProc import httpProc
|
||||
from plogical.mailUtilities import mailUtilities
|
||||
@@ -12,6 +13,11 @@ from plogical.acl import ACLManager
|
||||
from .models import PDNSStatus, SlaveServers
|
||||
from .serviceManager import ServiceManager
|
||||
from plogical.processUtilities import ProcessUtilities
|
||||
from .application_detection import managed_apps_os_support
|
||||
from .application_page_meta import (
|
||||
build_manage_applications_page_data,
|
||||
get_application_meta_response_dict,
|
||||
)
|
||||
# Create your views here.
|
||||
|
||||
def managePowerDNS(request):
|
||||
@@ -270,43 +276,57 @@ def saveStatus(request):
|
||||
return HttpResponse(json_data)
|
||||
|
||||
def manageApplications(request):
|
||||
services = []
|
||||
services, application_meta_bootstrap_json = build_manage_applications_page_data(
|
||||
'8', '3'
|
||||
)
|
||||
|
||||
## ElasticSearch
|
||||
|
||||
esPath = '/home/cyberpanel/elasticsearch'
|
||||
rPath = '/home/cyberpanel/redis'
|
||||
rmqPath = '/home/cyberpanel/rabbitmq'
|
||||
|
||||
if os.path.exists(esPath):
|
||||
installed = 'Installed'
|
||||
else:
|
||||
installed = 'Not-Installed'
|
||||
|
||||
if os.path.exists(rPath):
|
||||
rInstalled = 'Installed'
|
||||
else:
|
||||
rInstalled = 'Not-Installed'
|
||||
|
||||
if os.path.exists(rmqPath):
|
||||
rmqInstalled = 'Installed'
|
||||
else:
|
||||
rmqInstalled = 'Not-Installed'
|
||||
|
||||
elasticSearch = {'image': '/static/manageServices/images/elastic-search.png', 'name': 'Elasticsearch',
|
||||
'installed': installed}
|
||||
redis = {'image': '/static/manageServices/images/redis.png', 'name': 'Redis',
|
||||
'installed': rInstalled}
|
||||
rabbitmq = {'image': '/static/manageServices/images/rabbitmq-logo.svg', 'name': 'RabbitMQ',
|
||||
'installed': rmqInstalled}
|
||||
services.append(elasticSearch)
|
||||
services.append(redis)
|
||||
services.append(rabbitmq)
|
||||
|
||||
proc = httpProc(request, 'manageServices/applications.html',
|
||||
{'services': services}, 'admin')
|
||||
proc = httpProc(
|
||||
request,
|
||||
'manageServices/applications.html',
|
||||
{
|
||||
'services': services,
|
||||
'application_meta_bootstrap_json': application_meta_bootstrap_json,
|
||||
},
|
||||
'admin',
|
||||
)
|
||||
return proc.render()
|
||||
|
||||
|
||||
def applicationMeta(request):
|
||||
try:
|
||||
userID = request.session['userID']
|
||||
currentACL = ACLManager.loadedACL(userID)
|
||||
|
||||
if currentACL['admin'] != 1:
|
||||
return ACLManager.loadErrorJson()
|
||||
|
||||
data = {}
|
||||
if request.method == 'POST':
|
||||
data = json.loads(request.body)
|
||||
|
||||
requested_major = str(data.get('esMajor', '8'))
|
||||
if requested_major not in ('7', '8', '9'):
|
||||
requested_major = '8'
|
||||
|
||||
requested_rmq_stream = str(data.get('rabbitmqStream', '3')).strip()
|
||||
if requested_rmq_stream not in ('3', '4'):
|
||||
requested_rmq_stream = '3'
|
||||
|
||||
response_data = get_application_meta_response_dict(
|
||||
requested_major, requested_rmq_stream
|
||||
)
|
||||
|
||||
return HttpResponse(
|
||||
json.dumps(response_data, ensure_ascii=False),
|
||||
content_type='application/json; charset=utf-8',
|
||||
)
|
||||
|
||||
except BaseException as msg:
|
||||
return HttpResponse(
|
||||
json.dumps({'status': 0, 'error_message': str(msg)}, ensure_ascii=False),
|
||||
content_type='application/json; charset=utf-8',
|
||||
)
|
||||
|
||||
def removeInstall(request):
|
||||
try:
|
||||
userID = request.session['userID']
|
||||
@@ -321,22 +341,59 @@ def removeInstall(request):
|
||||
|
||||
status = data['status']
|
||||
appName = data['appName']
|
||||
version = str(data.get('version', 'latest')).strip() or 'latest'
|
||||
esMajor = str(data.get('esMajor', '8')).strip() or '8'
|
||||
if esMajor not in ('7', '8', '9'):
|
||||
esMajor = '8'
|
||||
rabbitmqStream = str(data.get('rabbitmqStream', '3')).strip() or '3'
|
||||
if rabbitmqStream not in ('3', '4'):
|
||||
rabbitmqStream = '3'
|
||||
confirmAction = bool(data.get('confirmAction', False))
|
||||
|
||||
support = managed_apps_os_support()
|
||||
if not support['supported']:
|
||||
data_ret = {'status': 0, 'error_message': support['reason']}
|
||||
json_data = json.dumps(data_ret)
|
||||
return HttpResponse(json_data)
|
||||
|
||||
if status in ('Removing', 'Upgrading') and not confirmAction:
|
||||
data_ret = {'status': 0, 'error_message': 'Action confirmation is required.'}
|
||||
json_data = json.dumps(data_ret)
|
||||
return HttpResponse(json_data)
|
||||
|
||||
if appName == 'Elasticsearch':
|
||||
if status == 'Installing':
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --function InstallElasticSearch'
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --app Elasticsearch --action install --version {0} --esMajor {1}'.format(
|
||||
shlex.quote(version), shlex.quote(esMajor)
|
||||
)
|
||||
elif status == 'Upgrading':
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --app Elasticsearch --action upgrade --version {0} --esMajor {1}'.format(
|
||||
shlex.quote(version), shlex.quote(esMajor)
|
||||
)
|
||||
else:
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --function RemoveElasticSearch'
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --app Elasticsearch --action remove'
|
||||
elif appName == 'Redis':
|
||||
if status == 'Installing':
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --function InstallRedis'
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --app Redis --action install --version {0}'.format(
|
||||
shlex.quote(version)
|
||||
)
|
||||
elif status == 'Upgrading':
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --app Redis --action upgrade --version {0}'.format(
|
||||
shlex.quote(version)
|
||||
)
|
||||
else:
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --function RemoveRedis'
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --app Redis --action remove'
|
||||
elif appName == 'RabbitMQ':
|
||||
if status == 'Installing':
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --function InstallRabbitMQ'
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --app RabbitMQ --action install --version {0} --rabbitmqStream {1}'.format(
|
||||
shlex.quote(version), shlex.quote(rabbitmqStream)
|
||||
)
|
||||
elif status == 'Upgrading':
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --app RabbitMQ --action upgrade --version {0} --rabbitmqStream {1}'.format(
|
||||
shlex.quote(version), shlex.quote(rabbitmqStream)
|
||||
)
|
||||
else:
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --function RemoveRabbitMQ'
|
||||
command = '/usr/local/CyberCP/bin/python /usr/local/CyberCP/manageServices/serviceManager.py --app RabbitMQ --action remove'
|
||||
else:
|
||||
data_ret = {'status': 0, 'error_message': 'Unknown application selected.'}
|
||||
json_data = json.dumps(data_ret)
|
||||
|
||||
@@ -4464,6 +4464,56 @@ class Migration(migrations.Migration):
|
||||
except BaseException as msg:
|
||||
Upgrade.stdOut('RabbitMQ migration failed: ' + str(msg), 0)
|
||||
|
||||
@staticmethod
|
||||
def redisMigrations():
|
||||
marker_path = '/home/cyberpanel/redis'
|
||||
redis_binary = '/usr/bin/redis-server'
|
||||
redis_service_files = [
|
||||
'/usr/lib/systemd/system/redis.service',
|
||||
'/lib/systemd/system/redis.service'
|
||||
]
|
||||
|
||||
try:
|
||||
redis_installed = os.path.exists(redis_binary) or any(os.path.exists(path) for path in redis_service_files)
|
||||
if redis_installed:
|
||||
if not os.path.exists(marker_path):
|
||||
writeToFile = open(marker_path, 'w+')
|
||||
writeToFile.close()
|
||||
Upgrade.stdOut('Redis detected during upgrade. Marker file created.', 0)
|
||||
Upgrade.executioner('systemctl enable redis', 'Enable Redis service', 0)
|
||||
Upgrade.executioner('systemctl start redis', 'Start Redis service', 0)
|
||||
else:
|
||||
if os.path.exists(marker_path):
|
||||
os.remove(marker_path)
|
||||
Upgrade.stdOut('Redis marker removed because service is not installed.', 0)
|
||||
except BaseException as msg:
|
||||
Upgrade.stdOut('Redis migration failed: ' + str(msg), 0)
|
||||
|
||||
@staticmethod
|
||||
def elasticSearchMigrations():
|
||||
marker_path = '/home/cyberpanel/elasticsearch'
|
||||
es_binary = '/usr/share/elasticsearch/bin/elasticsearch'
|
||||
es_service_files = [
|
||||
'/usr/lib/systemd/system/elasticsearch.service',
|
||||
'/lib/systemd/system/elasticsearch.service'
|
||||
]
|
||||
|
||||
try:
|
||||
es_installed = os.path.exists(es_binary) or any(os.path.exists(path) for path in es_service_files)
|
||||
if es_installed:
|
||||
if not os.path.exists(marker_path):
|
||||
writeToFile = open(marker_path, 'w+')
|
||||
writeToFile.close()
|
||||
Upgrade.stdOut('Elasticsearch detected during upgrade. Marker file created.', 0)
|
||||
Upgrade.executioner('systemctl enable elasticsearch', 'Enable Elasticsearch service', 0)
|
||||
Upgrade.executioner('systemctl start elasticsearch', 'Start Elasticsearch service', 0)
|
||||
else:
|
||||
if os.path.exists(marker_path):
|
||||
os.remove(marker_path)
|
||||
Upgrade.stdOut('Elasticsearch marker removed because service is not installed.', 0)
|
||||
except BaseException as msg:
|
||||
Upgrade.stdOut('Elasticsearch migration failed: ' + str(msg), 0)
|
||||
|
||||
@staticmethod
|
||||
def backupCriticalFiles():
|
||||
"""Backup all critical configuration files before upgrade"""
|
||||
@@ -6682,6 +6732,8 @@ slowlog = /var/log/php{version}-fpm-slow.log
|
||||
Upgrade.setupWebmail()
|
||||
Upgrade.setupSieve()
|
||||
Upgrade.enableServices()
|
||||
Upgrade.elasticSearchMigrations()
|
||||
Upgrade.redisMigrations()
|
||||
Upgrade.rabbitMQMigrations()
|
||||
|
||||
# Apply AlmaLinux 9 fixes before other installations
|
||||
|
||||
@@ -428,10 +428,411 @@ app.controller('pureFTPD', function ($scope, $http, $timeout, $window) {
|
||||
|
||||
app.controller('manageApplications', function ($scope, $http, $timeout, $window) {
|
||||
|
||||
/**
|
||||
* Normalize entries from applicationMeta (strings, numbers, or rare object shapes)
|
||||
* so version pickers never show blank rows. CyberPanel uses {$ ... $} interpolation
|
||||
* in templates; list labels use versionLabel() for consistent display.
|
||||
*/
|
||||
function normalizeVersionToken(v) {
|
||||
if (v === null || v === undefined) {
|
||||
return '';
|
||||
}
|
||||
if (v === 'latest') {
|
||||
return 'latest';
|
||||
}
|
||||
if (typeof v === 'number' && isFinite(v)) {
|
||||
return String(v);
|
||||
}
|
||||
if (angular.isObject(v)) {
|
||||
var o = v;
|
||||
var cand = o.version || o.Version || o.value || o.name || o.ver || o.label;
|
||||
if (cand !== undefined && cand !== null) {
|
||||
return String(cand).trim();
|
||||
}
|
||||
try {
|
||||
return JSON.stringify(o);
|
||||
} catch (ignore) {
|
||||
return '';
|
||||
}
|
||||
}
|
||||
return String(v).trim();
|
||||
}
|
||||
|
||||
function sanitizeVersionsArray(vers) {
|
||||
if (!angular.isArray(vers)) {
|
||||
return [];
|
||||
}
|
||||
var out = [];
|
||||
var seen = {};
|
||||
vers.forEach(function (raw) {
|
||||
var t = normalizeVersionToken(raw);
|
||||
if (!t || t === 'latest') {
|
||||
return;
|
||||
}
|
||||
if (!seen[t]) {
|
||||
seen[t] = true;
|
||||
out.push(t);
|
||||
}
|
||||
});
|
||||
return out;
|
||||
}
|
||||
|
||||
$scope.versionLabel = function (v) {
|
||||
if (v === 'latest') {
|
||||
return 'latest';
|
||||
}
|
||||
var t = normalizeVersionToken(v);
|
||||
return t || '(unknown)';
|
||||
};
|
||||
|
||||
$scope.versionTrackId = function (idx, v) {
|
||||
return String(idx) + '|' + $scope.versionLabel(v);
|
||||
};
|
||||
|
||||
/* false = long-running install/remove/poll (show spinners); true = idle */
|
||||
$scope.cyberpanelLoading = true;
|
||||
/** Background applicationMeta refresh — separate from cyberpanelLoading so the page does not “freeze” on every modal open. */
|
||||
$scope.appsMetaRefreshing = false;
|
||||
$scope.apps = [
|
||||
{name: 'Elasticsearch', image: '/static/manageServices/images/elastic-search.png'},
|
||||
{name: 'Redis', image: '/static/manageServices/images/redis.png'},
|
||||
{name: 'RabbitMQ', image: '/static/manageServices/images/rabbitmq-logo.svg'}
|
||||
];
|
||||
|
||||
$scope.removeInstall = function (appName, status) {
|
||||
(function mergeMetaBootstrap() {
|
||||
var el = document.getElementById('manageApplicationsMetaBootstrap');
|
||||
if (!el || !el.textContent) {
|
||||
return;
|
||||
}
|
||||
var raw = el.textContent.trim();
|
||||
if (!raw) {
|
||||
return;
|
||||
}
|
||||
try {
|
||||
var boot = JSON.parse(raw);
|
||||
if (!boot || Number(boot.status) !== 1) {
|
||||
return;
|
||||
}
|
||||
var appMap = {};
|
||||
(boot.apps || []).forEach(function (a) {
|
||||
if (a && a.name) {
|
||||
appMap[a.name] = a;
|
||||
}
|
||||
});
|
||||
$scope.apps = $scope.apps.map(function (baseApp) {
|
||||
var meta = appMap[baseApp.name] || {};
|
||||
var vers = meta.versions;
|
||||
if (!angular.isArray(vers)) {
|
||||
vers = [];
|
||||
}
|
||||
vers = sanitizeVersionsArray(vers);
|
||||
return {
|
||||
name: baseApp.name,
|
||||
image: baseApp.image,
|
||||
installed: !!meta.installed,
|
||||
installedVersion: meta.installedVersion || '',
|
||||
updateAvailable: !!meta.updateAvailable,
|
||||
crossBranchUpdateSuggested: !!meta.crossBranchUpdateSuggested,
|
||||
versions: vers,
|
||||
latestAvailable: meta.latestAvailable || '',
|
||||
latestOverall: meta.latestOverall || ''
|
||||
};
|
||||
});
|
||||
} catch (ignore) {
|
||||
/* keep bare apps list */
|
||||
}
|
||||
})();
|
||||
|
||||
$scope.selectedVersion = 'latest';
|
||||
/** Row highlight uses $index so only one row looks selected (avoids sticky :focus / repeater quirks). */
|
||||
$scope.selectedVersionRowIndex = 0;
|
||||
$scope.selectedVersions = ['latest'];
|
||||
|
||||
$scope.recalcSelectedVersionRowIndex = function () {
|
||||
var list = $scope.selectedVersions || [];
|
||||
var sel = $scope.selectedVersion;
|
||||
var i = list.indexOf(sel);
|
||||
if (i < 0) {
|
||||
var normSel = normalizeVersionToken(sel);
|
||||
if (normSel) {
|
||||
for (var j = 0; j < list.length; j += 1) {
|
||||
if (normalizeVersionToken(list[j]) === normSel) {
|
||||
i = j;
|
||||
$scope.selectedVersion = list[j];
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
$scope.selectedVersionRowIndex = (i >= 0) ? i : 0;
|
||||
};
|
||||
|
||||
$scope.selectManagedAppVersion = function (idx, v, $event) {
|
||||
var n = (typeof idx === 'number') ? idx : parseInt(idx, 10);
|
||||
if (!isFinite(n) || n < 0) {
|
||||
n = 0;
|
||||
}
|
||||
$scope.selectedVersionRowIndex = n;
|
||||
$scope.selectedVersion = v;
|
||||
if ($event && $event.target && typeof $event.target.blur === 'function') {
|
||||
$event.target.blur();
|
||||
} else if (typeof document !== 'undefined' && document.activeElement && typeof document.activeElement.blur === 'function') {
|
||||
document.activeElement.blur();
|
||||
}
|
||||
};
|
||||
$scope.selectedEsMajor = '8';
|
||||
$scope.selectedRabbitmqStream = '3';
|
||||
$scope.confirmAction = false;
|
||||
$scope.selectedCurrentVersion = '';
|
||||
|
||||
/**
|
||||
* When the install/upgrade modal is open, re-apply version list from latest applicationMeta.
|
||||
* (Page-load meta can be empty for ES if dnf was slow; opening the modal must refetch.)
|
||||
*/
|
||||
$scope.syncModalVersionLists = function () {
|
||||
if (!$scope.appName || ($scope.status !== 'Installing' && $scope.status !== 'Upgrading')) {
|
||||
return;
|
||||
}
|
||||
if ($scope.appName !== 'Elasticsearch' && $scope.appName !== 'Redis' && $scope.appName !== 'RabbitMQ') {
|
||||
return;
|
||||
}
|
||||
var meta = $scope.findAppMeta($scope.appName);
|
||||
var vers = sanitizeVersionsArray((meta && meta.versions) ? meta.versions : []);
|
||||
$scope.selectedVersions = ['latest'].concat(vers);
|
||||
var curRaw = (meta && meta.installedVersion) ? meta.installedVersion : ($scope.selectedCurrentVersion || '');
|
||||
var cur = normalizeVersionToken(curRaw) || String(curRaw || '').trim();
|
||||
if (cur && $scope.selectedVersions.indexOf(cur) === -1) {
|
||||
$scope.selectedVersions.push(cur);
|
||||
}
|
||||
if (cur) {
|
||||
$scope.selectedCurrentVersion = cur;
|
||||
}
|
||||
var prevSel = $scope.selectedVersion;
|
||||
if (prevSel && $scope.selectedVersions.indexOf(prevSel) !== -1) {
|
||||
$scope.selectedVersion = prevSel;
|
||||
} else {
|
||||
$scope.selectedVersion = 'latest';
|
||||
}
|
||||
var realVers = ($scope.selectedVersions || []).filter(function (v) {
|
||||
return v && v !== 'latest';
|
||||
});
|
||||
$scope.repoShowsOnlyOneStream = ($scope.status === 'Upgrading' && realVers.length <= 1);
|
||||
$scope.recalcSelectedVersionRowIndex();
|
||||
};
|
||||
|
||||
$scope.refreshMeta = function () {
|
||||
$scope.appsMetaRefreshing = true;
|
||||
var url = "/manageservices/applicationMeta";
|
||||
var data = {
|
||||
esMajor: $scope.selectedEsMajor,
|
||||
rabbitmqStream: $scope.selectedRabbitmqStream
|
||||
};
|
||||
var config = {
|
||||
headers: {
|
||||
'Content-Type': 'application/json;charset=UTF-8',
|
||||
'X-CSRFToken': getCookie('csrftoken')
|
||||
},
|
||||
transformRequest: function (payload) {
|
||||
return angular.toJson(payload);
|
||||
}
|
||||
};
|
||||
|
||||
return $http.post(url, data, config).then(function (response) {
|
||||
$scope.appsMetaRefreshing = false;
|
||||
var payload = response.data;
|
||||
var ok = payload && (payload.status === 1 || payload.status === '1');
|
||||
if (ok) {
|
||||
var appMap = {};
|
||||
(payload.apps || []).forEach(function (app) {
|
||||
appMap[app.name] = app;
|
||||
});
|
||||
$scope.apps = $scope.apps.map(function (baseApp) {
|
||||
var meta = appMap[baseApp.name] || {};
|
||||
var vers = meta.versions;
|
||||
if (!angular.isArray(vers)) {
|
||||
vers = [];
|
||||
}
|
||||
vers = sanitizeVersionsArray(vers);
|
||||
return {
|
||||
name: baseApp.name,
|
||||
image: baseApp.image,
|
||||
installed: !!meta.installed,
|
||||
installedVersion: meta.installedVersion || '',
|
||||
updateAvailable: !!meta.updateAvailable,
|
||||
crossBranchUpdateSuggested: !!meta.crossBranchUpdateSuggested,
|
||||
versions: vers,
|
||||
latestAvailable: meta.latestAvailable || '',
|
||||
latestOverall: meta.latestOverall || ''
|
||||
};
|
||||
});
|
||||
$scope.syncModalVersionLists();
|
||||
} else {
|
||||
new PNotify({
|
||||
title: 'Operation Failed!',
|
||||
text: (payload && (payload.error_message || payload.errorMessage)) || 'Could not load application metadata.',
|
||||
type: 'error'
|
||||
});
|
||||
}
|
||||
}, function () {
|
||||
$scope.appsMetaRefreshing = false;
|
||||
new PNotify({
|
||||
title: 'Operation Failed!',
|
||||
text: 'Could not connect to server, please refresh this page',
|
||||
type: 'error'
|
||||
});
|
||||
});
|
||||
};
|
||||
|
||||
$scope.prepareAction = function (service, status, bootstrapInstalledVersion) {
|
||||
if (bootstrapInstalledVersion === undefined || bootstrapInstalledVersion === null) {
|
||||
bootstrapInstalledVersion = '';
|
||||
} else {
|
||||
bootstrapInstalledVersion = String(bootstrapInstalledVersion).trim();
|
||||
}
|
||||
$scope.status = status;
|
||||
$scope.appName = service.name;
|
||||
$scope.confirmAction = false;
|
||||
var effectiveInstalled = (service.installedVersion || bootstrapInstalledVersion || '').trim();
|
||||
$scope.selectedCurrentVersion = effectiveInstalled;
|
||||
|
||||
if (service.name === 'RabbitMQ') {
|
||||
if (effectiveInstalled && /^4\./.test(effectiveInstalled)) {
|
||||
$scope.selectedRabbitmqStream = '4';
|
||||
} else if (effectiveInstalled && /^3\./.test(effectiveInstalled)) {
|
||||
$scope.selectedRabbitmqStream = '3';
|
||||
}
|
||||
}
|
||||
|
||||
if (service.name === 'Elasticsearch' && effectiveInstalled) {
|
||||
var iv = effectiveInstalled;
|
||||
if (/^9\./.test(iv)) {
|
||||
$scope.selectedEsMajor = '9';
|
||||
} else if (/^8\./.test(iv)) {
|
||||
$scope.selectedEsMajor = '8';
|
||||
} else if (/^7\./.test(iv)) {
|
||||
$scope.selectedEsMajor = '7';
|
||||
}
|
||||
}
|
||||
|
||||
$scope.selectedVersions = ['latest'];
|
||||
var svcVers = sanitizeVersionsArray(service.versions || []);
|
||||
if (svcVers.length > 0) {
|
||||
$scope.selectedVersions = ['latest'].concat(svcVers);
|
||||
}
|
||||
var curPick = normalizeVersionToken(effectiveInstalled) || effectiveInstalled;
|
||||
if (curPick && $scope.selectedVersions.indexOf(curPick) === -1) {
|
||||
$scope.selectedVersions.push(curPick);
|
||||
}
|
||||
$scope.selectedVersion = 'latest';
|
||||
$scope.requestData = '';
|
||||
|
||||
var realVers = ($scope.selectedVersions || []).filter(function (v) {
|
||||
return v && v !== 'latest';
|
||||
});
|
||||
$scope.repoShowsOnlyOneStream = ($scope.status === 'Upgrading' && realVers.length <= 1);
|
||||
$scope.recalcSelectedVersionRowIndex();
|
||||
};
|
||||
|
||||
$scope.findAppMeta = function (appName) {
|
||||
var found = null;
|
||||
($scope.apps || []).forEach(function (item) {
|
||||
if (item.name === appName) {
|
||||
found = item;
|
||||
}
|
||||
});
|
||||
return found || {};
|
||||
};
|
||||
|
||||
$scope.prepareActionByName = function (appName, status, bootstrapInstalledVersion) {
|
||||
var meta = $scope.findAppMeta(appName);
|
||||
if (!meta.name) {
|
||||
meta = {name: appName, versions: []};
|
||||
}
|
||||
var mver = meta.versions;
|
||||
if (!angular.isArray(mver)) {
|
||||
mver = [];
|
||||
}
|
||||
var merged = {
|
||||
name: meta.name,
|
||||
image: meta.image,
|
||||
installed: meta.installed,
|
||||
installedVersion: meta.installedVersion || '',
|
||||
versions: mver
|
||||
};
|
||||
$scope.prepareAction(merged, status, bootstrapInstalledVersion);
|
||||
};
|
||||
|
||||
/**
|
||||
* Prepare scope then show modal (do not use data-toggle + ng-click — Bootstrap can
|
||||
* open the dialog before Angular runs ng-click, leaving status/appName unset).
|
||||
* For managed apps, wait for applicationMeta so version lists and installedVersion are correct (upgrade vs downgrade).
|
||||
*/
|
||||
$scope.openApplicationsModal = function (appName, status, bootstrapInstalledVersion) {
|
||||
var needMeta = (appName === 'RabbitMQ' || appName === 'Elasticsearch' || appName === 'Redis');
|
||||
var showModal = function () {
|
||||
if (typeof window.jQuery !== 'undefined' && jQuery('#settings').modal) {
|
||||
jQuery('#settings').modal('show');
|
||||
}
|
||||
};
|
||||
if (needMeta) {
|
||||
// applicationMeta uses selectedRabbitmqStream / selectedEsMajor. If the user left
|
||||
// 4.x selected earlier, the first POST would ask for 4.x while the node is on 3.x —
|
||||
// filtered versions come back empty and the Version dropdown shows only "latest".
|
||||
var bootVer = (bootstrapInstalledVersion !== undefined && bootstrapInstalledVersion !== null)
|
||||
? String(bootstrapInstalledVersion).trim()
|
||||
: '';
|
||||
if (!bootVer && appName) {
|
||||
var metaEarly = $scope.findAppMeta(appName);
|
||||
if (metaEarly && metaEarly.installedVersion) {
|
||||
bootVer = String(metaEarly.installedVersion).trim();
|
||||
}
|
||||
}
|
||||
if (appName === 'RabbitMQ' && bootVer) {
|
||||
if (/^4\./.test(bootVer)) {
|
||||
$scope.selectedRabbitmqStream = '4';
|
||||
} else if (/^3\./.test(bootVer)) {
|
||||
$scope.selectedRabbitmqStream = '3';
|
||||
}
|
||||
}
|
||||
if (appName === 'Elasticsearch' && bootVer) {
|
||||
var biv = bootVer;
|
||||
if (/^9\./.test(biv)) {
|
||||
$scope.selectedEsMajor = '9';
|
||||
} else if (/^8\./.test(biv)) {
|
||||
$scope.selectedEsMajor = '8';
|
||||
} else if (/^7\./.test(biv)) {
|
||||
$scope.selectedEsMajor = '7';
|
||||
}
|
||||
}
|
||||
// Open immediately with bootstrap / in-memory meta; refresh in background.
|
||||
// Blocking on applicationMeta made "Change version" feel frozen (dnf/repoquery).
|
||||
$scope.appName = appName;
|
||||
$scope.status = status;
|
||||
$scope.prepareActionByName(appName, status, bootstrapInstalledVersion);
|
||||
showModal();
|
||||
$timeout(function () {
|
||||
$scope.refreshMeta();
|
||||
}, 0);
|
||||
} else {
|
||||
$scope.prepareActionByName(appName, status, bootstrapInstalledVersion);
|
||||
showModal();
|
||||
}
|
||||
};
|
||||
|
||||
$scope.runAction = function () {
|
||||
var appName = $scope.appName;
|
||||
var status = $scope.status;
|
||||
if (!appName || !status) {
|
||||
return;
|
||||
}
|
||||
if ((status === 'Removing' || status === 'Upgrading') && !$scope.confirmAction) {
|
||||
new PNotify({
|
||||
title: 'Confirmation Required',
|
||||
text: 'Please confirm this action before proceeding.',
|
||||
type: 'warning'
|
||||
});
|
||||
return;
|
||||
}
|
||||
$scope.status = status;
|
||||
$scope.appName = appName;
|
||||
|
||||
@@ -441,7 +842,11 @@ app.controller('manageApplications', function ($scope, $http, $timeout, $window)
|
||||
|
||||
var data = {
|
||||
appName: appName,
|
||||
status: status
|
||||
status: status,
|
||||
version: $scope.selectedVersion || 'latest',
|
||||
esMajor: $scope.selectedEsMajor || '8',
|
||||
rabbitmqStream: $scope.selectedRabbitmqStream || '3',
|
||||
confirmAction: $scope.confirmAction === true
|
||||
};
|
||||
|
||||
var config = {
|
||||
@@ -524,6 +929,9 @@ app.controller('manageApplications', function ($scope, $http, $timeout, $window)
|
||||
|
||||
}
|
||||
|
||||
// Do not fetch package metadata on page load; it can block workers under DNF load.
|
||||
// Metadata is fetched on-demand when opening install/version-change modals.
|
||||
|
||||
});
|
||||
|
||||
/* Java script code */
|
||||
@@ -848,9 +848,20 @@ app.controller('createACLCTRL', function ($scope, $http) {
|
||||
|
||||
var url = "/users/createACLFunc";
|
||||
|
||||
var aclNameTrimmed = ($scope.aclName !== undefined && $scope.aclName !== null) ? String($scope.aclName).trim() : '';
|
||||
if (!aclNameTrimmed) {
|
||||
$scope.aclLoading = true;
|
||||
safePNotify({
|
||||
title: 'Error!',
|
||||
text: 'Please enter a name for this ACL.',
|
||||
type: 'error'
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
var data = {
|
||||
|
||||
aclName: $scope.aclName,
|
||||
aclName: aclNameTrimmed,
|
||||
makeAdmin: $scope.makeAdmin,
|
||||
|
||||
//
|
||||
@@ -943,10 +954,10 @@ app.controller('createACLCTRL', function ($scope, $http) {
|
||||
type: 'success'
|
||||
});
|
||||
} else {
|
||||
|
||||
var errText = (response.data && (response.data.errorMessage || response.data.error_message)) ? (response.data.errorMessage || response.data.error_message) : 'Unknown error';
|
||||
safePNotify({
|
||||
title: 'Error!',
|
||||
text: response.data.errorMessage,
|
||||
text: errText,
|
||||
type: 'error'
|
||||
});
|
||||
|
||||
|
||||
@@ -3,6 +3,7 @@
|
||||
|
||||
from django.shortcuts import render, redirect
|
||||
from django.http import HttpResponse
|
||||
from django.utils.translation import gettext as _
|
||||
from django.db import models
|
||||
from django.db.utils import IntegrityError, ProgrammingError, OperationalError
|
||||
from django.views.decorators.csrf import ensure_csrf_cookie
|
||||
@@ -765,9 +766,17 @@ def createACLFunc(request):
|
||||
if currentACL['admin'] == 1:
|
||||
data = json.loads(request.body)
|
||||
|
||||
acl_name_raw = data.get('aclName', '') or ''
|
||||
acl_name = str(acl_name_raw).strip()
|
||||
if not acl_name:
|
||||
msg = str(_('Please enter a name for this ACL.'))
|
||||
err_body = {'status': 0, 'error_message': msg, 'errorMessage': msg}
|
||||
return HttpResponse(json.dumps(err_body), content_type='application/json')
|
||||
data['aclName'] = acl_name
|
||||
|
||||
## Version Management
|
||||
|
||||
if data['makeAdmin']:
|
||||
if data.get('makeAdmin'):
|
||||
data['adminStatus'] = 1
|
||||
else:
|
||||
data['adminStatus'] = 0
|
||||
|
||||
@@ -1,322 +0,0 @@
|
||||
{% extends "baseTemplate/index.html" %}
|
||||
{% load static %}
|
||||
|
||||
{% block title %}
|
||||
Security Management - CyberPanel
|
||||
{% endblock %}
|
||||
|
||||
{% block content %}
|
||||
<div class="container-fluid">
|
||||
<div class="row">
|
||||
<div class="col-12">
|
||||
<div class="card">
|
||||
<div class="card-header">
|
||||
<h3 class="card-title">
|
||||
<i class="fas fa-shield-alt"></i>
|
||||
Security Management
|
||||
</h3>
|
||||
</div>
|
||||
<div class="card-body">
|
||||
<!-- Security Alerts Section -->
|
||||
<div class="row mb-4">
|
||||
<div class="col-12">
|
||||
<div class="alert alert-warning">
|
||||
<h5><i class="fas fa-exclamation-triangle"></i> Security Alerts</h5>
|
||||
<p>Monitor and manage security threats detected by the system.</p>
|
||||
<button class="btn btn-warning" onclick="refreshSecurityAlerts()">
|
||||
<i class="fas fa-sync"></i> Refresh Alerts
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Security Alerts List -->
|
||||
<div class="row">
|
||||
<div class="col-12">
|
||||
<div class="card">
|
||||
<div class="card-header">
|
||||
<h4>Recent Security Alerts</h4>
|
||||
</div>
|
||||
<div class="card-body">
|
||||
<div id="securityAlertsContainer">
|
||||
<!-- Alerts will be loaded here -->
|
||||
<div class="text-center">
|
||||
<i class="fas fa-spinner fa-spin"></i> Loading security alerts...
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Blocked IPs Management -->
|
||||
<div class="row mt-4">
|
||||
<div class="col-12">
|
||||
<div class="card">
|
||||
<div class="card-header">
|
||||
<h4>Blocked IP Addresses</h4>
|
||||
<button class="btn btn-success btn-sm" onclick="refreshBlockedIPs()">
|
||||
<i class="fas fa-sync"></i> Refresh
|
||||
</button>
|
||||
</div>
|
||||
<div class="card-body">
|
||||
<div class="table-responsive">
|
||||
<table class="table table-striped" id="blockedIPsTable">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>IP Address</th>
|
||||
<th>Blocked At</th>
|
||||
<th>Reason</th>
|
||||
<th>Actions</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody id="blockedIPsTableBody">
|
||||
<tr>
|
||||
<td colspan="4" class="text-center">
|
||||
<i class="fas fa-spinner fa-spin"></i> Loading...
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Manual IP Blocking -->
|
||||
<div class="row mt-4">
|
||||
<div class="col-12">
|
||||
<div class="card">
|
||||
<div class="card-header">
|
||||
<h4>Manual IP Blocking</h4>
|
||||
</div>
|
||||
<div class="card-body">
|
||||
<form id="blockIPForm">
|
||||
<div class="row">
|
||||
<div class="col-md-6">
|
||||
<div class="form-group">
|
||||
<label for="ipAddress">IP Address</label>
|
||||
<input type="text" class="form-control" id="ipAddress" name="ip_address" placeholder="192.168.1.100" required>
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-md-6">
|
||||
<div class="form-group">
|
||||
<label for="blockReason">Reason</label>
|
||||
<input type="text" class="form-control" id="blockReason" name="reason" placeholder="Suspicious activity" value="Manual block via CyberPanel">
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<button type="submit" class="btn btn-danger">
|
||||
<i class="fas fa-ban"></i> Block IP Address
|
||||
</button>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
// Sample security alerts data (in a real implementation, this would come from the backend)
|
||||
const sampleAlerts = [
|
||||
{
|
||||
id: 1,
|
||||
type: 'brute_force',
|
||||
ip: '129.212.176.254',
|
||||
attempts: 85,
|
||||
severity: 'HIGH',
|
||||
timestamp: '2024-01-15 14:30:25',
|
||||
description: 'IP address 129.212.176.254 has made 85 failed password attempts. This indicates a potential brute force attack.'
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
type: 'brute_force',
|
||||
ip: '177.10.47.186',
|
||||
attempts: 10,
|
||||
severity: 'HIGH',
|
||||
timestamp: '2024-01-15 14:25:10',
|
||||
description: 'IP address 177.10.47.186 has made 10 failed password attempts. This indicates a potential brute force attack.'
|
||||
}
|
||||
];
|
||||
|
||||
function refreshSecurityAlerts() {
|
||||
const container = document.getElementById('securityAlertsContainer');
|
||||
|
||||
if (sampleAlerts.length === 0) {
|
||||
container.innerHTML = '<div class="alert alert-info">No security alerts found.</div>';
|
||||
return;
|
||||
}
|
||||
|
||||
let html = '';
|
||||
sampleAlerts.forEach(alert => {
|
||||
const severityClass = alert.severity === 'HIGH' ? 'danger' : alert.severity === 'MEDIUM' ? 'warning' : 'info';
|
||||
|
||||
html += `
|
||||
<div class="alert alert-${severityClass} mb-3">
|
||||
<div class="d-flex justify-content-between align-items-start">
|
||||
<div>
|
||||
<h6 class="alert-heading">
|
||||
<i class="fas fa-exclamation-triangle"></i>
|
||||
${alert.type.replace('_', ' ').toUpperCase()} Attack Detected
|
||||
</h6>
|
||||
<p class="mb-2">${alert.description}</p>
|
||||
<div class="row">
|
||||
<div class="col-md-3">
|
||||
<strong>IP Address:</strong> ${alert.ip}
|
||||
</div>
|
||||
<div class="col-md-3">
|
||||
<strong>Failed Attempts:</strong> ${alert.attempts}
|
||||
</div>
|
||||
<div class="col-md-3">
|
||||
<strong>Attack Type:</strong> Brute Force
|
||||
</div>
|
||||
<div class="col-md-3">
|
||||
<strong>Time:</strong> ${alert.timestamp}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="ml-3">
|
||||
<span class="badge badge-${severityClass}">${alert.severity}</span>
|
||||
<div class="mt-2">
|
||||
<button class="btn btn-sm btn-danger" onclick="blockIP('${alert.ip}', 'Brute force attack - ${alert.attempts} attempts')">
|
||||
<i class="fas fa-ban"></i> Block IP
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
`;
|
||||
});
|
||||
|
||||
container.innerHTML = html;
|
||||
}
|
||||
|
||||
function blockIP(ipAddress, reason) {
|
||||
if (!confirm(`Are you sure you want to block IP address ${ipAddress}?`)) {
|
||||
return;
|
||||
}
|
||||
|
||||
const formData = {
|
||||
'csrfmiddlewaretoken': '{{ csrf_token }}',
|
||||
'ip_address': ipAddress,
|
||||
'reason': reason
|
||||
};
|
||||
|
||||
$.post('{% url "blockIPAddress" %}', formData, function(data) {
|
||||
if (data.status === 1) {
|
||||
showNotification('success', data.message);
|
||||
refreshBlockedIPs();
|
||||
refreshSecurityAlerts(); // Refresh to remove the alert or update its status
|
||||
} else {
|
||||
showNotification('error', data.message);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
function unblockIP(ipAddress) {
|
||||
if (!confirm(`Are you sure you want to unblock IP address ${ipAddress}?`)) {
|
||||
return;
|
||||
}
|
||||
|
||||
const formData = {
|
||||
'csrfmiddlewaretoken': '{{ csrf_token }}',
|
||||
'ip_address': ipAddress
|
||||
};
|
||||
|
||||
$.post('{% url "unblockIPAddress" %}', formData, function(data) {
|
||||
if (data.status === 1) {
|
||||
showNotification('success', data.message);
|
||||
refreshBlockedIPs();
|
||||
} else {
|
||||
showNotification('error', data.message);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
function refreshBlockedIPs() {
|
||||
$.post('{% url "getBlockedIPs" %}', {
|
||||
'csrfmiddlewaretoken': '{{ csrf_token }}'
|
||||
}, function(data) {
|
||||
if (data.status === 1) {
|
||||
displayBlockedIPs(data.blocked_ips);
|
||||
} else {
|
||||
showNotification('error', data.message);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
function displayBlockedIPs(blockedIPs) {
|
||||
const tbody = document.getElementById('blockedIPsTableBody');
|
||||
|
||||
if (blockedIPs.length === 0) {
|
||||
tbody.innerHTML = '<tr><td colspan="4" class="text-center">No blocked IP addresses found</td></tr>';
|
||||
return;
|
||||
}
|
||||
|
||||
let html = '';
|
||||
blockedIPs.forEach(ip => {
|
||||
html += `
|
||||
<tr>
|
||||
<td>${ip}</td>
|
||||
<td>N/A</td>
|
||||
<td>Blocked via CyberPanel</td>
|
||||
<td>
|
||||
<button class="btn btn-sm btn-warning" onclick="unblockIP('${ip}')">
|
||||
<i class="fas fa-unlock"></i> Unblock
|
||||
</button>
|
||||
</td>
|
||||
</tr>
|
||||
`;
|
||||
});
|
||||
|
||||
tbody.innerHTML = html;
|
||||
}
|
||||
|
||||
function showNotification(type, message) {
|
||||
const alertClass = type === 'success' ? 'alert-success' : 'alert-danger';
|
||||
const icon = type === 'success' ? 'fa-check-circle' : 'fa-exclamation-circle';
|
||||
|
||||
const notification = `
|
||||
<div class="alert ${alertClass} alert-dismissible fade show" role="alert">
|
||||
<i class="fas ${icon}"></i> ${message}
|
||||
<button type="button" class="close" data-dismiss="alert">
|
||||
<span>×</span>
|
||||
</button>
|
||||
</div>
|
||||
`;
|
||||
|
||||
$('.card-body').prepend(notification);
|
||||
|
||||
setTimeout(() => {
|
||||
$('.alert').fadeOut();
|
||||
}, 5000);
|
||||
}
|
||||
|
||||
// Handle manual IP blocking form
|
||||
$(document).ready(function() {
|
||||
$('#blockIPForm').on('submit', function(e) {
|
||||
e.preventDefault();
|
||||
|
||||
const ipAddress = $('#ipAddress').val();
|
||||
const reason = $('#blockReason').val();
|
||||
|
||||
if (!ipAddress) {
|
||||
showNotification('error', 'Please enter an IP address');
|
||||
return;
|
||||
}
|
||||
|
||||
blockIP(ipAddress, reason);
|
||||
$('#blockIPForm')[0].reset();
|
||||
});
|
||||
|
||||
// Load initial data
|
||||
refreshSecurityAlerts();
|
||||
refreshBlockedIPs();
|
||||
});
|
||||
</script>
|
||||
{% endblock %}
|
||||
@@ -215,13 +215,6 @@ urlpatterns = [
|
||||
path('getBandwidthResetLogs', views.getBandwidthResetLogs, name='getBandwidthResetLogs'),
|
||||
path('scheduleBandwidthReset', views.scheduleBandwidthReset, name='scheduleBandwidthReset'),
|
||||
|
||||
# Security Management
|
||||
path('securityManagement', views.securityManagementPage, name='securityManagementPage'),
|
||||
|
||||
# IP Blocking
|
||||
path('blockIPAddress', views.blockIPAddress, name='blockIPAddress'),
|
||||
path('unblockIPAddress', views.unblockIPAddress, name='unblockIPAddress'),
|
||||
path('getBlockedIPs', views.getBlockedIPs, name='getBlockedIPs'),
|
||||
path('checkIPStatus', views.checkIPStatus, name='checkIPStatus'),
|
||||
|
||||
# Catch all for domains (must be last)
|
||||
|
||||
@@ -2242,15 +2242,6 @@ def bandwidthManagementPage(request):
|
||||
except KeyError:
|
||||
return redirect(loadLoginPage)
|
||||
|
||||
def securityManagementPage(request):
|
||||
"""Render the Security Management page."""
|
||||
try:
|
||||
userID = request.session['userID']
|
||||
proc = httpProc(request, 'websiteFunctions/securityManagement.html', {}, 'admin')
|
||||
return proc.render()
|
||||
except KeyError:
|
||||
return redirect(loadLoginPage)
|
||||
|
||||
def getFTPQuotaStatus(request):
|
||||
try:
|
||||
userID = request.session['userID']
|
||||
@@ -2308,31 +2299,6 @@ def scheduleBandwidthReset(request):
|
||||
except KeyError:
|
||||
return redirect(loadLoginPage)
|
||||
|
||||
# IP Blocking Views
|
||||
def blockIPAddress(request):
|
||||
try:
|
||||
userID = request.session['userID']
|
||||
wm = WebsiteManager()
|
||||
return wm.blockIPAddress(userID, request.POST)
|
||||
except KeyError:
|
||||
return redirect(loadLoginPage)
|
||||
|
||||
def unblockIPAddress(request):
|
||||
try:
|
||||
userID = request.session['userID']
|
||||
wm = WebsiteManager()
|
||||
return wm.unblockIPAddress(userID, request.POST)
|
||||
except KeyError:
|
||||
return redirect(loadLoginPage)
|
||||
|
||||
def getBlockedIPs(request):
|
||||
try:
|
||||
userID = request.session['userID']
|
||||
wm = WebsiteManager()
|
||||
return wm.getBlockedIPs(userID, request.POST)
|
||||
except KeyError:
|
||||
return redirect(loadLoginPage)
|
||||
|
||||
def checkIPStatus(request):
|
||||
try:
|
||||
userID = request.session['userID']
|
||||
|
||||
@@ -9322,141 +9322,6 @@ StrictHostKeyChecking no
|
||||
json_data = json.dumps(data_ret)
|
||||
return HttpResponse(json_data)
|
||||
|
||||
def blockIPAddress(self, userID=None, data=None):
|
||||
"""
|
||||
Block an IP address
|
||||
"""
|
||||
try:
|
||||
currentACL = ACLManager.loadedACL(userID)
|
||||
admin = Administrator.objects.get(pk=userID)
|
||||
|
||||
# Check if user has permission
|
||||
if not (currentACL.get('admin', 0) == 1):
|
||||
return ACLManager.loadErrorJson('status', 0)
|
||||
|
||||
ip_address = data.get('ip_address')
|
||||
reason = data.get('reason', 'Manual block via CyberPanel')
|
||||
|
||||
if not ip_address:
|
||||
data_ret = {
|
||||
'status': 0,
|
||||
'message': 'IP address is required'
|
||||
}
|
||||
json_data = json.dumps(data_ret)
|
||||
return HttpResponse(json_data)
|
||||
|
||||
# Import firewall utilities
|
||||
from plogical.firewallUtilities import FirewallUtilities
|
||||
|
||||
# Block the IP
|
||||
success, message = FirewallUtilities.blockIP(ip_address, reason)
|
||||
|
||||
if success:
|
||||
data_ret = {
|
||||
'status': 1,
|
||||
'message': message
|
||||
}
|
||||
else:
|
||||
data_ret = {
|
||||
'status': 0,
|
||||
'message': message
|
||||
}
|
||||
|
||||
json_data = json.dumps(data_ret)
|
||||
return HttpResponse(json_data)
|
||||
|
||||
except Exception as e:
|
||||
data_ret = {
|
||||
'status': 0,
|
||||
'message': f'Error blocking IP: {str(e)}'
|
||||
}
|
||||
json_data = json.dumps(data_ret)
|
||||
return HttpResponse(json_data)
|
||||
|
||||
def unblockIPAddress(self, userID=None, data=None):
|
||||
"""
|
||||
Unblock an IP address
|
||||
"""
|
||||
try:
|
||||
currentACL = ACLManager.loadedACL(userID)
|
||||
admin = Administrator.objects.get(pk=userID)
|
||||
|
||||
# Check if user has permission
|
||||
if not (currentACL.get('admin', 0) == 1):
|
||||
return ACLManager.loadErrorJson('status', 0)
|
||||
|
||||
ip_address = data.get('ip_address')
|
||||
|
||||
if not ip_address:
|
||||
data_ret = {
|
||||
'status': 0,
|
||||
'message': 'IP address is required'
|
||||
}
|
||||
json_data = json.dumps(data_ret)
|
||||
return HttpResponse(json_data)
|
||||
|
||||
# Import firewall utilities
|
||||
from plogical.firewallUtilities import FirewallUtilities
|
||||
|
||||
# Unblock the IP
|
||||
success, message = FirewallUtilities.unblockIP(ip_address)
|
||||
|
||||
if success:
|
||||
data_ret = {
|
||||
'status': 1,
|
||||
'message': message
|
||||
}
|
||||
else:
|
||||
data_ret = {
|
||||
'status': 0,
|
||||
'message': message
|
||||
}
|
||||
|
||||
json_data = json.dumps(data_ret)
|
||||
return HttpResponse(json_data)
|
||||
|
||||
except Exception as e:
|
||||
data_ret = {
|
||||
'status': 0,
|
||||
'message': f'Error unblocking IP: {str(e)}'
|
||||
}
|
||||
json_data = json.dumps(data_ret)
|
||||
return HttpResponse(json_data)
|
||||
|
||||
def getBlockedIPs(self, userID=None, data=None):
|
||||
"""
|
||||
Get list of blocked IP addresses
|
||||
"""
|
||||
try:
|
||||
currentACL = ACLManager.loadedACL(userID)
|
||||
admin = Administrator.objects.get(pk=userID)
|
||||
|
||||
# Check if user has permission
|
||||
if not (currentACL.get('admin', 0) == 1):
|
||||
return ACLManager.loadErrorJson('status', 0)
|
||||
|
||||
# Import firewall utilities
|
||||
from plogical.firewallUtilities import FirewallUtilities
|
||||
|
||||
# Get blocked IPs
|
||||
blocked_ips = FirewallUtilities.getBlockedIPs()
|
||||
|
||||
data_ret = {
|
||||
'status': 1,
|
||||
'blocked_ips': blocked_ips
|
||||
}
|
||||
|
||||
json_data = json.dumps(data_ret)
|
||||
return HttpResponse(json_data)
|
||||
|
||||
except Exception as e:
|
||||
data_ret = {
|
||||
'status': 0,
|
||||
'message': f'Error getting blocked IPs: {str(e)}'
|
||||
}
|
||||
json_data = json.dumps(data_ret)
|
||||
return HttpResponse(json_data)
|
||||
|
||||
def checkIPStatus(self, userID=None, data=None):
|
||||
"""
|
||||
Check if an IP is blocked
|
||||
|
||||
Reference in New Issue
Block a user