Fix: check if relation creates existing field name (#3433)
* Fix: check if relation creates existing field name * fix rebase * add object name to performance log
This commit is contained in:
@ -6,6 +6,7 @@ import {
|
|||||||
ForbiddenError,
|
ForbiddenError,
|
||||||
ValidationError,
|
ValidationError,
|
||||||
NotFoundError,
|
NotFoundError,
|
||||||
|
ConflictError,
|
||||||
} from 'src/filters/utils/graphql-errors.util';
|
} from 'src/filters/utils/graphql-errors.util';
|
||||||
import { ExceptionHandlerService } from 'src/integrations/exception-handler/exception-handler.service';
|
import { ExceptionHandlerService } from 'src/integrations/exception-handler/exception-handler.service';
|
||||||
|
|
||||||
@ -14,6 +15,7 @@ const graphQLPredefinedExceptions = {
|
|||||||
401: AuthenticationError,
|
401: AuthenticationError,
|
||||||
403: ForbiddenError,
|
403: ForbiddenError,
|
||||||
404: NotFoundError,
|
404: NotFoundError,
|
||||||
|
409: ConflictError,
|
||||||
};
|
};
|
||||||
|
|
||||||
export const handleExceptionAndConvertToGraphQLError = (
|
export const handleExceptionAndConvertToGraphQLError = (
|
||||||
|
|||||||
@ -141,3 +141,11 @@ export class NotFoundError extends BaseGraphQLError {
|
|||||||
Object.defineProperty(this, 'name', { value: 'NotFoundError' });
|
Object.defineProperty(this, 'name', { value: 'NotFoundError' });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class ConflictError extends BaseGraphQLError {
|
||||||
|
constructor(message: string, extensions?: Record<string, any>) {
|
||||||
|
super(message, 'CONFLICT', extensions);
|
||||||
|
|
||||||
|
Object.defineProperty(this, 'name', { value: 'ConflictError' });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@ -1,5 +1,6 @@
|
|||||||
import {
|
import {
|
||||||
BadRequestException,
|
BadRequestException,
|
||||||
|
ConflictException,
|
||||||
Injectable,
|
Injectable,
|
||||||
NotFoundException,
|
NotFoundException,
|
||||||
} from '@nestjs/common';
|
} from '@nestjs/common';
|
||||||
@ -38,149 +39,42 @@ export class RelationMetadataService extends TypeOrmQueryService<RelationMetadat
|
|||||||
super(relationMetadataRepository);
|
super(relationMetadataRepository);
|
||||||
}
|
}
|
||||||
|
|
||||||
override async deleteOne(id: string): Promise<RelationMetadataEntity> {
|
|
||||||
// TODO: This logic is duplicated with the BeforeDeleteOneRelation hook
|
|
||||||
const relationMetadata = await this.relationMetadataRepository.findOne({
|
|
||||||
where: { id },
|
|
||||||
relations: ['fromFieldMetadata', 'toFieldMetadata'],
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!relationMetadata) {
|
|
||||||
throw new NotFoundException('Relation does not exist');
|
|
||||||
}
|
|
||||||
|
|
||||||
const deletedRelationMetadata = super.deleteOne(id);
|
|
||||||
|
|
||||||
// TODO: Move to a cdc scheduler
|
|
||||||
this.fieldMetadataService.deleteMany({
|
|
||||||
id: {
|
|
||||||
in: [
|
|
||||||
relationMetadata.fromFieldMetadataId,
|
|
||||||
relationMetadata.toFieldMetadataId,
|
|
||||||
],
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
return deletedRelationMetadata;
|
|
||||||
}
|
|
||||||
|
|
||||||
override async createOne(
|
override async createOne(
|
||||||
relationMetadataInput: CreateRelationInput,
|
relationMetadataInput: CreateRelationInput,
|
||||||
): Promise<RelationMetadataEntity> {
|
): Promise<RelationMetadataEntity> {
|
||||||
if (
|
const objectMetadataMap = await this.getObjectMetadataMap(
|
||||||
relationMetadataInput.relationType === RelationMetadataType.MANY_TO_MANY
|
relationMetadataInput,
|
||||||
) {
|
|
||||||
throw new BadRequestException(
|
|
||||||
'Many to many relations are not supported yet',
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Relation types
|
|
||||||
*
|
|
||||||
* MANY TO MANY:
|
|
||||||
* FROM Ǝ-E TO (NOT YET SUPPORTED)
|
|
||||||
*
|
|
||||||
* ONE TO MANY:
|
|
||||||
* FROM --E TO (host the id in the TO table)
|
|
||||||
*
|
|
||||||
* ONE TO ONE:
|
|
||||||
* FROM --- TO (host the id in the TO table)
|
|
||||||
*/
|
|
||||||
|
|
||||||
const objectMetadataEntries =
|
|
||||||
await this.objectMetadataService.findManyWithinWorkspace(
|
|
||||||
relationMetadataInput.workspaceId,
|
|
||||||
{
|
|
||||||
where: {
|
|
||||||
id: In([
|
|
||||||
relationMetadataInput.fromObjectMetadataId,
|
|
||||||
relationMetadataInput.toObjectMetadataId,
|
|
||||||
]),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
const objectMetadataMap = objectMetadataEntries.reduce(
|
|
||||||
(acc, curr) => {
|
|
||||||
acc[curr.id] = curr;
|
|
||||||
|
|
||||||
return acc;
|
|
||||||
},
|
|
||||||
{} as { [key: string]: ObjectMetadataEntity },
|
|
||||||
);
|
);
|
||||||
|
|
||||||
if (
|
await this.validateCreateRelationMetadataInput(
|
||||||
objectMetadataMap[relationMetadataInput.fromObjectMetadataId] ===
|
relationMetadataInput,
|
||||||
undefined ||
|
objectMetadataMap,
|
||||||
objectMetadataMap[relationMetadataInput.toObjectMetadataId] === undefined
|
);
|
||||||
) {
|
|
||||||
throw new NotFoundException(
|
|
||||||
'Can\t find an existing object matching fromObjectMetadataId or toObjectMetadataId',
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
// NOTE: this logic is called to create relation through metadata graphql endpoint (so only for custom field relations)
|
||||||
|
const isCustom = true;
|
||||||
const baseColumnName = `${camelCase(relationMetadataInput.toName)}Id`;
|
const baseColumnName = `${camelCase(relationMetadataInput.toName)}Id`;
|
||||||
|
|
||||||
// TODO: this logic is called to create relation through metadata graphql endpoint (so only for custom field relations)
|
|
||||||
const isCustom = true;
|
|
||||||
const foreignKeyColumnName = isCustom
|
const foreignKeyColumnName = isCustom
|
||||||
? createCustomColumnName(baseColumnName)
|
? createCustomColumnName(baseColumnName)
|
||||||
: baseColumnName;
|
: baseColumnName;
|
||||||
|
|
||||||
const createdFields = await this.fieldMetadataService.createMany([
|
const createdFields = await this.fieldMetadataService.createMany([
|
||||||
// FROM
|
this.createFieldMetadataForRelationMetadata(
|
||||||
{
|
relationMetadataInput,
|
||||||
name: relationMetadataInput.fromName,
|
'from',
|
||||||
label: relationMetadataInput.fromLabel,
|
isCustom,
|
||||||
description: relationMetadataInput.fromDescription,
|
),
|
||||||
icon: relationMetadataInput.fromIcon,
|
this.createFieldMetadataForRelationMetadata(
|
||||||
isCustom: true,
|
relationMetadataInput,
|
||||||
targetColumnMap: {},
|
'to',
|
||||||
isActive: true,
|
isCustom,
|
||||||
isNullable: true,
|
),
|
||||||
type: FieldMetadataType.RELATION,
|
this.createForeignKeyFieldMetadata(
|
||||||
objectMetadataId: relationMetadataInput.fromObjectMetadataId,
|
relationMetadataInput,
|
||||||
workspaceId: relationMetadataInput.workspaceId,
|
baseColumnName,
|
||||||
},
|
foreignKeyColumnName,
|
||||||
// TO
|
),
|
||||||
{
|
|
||||||
name: relationMetadataInput.toName,
|
|
||||||
label: relationMetadataInput.toLabel,
|
|
||||||
description: relationMetadataInput.toDescription,
|
|
||||||
icon: relationMetadataInput.toIcon,
|
|
||||||
isCustom: true,
|
|
||||||
targetColumnMap: {
|
|
||||||
value: isCustom
|
|
||||||
? createCustomColumnName(relationMetadataInput.toName)
|
|
||||||
: relationMetadataInput.toName,
|
|
||||||
},
|
|
||||||
isActive: true,
|
|
||||||
isNullable: true,
|
|
||||||
type: FieldMetadataType.RELATION,
|
|
||||||
objectMetadataId: relationMetadataInput.toObjectMetadataId,
|
|
||||||
workspaceId: relationMetadataInput.workspaceId,
|
|
||||||
},
|
|
||||||
// FOREIGN KEY
|
|
||||||
{
|
|
||||||
name: baseColumnName,
|
|
||||||
label: `${relationMetadataInput.toLabel} Foreign Key`,
|
|
||||||
description: relationMetadataInput.toDescription
|
|
||||||
? `${relationMetadataInput.toDescription} Foreign Key`
|
|
||||||
: undefined,
|
|
||||||
icon: undefined,
|
|
||||||
isCustom: true,
|
|
||||||
targetColumnMap: {
|
|
||||||
value: foreignKeyColumnName,
|
|
||||||
},
|
|
||||||
isActive: true,
|
|
||||||
isNullable: true,
|
|
||||||
// Should not be visible on the front side
|
|
||||||
isSystem: true,
|
|
||||||
type: FieldMetadataType.UUID,
|
|
||||||
objectMetadataId: relationMetadataInput.toObjectMetadataId,
|
|
||||||
workspaceId: relationMetadataInput.workspaceId,
|
|
||||||
},
|
|
||||||
]);
|
]);
|
||||||
|
|
||||||
const createdFieldMap = createdFields.reduce((acc, fieldMetadata) => {
|
const createdFieldMap = createdFields.reduce((acc, fieldMetadata) => {
|
||||||
@ -197,6 +91,86 @@ export class RelationMetadataService extends TypeOrmQueryService<RelationMetadat
|
|||||||
toFieldMetadataId: createdFieldMap[relationMetadataInput.toName].id,
|
toFieldMetadataId: createdFieldMap[relationMetadataInput.toName].id,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
await this.createWorkspaceCustomMigration(
|
||||||
|
relationMetadataInput,
|
||||||
|
objectMetadataMap,
|
||||||
|
foreignKeyColumnName,
|
||||||
|
);
|
||||||
|
|
||||||
|
await this.workspaceMigrationRunnerService.executeMigrationFromPendingMigrations(
|
||||||
|
relationMetadataInput.workspaceId,
|
||||||
|
);
|
||||||
|
|
||||||
|
return createdRelationMetadata;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async validateCreateRelationMetadataInput(
|
||||||
|
relationMetadataInput: CreateRelationInput,
|
||||||
|
objectMetadataMap: { [key: string]: ObjectMetadataEntity },
|
||||||
|
) {
|
||||||
|
if (
|
||||||
|
relationMetadataInput.relationType === RelationMetadataType.MANY_TO_MANY
|
||||||
|
) {
|
||||||
|
throw new BadRequestException(
|
||||||
|
'Many to many relations are not supported yet',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
objectMetadataMap[relationMetadataInput.fromObjectMetadataId] ===
|
||||||
|
undefined ||
|
||||||
|
objectMetadataMap[relationMetadataInput.toObjectMetadataId] === undefined
|
||||||
|
) {
|
||||||
|
throw new NotFoundException(
|
||||||
|
'Can\t find an existing object matching with fromObjectMetadataId or toObjectMetadataId',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.checkIfFieldMetadataRelationNameExists(
|
||||||
|
relationMetadataInput,
|
||||||
|
objectMetadataMap,
|
||||||
|
'from',
|
||||||
|
);
|
||||||
|
await this.checkIfFieldMetadataRelationNameExists(
|
||||||
|
relationMetadataInput,
|
||||||
|
objectMetadataMap,
|
||||||
|
'to',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async checkIfFieldMetadataRelationNameExists(
|
||||||
|
relationMetadataInput: CreateRelationInput,
|
||||||
|
objectMetadataMap: { [key: string]: ObjectMetadataEntity },
|
||||||
|
relationDirection: 'from' | 'to',
|
||||||
|
) {
|
||||||
|
const fieldAlreadyExists =
|
||||||
|
await this.fieldMetadataService.findOneWithinWorkspace(
|
||||||
|
relationMetadataInput.workspaceId,
|
||||||
|
{
|
||||||
|
where: {
|
||||||
|
name: relationMetadataInput[`${relationDirection}Name`],
|
||||||
|
objectMetadataId:
|
||||||
|
relationMetadataInput[`${relationDirection}ObjectMetadataId`],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
if (fieldAlreadyExists) {
|
||||||
|
throw new ConflictException(
|
||||||
|
`Field on ${
|
||||||
|
objectMetadataMap[
|
||||||
|
relationMetadataInput[`${relationDirection}ObjectMetadataId`]
|
||||||
|
].nameSingular
|
||||||
|
} already exists`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async createWorkspaceCustomMigration(
|
||||||
|
relationMetadataInput: CreateRelationInput,
|
||||||
|
objectMetadataMap: { [key: string]: ObjectMetadataEntity },
|
||||||
|
foreignKeyColumnName: string,
|
||||||
|
) {
|
||||||
await this.workspaceMigrationService.createCustomMigration(
|
await this.workspaceMigrationService.createCustomMigration(
|
||||||
relationMetadataInput.workspaceId,
|
relationMetadataInput.workspaceId,
|
||||||
[
|
[
|
||||||
@ -237,12 +211,81 @@ export class RelationMetadataService extends TypeOrmQueryService<RelationMetadat
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
|
}
|
||||||
|
|
||||||
await this.workspaceMigrationRunnerService.executeMigrationFromPendingMigrations(
|
private createFieldMetadataForRelationMetadata(
|
||||||
relationMetadataInput.workspaceId,
|
relationMetadataInput: CreateRelationInput,
|
||||||
|
relationDirection: 'from' | 'to',
|
||||||
|
isCustom: boolean,
|
||||||
|
) {
|
||||||
|
return {
|
||||||
|
name: relationMetadataInput[`${relationDirection}Name`],
|
||||||
|
label: relationMetadataInput[`${relationDirection}Label`],
|
||||||
|
description: relationMetadataInput[`${relationDirection}Description`],
|
||||||
|
icon: relationMetadataInput[`${relationDirection}Icon`],
|
||||||
|
isCustom: true,
|
||||||
|
targetColumnMap:
|
||||||
|
relationDirection === 'to'
|
||||||
|
? isCustom
|
||||||
|
? createCustomColumnName(relationMetadataInput.toName)
|
||||||
|
: relationMetadataInput.toName
|
||||||
|
: {},
|
||||||
|
isActive: true,
|
||||||
|
isNullable: true,
|
||||||
|
type: FieldMetadataType.RELATION,
|
||||||
|
objectMetadataId:
|
||||||
|
relationMetadataInput[`${relationDirection}ObjectMetadataId`],
|
||||||
|
workspaceId: relationMetadataInput.workspaceId,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private createForeignKeyFieldMetadata(
|
||||||
|
relationMetadataInput: CreateRelationInput,
|
||||||
|
baseColumnName: string,
|
||||||
|
foreignKeyColumnName: string,
|
||||||
|
) {
|
||||||
|
return {
|
||||||
|
name: baseColumnName,
|
||||||
|
label: `${relationMetadataInput.toLabel} Foreign Key`,
|
||||||
|
description: relationMetadataInput.toDescription
|
||||||
|
? `${relationMetadataInput.toDescription} Foreign Key`
|
||||||
|
: undefined,
|
||||||
|
icon: undefined,
|
||||||
|
isCustom: true,
|
||||||
|
targetColumnMap: { value: foreignKeyColumnName },
|
||||||
|
isActive: true,
|
||||||
|
isNullable: true,
|
||||||
|
isSystem: true,
|
||||||
|
type: FieldMetadataType.UUID,
|
||||||
|
objectMetadataId: relationMetadataInput.toObjectMetadataId,
|
||||||
|
workspaceId: relationMetadataInput.workspaceId,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private async getObjectMetadataMap(
|
||||||
|
relationMetadataInput: CreateRelationInput,
|
||||||
|
): Promise<{ [key: string]: ObjectMetadataEntity }> {
|
||||||
|
const objectMetadataEntries =
|
||||||
|
await this.objectMetadataService.findManyWithinWorkspace(
|
||||||
|
relationMetadataInput.workspaceId,
|
||||||
|
{
|
||||||
|
where: {
|
||||||
|
id: In([
|
||||||
|
relationMetadataInput.fromObjectMetadataId,
|
||||||
|
relationMetadataInput.toObjectMetadataId,
|
||||||
|
]),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
return objectMetadataEntries.reduce(
|
||||||
|
(acc, curr) => {
|
||||||
|
acc[curr.id] = curr;
|
||||||
|
|
||||||
|
return acc;
|
||||||
|
},
|
||||||
|
{} as { [key: string]: ObjectMetadataEntity },
|
||||||
);
|
);
|
||||||
|
|
||||||
return createdRelationMetadata;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async findOneWithinWorkspace(
|
public async findOneWithinWorkspace(
|
||||||
@ -258,4 +301,30 @@ export class RelationMetadataService extends TypeOrmQueryService<RelationMetadat
|
|||||||
relations: ['fromFieldMetadata', 'toFieldMetadata'],
|
relations: ['fromFieldMetadata', 'toFieldMetadata'],
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override async deleteOne(id: string): Promise<RelationMetadataEntity> {
|
||||||
|
// TODO: This logic is duplicated with the BeforeDeleteOneRelation hook
|
||||||
|
const relationMetadata = await this.relationMetadataRepository.findOne({
|
||||||
|
where: { id },
|
||||||
|
relations: ['fromFieldMetadata', 'toFieldMetadata'],
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!relationMetadata) {
|
||||||
|
throw new NotFoundException('Relation does not exist');
|
||||||
|
}
|
||||||
|
|
||||||
|
const deletedRelationMetadata = super.deleteOne(id);
|
||||||
|
|
||||||
|
// TODO: Move to a cdc scheduler
|
||||||
|
this.fieldMetadataService.deleteMany({
|
||||||
|
id: {
|
||||||
|
in: [
|
||||||
|
relationMetadata.fromFieldMetadataId,
|
||||||
|
relationMetadata.toFieldMetadataId,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return deletedRelationMetadata;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -76,7 +76,11 @@ export class WorkspaceQueryRunnerService {
|
|||||||
const result = await this.execute(query, workspaceId);
|
const result = await this.execute(query, workspaceId);
|
||||||
const end = performance.now();
|
const end = performance.now();
|
||||||
|
|
||||||
console.log(`query time: ${end - start} ms`);
|
console.log(
|
||||||
|
`query time: ${end - start} ms on query ${
|
||||||
|
options.objectMetadataItem.nameSingular
|
||||||
|
}`,
|
||||||
|
);
|
||||||
|
|
||||||
return this.parseResult<IConnection<Record>>(
|
return this.parseResult<IConnection<Record>>(
|
||||||
result,
|
result,
|
||||||
|
|||||||
Reference in New Issue
Block a user