-
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
6ba7122
commit 28dbaae
Showing
41 changed files
with
1,318 additions
and
665 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 | ||
--- | ||
|
||
Alerts can now be accessed via the cmd+k menu. |
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 @@ | ||
--- | ||
'@siafoundation/design-system': minor | ||
--- | ||
|
||
Removed AlertsDialog. |
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 @@ | ||
--- | ||
'@siafoundation/hostd-types': minor | ||
--- | ||
|
||
Added named AlertData type. |
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 @@ | ||
--- | ||
'@siafoundation/e2e': minor | ||
--- | ||
|
||
Added continueToClickUntil. |
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,90 @@ | ||
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 { 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() | ||
try { | ||
fs.rmSync(dirPath, { recursive: true }) | ||
} catch (e) { | ||
console.error(e) | ||
} | ||
}) | ||
|
||
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.') | ||
) | ||
await navigateToVolumes({ page }) | ||
await deleteVolume(page, name, dirPath) | ||
}) | ||
|
||
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> | ||
</> | ||
) | ||
} |
Oops, something went wrong.