Files
domnitor/app/Helpers/LayoutHelper.php
Hosteroid 30a139d765 Handle hotfix updates and stale commit cache
Treat file-only/hotfix updates (identified by commit SHA) as non-version changes and clear stale commit-cache so the UI no longer reports an available update after a hotfix. UpdateService now clears commits_behind_count and latest_remote_sha when no new commits are found. LayoutHelper and settings view consider installed_commit_sha vs latest_remote_sha and set commitsBehind to 0 when they match. NotificationService detects commit SHAs for the target version and emits a clearer "hotfix {sha}" message for file-only updates.
2026-02-11 19:24:39 +02:00

238 lines
7.9 KiB
PHP

<?php
namespace App\Helpers;
use App\Models\Notification;
use App\Models\Setting;
class LayoutHelper
{
/**
* Get notifications for the top nav dropdown
*/
public static function getNotifications(int $userId): array
{
try {
$notificationModel = new Notification();
$notifications = $notificationModel->getRecentUnread($userId, 4);
$unreadCount = $notificationModel->getUnreadCount($userId);
// Format each notification
foreach ($notifications as &$notif) {
$notif['time_ago'] = self::timeAgo($notif['created_at']);
$notif['icon'] = self::getNotificationIcon($notif['type']);
$notif['color'] = self::getNotificationColor($notif['type']);
$notif['login_data'] = self::parseLoginData($notif);
}
return [
'items' => $notifications,
'unread_count' => $unreadCount
];
} catch (\Exception $e) {
// If table doesn't exist yet
return ['items' => [], 'unread_count' => 0];
}
}
/**
* Get domain statistics (centralized function for views)
*/
public static function getDomainStats(): array
{
$domainModel = new \App\Models\Domain();
$userId = \Core\Auth::id();
$settingModel = new \App\Models\Setting();
$isolationMode = $settingModel->getValue('user_isolation_mode', 'shared');
if ($isolationMode === 'isolated') {
return $domainModel->getStatistics($userId);
} else {
return $domainModel->getStatistics();
}
}
/**
* Parse session_new notification message (JSON)
* Returns structured data for rich display, or null if not parseable
*/
public static function parseLoginData(array $notification): ?array
{
if ($notification['type'] !== 'session_new' && $notification['type'] !== 'session_failed') {
return null;
}
$data = json_decode($notification['message'] ?? '', true);
if (is_array($data) && isset($data['ip'])) {
return $data;
}
return null;
}
/**
* Format session_new notification for dropdown display (compact)
*/
public static function formatLoginDropdown(array $loginData): string
{
$parts = [];
if ($loginData['city'] !== 'Unknown' && $loginData['city'] !== 'Local') {
$parts[] = $loginData['city'];
}
if ($loginData['country'] !== 'Unknown' && $loginData['country'] !== 'Local') {
$parts[] = $loginData['country'];
}
$location = !empty($parts) ? implode(', ', $parts) : $loginData['ip'];
$browser = $loginData['browser'] ?? 'Unknown';
return "{$location} · {$browser}";
}
/**
* Convert timestamp to "time ago" format
*/
private static function timeAgo(string $datetime): string
{
$timestamp = strtotime($datetime);
$diff = time() - $timestamp;
if ($diff < 60) return 'just now';
if ($diff < 3600) {
$mins = floor($diff / 60);
return $mins . ' min' . ($mins > 1 ? 's' : '') . ' ago';
}
if ($diff < 86400) {
$hours = floor($diff / 3600);
return $hours . ' hour' . ($hours > 1 ? 's' : '') . ' ago';
}
$days = floor($diff / 86400);
return $days . ' day' . ($days > 1 ? 's' : '') . ' ago';
}
/**
* Get notification icon based on type
*/
private static function getNotificationIcon(string $type): string
{
return match($type) {
'domain_expiring' => 'exclamation-triangle',
'domain_expired', 'domain_expired_status' => 'times-circle',
'domain_available' => 'check-circle',
'domain_registered' => 'globe',
'domain_redemption' => 'hourglass-half',
'domain_pending_delete' => 'trash-alt',
'domain_updated' => 'sync-alt',
'session_new' => 'sign-in-alt',
'session_failed' => 'shield-alt',
'whois_failed' => 'exclamation-circle',
'system_welcome' => 'hand-sparkles',
'system_upgrade' => 'arrow-up',
'update_available' => 'cloud-download-alt',
default => 'bell'
};
}
/**
* Get notification color based on type
*/
private static function getNotificationColor(string $type): string
{
return match($type) {
'domain_expiring' => 'orange',
'domain_expired', 'domain_expired_status' => 'red',
'domain_available' => 'blue',
'domain_registered' => 'green',
'domain_redemption' => 'amber',
'domain_pending_delete' => 'rose',
'domain_updated' => 'green',
'session_new' => 'blue',
'session_failed' => 'red',
'whois_failed' => 'gray',
'system_welcome' => 'purple',
'system_upgrade' => 'indigo',
'update_available' => 'blue',
default => 'gray'
};
}
/**
* Get application settings
*/
public static function getAppSettings(): array
{
try {
$settingModel = new Setting();
$appSettings = $settingModel->getAppSettings();
return [
'app_name' => htmlspecialchars($appSettings['app_name']),
'app_timezone' => $appSettings['app_timezone'],
'app_version' => $appSettings['app_version']
];
} catch (\Exception $e) {
// Fallback defaults
$settingModel = new Setting();
return [
'app_name' => 'Domain Monitor',
'app_timezone' => 'UTC',
'app_version' => $settingModel->getAppVersion()
];
}
}
/**
* Get update badge info for the top menu (admin only).
* Uses cached update check data; no GitHub API call.
* Returns ['show' => bool, 'available' => bool, 'label' => string].
*/
public static function getUpdateBadgeInfo(): array
{
try {
$settingModel = new Setting();
$updateSettings = $settingModel->getUpdateSettings();
$badgeEnabled = ($updateSettings['update_badge_enabled'] ?? '1') !== '0';
if (!$badgeEnabled) {
return ['show' => false, 'available' => false, 'label' => ''];
}
$current = $settingModel->getAppVersion();
$latestVersion = $updateSettings['latest_available_version'] ?? null;
$channel = $updateSettings['update_channel'] ?? 'stable';
$commitsBehind = (int) ($updateSettings['commits_behind_count'] ?? 0);
$installedSha = $updateSettings['installed_commit_sha'] ?? '';
$remoteSha = $updateSettings['latest_remote_sha'] ?? '';
// If installed SHA matches remote SHA, the cached commits_behind is stale
if ($installedSha !== '' && $remoteSha !== '' && str_starts_with($installedSha, $remoteSha)) {
$commitsBehind = 0;
}
$available = false;
$label = '';
if ($latestVersion && version_compare($latestVersion, $current, '>')) {
$available = true;
$label = 'v' . $latestVersion;
}
if ($channel === 'latest' && $commitsBehind > 0 && !$available) {
$available = true;
$label = $commitsBehind . ' commit' . ($commitsBehind !== 1 ? 's' : '');
}
return [
'show' => $available,
'available' => $available,
'label' => $label,
];
} catch (\Exception $e) {
return ['show' => false, 'available' => false, 'label' => ''];
}
}
}