From f49cc8cd334861a43b090b6c2085a49bdad89769 Mon Sep 17 00:00:00 2001 From: Thomas Heartman Date: Wed, 13 Sep 2023 08:22:55 +0200 Subject: [PATCH] feat: update UI to add hints about implicit ^ and $ (#4667) This PR updates the UI to reflect the changes to the implicit ^ and $ that we now add. The changes are: 1. Show input adornments for ^ and $ when you create a pattern. 2. Mention that ^ and $ are added implicitly in description. 3. Checks the example you provide against the pattern with added ^ and $ + adds a test for that. Points 1 and 2: ![image](https://github.com/Unleash/unleash/assets/17786332/88c610b4-444b-4a83-a50a-4b7639614a86) ## Discussion point: I have not touched the information about the pattern yet as the PR that updates that is still in review (#4656), but it would be prudent to also update that info to make it clearer. I can address that in a follow-up PR. --- .../FeatureNamingPatternInfo.tsx | 2 +- .../ProjectForm/FeatureFlagNamingTooltip.tsx | 3 +- .../Project/ProjectForm/ProjectForm.tsx | 58 +++++++++++++------ .../validate-feature-naming.test.ts | 10 ++++ 4 files changed, 51 insertions(+), 22 deletions(-) diff --git a/frontend/src/component/feature/FeatureNamingPatternInfo/FeatureNamingPatternInfo.tsx b/frontend/src/component/feature/FeatureNamingPatternInfo/FeatureNamingPatternInfo.tsx index 53025f6e6b..ed11b501ca 100644 --- a/frontend/src/component/feature/FeatureNamingPatternInfo/FeatureNamingPatternInfo.tsx +++ b/frontend/src/component/feature/FeatureNamingPatternInfo/FeatureNamingPatternInfo.tsx @@ -37,7 +37,7 @@ export const FeatureNamingPatternInfo: React.FC = ({
Pattern
- {featureNaming.pattern} + ^{featureNaming.pattern}$
{ const X = 'X'; - const Y = 'Y'; const nx = 'n{X,}'; const nxy = 'n{X,Y}'; return ( @@ -14,7 +13,7 @@ export const FeatureFlagNamingTooltip: FC = () => {

Enforce a naming convention for feature flags


-

{`eg. ^[A-Za-z0-9]{2}[.][a-z]{4,12}$ matches 'a1.project'`}

+

{`eg. [A-Za-z0-9]{2}[.][a-z]{4,12} matches 'a1.project'`}

Brackets:

diff --git a/frontend/src/component/project/Project/ProjectForm/ProjectForm.tsx b/frontend/src/component/project/Project/ProjectForm/ProjectForm.tsx index dab56a1bdc..e6df250fea 100644 --- a/frontend/src/component/project/Project/ProjectForm/ProjectForm.tsx +++ b/frontend/src/component/project/Project/ProjectForm/ProjectForm.tsx @@ -4,7 +4,7 @@ import { StickinessSelect } from 'component/feature/StrategyTypes/FlexibleStrate import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender'; import Select from 'component/common/select'; import { ProjectMode } from '../hooks/useProjectForm'; -import { Box, styled, TextField } from '@mui/material'; +import { Box, InputAdornment, styled, TextField } from '@mui/material'; import { CollaborationModeTooltip } from './CollaborationModeTooltip'; import Input from 'component/common/Input/Input'; import { FeatureTogglesLimitTooltip } from './FeatureTogglesLimitTooltip'; @@ -106,6 +106,10 @@ const StyledFlagNamingContainer = styled('div')(({ theme }) => ({ '& > *': { width: '100%' }, })); +const StyledPatternNamingExplanation = styled('div')(({ theme }) => ({ + 'p + p': { marginTop: theme.spacing(1) }, +})); + export const validateFeatureNamingExample = ({ pattern, example, @@ -118,7 +122,7 @@ export const validateFeatureNamingExample = ({ if (featureNamingPatternError || !example || !pattern) { return { state: 'valid' }; } else if (example && pattern) { - const regex = new RegExp(pattern); + const regex = new RegExp(`^${pattern}$`); const matches = regex.test(example); if (!matches) { return { state: 'invalid', reason: 'Example does not match regex' }; @@ -354,29 +358,45 @@ const ProjectForm: React.FC = ({ -

- A feature flag naming pattern is a{' '} - - JavaScript RegEx - {' '} - used to enforce feature flag names within - this project. -

-

- Leave it empty if you don’t want to add a - naming pattern. -

+ +

+ Define a{' '} + + JavaScript RegEx + {' '} + used to enforce feature flag names + within this project. The regex will be + surrounded by a leading ^{' '} + and a trailing $. +

+

+ Leave it empty if you don’t want to add + a naming pattern. +

+
+ ^ + + ), + endAdornment: ( + + $ + + ), + }} type={'text'} value={featureNamingPattern || ''} error={Boolean(errors.featureNamingPattern)} diff --git a/frontend/src/component/project/Project/ProjectForm/validate-feature-naming.test.ts b/frontend/src/component/project/Project/ProjectForm/validate-feature-naming.test.ts index f54233fd9e..df14b1929c 100644 --- a/frontend/src/component/project/Project/ProjectForm/validate-feature-naming.test.ts +++ b/frontend/src/component/project/Project/ProjectForm/validate-feature-naming.test.ts @@ -47,4 +47,14 @@ describe('validateFeatureNaming', () => { expect(result.state).toBe(state); } ); + + test('the pattern gets an implicit leading ^ and trailing $ added', () => { + const result = validateFeatureNamingExample({ + pattern: '[a-z]+', + example: 'not.valid', + featureNamingPatternError: undefined, + }); + + expect(result.state).toBe('invalid'); + }); });