fix: resolve 6 reporting issues - SQL schema mismatches, duplicate type, UI dropdown, chart scale, CSV error handling
All checks were successful
CI Pipeline / Rust Format Check (push) Successful in 4s
CI Pipeline / Clippy Lints (push) Successful in 57s
CI Pipeline / Rust Unit Tests (push) Successful in 1m17s
CI Pipeline / Security Audit (push) Successful in 5s
CI Pipeline / Frontend Lint & Type Check (push) Successful in 15s
CI Pipeline / Build .deb & Release (push) Has been skipped
All checks were successful
CI Pipeline / Rust Format Check (push) Successful in 4s
CI Pipeline / Clippy Lints (push) Successful in 57s
CI Pipeline / Rust Unit Tests (push) Successful in 1m17s
CI Pipeline / Security Audit (push) Successful in 5s
CI Pipeline / Frontend Lint & Type Check (push) Successful in 15s
CI Pipeline / Build .deb & Release (push) Has been skipped
1. Compliance CSV/PDF: Replace non-existent pd.total_packages with jsonb_array_length(pd.installed_packages) and pd.pending_patches with pd.patch_count. Fix GROUP BY to match new columns. 2. Vulnerability CSV/PDF: Replace non-existent pd.cve_data with jsonb_array_elements on pd.available_patches JSONB, extracting cve_ids via nested lateral join. Replace pd.updated_at with pd.polled_at (actual column name). 3. TypeScript: Remove duplicate PollingConfig interface declaration in frontend/src/types/index.ts. 4. ReportsPage: Replace Group ID text field with Select dropdown populated from GET /api/v1/groups, showing group names instead of requiring UUID input. 5. PDF charts: Increase embed_image scale from 0.18 to 0.28 for better visibility on A4 landscape pages. 6. Vulnerability CSV: Remove invalid (no data) comment row on query failure; return header-only CSV instead to maintain valid CSV format.
This commit is contained in:
14
Cargo.lock
generated
14
Cargo.lock
generated
@ -2206,7 +2206,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "pm-agent-client"
|
name = "pm-agent-client"
|
||||||
version = "0.1.2"
|
version = "0.1.3"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"chrono",
|
"chrono",
|
||||||
@ -2223,7 +2223,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "pm-auth"
|
name = "pm-auth"
|
||||||
version = "0.1.2"
|
version = "0.1.3"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"argon2",
|
"argon2",
|
||||||
@ -2250,7 +2250,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "pm-ca"
|
name = "pm-ca"
|
||||||
version = "0.1.2"
|
version = "0.1.3"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"chrono",
|
"chrono",
|
||||||
@ -2273,7 +2273,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "pm-core"
|
name = "pm-core"
|
||||||
version = "0.1.2"
|
version = "0.1.3"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"aes-gcm",
|
"aes-gcm",
|
||||||
"anyhow",
|
"anyhow",
|
||||||
@ -2297,7 +2297,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "pm-reports"
|
name = "pm-reports"
|
||||||
version = "0.1.2"
|
version = "0.1.3"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"chrono",
|
"chrono",
|
||||||
@ -2318,7 +2318,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "pm-web"
|
name = "pm-web"
|
||||||
version = "0.1.2"
|
version = "0.1.3"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"axum",
|
"axum",
|
||||||
@ -2355,7 +2355,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "pm-worker"
|
name = "pm-worker"
|
||||||
version = "0.1.2"
|
version = "0.1.3"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"chrono",
|
"chrono",
|
||||||
|
|||||||
@ -27,10 +27,10 @@ SELECT
|
|||||||
h.fqdn,
|
h.fqdn,
|
||||||
h.health_status::text AS health_status,
|
h.health_status::text AS health_status,
|
||||||
h.last_patch_at,
|
h.last_patch_at,
|
||||||
COALESCE(pd.total_packages, 0) AS total_packages,
|
COALESCE(jsonb_array_length(pd.installed_packages), 0) AS total_packages,
|
||||||
COALESCE(pd.pending_patches, 0) AS pending_patches,
|
COALESCE(pd.patch_count, 0) AS pending_patches,
|
||||||
CASE WHEN COALESCE(pd.total_packages,0) = 0 THEN 100.0
|
CASE WHEN COALESCE(jsonb_array_length(pd.installed_packages), 0) = 0 THEN 100.0
|
||||||
ELSE ROUND((1.0 - pd.pending_patches::float / NULLIF(pd.total_packages,0)) * 100, 1)
|
ELSE ROUND((1.0 - pd.patch_count::float / NULLIF(jsonb_array_length(pd.installed_packages), 0)) * 100, 1)
|
||||||
END AS compliance_pct,
|
END AS compliance_pct,
|
||||||
COALESCE(string_agg(DISTINCT g.name, ', '), '') AS group_names
|
COALESCE(string_agg(DISTINCT g.name, ', '), '') AS group_names
|
||||||
FROM hosts h
|
FROM hosts h
|
||||||
@ -40,7 +40,7 @@ LEFT JOIN groups g ON g.id = hg.group_id
|
|||||||
WHERE h.id IN (
|
WHERE h.id IN (
|
||||||
SELECT host_id FROM host_groups WHERE group_id = $1
|
SELECT host_id FROM host_groups WHERE group_id = $1
|
||||||
)
|
)
|
||||||
GROUP BY h.id, pd.total_packages, pd.pending_patches
|
GROUP BY h.id, pd.installed_packages, pd.patch_count
|
||||||
ORDER BY compliance_pct ASC
|
ORDER BY compliance_pct ASC
|
||||||
",
|
",
|
||||||
)
|
)
|
||||||
@ -57,17 +57,17 @@ SELECT
|
|||||||
h.fqdn,
|
h.fqdn,
|
||||||
h.health_status::text AS health_status,
|
h.health_status::text AS health_status,
|
||||||
h.last_patch_at,
|
h.last_patch_at,
|
||||||
COALESCE(pd.total_packages, 0) AS total_packages,
|
COALESCE(jsonb_array_length(pd.installed_packages), 0) AS total_packages,
|
||||||
COALESCE(pd.pending_patches, 0) AS pending_patches,
|
COALESCE(pd.patch_count, 0) AS pending_patches,
|
||||||
CASE WHEN COALESCE(pd.total_packages,0) = 0 THEN 100.0
|
CASE WHEN COALESCE(jsonb_array_length(pd.installed_packages), 0) = 0 THEN 100.0
|
||||||
ELSE ROUND((1.0 - pd.pending_patches::float / NULLIF(pd.total_packages,0)) * 100, 1)
|
ELSE ROUND((1.0 - pd.patch_count::float / NULLIF(jsonb_array_length(pd.installed_packages), 0)) * 100, 1)
|
||||||
END AS compliance_pct,
|
END AS compliance_pct,
|
||||||
COALESCE(string_agg(DISTINCT g.name, ', '), '') AS group_names
|
COALESCE(string_agg(DISTINCT g.name, ', '), '') AS group_names
|
||||||
FROM hosts h
|
FROM hosts h
|
||||||
LEFT JOIN host_patch_data pd ON pd.host_id = h.id
|
LEFT JOIN host_patch_data pd ON pd.host_id = h.id
|
||||||
LEFT JOIN host_groups hg ON hg.host_id = h.id
|
LEFT JOIN host_groups hg ON hg.host_id = h.id
|
||||||
LEFT JOIN groups g ON g.id = hg.group_id
|
LEFT JOIN groups g ON g.id = hg.group_id
|
||||||
GROUP BY h.id, pd.total_packages, pd.pending_patches
|
GROUP BY h.id, pd.installed_packages, pd.patch_count
|
||||||
ORDER BY compliance_pct ASC
|
ORDER BY compliance_pct ASC
|
||||||
",
|
",
|
||||||
)
|
)
|
||||||
@ -220,19 +220,19 @@ SELECT
|
|||||||
h.id::text AS host_id,
|
h.id::text AS host_id,
|
||||||
h.display_name,
|
h.display_name,
|
||||||
h.fqdn,
|
h.fqdn,
|
||||||
cve.cve_id,
|
cve_id,
|
||||||
cve.package_name,
|
patch->>'name' AS package_name,
|
||||||
cve.severity,
|
patch->>'severity' AS severity,
|
||||||
cve.available_version,
|
patch->>'available_version' AS available_version,
|
||||||
pd.updated_at AS last_seen_at
|
pd.polled_at AS last_seen_at
|
||||||
FROM hosts h
|
FROM hosts h
|
||||||
JOIN host_patch_data pd ON pd.host_id = h.id
|
JOIN host_patch_data pd ON pd.host_id = h.id
|
||||||
CROSS JOIN LATERAL jsonb_to_recordset(COALESCE(pd.cve_data, '[]'::jsonb))
|
CROSS JOIN LATERAL jsonb_array_elements(COALESCE(pd.available_patches, '[]'::jsonb)) AS patch
|
||||||
AS cve(cve_id text, package_name text, severity text, available_version text)
|
CROSS JOIN LATERAL jsonb_array_elements_text(COALESCE(patch->'cve_ids', '[]'::jsonb)) AS cve_id
|
||||||
WHERE ($1::timestamptz IS NULL OR pd.updated_at >= $1)
|
WHERE ($1::timestamptz IS NULL OR pd.polled_at >= $1)
|
||||||
AND ($2::timestamptz IS NULL OR pd.updated_at <= $2)
|
AND ($2::timestamptz IS NULL OR pd.polled_at <= $2)
|
||||||
ORDER BY
|
ORDER BY
|
||||||
CASE cve.severity
|
CASE patch->>'severity'
|
||||||
WHEN 'critical' THEN 1
|
WHEN 'critical' THEN 1
|
||||||
WHEN 'high' THEN 2
|
WHEN 'high' THEN 2
|
||||||
WHEN 'medium' THEN 3
|
WHEN 'medium' THEN 3
|
||||||
@ -274,18 +274,8 @@ ORDER BY
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
tracing::warn!(error = %e, "vulnerability query failed — returning empty rows");
|
tracing::warn!(error = %e, "vulnerability query failed — returning header-only CSV");
|
||||||
// write a comment row indicating empty data
|
// Return header-only CSV (no invalid comment rows)
|
||||||
wtr.write_record(&[
|
|
||||||
"(no data)",
|
|
||||||
"",
|
|
||||||
"",
|
|
||||||
"",
|
|
||||||
"",
|
|
||||||
"",
|
|
||||||
"",
|
|
||||||
&format!("query error: {}", e),
|
|
||||||
])?;
|
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -250,15 +250,15 @@ async fn compliance_pdf(pool: &sqlx::PgPool, params: &ReportParams) -> anyhow::R
|
|||||||
sqlx::query(
|
sqlx::query(
|
||||||
"
|
"
|
||||||
SELECT h.display_name, h.fqdn,
|
SELECT h.display_name, h.fqdn,
|
||||||
COALESCE(pd.total_packages,0) AS total_packages,
|
COALESCE(jsonb_array_length(pd.installed_packages),0) AS total_packages,
|
||||||
COALESCE(pd.pending_patches,0) AS pending_patches,
|
COALESCE(pd.patch_count,0) AS pending_patches,
|
||||||
CASE WHEN COALESCE(pd.total_packages,0)=0 THEN 100.0
|
CASE WHEN COALESCE(jsonb_array_length(pd.installed_packages),0)=0 THEN 100.0
|
||||||
ELSE ROUND((1.0-pd.pending_patches::float/NULLIF(pd.total_packages,0))*100,1)
|
ELSE ROUND((1.0-pd.patch_count::float/NULLIF(jsonb_array_length(pd.installed_packages),0))*100,1)
|
||||||
END AS compliance_pct,
|
END AS compliance_pct,
|
||||||
h.health_status::text AS health_status
|
h.health_status::text AS health_status
|
||||||
FROM hosts h LEFT JOIN host_patch_data pd ON pd.host_id=h.id
|
FROM hosts h LEFT JOIN host_patch_data pd ON pd.host_id=h.id
|
||||||
WHERE h.id IN (SELECT host_id FROM host_groups WHERE group_id=$1)
|
WHERE h.id IN (SELECT host_id FROM host_groups WHERE group_id=$1)
|
||||||
GROUP BY h.id,pd.total_packages,pd.pending_patches
|
GROUP BY h.id, pd.installed_packages, pd.patch_count
|
||||||
ORDER BY compliance_pct ASC",
|
ORDER BY compliance_pct ASC",
|
||||||
)
|
)
|
||||||
.bind(gid)
|
.bind(gid)
|
||||||
@ -269,14 +269,14 @@ ORDER BY compliance_pct ASC",
|
|||||||
sqlx::query(
|
sqlx::query(
|
||||||
"
|
"
|
||||||
SELECT h.display_name, h.fqdn,
|
SELECT h.display_name, h.fqdn,
|
||||||
COALESCE(pd.total_packages,0) AS total_packages,
|
COALESCE(jsonb_array_length(pd.installed_packages),0) AS total_packages,
|
||||||
COALESCE(pd.pending_patches,0) AS pending_patches,
|
COALESCE(pd.patch_count,0) AS pending_patches,
|
||||||
CASE WHEN COALESCE(pd.total_packages,0)=0 THEN 100.0
|
CASE WHEN COALESCE(jsonb_array_length(pd.installed_packages),0)=0 THEN 100.0
|
||||||
ELSE ROUND((1.0-pd.pending_patches::float/NULLIF(pd.total_packages,0))*100,1)
|
ELSE ROUND((1.0-pd.patch_count::float/NULLIF(jsonb_array_length(pd.installed_packages),0))*100,1)
|
||||||
END AS compliance_pct,
|
END AS compliance_pct,
|
||||||
h.health_status::text AS health_status
|
h.health_status::text AS health_status
|
||||||
FROM hosts h LEFT JOIN host_patch_data pd ON pd.host_id=h.id
|
FROM hosts h LEFT JOIN host_patch_data pd ON pd.host_id=h.id
|
||||||
GROUP BY h.id,pd.total_packages,pd.pending_patches
|
GROUP BY h.id, pd.installed_packages, pd.patch_count
|
||||||
ORDER BY compliance_pct ASC",
|
ORDER BY compliance_pct ASC",
|
||||||
)
|
)
|
||||||
.fetch_all(pool)
|
.fetch_all(pool)
|
||||||
@ -333,7 +333,7 @@ ORDER BY compliance_pct ASC",
|
|||||||
Ok((raw, w, h)) => {
|
Ok((raw, w, h)) => {
|
||||||
pdf.new_page();
|
pdf.new_page();
|
||||||
pdf.write_text("Compliance Chart", 16.0, MARGIN, 200.0, true);
|
pdf.write_text("Compliance Chart", 16.0, MARGIN, 200.0, true);
|
||||||
if let Err(e) = pdf.embed_image(raw, w, h, MARGIN, 10.0, 0.18, 0.18) {
|
if let Err(e) = pdf.embed_image(raw, w, h, MARGIN, 10.0, 0.28, 0.28) {
|
||||||
tracing::warn!(error = %e, "chart embed failed");
|
tracing::warn!(error = %e, "chart embed failed");
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -432,7 +432,7 @@ ORDER BY pjh.started_at DESC",
|
|||||||
Ok((raw, w, h)) => {
|
Ok((raw, w, h)) => {
|
||||||
pdf.new_page();
|
pdf.new_page();
|
||||||
pdf.write_text("Patch Activity Chart", 16.0, MARGIN, 200.0, true);
|
pdf.write_text("Patch Activity Chart", 16.0, MARGIN, 200.0, true);
|
||||||
if let Err(e) = pdf.embed_image(raw, w, h, MARGIN, 10.0, 0.18, 0.18) {
|
if let Err(e) = pdf.embed_image(raw, w, h, MARGIN, 10.0, 0.28, 0.28) {
|
||||||
tracing::warn!(error = %e, "chart embed failed");
|
tracing::warn!(error = %e, "chart embed failed");
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -451,14 +451,17 @@ async fn vulnerability_pdf(pool: &sqlx::PgPool, params: &ReportParams) -> anyhow
|
|||||||
// Query DB FIRST (before creating any non-Send PdfBuilder)
|
// Query DB FIRST (before creating any non-Send PdfBuilder)
|
||||||
let query_result = sqlx::query("
|
let query_result = sqlx::query("
|
||||||
SELECT h.display_name, h.fqdn,
|
SELECT h.display_name, h.fqdn,
|
||||||
cve.cve_id, cve.package_name, cve.severity, cve.available_version,
|
cve_id,
|
||||||
pd.updated_at AS last_seen_at
|
patch->>'name' AS package_name,
|
||||||
|
patch->>'severity' AS severity,
|
||||||
|
patch->>'available_version' AS available_version,
|
||||||
|
pd.polled_at AS last_seen_at
|
||||||
FROM hosts h JOIN host_patch_data pd ON pd.host_id=h.id
|
FROM hosts h JOIN host_patch_data pd ON pd.host_id=h.id
|
||||||
CROSS JOIN LATERAL jsonb_to_recordset(COALESCE(pd.cve_data,'[]'::jsonb))
|
CROSS JOIN LATERAL jsonb_array_elements(COALESCE(pd.available_patches,'[]'::jsonb)) AS patch
|
||||||
AS cve(cve_id text,package_name text,severity text,available_version text)
|
CROSS JOIN LATERAL jsonb_array_elements_text(COALESCE(patch->'cve_ids','[]'::jsonb)) AS cve_id
|
||||||
WHERE ($1::timestamptz IS NULL OR pd.updated_at>=$1)
|
WHERE ($1::timestamptz IS NULL OR pd.polled_at>=$1)
|
||||||
AND ($2::timestamptz IS NULL OR pd.updated_at<=$2)
|
AND ($2::timestamptz IS NULL OR pd.polled_at<=$2)
|
||||||
ORDER BY CASE cve.severity WHEN 'critical' THEN 1 WHEN 'high' THEN 2 WHEN 'medium' THEN 3 ELSE 4 END,
|
ORDER BY CASE patch->>'severity' WHEN 'critical' THEN 1 WHEN 'high' THEN 2 WHEN 'medium' THEN 3 ELSE 4 END,
|
||||||
h.display_name")
|
h.display_name")
|
||||||
.bind(params.from).bind(params.to).fetch_all(pool).await;
|
.bind(params.from).bind(params.to).fetch_all(pool).await;
|
||||||
// Now create PdfBuilder (non-Send Rc types) after all awaits
|
// Now create PdfBuilder (non-Send Rc types) after all awaits
|
||||||
|
|||||||
@ -1,4 +1,4 @@
|
|||||||
import { useState } from 'react'
|
import { useEffect, useState } from 'react'
|
||||||
import {
|
import {
|
||||||
Alert,
|
Alert,
|
||||||
Box,
|
Box,
|
||||||
@ -22,8 +22,8 @@ import {
|
|||||||
import DescriptionIcon from '@mui/icons-material/Description'
|
import DescriptionIcon from '@mui/icons-material/Description'
|
||||||
import PictureAsPdfIcon from '@mui/icons-material/PictureAsPdf'
|
import PictureAsPdfIcon from '@mui/icons-material/PictureAsPdf'
|
||||||
import VerifiedUserIcon from '@mui/icons-material/VerifiedUser'
|
import VerifiedUserIcon from '@mui/icons-material/VerifiedUser'
|
||||||
import { reportsApi, settingsApi } from '../api/client'
|
import { reportsApi, settingsApi, apiClient } from '../api/client'
|
||||||
import type { ReportType, ReportFormat, AuditIntegrityResult } from '../types'
|
import type { ReportType, ReportFormat, AuditIntegrityResult, Group } from '../types'
|
||||||
|
|
||||||
// ── Report metadata ───────────────────────────────────────────────────────────
|
// ── Report metadata ───────────────────────────────────────────────────────────
|
||||||
|
|
||||||
@ -97,11 +97,20 @@ export default function ReportsPage() {
|
|||||||
const [fromDate, setFromDate] = useState<string>(defaultFromDate())
|
const [fromDate, setFromDate] = useState<string>(defaultFromDate())
|
||||||
const [toDate, setToDate] = useState<string>(defaultToDate())
|
const [toDate, setToDate] = useState<string>(defaultToDate())
|
||||||
const [groupId, setGroupId] = useState<string>('')
|
const [groupId, setGroupId] = useState<string>('')
|
||||||
|
const [groups, setGroups] = useState<Group[]>([])
|
||||||
const [downloading, setDownloading] = useState(false)
|
const [downloading, setDownloading] = useState(false)
|
||||||
const [error, setError] = useState<string | null>(null)
|
const [error, setError] = useState<string | null>(null)
|
||||||
const [verifyingIntegrity, setVerifyingIntegrity] = useState(false)
|
const [verifyingIntegrity, setVerifyingIntegrity] = useState(false)
|
||||||
const [integrityResult, setIntegrityResult] = useState<AuditIntegrityResult | null>(null)
|
const [integrityResult, setIntegrityResult] = useState<AuditIntegrityResult | null>(null)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
apiClient.get<Group[]>('/groups').then((res) => {
|
||||||
|
setGroups(res.data)
|
||||||
|
}).catch(() => {
|
||||||
|
// Groups fetch is optional; silently ignore errors
|
||||||
|
})
|
||||||
|
}, [])
|
||||||
|
|
||||||
const info = REPORT_INFO[reportType]
|
const info = REPORT_INFO[reportType]
|
||||||
|
|
||||||
const handleDownload = async (format: ReportFormat) => {
|
const handleDownload = async (format: ReportFormat) => {
|
||||||
@ -111,7 +120,7 @@ export default function ReportsPage() {
|
|||||||
const params: Record<string, string> = {}
|
const params: Record<string, string> = {}
|
||||||
if (fromDate) params.from = new Date(fromDate).toISOString()
|
if (fromDate) params.from = new Date(fromDate).toISOString()
|
||||||
if (toDate) params.to = new Date(toDate + 'T23:59:59Z').toISOString()
|
if (toDate) params.to = new Date(toDate + 'T23:59:59Z').toISOString()
|
||||||
if (reportType === 'compliance' && groupId.trim()) params.group_id = groupId.trim()
|
if (reportType === 'compliance' && groupId) params.group_id = groupId
|
||||||
|
|
||||||
const res = await reportsApi.download(reportType, format, params)
|
const res = await reportsApi.download(reportType, format, params)
|
||||||
|
|
||||||
@ -203,13 +212,21 @@ export default function ReportsPage() {
|
|||||||
{/* Group Filter — compliance only */}
|
{/* Group Filter — compliance only */}
|
||||||
{reportType === 'compliance' && (
|
{reportType === 'compliance' && (
|
||||||
<FormControl fullWidth sx={{ mb: 2 }}>
|
<FormControl fullWidth sx={{ mb: 2 }}>
|
||||||
<TextField
|
<InputLabel id="group-filter-label">Group (optional)</InputLabel>
|
||||||
label="Group ID (optional)"
|
<Select
|
||||||
|
labelId="group-filter-label"
|
||||||
value={groupId}
|
value={groupId}
|
||||||
|
label="Group (optional)"
|
||||||
onChange={(e) => setGroupId(e.target.value)}
|
onChange={(e) => setGroupId(e.target.value)}
|
||||||
placeholder="e.g. 550e8400-e29b-41d4-a716-446655440000"
|
>
|
||||||
/>
|
<MenuItem value="">All Groups</MenuItem>
|
||||||
<FormHelperText>Filter compliance report by a specific group UUID</FormHelperText>
|
{groups.map((g) => (
|
||||||
|
<MenuItem key={g.id} value={g.id}>
|
||||||
|
{g.name}
|
||||||
|
</MenuItem>
|
||||||
|
))}
|
||||||
|
</Select>
|
||||||
|
<FormHelperText>Filter compliance report by a specific group</FormHelperText>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
|||||||
@ -261,10 +261,6 @@ export interface SmtpConfig {
|
|||||||
tls_mode: string
|
tls_mode: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface PollingConfig {
|
|
||||||
health_poll_interval_secs: number
|
|
||||||
patch_poll_interval_secs: number
|
|
||||||
}
|
|
||||||
export interface PollingConfig {
|
export interface PollingConfig {
|
||||||
health_poll_interval_secs: number
|
health_poll_interval_secs: number
|
||||||
patch_poll_interval_secs: number
|
patch_poll_interval_secs: number
|
||||||
|
|||||||
Reference in New Issue
Block a user