Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(diagnostics): Add Dashboard card for invoking diagnostic operations #1426

Merged
merged 23 commits into from
Oct 1, 2024
Merged
Show file tree
Hide file tree
Changes from 6 commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
d9ac002
Initial implementation
Josh-Matsuoka Sep 24, 2024
db065a4
Fix incorrect API call
Josh-Matsuoka Sep 24, 2024
36576c1
Fix API path
Josh-Matsuoka Sep 24, 2024
e8d9713
Fix API request
Josh-Matsuoka Sep 24, 2024
5c3a96a
Run Prettier
Josh-Matsuoka Sep 24, 2024
d934f92
Merge branch 'main' into diagnostics-card-1
Josh-Matsuoka Sep 24, 2024
ec57f58
Fix error handling
Josh-Matsuoka Sep 25, 2024
5ad55e9
Add loading spinner to diagnostic card
Josh-Matsuoka Sep 27, 2024
ad18b1b
Move DiagnosticsCard, clean up localization
Josh-Matsuoka Oct 1, 2024
d0374c6
Merge branch 'main' into diagnostics-card-1
Josh-Matsuoka Oct 1, 2024
4de2d58
Merge remote-tracking branch 'upstream/main' into diagnostics-card-1
andrewazores Oct 1, 2024
c1628d0
Remove refresh button, cleanup
Josh-Matsuoka Oct 1, 2024
115fd63
Merge remote-tracking branch 'jmatsuok/diagnostics-card-1' into diagn…
Josh-Matsuoka Oct 1, 2024
f2913dc
remove unused translation component
andrewazores Oct 1, 2024
0facbbf
remove dead code
andrewazores Oct 1, 2024
30351c9
restore default card header actions
andrewazores Oct 1, 2024
c9e3099
inline props, remove unused props
andrewazores Oct 1, 2024
d8aceef
remove unused title prop
andrewazores Oct 1, 2024
ca8fde9
ensure observables complete after first emission
andrewazores Oct 1, 2024
4bfdef2
loading state cleanup
andrewazores Oct 1, 2024
420a52b
pass diagnostics failure back to card for handling
andrewazores Oct 1, 2024
4f5896d
error notification styling
andrewazores Oct 1, 2024
76a3344
lint
andrewazores Oct 1, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions locales/en/public.json
Original file line number Diff line number Diff line change
Expand Up @@ -157,6 +157,10 @@
"MBEAN_METRICS_CARD_DESCRIPTION": "Display common performance metrics from current MBean data.",
"MBEAN_METRICS_CARD_DESCRIPTION_FULL": "Display a single performance metric from a list of supported MBeans.",
"MBEAN_METRICS_CARD_TITLE": "MBean Metrics Chart",
"DIAGNOSTICS_CARD_DESCRIPTION": "Perform diagnostic operations on the target.",
"DIAGNOSTICS_CARD_DESCRIPTION_FULL": "Perform diagonstic operations from a list of supported operations on the target.",
"DIAGNOSTICS_CARD_TITLE": "Diagnostics",
"DIAGNOSTICS_GC_BUTTON": "Start Garbage Collection",
Josh-Matsuoka marked this conversation as resolved.
Show resolved Hide resolved
"NO_RECORDING": {
"DESCRIPTION": "Metrics cards display data taken from running Flight Recordings with the label <label>origin={{recordingName}}</label>. No such Recordings are currently available.",
"TITLE": "No source Recording"
Expand Down
186 changes: 186 additions & 0 deletions src/app/Dashboard/Charts/diagnostics/DiagnosticsCard.tsx
Josh-Matsuoka marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,186 @@
/*
* Copyright The Cryostat Authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

import {
DashboardCardTypeProps,
DashboardCardFC,
DashboardCardSizes,
DashboardCardDescriptor,
} from '@app/Dashboard/types';
import { ErrorView } from '@app/ErrorView/ErrorView';
import { FeatureLevel } from '@app/Shared/Services/service.types';
import { ServiceContext } from '@app/Shared/Services/Services';
import { useSubscriptions } from '@app/utils/hooks/useSubscriptions';
import {
Bullseye,
Button,
CardBody,
CardHeader,
CardTitle,
EmptyState,
EmptyStateBody,
EmptyStateIcon,
EmptyStateVariant,
Label,
EmptyStateHeader,
EmptyStateFooter,
} from '@patternfly/react-core';
import { DataSourceIcon, SyncAltIcon, TachometerAltIcon } from '@patternfly/react-icons';
import * as React from 'react';
import { Trans, useTranslation } from 'react-i18next';
import { DashboardCard } from '../../DashboardCard';

export interface DiagnosticsCardProps extends DashboardCardTypeProps {
chartKind: string;
duration: number;
period: number;
}

// TODO are these needed?
export enum DiagnosticsCardKind {}

export function kindToId(kind: string): number {
return DiagnosticsCardKind[kind];
}

export const DiagnosticsCard: DashboardCardFC<DiagnosticsCardProps> = (props) => {
const { t } = useTranslation();
const serviceContext = React.useContext(ServiceContext);
const addSubscription = useSubscriptions();

const handleError = React.useCallback((error) => {
return <ErrorView title={'Error executing diagnostic command'} message={`${error.message}`} />;
Josh-Matsuoka marked this conversation as resolved.
Show resolved Hide resolved
}, []);

const handleGC = React.useCallback(() => {
addSubscription(
serviceContext.api.runGC().subscribe({
error: (err) => handleError(err),
Josh-Matsuoka marked this conversation as resolved.
Show resolved Hide resolved
}),
);
}, [addSubscription, serviceContext.api, handleError]);

const GCButton = React.useMemo(() => {
return (
<Button
key={0}
aria-label={t('DIAGNOSTICS_GC_BUTTON', { chartKind: props.chartKind })}
onClick={handleGC}
variant="plain"
icon={<SyncAltIcon />}
isDisabled={false}
/>
);
}, [t, props.chartKind, handleGC]);

const actions = React.useMemo(() => {
const a = props.actions || [];
return [GCButton, ...a];
}, [props.actions, GCButton]);

const header = React.useMemo(() => {
return (
<CardHeader
actions={{
actions: <>{actions}</>,
hasNoOffset: false,
className: undefined,
}}
>
<CardTitle>
{t('CHART_CARD.TITLE', { chartKind: props.chartKind, duration: props.duration, period: props.period })}
Josh-Matsuoka marked this conversation as resolved.
Show resolved Hide resolved
</CardTitle>
</CardHeader>
);
}, [props.chartKind, props.duration, props.period, t, actions]);

return (
<DashboardCard
id={props.chartKind + '-chart-card'}
dashboardId={props.dashboardId}
cardSizes={DiagnosticsCardSizes}
isCompact
cardHeader={header}
title={props.chartKind}
isDraggable={props.isDraggable}
isResizable={props.isResizable}
isFullHeight={props.isFullHeight}
>
<CardBody>
<Bullseye>
<EmptyState variant={EmptyStateVariant.lg}>
<EmptyStateHeader
titleText={<>{t('CHART_CARD.DIAGNOSTICS_CARD_TITLE')}</>}
icon={<EmptyStateIcon icon={DataSourceIcon} />}
headingLevel="h2"
/>
<EmptyStateBody>
<Trans
t={t}
values={{ recordingName: 'RECORDING_NAME' }}
Josh-Matsuoka marked this conversation as resolved.
Show resolved Hide resolved
components={{ label: <Label color="blue" isCompact /> }}
>
CHART_CARD.DIAGNOSTICS_CARD_DESCRIPTION
</Trans>
</EmptyStateBody>
<EmptyStateFooter>
<Button variant="primary" onClick={handleGC}>
{t('CHART_CARD.DIAGNOSTICS_GC_BUTTON')}
</Button>
</EmptyStateFooter>
</EmptyState>
</Bullseye>
</CardBody>
</DashboardCard>
);
};

DiagnosticsCard.cardComponentName = 'DiagnosticsCard';

export const DiagnosticsCardSizes: DashboardCardSizes = {
span: {
minimum: 3,
default: 4,
maximum: 12,
},
height: {
// TODO: implement height resizing
minimum: Number.NaN,
default: Number.NaN,
maximum: Number.NaN,
},
};

export const DiagnosticsCardDescriptor: DashboardCardDescriptor = {
featureLevel: FeatureLevel.BETA,
title: 'CHART_CARD.DIAGNOSTICS_CARD_TITLE',
cardSizes: DiagnosticsCardSizes,
description: 'CHART_CARD.DIAGNOSTICS_CARD_DESCRIPTION',
descriptionFull: 'CHART_CARD.DIAGNOSTICS_CARD_DESCRIPTION_FULL',
component: DiagnosticsCard,
propControls: [],
icon: <TachometerAltIcon />,
labels: [
{
content: 'Beta',
color: 'cyan',
},
{
content: 'Diagnostics',
color: 'blue',
},
],
};
2 changes: 2 additions & 0 deletions src/app/Dashboard/utils.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ import * as React from 'react';
import { useTranslation } from 'react-i18next';
import { useDispatch } from 'react-redux';
import { AutomatedAnalysisCardDescriptor } from './AutomatedAnalysis/AutomatedAnalysisCard';
import { DiagnosticsCardDescriptor } from './Charts/diagnostics/DiagnosticsCard';
import { JFRMetricsChartCardDescriptor } from './Charts/jfr/JFRMetricsChartCard';
import { MBeanMetricsChartCardDescriptor } from './Charts/mbean/MBeanMetricsChartCard';
import { JvmDetailsCardDescriptor } from './JvmDetails/JvmDetailsCard';
Expand Down Expand Up @@ -165,6 +166,7 @@ export const getDashboardCards: (featureLevel?: FeatureLevel) => DashboardCardDe
AutomatedAnalysisCardDescriptor,
JFRMetricsChartCardDescriptor,
MBeanMetricsChartCardDescriptor,
DiagnosticsCardDescriptor,
];
return cards.filter((card) => card.featureLevel >= featureLevel);
};
Expand Down
14 changes: 14 additions & 0 deletions src/app/Shared/Services/Api.service.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -671,6 +671,20 @@ export class ApiService {
);
}

runGC(): Observable<boolean> {
return this.target.target().pipe(
concatMap((target) =>
this.sendRequest('beta', `diagnostics/targets/${target?.id}/gc`, {
method: 'POST',
}).pipe(
map((resp) => resp.ok),
catchError(() => of(false)),
first(),
),
),
);
}

insertProbes(templateName: string): Observable<boolean> {
return this.target.target().pipe(
concatMap((target) =>
Expand Down
Loading