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

[8.13][Security Solution][Endpoint] Add beta badge to sentinel one connector cards/flyout and responder/isolation action flyouts #176228

Merged
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
Show all changes
22 commits
Select commit Hold shift + click to select a range
52e7fee
add beta badge to sentinel one connector cards/flyout
ashokaditya Feb 5, 2024
a5dedfb
add beta tag to sentinel one responder
ashokaditya Feb 7, 2024
2ec1340
Add beta badge to alert isolate/release action flyout
ashokaditya Feb 7, 2024
41a950b
Update header.test.tsx
ashokaditya Feb 7, 2024
af0bb4b
Update beta badge tooltip text
ashokaditya Feb 7, 2024
194852f
add tests
ashokaditya Feb 7, 2024
5f09dc0
[CI] Auto-commit changed files from 'node scripts/eslint --no-cache -…
kibanamachine Feb 7, 2024
9797a21
beta badges for responder and isolate/release flyout
ashokaditya Feb 7, 2024
fcb7684
update tests
ashokaditya Feb 7, 2024
6d2a68e
[CI] Auto-commit changed files from 'node scripts/lint_ts_projects --…
kibanamachine Feb 7, 2024
6bb9d7f
fix lint and import
ashokaditya Feb 7, 2024
8d46b7e
[CI] Auto-commit changed files from 'node scripts/lint_ts_projects --…
kibanamachine Feb 7, 2024
7818a8f
Connector shows tech preview badge without ff
ashokaditya Feb 8, 2024
a8d80a1
update tests
ashokaditya Feb 8, 2024
e99103b
Merge branch 'main' into task/dw-beta-badge-sentinel-one-8526
ashokaditya Feb 8, 2024
f1d7821
Merge branch 'main' into task/dw-beta-badge-sentinel-one-8526
ashokaditya Feb 9, 2024
0a4ab17
Merge branch 'main' into task/dw-beta-badge-sentinel-one-8526
ashokaditya Feb 9, 2024
8b64722
Merge branch 'main' into task/dw-beta-badge-sentinel-one-8526
ashokaditya Feb 9, 2024
a1081c2
Merge branch 'main' into task/dw-beta-badge-sentinel-one-8526
ashokaditya Feb 9, 2024
3e32ee3
Merge branch 'main' into task/dw-beta-badge-sentinel-one-8526
ashokaditya Feb 12, 2024
a6dad82
Merge branch 'main' into task/dw-beta-badge-sentinel-one-8526
ashokaditya Feb 12, 2024
f0cde7a
Merge branch 'main' into task/dw-beta-badge-sentinel-one-8526
ashokaditya Feb 12, 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
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,11 @@ export const BETA = i18n.translate('xpack.securitySolution.pages.common.beta', {
defaultMessage: 'Beta',
});

export const BETA_TOOLTIP = i18n.translate('xpack.securitySolution.pages.common.beta.tooltip', {
defaultMessage:
'This functionality is in beta and is subject to change. The design and code is less mature than official GA features and is being provided as-is with no warranties. Beta features are not subject to the support SLA of official GA features.',
});

export const UPDATE_ALERT_STATUS_FAILED = (conflicts: number) =>
i18n.translate('xpack.securitySolution.pages.common.updateAlertStatusFailed', {
values: { conflicts },
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,9 +11,13 @@ import { __IntlProvider as IntlProvider } from '@kbn/i18n-react';
import { useIsolateHostPanelContext } from './context';
import { PanelHeader } from './header';
import { FLYOUT_HEADER_TITLE_TEST_ID } from './test_ids';
import { isAlertFromSentinelOneEvent } from '../../../common/utils/sentinelone_alert_check';

jest.mock('../../../common/utils/sentinelone_alert_check');
jest.mock('./context');

const mockIsAlertFromSentinelOneEvent = isAlertFromSentinelOneEvent as jest.Mock;

const renderPanelHeader = () =>
render(
<IntlProvider locale="en">
Expand All @@ -39,4 +43,19 @@ describe('<PanelHeader />', () => {
expect(getByTestId(FLYOUT_HEADER_TITLE_TEST_ID)).toBeInTheDocument();
expect(getByTestId(FLYOUT_HEADER_TITLE_TEST_ID)).toHaveTextContent('Release host');
});

it.each(['isolateHost', 'unisolateHost'])(
logeekal marked this conversation as resolved.
Show resolved Hide resolved
'should display beta badge on %s host message for SentinelOne alerts',
(action) => {
(useIsolateHostPanelContext as jest.Mock).mockReturnValue({
isolateAction: action,
});
mockIsAlertFromSentinelOneEvent.mockReturnValue(true);

const { getByTestId } = renderPanelHeader();

expect(getByTestId(FLYOUT_HEADER_TITLE_TEST_ID)).toBeInTheDocument();
expect(getByTestId(FLYOUT_HEADER_TITLE_TEST_ID)).toHaveTextContent('Beta');
}
);
});
Original file line number Diff line number Diff line change
Expand Up @@ -5,10 +5,12 @@
* 2.0.
*/

import { EuiTitle } from '@elastic/eui';
import { EuiBetaBadge, EuiFlexGroup, EuiFlexItem, EuiTitle } from '@elastic/eui';
import type { FC } from 'react';
import React from 'react';
import { FormattedMessage } from '@kbn/i18n-react';
import { BETA, BETA_TOOLTIP } from '../../../common/translations';
import { isAlertFromSentinelOneEvent } from '../../../common/utils/sentinelone_alert_check';
import { useIsolateHostPanelContext } from './context';
import { FLYOUT_HEADER_TITLE_TEST_ID } from './test_ids';
import { FlyoutHeader } from '../../shared/components/flyout_header';
Expand All @@ -17,20 +19,31 @@ import { FlyoutHeader } from '../../shared/components/flyout_header';
* Document details expandable right section header for the isolate host panel
*/
export const PanelHeader: FC = () => {
const { isolateAction } = useIsolateHostPanelContext();
const { isolateAction, dataFormattedForFieldBrowser: data } = useIsolateHostPanelContext();
const isSentinelOneAlert = isAlertFromSentinelOneEvent({ data });

const title =
isolateAction === 'isolateHost' ? (
<FormattedMessage
id="xpack.securitySolution.flyout.isolateHost.isolateTitle"
defaultMessage="Isolate host"
/>
) : (
<FormattedMessage
id="xpack.securitySolution.flyout.isolateHost.releaseTitle"
defaultMessage="Release host"
/>
);
const title = (
<EuiFlexGroup responsive gutterSize="s">
<EuiFlexItem grow={false}>
{isolateAction === 'isolateHost' ? (
<FormattedMessage
id="xpack.securitySolution.flyout.isolateHost.isolateTitle"
defaultMessage="Isolate host"
/>
) : (
<FormattedMessage
id="xpack.securitySolution.flyout.isolateHost.releaseTitle"
defaultMessage="Release host"
/>
)}
</EuiFlexItem>
{isSentinelOneAlert && (
<EuiFlexItem grow={false}>
<EuiBetaBadge label={BETA} tooltipContent={BETA_TOOLTIP} />
</EuiFlexItem>
)}
</EuiFlexGroup>
);

return (
<FlyoutHeader>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,27 +5,33 @@
* 2.0.
*/

import React, { useCallback } from 'react';
import { useLicense } from '../../common/hooks/use_license';
import type { ImmutableArray } from '../../../common/endpoint/types';
import React, {useCallback} from 'react';
import {EuiBetaBadge, EuiFlexGroup, EuiFlexItem} from '@elastic/eui';
import {BETA, BETA_TOOLTIP} from '../../common/translations';
import {useLicense} from '../../common/hooks/use_license';
import type {ImmutableArray} from '../../../common/endpoint/types';
import {
type ConsoleResponseActionCommands,
RESPONSE_CONSOLE_COMMAND_TO_API_COMMAND_MAP,
type ResponseActionAgentType,
} from '../../../common/endpoint/service/response_actions/constants';
import { isResponseActionSupported } from '../../../common/endpoint/service/response_actions/is_response_action_supported';
import { HeaderSentinelOneInfo } from '../components/endpoint_responder/components/header_info/sentinel_one/header_sentinel_one_info';
import {
isResponseActionSupported
} from '../../../common/endpoint/service/response_actions/is_response_action_supported';
import {
HeaderSentinelOneInfo
} from '../components/endpoint_responder/components/header_info/sentinel_one/header_sentinel_one_info';

import { useUserPrivileges } from '../../common/components/user_privileges';
import {useUserPrivileges} from '../../common/components/user_privileges';
import {
ActionLogButton,
getEndpointConsoleCommands,
HeaderEndpointInfo,
OfflineCallout,
} from '../components/endpoint_responder';
import { useConsoleManager } from '../components/console';
import { MissingEncryptionKeyCallout } from '../components/missing_encryption_key_callout';
import { RESPONDER_PAGE_TITLE } from './translations';
import {useConsoleManager} from '../components/console';
import {MissingEncryptionKeyCallout} from '../components/missing_encryption_key_callout';
import {RESPONDER_PAGE_TITLE} from './translations';

type ShowResponseActionsConsole = (props: ResponderInfoProps) => void;

Expand Down Expand Up @@ -126,7 +132,19 @@ export const useWithShowResponder = (): ShowResponseActionsConsole => {
hostName,
},
consoleProps,
PageTitleComponent: () => <>{RESPONDER_PAGE_TITLE}</>,
PageTitleComponent: () => {
if (agentType === 'sentinel_one') {
return (
<EuiFlexGroup>
<EuiFlexItem>{RESPONDER_PAGE_TITLE}</EuiFlexItem>
<EuiFlexItem grow={false}>
<EuiBetaBadge label={BETA} tooltipContent={BETA_TOOLTIP} />
</EuiFlexItem>
</EuiFlexGroup>
);
}
return <>{RESPONDER_PAGE_TITLE}</>;
},
ActionComponents: endpointPrivileges.canReadActionsLogManagement
? [ActionLogButton]
: undefined,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,9 +17,9 @@ import {
SUB_ACTION,
} from '../../../common/sentinelone/constants';
import type {
SentinelOneActionParams,
SentinelOneConfig,
SentinelOneSecrets,
SentinelOneActionParams,
} from '../../../common/sentinelone/types';

interface ValidationErrors {
Expand All @@ -35,7 +35,7 @@ export function getConnectorType(): ConnectorTypeModel<
id: SENTINELONE_CONNECTOR_ID,
actionTypeTitle: SENTINELONE_TITLE,
iconClass: lazy(() => import('./logo')),
isExperimental: true,
isBeta: true,
selectMessage: i18n.translate(
'xpack.stackConnectors.security.sentinelone.config.selectMessageText',
{
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,17 +6,16 @@
*/

import React, { useEffect, useState } from 'react';
import { EuiFlexItem, EuiCard, EuiIcon, EuiFlexGrid, EuiSpacer } from '@elastic/eui';
import { EuiCard, EuiFlexGrid, EuiFlexItem, EuiIcon, EuiSpacer, EuiToolTip } from '@elastic/eui';
import { i18n } from '@kbn/i18n';
import { EuiToolTip } from '@elastic/eui';
import { FormattedMessage } from '@kbn/i18n-react';
import { ActionType, ActionTypeIndex, ActionTypeRegistryContract } from '../../../types';
import { loadActionTypes } from '../../lib/action_connector_api';
import { actionTypeCompare } from '../../lib/action_type_compare';
import { checkActionTypeEnabled } from '../../lib/check_action_type_enabled';
import { useKibana } from '../../../common/lib/kibana';
import { SectionLoading } from '../../components/section_loading';
import { betaBadgeProps } from './beta_badge_props';
import { betaBadgeProps, technicalPreviewBadgeProps } from './beta_badge_props';

interface Props {
onActionTypeChange: (actionType: ActionType) => void;
Expand Down Expand Up @@ -77,12 +76,13 @@ export const ActionTypeMenu = ({
})();
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);

const registeredActionTypes = Object.entries(actionTypesIndex ?? [])
.filter(
([id, details]) =>
actionTypeRegistry.has(id) &&
details.enabledInConfig === true &&
!actionTypeRegistry.get(id).hideInUi
!actionTypeRegistry.get(id).hideInUi &&
details.enabledInConfig
)
.map(([id, actionType]) => {
const actionTypeModel = actionTypeRegistry.get(id);
Expand All @@ -91,6 +91,7 @@ export const ActionTypeMenu = ({
selectMessage: actionTypeModel ? actionTypeModel.selectMessage : '',
actionType,
name: actionType.name,
isBeta: actionTypeModel.isBeta,
isExperimental: actionTypeModel.isExperimental,
};
});
Expand All @@ -101,7 +102,13 @@ export const ActionTypeMenu = ({
const checkEnabledResult = checkActionTypeEnabled(item.actionType);
const card = (
<EuiCard
betaBadgeProps={item.isExperimental ? betaBadgeProps : undefined}
betaBadgeProps={
item.isBeta
? betaBadgeProps
: item.isExperimental
? technicalPreviewBadgeProps
: undefined
}
titleSize="xs"
data-test-subj={`${item.actionType.id}-card`}
icon={<EuiIcon size="xl" type={item.iconClass} />}
Expand All @@ -117,7 +124,7 @@ export const ActionTypeMenu = ({
return (
<EuiFlexItem key={index}>
{checkEnabledResult.isEnabled && card}
{checkEnabledResult.isEnabled === false && (
{!checkEnabledResult.isEnabled && (
<EuiToolTip position="top" content={checkEnabledResult.message}>
{card}
</EuiToolTip>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,16 @@
import { i18n } from '@kbn/i18n';

export const betaBadgeProps = {
label: i18n.translate('xpack.triggersActionsUI.betaBadgeLabel', {
defaultMessage: 'Beta',
}),
tooltipContent: i18n.translate('xpack.triggersActionsUI.betaBadgeDescription', {
defaultMessage:
'This functionality is in beta and is subject to change. The design and code is less mature than official GA features and is being provided as-is with no warranties. Beta features are not subject to the support SLA of official GA features.',
}),
};

export const technicalPreviewBadgeProps = {
label: i18n.translate('xpack.triggersActionsUI.technicalPreviewBadgeLabel', {
defaultMessage: 'Technical preview',
}),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,25 +8,26 @@
import React, { memo } from 'react';
import {
EuiBadge,
EuiTitle,
EuiBetaBadge,
EuiFlexGroup,
EuiFlexItem,
EuiFlyoutHeader,
EuiIcon,
EuiSpacer,
EuiText,
EuiFlyoutHeader,
EuiTitle,
IconType,
EuiSpacer,
EuiBetaBadge,
} from '@elastic/eui';
import { FormattedMessage } from '@kbn/i18n-react';
import { betaBadgeProps } from '../beta_badge_props';
import { betaBadgeProps, technicalPreviewBadgeProps } from '../beta_badge_props';

interface Props {
icon?: IconType | null;
actionTypeName?: string | null;
actionTypeMessage?: string | null;
compatibility?: string[] | null;
isExperimental?: boolean;
isBeta?: boolean;
}

const FlyoutHeaderComponent: React.FC<Props> = ({
Expand All @@ -35,6 +36,7 @@ const FlyoutHeaderComponent: React.FC<Props> = ({
actionTypeMessage,
compatibility,
isExperimental,
isBeta,
}) => {
return (
<EuiFlyoutHeader hasBorder data-test-subj="create-connector-flyout-header">
Expand All @@ -61,14 +63,23 @@ const FlyoutHeaderComponent: React.FC<Props> = ({
</h3>
</EuiTitle>
</EuiFlexItem>
{actionTypeName && isExperimental && (
<EuiFlexItem grow={false}>
<EuiBetaBadge
label={betaBadgeProps.label}
tooltipContent={betaBadgeProps.tooltipContent}
/>
</EuiFlexItem>
)}
{actionTypeName
? isBeta && (
<EuiFlexItem grow={false}>
<EuiBetaBadge
label={betaBadgeProps.label}
tooltipContent={betaBadgeProps.tooltipContent}
/>
</EuiFlexItem>
)
: isExperimental && (
<EuiFlexItem grow={false}>
<EuiBetaBadge
label={technicalPreviewBadgeProps.label}
tooltipContent={technicalPreviewBadgeProps.tooltipContent}
/>
</EuiFlexItem>
)}
</EuiFlexGroup>
<EuiText size="s" color="subdued">
{actionTypeMessage}
Expand Down
Loading