Skip to content
This repository has been archived by the owner on Nov 28, 2024. It is now read-only.

Commit

Permalink
Merge pull request #95 from upalatucci/filter-by-lldp
Browse files Browse the repository at this point in the history
CNV-39761: filter lldp enabled or disabled
  • Loading branch information
openshift-merge-bot[bot] authored Mar 22, 2024
2 parents 3722840 + f82aaf2 commit 0fa9ce2
Show file tree
Hide file tree
Showing 6 changed files with 95 additions and 10 deletions.
40 changes: 40 additions & 0 deletions cypress/e2e/StatusList.spec.cy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,8 @@ import {
EXPAND_INTERFACES_LIST_TEST_ID,
INTERFACE_DRAWER_TEST_ID,
LLDP_DRAWER_DETAILS_SECTION_TEST_ID,
LLDP_ENABLED_FILTER,
ROW_FILTERS_BUTTON,
} from '../support/selectors';

describe('NodeNetworkState list', () => {
Expand Down Expand Up @@ -83,4 +85,42 @@ describe('NodeNetworkState list', () => {
.should('contain', 'VLANS');
});
});

it('filter by lldp', () => {
cy.intercept('GET', '/api/kubernetes/apis/nmstate.io/v1beta1/nodenetworkstates*', {
fixture: 'NodeNetworkStatusWithLLDP.json',
}).as('getStatuses');

cy.fixture('NodeNetworkStatusWithLLDP.json').then((nnsResponse) => {
const nns = nnsResponse.items[0];
const ifaceWithLLDP = nns.status.currentState.interfaces.find((iface) => iface.lldp?.enabled);
const ifaceWithoutLLDP = nns.status.currentState.interfaces.find(
(iface) => !iface.lldp?.enabled,
);

cy.visit('/k8s/cluster/nmstate.io~v1beta1~NodeNetworkState');

cy.wait(['@getStatuses'], { timeout: 40000 });

cy.get('table').should('contain', nns.metadata.name);

// open filter toolbar
cy.get(ROW_FILTERS_BUTTON).click();

cy.get(LLDP_ENABLED_FILTER).check();

// close filter toolbar
cy.get(ROW_FILTERS_BUTTON).click();

cy.get('table').should('contain', nns.metadata.name);
cy.get(EXPAND_INTERFACES_LIST_TEST_ID).click();
cy.byTestID(EXPAND_INTERFACE_INFO).find('button').click();

cy.byTestID(`${ifaceWithLLDP.type}-${ifaceWithLLDP.name}-open-drawer`).contains(
ifaceWithLLDP.name,
);

cy.byTestID(`${ifaceWithoutLLDP.type}-expandable-section-toggle`).should('not.exist');
});
});
});
4 changes: 4 additions & 0 deletions cypress/support/selectors.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,3 +2,7 @@ export const EXPAND_INTERFACES_LIST_TEST_ID = '#expand-interfaces-list0';
export const EXPAND_INTERFACE_INFO = 'ethernet-expandable-section-toggle';
export const INTERFACE_DRAWER_TEST_ID = 'interface-drawer';
export const LLDP_DRAWER_DETAILS_SECTION_TEST_ID = 'lldp-section';

export const ROW_FILTERS_BUTTON = 'button.pf-v5-c-select__toggle';

export const LLDP_ENABLED_FILTER = '#enabled[type="checkbox"]';
3 changes: 3 additions & 0 deletions src/utils/helpers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,9 @@ import {

import { ALL_NAMESPACES_SESSION_KEY } from './constants';

export const isEmpty = (obj) =>
[Array, Object].includes((obj || {}).constructor) && !Object.entries(obj || {}).length;

export const ensurePath = <T extends object>(data: T, paths: string | string[]) => {
let current = data;

Expand Down
15 changes: 11 additions & 4 deletions src/views/states/list/components/utils.ts
Original file line number Diff line number Diff line change
@@ -1,29 +1,36 @@
import { InterfaceType, NodeNetworkConfigurationInterface } from '@types';
import { isEmpty } from '@utils/helpers';

import { FILTER_TYPES } from '../constants';
import { FILTER_TYPES, LLDP_ENABLED } from '../constants';
import { searchInterfaceByIP } from '../utilts';

export const interfaceFilters: Record<
string,
(selectedInput: string[], obj: NodeNetworkConfigurationInterface) => boolean
> = {
[FILTER_TYPES.INTERFACE_STATE]: (selectedInput, obj) => {
if (!selectedInput.length) return true;
if (isEmpty(selectedInput)) return true;
return selectedInput.some((status) => obj.state.toLowerCase() === status);
},
[FILTER_TYPES.INTERFACE_TYPE]: (selectedInput, obj) => {
if (!selectedInput.length) return true;
if (isEmpty(selectedInput)) return true;
return selectedInput.some((interfaceType) => obj.type === interfaceType);
},
[FILTER_TYPES.IP_FILTER]: (selectedInput, obj) => {
if (!selectedInput.length) return true;
if (isEmpty(selectedInput)) return true;
return selectedInput.some((ipType) => !!obj[ipType]);
},
[FILTER_TYPES.IP_ADDRESS]: (selectedInput, obj) => {
const searchIPAddress = selectedInput?.[0];

return searchInterfaceByIP(searchIPAddress, obj);
},
[FILTER_TYPES.LLDP]: (selectedInput, obj) => {
if (isEmpty(selectedInput)) return true;
return selectedInput.some((status) =>
status === LLDP_ENABLED ? Boolean(obj?.lldp?.enabled) : !obj?.lldp?.enabled,
);
},
} as const;

export const filterInterfaces = (
Expand Down
4 changes: 4 additions & 0 deletions src/views/states/list/constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,4 +9,8 @@ export const FILTER_TYPES = {
INTERFACE_TYPE: 'interface-type',
IP_FILTER: 'ip-filter',
IP_ADDRESS: 'ip-address',
LLDP: 'lldp',
} as const;

export const LLDP_ENABLED = 'enabled';
export const LLDP_DISABLED = 'disabled';
39 changes: 33 additions & 6 deletions src/views/states/list/hooks/useStateFilters.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,9 @@ import { useNMStateTranslation } from 'src/utils/hooks/useNMStateTranslation';

import { RowFilter, RowSearchFilter } from '@openshift-console/dynamic-plugin-sdk';
import { InterfaceType, NodeNetworkConfigurationInterface, V1beta1NodeNetworkState } from '@types';
import { isEmpty } from '@utils/helpers';

import { FILTER_TYPES } from '../constants';
import { FILTER_TYPES, LLDP_DISABLED, LLDP_ENABLED } from '../constants';
import { searchInterfaceByIP } from '../utilts';

export const useStateSearchFilters = (): RowSearchFilter<V1beta1NodeNetworkState>[] => {
Expand All @@ -30,12 +31,38 @@ export const useStateFilters = (): RowFilter<V1beta1NodeNetworkState>[] => {
const { t } = useNMStateTranslation();

return [
{
filterGroupName: t('LLDP'),
type: FILTER_TYPES.LLDP,
filter: (selectedLLDPStatus, obj) => {
if (isEmpty(selectedLLDPStatus.selected)) return true;
return selectedLLDPStatus.selected.some((status) =>
obj?.status?.currentState?.interfaces.some((iface) =>
status === LLDP_ENABLED ? Boolean(iface?.lldp?.enabled) : !iface?.lldp?.enabled,
),
);
},
isMatch: (obj, status) =>
obj?.status?.currentState?.interfaces.some((iface) =>
status === LLDP_ENABLED ? Boolean(iface?.lldp?.enabled) : !iface?.lldp?.enabled,
),
items: [
{
id: LLDP_ENABLED,
title: t('Enabled'),
},
{
id: LLDP_DISABLED,
title: t('Disabled'),
},
],
},
{
filterGroupName: t('Interface state'),
type: FILTER_TYPES.INTERFACE_STATE,
filter: (selectedIpTypes, obj) => {
if (!selectedIpTypes.selected.length) return true;
return selectedIpTypes.selected.some((status) =>
filter: (selectedInfaceState, obj) => {
if (isEmpty(selectedInfaceState.selected)) return true;
return selectedInfaceState.selected.some((status) =>
obj?.status?.currentState?.interfaces.some(
(iface) => iface.state.toLowerCase() === status,
),
Expand All @@ -58,7 +85,7 @@ export const useStateFilters = (): RowFilter<V1beta1NodeNetworkState>[] => {
filterGroupName: t('Interface type'),
type: FILTER_TYPES.INTERFACE_TYPE,
filter: (selectedInterfaceTypes, obj) => {
if (!selectedInterfaceTypes.selected.length) return true;
if (isEmpty(selectedInterfaceTypes.selected)) return true;
return selectedInterfaceTypes.selected.some((interfaceType) =>
obj?.status?.currentState?.interfaces.some(
(iface: NodeNetworkConfigurationInterface) => iface.type === interfaceType,
Expand All @@ -84,7 +111,7 @@ export const useStateFilters = (): RowFilter<V1beta1NodeNetworkState>[] => {
filterGroupName: t('IP'),
type: FILTER_TYPES.IP_FILTER,
filter: (selectedIpTypes, obj) => {
if (!selectedIpTypes.selected.length) return true;
if (isEmpty(selectedIpTypes.selected)) return true;
return selectedIpTypes.selected.some((ipType) =>
obj?.status?.currentState?.interfaces.some((i) => !!i[ipType]),
);
Expand Down

0 comments on commit 0fa9ce2

Please sign in to comment.