Remove number from label identifier list (#12831)
This commit is contained in:
@ -3,6 +3,7 @@ import { DropdownContent } from '@/ui/layout/dropdown/components/DropdownContent
|
|||||||
import { DropdownMenuItemsContainer } from '@/ui/layout/dropdown/components/DropdownMenuItemsContainer';
|
import { DropdownMenuItemsContainer } from '@/ui/layout/dropdown/components/DropdownMenuItemsContainer';
|
||||||
import { GenericDropdownContentWidth } from '@/ui/layout/dropdown/constants/GenericDropdownContentWidth';
|
import { GenericDropdownContentWidth } from '@/ui/layout/dropdown/constants/GenericDropdownContentWidth';
|
||||||
import { useDropdown } from '@/ui/layout/dropdown/hooks/useDropdown';
|
import { useDropdown } from '@/ui/layout/dropdown/hooks/useDropdown';
|
||||||
|
import { isDefined } from 'twenty-shared/utils';
|
||||||
import {
|
import {
|
||||||
IconArchive,
|
IconArchive,
|
||||||
IconDotsVertical,
|
IconDotsVertical,
|
||||||
@ -65,14 +66,14 @@ export const SettingsObjectFieldActiveActionDropdown = ({
|
|||||||
LeftIcon={isCustomField ? IconPencil : IconEye}
|
LeftIcon={isCustomField ? IconPencil : IconEye}
|
||||||
onClick={handleEdit}
|
onClick={handleEdit}
|
||||||
/>
|
/>
|
||||||
{!!onSetAsLabelIdentifier && (
|
{isDefined(onSetAsLabelIdentifier) && (
|
||||||
<MenuItem
|
<MenuItem
|
||||||
text="Set as record text"
|
text="Set as record text"
|
||||||
LeftIcon={IconTextSize}
|
LeftIcon={IconTextSize}
|
||||||
onClick={handleSetAsLabelIdentifier}
|
onClick={handleSetAsLabelIdentifier}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{!!onDeactivate && (
|
{isDefined(onDeactivate) && (
|
||||||
<MenuItem
|
<MenuItem
|
||||||
text="Deactivate"
|
text="Deactivate"
|
||||||
LeftIcon={IconArchive}
|
LeftIcon={IconArchive}
|
||||||
|
|||||||
@ -1,4 +1,3 @@
|
|||||||
import { LABEL_IDENTIFIER_FIELD_METADATA_TYPES } from '@/object-metadata/constants/LabelIdentifierFieldMetadataTypes';
|
|
||||||
import { useFieldMetadataItem } from '@/object-metadata/hooks/useFieldMetadataItem';
|
import { useFieldMetadataItem } from '@/object-metadata/hooks/useFieldMetadataItem';
|
||||||
import { useGetRelationMetadata } from '@/object-metadata/hooks/useGetRelationMetadata';
|
import { useGetRelationMetadata } from '@/object-metadata/hooks/useGetRelationMetadata';
|
||||||
import { useUpdateOneObjectMetadataItem } from '@/object-metadata/hooks/useUpdateOneObjectMetadataItem';
|
import { useUpdateOneObjectMetadataItem } from '@/object-metadata/hooks/useUpdateOneObjectMetadataItem';
|
||||||
@ -19,7 +18,10 @@ import { useTheme } from '@emotion/react';
|
|||||||
import styled from '@emotion/styled';
|
import styled from '@emotion/styled';
|
||||||
import { useMemo } from 'react';
|
import { useMemo } from 'react';
|
||||||
import { useRecoilState, useRecoilValue } from 'recoil';
|
import { useRecoilState, useRecoilValue } from 'recoil';
|
||||||
import { isDefined } from 'twenty-shared/utils';
|
import {
|
||||||
|
isDefined,
|
||||||
|
isLabelIdentifierFieldMetadataTypes,
|
||||||
|
} from 'twenty-shared/utils';
|
||||||
import { IconMinus, IconPlus, useIcons } from 'twenty-ui/display';
|
import { IconMinus, IconPlus, useIcons } from 'twenty-ui/display';
|
||||||
import { LightIconButton } from 'twenty-ui/input';
|
import { LightIconButton } from 'twenty-ui/input';
|
||||||
import { UndecoratedLink } from 'twenty-ui/navigation';
|
import { UndecoratedLink } from 'twenty-ui/navigation';
|
||||||
@ -102,7 +104,7 @@ export const SettingsObjectFieldItemTableRow = ({
|
|||||||
const canBeSetAsLabelIdentifier =
|
const canBeSetAsLabelIdentifier =
|
||||||
objectMetadataItem.isCustom &&
|
objectMetadataItem.isCustom &&
|
||||||
!isLabelIdentifier &&
|
!isLabelIdentifier &&
|
||||||
LABEL_IDENTIFIER_FIELD_METADATA_TYPES.includes(fieldMetadataItem.type);
|
isLabelIdentifierFieldMetadataTypes(fieldMetadataItem.type);
|
||||||
|
|
||||||
const linkToNavigate = getSettingsPath(SettingsPath.ObjectFieldEdit, {
|
const linkToNavigate = getSettingsPath(SettingsPath.ObjectFieldEdit, {
|
||||||
objectNamePlural: objectMetadataItem.namePlural,
|
objectNamePlural: objectMetadataItem.namePlural,
|
||||||
|
|||||||
@ -3,7 +3,6 @@ import { useMemo } from 'react';
|
|||||||
import { Controller, useForm } from 'react-hook-form';
|
import { Controller, useForm } from 'react-hook-form';
|
||||||
import { ZodError, isDirty, z } from 'zod';
|
import { ZodError, isDirty, z } from 'zod';
|
||||||
|
|
||||||
import { LABEL_IDENTIFIER_FIELD_METADATA_TYPES } from '@/object-metadata/constants/LabelIdentifierFieldMetadataTypes';
|
|
||||||
import { useUpdateOneObjectMetadataItem } from '@/object-metadata/hooks/useUpdateOneObjectMetadataItem';
|
import { useUpdateOneObjectMetadataItem } from '@/object-metadata/hooks/useUpdateOneObjectMetadataItem';
|
||||||
import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem';
|
import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem';
|
||||||
import { getActiveFieldMetadataItems } from '@/object-metadata/utils/getActiveFieldMetadataItems';
|
import { getActiveFieldMetadataItems } from '@/object-metadata/utils/getActiveFieldMetadataItems';
|
||||||
@ -14,6 +13,7 @@ import { Select } from '@/ui/input/components/Select';
|
|||||||
import { zodResolver } from '@hookform/resolvers/zod';
|
import { zodResolver } from '@hookform/resolvers/zod';
|
||||||
import { t } from '@lingui/core/macro';
|
import { t } from '@lingui/core/macro';
|
||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
|
import { isLabelIdentifierFieldMetadataTypes } from 'twenty-shared/utils';
|
||||||
import { IconCircleOff, IconPlus, useIcons } from 'twenty-ui/display';
|
import { IconCircleOff, IconPlus, useIcons } from 'twenty-ui/display';
|
||||||
import { SelectOption } from 'twenty-ui/input';
|
import { SelectOption } from 'twenty-ui/input';
|
||||||
|
|
||||||
@ -84,7 +84,7 @@ export const SettingsDataModelObjectIdentifiersForm = ({
|
|||||||
getActiveFieldMetadataItems(objectMetadataItem)
|
getActiveFieldMetadataItems(objectMetadataItem)
|
||||||
.filter(
|
.filter(
|
||||||
({ id, type }) =>
|
({ id, type }) =>
|
||||||
LABEL_IDENTIFIER_FIELD_METADATA_TYPES.includes(type) ||
|
isLabelIdentifierFieldMetadataTypes(type) ||
|
||||||
objectMetadataItem.labelIdentifierFieldMetadataId === id,
|
objectMetadataItem.labelIdentifierFieldMetadataId === id,
|
||||||
)
|
)
|
||||||
.map<SelectOption<string | null>>((fieldMetadataItem) => ({
|
.map<SelectOption<string | null>>((fieldMetadataItem) => ({
|
||||||
|
|||||||
@ -1,12 +1,14 @@
|
|||||||
import { Field, InputType } from '@nestjs/graphql';
|
import { Field, InputType } from '@nestjs/graphql';
|
||||||
|
|
||||||
import { BeforeUpdateOne } from '@ptc-org/nestjs-query-graphql';
|
import { BeforeUpdateOne } from '@ptc-org/nestjs-query-graphql';
|
||||||
|
import { Type } from 'class-transformer';
|
||||||
import {
|
import {
|
||||||
IsBoolean,
|
IsBoolean,
|
||||||
IsNotEmpty,
|
IsNotEmpty,
|
||||||
IsOptional,
|
IsOptional,
|
||||||
IsString,
|
IsString,
|
||||||
IsUUID,
|
IsUUID,
|
||||||
|
ValidateNested,
|
||||||
} from 'class-validator';
|
} from 'class-validator';
|
||||||
|
|
||||||
import { UUIDScalarType } from 'src/engine/api/graphql/workspace-schema-builder/graphql-types/scalars';
|
import { UUIDScalarType } from 'src/engine/api/graphql/workspace-schema-builder/graphql-types/scalars';
|
||||||
@ -76,6 +78,8 @@ export class UpdateObjectPayload {
|
|||||||
@InputType()
|
@InputType()
|
||||||
@BeforeUpdateOne(BeforeUpdateOneObject)
|
@BeforeUpdateOne(BeforeUpdateOneObject)
|
||||||
export class UpdateOneObjectInput {
|
export class UpdateOneObjectInput {
|
||||||
|
@Type(() => UpdateObjectPayload)
|
||||||
|
@ValidateNested()
|
||||||
@Field(() => UpdateObjectPayload)
|
@Field(() => UpdateObjectPayload)
|
||||||
update: UpdateObjectPayload;
|
update: UpdateObjectPayload;
|
||||||
|
|
||||||
|
|||||||
@ -656,34 +656,6 @@ describe('BeforeUpdateOneObject', () => {
|
|||||||
expect(result).toEqual(instance);
|
expect(result).toEqual(instance);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw BadRequestException if label identifier field does not exist', async () => {
|
|
||||||
const labelIdentifierFieldId = 'nonexistent-field-id';
|
|
||||||
const instance: UpdateOneInputType<UpdateObjectPayloadForTest> = {
|
|
||||||
id: mockObjectId,
|
|
||||||
update: {
|
|
||||||
labelIdentifierFieldMetadataId: labelIdentifierFieldId,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
const mockObject: Partial<ObjectMetadataEntity> = {
|
|
||||||
id: mockObjectId,
|
|
||||||
isCustom: true,
|
|
||||||
};
|
|
||||||
|
|
||||||
jest
|
|
||||||
.spyOn(objectMetadataService, 'findOneWithinWorkspace')
|
|
||||||
.mockResolvedValue(mockObject as ObjectMetadataEntity);
|
|
||||||
|
|
||||||
jest.spyOn(fieldMetadataRepository, 'findBy').mockResolvedValue([]);
|
|
||||||
|
|
||||||
await expect(
|
|
||||||
hook.run(instance as UpdateOneInputType<UpdateObjectPayload>, {
|
|
||||||
workspaceId: mockWorkspaceId,
|
|
||||||
locale: undefined,
|
|
||||||
}),
|
|
||||||
).rejects.toThrow('This label identifier does not exist');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should validate image identifier field correctly for custom objects', async () => {
|
it('should validate image identifier field correctly for custom objects', async () => {
|
||||||
const imageIdentifierFieldId = 'image-field-id';
|
const imageIdentifierFieldId = 'image-field-id';
|
||||||
const instance: UpdateOneInputType<UpdateObjectPayloadForTest> = {
|
const instance: UpdateOneInputType<UpdateObjectPayloadForTest> = {
|
||||||
@ -722,32 +694,4 @@ describe('BeforeUpdateOneObject', () => {
|
|||||||
|
|
||||||
expect(result).toEqual(instance);
|
expect(result).toEqual(instance);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw BadRequestException if image identifier field does not exist', async () => {
|
|
||||||
const imageIdentifierFieldId = 'nonexistent-field-id';
|
|
||||||
const instance: UpdateOneInputType<UpdateObjectPayloadForTest> = {
|
|
||||||
id: mockObjectId,
|
|
||||||
update: {
|
|
||||||
imageIdentifierFieldMetadataId: imageIdentifierFieldId,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
const mockObject: Partial<ObjectMetadataEntity> = {
|
|
||||||
id: mockObjectId,
|
|
||||||
isCustom: true,
|
|
||||||
};
|
|
||||||
|
|
||||||
jest
|
|
||||||
.spyOn(objectMetadataService, 'findOneWithinWorkspace')
|
|
||||||
.mockResolvedValue(mockObject as ObjectMetadataEntity);
|
|
||||||
|
|
||||||
jest.spyOn(fieldMetadataRepository, 'findBy').mockResolvedValue([]);
|
|
||||||
|
|
||||||
await expect(
|
|
||||||
hook.run(instance as UpdateOneInputType<UpdateObjectPayload>, {
|
|
||||||
workspaceId: mockWorkspaceId,
|
|
||||||
locale: undefined,
|
|
||||||
}),
|
|
||||||
).rejects.toThrow('This image identifier does not exist');
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|||||||
@ -12,7 +12,7 @@ import {
|
|||||||
} from '@ptc-org/nestjs-query-graphql';
|
} from '@ptc-org/nestjs-query-graphql';
|
||||||
import { APP_LOCALES, SOURCE_LOCALE } from 'twenty-shared/translations';
|
import { APP_LOCALES, SOURCE_LOCALE } from 'twenty-shared/translations';
|
||||||
import { isDefined } from 'twenty-shared/utils';
|
import { isDefined } from 'twenty-shared/utils';
|
||||||
import { Equal, In, Repository } from 'typeorm';
|
import { Repository } from 'typeorm';
|
||||||
|
|
||||||
import { generateMessageId } from 'src/engine/core-modules/i18n/utils/generateMessageId';
|
import { generateMessageId } from 'src/engine/core-modules/i18n/utils/generateMessageId';
|
||||||
import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity';
|
import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity';
|
||||||
@ -57,8 +57,6 @@ export class BeforeUpdateOneObject<T extends UpdateObjectPayload>
|
|||||||
return this.handleStandardObjectUpdate(instance, objectMetadata, locale);
|
return this.handleStandardObjectUpdate(instance, objectMetadata, locale);
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.validateIdentifierFields(instance, workspaceId);
|
|
||||||
|
|
||||||
return instance;
|
return instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -437,56 +435,4 @@ export class BeforeUpdateOneObject<T extends UpdateObjectPayload>
|
|||||||
locale,
|
locale,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
private async validateIdentifierFields(
|
|
||||||
instance: UpdateOneInputType<T>,
|
|
||||||
workspaceId: string,
|
|
||||||
): Promise<void> {
|
|
||||||
if (
|
|
||||||
!instance.update.labelIdentifierFieldMetadataId &&
|
|
||||||
!instance.update.imageIdentifierFieldMetadataId
|
|
||||||
) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const fields = await this.fieldMetadataRepository.findBy({
|
|
||||||
workspaceId: Equal(workspaceId),
|
|
||||||
objectMetadataId: Equal(instance.id.toString()),
|
|
||||||
id: In(
|
|
||||||
[
|
|
||||||
instance.update.labelIdentifierFieldMetadataId,
|
|
||||||
instance.update.imageIdentifierFieldMetadataId,
|
|
||||||
].filter((id) => id !== null),
|
|
||||||
),
|
|
||||||
});
|
|
||||||
|
|
||||||
const fieldIds = fields.map((field) => field.id);
|
|
||||||
|
|
||||||
this.validateLabelIdentifier(instance, fieldIds);
|
|
||||||
this.validateImageIdentifier(instance, fieldIds);
|
|
||||||
}
|
|
||||||
|
|
||||||
private validateLabelIdentifier(
|
|
||||||
instance: UpdateOneInputType<T>,
|
|
||||||
fieldIds: string[],
|
|
||||||
): void {
|
|
||||||
if (
|
|
||||||
instance.update.labelIdentifierFieldMetadataId &&
|
|
||||||
!fieldIds.includes(instance.update.labelIdentifierFieldMetadataId)
|
|
||||||
) {
|
|
||||||
throw new BadRequestException('This label identifier does not exist');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private validateImageIdentifier(
|
|
||||||
instance: UpdateOneInputType<T>,
|
|
||||||
fieldIds: string[],
|
|
||||||
): void {
|
|
||||||
if (
|
|
||||||
instance.update.imageIdentifierFieldMetadataId &&
|
|
||||||
!fieldIds.includes(instance.update.imageIdentifierFieldMetadataId)
|
|
||||||
) {
|
|
||||||
throw new BadRequestException('This image identifier does not exist');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -34,6 +34,7 @@ import {
|
|||||||
import { RemoteTableRelationsService } from 'src/engine/metadata-modules/remote-server/remote-table/remote-table-relations/remote-table-relations.service';
|
import { RemoteTableRelationsService } from 'src/engine/metadata-modules/remote-server/remote-table/remote-table-relations/remote-table-relations.service';
|
||||||
import { SearchVectorService } from 'src/engine/metadata-modules/search-vector/search-vector.service';
|
import { SearchVectorService } from 'src/engine/metadata-modules/search-vector/search-vector.service';
|
||||||
import { ObjectMetadataItemWithFieldMaps } from 'src/engine/metadata-modules/types/object-metadata-item-with-field-maps';
|
import { ObjectMetadataItemWithFieldMaps } from 'src/engine/metadata-modules/types/object-metadata-item-with-field-maps';
|
||||||
|
import { validateMetadataIdentifierFieldMetadataIds } from 'src/engine/metadata-modules/utils/validate-metadata-identifier-field-metadata-id.utils';
|
||||||
import { validateNameAndLabelAreSyncOrThrow } from 'src/engine/metadata-modules/utils/validate-name-and-label-are-sync-or-throw.util';
|
import { validateNameAndLabelAreSyncOrThrow } from 'src/engine/metadata-modules/utils/validate-name-and-label-are-sync-or-throw.util';
|
||||||
import { validatesNoOtherObjectWithSameNameExistsOrThrows } from 'src/engine/metadata-modules/utils/validate-no-other-object-with-same-name-exists-or-throw.util';
|
import { validatesNoOtherObjectWithSameNameExistsOrThrows } from 'src/engine/metadata-modules/utils/validate-no-other-object-with-same-name-exists-or-throw.util';
|
||||||
import { WorkspaceMetadataCacheService } from 'src/engine/metadata-modules/workspace-metadata-cache/services/workspace-metadata-cache.service';
|
import { WorkspaceMetadataCacheService } from 'src/engine/metadata-modules/workspace-metadata-cache/services/workspace-metadata-cache.service';
|
||||||
@ -300,6 +301,14 @@ export class ObjectMetadataService extends TypeOrmQueryService<ObjectMetadataEnt
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
validateMetadataIdentifierFieldMetadataIds({
|
||||||
|
fieldMetadataItems: Object.values(existingObjectMetadata.fieldsById),
|
||||||
|
labelIdentifierFieldMetadataId:
|
||||||
|
inputPayload.labelIdentifierFieldMetadataId,
|
||||||
|
imageIdentifierFieldMetadataId:
|
||||||
|
inputPayload.imageIdentifierFieldMetadataId,
|
||||||
|
});
|
||||||
|
|
||||||
const updatedObject = await super.updateOne(inputId, inputPayload);
|
const updatedObject = await super.updateOne(inputId, inputPayload);
|
||||||
|
|
||||||
await this.handleObjectNameAndLabelUpdates(
|
await this.handleObjectNameAndLabelUpdates(
|
||||||
|
|||||||
@ -0,0 +1,94 @@
|
|||||||
|
import {
|
||||||
|
isDefined,
|
||||||
|
isLabelIdentifierFieldMetadataTypes,
|
||||||
|
} from 'twenty-shared/utils';
|
||||||
|
|
||||||
|
import { FieldMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/field-metadata.interface';
|
||||||
|
|
||||||
|
import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity';
|
||||||
|
import {
|
||||||
|
ObjectMetadataException,
|
||||||
|
ObjectMetadataExceptionCode,
|
||||||
|
} from 'src/engine/metadata-modules/object-metadata/object-metadata.exception';
|
||||||
|
|
||||||
|
type Validator = {
|
||||||
|
validator: (args: {
|
||||||
|
fieldMetadataId: string;
|
||||||
|
matchingFieldMetadata?: FieldMetadataEntity | FieldMetadataInterface;
|
||||||
|
}) => boolean;
|
||||||
|
label: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
type ValidateMetadataIdentifierFieldMetadataIdOrThrowArgs = {
|
||||||
|
fieldMetadataId: string;
|
||||||
|
fieldMetadataItems: FieldMetadataEntity[] | FieldMetadataInterface[];
|
||||||
|
validators: Validator[];
|
||||||
|
};
|
||||||
|
const validatorRunner = ({
|
||||||
|
fieldMetadataId,
|
||||||
|
fieldMetadataItems,
|
||||||
|
validators,
|
||||||
|
}: ValidateMetadataIdentifierFieldMetadataIdOrThrowArgs): void => {
|
||||||
|
const matchingFieldMetadata = fieldMetadataItems.find(
|
||||||
|
(fieldMetadata) => fieldMetadata.id === fieldMetadataId,
|
||||||
|
);
|
||||||
|
|
||||||
|
validators.forEach(({ label, validator }) => {
|
||||||
|
if (validator({ fieldMetadataId, matchingFieldMetadata })) {
|
||||||
|
throw new ObjectMetadataException(
|
||||||
|
label,
|
||||||
|
ObjectMetadataExceptionCode.INVALID_OBJECT_INPUT,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
type ValidateMetadataIdentifierFieldMetadataIdsArgs = {
|
||||||
|
labelIdentifierFieldMetadataId: string | undefined;
|
||||||
|
imageIdentifierFieldMetadataId: string | undefined;
|
||||||
|
fieldMetadataItems: FieldMetadataEntity[] | FieldMetadataInterface[];
|
||||||
|
};
|
||||||
|
export const validateMetadataIdentifierFieldMetadataIds = ({
|
||||||
|
imageIdentifierFieldMetadataId,
|
||||||
|
labelIdentifierFieldMetadataId,
|
||||||
|
fieldMetadataItems,
|
||||||
|
}: ValidateMetadataIdentifierFieldMetadataIdsArgs) => {
|
||||||
|
const isMatchingFieldMetadataDefined: Validator['validator'] = ({
|
||||||
|
matchingFieldMetadata,
|
||||||
|
}) => !isDefined(matchingFieldMetadata);
|
||||||
|
|
||||||
|
if (isDefined(labelIdentifierFieldMetadataId)) {
|
||||||
|
validatorRunner({
|
||||||
|
fieldMetadataId: labelIdentifierFieldMetadataId,
|
||||||
|
fieldMetadataItems,
|
||||||
|
validators: [
|
||||||
|
{
|
||||||
|
validator: isMatchingFieldMetadataDefined,
|
||||||
|
label:
|
||||||
|
'labelIdentifierFieldMetadataId validation failed: related field metadata not found',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
validator: ({ matchingFieldMetadata }) =>
|
||||||
|
isDefined(matchingFieldMetadata) &&
|
||||||
|
!isLabelIdentifierFieldMetadataTypes(matchingFieldMetadata.type),
|
||||||
|
label:
|
||||||
|
'labelIdentifierFieldMetadataId validation failed: it must be a TEXT or FULL_NAME field metadata type id',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isDefined(imageIdentifierFieldMetadataId)) {
|
||||||
|
validatorRunner({
|
||||||
|
fieldMetadataId: imageIdentifierFieldMetadataId,
|
||||||
|
fieldMetadataItems,
|
||||||
|
validators: [
|
||||||
|
{
|
||||||
|
validator: isMatchingFieldMetadataDefined,
|
||||||
|
label:
|
||||||
|
'imageIdentifierFieldMetadataId validation failed: related field metadata not found',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
@ -0,0 +1,34 @@
|
|||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`Object metadata update should fail when labelIdentifier is not a TEXT or NAME field 1`] = `
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"extensions": {
|
||||||
|
"code": "BAD_USER_INPUT",
|
||||||
|
},
|
||||||
|
"message": "labelIdentifierFieldMetadataId validation failed: it must be a TEXT or FULL_NAME field metadata type id",
|
||||||
|
},
|
||||||
|
]
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`Object metadata update should fail when labelIdentifier is not a known field metadata id 1`] = `
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"extensions": {
|
||||||
|
"code": "BAD_USER_INPUT",
|
||||||
|
},
|
||||||
|
"message": "labelIdentifierFieldMetadataId validation failed: related field metadata not found",
|
||||||
|
},
|
||||||
|
]
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`Object metadata update should fail when labelIdentifier is not a uuid 1`] = `
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"extensions": {
|
||||||
|
"code": "BAD_USER_INPUT",
|
||||||
|
},
|
||||||
|
"message": "labelIdentifierFieldMetadataId must be a UUID",
|
||||||
|
},
|
||||||
|
]
|
||||||
|
`;
|
||||||
@ -0,0 +1,96 @@
|
|||||||
|
import { createOneFieldMetadata } from 'test/integration/metadata/suites/field-metadata/utils/create-one-field-metadata.util';
|
||||||
|
import { createOneObjectMetadata } from 'test/integration/metadata/suites/object-metadata/utils/create-one-object-metadata.util';
|
||||||
|
import { deleteOneObjectMetadata } from 'test/integration/metadata/suites/object-metadata/utils/delete-one-object-metadata.util';
|
||||||
|
import { getMockCreateObjectInput } from 'test/integration/metadata/suites/object-metadata/utils/generate-mock-create-object-metadata-input';
|
||||||
|
import { updateOneObjectMetadata } from 'test/integration/metadata/suites/object-metadata/utils/update-one-object-metadata.util';
|
||||||
|
import { EachTestingContext } from 'twenty-shared/testing';
|
||||||
|
import { FieldMetadataType } from 'twenty-shared/types';
|
||||||
|
|
||||||
|
import { UpdateObjectPayload } from 'src/engine/metadata-modules/object-metadata/dtos/update-object.input';
|
||||||
|
|
||||||
|
type TestingRuntimeContext = {
|
||||||
|
objectMetadataId: string;
|
||||||
|
numberFieldMetadataId: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
type CreateOneObjectMetadataItemTestingContext = EachTestingContext<
|
||||||
|
| ((args: TestingRuntimeContext) => Partial<UpdateObjectPayload>)
|
||||||
|
| Partial<UpdateObjectPayload>
|
||||||
|
>[];
|
||||||
|
|
||||||
|
const labelIdentifierFailingTestsUseCase: CreateOneObjectMetadataItemTestingContext =
|
||||||
|
[
|
||||||
|
{
|
||||||
|
title: 'when labelIdentifier is not a uuid',
|
||||||
|
context: {
|
||||||
|
labelIdentifierFieldMetadataId: 'not-a-uuid',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'when labelIdentifier is not a known field metadata id',
|
||||||
|
context: {
|
||||||
|
labelIdentifierFieldMetadataId: '42422020-f49c-4159-8751-76a24f47b360',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'when labelIdentifier is not a TEXT or NAME field',
|
||||||
|
context: ({ numberFieldMetadataId }) => ({
|
||||||
|
labelIdentifierFieldMetadataId: numberFieldMetadataId,
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const allTestsUseCases = [...labelIdentifierFailingTestsUseCase];
|
||||||
|
|
||||||
|
describe('Object metadata update should fail', () => {
|
||||||
|
let objectMetadataId: string;
|
||||||
|
let numberFieldMetadataId: string;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
const { data } = await createOneObjectMetadata({
|
||||||
|
input: getMockCreateObjectInput(),
|
||||||
|
});
|
||||||
|
|
||||||
|
objectMetadataId = data.createOneObject.id;
|
||||||
|
|
||||||
|
const {
|
||||||
|
data: { createOneField },
|
||||||
|
} = await createOneFieldMetadata({
|
||||||
|
input: {
|
||||||
|
objectMetadataId: objectMetadataId,
|
||||||
|
name: 'testName',
|
||||||
|
label: 'Test name',
|
||||||
|
isLabelSyncedWithName: true,
|
||||||
|
type: FieldMetadataType.NUMBER,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
numberFieldMetadataId = createOneField.id;
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(async () => {
|
||||||
|
await deleteOneObjectMetadata({
|
||||||
|
input: {
|
||||||
|
idToDelete: objectMetadataId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it.each(allTestsUseCases)('$title', async ({ context }) => {
|
||||||
|
const updatePayload =
|
||||||
|
typeof context === 'function'
|
||||||
|
? context({ numberFieldMetadataId, objectMetadataId })
|
||||||
|
: context;
|
||||||
|
|
||||||
|
const { errors } = await updateOneObjectMetadata({
|
||||||
|
input: {
|
||||||
|
idToUpdate: objectMetadataId,
|
||||||
|
updatePayload,
|
||||||
|
},
|
||||||
|
expectToFail: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(errors).toBeDefined();
|
||||||
|
expect(errors).toMatchSnapshot();
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -1,7 +1,6 @@
|
|||||||
import { FieldMetadataType } from '~/generated-metadata/graphql';
|
import { FieldMetadataType } from "@/types";
|
||||||
|
|
||||||
export const LABEL_IDENTIFIER_FIELD_METADATA_TYPES = [
|
export const LABEL_IDENTIFIER_FIELD_METADATA_TYPES = [
|
||||||
FieldMetadataType.NUMBER,
|
|
||||||
FieldMetadataType.TEXT,
|
FieldMetadataType.TEXT,
|
||||||
FieldMetadataType.FULL_NAME,
|
FieldMetadataType.FULL_NAME,
|
||||||
];
|
];
|
||||||
@ -10,6 +10,7 @@
|
|||||||
export { FIELD_FOR_TOTAL_COUNT_AGGREGATE_OPERATION } from './FieldForTotalCountAggregateOperation';
|
export { FIELD_FOR_TOTAL_COUNT_AGGREGATE_OPERATION } from './FieldForTotalCountAggregateOperation';
|
||||||
export { MAX_OPTIONS_TO_DISPLAY } from './FieldMetadataMaxOptionsToDisplay';
|
export { MAX_OPTIONS_TO_DISPLAY } from './FieldMetadataMaxOptionsToDisplay';
|
||||||
export { FIELD_RESTRICTED_ADDITIONAL_PERMISSIONS_REQUIRED } from './FieldRestrictedAdditionalPermissionsRequired';
|
export { FIELD_RESTRICTED_ADDITIONAL_PERMISSIONS_REQUIRED } from './FieldRestrictedAdditionalPermissionsRequired';
|
||||||
|
export { LABEL_IDENTIFIER_FIELD_METADATA_TYPES } from './LabelIdentifierFieldMetadataTypes';
|
||||||
export { PermissionsOnAllObjectRecords } from './PermissionsOnAllObjectRecords';
|
export { PermissionsOnAllObjectRecords } from './PermissionsOnAllObjectRecords';
|
||||||
export { QUERY_MAX_RECORDS } from './QueryMaxRecords';
|
export { QUERY_MAX_RECORDS } from './QueryMaxRecords';
|
||||||
export { STANDARD_OBJECT_RECORDS_UNDER_OBJECT_RECORDS_PERMISSIONS } from './StandardObjectRecordsUnderObjectRecordsPermissions';
|
export { STANDARD_OBJECT_RECORDS_UNDER_OBJECT_RECORDS_PERMISSIONS } from './StandardObjectRecordsUnderObjectRecordsPermissions';
|
||||||
|
|||||||
@ -25,6 +25,7 @@ export { getUrlHostnameOrThrow } from './url/getUrlHostnameOrThrow';
|
|||||||
export { isValidHostname } from './url/isValidHostname';
|
export { isValidHostname } from './url/isValidHostname';
|
||||||
export { isValidUrl } from './url/isValidUrl';
|
export { isValidUrl } from './url/isValidUrl';
|
||||||
export { isDefined } from './validation/isDefined';
|
export { isDefined } from './validation/isDefined';
|
||||||
|
export { isLabelIdentifierFieldMetadataTypes } from './validation/isLabelIdentifierFieldMetadataTypes';
|
||||||
export { isValidLocale } from './validation/isValidLocale';
|
export { isValidLocale } from './validation/isValidLocale';
|
||||||
export { isValidUuid } from './validation/isValidUuid';
|
export { isValidUuid } from './validation/isValidUuid';
|
||||||
export { isValidVariable } from './validation/isValidVariable';
|
export { isValidVariable } from './validation/isValidVariable';
|
||||||
|
|||||||
@ -0,0 +1,7 @@
|
|||||||
|
import { LABEL_IDENTIFIER_FIELD_METADATA_TYPES } from '@/constants/LabelIdentifierFieldMetadataTypes';
|
||||||
|
import { FieldMetadataType } from '@/types';
|
||||||
|
|
||||||
|
export const isLabelIdentifierFieldMetadataTypes = (
|
||||||
|
value: FieldMetadataType,
|
||||||
|
): value is (typeof LABEL_IDENTIFIER_FIELD_METADATA_TYPES)[number] =>
|
||||||
|
LABEL_IDENTIFIER_FIELD_METADATA_TYPES.includes(value);
|
||||||
Reference in New Issue
Block a user