From 5251494f7a87edfc6d70813230bf0c97aa1fd243 Mon Sep 17 00:00:00 2001 From: Myron Blair Date: Mon, 8 Jun 2026 11:35:12 +0000 Subject: [PATCH] =?UTF-8?q?feat:=20full=20PHP=20Manager=20=E2=80=94=20vers?= =?UTF-8?q?ion=20install/remove,=20per-version=20extension=20management?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit php.php: install-version, remove-version, version-extensions, install-extension, remove-extension, fpm-action endpoints. versions now returns fpm_active status and panel_php (current runtime version). admin.js phpManager(): grid of installed/not-installed versions with Install/ Remove/Restart FPM buttons; Extensions panel slides in below with filterable list, dropdown of common extensions + custom input, per-extension Remove buttons. Panel PHP info card shows which version NovaCPX runs on. Co-Authored-By: Claude Sonnet 4.6 --- panel/api/endpoints/php.php | 92 ++++++++++++++++++- panel/public/assets/js/admin.js | 152 +++++++++++++++++++++++++++++--- 2 files changed, 229 insertions(+), 15 deletions(-) diff --git a/panel/api/endpoints/php.php b/panel/api/endpoints/php.php index 599b906..21af62e 100644 --- a/panel/api/endpoints/php.php +++ b/panel/api/endpoints/php.php @@ -26,10 +26,96 @@ match ($action) { 'versions' => (function() { $versions = []; foreach (['7.4','8.1','8.2','8.3'] as $v) { - $installed = file_exists("/usr/bin/php{$v}"); - $versions[] = ['version' => $v, 'installed' => $installed, 'is_default' => $v === PHP_DEFAULT]; + $installed = file_exists("/usr/bin/php{$v}"); + $fpmActive = $installed ? trim(shell_exec("systemctl is-active php{$v}-fpm 2>/dev/null") ?: '') : ''; + $versions[] = [ + 'version' => $v, + 'installed' => $installed, + 'fpm_active' => $fpmActive === 'active', + 'is_default' => $v === PHP_DEFAULT, + ]; } - Response::success($versions); + // Detect which version the panel itself runs on (highest installed) + $panelVer = PHP_MAJOR_VERSION . '.' . PHP_MINOR_VERSION; + Response::success(['versions' => $versions, 'panel_php' => $panelVer]); + })(), + + 'install-version' => (function() use ($body) { + Auth::getInstance()->require('admin'); + $ver = $body['version'] ?? ''; + if (!preg_match('/^[78]\.\d$/', $ver)) Response::error("Invalid version"); + $pkgs = "php{$ver} php{$ver}-fpm php{$ver}-cli php{$ver}-common php{$ver}-mysql php{$ver}-curl php{$ver}-gd php{$ver}-xml php{$ver}-mbstring php{$ver}-zip php{$ver}-intl php{$ver}-bcmath php{$ver}-soap php{$ver}-opcache"; + $out = shell_exec("apt-get install -y $pkgs 2>&1"); + shell_exec("systemctl enable php{$ver}-fpm && systemctl start php{$ver}-fpm 2>/dev/null"); + audit('php.install-version', $ver); + Response::success(['output' => substr($out ?: '', -1000)], "PHP $ver installed"); + })(), + + 'remove-version' => (function() use ($body) { + Auth::getInstance()->require('admin'); + $ver = $body['version'] ?? ''; + if (!preg_match('/^[78]\.\d$/', $ver)) Response::error("Invalid version"); + if ($ver === PHP_DEFAULT) Response::error("Cannot remove the panel's default PHP version"); + shell_exec("systemctl stop php{$ver}-fpm 2>/dev/null || true"); + $out = shell_exec("apt-get remove -y php{$ver}* 2>&1"); + audit('php.remove-version', $ver); + Response::success(['output' => substr($out ?: '', -1000)], "PHP $ver removed"); + })(), + + 'version-extensions' => (function() use ($body) { + Auth::getInstance()->require('admin'); + $ver = $body['version'] ?? $_GET['version'] ?? ''; + if (!preg_match('/^[78]\.\d$/', $ver)) Response::error("Invalid version"); + if (!file_exists("/usr/bin/php{$ver}")) Response::error("PHP $ver not installed"); + $out = shell_exec("php{$ver} -m 2>/dev/null") ?: ''; + $exts = array_values(array_filter(explode("\n", trim($out)), fn($l) => $l && !str_starts_with($l, '['))); + + // Available apt packages for this version (common ones) + $common = ['bcmath','bz2','curl','gd','gmp','igbinary','imagick','imap','intl','ldap','mbstring', + 'memcached','mongodb','msgpack','mysql','odbc','opcache','pdo','pdo-mysql','pdo-pgsql', + 'pgsql','redis','soap','sqlite3','tidy','tokenizer','uuid','xml','xmlrpc','xsl','zip']; + $available = array_map(fn($e) => "php{$ver}-{$e}", $common); + Response::success(['version' => $ver, 'installed' => $exts, 'available' => $available]); + })(), + + 'install-extension' => (function() use ($body) { + Auth::getInstance()->require('admin'); + $ver = $body['version'] ?? ''; + $ext = preg_replace('/[^a-z0-9\-]/', '', strtolower($body['extension'] ?? '')); + if (!preg_match('/^[78]\.\d$/', $ver) || !$ext) Response::error("Invalid input"); + $pkg = "php{$ver}-{$ext}"; + $out = shell_exec("apt-get install -y $pkg 2>&1"); + if (str_contains($out ?: '', 'Unable to locate') || str_contains($out ?: '', 'E:')) { + // Try pecl fallback + $out2 = shell_exec("php{$ver} /usr/bin/pecl install {$ext} 2>&1"); + $out .= "\n[pecl] " . $out2; + } + shell_exec("systemctl reload php{$ver}-fpm 2>/dev/null || true"); + audit('php.install-extension', "$ver/$ext"); + Response::success(['output' => substr($out ?: '', -1000)], "Extension $ext installed for PHP $ver"); + })(), + + 'remove-extension' => (function() use ($body) { + Auth::getInstance()->require('admin'); + $ver = $body['version'] ?? ''; + $ext = preg_replace('/[^a-z0-9\-]/', '', strtolower($body['extension'] ?? '')); + if (!preg_match('/^[78]\.\d$/', $ver) || !$ext) Response::error("Invalid input"); + $pkg = "php{$ver}-{$ext}"; + $out = shell_exec("apt-get remove -y $pkg 2>&1"); + shell_exec("systemctl reload php{$ver}-fpm 2>/dev/null || true"); + audit('php.remove-extension', "$ver/$ext"); + Response::success(['output' => substr($out ?: '', -1000)], "Extension $ext removed from PHP $ver"); + })(), + + 'fpm-action' => (function() use ($body) { + Auth::getInstance()->require('admin'); + $ver = $body['version'] ?? ''; + $cmd = $body['command'] ?? 'restart'; + if (!preg_match('/^[78]\.\d$/', $ver)) Response::error("Invalid version"); + if (!in_array($cmd, ['start','stop','restart','reload'])) Response::error("Invalid command"); + shell_exec("systemctl $cmd php{$ver}-fpm 2>/dev/null"); + audit('php.fpm-action', "$cmd php{$ver}-fpm"); + Response::success(null, "php{$ver}-fpm {$cmd}ed"); })(), 'switch-version' => (function() use ($body, $accountId) { diff --git a/panel/public/assets/js/admin.js b/panel/public/assets/js/admin.js index ecabbd4..4f3a842 100644 --- a/panel/public/assets/js/admin.js +++ b/panel/public/assets/js/admin.js @@ -457,28 +457,156 @@ // ── PHP Manager ──────────────────────────────────────────────────────────── async function phpManager() { + const res = await Nova.api('php', 'versions'); + const data = res?.data || {}; + const vers = data.versions || []; + const panelPhp = data.panel_php || '—'; + return ` -
-
PHP Version Manager
+ + +
+
Panel PHP
+
+

NovaCPX itself runs on PHP ${panelPhp} (always the highest installed version, updated automatically when a new version is installed).

+
+
+ +
+
Installed Versions
-

Manage installed PHP versions and global extensions.

- ${['7.4','8.1','8.2','8.3'].map(v => ` + ${vers.map(v => `
-
PHP ${v}
-
${Nova.badge('Active','green')}
-
- +
+ PHP ${v.version} + ${v.installed ? Nova.badge(v.fpm_active ? 'active' : 'stopped', v.fpm_active ? 'green' : 'yellow') : Nova.badge('not installed','muted')} +
+ ${v.is_default ? `
Panel default
` : ''} +
+ ${v.installed ? ` + + + ${!v.is_default ? `` : ''} + ` : ` + + `}
`).join('')}
-
-

Global PHP Extensions

-

Extensions installed across all PHP versions: mbstring, curl, gd, xml, zip, opcache, redis, imagick, pdo, pdo_mysql, pdo_pgsql

+
+
+ +`; + } + + window.phpInstallVersion = (ver) => { + Nova.confirm(`Install PHP ${ver}? This will run apt-get and may take a minute.`, async () => { + Nova.toast(`Installing PHP ${ver}…`, 'info', 15000); + const r = await Nova.api('php', 'install-version', { method: 'POST', body: { version: ver } }); + if (r?.success) { Nova.toast(`PHP ${ver} installed`, 'success'); adminPage('php-manager'); } + else Nova.toast(r?.message || 'Install failed', 'error'); + }); + }; + + window.phpRemoveVersion = (ver) => { + Nova.confirm(`Remove PHP ${ver}? All FPM pools for this version will stop.`, async () => { + const r = await Nova.api('php', 'remove-version', { method: 'POST', body: { version: ver } }); + if (r?.success) { Nova.toast(`PHP ${ver} removed`, 'success'); adminPage('php-manager'); } + else Nova.toast(r?.message || 'Remove failed', 'error'); + }, true); + }; + + window.phpFpmAction = async (ver, cmd) => { + const r = await Nova.api('php', 'fpm-action', { method: 'POST', body: { version: ver, command: cmd } }); + if (r?.success) { Nova.toast(r.message, 'success'); adminPage('php-manager'); } + else Nova.toast(r?.message || 'Action failed', 'error'); + }; + + window.phpExtModal = async (ver) => { + const panel = document.getElementById('php-ext-panel'); + if (!panel) return; + panel.style.display = ''; + panel.innerHTML = `

Loading extensions for PHP ${ver}…

`; + panel.scrollIntoView({ behavior: 'smooth' }); + + const r = await Nova.api('php', 'version-extensions', { params: { version: ver } }); + if (!r?.success) { panel.innerHTML = `

${r?.message || 'Failed to load'}

`; return; } + + const installed = r.data.installed || []; + const available = r.data.available || []; + const notInstalled = available.filter(pkg => { + const ext = pkg.replace(/^php[\d.]+-/, ''); + return !installed.some(i => i.toLowerCase() === ext.toLowerCase() || i.toLowerCase().replace('_','-') === ext.toLowerCase()); + }); + + panel.innerHTML = ` +
+
+ PHP ${ver} Extensions +
+ + +
+
+
+
+ Add extension +
+ + or + + +
+
+
+ + + + ${installed.map(e => ` + + + + `).join('')} + +
ExtensionAction
${e} + +
`; - } + }; + + window.phpExtFilter = (q) => { + document.querySelectorAll('.php-ext-row').forEach(row => { + row.style.display = row.dataset.ext.includes(q.toLowerCase()) ? '' : 'none'; + }); + }; + + window.phpExtInstall = async (ver) => { + const sel = document.getElementById('php-ext-add-sel')?.value; + const custom = document.getElementById('php-ext-add-custom')?.value?.trim(); + const ext = custom || sel; + if (!ext) { Nova.toast('Choose or type an extension name', 'error'); return; } + Nova.toast(`Installing ${ext} for PHP ${ver}…`, 'info', 15000); + const r = await Nova.api('php', 'install-extension', { method: 'POST', body: { version: ver, extension: ext } }); + if (r?.success) { Nova.toast(r.message, 'success'); phpExtModal(ver); } + else Nova.toast(r?.message || 'Install failed', 'error'); + }; + + window.phpExtRemove = (ver, ext) => { + Nova.confirm(`Remove extension ${ext} from PHP ${ver}?`, async () => { + const r = await Nova.api('php', 'remove-extension', { method: 'POST', body: { version: ver, extension: ext } }); + if (r?.success) { Nova.toast(r.message, 'success'); phpExtModal(ver); } + else Nova.toast(r?.message || 'Remove failed', 'error'); + }, true); + }; // ── Notifications (#25) ─────────────────────────────────────────────────── async function notifications() {