Fix transliteration for metadata + transliterate select options (#5430)
## Context Fixes #5403 Transliteration is now integrated to form validation through the schema. While it does not impede inputting an invalid value, it impedes submitting a form that will fail as the transliteration is not possible. Until then we were only performing the transliteration at save time in the front-end, but it's best to provide the information as soon as possible. Later we will add helpers to guide the user (eg "This name is not valid": https://github.com/twentyhq/twenty/issues/5428). --------- Co-authored-by: Charles Bochet <charles@twenty.com>
This commit is contained in:
@ -18,7 +18,9 @@ export const RightDrawerCalendarEvent = () => {
|
|||||||
onCompleted: (record) => setRecords([record]),
|
onCompleted: (record) => setRecords([record]),
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!calendarEvent) return null;
|
if (!calendarEvent) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
return <CalendarEventDetails calendarEvent={calendarEvent} />;
|
return <CalendarEventDetails calendarEvent={calendarEvent} />;
|
||||||
};
|
};
|
||||||
|
|||||||
@ -1,5 +1,5 @@
|
|||||||
import { FieldMetadataItem } from '@/object-metadata/types/FieldMetadataItem';
|
import { FieldMetadataItem } from '@/object-metadata/types/FieldMetadataItem';
|
||||||
import { formatMetadataLabelToMetadataNameOrThrows } from '~/pages/settings/data-model/utils/format-metadata-label-to-name.util';
|
import { computeMetadataNameFromLabelOrThrow } from '~/pages/settings/data-model/utils/compute-metadata-name-from-label.utils';
|
||||||
|
|
||||||
export const formatFieldMetadataItemInput = (
|
export const formatFieldMetadataItemInput = (
|
||||||
input: Partial<
|
input: Partial<
|
||||||
@ -16,7 +16,7 @@ export const formatFieldMetadataItemInput = (
|
|||||||
description: input.description?.trim() ?? null,
|
description: input.description?.trim() ?? null,
|
||||||
icon: input.icon,
|
icon: input.icon,
|
||||||
label,
|
label,
|
||||||
name: label ? formatMetadataLabelToMetadataNameOrThrows(label) : undefined,
|
name: label ? computeMetadataNameFromLabelOrThrow(label) : undefined,
|
||||||
options: input.options,
|
options: input.options,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
@ -0,0 +1,24 @@
|
|||||||
|
import { metadataLabelSchema } from '@/object-metadata/validation-schemas/metadataLabelSchema';
|
||||||
|
|
||||||
|
describe('metadataLabelSchema', () => {
|
||||||
|
it('validates a valid label', () => {
|
||||||
|
// Given
|
||||||
|
const validMetadataLabel = 'Option 1';
|
||||||
|
|
||||||
|
// When
|
||||||
|
const result = metadataLabelSchema.parse(validMetadataLabel);
|
||||||
|
|
||||||
|
// Then
|
||||||
|
expect(result).toEqual(validMetadataLabel);
|
||||||
|
});
|
||||||
|
it('validates a label with non-latin characters', () => {
|
||||||
|
// Given
|
||||||
|
const validMetadataLabel = 'עִבְרִי';
|
||||||
|
|
||||||
|
// When
|
||||||
|
const result = metadataLabelSchema.parse(validMetadataLabel);
|
||||||
|
|
||||||
|
// Then
|
||||||
|
expect(result).toEqual(validMetadataLabel);
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -1,7 +1,23 @@
|
|||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
import { METADATA_LABEL_VALID_PATTERN } from '~/pages/settings/data-model/constants/MetadataLabelValidPattern';
|
||||||
|
import { computeMetadataNameFromLabelOrThrow } from '~/pages/settings/data-model/utils/compute-metadata-name-from-label.utils';
|
||||||
|
|
||||||
export const metadataLabelSchema = z
|
export const metadataLabelSchema = z
|
||||||
.string()
|
.string()
|
||||||
.trim()
|
.trim()
|
||||||
.min(1)
|
.min(1)
|
||||||
.regex(/^[a-zA-Z][a-zA-Z0-9 ()]*$/);
|
.regex(METADATA_LABEL_VALID_PATTERN)
|
||||||
|
.refine(
|
||||||
|
(label) => {
|
||||||
|
try {
|
||||||
|
computeMetadataNameFromLabelOrThrow(label);
|
||||||
|
return true;
|
||||||
|
} catch (error) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: 'Label is not formattable',
|
||||||
|
},
|
||||||
|
); // allows non-latin char
|
||||||
|
|||||||
@ -3,6 +3,7 @@ import { z } from 'zod';
|
|||||||
import { FieldMetadataItemOption } from '@/object-metadata/types/FieldMetadataItem';
|
import { FieldMetadataItemOption } from '@/object-metadata/types/FieldMetadataItem';
|
||||||
import { getOptionValueFromLabel } from '@/settings/data-model/fields/forms/utils/getOptionValueFromLabel';
|
import { getOptionValueFromLabel } from '@/settings/data-model/fields/forms/utils/getOptionValueFromLabel';
|
||||||
import { themeColorSchema } from '@/ui/theme/utils/themeColorSchema';
|
import { themeColorSchema } from '@/ui/theme/utils/themeColorSchema';
|
||||||
|
import { computeOptionValueFromLabelOrThrow } from '~/pages/settings/data-model/utils/compute-option-value-from-label.utils';
|
||||||
|
|
||||||
const selectOptionSchema = z
|
const selectOptionSchema = z
|
||||||
.object({
|
.object({
|
||||||
@ -14,7 +15,20 @@ const selectOptionSchema = z
|
|||||||
})
|
})
|
||||||
.refine((option) => option.value === getOptionValueFromLabel(option.label), {
|
.refine((option) => option.value === getOptionValueFromLabel(option.label), {
|
||||||
message: 'Value does not match label',
|
message: 'Value does not match label',
|
||||||
}) satisfies z.ZodType<FieldMetadataItemOption>;
|
})
|
||||||
|
.refine(
|
||||||
|
(option) => {
|
||||||
|
try {
|
||||||
|
computeOptionValueFromLabelOrThrow(option.label);
|
||||||
|
return true;
|
||||||
|
} catch (error) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: 'Label is not transliterable',
|
||||||
|
},
|
||||||
|
) satisfies z.ZodType<FieldMetadataItemOption>;
|
||||||
|
|
||||||
export const selectOptionsSchema = z
|
export const selectOptionsSchema = z
|
||||||
.array(selectOptionSchema)
|
.array(selectOptionSchema)
|
||||||
|
|||||||
@ -1,15 +1,14 @@
|
|||||||
import snakeCase from 'lodash.snakecase';
|
import snakeCase from 'lodash.snakecase';
|
||||||
|
|
||||||
export const getOptionValueFromLabel = (label: string) => {
|
import { computeOptionValueFromLabelOrThrow } from '~/pages/settings/data-model/utils/compute-option-value-from-label.utils';
|
||||||
// Remove accents
|
|
||||||
const unaccentedLabel = label
|
|
||||||
.normalize('NFD')
|
|
||||||
.replace(/[\u0300-\u036f]/g, '');
|
|
||||||
// Remove special characters
|
|
||||||
const noSpecialCharactersLabel = unaccentedLabel.replace(
|
|
||||||
/[^a-zA-Z0-9 ]/g,
|
|
||||||
'',
|
|
||||||
);
|
|
||||||
|
|
||||||
return snakeCase(noSpecialCharactersLabel).toUpperCase();
|
export const getOptionValueFromLabel = (label: string) => {
|
||||||
|
let transliteratedLabel = label;
|
||||||
|
try {
|
||||||
|
transliteratedLabel = computeOptionValueFromLabelOrThrow(label);
|
||||||
|
} catch (error) {
|
||||||
|
return label;
|
||||||
|
}
|
||||||
|
|
||||||
|
return snakeCase(transliteratedLabel).toUpperCase();
|
||||||
};
|
};
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
import { objectMetadataItemSchema } from '@/object-metadata/validation-schemas/objectMetadataItemSchema';
|
import { objectMetadataItemSchema } from '@/object-metadata/validation-schemas/objectMetadataItemSchema';
|
||||||
import { CreateObjectInput } from '~/generated-metadata/graphql';
|
import { CreateObjectInput } from '~/generated-metadata/graphql';
|
||||||
import { formatMetadataLabelToMetadataNameOrThrows } from '~/pages/settings/data-model/utils/format-metadata-label-to-name.util';
|
import { computeMetadataNameFromLabelOrThrow } from '~/pages/settings/data-model/utils/compute-metadata-name-from-label.utils';
|
||||||
|
|
||||||
export const settingsCreateObjectInputSchema = objectMetadataItemSchema
|
export const settingsCreateObjectInputSchema = objectMetadataItemSchema
|
||||||
.pick({
|
.pick({
|
||||||
@ -11,8 +11,6 @@ export const settingsCreateObjectInputSchema = objectMetadataItemSchema
|
|||||||
})
|
})
|
||||||
.transform<CreateObjectInput>((value) => ({
|
.transform<CreateObjectInput>((value) => ({
|
||||||
...value,
|
...value,
|
||||||
nameSingular: formatMetadataLabelToMetadataNameOrThrows(
|
nameSingular: computeMetadataNameFromLabelOrThrow(value.labelSingular),
|
||||||
value.labelSingular,
|
namePlural: computeMetadataNameFromLabelOrThrow(value.labelPlural),
|
||||||
),
|
|
||||||
namePlural: formatMetadataLabelToMetadataNameOrThrows(value.labelPlural),
|
|
||||||
}));
|
}));
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
import { objectMetadataItemSchema } from '@/object-metadata/validation-schemas/objectMetadataItemSchema';
|
import { objectMetadataItemSchema } from '@/object-metadata/validation-schemas/objectMetadataItemSchema';
|
||||||
import { UpdateObjectPayload } from '~/generated-metadata/graphql';
|
import { UpdateObjectPayload } from '~/generated-metadata/graphql';
|
||||||
import { formatMetadataLabelToMetadataNameOrThrows } from '~/pages/settings/data-model/utils/format-metadata-label-to-name.util';
|
import { computeMetadataNameFromLabelOrThrow } from '~/pages/settings/data-model/utils/compute-metadata-name-from-label.utils';
|
||||||
|
|
||||||
export const settingsUpdateObjectInputSchema = objectMetadataItemSchema
|
export const settingsUpdateObjectInputSchema = objectMetadataItemSchema
|
||||||
.pick({
|
.pick({
|
||||||
@ -16,9 +16,9 @@ export const settingsUpdateObjectInputSchema = objectMetadataItemSchema
|
|||||||
.transform<UpdateObjectPayload>((value) => ({
|
.transform<UpdateObjectPayload>((value) => ({
|
||||||
...value,
|
...value,
|
||||||
nameSingular: value.labelSingular
|
nameSingular: value.labelSingular
|
||||||
? formatMetadataLabelToMetadataNameOrThrows(value.labelSingular)
|
? computeMetadataNameFromLabelOrThrow(value.labelSingular)
|
||||||
: undefined,
|
: undefined,
|
||||||
namePlural: value.labelPlural
|
namePlural: value.labelPlural
|
||||||
? formatMetadataLabelToMetadataNameOrThrows(value.labelPlural)
|
? computeMetadataNameFromLabelOrThrow(value.labelPlural)
|
||||||
: undefined,
|
: undefined,
|
||||||
}));
|
}));
|
||||||
|
|||||||
@ -0,0 +1 @@
|
|||||||
|
export const METADATA_LABEL_VALID_PATTERN = /^[^0-9].*$/;
|
||||||
@ -0,0 +1 @@
|
|||||||
|
export const METADATA_NAME_VALID_PATTERN = /^[a-zA-Z][a-zA-Z0-9]*$/;
|
||||||
@ -0,0 +1 @@
|
|||||||
|
export const OPTION_VALUE_VALID_PATTERN = /^[a-zA-Z0-9]+$/;
|
||||||
@ -0,0 +1,27 @@
|
|||||||
|
import { computeMetadataNameFromLabelOrThrow } from '~/pages/settings/data-model/utils/compute-metadata-name-from-label.utils';
|
||||||
|
|
||||||
|
describe('computeMetadataNameFromLabel', () => {
|
||||||
|
it('throws if empty label', () => {
|
||||||
|
const label = ' ';
|
||||||
|
|
||||||
|
expect(() => computeMetadataNameFromLabelOrThrow(label)).toThrow();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('computes name for 1 char long label', () => {
|
||||||
|
const label = 'a';
|
||||||
|
|
||||||
|
expect(computeMetadataNameFromLabelOrThrow(label)).toEqual('a');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('throws if label starts with digits', () => {
|
||||||
|
const label = '1string';
|
||||||
|
|
||||||
|
expect(() => computeMetadataNameFromLabelOrThrow(label)).toThrow();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('computes name for label with non-latin char', () => {
|
||||||
|
const label = 'λλλ!';
|
||||||
|
|
||||||
|
expect(computeMetadataNameFromLabelOrThrow(label)).toEqual('lll');
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -0,0 +1,27 @@
|
|||||||
|
import { computeOptionValueFromLabelOrThrow } from '~/pages/settings/data-model/utils/compute-option-value-from-label.utils';
|
||||||
|
|
||||||
|
describe('computeOptionValueFromLabel', () => {
|
||||||
|
it('throws if empty label', () => {
|
||||||
|
const label = ' ';
|
||||||
|
|
||||||
|
expect(() => computeOptionValueFromLabelOrThrow(label)).toThrow();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('computes name for 1 char long label', () => {
|
||||||
|
const label = 'a';
|
||||||
|
|
||||||
|
expect(computeOptionValueFromLabelOrThrow(label)).toEqual('a');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('compute name if starts with digits', () => {
|
||||||
|
const label = '1';
|
||||||
|
|
||||||
|
expect(computeOptionValueFromLabelOrThrow(label)).toEqual('1');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('computes name for label with non-latin char', () => {
|
||||||
|
const label = 'λλλ';
|
||||||
|
|
||||||
|
expect(computeOptionValueFromLabelOrThrow(label)).toEqual('lll');
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -1,57 +0,0 @@
|
|||||||
import { formatMetadataLabelToMetadataNameOrThrows } from '~/pages/settings/data-model/utils/format-metadata-label-to-name.util';
|
|
||||||
|
|
||||||
const VALID_STRING_PATTERN = /^[a-zA-Z][a-zA-Z0-9 ]*$/;
|
|
||||||
|
|
||||||
describe('formatMetadataLabelToMetadataNameOrThrows', () => {
|
|
||||||
it('leaves strings unchanged if only latin characters', () => {
|
|
||||||
const input = 'testName';
|
|
||||||
|
|
||||||
expect(
|
|
||||||
formatMetadataLabelToMetadataNameOrThrows(input).match(
|
|
||||||
VALID_STRING_PATTERN,
|
|
||||||
)?.length,
|
|
||||||
).toBe(1);
|
|
||||||
expect(formatMetadataLabelToMetadataNameOrThrows(input)).toEqual(input);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('leaves strings unchanged if only latin characters and digits', () => {
|
|
||||||
const input = 'testName123';
|
|
||||||
|
|
||||||
expect(
|
|
||||||
formatMetadataLabelToMetadataNameOrThrows(input).match(
|
|
||||||
VALID_STRING_PATTERN,
|
|
||||||
)?.length,
|
|
||||||
).toBe(1);
|
|
||||||
expect(formatMetadataLabelToMetadataNameOrThrows(input)).toEqual(input);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('format strings with non latin characters', () => {
|
|
||||||
const input = 'בְרִבְרִ';
|
|
||||||
const expected = 'bRibRi';
|
|
||||||
|
|
||||||
expect(
|
|
||||||
formatMetadataLabelToMetadataNameOrThrows(input).match(
|
|
||||||
VALID_STRING_PATTERN,
|
|
||||||
)?.length,
|
|
||||||
).toBe(1);
|
|
||||||
expect(formatMetadataLabelToMetadataNameOrThrows(input)).toEqual(expected);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('format strings with mixed characters', () => {
|
|
||||||
const input = 'aa2בְרִבְרִ';
|
|
||||||
const expected = 'aa2BRibRi';
|
|
||||||
|
|
||||||
expect(
|
|
||||||
formatMetadataLabelToMetadataNameOrThrows(input).match(
|
|
||||||
VALID_STRING_PATTERN,
|
|
||||||
)?.length,
|
|
||||||
).toBe(1);
|
|
||||||
expect(formatMetadataLabelToMetadataNameOrThrows(input)).toEqual(expected);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('throws error if could not format', () => {
|
|
||||||
const input = '$$$***';
|
|
||||||
|
|
||||||
expect(() => formatMetadataLabelToMetadataNameOrThrows(input)).toThrow();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@ -0,0 +1,6 @@
|
|||||||
|
import { METADATA_NAME_VALID_PATTERN } from '~/pages/settings/data-model/constants/MetadataNameValidPattern';
|
||||||
|
import { transliterateAndFormatOrThrow } from '~/pages/settings/data-model/utils/transliterate-and-format.utils';
|
||||||
|
|
||||||
|
export const computeMetadataNameFromLabelOrThrow = (label: string): string => {
|
||||||
|
return transliterateAndFormatOrThrow(label, METADATA_NAME_VALID_PATTERN);
|
||||||
|
};
|
||||||
@ -0,0 +1,6 @@
|
|||||||
|
import { OPTION_VALUE_VALID_PATTERN } from '~/pages/settings/data-model/constants/OptionValueValidPattern';
|
||||||
|
import { transliterateAndFormatOrThrow } from '~/pages/settings/data-model/utils/transliterate-and-format.utils';
|
||||||
|
|
||||||
|
export const computeOptionValueFromLabelOrThrow = (label: string): string => {
|
||||||
|
return transliterateAndFormatOrThrow(label, OPTION_VALUE_VALID_PATTERN);
|
||||||
|
};
|
||||||
@ -3,14 +3,13 @@ import { slugify, transliterate } from 'transliteration';
|
|||||||
|
|
||||||
import { isDefined } from '~/utils/isDefined';
|
import { isDefined } from '~/utils/isDefined';
|
||||||
|
|
||||||
const VALID_STRING_PATTERN = /^[a-zA-Z][a-zA-Z0-9 ]*$/;
|
export const transliterateAndFormatOrThrow = (
|
||||||
|
|
||||||
export const formatMetadataLabelToMetadataNameOrThrows = (
|
|
||||||
string: string,
|
string: string,
|
||||||
|
validStringPattern: RegExp,
|
||||||
): string => {
|
): string => {
|
||||||
let formattedString = string;
|
let formattedString = string;
|
||||||
|
|
||||||
if (isDefined(formattedString.match(VALID_STRING_PATTERN))) {
|
if (isDefined(formattedString.match(validStringPattern))) {
|
||||||
return toCamelCase(formattedString);
|
return toCamelCase(formattedString);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -18,7 +17,7 @@ export const formatMetadataLabelToMetadataNameOrThrows = (
|
|||||||
slugify(transliterate(formattedString, { trim: true })),
|
slugify(transliterate(formattedString, { trim: true })),
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!formattedString.match(VALID_STRING_PATTERN)) {
|
if (!formattedString.match(validStringPattern)) {
|
||||||
throw new Error(`"${string}" is not a valid name`);
|
throw new Error(`"${string}" is not a valid name`);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -4,7 +4,7 @@ import {
|
|||||||
ValidationArguments,
|
ValidationArguments,
|
||||||
} from 'class-validator';
|
} from 'class-validator';
|
||||||
|
|
||||||
const graphQLEnumNameRegex = /^[_A-Za-z][_0-9A-Za-z]+$/;
|
const graphQLEnumNameRegex = /^[_A-Za-z][_0-9A-Za-z]*$/;
|
||||||
|
|
||||||
export function IsValidGraphQLEnumName(validationOptions?: ValidationOptions) {
|
export function IsValidGraphQLEnumName(validationOptions?: ValidationOptions) {
|
||||||
return function (object: object, propertyName: string) {
|
return function (object: object, propertyName: string) {
|
||||||
|
|||||||
Reference in New Issue
Block a user