import { BadRequestException, Injectable, NotFoundException, } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Equal, In, Repository } from 'typeorm'; import { TypeOrmQueryService } from '@ptc-org/nestjs-query-typeorm'; import { TenantMigrationService } from 'src/metadata/tenant-migration/tenant-migration.service'; import { TenantMigrationRunnerService } from 'src/tenant-migration-runner/tenant-migration-runner.service'; import { TenantMigrationTableAction } from 'src/metadata/tenant-migration/tenant-migration.entity'; import { ObjectMetadataEntity } from './object-metadata.entity'; import { CreateObjectInput } from './dtos/create-object.input'; @Injectable() export class ObjectMetadataService extends TypeOrmQueryService { constructor( @InjectRepository(ObjectMetadataEntity, 'metadata') private readonly objectMetadataRepository: Repository, private readonly tenantMigrationService: TenantMigrationService, private readonly migrationRunnerService: TenantMigrationRunnerService, ) { super(objectMetadataRepository); } override async deleteOne(id: string): Promise { const objectMetadata = await this.objectMetadataRepository.findOne({ where: { id }, }); if (!objectMetadata) { throw new NotFoundException('Object does not exist'); } if (!objectMetadata.isCustom) { throw new BadRequestException("Standard Objects can't be deleted"); } if (objectMetadata.isActive) { throw new BadRequestException("Active objects can't be deleted"); } return super.deleteOne(id); } override async createOne( record: CreateObjectInput, ): Promise { const createdObjectMetadata = await super.createOne({ ...record, targetTableName: `_${record.nameSingular}`, isActive: true, isCustom: true, isSystem: false, }); await this.tenantMigrationService.createCustomMigration( createdObjectMetadata.workspaceId, [ { name: createdObjectMetadata.targetTableName, action: 'create', } satisfies TenantMigrationTableAction, ], ); await this.migrationRunnerService.executeMigrationFromPendingMigrations( createdObjectMetadata.workspaceId, ); return createdObjectMetadata; } public async getObjectMetadataFromWorkspaceId(workspaceId: string) { return this.objectMetadataRepository.find({ where: { workspaceId }, relations: [ 'fields', 'fields.fromRelationMetadata', 'fields.fromRelationMetadata.fromObjectMetadata', 'fields.fromRelationMetadata.toObjectMetadata', 'fields.fromRelationMetadata.toObjectMetadata.fields', 'fields.toRelationMetadata', 'fields.toRelationMetadata.fromObjectMetadata', 'fields.toRelationMetadata.fromObjectMetadata.fields', 'fields.toRelationMetadata.toObjectMetadata', ], }); } public async getObjectMetadataFromDataSourceId(dataSourceId: string) { return this.objectMetadataRepository.find({ where: { dataSourceId }, relations: [ 'fields', 'fields.fromRelationMetadata', 'fields.fromRelationMetadata.fromObjectMetadata', 'fields.fromRelationMetadata.toObjectMetadata', 'fields.fromRelationMetadata.toObjectMetadata.fields', 'fields.toRelationMetadata', 'fields.toRelationMetadata.fromObjectMetadata', 'fields.toRelationMetadata.fromObjectMetadata.fields', 'fields.toRelationMetadata.toObjectMetadata', ], }); } public async findOneWithinWorkspace( objectMetadataId: string, workspaceId: string, ) { return this.objectMetadataRepository.findOne({ where: { id: objectMetadataId, workspaceId }, }); } public async findManyWithinWorkspace( objectMetadataIds: string[], workspaceId: string, ) { return this.objectMetadataRepository.findBy({ id: In(objectMetadataIds), workspaceId: Equal(workspaceId), }); } public async deleteObjectsMetadata(workspaceId: string) { await this.objectMetadataRepository.delete({ workspaceId }); } }