From 1aa1350b98a00bb73618fbdb89879faf561f077c Mon Sep 17 00:00:00 2001 From: Wes Date: Tue, 3 Sep 2024 14:33:44 -0700 Subject: [PATCH] feat: add infrastructure tabs and content (#2588) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Fixes #2552 ![Screenshot 2024-09-03 at 1 44 20 PM](https://github.com/user-attachments/assets/44f17685-5ee8-4a26-b2f5-b0bfa210a642) ![Screenshot 2024-09-03 at 1 44 27 PM](https://github.com/user-attachments/assets/82f5aa35-cfb7-4718-9670-52f665ce6cc3) ![Screenshot 2024-09-03 at 1 44 30 PM](https://github.com/user-attachments/assets/90c07eeb-daca-416f-87a2-ed49a01e85bc) ![Screenshot 2024-09-03 at 1 44 35 PM](https://github.com/user-attachments/assets/0542b7c0-7dd7-45cf-b9c7-bf2539e90ff3) --- frontend/.gitignore | 3 ++ frontend/console/e2e/infrastructure.spec.ts | 2 +- frontend/console/src/components/List.tsx | 24 +++++++++ .../src/components/StatusIndicator.tsx | 24 +++++++++ .../infrastructure/ControllersList.tsx | 30 ++++++++++- .../infrastructure/DeploymentsList.tsx | 39 ++++++++++++++- .../infrastructure/InfrastructurePage.tsx | 24 +++++---- .../features/infrastructure/RoutesList.tsx | 28 ++++++++++- .../features/infrastructure/RunnersList.tsx | 50 ++++++++++++++++++- .../infrastructure/infrastructure.utils.ts | 18 +++++++ .../src/providers/routing-provider.tsx | 3 +- 11 files changed, 224 insertions(+), 21 deletions(-) create mode 100644 frontend/console/src/components/List.tsx create mode 100644 frontend/console/src/components/StatusIndicator.tsx create mode 100644 frontend/console/src/features/infrastructure/infrastructure.utils.ts diff --git a/frontend/.gitignore b/frontend/.gitignore index 8379000fd8..15a1e99fc3 100644 --- a/frontend/.gitignore +++ b/frontend/.gitignore @@ -7,6 +7,9 @@ yarn-error.log* pnpm-debug.log* lerna-debug.log* +playwright-report/ +test-results/ + package-lock.json node_modules dist diff --git a/frontend/console/e2e/infrastructure.spec.ts b/frontend/console/e2e/infrastructure.spec.ts index 8d8ca45bff..38cb2f64e2 100644 --- a/frontend/console/e2e/infrastructure.spec.ts +++ b/frontend/console/e2e/infrastructure.spec.ts @@ -3,7 +3,7 @@ import { expect, ftlTest } from './ftl-test' ftlTest('shows infrastructure', async ({ page }) => { const infrastructureNavItem = page.getByRole('link', { name: 'Infrastructure' }) await infrastructureNavItem.click() - await expect(page).toHaveURL(/\/infrastructure$/) + await expect(page).toHaveURL(/\/infrastructure\/controllers$/) const controllersTab = await page.getByRole('button', { name: 'Controllers' }) await expect(controllersTab).toBeVisible() diff --git a/frontend/console/src/components/List.tsx b/frontend/console/src/components/List.tsx new file mode 100644 index 0000000000..8733d182f6 --- /dev/null +++ b/frontend/console/src/components/List.tsx @@ -0,0 +1,24 @@ +import { classNames } from '../utils' + +type ListProps = { + items: T[] + renderItem: (item: T) => React.ReactNode + onClick?: (item: T) => void + className?: string +} + +export const List = ({ items, renderItem, onClick, className }: ListProps) => { + return ( +
    + {items.map((item, index) => ( +
  • onClick(item) : undefined} + > + {renderItem(item)} +
  • + ))} +
+ ) +} diff --git a/frontend/console/src/components/StatusIndicator.tsx b/frontend/console/src/components/StatusIndicator.tsx new file mode 100644 index 0000000000..7d7325a685 --- /dev/null +++ b/frontend/console/src/components/StatusIndicator.tsx @@ -0,0 +1,24 @@ +import type React from 'react' +import { classNames } from '../utils' + +type StatusIndicatorProps = { + state: 'success' | 'error' | 'idle' + text?: string +} + +export const StatusIndicator: React.FC = ({ state, text }) => { + const backgrounds = { + idle: 'text-gray-500 bg-gray-500/20 dark:bg-gray-100/10', + success: 'text-green-500 bg-emerald-500/20 dark:text-green-400 dark:bg-green-400/10 ', + error: 'text-rose-500 bg-rose-500/20 dark:text-rose-400 dark:bg-rose-400/10', + } + + return ( +
+
+
+
+ {text &&

{text}

} +
+ ) +} diff --git a/frontend/console/src/features/infrastructure/ControllersList.tsx b/frontend/console/src/features/infrastructure/ControllersList.tsx index 299186833f..a071f558d1 100644 --- a/frontend/console/src/features/infrastructure/ControllersList.tsx +++ b/frontend/console/src/features/infrastructure/ControllersList.tsx @@ -1,3 +1,29 @@ -export const ControllersList = () => { - return <>Controllers Content +import { Badge } from '../../components/Badge' +import { List } from '../../components/List' +import type { StatusResponse_Controller } from '../../protos/xyz/block/ftl/v1/ftl_pb' + +export const ControllersList = ({ controllers }: { controllers: StatusResponse_Controller[] }) => { + return ( + ( + <> +
+
+

+ + {controller.key} +

+

{controller.endpoint}

+
+
+
+
+ +
+
+ + )} + /> + ) } diff --git a/frontend/console/src/features/infrastructure/DeploymentsList.tsx b/frontend/console/src/features/infrastructure/DeploymentsList.tsx index cd0e9296e0..2545e9ff38 100644 --- a/frontend/console/src/features/infrastructure/DeploymentsList.tsx +++ b/frontend/console/src/features/infrastructure/DeploymentsList.tsx @@ -1,3 +1,38 @@ -export const DeploymentsList = () => { - return <>Deployments Content +import { AttributeBadge } from '../../components' +import { Badge } from '../../components/Badge' +import { List } from '../../components/List' +import type { StatusResponse_Deployment } from '../../protos/xyz/block/ftl/v1/ftl_pb' +import { classNames } from '../../utils' +import { deploymentTextColor } from '../deployments/deployment.utils' +import { renderValue } from './infrastructure.utils' + +export const DeploymentsList = ({ deployments }: { deployments: StatusResponse_Deployment[] }) => { + return ( + ( +
+
+
+
+

{deployment.name}

+ +
+ +

{deployment.key}

+
+
+
+
+ + + {Object.entries(deployment.labels?.fields || {}).map(([key, value]) => ( + + ))} +
+
+
+ )} + /> + ) } diff --git a/frontend/console/src/features/infrastructure/InfrastructurePage.tsx b/frontend/console/src/features/infrastructure/InfrastructurePage.tsx index e031445c62..2c652c96a6 100644 --- a/frontend/console/src/features/infrastructure/InfrastructurePage.tsx +++ b/frontend/console/src/features/infrastructure/InfrastructurePage.tsx @@ -1,5 +1,5 @@ import { useEffect, useState } from 'react' -import { useSearchParams } from 'react-router-dom' +import { useLocation, useNavigate } from 'react-router-dom' import { useStatus } from '../../api/status/use-status' import { Tabs } from '../../components/Tabs' import { ControllersList } from './ControllersList' @@ -9,7 +9,8 @@ import { RunnersList } from './RunnersList' export const InfrastructurePage = () => { const status = useStatus() - const [searchParams, setSearchParams] = useSearchParams() + const navigate = useNavigate() + const location = useLocation() const [tabs, setTabs] = useState([ { name: 'Controllers', id: 'controllers' }, @@ -41,30 +42,31 @@ export const InfrastructurePage = () => { ) }, [status.data]) - const handleTabClick = (tabId: string) => { - setSearchParams({ tab: tabId }) - } + const currentTab = location.pathname.split('/').pop() - const currentTab = searchParams.get('tab') || tabs[0].id const renderTabContent = () => { switch (currentTab) { case 'controllers': - return + return case 'runners': - return + return case 'deployments': - return + return case 'routes': - return + return default: return <> } } + const handleTabClick = (tabId: string) => { + navigate(`/infrastructure/${tabId}`) + } + return (
-
{renderTabContent()}
+
{renderTabContent()}
) } diff --git a/frontend/console/src/features/infrastructure/RoutesList.tsx b/frontend/console/src/features/infrastructure/RoutesList.tsx index eea618d767..ff69466be3 100644 --- a/frontend/console/src/features/infrastructure/RoutesList.tsx +++ b/frontend/console/src/features/infrastructure/RoutesList.tsx @@ -1,3 +1,27 @@ -export const RoutesList = () => { - return <>Routes Content +import { AttributeBadge } from '../../components' +import { List } from '../../components/List' +import type { StatusResponse_Route } from '../../protos/xyz/block/ftl/v1/ftl_pb' + +export const RoutesList = ({ routes }: { routes: StatusResponse_Route[] }) => { + return ( + ( +
+
+
+
{route.module}
+

{route.endpoint}

+
+
+
+
+ + +
+
+
+ )} + /> + ) } diff --git a/frontend/console/src/features/infrastructure/RunnersList.tsx b/frontend/console/src/features/infrastructure/RunnersList.tsx index 3daf525342..6a3860e289 100644 --- a/frontend/console/src/features/infrastructure/RunnersList.tsx +++ b/frontend/console/src/features/infrastructure/RunnersList.tsx @@ -1,3 +1,49 @@ -export const RunnersList = () => { - return <>Runners Content +import { AttributeBadge } from '../../components' +import { List } from '../../components/List' +import { StatusIndicator } from '../../components/StatusIndicator' +import { RunnerState, type StatusResponse_Runner } from '../../protos/xyz/block/ftl/v1/ftl_pb' +import { classNames } from '../../utils' +import { deploymentTextColor } from '../deployments/deployment.utils' +import { renderValue } from './infrastructure.utils' + +export const RunnersList = ({ runners }: { runners: StatusResponse_Runner[] }) => { + return ( + ( + <> +
+
+

{runner.key}

+

{runner.endpoint}

+
+ {status(runner.state)} + {runner.deployment &&

{runner.deployment}

} +
+
+
+
+
+ {Object.entries(runner.labels?.fields || {}).map(([key, value]) => ( + + ))} +
+
+ + )} + /> + ) +} + +const status = (state: RunnerState) => { + switch (state) { + case RunnerState.RUNNER_ASSIGNED: + return + case RunnerState.RUNNER_RESERVED: + return + case RunnerState.RUNNER_DEAD: + return + case RunnerState.RUNNER_IDLE: + return + } } diff --git a/frontend/console/src/features/infrastructure/infrastructure.utils.ts b/frontend/console/src/features/infrastructure/infrastructure.utils.ts new file mode 100644 index 0000000000..1ebecb4b33 --- /dev/null +++ b/frontend/console/src/features/infrastructure/infrastructure.utils.ts @@ -0,0 +1,18 @@ +import type { Value } from '@bufbuild/protobuf' + +export const renderValue = (value: Value): string => { + switch (value.kind?.case) { + case 'numberValue': + return value.kind.value.toString() + case 'stringValue': + return value.kind.value + case 'boolValue': + return value.kind.value ? 'true' : 'false' + case 'structValue': + return value.kind.value.toJsonString() + case 'listValue': + return value.kind.value.values.map(renderValue).join(', ') + default: + return '' + } +} diff --git a/frontend/console/src/providers/routing-provider.tsx b/frontend/console/src/providers/routing-provider.tsx index 86dfaeefcc..9e1025652d 100644 --- a/frontend/console/src/providers/routing-provider.tsx +++ b/frontend/console/src/providers/routing-provider.tsx @@ -18,7 +18,8 @@ const router = createBrowserRouter( } />} /> } />} /> } /> - } /> + } /> + } /> } />