Compare commits

..

1 Commits

Author SHA1 Message Date
renovate[bot]
186fb1b4c7 Update dependency @vitejs/plugin-react to v5 2025-11-18 18:17:16 +00:00
17 changed files with 382 additions and 710 deletions

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

View File

@@ -1,6 +1,6 @@
{ {
"name": "patchmon-backend", "name": "patchmon-backend",
"version": "1.3.6", "version": "1.3.5",
"description": "Backend API for Linux Patch Monitoring System", "description": "Backend API for Linux Patch Monitoring System",
"license": "AGPL-3.0", "license": "AGPL-3.0",
"main": "src/server.js", "main": "src/server.js",

View File

@@ -103,7 +103,6 @@ model hosts {
gateway_ip String? gateway_ip String?
hostname String? hostname String?
kernel_version String? kernel_version String?
installed_kernel_version String?
load_average Json? load_average Json?
network_interfaces Json? network_interfaces Json?
ram_installed Int? ram_installed Int?

View File

@@ -506,10 +506,6 @@ router.post(
.optional() .optional()
.isString() .isString()
.withMessage("Kernel version must be a string"), .withMessage("Kernel version must be a string"),
body("installedKernelVersion")
.optional()
.isString()
.withMessage("Installed kernel version must be a string"),
body("selinuxStatus") body("selinuxStatus")
.optional() .optional()
.isIn(["enabled", "disabled", "permissive"]) .isIn(["enabled", "disabled", "permissive"])
@@ -591,8 +587,6 @@ router.post(
// System Information // System Information
if (req.body.kernelVersion) if (req.body.kernelVersion)
updateData.kernel_version = req.body.kernelVersion; updateData.kernel_version = req.body.kernelVersion;
if (req.body.installedKernelVersion)
updateData.installed_kernel_version = req.body.installedKernelVersion;
if (req.body.selinuxStatus) if (req.body.selinuxStatus)
updateData.selinux_status = req.body.selinuxStatus; updateData.selinux_status = req.body.selinuxStatus;
if (req.body.systemUptime) if (req.body.systemUptime)

View File

@@ -6,5 +6,5 @@ VITE_API_URL=http://localhost:3001/api/v1
# Application Metadata # Application Metadata
VITE_APP_NAME=PatchMon VITE_APP_NAME=PatchMon
VITE_APP_VERSION=1.3.6 VITE_APP_VERSION=1.3.5

View File

@@ -1,7 +1,7 @@
{ {
"name": "patchmon-frontend", "name": "patchmon-frontend",
"private": true, "private": true,
"version": "1.3.6", "version": "1.3.5",
"license": "AGPL-3.0", "license": "AGPL-3.0",
"type": "module", "type": "module",
"scripts": { "scripts": {
@@ -32,7 +32,7 @@
"devDependencies": { "devDependencies": {
"@types/react": "^18.3.14", "@types/react": "^18.3.14",
"@types/react-dom": "^18.3.1", "@types/react-dom": "^18.3.1",
"@vitejs/plugin-react": "^4.3.4", "@vitejs/plugin-react": "^5.0.0",
"autoprefixer": "^10.4.20", "autoprefixer": "^10.4.20",
"postcss": "^8.5.6", "postcss": "^8.5.6",
"tailwindcss": "^3.4.17", "tailwindcss": "^3.4.17",

View File

@@ -215,7 +215,7 @@ const GlobalSearch = () => {
<input <input
ref={inputRef} ref={inputRef}
type="text" type="text"
className="block w-full rounded-lg border border-secondary-200 bg-white py-2.5 sm:py-2 pl-10 pr-10 text-sm text-secondary-900 placeholder-secondary-500 focus:border-primary-500 focus:outline-none focus:ring-2 focus:ring-primary-500 dark:border-secondary-600 dark:bg-secondary-700 dark:text-white dark:placeholder-secondary-400 min-h-[44px]" className="block w-full rounded-lg border border-secondary-200 bg-white py-2 pl-10 pr-10 text-sm text-secondary-900 placeholder-secondary-500 focus:border-primary-500 focus:outline-none focus:ring-2 focus:ring-primary-500 dark:border-secondary-600 dark:bg-secondary-700 dark:text-white dark:placeholder-secondary-400"
placeholder="Search hosts, packages, repos, users..." placeholder="Search hosts, packages, repos, users..."
value={query} value={query}
onChange={handleInputChange} onChange={handleInputChange}
@@ -228,8 +228,7 @@ const GlobalSearch = () => {
<button <button
type="button" type="button"
onClick={handleClear} onClick={handleClear}
className="absolute inset-y-0 right-0 flex items-center pr-3 text-secondary-400 hover:text-secondary-600 min-w-[44px] min-h-[44px] justify-center" className="absolute inset-y-0 right-0 flex items-center pr-3 text-secondary-400 hover:text-secondary-600"
aria-label="Clear search"
> >
<X className="h-4 w-4" /> <X className="h-4 w-4" />
</button> </button>
@@ -238,9 +237,9 @@ const GlobalSearch = () => {
{/* Dropdown Results */} {/* Dropdown Results */}
{isOpen && ( {isOpen && (
<div className="absolute z-50 mt-2 w-full sm:w-[calc(100vw-2rem)] sm:max-w-md rounded-lg border border-secondary-200 bg-white shadow-lg dark:border-secondary-600 dark:bg-secondary-800 left-0 sm:left-auto right-0 sm:right-auto"> <div className="absolute z-50 mt-2 w-full rounded-lg border border-secondary-200 bg-white shadow-lg dark:border-secondary-600 dark:bg-secondary-800">
{isLoading ? ( {isLoading ? (
<div className="px-4 py-2 text-center text-sm text-secondary-500 dark:text-white/70"> <div className="px-4 py-2 text-center text-sm text-secondary-500">
Searching... Searching...
</div> </div>
) : hasResults ? ( ) : hasResults ? (
@@ -248,7 +247,7 @@ const GlobalSearch = () => {
{/* Hosts */} {/* Hosts */}
{results.hosts?.length > 0 && ( {results.hosts?.length > 0 && (
<div> <div>
<div className="sticky top-0 z-10 bg-secondary-50 px-3 py-1.5 text-xs font-semibold uppercase tracking-wider text-secondary-500 dark:bg-secondary-700 dark:text-white/80"> <div className="sticky top-0 z-10 bg-secondary-50 px-3 py-1.5 text-xs font-semibold uppercase tracking-wider text-secondary-500 dark:bg-secondary-700 dark:text-secondary-400">
Hosts Hosts
</div> </div>
{results.hosts.map((host, _idx) => { {results.hosts.map((host, _idx) => {
@@ -261,7 +260,7 @@ const GlobalSearch = () => {
type="button" type="button"
key={host.id} key={host.id}
onClick={() => handleResultClick(host)} onClick={() => handleResultClick(host)}
className={`flex w-full items-center gap-2 px-3 py-3 sm:py-1.5 text-left transition-colors min-h-[44px] ${ className={`flex w-full items-center gap-2 px-3 py-1.5 text-left transition-colors ${
globalIdx === selectedIndex globalIdx === selectedIndex
? "bg-primary-50 dark:bg-primary-900/20" ? "bg-primary-50 dark:bg-primary-900/20"
: "hover:bg-secondary-50 dark:hover:bg-secondary-700" : "hover:bg-secondary-50 dark:hover:bg-secondary-700"
@@ -272,14 +271,12 @@ const GlobalSearch = () => {
<span className="text-sm font-medium text-secondary-900 dark:text-white truncate"> <span className="text-sm font-medium text-secondary-900 dark:text-white truncate">
{display.primary} {display.primary}
</span> </span>
<span className="text-xs text-secondary-400 dark:text-white/50"> <span className="text-xs text-secondary-400"></span>
<span className="text-xs text-secondary-500 dark:text-secondary-400 truncate">
</span>
<span className="text-xs text-secondary-500 dark:text-white/70 truncate">
{display.secondary} {display.secondary}
</span> </span>
</div> </div>
<div className="flex-shrink-0 text-xs text-secondary-400 dark:text-white/60"> <div className="flex-shrink-0 text-xs text-secondary-400">
{host.os_type} {host.os_type}
</div> </div>
</button> </button>
@@ -291,7 +288,7 @@ const GlobalSearch = () => {
{/* Packages */} {/* Packages */}
{results.packages?.length > 0 && ( {results.packages?.length > 0 && (
<div> <div>
<div className="sticky top-0 z-10 bg-secondary-50 px-3 py-1.5 text-xs font-semibold uppercase tracking-wider text-secondary-500 dark:bg-secondary-700 dark:text-white/80"> <div className="sticky top-0 z-10 bg-secondary-50 px-3 py-1.5 text-xs font-semibold uppercase tracking-wider text-secondary-500 dark:bg-secondary-700 dark:text-secondary-400">
Packages Packages
</div> </div>
{results.packages.map((pkg, _idx) => { {results.packages.map((pkg, _idx) => {
@@ -304,7 +301,7 @@ const GlobalSearch = () => {
type="button" type="button"
key={pkg.id} key={pkg.id}
onClick={() => handleResultClick(pkg)} onClick={() => handleResultClick(pkg)}
className={`flex w-full items-center gap-2 px-3 py-3 sm:py-1.5 text-left transition-colors min-h-[44px] ${ className={`flex w-full items-center gap-2 px-3 py-1.5 text-left transition-colors ${
globalIdx === selectedIndex globalIdx === selectedIndex
? "bg-primary-50 dark:bg-primary-900/20" ? "bg-primary-50 dark:bg-primary-900/20"
: "hover:bg-secondary-50 dark:hover:bg-secondary-700" : "hover:bg-secondary-50 dark:hover:bg-secondary-700"
@@ -320,13 +317,13 @@ const GlobalSearch = () => {
<span className="text-xs text-secondary-400"> <span className="text-xs text-secondary-400">
</span> </span>
<span className="text-xs text-secondary-500 dark:text-white/70 truncate"> <span className="text-xs text-secondary-500 dark:text-secondary-400 truncate">
{display.secondary} {display.secondary}
</span> </span>
</> </>
)} )}
</div> </div>
<div className="flex-shrink-0 text-xs text-secondary-400 dark:text-white/60"> <div className="flex-shrink-0 text-xs text-secondary-400">
{pkg.host_count} hosts {pkg.host_count} hosts
</div> </div>
</button> </button>
@@ -338,7 +335,7 @@ const GlobalSearch = () => {
{/* Repositories */} {/* Repositories */}
{results.repositories?.length > 0 && ( {results.repositories?.length > 0 && (
<div> <div>
<div className="sticky top-0 z-10 bg-secondary-50 px-3 py-1.5 text-xs font-semibold uppercase tracking-wider text-secondary-500 dark:bg-secondary-700 dark:text-white/80"> <div className="sticky top-0 z-10 bg-secondary-50 px-3 py-1.5 text-xs font-semibold uppercase tracking-wider text-secondary-500 dark:bg-secondary-700 dark:text-secondary-400">
Repositories Repositories
</div> </div>
{results.repositories.map((repo, _idx) => { {results.repositories.map((repo, _idx) => {
@@ -351,7 +348,7 @@ const GlobalSearch = () => {
type="button" type="button"
key={repo.id} key={repo.id}
onClick={() => handleResultClick(repo)} onClick={() => handleResultClick(repo)}
className={`flex w-full items-center gap-2 px-3 py-3 sm:py-1.5 text-left transition-colors min-h-[44px] ${ className={`flex w-full items-center gap-2 px-3 py-1.5 text-left transition-colors ${
globalIdx === selectedIndex globalIdx === selectedIndex
? "bg-primary-50 dark:bg-primary-900/20" ? "bg-primary-50 dark:bg-primary-900/20"
: "hover:bg-secondary-50 dark:hover:bg-secondary-700" : "hover:bg-secondary-50 dark:hover:bg-secondary-700"
@@ -362,14 +359,12 @@ const GlobalSearch = () => {
<span className="text-sm font-medium text-secondary-900 dark:text-white truncate"> <span className="text-sm font-medium text-secondary-900 dark:text-white truncate">
{display.primary} {display.primary}
</span> </span>
<span className="text-xs text-secondary-400 dark:text-white/50"> <span className="text-xs text-secondary-400"></span>
<span className="text-xs text-secondary-500 dark:text-secondary-400 truncate">
</span>
<span className="text-xs text-secondary-500 dark:text-white/70 truncate">
{display.secondary} {display.secondary}
</span> </span>
</div> </div>
<div className="flex-shrink-0 text-xs text-secondary-400 dark:text-white/60"> <div className="flex-shrink-0 text-xs text-secondary-400">
{repo.host_count} hosts {repo.host_count} hosts
</div> </div>
</button> </button>
@@ -381,7 +376,7 @@ const GlobalSearch = () => {
{/* Users */} {/* Users */}
{results.users?.length > 0 && ( {results.users?.length > 0 && (
<div> <div>
<div className="sticky top-0 z-10 bg-secondary-50 px-3 py-1.5 text-xs font-semibold uppercase tracking-wider text-secondary-500 dark:bg-secondary-700 dark:text-white/80"> <div className="sticky top-0 z-10 bg-secondary-50 px-3 py-1.5 text-xs font-semibold uppercase tracking-wider text-secondary-500 dark:bg-secondary-700 dark:text-secondary-400">
Users Users
</div> </div>
{results.users.map((user, _idx) => { {results.users.map((user, _idx) => {
@@ -394,7 +389,7 @@ const GlobalSearch = () => {
type="button" type="button"
key={user.id} key={user.id}
onClick={() => handleResultClick(user)} onClick={() => handleResultClick(user)}
className={`flex w-full items-center gap-2 px-3 py-3 sm:py-1.5 text-left transition-colors min-h-[44px] ${ className={`flex w-full items-center gap-2 px-3 py-1.5 text-left transition-colors ${
globalIdx === selectedIndex globalIdx === selectedIndex
? "bg-primary-50 dark:bg-primary-900/20" ? "bg-primary-50 dark:bg-primary-900/20"
: "hover:bg-secondary-50 dark:hover:bg-secondary-700" : "hover:bg-secondary-50 dark:hover:bg-secondary-700"
@@ -405,14 +400,12 @@ const GlobalSearch = () => {
<span className="text-sm font-medium text-secondary-900 dark:text-white truncate"> <span className="text-sm font-medium text-secondary-900 dark:text-white truncate">
{display.primary} {display.primary}
</span> </span>
<span className="text-xs text-secondary-400 dark:text-white/50"> <span className="text-xs text-secondary-400"></span>
<span className="text-xs text-secondary-500 dark:text-secondary-400 truncate">
</span>
<span className="text-xs text-secondary-500 dark:text-white/70 truncate">
{display.secondary} {display.secondary}
</span> </span>
</div> </div>
<div className="flex-shrink-0 text-xs text-secondary-400 dark:text-white/60"> <div className="flex-shrink-0 text-xs text-secondary-400">
{user.role} {user.role}
</div> </div>
</button> </button>
@@ -422,7 +415,7 @@ const GlobalSearch = () => {
)} )}
</div> </div>
) : query.trim() ? ( ) : query.trim() ? (
<div className="px-4 py-2 text-center text-sm text-secondary-500 dark:text-white/70"> <div className="px-4 py-2 text-center text-sm text-secondary-500">
No results found for "{query}" No results found for "{query}"
</div> </div>
) : null} ) : null}

File diff suppressed because one or more lines are too long

View File

@@ -22,6 +22,7 @@ import {
Server, Server,
Settings, Settings,
Shield, Shield,
TrendingUp,
Users, Users,
WifiOff, WifiOff,
} from "lucide-react"; } from "lucide-react";
@@ -58,7 +59,6 @@ const Dashboard = () => {
const [packageTrendsHost, setPackageTrendsHost] = useState("all"); // host filter const [packageTrendsHost, setPackageTrendsHost] = useState("all"); // host filter
const [systemStatsJobId, setSystemStatsJobId] = useState(null); // Track job ID for system statistics const [systemStatsJobId, setSystemStatsJobId] = useState(null); // Track job ID for system statistics
const [isTriggeringJob, setIsTriggeringJob] = useState(false); const [isTriggeringJob, setIsTriggeringJob] = useState(false);
const [isMobile, setIsMobile] = useState(window.innerWidth < 640);
const navigate = useNavigate(); const navigate = useNavigate();
const { isDark } = useTheme(); const { isDark } = useTheme();
const { user } = useAuth(); const { user } = useAuth();
@@ -312,15 +312,6 @@ const Dashboard = () => {
}; };
}, []); }, []);
// Track window size for responsive chart options
useEffect(() => {
const handleResize = () => {
setIsMobile(window.innerWidth < 640);
};
window.addEventListener("resize", handleResize);
return () => window.removeEventListener("resize", handleResize);
}, []);
// Helper function to check if a card should be displayed // Helper function to check if a card should be displayed
const isCardEnabled = (cardId) => { const isCardEnabled = (cardId) => {
const card = cardPreferences.find((c) => c.cardId === cardId); const card = cardPreferences.find((c) => c.cardId === cardId);
@@ -367,11 +358,11 @@ const Dashboard = () => {
const getGroupClassName = (cardType) => { const getGroupClassName = (cardType) => {
switch (cardType) { switch (cardType) {
case "stats": case "stats":
return "grid grid-cols-2 sm:grid-cols-3 lg:grid-cols-5 gap-3 sm:gap-4"; return "grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-5 gap-4";
case "charts": case "charts":
return "grid grid-cols-1 lg:grid-cols-3 gap-4 sm:gap-6"; return "grid grid-cols-1 lg:grid-cols-3 gap-6";
case "widecharts": case "widecharts":
return "grid grid-cols-1 lg:grid-cols-3 gap-4 sm:gap-6"; return "grid grid-cols-1 lg:grid-cols-3 gap-6";
case "fullwidth": case "fullwidth":
return "space-y-6"; return "space-y-6";
default: default:
@@ -386,7 +377,7 @@ const Dashboard = () => {
return ( return (
<button <button
type="button" type="button"
className="card p-3 sm:p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left min-h-[44px]" className="card p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left"
onClick={handleNeedsRebootClick} onClick={handleNeedsRebootClick}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
@@ -414,7 +405,7 @@ const Dashboard = () => {
return ( return (
<button <button
type="button" type="button"
className="card p-3 sm:p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left min-h-[44px]" className="card p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left"
onClick={handleTotalHostsClick} onClick={handleTotalHostsClick}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
@@ -443,7 +434,7 @@ const Dashboard = () => {
return ( return (
<button <button
type="button" type="button"
className="card p-3 sm:p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left min-h-[44px]" className="card p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left"
onClick={handleHostsNeedingUpdatesClick} onClick={handleHostsNeedingUpdatesClick}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
@@ -472,7 +463,7 @@ const Dashboard = () => {
return ( return (
<button <button
type="button" type="button"
className="card p-3 sm:p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left min-h-[44px]" className="card p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left"
onClick={handleUpToDateClick} onClick={handleUpToDateClick}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
@@ -501,7 +492,7 @@ const Dashboard = () => {
return ( return (
<button <button
type="button" type="button"
className="card p-3 sm:p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left min-h-[44px]" className="card p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left"
onClick={handleOutdatedPackagesClick} onClick={handleOutdatedPackagesClick}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
@@ -530,7 +521,7 @@ const Dashboard = () => {
return ( return (
<button <button
type="button" type="button"
className="card p-3 sm:p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left min-h-[44px]" className="card p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left"
onClick={handleSecurityUpdatesClick} onClick={handleSecurityUpdatesClick}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
@@ -559,7 +550,7 @@ const Dashboard = () => {
return ( return (
<button <button
type="button" type="button"
className="card p-3 sm:p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left min-h-[44px]" className="card p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left"
onClick={handleHostGroupsClick} onClick={handleHostGroupsClick}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
@@ -588,7 +579,7 @@ const Dashboard = () => {
return ( return (
<button <button
type="button" type="button"
className="card p-3 sm:p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left min-h-[44px]" className="card p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left"
onClick={handleUsersClick} onClick={handleUsersClick}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
@@ -617,7 +608,7 @@ const Dashboard = () => {
return ( return (
<button <button
type="button" type="button"
className="card p-3 sm:p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left min-h-[44px]" className="card p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left"
onClick={handleRepositoriesClick} onClick={handleRepositoriesClick}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
@@ -750,7 +741,7 @@ const Dashboard = () => {
case "osDistribution": case "osDistribution":
return ( return (
<div className="card p-4 sm:p-6 w-full"> <div className="card p-6 w-full">
<h3 className="text-lg font-medium text-secondary-900 dark:text-white mb-4"> <h3 className="text-lg font-medium text-secondary-900 dark:text-white mb-4">
OS Distribution OS Distribution
</h3> </h3>
@@ -764,7 +755,7 @@ const Dashboard = () => {
case "osDistributionDoughnut": case "osDistributionDoughnut":
return ( return (
<div className="card p-4 sm:p-6 w-full"> <div className="card p-6 w-full">
<h3 className="text-lg font-medium text-secondary-900 dark:text-white mb-4"> <h3 className="text-lg font-medium text-secondary-900 dark:text-white mb-4">
OS Distribution OS Distribution
</h3> </h3>
@@ -778,7 +769,7 @@ const Dashboard = () => {
case "osDistributionBar": case "osDistributionBar":
return ( return (
<div className="card p-4 sm:p-6 w-full"> <div className="card p-6 w-full">
<h3 className="text-lg font-medium text-secondary-900 dark:text-white mb-4"> <h3 className="text-lg font-medium text-secondary-900 dark:text-white mb-4">
OS Distribution OS Distribution
</h3> </h3>
@@ -792,7 +783,7 @@ const Dashboard = () => {
return ( return (
<button <button
type="button" type="button"
className="card p-4 sm:p-6 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left" className="card p-6 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 w-full text-left"
onClick={handleUpdateStatusClick} onClick={handleUpdateStatusClick}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
@@ -817,7 +808,7 @@ const Dashboard = () => {
case "packagePriority": case "packagePriority":
return ( return (
<div className="card p-4 sm:p-6 w-full"> <div className="card p-6 w-full">
<h3 className="text-lg font-medium text-secondary-900 dark:text-white mb-4"> <h3 className="text-lg font-medium text-secondary-900 dark:text-white mb-4">
Outdated Packages by Priority Outdated Packages by Priority
</h3> </h3>
@@ -834,13 +825,13 @@ const Dashboard = () => {
case "packageTrends": case "packageTrends":
return ( return (
<div className="card p-4 sm:p-6 w-full"> <div className="card p-6 w-full">
<div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-4 mb-4"> <div className="flex items-center justify-between mb-4">
<h3 className="text-lg font-medium text-secondary-900 dark:text-white"> <h3 className="text-lg font-medium text-secondary-900 dark:text-white">
Package Trends Over Time Package Trends Over Time
</h3> </h3>
<div className="flex flex-col gap-2"> <div className="flex flex-col gap-2">
<div className="flex flex-col sm:flex-row items-stretch sm:items-center gap-2 sm:gap-3"> <div className="flex items-center gap-3">
{/* Refresh Button */} {/* Refresh Button */}
<button <button
type="button" type="button"
@@ -878,7 +869,7 @@ const Dashboard = () => {
} }
}} }}
disabled={packageTrendsFetching || isTriggeringJob} disabled={packageTrendsFetching || isTriggeringJob}
className="px-3 py-2.5 sm:py-1.5 text-sm border border-secondary-300 dark:border-secondary-600 rounded-md bg-white dark:bg-secondary-800 text-secondary-900 dark:text-white hover:bg-secondary-50 dark:hover:bg-secondary-700 focus:ring-2 focus:ring-primary-500 focus:border-primary-500 disabled:opacity-50 disabled:cursor-not-allowed flex items-center justify-center gap-2 min-h-[44px]" className="px-3 py-1.5 text-sm border border-secondary-300 dark:border-secondary-600 rounded-md bg-white dark:bg-secondary-800 text-secondary-900 dark:text-white hover:bg-secondary-50 dark:hover:bg-secondary-700 focus:ring-2 focus:ring-primary-500 focus:border-primary-500 disabled:opacity-50 disabled:cursor-not-allowed flex items-center gap-2"
title={ title={
packageTrendsHost === "all" packageTrendsHost === "all"
? "Trigger system statistics collection" ? "Trigger system statistics collection"
@@ -899,7 +890,7 @@ const Dashboard = () => {
<select <select
value={packageTrendsPeriod} value={packageTrendsPeriod}
onChange={(e) => setPackageTrendsPeriod(e.target.value)} onChange={(e) => setPackageTrendsPeriod(e.target.value)}
className="px-3 py-2.5 sm:py-1.5 text-sm border border-secondary-300 dark:border-secondary-600 rounded-md bg-white dark:bg-secondary-800 text-secondary-900 dark:text-white focus:ring-2 focus:ring-primary-500 focus:border-primary-500 min-h-[44px]" className="px-3 py-1.5 text-sm border border-secondary-300 dark:border-secondary-600 rounded-md bg-white dark:bg-secondary-800 text-secondary-900 dark:text-white focus:ring-2 focus:ring-primary-500 focus:border-primary-500"
> >
<option value="1">Last 24 hours</option> <option value="1">Last 24 hours</option>
<option value="7">Last 7 days</option> <option value="7">Last 7 days</option>
@@ -917,7 +908,7 @@ const Dashboard = () => {
// Clear job ID message when host selection changes // Clear job ID message when host selection changes
setSystemStatsJobId(null); setSystemStatsJobId(null);
}} }}
className="px-3 py-2.5 sm:py-1.5 text-sm border border-secondary-300 dark:border-secondary-600 rounded-md bg-white dark:bg-secondary-800 text-secondary-900 dark:text-white focus:ring-2 focus:ring-primary-500 focus:border-primary-500 min-h-[44px]" className="px-3 py-1.5 text-sm border border-secondary-300 dark:border-secondary-600 rounded-md bg-white dark:bg-secondary-800 text-secondary-900 dark:text-white focus:ring-2 focus:ring-primary-500 focus:border-primary-500"
> >
<option value="all">All Hosts</option> <option value="all">All Hosts</option>
{packageTrendsData?.hosts?.length > 0 ? ( {packageTrendsData?.hosts?.length > 0 ? (
@@ -937,7 +928,7 @@ const Dashboard = () => {
</div> </div>
{/* Job ID Message */} {/* Job ID Message */}
{systemStatsJobId && packageTrendsHost === "all" && ( {systemStatsJobId && packageTrendsHost === "all" && (
<p className="text-xs text-secondary-600 dark:text-white/70 ml-1"> <p className="text-xs text-secondary-600 dark:text-secondary-400 ml-1">
Ran collection job #{systemStatsJobId} Ran collection job #{systemStatsJobId}
</p> </p>
)} )}
@@ -955,7 +946,7 @@ const Dashboard = () => {
options={packageTrendsChartOptions} options={packageTrendsChartOptions}
/> />
) : ( ) : (
<div className="flex items-center justify-center h-full text-secondary-500 dark:text-white/70"> <div className="flex items-center justify-center h-full text-secondary-500 dark:text-secondary-400">
No data available No data available
</div> </div>
)} )}
@@ -993,7 +984,7 @@ const Dashboard = () => {
: 0; : 0;
return ( return (
<div className="card p-4 sm:p-6"> <div className="card p-6">
<div className="flex items-center justify-between mb-4"> <div className="flex items-center justify-between mb-4">
<h3 className="text-lg font-medium text-secondary-900 dark:text-white"> <h3 className="text-lg font-medium text-secondary-900 dark:text-white">
System Overview System Overview
@@ -1004,10 +995,10 @@ const Dashboard = () => {
<div className="text-2xl font-bold text-primary-600"> <div className="text-2xl font-bold text-primary-600">
{updatePercentage}% {updatePercentage}%
</div> </div>
<div className="text-sm text-secondary-500 dark:text-white/70"> <div className="text-sm text-secondary-500 dark:text-secondary-300">
Need Updates Need Updates
</div> </div>
<div className="text-xs text-secondary-400 dark:text-white/60"> <div className="text-xs text-secondary-400 dark:text-secondary-500">
{stats.cards.hostsNeedingUpdates}/{stats.cards.totalHosts}{" "} {stats.cards.hostsNeedingUpdates}/{stats.cards.totalHosts}{" "}
hosts hosts
</div> </div>
@@ -1016,10 +1007,10 @@ const Dashboard = () => {
<div className="text-2xl font-bold text-danger-600"> <div className="text-2xl font-bold text-danger-600">
{stats.cards.securityUpdates} {stats.cards.securityUpdates}
</div> </div>
<div className="text-sm text-secondary-500 dark:text-white/70"> <div className="text-sm text-secondary-500 dark:text-secondary-300">
Security Issues Security Issues
</div> </div>
<div className="text-xs text-secondary-400 dark:text-white/60"> <div className="text-xs text-secondary-400 dark:text-secondary-500">
{securityPercentage}% of updates {securityPercentage}% of updates
</div> </div>
</div> </div>
@@ -1027,10 +1018,10 @@ const Dashboard = () => {
<div className="text-2xl font-bold text-success-600"> <div className="text-2xl font-bold text-success-600">
{onlinePercentage}% {onlinePercentage}%
</div> </div>
<div className="text-sm text-secondary-500 dark:text-white/70"> <div className="text-sm text-secondary-500 dark:text-secondary-300">
Online Online
</div> </div>
<div className="text-xs text-secondary-400 dark:text-white/60"> <div className="text-xs text-secondary-400 dark:text-secondary-500">
{onlineHosts}/{stats.cards.totalHosts} hosts {onlineHosts}/{stats.cards.totalHosts} hosts
</div> </div>
</div> </div>
@@ -1038,10 +1029,10 @@ const Dashboard = () => {
<div className="text-2xl font-bold text-secondary-600"> <div className="text-2xl font-bold text-secondary-600">
{avgPackagesPerHost} {avgPackagesPerHost}
</div> </div>
<div className="text-sm text-secondary-500 dark:text-white/70"> <div className="text-sm text-secondary-500 dark:text-secondary-300">
Avg per Host Avg per Host
</div> </div>
<div className="text-xs text-secondary-400 dark:text-white/60"> <div className="text-xs text-secondary-400 dark:text-secondary-500">
outdated packages outdated packages
</div> </div>
</div> </div>
@@ -1052,7 +1043,7 @@ const Dashboard = () => {
case "recentUsers": case "recentUsers":
return ( return (
<div className="card p-4 sm:p-6"> <div className="card p-6">
<h3 className="text-lg font-medium text-secondary-900 dark:text-white mb-4"> <h3 className="text-lg font-medium text-secondary-900 dark:text-white mb-4">
Recent Users Logged in Recent Users Logged in
</h3> </h3>
@@ -1066,7 +1057,7 @@ const Dashboard = () => {
<div className="text-sm font-medium text-secondary-900 dark:text-white"> <div className="text-sm font-medium text-secondary-900 dark:text-white">
{u.username} {u.username}
</div> </div>
<div className="text-sm text-secondary-500 dark:text-white/70"> <div className="text-sm text-secondary-500 dark:text-secondary-400">
{u.last_login {u.last_login
? formatRelativeTime(u.last_login) ? formatRelativeTime(u.last_login)
: "Never"} : "Never"}
@@ -1074,7 +1065,7 @@ const Dashboard = () => {
</div> </div>
))} ))}
{(!recentUsers || recentUsers.length === 0) && ( {(!recentUsers || recentUsers.length === 0) && (
<div className="text-center text-secondary-500 dark:text-white/70 py-4"> <div className="text-center text-secondary-500 dark:text-secondary-400 py-4">
No users found No users found
</div> </div>
)} )}
@@ -1085,7 +1076,7 @@ const Dashboard = () => {
case "recentCollection": case "recentCollection":
return ( return (
<div className="card p-4 sm:p-6"> <div className="card p-6">
<h3 className="text-lg font-medium text-secondary-900 dark:text-white mb-4"> <h3 className="text-lg font-medium text-secondary-900 dark:text-white mb-4">
Recent Collection Recent Collection
</h3> </h3>
@@ -1103,7 +1094,7 @@ const Dashboard = () => {
> >
{host.friendly_name || host.hostname} {host.friendly_name || host.hostname}
</button> </button>
<div className="text-sm text-secondary-500 dark:text-white/70"> <div className="text-sm text-secondary-500 dark:text-secondary-400">
{host.last_update {host.last_update
? formatRelativeTime(host.last_update) ? formatRelativeTime(host.last_update)
: "Never"} : "Never"}
@@ -1111,7 +1102,7 @@ const Dashboard = () => {
</div> </div>
))} ))}
{(!recentCollection || recentCollection.length === 0) && ( {(!recentCollection || recentCollection.length === 0) && (
<div className="text-center text-secondary-500 dark:text-white/70 py-4"> <div className="text-center text-secondary-500 dark:text-secondary-400 py-4">
No hosts found No hosts found
</div> </div>
)} )}
@@ -1163,13 +1154,13 @@ const Dashboard = () => {
maintainAspectRatio: false, maintainAspectRatio: false,
plugins: { plugins: {
legend: { legend: {
position: isMobile ? "bottom" : "right", position: "right",
labels: { labels: {
color: isDark ? "#ffffff" : "#374151", color: isDark ? "#ffffff" : "#374151",
font: { font: {
size: isMobile ? 10 : 12, size: 12,
}, },
padding: isMobile ? 10 : 15, padding: 15,
usePointStyle: true, usePointStyle: true,
pointStyle: "circle", pointStyle: "circle",
}, },
@@ -1177,7 +1168,7 @@ const Dashboard = () => {
}, },
layout: { layout: {
padding: { padding: {
right: isMobile ? 10 : 20, right: 20,
}, },
}, },
onClick: handleOSChartClick, onClick: handleOSChartClick,
@@ -1188,13 +1179,13 @@ const Dashboard = () => {
maintainAspectRatio: false, maintainAspectRatio: false,
plugins: { plugins: {
legend: { legend: {
position: isMobile ? "bottom" : "right", position: "right",
labels: { labels: {
color: isDark ? "#ffffff" : "#374151", color: isDark ? "#ffffff" : "#374151",
font: { font: {
size: isMobile ? 10 : 12, size: 12,
}, },
padding: isMobile ? 10 : 15, padding: 15,
usePointStyle: true, usePointStyle: true,
pointStyle: "circle", pointStyle: "circle",
}, },
@@ -1202,7 +1193,7 @@ const Dashboard = () => {
}, },
layout: { layout: {
padding: { padding: {
right: isMobile ? 10 : 20, right: 20,
}, },
}, },
onClick: handleOSChartClick, onClick: handleOSChartClick,
@@ -1593,10 +1584,10 @@ const Dashboard = () => {
{/* Page Header */} {/* Page Header */}
<div className="flex items-center justify-between"> <div className="flex items-center justify-between">
<div> <div>
<h1 className="text-xl sm:text-2xl font-semibold text-secondary-900 dark:text-white"> <h1 className="text-2xl font-semibold text-secondary-900 dark:text-white">
Welcome back, {user?.first_name || user?.username || "User"} 👋 Welcome back, {user?.first_name || user?.username || "User"} 👋
</h1> </h1>
<p className="text-sm text-secondary-600 dark:text-white/80 mt-1"> <p className="text-sm text-secondary-600 dark:text-secondary-400 mt-1">
Overview of your PatchMon infrastructure Overview of your PatchMon infrastructure
</p> </p>
</div> </div>
@@ -1604,7 +1595,7 @@ const Dashboard = () => {
<button <button
type="button" type="button"
onClick={() => setShowSettingsModal(true)} onClick={() => setShowSettingsModal(true)}
className="hidden md:flex btn-outline items-center gap-2" className="btn-outline flex items-center gap-2"
title="Customize dashboard layout" title="Customize dashboard layout"
> >
<Settings className="h-4 w-4" /> <Settings className="h-4 w-4" />
@@ -1613,7 +1604,7 @@ const Dashboard = () => {
type="button" type="button"
onClick={() => refetch()} onClick={() => refetch()}
disabled={isFetching} disabled={isFetching}
className="btn-outline flex items-center gap-2 min-h-[44px] min-w-[44px] justify-center" className="btn-outline flex items-center gap-2"
title="Refresh dashboard data" title="Refresh dashboard data"
> >
<RefreshCw <RefreshCw

View File

@@ -1013,25 +1013,29 @@ const HostDetail = () => {
</div> </div>
)} )}
{host.kernel_version && ( {(host.kernel_version ||
<div> host.installed_kernel_version) && (
<p className="text-xs text-secondary-500 dark:text-secondary-300"> <div className="flex flex-col gap-2">
Running Kernel {host.kernel_version && (
</p> <div>
<p className="font-medium text-secondary-900 dark:text-white font-mono text-sm break-all"> <p className="text-xs text-secondary-500 dark:text-secondary-300 mb-1">
{host.kernel_version} Running Kernel
</p> </p>
</div> <p className="font-medium text-secondary-900 dark:text-white font-mono text-sm break-all">
)} {host.kernel_version}
</p>
{host.installed_kernel_version && ( </div>
<div> )}
<p className="text-xs text-secondary-500 dark:text-secondary-300"> {host.installed_kernel_version && (
Installed Kernel <div>
</p> <p className="text-xs text-secondary-500 dark:text-secondary-300 mb-1">
<p className="font-medium text-secondary-900 dark:text-white font-mono text-sm break-all"> Installed Kernel
{host.installed_kernel_version} </p>
</p> <p className="font-medium text-secondary-900 dark:text-white font-mono text-sm break-all">
{host.installed_kernel_version}
</p>
</div>
)}
</div> </div>
)} )}

View File

@@ -13,7 +13,6 @@ import {
Eye as EyeIcon, Eye as EyeIcon,
EyeOff as EyeOffIcon, EyeOff as EyeOffIcon,
Filter, Filter,
FolderPlus,
GripVertical, GripVertical,
Plus, Plus,
RefreshCw, RefreshCw,
@@ -22,6 +21,7 @@ import {
Server, Server,
Square, Square,
Trash2, Trash2,
Users,
Wifi, Wifi,
X, X,
} from "lucide-react"; } from "lucide-react";
@@ -94,8 +94,8 @@ const AddHostModal = ({ isOpen, onClose, onSuccess }) => {
if (!isOpen) return null; if (!isOpen) return null;
return ( return (
<div className="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center z-50 p-4"> <div className="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center z-50">
<div className="bg-white dark:bg-secondary-800 rounded-lg p-4 sm:p-6 w-full max-w-md max-h-[90vh] overflow-y-auto"> <div className="bg-white dark:bg-secondary-800 rounded-lg p-6 w-full max-w-md">
<div className="flex justify-between items-center mb-4"> <div className="flex justify-between items-center mb-4">
<h3 className="text-lg font-medium text-secondary-900 dark:text-white"> <h3 className="text-lg font-medium text-secondary-900 dark:text-white">
Add New Host Add New Host
@@ -125,7 +125,7 @@ const AddHostModal = ({ isOpen, onClose, onSuccess }) => {
onChange={(e) => onChange={(e) =>
setFormData({ ...formData, friendly_name: e.target.value }) setFormData({ ...formData, friendly_name: e.target.value })
} }
className="block w-full px-3 py-3 sm:py-2.5 text-base border-2 border-secondary-300 dark:border-secondary-600 rounded-lg shadow-sm focus:ring-2 focus:ring-primary-500 focus:border-primary-500 bg-white dark:bg-secondary-700 text-secondary-900 dark:text-white transition-all duration-200 min-h-[44px]" className="block w-full px-3 py-2.5 text-base border-2 border-secondary-300 dark:border-secondary-600 rounded-lg shadow-sm focus:ring-2 focus:ring-primary-500 focus:border-primary-500 bg-white dark:bg-secondary-700 text-secondary-900 dark:text-white transition-all duration-200"
placeholder="server.example.com" placeholder="server.example.com"
/> />
<p className="mt-2 text-sm text-secondary-500 dark:text-secondary-400"> <p className="mt-2 text-sm text-secondary-500 dark:text-secondary-400">
@@ -197,18 +197,18 @@ const AddHostModal = ({ isOpen, onClose, onSuccess }) => {
</div> </div>
)} )}
<div className="flex flex-col-reverse sm:flex-row sm:justify-end gap-3 pt-2"> <div className="flex justify-end space-x-3 pt-2">
<button <button
type="button" type="button"
onClick={onClose} onClick={onClose}
className="px-6 py-3 text-sm font-medium text-secondary-700 dark:text-secondary-200 bg-white dark:bg-secondary-700 border-2 border-secondary-300 dark:border-secondary-600 rounded-lg hover:bg-secondary-50 dark:hover:bg-secondary-600 transition-all duration-200 min-h-[44px] w-full sm:w-auto" className="px-6 py-3 text-sm font-medium text-secondary-700 dark:text-secondary-200 bg-white dark:bg-secondary-700 border-2 border-secondary-300 dark:border-secondary-600 rounded-lg hover:bg-secondary-50 dark:hover:bg-secondary-600 transition-all duration-200"
> >
Cancel Cancel
</button> </button>
<button <button
type="submit" type="submit"
disabled={isSubmitting} disabled={isSubmitting}
className="px-6 py-3 text-sm font-medium text-white bg-primary-600 border-2 border-transparent rounded-lg hover:bg-primary-700 disabled:opacity-50 transition-all duration-200 min-h-[44px] w-full sm:w-auto" className="px-6 py-3 text-sm font-medium text-white bg-primary-600 border-2 border-transparent rounded-lg hover:bg-primary-700 disabled:opacity-50 transition-all duration-200"
> >
{isSubmitting ? "Creating..." : "Create Host"} {isSubmitting ? "Creating..." : "Create Host"}
</button> </button>
@@ -649,27 +649,11 @@ const Hosts = () => {
); );
}; };
const handleSelectAll = (hostsToSelect) => { const handleSelectAll = () => {
const hostIdsToSelect = hostsToSelect.map((host) => host.id); if (selectedHosts.length === hosts.length) {
const allSelected = hostIdsToSelect.every((id) => setSelectedHosts([]);
selectedHosts.includes(id),
);
if (allSelected) {
// Deselect all hosts in this group
setSelectedHosts((prev) =>
prev.filter((id) => !hostIdsToSelect.includes(id)),
);
} else { } else {
// Select all hosts in this group (merge with existing selections) setSelectedHosts(hosts.map((host) => host.id));
setSelectedHosts((prev) => {
const newSelection = [...prev];
hostIdsToSelect.forEach((id) => {
if (!newSelection.includes(id)) {
newSelection.push(id);
}
});
return newSelection;
});
} }
}; };
@@ -1122,17 +1106,11 @@ const Hosts = () => {
} }
> >
<div <div
className={`w-2 h-2 rounded-full ${wsStatus.connected ? "mr-1.5" : "mr-1.5 md:mr-0"} ${ className={`w-2 h-2 rounded-full mr-1.5 ${
wsStatus.connected ? "bg-green-500 animate-pulse" : "bg-red-500" wsStatus.connected ? "bg-green-500 animate-pulse" : "bg-red-500"
}`} }`}
></div> ></div>
<span className="hidden md:inline"> {wsStatus.connected ? (wsStatus.secure ? "WSS" : "WS") : "Offline"}
{wsStatus.connected
? wsStatus.secure
? "WSS"
: "WS"
: "Offline"}
</span>
</span> </span>
); );
} }
@@ -1307,14 +1285,14 @@ const Hosts = () => {
} }
return ( return (
<div className="min-h-0 flex flex-col md:h-[calc(100vh-7rem)] md:overflow-hidden"> <div className="h-[calc(100vh-7rem)] flex flex-col overflow-hidden">
{/* Page Header */} {/* Page Header */}
<div className="flex items-center justify-between mb-6"> <div className="flex items-center justify-between mb-6">
<div> <div>
<h1 className="text-2xl font-semibold text-secondary-900 dark:text-white"> <h1 className="text-2xl font-semibold text-secondary-900 dark:text-white">
Hosts Hosts
</h1> </h1>
<p className="text-sm text-secondary-600 dark:text-white/80 mt-1"> <p className="text-sm text-secondary-600 dark:text-secondary-400 mt-1">
Manage and monitor your connected hosts Manage and monitor your connected hosts
</p> </p>
</div> </div>
@@ -1342,7 +1320,7 @@ const Hosts = () => {
</div> </div>
{/* Stats Summary */} {/* Stats Summary */}
<div className="grid grid-cols-2 sm:grid-cols-4 gap-3 sm:gap-4 mb-6"> <div className="grid grid-cols-1 sm:grid-cols-4 gap-4 mb-6">
<button <button
type="button" type="button"
className="card p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 text-left w-full" className="card p-4 cursor-pointer hover:shadow-card-hover dark:hover:shadow-card-hover-dark transition-shadow duration-200 text-left w-full"
@@ -1426,7 +1404,7 @@ const Hosts = () => {
<span className="text-sm font-medium text-secondary-900 dark:text-white"> <span className="text-sm font-medium text-secondary-900 dark:text-white">
{connectedCount} {connectedCount}
</span> </span>
<span className="text-xs text-secondary-500 dark:text-secondary-400 hidden sm:inline"> <span className="text-xs text-secondary-500 dark:text-secondary-400">
Connected Connected
</span> </span>
</div> </div>
@@ -1435,7 +1413,7 @@ const Hosts = () => {
<span className="text-sm font-medium text-secondary-900 dark:text-white"> <span className="text-sm font-medium text-secondary-900 dark:text-white">
{offlineCount} {offlineCount}
</span> </span>
<span className="text-xs text-secondary-500 dark:text-secondary-400 hidden sm:inline"> <span className="text-xs text-secondary-500 dark:text-secondary-400">
Offline Offline
</span> </span>
</div> </div>
@@ -1448,39 +1426,37 @@ const Hosts = () => {
</div> </div>
{/* Hosts List */} {/* Hosts List */}
<div className="card flex-1 flex flex-col md:overflow-hidden min-h-0"> <div className="card flex-1 flex flex-col overflow-hidden min-h-0">
<div className="px-4 py-4 sm:p-4 flex-1 flex flex-col md:overflow-hidden min-h-0"> <div className="px-4 py-4 sm:p-4 flex-1 flex flex-col overflow-hidden min-h-0">
<div className="flex flex-col sm:flex-row sm:items-center sm:justify-end gap-3 mb-4"> <div className="flex items-center justify-end mb-4">
{selectedHosts.length > 0 && ( {selectedHosts.length > 0 && (
<div className="flex flex-wrap items-center gap-2 sm:gap-3"> <div className="flex items-center gap-3">
<span className="text-sm text-secondary-600 dark:text-white/80 flex-shrink-0"> <span className="text-sm text-secondary-600">
{selectedHosts.length} host {selectedHosts.length} host
{selectedHosts.length !== 1 ? "s" : ""} selected {selectedHosts.length !== 1 ? "s" : ""} selected
</span> </span>
<button <button
type="button" type="button"
onClick={() => setShowBulkAssignModal(true)} onClick={() => setShowBulkAssignModal(true)}
className="btn-outline flex items-center gap-1.5 sm:gap-2 px-3 sm:px-4 py-2 min-h-[44px] text-xs sm:text-sm" className="btn-outline flex items-center gap-2"
> >
<FolderPlus className="h-4 w-4 flex-shrink-0" /> <Users className="h-4 w-4" />
<span className="hidden sm:inline">Assign to Group</span> Assign to Group
<span className="sm:hidden">Assign</span>
</button> </button>
<button <button
type="button" type="button"
onClick={() => setShowBulkDeleteModal(true)} onClick={() => setShowBulkDeleteModal(true)}
className="btn-danger flex items-center gap-1.5 sm:gap-2 px-3 sm:px-4 py-2 min-h-[44px] text-xs sm:text-sm" className="btn-danger flex items-center gap-2"
> >
<Trash2 className="h-4 w-4 flex-shrink-0" /> <Trash2 className="h-4 w-4" />
<span>Delete</span> Delete
</button> </button>
<button <button
type="button" type="button"
onClick={() => setSelectedHosts([])} onClick={() => setSelectedHosts([])}
className="text-xs sm:text-sm text-secondary-500 dark:text-white/70 hover:text-secondary-700 dark:hover:text-white/90 min-h-[44px] px-2" className="text-sm text-secondary-500 hover:text-secondary-700"
> >
<span className="hidden sm:inline">Clear Selection</span> Clear Selection
<span className="sm:hidden">Clear</span>
</button> </button>
</div> </div>
)} )}
@@ -1502,28 +1478,28 @@ const Hosts = () => {
/> />
</div> </div>
</div> </div>
<div className="flex flex-wrap gap-2"> <div className="flex gap-2">
<button <button
type="button" type="button"
onClick={() => setShowFilters(!showFilters)} onClick={() => setShowFilters(!showFilters)}
className={`btn-outline flex items-center gap-1.5 sm:gap-2 px-2 sm:px-4 py-2 min-h-[44px] text-xs sm:text-sm ${showFilters ? "bg-primary-50 border-primary-300" : ""}`} className={`btn-outline flex items-center gap-2 ${showFilters ? "bg-primary-50 border-primary-300" : ""}`}
> >
<Filter className="h-4 w-4 flex-shrink-0" /> <Filter className="h-4 w-4" />
<span className="hidden sm:inline">Filters</span> Filters
</button> </button>
<button <button
type="button" type="button"
onClick={() => setShowColumnSettings(true)} onClick={() => setShowColumnSettings(true)}
className="btn-outline flex items-center gap-1.5 sm:gap-2 px-2 sm:px-4 py-2 min-h-[44px] text-xs sm:text-sm" className="btn-outline flex items-center gap-2"
> >
<Columns className="h-4 w-4 flex-shrink-0" /> <Columns className="h-4 w-4" />
<span className="hidden sm:inline">Columns</span> Columns
</button> </button>
<div className="relative"> <div className="relative">
<select <select
value={groupBy} value={groupBy}
onChange={(e) => setGroupBy(e.target.value)} onChange={(e) => setGroupBy(e.target.value)}
className="appearance-none bg-white dark:bg-secondary-800 border-2 border-secondary-300 dark:border-secondary-600 rounded-lg px-2 py-2 pr-6 text-xs sm:text-sm focus:ring-2 focus:ring-primary-500 focus:border-primary-500 text-secondary-900 dark:text-white hover:border-secondary-400 dark:hover:border-secondary-500 transition-colors min-w-[100px] sm:min-w-[120px] min-h-[44px]" className="appearance-none bg-white dark:bg-secondary-800 border-2 border-secondary-300 dark:border-secondary-600 rounded-lg px-2 py-2 pr-6 text-sm focus:ring-2 focus:ring-primary-500 focus:border-primary-500 text-secondary-900 dark:text-white hover:border-secondary-400 dark:hover:border-secondary-500 transition-colors min-w-[120px]"
> >
<option value="none">No Grouping</option> <option value="none">No Grouping</option>
<option value="group">By Group</option> <option value="group">By Group</option>
@@ -1535,18 +1511,18 @@ const Hosts = () => {
<button <button
type="button" type="button"
onClick={() => setHideStale(!hideStale)} onClick={() => setHideStale(!hideStale)}
className={`btn-outline flex items-center gap-1.5 sm:gap-2 px-2 sm:px-4 py-2 min-h-[44px] text-xs sm:text-sm ${hideStale ? "bg-primary-50 border-primary-300" : ""}`} className={`btn-outline flex items-center gap-2 ${hideStale ? "bg-primary-50 border-primary-300" : ""}`}
> >
<AlertTriangle className="h-4 w-4 flex-shrink-0" /> <AlertTriangle className="h-4 w-4" />
<span className="hidden sm:inline">Hide Stale</span> Hide Stale
</button> </button>
</div> </div>
</div> </div>
{/* Advanced Filters */} {/* Advanced Filters */}
{showFilters && ( {showFilters && (
<div className="bg-secondary-50 dark:bg-secondary-700 p-3 sm:p-4 rounded-lg border dark:border-secondary-600"> <div className="bg-secondary-50 dark:bg-secondary-700 p-4 rounded-lg border dark:border-secondary-600">
<div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-4 gap-3 sm:gap-4"> <div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-4 gap-4">
<div> <div>
<label <label
htmlFor={hostGroupFilterId} htmlFor={hostGroupFilterId}
@@ -1558,7 +1534,7 @@ const Hosts = () => {
id={hostGroupFilterId} id={hostGroupFilterId}
value={groupFilter} value={groupFilter}
onChange={(e) => setGroupFilter(e.target.value)} onChange={(e) => setGroupFilter(e.target.value)}
className="w-full border border-secondary-300 dark:border-secondary-600 rounded-lg px-3 py-2.5 sm:py-2 focus:ring-2 focus:ring-primary-500 focus:border-primary-500 bg-white dark:bg-secondary-800 text-secondary-900 dark:text-white min-h-[44px]" className="w-full border border-secondary-300 dark:border-secondary-600 rounded-lg px-3 py-2 focus:ring-2 focus:ring-primary-500 focus:border-primary-500 bg-white dark:bg-secondary-800 text-secondary-900 dark:text-white"
> >
<option value="all">All Groups</option> <option value="all">All Groups</option>
<option value="ungrouped">Ungrouped</option> <option value="ungrouped">Ungrouped</option>
@@ -1580,7 +1556,7 @@ const Hosts = () => {
id={statusFilterId} id={statusFilterId}
value={statusFilter} value={statusFilter}
onChange={(e) => setStatusFilter(e.target.value)} onChange={(e) => setStatusFilter(e.target.value)}
className="w-full border border-secondary-300 dark:border-secondary-600 rounded-lg px-3 py-2.5 sm:py-2 focus:ring-2 focus:ring-primary-500 focus:border-primary-500 bg-white dark:bg-secondary-800 text-secondary-900 dark:text-white min-h-[44px]" className="w-full border border-secondary-300 dark:border-secondary-600 rounded-lg px-3 py-2 focus:ring-2 focus:ring-primary-500 focus:border-primary-500 bg-white dark:bg-secondary-800 text-secondary-900 dark:text-white"
> >
<option value="all">All Status</option> <option value="all">All Status</option>
<option value="active">Active</option> <option value="active">Active</option>
@@ -1600,7 +1576,7 @@ const Hosts = () => {
id={osFilterId} id={osFilterId}
value={osFilter} value={osFilter}
onChange={(e) => setOsFilter(e.target.value)} onChange={(e) => setOsFilter(e.target.value)}
className="w-full border border-secondary-300 dark:border-secondary-600 rounded-lg px-3 py-2.5 sm:py-2 focus:ring-2 focus:ring-primary-500 focus:border-primary-500 bg-white dark:bg-secondary-800 text-secondary-900 dark:text-white min-h-[44px]" className="w-full border border-secondary-300 dark:border-secondary-600 rounded-lg px-3 py-2 focus:ring-2 focus:ring-primary-500 focus:border-primary-500 bg-white dark:bg-secondary-800 text-secondary-900 dark:text-white"
> >
<option value="all">All OS</option> <option value="all">All OS</option>
{uniqueOsTypes.map((osType) => ( {uniqueOsTypes.map((osType) => (
@@ -1621,7 +1597,7 @@ const Hosts = () => {
setGroupBy("none"); setGroupBy("none");
setHideStale(false); setHideStale(false);
}} }}
className="btn-outline w-full min-h-[44px]" className="btn-outline w-full"
> >
Clear Filters Clear Filters
</button> </button>
@@ -1631,7 +1607,7 @@ const Hosts = () => {
)} )}
</div> </div>
<div className="flex-1 md:overflow-hidden"> <div className="flex-1 overflow-hidden">
{!hosts || hosts.length === 0 ? ( {!hosts || hosts.length === 0 ? (
<div className="text-center py-8"> <div className="text-center py-8">
<Server className="h-12 w-12 text-secondary-400 mx-auto mb-4" /> <Server className="h-12 w-12 text-secondary-400 mx-auto mb-4" />
@@ -1652,7 +1628,7 @@ const Hosts = () => {
</p> </p>
</div> </div>
) : ( ) : (
<div className="md:h-full overflow-auto"> <div className="h-full overflow-auto">
<div className="space-y-6"> <div className="space-y-6">
{Object.entries(groupedHosts).map( {Object.entries(groupedHosts).map(
([groupName, groupHosts]) => ( ([groupName, groupHosts]) => (
@@ -1666,246 +1642,24 @@ const Hosts = () => {
</div> </div>
)} )}
{/* Mobile Card Layout */} {/* Table for this group */}
<div className="md:hidden space-y-3"> <div className="overflow-x-auto">
{groupHosts.map((host) => { <table className="min-w-full divide-y divide-secondary-200 dark:divide-secondary-600">
const isInactive =
(host.effectiveStatus || host.status) ===
"inactive";
const isSelected = selectedHosts.includes(host.id);
const wsStatus = wsStatusMap[host.api_id];
const groupIds =
host.host_group_memberships?.map(
(membership) => membership.host_groups.id,
) || [];
const groups =
hostGroups?.filter((g) =>
groupIds.includes(g.id),
) || [];
return (
<div
key={host.id}
className={`card p-4 space-y-3 ${
isSelected
? "ring-2 ring-primary-500 bg-primary-50 dark:bg-primary-900/20"
: isInactive
? "bg-red-50 dark:bg-red-900/20"
: ""
}`}
>
{/* Header with select and main info */}
<div className="flex items-start justify-between gap-3">
<div className="flex items-center gap-2 flex-1 min-w-0">
{visibleColumns.some(
(col) => col.id === "select",
) && (
<button
type="button"
onClick={() =>
handleSelectHost(host.id)
}
className="flex-shrink-0 min-w-[44px] min-h-[44px] flex items-center justify-center"
>
{isSelected ? (
<CheckSquare className="h-5 w-5 text-primary-600" />
) : (
<Square className="h-5 w-5 text-secondary-400" />
)}
</button>
)}
<div className="flex-1 min-w-0">
{visibleColumns.some(
(col) => col.id === "host",
) && (
<Link
to={`/hosts/${host.id}`}
className="text-base font-semibold text-secondary-900 dark:text-white hover:text-primary-600 dark:hover:text-primary-400 block truncate"
>
{host.friendly_name || "Unnamed Host"}
</Link>
)}
{visibleColumns.some(
(col) => col.id === "hostname",
) &&
host.hostname && (
<div className="text-sm text-secondary-500 dark:text-secondary-400 font-mono truncate">
{host.hostname}
</div>
)}
</div>
</div>
{visibleColumns.some(
(col) => col.id === "actions",
) && (
<Link
to={`/hosts/${host.id}`}
className="btn-primary text-sm px-3 py-2 min-h-[44px] flex items-center gap-1 flex-shrink-0"
>
View
<ExternalLink className="h-4 w-4" />
</Link>
)}
</div>
{/* Status and connection info */}
<div className="flex flex-wrap items-center gap-2">
{visibleColumns.some(
(col) => col.id === "status",
) && (
<span className="inline-flex items-center px-2 py-1 rounded-md text-xs font-medium bg-secondary-100 text-secondary-700 dark:bg-secondary-700 dark:text-secondary-300">
{(host.effectiveStatus || host.status)
.charAt(0)
.toUpperCase() +
(
host.effectiveStatus || host.status
).slice(1)}
</span>
)}
{visibleColumns.some(
(col) => col.id === "ws_status",
) &&
wsStatus && (
<span
className={`inline-flex items-center px-2 py-1 rounded-md text-xs font-medium ${
wsStatus.connected
? "bg-green-100 text-green-800 dark:bg-green-900 dark:text-green-200"
: "bg-red-100 text-red-800 dark:bg-red-900 dark:text-red-200"
}`}
>
<div
className={`w-2 h-2 rounded-full ${wsStatus.connected ? "mr-1.5" : "mr-1.5 md:mr-0"} ${
wsStatus.connected
? "bg-green-500 animate-pulse"
: "bg-red-500"
}`}
></div>
<span className="hidden md:inline">
{wsStatus.connected
? wsStatus.secure
? "WSS"
: "WS"
: "Offline"}
</span>
</span>
)}
{visibleColumns.some(
(col) => col.id === "needs_reboot",
) &&
(host.needs_reboot ? (
<span className="inline-flex items-center gap-1 px-2 py-1 rounded-md text-xs font-medium bg-orange-100 text-orange-800 dark:bg-orange-900 dark:text-orange-200">
<RotateCcw className="h-3 w-3" />
Reboot Required
</span>
) : null)}
</div>
{/* OS and Group info */}
<div className="flex flex-wrap items-center gap-3 text-sm">
{visibleColumns.some(
(col) => col.id === "os",
) && (
<div className="flex items-center gap-2">
<OSIcon
osType={host.os_type}
className="h-4 w-4"
/>
<span className="text-secondary-700 dark:text-secondary-300">
{getOSDisplayName(host.os_type)}
</span>
</div>
)}
{visibleColumns.some(
(col) => col.id === "group",
) &&
groups.length > 0 && (
<div className="flex items-center gap-1 flex-wrap">
<span className="text-secondary-500 dark:text-secondary-400">
Groups:
</span>
{groups.map((g, idx) => (
<span
key={g.id}
className="text-secondary-700 dark:text-secondary-300"
>
{g.name}
{idx < groups.length - 1 ? "," : ""}
</span>
))}
</div>
)}
</div>
{/* Updates info */}
<div className="flex items-center gap-4 pt-2 border-t border-secondary-200 dark:border-secondary-600">
{visibleColumns.some(
(col) => col.id === "updates",
) && (
<button
type="button"
onClick={() =>
navigate(
`/packages?host=${host.id}&filter=outdated`,
)
}
className="text-sm text-primary-600 hover:text-primary-900 dark:text-primary-400 dark:hover:text-primary-300 font-medium min-h-[44px] flex items-center"
>
{host.updatesCount || 0} Updates
</button>
)}
{visibleColumns.some(
(col) => col.id === "security_updates",
) && (
<button
type="button"
onClick={() =>
navigate(
`/packages?host=${host.id}&filter=security-updates`,
)
}
className="text-sm text-red-600 hover:text-red-900 dark:text-red-400 dark:hover:text-red-300 font-medium min-h-[44px] flex items-center"
>
{host.securityUpdatesCount || 0} Security
</button>
)}
{visibleColumns.some(
(col) => col.id === "last_update",
) && (
<div className="text-xs text-secondary-500 dark:text-secondary-400 ml-auto">
Updated{" "}
{formatRelativeTime(host.last_update)}
</div>
)}
</div>
</div>
);
})}
</div>
{/* Desktop Table Layout */}
<div className="hidden md:block overflow-x-auto">
<table
className="min-w-full divide-y divide-secondary-200 dark:divide-secondary-600"
style={{ minWidth: "max-content" }}
>
<thead className="bg-secondary-50 dark:bg-secondary-700"> <thead className="bg-secondary-50 dark:bg-secondary-700">
<tr> <tr>
{visibleColumns.map((column) => ( {visibleColumns.map((column) => (
<th <th
key={column.id} key={column.id}
className="px-3 sm:px-4 py-2 text-center text-xs font-medium text-secondary-500 dark:text-secondary-300 uppercase tracking-wider whitespace-nowrap" className="px-4 py-2 text-center text-xs font-medium text-secondary-500 dark:text-secondary-300 uppercase tracking-wider"
> >
{column.id === "select" ? ( {column.id === "select" ? (
<button <button
type="button" type="button"
onClick={() => onClick={handleSelectAll}
handleSelectAll(groupHosts)
}
className="flex items-center gap-2 hover:text-secondary-700" className="flex items-center gap-2 hover:text-secondary-700"
> >
{groupHosts.every((host) => {selectedHosts.length ===
selectedHosts.includes(host.id), groupHosts.length ? (
) ? (
<CheckSquare className="h-4 w-4" /> <CheckSquare className="h-4 w-4" />
) : ( ) : (
<Square className="h-4 w-4" /> <Square className="h-4 w-4" />
@@ -2069,7 +1823,7 @@ const Hosts = () => {
{visibleColumns.map((column) => ( {visibleColumns.map((column) => (
<td <td
key={column.id} key={column.id}
className="px-3 sm:px-4 py-2 whitespace-nowrap text-center" className="px-4 py-2 whitespace-nowrap text-center"
> >
{renderCellContent(column, host)} {renderCellContent(column, host)}
</td> </td>
@@ -2411,6 +2165,7 @@ const ColumnSettingsModal = ({
key={column.id} key={column.id}
type="button" type="button"
draggable draggable
tabIndex={0}
aria-label={`Drag to reorder ${column.label} column`} aria-label={`Drag to reorder ${column.label} column`}
onDragStart={(e) => handleDragStart(e, index)} onDragStart={(e) => handleDragStart(e, index)}
onDragOver={handleDragOver} onDragOver={handleDragOver}
@@ -2421,7 +2176,7 @@ const ColumnSettingsModal = ({
// Focus handling for keyboard users // Focus handling for keyboard users
} }
}} }}
className={`flex items-center justify-between p-2.5 border rounded-lg cursor-move w-full transition-colors ${ className={`flex items-center justify-between p-2.5 border rounded-lg cursor-move w-full text-left transition-colors ${
draggedIndex === index draggedIndex === index
? "opacity-50" ? "opacity-50"
: "hover:bg-secondary-50 dark:hover:bg-secondary-700" : "hover:bg-secondary-50 dark:hover:bg-secondary-700"
@@ -2435,20 +2190,12 @@ const ColumnSettingsModal = ({
</div> </div>
<button <button
type="button" type="button"
onClick={(e) => { onClick={() => onToggleVisibility(column.id)}
e.stopPropagation(); className={`p-1 rounded transition-colors flex-shrink-0 ${
onToggleVisibility(column.id);
}}
className={`p-1 rounded transition-colors flex-shrink-0 min-w-[44px] min-h-[44px] flex items-center justify-center ${
column.visible column.visible
? "text-primary-600 hover:text-primary-700 dark:text-primary-400 dark:hover:text-primary-300" ? "text-primary-600 hover:text-primary-700 dark:text-primary-400 dark:hover:text-primary-300"
: "text-secondary-400 hover:text-secondary-600 dark:text-secondary-500 dark:hover:text-secondary-300" : "text-secondary-400 hover:text-secondary-600 dark:text-secondary-500 dark:hover:text-secondary-300"
}`} }`}
aria-label={
column.visible
? `Hide ${column.label} column`
: `Show ${column.label} column`
}
> >
{column.visible ? ( {column.visible ? (
<EyeIcon className="h-4 w-4" /> <EyeIcon className="h-4 w-4" />

View File

@@ -1586,7 +1586,7 @@ const Integrations = () => {
<div className="flex items-center gap-2"> <div className="flex items-center gap-2">
<input <input
type="text" type="text"
value={`curl ${curl_flags} "${getEnrollmentUrl()}" | ${selected_script_type === "proxmox-lxc" ? "bash" : "sh"}`} value={`curl ${curl_flags} "${getEnrollmentUrl()}" | sh`}
readOnly readOnly
className="flex-1 px-3 py-2 border border-secondary-300 dark:border-secondary-600 rounded-md bg-secondary-50 dark:bg-secondary-900 text-secondary-900 dark:text-white font-mono text-xs" className="flex-1 px-3 py-2 border border-secondary-300 dark:border-secondary-600 rounded-md bg-secondary-50 dark:bg-secondary-900 text-secondary-900 dark:text-white font-mono text-xs"
/> />
@@ -1594,7 +1594,7 @@ const Integrations = () => {
type="button" type="button"
onClick={() => onClick={() =>
copy_to_clipboard( copy_to_clipboard(
`curl ${curl_flags} "${getEnrollmentUrl()}" | ${selected_script_type === "proxmox-lxc" ? "bash" : "sh"}`, `curl ${curl_flags} "${getEnrollmentUrl()}" | sh`,
"enrollment-command", "enrollment-command",
) )
} }

216
package-lock.json generated
View File

@@ -1,12 +1,12 @@
{ {
"name": "patchmon", "name": "patchmon",
"version": "1.3.4", "version": "1.3.5",
"lockfileVersion": 3, "lockfileVersion": 3,
"requires": true, "requires": true,
"packages": { "packages": {
"": { "": {
"name": "patchmon", "name": "patchmon",
"version": "1.3.4", "version": "1.3.5",
"license": "AGPL-3.0", "license": "AGPL-3.0",
"workspaces": [ "workspaces": [
"backend", "backend",
@@ -23,7 +23,7 @@
}, },
"backend": { "backend": {
"name": "patchmon-backend", "name": "patchmon-backend",
"version": "1.3.4", "version": "1.3.5",
"license": "AGPL-3.0", "license": "AGPL-3.0",
"dependencies": { "dependencies": {
"@bull-board/api": "^6.13.1", "@bull-board/api": "^6.13.1",
@@ -59,7 +59,7 @@
}, },
"frontend": { "frontend": {
"name": "patchmon-frontend", "name": "patchmon-frontend",
"version": "1.3.4", "version": "1.3.5",
"license": "AGPL-3.0", "license": "AGPL-3.0",
"dependencies": { "dependencies": {
"@dnd-kit/core": "^6.3.1", "@dnd-kit/core": "^6.3.1",
@@ -83,7 +83,7 @@
"devDependencies": { "devDependencies": {
"@types/react": "^18.3.14", "@types/react": "^18.3.14",
"@types/react-dom": "^18.3.1", "@types/react-dom": "^18.3.1",
"@vitejs/plugin-react": "^4.3.4", "@vitejs/plugin-react": "^5.0.0",
"autoprefixer": "^10.4.20", "autoprefixer": "^10.4.20",
"postcss": "^8.5.6", "postcss": "^8.5.6",
"tailwindcss": "^3.4.17", "tailwindcss": "^3.4.17",
@@ -131,19 +131,22 @@
} }
}, },
"node_modules/@babel/core": { "node_modules/@babel/core": {
"version": "7.28.4", "version": "7.28.5",
"resolved": "https://registry.npmjs.org/@babel/core/-/core-7.28.5.tgz",
"integrity": "sha512-e7jT4DxYvIDLk1ZHmU/m/mB19rex9sv0c2ftBtjSBv+kVM/902eh0fINUzD7UwLLNR+jU585GxUJ8/EBfAM5fw==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"peer": true,
"dependencies": { "dependencies": {
"@babel/code-frame": "^7.27.1", "@babel/code-frame": "^7.27.1",
"@babel/generator": "^7.28.3", "@babel/generator": "^7.28.5",
"@babel/helper-compilation-targets": "^7.27.2", "@babel/helper-compilation-targets": "^7.27.2",
"@babel/helper-module-transforms": "^7.28.3", "@babel/helper-module-transforms": "^7.28.3",
"@babel/helpers": "^7.28.4", "@babel/helpers": "^7.28.4",
"@babel/parser": "^7.28.4", "@babel/parser": "^7.28.5",
"@babel/template": "^7.27.2", "@babel/template": "^7.27.2",
"@babel/traverse": "^7.28.4", "@babel/traverse": "^7.28.5",
"@babel/types": "^7.28.4", "@babel/types": "^7.28.5",
"@jridgewell/remapping": "^2.3.5", "@jridgewell/remapping": "^2.3.5",
"convert-source-map": "^2.0.0", "convert-source-map": "^2.0.0",
"debug": "^4.1.0", "debug": "^4.1.0",
@@ -160,12 +163,14 @@
} }
}, },
"node_modules/@babel/generator": { "node_modules/@babel/generator": {
"version": "7.28.3", "version": "7.28.5",
"resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.28.5.tgz",
"integrity": "sha512-3EwLFhZ38J4VyIP6WNtt2kUdW9dokXA9Cr4IVIFHuCpZ3H8/YFOl5JjZHisrn1fATPBmKKqXzDFvh9fUwHz6CQ==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@babel/parser": "^7.28.3", "@babel/parser": "^7.28.5",
"@babel/types": "^7.28.2", "@babel/types": "^7.28.5",
"@jridgewell/gen-mapping": "^0.3.12", "@jridgewell/gen-mapping": "^0.3.12",
"@jridgewell/trace-mapping": "^0.3.28", "@jridgewell/trace-mapping": "^0.3.28",
"jsesc": "^3.0.2" "jsesc": "^3.0.2"
@@ -242,7 +247,9 @@
} }
}, },
"node_modules/@babel/helper-validator-identifier": { "node_modules/@babel/helper-validator-identifier": {
"version": "7.27.1", "version": "7.28.5",
"resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz",
"integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"engines": { "engines": {
@@ -270,11 +277,13 @@
} }
}, },
"node_modules/@babel/parser": { "node_modules/@babel/parser": {
"version": "7.28.4", "version": "7.28.5",
"resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.28.5.tgz",
"integrity": "sha512-KKBU1VGYR7ORr3At5HAtUQ+TV3SzRCXmA/8OdDZiLDBIZxVyzXuztPjfLd3BV1PRAQGCMWWSHYhL0F8d5uHBDQ==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@babel/types": "^7.28.4" "@babel/types": "^7.28.5"
}, },
"bin": { "bin": {
"parser": "bin/babel-parser.js" "parser": "bin/babel-parser.js"
@@ -333,16 +342,18 @@
} }
}, },
"node_modules/@babel/traverse": { "node_modules/@babel/traverse": {
"version": "7.28.4", "version": "7.28.5",
"resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.28.5.tgz",
"integrity": "sha512-TCCj4t55U90khlYkVV/0TfkJkAkUg3jZFA3Neb7unZT8CPok7iiRfaX0F+WnqWqt7OxhOn0uBKXCw4lbL8W0aQ==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@babel/code-frame": "^7.27.1", "@babel/code-frame": "^7.27.1",
"@babel/generator": "^7.28.3", "@babel/generator": "^7.28.5",
"@babel/helper-globals": "^7.28.0", "@babel/helper-globals": "^7.28.0",
"@babel/parser": "^7.28.4", "@babel/parser": "^7.28.5",
"@babel/template": "^7.27.2", "@babel/template": "^7.27.2",
"@babel/types": "^7.28.4", "@babel/types": "^7.28.5",
"debug": "^4.3.1" "debug": "^4.3.1"
}, },
"engines": { "engines": {
@@ -350,12 +361,14 @@
} }
}, },
"node_modules/@babel/types": { "node_modules/@babel/types": {
"version": "7.28.4", "version": "7.28.5",
"resolved": "https://registry.npmjs.org/@babel/types/-/types-7.28.5.tgz",
"integrity": "sha512-qQ5m48eI/MFLQ5PxQj4PFaprjyCTLI37ElWMmNs0K8Lk3dVeOdNpB3ks8jc7yM5CDmVC73eMVk/trk3fgmrUpA==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@babel/helper-string-parser": "^7.27.1", "@babel/helper-string-parser": "^7.27.1",
"@babel/helper-validator-identifier": "^7.27.1" "@babel/helper-validator-identifier": "^7.28.5"
}, },
"engines": { "engines": {
"node": ">=6.9.0" "node": ">=6.9.0"
@@ -547,6 +560,7 @@
"node_modules/@bull-board/ui": { "node_modules/@bull-board/ui": {
"version": "6.13.1", "version": "6.13.1",
"license": "MIT", "license": "MIT",
"peer": true,
"dependencies": { "dependencies": {
"@bull-board/api": "6.13.1" "@bull-board/api": "6.13.1"
} }
@@ -580,6 +594,7 @@
"node_modules/@dnd-kit/core": { "node_modules/@dnd-kit/core": {
"version": "6.3.1", "version": "6.3.1",
"license": "MIT", "license": "MIT",
"peer": true,
"dependencies": { "dependencies": {
"@dnd-kit/accessibility": "^3.1.1", "@dnd-kit/accessibility": "^3.1.1",
"@dnd-kit/utilities": "^3.2.2", "@dnd-kit/utilities": "^3.2.2",
@@ -895,7 +910,9 @@
} }
}, },
"node_modules/@rolldown/pluginutils": { "node_modules/@rolldown/pluginutils": {
"version": "1.0.0-beta.27", "version": "1.0.0-beta.47",
"resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.47.tgz",
"integrity": "sha512-8QagwMH3kNCuzD8EWL8R2YPW5e4OrHNSAHRFDdmFqEwEaD/KcNKjVoumo+gP2vW5eKB2UPbM6vTYiGZX0ixLnw==",
"dev": true, "dev": true,
"license": "MIT" "license": "MIT"
}, },
@@ -1020,6 +1037,7 @@
"version": "18.3.24", "version": "18.3.24",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"peer": true,
"dependencies": { "dependencies": {
"@types/prop-types": "*", "@types/prop-types": "*",
"csstype": "^3.0.2" "csstype": "^3.0.2"
@@ -1038,19 +1056,21 @@
"license": "MIT" "license": "MIT"
}, },
"node_modules/@vitejs/plugin-react": { "node_modules/@vitejs/plugin-react": {
"version": "4.7.0", "version": "5.1.1",
"resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-5.1.1.tgz",
"integrity": "sha512-WQfkSw0QbQ5aJ2CHYw23ZGkqnRwqKHD/KYsMeTkZzPT4Jcf0DcBxBtwMJxnu6E7oxw5+JC6ZAiePgh28uJ1HBA==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@babel/core": "^7.28.0", "@babel/core": "^7.28.5",
"@babel/plugin-transform-react-jsx-self": "^7.27.1", "@babel/plugin-transform-react-jsx-self": "^7.27.1",
"@babel/plugin-transform-react-jsx-source": "^7.27.1", "@babel/plugin-transform-react-jsx-source": "^7.27.1",
"@rolldown/pluginutils": "1.0.0-beta.27", "@rolldown/pluginutils": "1.0.0-beta.47",
"@types/babel__core": "^7.20.5", "@types/babel__core": "^7.20.5",
"react-refresh": "^0.17.0" "react-refresh": "^0.18.0"
}, },
"engines": { "engines": {
"node": "^14.18.0 || >=16.0.0" "node": "^20.19.0 || >=22.12.0"
}, },
"peerDependencies": { "peerDependencies": {
"vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0" "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0"
@@ -1267,6 +1287,7 @@
} }
], ],
"license": "MIT", "license": "MIT",
"peer": true,
"dependencies": { "dependencies": {
"baseline-browser-mapping": "^2.8.3", "baseline-browser-mapping": "^2.8.3",
"caniuse-lite": "^1.0.30001741", "caniuse-lite": "^1.0.30001741",
@@ -1456,6 +1477,7 @@
"node_modules/chart.js": { "node_modules/chart.js": {
"version": "4.5.0", "version": "4.5.0",
"license": "MIT", "license": "MIT",
"peer": true,
"dependencies": { "dependencies": {
"@kurkle/color": "^0.3.0" "@kurkle/color": "^0.3.0"
}, },
@@ -2030,6 +2052,7 @@
"node_modules/express": { "node_modules/express": {
"version": "4.21.2", "version": "4.21.2",
"license": "MIT", "license": "MIT",
"peer": true,
"dependencies": { "dependencies": {
"accepts": "~1.3.8", "accepts": "~1.3.8",
"array-flatten": "1.1.1", "array-flatten": "1.1.1",
@@ -2795,6 +2818,76 @@
"lefthook-windows-x64": "1.13.5" "lefthook-windows-x64": "1.13.5"
} }
}, },
"node_modules/lefthook-darwin-arm64": {
"version": "1.13.5",
"resolved": "https://registry.npmjs.org/lefthook-darwin-arm64/-/lefthook-darwin-arm64-1.13.5.tgz",
"integrity": "sha512-BYt5CnAOXasVCS6i+A4ljUo9xru/B5uMFD6EWHhs3R26jGF7mBSDxM3ErzXTUaJRTP0kQI/XBmgqBryBqoqZOQ==",
"cpu": [
"arm64"
],
"dev": true,
"license": "MIT",
"optional": true,
"os": [
"darwin"
]
},
"node_modules/lefthook-darwin-x64": {
"version": "1.13.5",
"resolved": "https://registry.npmjs.org/lefthook-darwin-x64/-/lefthook-darwin-x64-1.13.5.tgz",
"integrity": "sha512-ZDtLBzvI5e26C/RZ4irOHpELTd22x9lDTgF2+eCYcnrBWOkB7800V8tuAvBybsLGvg6JwKjFxn+NTRNZnCC2hw==",
"cpu": [
"x64"
],
"dev": true,
"license": "MIT",
"optional": true,
"os": [
"darwin"
]
},
"node_modules/lefthook-freebsd-arm64": {
"version": "1.13.5",
"resolved": "https://registry.npmjs.org/lefthook-freebsd-arm64/-/lefthook-freebsd-arm64-1.13.5.tgz",
"integrity": "sha512-uQ/kQZSSedw74aGCpsfOPN4yVt3klg8grOP6gHQOCRUMv5oK/Lj3pe1PylpTuuhxWORWRzkauPMot26J0OZZdA==",
"cpu": [
"arm64"
],
"dev": true,
"license": "MIT",
"optional": true,
"os": [
"freebsd"
]
},
"node_modules/lefthook-freebsd-x64": {
"version": "1.13.5",
"resolved": "https://registry.npmjs.org/lefthook-freebsd-x64/-/lefthook-freebsd-x64-1.13.5.tgz",
"integrity": "sha512-6czek8XagVrI7ExURawkfrfX40Qjc/wktc8bLq/iXfRlmdvKDMrx2FrA82mDfEVCAEz+tTvkteK1TfR3icYF3Q==",
"cpu": [
"x64"
],
"dev": true,
"license": "MIT",
"optional": true,
"os": [
"freebsd"
]
},
"node_modules/lefthook-linux-arm64": {
"version": "1.13.5",
"resolved": "https://registry.npmjs.org/lefthook-linux-arm64/-/lefthook-linux-arm64-1.13.5.tgz",
"integrity": "sha512-MjWtiuW1br+rpTtgG1KGV53mSGtL5MWQwgafYzrFleJ89fKb86F4TD/4mVNzk5thmZ+HVPZw9bRZGUHFBnNJWg==",
"cpu": [
"arm64"
],
"dev": true,
"license": "MIT",
"optional": true,
"os": [
"linux"
]
},
"node_modules/lefthook-linux-x64": { "node_modules/lefthook-linux-x64": {
"version": "1.13.5", "version": "1.13.5",
"cpu": [ "cpu": [
@@ -2807,6 +2900,62 @@
"linux" "linux"
] ]
}, },
"node_modules/lefthook-openbsd-arm64": {
"version": "1.13.5",
"resolved": "https://registry.npmjs.org/lefthook-openbsd-arm64/-/lefthook-openbsd-arm64-1.13.5.tgz",
"integrity": "sha512-lYXrWf0/hBrwtG8ceaHq886bcqRKh3Lfv+jZJs+ykMLB6L/kaqk8tA4V2NHWydQ5h56o45ugs/580nMz36ZdRg==",
"cpu": [
"arm64"
],
"dev": true,
"license": "MIT",
"optional": true,
"os": [
"openbsd"
]
},
"node_modules/lefthook-openbsd-x64": {
"version": "1.13.5",
"resolved": "https://registry.npmjs.org/lefthook-openbsd-x64/-/lefthook-openbsd-x64-1.13.5.tgz",
"integrity": "sha512-Ba1JrsRbfan4WKd8Q7gUhTxCUuppXzirDObd3JxpLRSLxA47yxhjMv7KByDunRDTvzTgsXoykZI6mPupkc1JiQ==",
"cpu": [
"x64"
],
"dev": true,
"license": "MIT",
"optional": true,
"os": [
"openbsd"
]
},
"node_modules/lefthook-windows-arm64": {
"version": "1.13.5",
"resolved": "https://registry.npmjs.org/lefthook-windows-arm64/-/lefthook-windows-arm64-1.13.5.tgz",
"integrity": "sha512-Y/CpmEIb0hlFe+kTT/efWgX6+/gUTp5NItTF+gmUrY1/G/bTLIxdIRS7WpodVM0MEN24sOrQVTSi9DN9FvGoGg==",
"cpu": [
"arm64"
],
"dev": true,
"license": "MIT",
"optional": true,
"os": [
"win32"
]
},
"node_modules/lefthook-windows-x64": {
"version": "1.13.5",
"resolved": "https://registry.npmjs.org/lefthook-windows-x64/-/lefthook-windows-x64-1.13.5.tgz",
"integrity": "sha512-WJBqGNBlFJnunRwy12QyaDHdGULtostPqpYSZSS4boFJDY0lP5qtz9lAGmJ49aA5GQ19jrnDjGLwVPFiwIqksQ==",
"cpu": [
"x64"
],
"dev": true,
"license": "MIT",
"optional": true,
"os": [
"win32"
]
},
"node_modules/lilconfig": { "node_modules/lilconfig": {
"version": "3.1.3", "version": "3.1.3",
"dev": true, "dev": true,
@@ -3419,6 +3568,7 @@
} }
], ],
"license": "MIT", "license": "MIT",
"peer": true,
"dependencies": { "dependencies": {
"nanoid": "^3.3.11", "nanoid": "^3.3.11",
"picocolors": "^1.1.1", "picocolors": "^1.1.1",
@@ -3548,6 +3698,7 @@
"devOptional": true, "devOptional": true,
"hasInstallScript": true, "hasInstallScript": true,
"license": "Apache-2.0", "license": "Apache-2.0",
"peer": true,
"dependencies": { "dependencies": {
"@prisma/config": "6.16.2", "@prisma/config": "6.16.2",
"@prisma/engines": "6.16.2" "@prisma/engines": "6.16.2"
@@ -3737,6 +3888,7 @@
"node_modules/react": { "node_modules/react": {
"version": "18.3.1", "version": "18.3.1",
"license": "MIT", "license": "MIT",
"peer": true,
"dependencies": { "dependencies": {
"loose-envify": "^1.1.0" "loose-envify": "^1.1.0"
}, },
@@ -3755,6 +3907,7 @@
"node_modules/react-dom": { "node_modules/react-dom": {
"version": "18.3.1", "version": "18.3.1",
"license": "MIT", "license": "MIT",
"peer": true,
"dependencies": { "dependencies": {
"loose-envify": "^1.1.0", "loose-envify": "^1.1.0",
"scheduler": "^0.23.2" "scheduler": "^0.23.2"
@@ -3771,7 +3924,9 @@
} }
}, },
"node_modules/react-refresh": { "node_modules/react-refresh": {
"version": "0.17.0", "version": "0.18.0",
"resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.18.0.tgz",
"integrity": "sha512-QgT5//D3jfjJb6Gsjxv0Slpj23ip+HtOpnNgnb2S5zU3CB26G/IDPGoy4RJB42wzFE46DRsstbW6tKHoKbhAxw==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"engines": { "engines": {
@@ -4472,6 +4627,7 @@
"version": "4.0.3", "version": "4.0.3",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"peer": true,
"engines": { "engines": {
"node": ">=12" "node": ">=12"
}, },
@@ -4624,6 +4780,7 @@
"version": "7.1.7", "version": "7.1.7",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"peer": true,
"dependencies": { "dependencies": {
"esbuild": "^0.25.0", "esbuild": "^0.25.0",
"fdir": "^6.5.0", "fdir": "^6.5.0",
@@ -4713,6 +4870,7 @@
"version": "4.0.3", "version": "4.0.3",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"peer": true,
"engines": { "engines": {
"node": ">=12" "node": ">=12"
}, },

View File

@@ -1,6 +1,6 @@
{ {
"name": "patchmon", "name": "patchmon",
"version": "1.3.6", "version": "1.3.5",
"description": "Linux Patch Monitoring System", "description": "Linux Patch Monitoring System",
"license": "AGPL-3.0", "license": "AGPL-3.0",
"private": true, "private": true,