1
0
mirror of https://github.com/Unleash/unleash.git synced 2025-01-25 00:07:47 +01:00

Fix/constraint with legal value that has been deleted (#4473)

When editing a constraint that uses a context field with legal values:

if the contraint has a value that has been deleted from the legal values
of the context field:
- Show the value and mark it as disabled
- On any change -> 'cleans'/removed the deleted legal values from the
constraint values

Closes:
[1-1209](https://linear.app/unleash/issue/1-1209/if-i-modified-the-legal-values-of-a-used-context-field)

---------

Signed-off-by: andreas-unleash <andreas@getunleash.ai>
This commit is contained in:
andreas-unleash 2023-08-11 13:44:28 +03:00 committed by GitHub
parent c1fe43fe64
commit f9a686ca81
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 101 additions and 16 deletions

View File

@ -1,4 +1,4 @@
import { IUnleashContextDefinition } from 'interfaces/context';
import { ILegalValue, IUnleashContextDefinition } from 'interfaces/context';
import { IConstraint } from 'interfaces/strategy';
import { DateSingleValue } from '../DateSingleValue/DateSingleValue';
import { FreeTextInput } from '../FreeTextInput/FreeTextInput';
@ -30,6 +30,25 @@ interface IResolveInputProps {
error: string;
}
const resolveLegalValues = (
values: IConstraint['values'],
legalValues: IUnleashContextDefinition['legalValues']
): { legalValues: ILegalValue[]; deletedLegalValues: ILegalValue[] } => {
const deletedLegalValues = (values || [])
.filter(
value =>
!(legalValues || []).some(
({ value: legalValue }) => legalValue === value
)
)
.map(v => ({ value: v, description: '' }));
return {
legalValues: legalValues || [],
deletedLegalValues,
};
};
export const ResolveInput = ({
input,
contextDefinition,
@ -43,20 +62,14 @@ export const ResolveInput = ({
const resolveInput = () => {
switch (input) {
case IN_OPERATORS_LEGAL_VALUES:
return (
<RestrictiveLegalValues
legalValues={contextDefinition.legalValues || []}
values={localConstraint.values || []}
setValues={setValues}
error={error}
setError={setError}
/>
);
case STRING_OPERATORS_LEGAL_VALUES:
return (
<>
<RestrictiveLegalValues
legalValues={contextDefinition.legalValues || []}
data={resolveLegalValues(
localConstraint.values,
contextDefinition.legalValues
)}
values={localConstraint.values || []}
setValues={setValues}
error={error}

View File

@ -0,0 +1,54 @@
import { render } from 'utils/testRenderer';
import { RestrictiveLegalValues } from './RestrictiveLegalValues';
import { screen } from '@testing-library/react';
import { vi } from 'vitest';
describe('RestictiveLegalValues', () => {
it('should show deleted legal values as disabled', async () => {
const value = 'some-value';
const values = [{ value }];
const legalValues = [{ value: 'some-other-value' }];
render(
<RestrictiveLegalValues
data={{ legalValues, deletedLegalValues: values }}
values={[value]}
setValues={vi.fn()}
error={''}
setError={vi.fn()}
/>
);
const input = await screen.findByDisplayValue('some-value');
expect(input).toBeInTheDocument();
expect(input).toHaveProperty('disabled', true);
expect(
await screen.findByDisplayValue('some-other-value')
).toBeInTheDocument();
});
it('should remove deleted legal values when editing values', async () => {
const value = 'some-value';
const deletedLegalValues = [{ value }];
const legalValues = [
{ value: 'some-other-value' },
{ value: 'value2' },
];
const setValues = vi.fn();
render(
<RestrictiveLegalValues
data={{ legalValues, deletedLegalValues }}
values={[value, 'value2']}
setValues={setValues}
error={''}
setError={vi.fn()}
/>
);
const btn = await screen.findByDisplayValue('some-other-value');
btn.click();
expect(setValues).toHaveBeenCalledWith(['value2', 'some-other-value']);
});
});

View File

@ -6,12 +6,15 @@ import { ConstraintValueSearch } from 'component/common/ConstraintAccordion/Cons
import { ConstraintFormHeader } from '../ConstraintFormHeader/ConstraintFormHeader';
import { ILegalValue } from 'interfaces/context';
import {
LegalValueLabel,
filterLegalValues,
LegalValueLabel,
} from '../LegalValueLabel/LegalValueLabel';
interface IRestrictiveLegalValuesProps {
legalValues: ILegalValue[];
data: {
legalValues: ILegalValue[];
deletedLegalValues: ILegalValue[];
};
values: string[];
setValues: (values: string[]) => void;
beforeValues?: JSX.Element;
@ -33,13 +36,14 @@ const createValuesMap = (values: string[]): IValuesMap => {
};
export const RestrictiveLegalValues = ({
legalValues,
data,
values,
setValues,
error,
setError,
}: IRestrictiveLegalValuesProps) => {
const [filter, setFilter] = useState('');
const { legalValues, deletedLegalValues } = data;
const filteredValues = filterLegalValues(legalValues, filter);
// Lazily initialise the values because there might be a lot of them.
@ -50,8 +54,19 @@ export const RestrictiveLegalValues = ({
setValuesMap(createValuesMap(values));
}, [values, setValuesMap]);
const cleanDeletedLegalValues = (constraintValues: string[]): string[] => {
const deletedValuesSet = new Set(
deletedLegalValues.map(({ value }) => value)
);
return (
constraintValues?.filter(value => !deletedValuesSet.has(value)) ||
[]
);
};
const onChange = (legalValue: string) => {
setError('');
if (valuesMap[legalValue]) {
const index = values.findIndex(value => value === legalValue);
const newValues = [...values];
@ -60,7 +75,7 @@ export const RestrictiveLegalValues = ({
return;
}
setValues([...values, legalValue]);
setValues([...cleanDeletedLegalValues(values), legalValue]);
};
return (
@ -77,7 +92,7 @@ export const RestrictiveLegalValues = ({
/>
}
/>
{filteredValues.map(match => (
{filteredValues.concat(deletedLegalValues).map(match => (
<LegalValueLabel
key={match.value}
legal={match}
@ -87,6 +102,9 @@ export const RestrictiveLegalValues = ({
onChange={() => onChange(match.value)}
name={match.value}
color="primary"
disabled={deletedLegalValues
.map(({ value }) => value)
.includes(match.value)}
/>
}
/>