-
Notifications
You must be signed in to change notification settings - Fork 10
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(hostd): upgraded alerts feature
- Loading branch information
1 parent
dc08559
commit b05a244
Showing
37 changed files
with
1,308 additions
and
667 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
--- | ||
'hostd': minor | ||
--- | ||
|
||
The hostd alerts feature is now a full page and matches the user experience of renterd alerts. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
import { Page } from '@playwright/test' | ||
import { maybeExpectAndReturn, step } from '@siafoundation/e2e' | ||
|
||
export const getAlertRows = (page: Page) => { | ||
return page.getByTestId('alertsTable').locator('tbody').getByRole('row') | ||
} | ||
|
||
export const getAlertRowsAll = step('get alert rows', async (page: Page) => { | ||
return getAlertRows(page).all() | ||
}) | ||
|
||
export const getAlertRowByIndex = step( | ||
'get alert row by index', | ||
async (page: Page, index: number, shouldExpect?: boolean) => { | ||
return maybeExpectAndReturn( | ||
page | ||
.getByTestId('alertsTable') | ||
.locator('tbody') | ||
.getByRole('row') | ||
.nth(index), | ||
shouldExpect | ||
) | ||
} | ||
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,84 @@ | ||
import { test, expect, Page } from '@playwright/test' | ||
import { navigateToAlerts, navigateToVolumes } from '../fixtures/navigate' | ||
import { afterTest, beforeTest } from '../fixtures/beforeTest' | ||
import { Alert } from '@siafoundation/hostd-types' | ||
import { getAlertRows } from '../fixtures/alerts' | ||
import { createVolume, deleteVolume } from '../fixtures/volumes' | ||
import fs from 'fs' | ||
import os from 'os' | ||
import { clickAndWait, continueToClickUntil } from '@siafoundation/e2e' | ||
|
||
let dirPath = '/' | ||
|
||
test.beforeEach(async ({ page }) => { | ||
await beforeTest(page) | ||
// Create a temporary directory. | ||
dirPath = fs.mkdtempSync(process.env.GITHUB_WORKSPACE || os.tmpdir()) | ||
}) | ||
|
||
test.afterEach(async () => { | ||
await afterTest() | ||
fs.rmSync(dirPath, { recursive: true }) | ||
}) | ||
|
||
test('filtering alerts', async ({ page }) => { | ||
await mockApiAlerts(page) | ||
await page.reload() | ||
|
||
await navigateToAlerts(page) | ||
|
||
// Check initial number of alerts. | ||
await expect(getAlertRows(page)).toHaveCount(2) | ||
|
||
// Verify alert content. | ||
await expect(page.getByText('Volume initialized')).toBeVisible() | ||
await expect(page.getByText('Volume warning')).toBeVisible() | ||
|
||
// Test filtering. | ||
await page.getByRole('button', { name: 'Info' }).click() | ||
await expect(getAlertRows(page)).toHaveCount(1) | ||
}) | ||
|
||
test('dismissing alerts', async ({ page }) => { | ||
const name = 'my-new-volume' | ||
await navigateToVolumes({ page }) | ||
await createVolume(page, name, dirPath) | ||
await navigateToAlerts(page) | ||
await expect(getAlertRows(page).getByText('Volume initialized')).toBeVisible() | ||
// Dismissing the alert too early will cause the alert to reappear | ||
// so maybe try to dismiss more than once. | ||
await continueToClickUntil( | ||
page.getByRole('button', { name: 'dismiss alert' }), | ||
page.getByText('There are currently no alerts.') | ||
) | ||
}) | ||
|
||
async function mockApiAlerts(page: Page) { | ||
const alerts: Alert[] = [ | ||
{ | ||
id: 'c39d09ee61a5d1dd9ad97015a0e87e9286f765bbf109cafad936d5a1aa843e54', | ||
severity: 'info', | ||
message: 'Volume initialized', | ||
data: { | ||
elapsed: 95823333, | ||
target: 2623, | ||
volumeID: 3, | ||
}, | ||
timestamp: '2025-01-10T10:17:52.365323-05:00', | ||
}, | ||
{ | ||
id: '93683b58d12c2de737a8849561b9f0dae07120eee3185f40da489d48585b416a', | ||
severity: 'warning', | ||
message: 'Volume warning', | ||
data: { | ||
elapsed: 257749500, | ||
targetSectors: 7868, | ||
volumeID: 2, | ||
}, | ||
timestamp: '2025-01-10T10:17:18.754568-05:00', | ||
}, | ||
] | ||
await page.route('**/api/alerts', async (route) => { | ||
await route.fulfill({ json: alerts }) | ||
}) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,54 @@ | ||
import { | ||
DropdownMenu, | ||
DropdownMenuItem, | ||
Button, | ||
DropdownMenuLeftSlot, | ||
DropdownMenuLabel, | ||
Text, | ||
} from '@siafoundation/design-system' | ||
import { CaretDown16, Checkmark16 } from '@siafoundation/react-icons' | ||
import { useAlerts } from '../../contexts/alerts' | ||
|
||
type Props = { | ||
id: string | ||
contentProps?: React.ComponentProps<typeof DropdownMenu>['contentProps'] | ||
buttonProps?: React.ComponentProps<typeof Button> | ||
} | ||
|
||
export function AlertContextMenu({ id, contentProps, buttonProps }: Props) { | ||
const { dismissOne } = useAlerts() | ||
|
||
return ( | ||
<DropdownMenu | ||
trigger={ | ||
<Button variant="ghost" icon="hover" {...buttonProps}> | ||
<CaretDown16 /> | ||
</Button> | ||
} | ||
contentProps={{ | ||
align: 'start', | ||
...contentProps, | ||
onClick: (e) => { | ||
e.stopPropagation() | ||
}, | ||
}} | ||
> | ||
<div className="px-1.5 py-1"> | ||
<Text size="14" weight="medium" color="subtle"> | ||
Alert {id.slice(0, 24)}... | ||
</Text> | ||
</div> | ||
<DropdownMenuLabel>Actions</DropdownMenuLabel> | ||
<DropdownMenuItem | ||
onSelect={() => { | ||
dismissOne(id) | ||
}} | ||
> | ||
<DropdownMenuLeftSlot> | ||
<Checkmark16 /> | ||
</DropdownMenuLeftSlot> | ||
Clear alert | ||
</DropdownMenuItem> | ||
</DropdownMenu> | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
import { AlertsViewDropdownMenu } from './AlertsViewDropdownMenu' | ||
|
||
export function AlertsActionsMenu() { | ||
return ( | ||
<div className="flex gap-2"> | ||
<AlertsViewDropdownMenu /> | ||
</div> | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
import { | ||
CommandGroup, | ||
CommandItemNav, | ||
CommandItemSearch, | ||
} from '../../CmdRoot/Item' | ||
import { Page } from '../../CmdRoot/types' | ||
import { useRouter } from 'next/router' | ||
import { useDialog } from '../../../contexts/dialog' | ||
import { routes } from '../../../config/routes' | ||
|
||
export const commandPage = { | ||
namespace: 'alerts', | ||
label: 'Alerts', | ||
} | ||
|
||
export function AlertsCmd({ | ||
currentPage, | ||
parentPage, | ||
pushPage, | ||
}: { | ||
currentPage: Page | ||
parentPage?: Page | ||
beforeSelect?: () => void | ||
afterSelect?: () => void | ||
pushPage: (page: Page) => void | ||
}) { | ||
const router = useRouter() | ||
const { closeDialog } = useDialog() | ||
return ( | ||
<> | ||
<CommandItemNav | ||
currentPage={currentPage} | ||
parentPage={parentPage} | ||
commandPage={parentPage} | ||
onSelect={() => { | ||
pushPage(commandPage) | ||
}} | ||
> | ||
{commandPage.label} | ||
</CommandItemNav> | ||
<CommandGroup currentPage={currentPage} commandPage={commandPage}> | ||
<CommandItemSearch | ||
currentPage={currentPage} | ||
commandPage={commandPage} | ||
onSelect={() => { | ||
router.push(routes.alerts.index) | ||
closeDialog() | ||
}} | ||
> | ||
View alerts | ||
</CommandItemSearch> | ||
</CommandGroup> | ||
</> | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,96 @@ | ||
'use client' | ||
|
||
import { Button, PaginatorKnownTotal, Text } from '@siafoundation/design-system' | ||
import { useAlerts } from '../../contexts/alerts' | ||
import { Checkmark16 } from '@siafoundation/react-icons' | ||
import { useCallback } from 'react' | ||
import { AlertSeverity } from '@siafoundation/hostd-types' | ||
|
||
export function AlertsFilterMenu() { | ||
const { | ||
offset, | ||
limit, | ||
totals, | ||
datasetPageTotal, | ||
datasetState, | ||
datasetPage, | ||
dismissMany, | ||
clientFilters, | ||
} = useAlerts() | ||
|
||
const severityFilter = clientFilters.filters.find((f) => f.id === 'severity') | ||
const setSeverityFilter = useCallback( | ||
(severity: AlertSeverity) => { | ||
if (severity === undefined) { | ||
clientFilters.removeFilter('severity') | ||
return | ||
} | ||
clientFilters.setFilter({ | ||
id: 'severity', | ||
value: severity, | ||
label: severity, | ||
fn: (a) => a.severity === severity, | ||
}) | ||
}, | ||
[clientFilters] | ||
) | ||
|
||
return ( | ||
<div className="flex gap-2 w-full items-center"> | ||
<Text weight="medium">Filter</Text> | ||
<div className="flex gap-1 items-center"> | ||
<Button | ||
variant={severityFilter === undefined ? 'active' : 'inactive'} | ||
onClick={() => setSeverityFilter(undefined)} | ||
> | ||
all ({totals.all}) | ||
</Button> | ||
<Button | ||
variant={severityFilter?.value === 'info' ? 'active' : 'inactive'} | ||
onClick={() => setSeverityFilter('info')} | ||
> | ||
info ({totals.info}) | ||
</Button> | ||
<Button | ||
variant={severityFilter?.value === 'warning' ? 'active' : 'inactive'} | ||
onClick={() => setSeverityFilter('warning')} | ||
> | ||
warning ({totals.warning}) | ||
</Button> | ||
<Button | ||
variant={severityFilter?.value === 'error' ? 'active' : 'inactive'} | ||
onClick={() => setSeverityFilter('error')} | ||
> | ||
error ({totals.error}) | ||
</Button> | ||
<Button | ||
variant={severityFilter?.value === 'critical' ? 'active' : 'inactive'} | ||
onClick={() => setSeverityFilter('critical')} | ||
> | ||
critical ({totals.critical}) | ||
</Button> | ||
</div> | ||
<div className="flex-1" /> | ||
{datasetState === 'loaded' && !!datasetPageTotal && ( | ||
<Button | ||
tip={severityFilter ? `dismiss ${datasetPageTotal}` : 'dismiss all'} | ||
onClick={() => { | ||
if (!datasetPage) { | ||
return | ||
} | ||
dismissMany(datasetPage.map((a) => a.id)) | ||
}} | ||
> | ||
<Checkmark16 /> | ||
Dismiss ({datasetPageTotal}) | ||
</Button> | ||
)} | ||
<PaginatorKnownTotal | ||
offset={offset} | ||
limit={limit} | ||
isLoading={datasetState === 'loading'} | ||
total={totals.all} | ||
/> | ||
</div> | ||
) | ||
} |
Oops, something went wrong.