@solidstarters/solid-core 1.2.169 → 1.2.171
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/config/iam.config.d.ts +2 -0
- package/dist/config/iam.config.d.ts.map +1 -1
- package/dist/config/iam.config.js +2 -1
- package/dist/config/iam.config.js.map +1 -1
- package/dist/controllers/authentication.controller.d.ts.map +1 -1
- package/dist/controllers/authentication.controller.js +0 -8
- package/dist/controllers/authentication.controller.js.map +1 -1
- package/dist/controllers/email-template.controller.d.ts +1 -1
- package/dist/controllers/email-template.controller.d.ts.map +1 -1
- package/dist/controllers/email-template.controller.js +3 -6
- package/dist/controllers/email-template.controller.js.map +1 -1
- package/dist/controllers/google-authentication.controller.d.ts +3 -3
- package/dist/controllers/google-authentication.controller.d.ts.map +1 -1
- package/dist/controllers/google-authentication.controller.js +5 -8
- package/dist/controllers/google-authentication.controller.js.map +1 -1
- package/dist/controllers/media.controller.d.ts +1 -1
- package/dist/controllers/media.controller.d.ts.map +1 -1
- package/dist/controllers/media.controller.js +1 -5
- package/dist/controllers/media.controller.js.map +1 -1
- package/dist/controllers/model-metadata.controller.d.ts.map +1 -1
- package/dist/controllers/model-metadata.controller.js +0 -5
- package/dist/controllers/model-metadata.controller.js.map +1 -1
- package/dist/controllers/otp-authentication.controller.js +0 -3
- package/dist/controllers/otp-authentication.controller.js.map +1 -1
- package/dist/controllers/service.controller.d.ts +2 -2
- package/dist/controllers/service.controller.d.ts.map +1 -1
- package/dist/controllers/service.controller.js +3 -7
- package/dist/controllers/service.controller.js.map +1 -1
- package/dist/controllers/sms-template.controller.d.ts.map +1 -1
- package/dist/controllers/sms-template.controller.js +0 -3
- package/dist/controllers/sms-template.controller.js.map +1 -1
- package/dist/entities/user.entity.d.ts +3 -0
- package/dist/entities/user.entity.d.ts.map +1 -1
- package/dist/entities/user.entity.js +13 -1
- package/dist/entities/user.entity.js.map +1 -1
- package/dist/filters/http-exception.filter.js +1 -1
- package/dist/filters/http-exception.filter.js.map +1 -1
- package/dist/helpers/error-mapper.service.d.ts.map +1 -1
- package/dist/helpers/error-mapper.service.js +1 -1
- package/dist/helpers/error-mapper.service.js.map +1 -1
- package/dist/helpers/field-crud-managers/PasswordFieldCrudManager.d.ts +2 -1
- package/dist/helpers/field-crud-managers/PasswordFieldCrudManager.d.ts.map +1 -1
- package/dist/helpers/field-crud-managers/PasswordFieldCrudManager.js +3 -3
- package/dist/helpers/field-crud-managers/PasswordFieldCrudManager.js.map +1 -1
- package/dist/helpers/model-metadata-helper.service.js.map +1 -1
- package/dist/passport-strategies/local.strategy.js +1 -1
- package/dist/passport-strategies/local.strategy.js.map +1 -1
- package/dist/seeders/seed-data/solid-core-metadata.json +43 -0
- package/dist/services/authentication.service.d.ts +2 -1
- package/dist/services/authentication.service.d.ts.map +1 -1
- package/dist/services/authentication.service.js +28 -5
- package/dist/services/authentication.service.js.map +1 -1
- package/dist/services/bcrypt.service.d.ts +11 -1
- package/dist/services/bcrypt.service.d.ts.map +1 -1
- package/dist/services/bcrypt.service.js +35 -5
- package/dist/services/bcrypt.service.js.map +1 -1
- package/dist/services/crud.service.d.ts.map +1 -1
- package/dist/services/crud.service.js +2 -1
- package/dist/services/crud.service.js.map +1 -1
- package/dist/services/export-transaction.service.d.ts +3 -1
- package/dist/services/export-transaction.service.d.ts.map +1 -1
- package/dist/services/export-transaction.service.js +70 -16
- package/dist/services/export-transaction.service.js.map +1 -1
- package/dist/services/hashing.service.d.ts +4 -1
- package/dist/services/hashing.service.d.ts.map +1 -1
- package/dist/services/hashing.service.js.map +1 -1
- package/dist/solid-core.module.d.ts.map +1 -1
- package/dist/solid-core.module.js +46 -61
- package/dist/solid-core.module.js.map +1 -1
- package/dist/subscribers/audit.subscriber.d.ts.map +1 -1
- package/dist/subscribers/audit.subscriber.js +1 -1
- package/dist/subscribers/audit.subscriber.js.map +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +1 -1
- package/src/config/iam.config.ts +2 -1
- package/src/controllers/authentication.controller.ts +8 -10
- package/src/controllers/email-template.controller.ts +7 -10
- package/src/controllers/google-authentication.controller.ts +9 -10
- package/src/controllers/media.controller.ts +5 -6
- package/src/controllers/model-metadata.controller.ts +5 -6
- package/src/controllers/otp-authentication.controller.ts +2 -2
- package/src/controllers/service.controller.ts +8 -9
- package/src/controllers/sms-template.controller.ts +3 -4
- package/src/entities/user.entity.ts +9 -0
- package/src/filters/http-exception.filter.ts +1 -1
- package/src/helpers/error-mapper.service.ts +1 -35
- package/src/helpers/field-crud-managers/PasswordFieldCrudManager.ts +4 -3
- package/src/helpers/model-metadata-helper.service.ts +1 -1
- package/src/passport-strategies/local.strategy.ts +1 -1
- package/src/seeders/seed-data/solid-core-metadata.json +43 -0
- package/src/services/authentication.service.ts +32 -3
- package/src/services/bcrypt.service.ts +45 -7
- package/src/services/crud.service.ts +2 -1
- package/src/services/export-transaction.service.ts +118 -55
- package/src/services/hashing.service.ts +5 -2
- package/src/solid-core.module.ts +59 -61
- package/src/subscribers/audit.subscriber.ts +6 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"crud.service.d.ts","sourceRoot":"","sources":["../../src/services/crud.service.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,aAAa,EAAc,MAAM,gBAAgB,CAAC;AAC3D,OAAO,EAAE,gBAAgB,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AAC3D,OAAO,EAAE,aAAa,EAAyD,MAAM,SAAS,CAAC;AAC/F,OAAO,EAAE,UAAU,EAAE,MAAM,+BAA+B,CAAC;AAC3D,OAAO,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAG3D,OAAO,EAAE,aAAa,EAAE,MAAM,mCAAmC,CAAC;AAsBlE,OAAO,EAAoB,gBAAgB,EAAE,MAAM,eAAe,CAAC;AACnE,OAAO,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;AAC1D,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAE7C,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAChE,OAAO,EAAE,qBAAqB,EAAE,MAAM,2BAA2B,CAAC;
|
|
1
|
+
{"version":3,"file":"crud.service.d.ts","sourceRoot":"","sources":["../../src/services/crud.service.ts"],"names":[],"mappings":"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"}
|
|
@@ -30,6 +30,7 @@ const class_validator_1 = require("class-validator");
|
|
|
30
30
|
const error_messages_1 = require("../constants/error-messages");
|
|
31
31
|
const success_messages_1 = require("../constants/success-messages");
|
|
32
32
|
const request_context_service_1 = require("./request-context.service");
|
|
33
|
+
const hashing_service_1 = require("./hashing.service");
|
|
33
34
|
class CRUDService {
|
|
34
35
|
constructor(modelMetadataService, moduleMetadataService, configService, fileService, discoveryService, crudHelperService, entityManager, repo, modelName, moduleName, moduleRef) {
|
|
35
36
|
this.modelMetadataService = modelMetadataService;
|
|
@@ -237,7 +238,7 @@ class CRUDService {
|
|
|
237
238
|
return new DateFieldCrudManager_1.DateFieldCrudManager(options);
|
|
238
239
|
}
|
|
239
240
|
case create_field_metadata_dto_1.SolidFieldType.password: {
|
|
240
|
-
const options = { ...commonOptions, min: fieldMetadata.min, max: fieldMetadata.max, regexPattern: fieldMetadata.regexPattern };
|
|
241
|
+
const options = { ...commonOptions, min: fieldMetadata.min, max: fieldMetadata.max, regexPattern: fieldMetadata.regexPattern, hashingService: this.moduleRef.get(hashing_service_1.HashingService, { strict: false }) };
|
|
241
242
|
return new PasswordFieldCrudManager_1.PasswordFieldCrudManager(options);
|
|
242
243
|
}
|
|
243
244
|
case create_field_metadata_dto_1.SolidFieldType.mediaSingle:
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"crud.service.js","sourceRoot":"","sources":["../../src/services/crud.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { BadRequestException, Inject, NotFoundException, Optional } from \"@nestjs/common\";\nimport { ConfigService, ConfigType } from \"@nestjs/config\";\nimport { DiscoveryService, ModuleRef } from \"@nestjs/core\";\nimport { EntityManager, In, IsNull, Not, QueryFailedError, SelectQueryBuilder } from \"typeorm\";\nimport { Repository } from \"typeorm/repository/Repository\";\nimport { BasicFilterDto } from \"../dtos/basic-filters.dto\";\nimport { ComputedFieldValueType, RelationType, SelectionValueType, SolidFieldType } from \"../dtos/create-field-metadata.dto\";\nimport { MediaStorageProviderType } from \"../dtos/create-media-storage-provider-metadata.dto\";\nimport { FieldMetadata } from \"../entities/field-metadata.entity\";\nimport { ModelMetadata } from \"../entities/model-metadata.entity\";\nimport { BigIntFieldCrudManager } from \"../helpers/field-crud-managers/BigIntFieldCrudManager\";\nimport { BooleanFieldCrudManager } from \"../helpers/field-crud-managers/BooleanFieldCrudManager\";\nimport { ComputedFieldCrudManager } from \"../helpers/field-crud-managers/ComputedFieldCrudManager\";\nimport { DateFieldCrudManager } from \"../helpers/field-crud-managers/DateFieldCrudManager\";\nimport { DecimalFieldCrudManager } from \"../helpers/field-crud-managers/DecimalFieldCrudManager\";\nimport { EmailFieldCrudManager, MAX_EMAIL_LENGTH } from \"../helpers/field-crud-managers/EmailFieldCrudManager\";\nimport { IntFieldCrudManager } from \"../helpers/field-crud-managers/IntFieldCrudManager\";\nimport { JsonFieldCrudManager } from \"../helpers/field-crud-managers/JsonFieldCrudManager\";\nimport { LongTextFieldCrudManager } from \"../helpers/field-crud-managers/LongTextFieldCrudManager\";\nimport { ManyToManyRelationFieldCrudManager, ManyToManyRelationFieldOptions } from \"../helpers/field-crud-managers/ManyToManyRelationFieldCrudManager\";\nimport { ManyToOneRelationFieldCrudManager, ManyToOneRelationFieldOptions } from \"../helpers/field-crud-managers/ManyToOneRelationFieldCrudManager\";\nimport { MediaFieldCrudManager, SolidMediaType } from \"../helpers/field-crud-managers/MediaFieldCrudManager\";\nimport { NoOpsFieldCrudManager } from \"../helpers/field-crud-managers/NoOpsFieldCrudManager\";\nimport { OneToManyRelationFieldCrudManager, OneToManyRelationFieldOptions } from \"../helpers/field-crud-managers/OneToManyRelationFieldCrudManager\";\nimport { PasswordFieldCrudManager } from \"../helpers/field-crud-managers/PasswordFieldCrudManager\";\nimport { RichTextFieldCrudManager } from \"../helpers/field-crud-managers/RichTextFieldCrudManager\";\nimport { SelectionDynamicFieldCrudManager } from \"../helpers/field-crud-managers/SelectionDynamicFieldCrudManager\";\nimport { SelectionStaticFieldCrudManager } from \"../helpers/field-crud-managers/SelectionStaticFieldCrudManager\";\nimport { ShortTextFieldCrudManager } from \"../helpers/field-crud-managers/ShortTextFieldCrudManager\";\nimport { UUIDFieldCrudManager } from \"../helpers/field-crud-managers/UUIDFieldCrudManager\";\nimport { FieldCrudManager, MediaWithFullUrl } from \"../interfaces\";\nimport { CrudHelperService } from \"./crud-helper.service\";\nimport { FileService } from \"./file.service\";\nimport { getMediaStorageProvider } from \"./mediaStorageProviders\";\nimport { ModelMetadataService } from \"./model-metadata.service\";\nimport { ModuleMetadataService } from \"./module-metadata.service\";\nimport { isArray } from \"class-validator\";\nimport { ERROR_MESSAGES } from \"src/constants/error-messages\";\nimport { SUCCESS_MESSAGES } from \"src/constants/success-messages\";\nimport { RequestContextService } from \"./request-context.service\";\n\nexport class CRUDService<T> { // Add two generic value i.e Person,CreatePersonDto, so we get the proper types in our service\n\n constructor(\n readonly modelMetadataService: ModelMetadataService,\n readonly moduleMetadataService: ModuleMetadataService,\n readonly configService: ConfigService,\n readonly fileService: FileService,\n readonly discoveryService: DiscoveryService,\n readonly crudHelperService: CrudHelperService,\n readonly entityManager: EntityManager,\n readonly repo: Repository<T>,\n readonly modelName: string,\n readonly moduleName: string,\n readonly moduleRef: ModuleRef,\n //We can just have the Model Entity here\n ) { }\n\n async create(createDto: any, files: Express.Multer.File[] = [], solidRequestContext: any = {}): Promise<T> {\n // This class will be extended by the generated service class i.e PersonService\n // The data required to identify the model and module name will be passed from the generate CrudService subclass\n //TODO: Algorithm to create the entity\n // 1. Fire a query and load all the fields in the provided model name for a particular module\n // FIXME This can be optimized to take in module name i.e (handle scenario wherein same model exists in multiple modules)\n let hasMediaFields = false;\n\n const model = await this.loadModel();\n // Check wheather user has create permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasCreatePermissionOnModel(solidRequestContext.activeUser, model.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n // const inverseRelationFields = await this.loadInverseRelationFields();\n const fieldsToProcess = [...model.fields];\n\n // 2. Loop through the fields with a switch statement\n // 3. Handle the fields based on field type\n for (const field of fieldsToProcess) {\n const transformed = await this.validateAndTransformDto(field, createDto, files, hasMediaFields);\n createDto = transformed.dto;\n hasMediaFields = transformed.hasMediaFields;\n };\n try {\n // 5. Save the entity\n // For media, we need to use a storage provider and save the media, then save the associated uri against the entity or media table\n const entity = this.repo.create(createDto);\n const savedEntity = await this.repo.save(entity) as T;\n\n // 6. Save the media\n if (hasMediaFields) {\n this.saveMedia(model, files, savedEntity);\n }\n return savedEntity;\n } catch (error) {\n if (error instanceof QueryFailedError && error.message.includes('duplicate key value violates unique constraint')) {\n throw new BadRequestException(ERROR_MESSAGES.DUPLICATE_ENTRY);\n }\n throw error;\n }\n }\n\n private async loadModel() {\n return await this.modelMetadataService.findOneBySingularName(this.modelName, {\n fields: {\n model: {\n userKeyField: true\n },\n mediaStorageProvider: true,\n },\n module: true,\n });\n }\n\n private async validateAndTransformDto(field: FieldMetadata, dto: any, files: Express.Multer.File[], hasMediaFields: boolean, isPartialUpdate: boolean = false, isUpdate: boolean = false) {\n const fieldManager: FieldCrudManager = await this.fieldCrudManager(field, this.entityManager, isPartialUpdate, isUpdate);\n const validationErrors = fieldManager.validate(dto, files);\n const errors = (validationErrors instanceof Promise) ? await validationErrors : validationErrors;\n if (errors.length > 0) {\n throw new BadRequestException(`Validation errors in ${field.name} is invalid i.e ${errors.map(e => e.error).join(', ')}`); //FIXME: Better to return a validation error object\n }\n const dtoOrPromise = fieldManager.transformForCreate(dto);\n dto = (dtoOrPromise instanceof Promise) ? await dtoOrPromise : dtoOrPromise;\n hasMediaFields = hasMediaFields || field.type === 'mediaSingle' || field.type === 'mediaMultiple';\n return { dto, hasMediaFields };\n }\n\n //FIXME: Need to make this saving media async. Use queues approach\n private saveMedia(model: ModelMetadata, files: Express.Multer.File[], savedEntity: T) {\n // Get all the media fields in the dto\n\n const mediaFields = model.fields.filter(field => field.type === 'mediaSingle' || field.type === 'mediaMultiple');\n\n // Depending upon media storage provider configured, get the appropriate storage provider\n mediaFields.forEach(async (mediaField) => {\n const media = files.filter(multerFile => multerFile.fieldname === mediaField.name);\n\n // If media is present, then save the media\n if (media.length > 0) {\n const storageProviderMetadata = mediaField.mediaStorageProvider;\n\n // Use the storage provider metadata to get the appropriate storage provider implementation\n const storageProviderType = storageProviderMetadata.type as MediaStorageProviderType;\n\n // Get the storage provider implementation\n const storageProvider = await getMediaStorageProvider(this.moduleRef, storageProviderType);\n\n //Commented the below code since we will be direclty images from server on call from ui \n // await storageProvider.delete(savedEntity, mediaField);\n await storageProvider.store(media, savedEntity, mediaField);\n }\n });\n }\n\n //TODO: Will the updates be partial i.e PATCH or full i.e PUT\n async update(id: number, updateDto: any, files: Express.Multer.File[] = [], isPartialUpdate: boolean = false, solidRequestContext: any = {}, isUpdate: boolean = false): Promise<T> {\n if (!id) {\n throw new Error(ERROR_MESSAGES.ID_REQUIRED_FOR_UPDATE);\n }\n isUpdate = true;\n const model = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasUpdatePermissionOnModel(solidRequestContext.activeUser, model.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n const entity = await this.repo.findOne({\n where: {\n //@ts-ignore\n id: id,\n }\n });\n if (!entity) {\n throw new Error(`Entity [${this.moduleName}.${this.modelName}] with id ${id} not found`);\n }\n\n updateDto.id = id;\n // This class will be extended by the generated service class i.e PersonService\n // The data required to identify the model and module name will be passed from the generate CrudService subclass\n //TODO: Algorithm to create the entity\n // 1. Fire a query and load all the fields in the provided model name for a particular module\n // FIXME This can be optimized to take in module name i.e (handle scenario wherein same model exists in multiple modules)\n let hasMediaFields = false;\n\n const fieldsToProcess = [...model.fields];\n\n // 2. Loop through the fields with a switch statement\n // 3. Handle the fields based on field type\n for (const field of fieldsToProcess) {\n const transformed = await this.validateAndTransformDto(field, updateDto, files, hasMediaFields, isPartialUpdate, isUpdate);\n updateDto = transformed.dto;\n hasMediaFields = transformed.hasMediaFields;\n }\n\n // 5. Save the entity\n // For media, we need to use a storage provider and save the media, then save the associated uri against the entity or media table\n const mergedEntity = this.repo.merge(entity, updateDto);\n const savedEntity = await this.repo.save(mergedEntity) as T;\n\n // 6. Save the media\n if (hasMediaFields) {\n this.saveMedia(model, files, savedEntity);\n }\n return savedEntity;\n }\n\n //TODO: Will the updates be partial i.e PATCH or full i.e PUT\n async delete(id: number, solidRequestContext: any = {}) {\n if (!id) {\n throw new Error(ERROR_MESSAGES.ID_REQUIRED_FOR_DELETE);\n }\n const loadedmodel = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasDeletePermissionOnModel(solidRequestContext.activeUser, loadedmodel.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n\n const model = await this.modelMetadataService.findOneBySingularName(this.modelName, {\n fields: {\n model: true,\n mediaStorageProvider: true,\n },\n module: true,\n });\n const entity = await this.repo.findOne({\n where: {\n //@ts-ignore\n id: id,\n }\n });\n if (!entity) {\n throw new Error(`Entity [${this.moduleName}.${this.modelName}] with id ${id} not found`);\n }\n\n // If the model has internationalisation enabled, delete children with defaultEntityLocaleId === this entity's id\n if (model.internationalisation) {\n // Find all child entities where defaultEntityLocaleId === this entity's id\n const childEntities = await this.repo.find({\n where: { defaultEntityLocaleId: id } as any\n });\n\n if (childEntities.length > 0) {\n if (model.enableSoftDelete === true) {\n await this.repo.softRemove(childEntities);\n } else {\n await this.repo.remove(childEntities);\n }\n }\n }\n\n if (model.enableSoftDelete === true) {\n await this.repo.softRemove(entity);\n return this.repo.save(entity);\n } else {\n return this.repo.remove(entity);\n }\n }\n\n private async fieldCrudManager(fieldMetadata: FieldMetadata, entityManager: EntityManager, isPartialUpdate: boolean = false, isUpdate: boolean = false) {\n const commonOptions = { required: fieldMetadata.required && !isPartialUpdate, fieldName: fieldMetadata.name, isUpdate };\n switch (fieldMetadata.type) {\n case SolidFieldType.shortText: {\n const options = { ...commonOptions, length: fieldMetadata.max, regexPattern: fieldMetadata.regexPattern };\n return new ShortTextFieldCrudManager(options);\n }\n case SolidFieldType.longtext: {\n const options = { ...commonOptions, regexPattern: fieldMetadata.regexPattern };\n return new LongTextFieldCrudManager(options);\n }\n case SolidFieldType.boolean: {\n const options = { ...commonOptions };\n return new BooleanFieldCrudManager(options);\n }\n case SolidFieldType.richText: {\n const options = { ...commonOptions, regexPattern: fieldMetadata.regexPattern };\n return new RichTextFieldCrudManager(options);\n }\n case SolidFieldType.json: {\n const options = { ...commonOptions };\n return new JsonFieldCrudManager(options);\n }\n case SolidFieldType.int: {\n const options = { ...commonOptions, min: fieldMetadata.min, max: fieldMetadata.max };\n return new IntFieldCrudManager(options);\n }\n case SolidFieldType.decimal: {\n const options = { ...commonOptions, min: fieldMetadata.min, max: fieldMetadata.max };\n return new DecimalFieldCrudManager(options);\n }\n case SolidFieldType.bigint: {\n const options = { ...commonOptions, min: fieldMetadata.min, max: fieldMetadata.max };\n return new BigIntFieldCrudManager(options);\n }\n case SolidFieldType.email: {\n const options = { ...commonOptions, max: fieldMetadata.max ?? MAX_EMAIL_LENGTH, regexPattern: fieldMetadata.regexPattern };\n return new EmailFieldCrudManager(options);\n }\n case SolidFieldType.date:\n case SolidFieldType.datetime: {\n const options = { ...commonOptions };\n return new DateFieldCrudManager(options);\n }\n case SolidFieldType.password: {\n const options = { ...commonOptions, min: fieldMetadata.min, max: fieldMetadata.max, regexPattern: fieldMetadata.regexPattern };\n return new PasswordFieldCrudManager(options);\n }\n case SolidFieldType.mediaSingle:\n case SolidFieldType.mediaMultiple: {\n // update will need to delete the existing media and save the new media \n // case 'mediaSingle':\n // Use the EntityController to extract uploaded content & pass to the entity service.\n // If embedded media, then the media uri will saved in the entity table,\n // else the uri will be saved in the media table\n // Plan the media table schema e.g id, uri, storageProvider, entity_id, entity_name, createdAt, updatedAt\n // break;\n // case 'mediaMultiple':\n // Use the EntityController to extract uploaded content & pass to the entity service.\n // If embedded media, then the media uri will saved in the entity table, else the uri will be saved in the media table\n // Plan the media table schema e.g id, uri, storageProvider, entity_id, entity_name, createdAt, updatedAt\n const options = { ...commonOptions, type: fieldMetadata.type as unknown as SolidMediaType };\n return new MediaFieldCrudManager(options);\n }\n case SolidFieldType.relation: {\n // Identify if the field is for the inverse side or not\n if (fieldMetadata.relationType === RelationType.manyToOne) {\n const relationCoModelUserKeyFieldName = await this.getUserKeyFieldNameForModel(fieldMetadata.relationCoModelSingularName);\n const manyToOneOptions: ManyToOneRelationFieldOptions = {\n ...commonOptions,\n relationCoModelSingularName: fieldMetadata.relationCoModelSingularName,\n // modelUserKeyFieldName: fieldMetadata.model.userKeyField?.name,\n modelSingularName: fieldMetadata.model.singularName,\n relationCoModelUserKeyFieldName: relationCoModelUserKeyFieldName,\n entityManager,\n }\n return new ManyToOneRelationFieldCrudManager(manyToOneOptions);\n }\n else if (fieldMetadata.relationType === RelationType.oneToMany) {\n const oneToManyOptions: OneToManyRelationFieldOptions = {\n ...commonOptions,\n relationCoModelSingularName: fieldMetadata.relationCoModelSingularName,\n modelSingularName: fieldMetadata.model.singularName,\n entityManager,\n inverseFieldName: fieldMetadata.relationCoModelFieldName,\n inverseRelationCoModelFieldName: fieldMetadata.name,\n }\n return new OneToManyRelationFieldCrudManager(oneToManyOptions);\n }\n else if (fieldMetadata.relationType === RelationType.manyTomany) {\n if (fieldMetadata.isRelationManyToManyOwner) {\n const manyToManyOptions: ManyToManyRelationFieldOptions = {\n ...commonOptions,\n relationCoModelSingularName: fieldMetadata.relationCoModelSingularName,\n modelSingularName: fieldMetadata.model.singularName,\n isInverseSide: false,\n entityManager,\n fieldName: fieldMetadata.name,\n }\n return new ManyToManyRelationFieldCrudManager(manyToManyOptions);\n }\n else {\n const inverseManyToManyOptions: ManyToManyRelationFieldOptions = {\n ...commonOptions,\n relationCoModelSingularName: fieldMetadata.relationCoModelSingularName,\n modelSingularName: fieldMetadata.model.singularName,\n isInverseSide: true,\n entityManager,\n fieldName: fieldMetadata.relationCoModelFieldName,\n relationCoModelFieldName: fieldMetadata.name,\n }\n return new ManyToManyRelationFieldCrudManager(inverseManyToManyOptions);\n }\n }\n else throw new Error(ERROR_MESSAGES.RELATION_TYPE_NOT_SUPPORTED);\n // return (fieldMetadata.relationType === 'many-to-one') ? new ManyToOneRelationFieldCrudManager(fieldMetadata, entityManager) : new ManyToManyRelationFieldCrudManager(fieldMetadata, entityManager); //FIXME many-to-many pending\n // ManyToOne -> fieldId. The value is saved as is. No transformation is required\n // OneToMany -> fieldIds. Get the value of the oneToMany field side. No transformation is required (While saving special provision to be made)\n // ManyToMany\n // break;\n }\n case SolidFieldType.selectionStatic: {\n\n // Validation against the selectionStatic values. No transformation is required\n // If the value is not in the selectionStatic values, then throw\n // Also validate against the selectionType\n const options = { ...commonOptions, selectionStaticValues: fieldMetadata.selectionStaticValues, selectionValueType: fieldMetadata.selectionValueType as SelectionValueType, isMultiSelect: fieldMetadata.isMultiSelect };\n return new SelectionStaticFieldCrudManager(options);\n }\n case SolidFieldType.selectionDynamic: {// [HOLD]\n // Default implementation using list of values.\n // ISelectionProvider interface to be implemented for dynamic selection\n // dataSource: string; // The name of the selection provider\n // filterSchema : json // This is a custom json object that every data source will handle accordingly. We could validate the query against the selection provider\n // values : string[]; // The values returned by the selection provider\n const options = { ...commonOptions, selectionDynamicProvider: fieldMetadata.selectionDynamicProvider, selectionDynamicProviderCtxt: fieldMetadata.selectionDynamicProviderCtxt, selectionValueType: fieldMetadata.selectionValueType as SelectionValueType, discoveryService: this.discoveryService, isMultiSelect: fieldMetadata.isMultiSelect };\n return new SelectionDynamicFieldCrudManager(options);\n }\n case SolidFieldType.uuid: {\n const options = { ...commonOptions };\n // If no value is provided, then generate a uuid. Add to the dto\n return new UUIDFieldCrudManager(options);\n }\n case SolidFieldType.computed: {\n\n // The value will be computed by the computed provider\n // Invoke the appropriate computed provider, get the value and add to the dto\n const options = { ...commonOptions, computedFieldProvider: fieldMetadata.computedFieldValueProvider, computedFieldValueProviderCtxt: fieldMetadata.computedFieldValueProviderCtxt, computedFieldValueType: fieldMetadata.computedFieldValueType as ComputedFieldValueType, discoveryService: this.discoveryService, skipComputation: this.isSkipComputation(isPartialUpdate, fieldMetadata) };\n return new ComputedFieldCrudManager(options);\n }\n default:\n return new NoOpsFieldCrudManager();\n }\n }\n\n private isSkipComputation(isPartialUpdate: boolean, computedFieldMetadata: FieldMetadata) {\n if (isPartialUpdate) return true; // If it is a partial update, then skip computation\n if (computedFieldMetadata.computedFieldTriggerConfig && computedFieldMetadata.computedFieldTriggerConfig.length > 0) {\n return true; // computedFieldTriggerConfig is a new field introduced as part of the IEntityComputedFieldProvider new interface, so this computation will be skiipped in crud service & will be called in the subscriber instead\n }\n return false; // If it is not a partial update, then do not skip computation\n }\n\n async find(basicFilterDto: BasicFilterDto, solidRequestContext: any = {}) {\n const alias = 'entity';\n // Extract the required keys from the input query\n let { limit, offset, populateMedia, populateGroup, groupFilter } = basicFilterDto;\n const { singularName, internationalisation, draftPublishWorkflow } = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasReadPermissionOnModel(solidRequestContext.activeUser, singularName);\n if (!hasPermission) {\n throw new BadRequestException('Forbidden');\n }\n }\n\n // Set the request filter in the request context service\n const requestContextService = this.moduleRef.get(RequestContextService, { strict: false });\n requestContextService.setRequestFilter(basicFilterDto);\n\n // Create above query on pincode table using query builder\n var qb: SelectQueryBuilder<T> = this.repo.createQueryBuilder(alias)\n qb = this.crudHelperService.buildFilterQuery(qb, basicFilterDto, alias);\n if (internationalisation && draftPublishWorkflow) {\n qb = this.crudHelperService.buildFilterQuery(qb, basicFilterDto, alias, internationalisation, draftPublishWorkflow, this.moduleRef);\n }\n\n if (basicFilterDto.groupBy) {\n // Get the records and the count\n const { groupMeta, groupRecords } = await this.handleGroupFind(qb, groupFilter, populateGroup, alias, populateMedia);\n const totalGroups = await this.crudHelperService.countGroupedRecords(qb, basicFilterDto, alias);\n qb = this.crudHelperService.buildFilterQuery(qb, basicFilterDto, alias);\n\n return {\n meta: {\n \"totalRecords\": totalGroups\n },\n groupMeta,\n groupRecords,\n }\n }\n else {\n // Get the records and the count\n const { meta, records } = await this.handleNonGroupFind(qb, populateMedia, offset, limit, alias);\n return {\n meta,\n records,\n }\n }\n }\n\n private async handleNonGroupFind(qb: SelectQueryBuilder<T>, populateMedia: string[], offset: number, limit: number, alias: string) {\n const [entities, count] = await qb.getManyAndCount();\n\n // Populate the entity with the media\n if (populateMedia && populateMedia.length > 0) {\n await this.handlePopulateMedia(populateMedia, entities);\n }\n\n return this.wrapFindResponse(offset, limit, count, entities);\n }\n\n private async handleGroupFind(qb: SelectQueryBuilder<T>, groupFilter: BasicFilterDto, populateGroup: boolean, alias: string, populateMedia: string[]) {\n const groupByResult = await qb.getRawMany();\n\n const groupMeta = [];\n const groupRecords = [];\n // For each group, get the records and the count\n for (const group of groupByResult) {\n if (populateGroup) {\n let groupByQb: SelectQueryBuilder<T> = this.repo.createQueryBuilder(alias);\n groupByQb = this.crudHelperService.buildFilterQuery(groupByQb, groupFilter, alias);\n groupByQb = this.crudHelperService.buildGroupByRecordsQuery(groupByQb, group, alias);\n const [entities, count] = await groupByQb.getManyAndCount();\n\n // Populate the entity with the media\n if (populateMedia && populateMedia.length > 0) {\n await this.handlePopulateMedia(populateMedia, entities);\n }\n const groupData = this.wrapFindResponse(groupFilter.offset, groupFilter.limit, count, entities);\n groupRecords.push(this.crudHelperService.createGroupRecords(group, alias, groupData));\n }\n groupMeta.push(this.crudHelperService.createGroupMeta(group, alias));\n }\n return { groupMeta, groupRecords };\n }\n\n private wrapFindResponse(offset: number, limit: number, count: number, entities: T[]) {\n const currentPage = Math.floor(offset / limit) + 1;\n const totalPages = Math.ceil(count / limit);\n\n const nextPage = currentPage < totalPages ? currentPage + 1 : null;\n const prevPage = currentPage > 1 ? currentPage - 1 : null;\n\n const r = {\n meta: {\n totalRecords: count,\n currentPage: currentPage,\n nextPage: nextPage,\n prevPage: prevPage,\n totalPages: totalPages,\n perPage: +limit,\n },\n records: entities\n };\n return r;\n }\n\n private async handlePopulateMedia(populateMedia: string[], entities: T[]) {\n const model = await this.entityManager.getRepository(ModelMetadata).findOne({\n where: {\n singularName: this.modelName,\n },\n relations: ['fields', 'fields.mediaStorageProvider', 'fields.model', 'module'],\n });\n\n // Will iterate through every entity & all populateMedia & call getMediaDetails for each field\n for (const entity of entities) {\n for (const mediaFieldPath of populateMedia) {\n await this.populateMediaObject(mediaFieldPath, model, entity);\n }\n }\n return entities;\n }\n\n // Adds the media with full URL to the entity / nested entity\n private async populateMediaObject(mediaFieldPath: string, model: ModelMetadata, entity: T) {\n if (mediaFieldPath.includes('.')) { // mediaFieldPath is a nested field\n const pathParts = mediaFieldPath.split('.');\n const mediaFieldMetadata = await this.getFieldMetadataRecursively(pathParts, model.fields);\n if (!mediaFieldMetadata) {\n throw new BadRequestException(`Media field ${mediaFieldPath} not found in model ${this.modelName}`);\n }\n\n // We can assume that the media field entity model is already populated as part of the entity data\n const mediaFieldEntities = this.getMediaFieldEntities(entity, pathParts);\n if (!mediaFieldEntities || mediaFieldEntities.length === 0) {\n return;//no need to populate data if relation not exists\n }\n // Populate the media field entities with the full URL\n for (const mediaFieldEntity of mediaFieldEntities) {\n const mediaWithFullUrl = await this.getMediaWithFullUrl(mediaFieldEntity, mediaFieldMetadata);\n this.appendMediaKey(mediaWithFullUrl, mediaFieldEntity, mediaFieldMetadata.name);\n // mediaFieldEntity['_media'][mediaFieldPath] = mediaWithFullUrl\n }\n }\n else {\n // mediaFieldPath is a single field\n const mediaFieldMetadata = model.fields.find(field => field.name === mediaFieldPath);\n if (!mediaFieldMetadata) {\n throw new BadRequestException(`Media field ${mediaFieldPath} not found in model ${this.modelName}`);\n }\n const mediaWithFullUrl = await this.getMediaWithFullUrl(entity, mediaFieldMetadata);\n this.appendMediaKey(mediaWithFullUrl, entity, mediaFieldPath);\n // entity['_media'][mediaFieldPath] = mediaWithFullUrl;\n }\n }\n\n // // Add the media with full URL to the entity\n private appendMediaKey(mediaWithFullUrl: MediaWithFullUrl[], entity: T, mediaFieldPath: string) {\n // if _media key already exists, append the new media to the existing array\n if (entity['_media']) {\n entity['_media'][mediaFieldPath] = mediaWithFullUrl;\n }\n else {\n entity['_media'] = {\n [mediaFieldPath]: mediaWithFullUrl\n };\n }\n }\n\n private getMediaFieldEntities(entity: T, mediaPathParts: string[]): T[] {\n let entityPart = entity;\n for (let i = 0; i < mediaPathParts.length - 1; i++) {\n const pathPart = mediaPathParts[i];\n if (entity[pathPart]) {\n entityPart = entity[pathPart];\n } else {\n throw new BadRequestException(`Media field ${pathPart} not found in entity ${JSON.stringify(entity)}`);\n }\n }\n return isArray(entityPart) ? entityPart : [entityPart];\n }\n\n async getMediaWithFullUrl(mediaEntity: any, mediaFieldMetadata: FieldMetadata): Promise<MediaWithFullUrl[]> {\n const storageProviderMetadata = mediaFieldMetadata.mediaStorageProvider;\n const storageProviderType = storageProviderMetadata.type as MediaStorageProviderType;\n const storageProvider = await getMediaStorageProvider(this.moduleRef, storageProviderType);\n const mediaDetails = await storageProvider.retrieve(mediaEntity, mediaFieldMetadata);\n return mediaDetails as MediaWithFullUrl[];\n }\n\n async findOne(id: number, query: any, solidRequestContext: any = {}) {\n const { populate = [], fields = [], populateMedia = [] } = query;\n\n // const normalizedFields = this.crudHelperService.normalize(fields);\n const normalizedPopulate = this.crudHelperService.normalize(populate);\n const normalizedPopulateMedia = this.crudHelperService.normalize(populateMedia);\n\n // if normalizedPopulateMedia, has any nested media paths, then add then to populate excluding the last part\n const additionalPopulate = this.crudHelperService.additionalRelationsRequiredForMediaPopulation(normalizedPopulateMedia);\n // Add the additional populate relations to the normalizedPopulate, if they are not already present\n normalizedPopulate.push(...additionalPopulate.filter((relation) => !normalizedPopulate.includes(relation)));\n\n const model = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasReadPermissionOnModel(solidRequestContext.activeUser, model.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n\n let entity = await this.repo.findOne({\n where: {\n //@ts-ignore\n id: id,\n },\n relations: normalizedPopulate,\n select: fields,\n });\n if (!entity) {\n throw new NotFoundException(`Entity [${this.moduleName}.${this.modelName}] with id ${id} not found`);\n }\n // Populate the entity with the media\n if (normalizedPopulateMedia.length > 0) {\n const populatedEntities = await this.handlePopulateMedia(normalizedPopulateMedia, [entity]);\n entity = populatedEntities[0] as Awaited<T>;\n }\n return entity;\n }\n\n async insertMany(createDtos: any[], filesArray: Express.Multer.File[][] = [], solidRequestContext: any = {}): Promise<T[]> {\n\n\n // if (createDtos.length !== filesArray.length) {\n // throw new BadRequestException('Mismatch between data objects and file arrays.');\n // }\n\n const loadedmodel = await this.loadModel();\n // Check wheather user has create permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasCreatePermissionOnModel(solidRequestContext.activeUser, loadedmodel.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n\n // Fetch model metadata once\n const model = await this.modelMetadataService.findOneBySingularName(this.modelName, {\n fields: {\n model: true,\n mediaStorageProvider: true,\n },\n module: true,\n });\n\n // Process each createDto in parallel\n const createAndSavePromises = createDtos.map(async (createDto, index) => {\n const files = []; // TODO, This is set, because we are not supporting files for insertMany currently\n let hasMediaFields = false;\n\n // Process each field\n for (const field of model.fields) {\n const fieldManager: FieldCrudManager = await this.fieldCrudManager(field, this.entityManager);\n const validationErrors = await fieldManager.validate(createDto, files);\n if (validationErrors.length > 0) {\n throw new BadRequestException(`Validation errors in ${field.name} are invalid: ${validationErrors.map(e => e.error).join(', ')}`);\n }\n createDto = await fieldManager.transformForCreate(createDto);\n hasMediaFields = hasMediaFields || field.type === 'mediaSingle' || field.type === 'mediaMultiple';\n }\n\n // Save the entity\n const entity = this.repo.create(createDto);\n const savedEntity = await this.repo.save(entity) as T;\n\n //Commented since currently Files are not supported for insertmany\n // if (hasMediaFields) {\n // await this.saveMedia(model, files, savedEntity);\n // }\n\n return savedEntity;\n });\n\n // Await all promises in parallel\n const savedEntities = await Promise.all(createAndSavePromises);\n\n return savedEntities;\n }\n\n async deleteMany(ids: number[], solidRequestContext: any = {}): Promise<any> {\n\n if (!ids || ids.length === 0) {\n throw new Error(ERROR_MESSAGES.DELETE_IDS_REQUIRED);\n }\n\n const loadedmodel = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasDeletePermissionOnModel(solidRequestContext.activeUser, loadedmodel.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n const model = await this.modelMetadataService.findOneBySingularName(this.modelName, {\n fields: {\n model: true,\n mediaStorageProvider: true,\n },\n module: true,\n });\n\n\n\n const removedEntities = [];\n for (let i = 0; i < ids.length; i++) {\n const id = ids[i]\n const entity = await this.repo.findOne({\n where: {\n //@ts-ignore\n id: id,\n }\n });\n removedEntities.push(entity);\n }\n if (model.enableSoftDelete === true) {\n await this.repo.softRemove(removedEntities);\n return this.repo.save(removedEntities);\n } else {\n return this.repo.remove(removedEntities);\n }\n // return removedEntities\n }\n\n async recover(id: number, solidRequestContext: any = {}) {\n try {\n const loadedmodel = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasRecoverPermissionOnModel(solidRequestContext.activeUser, loadedmodel.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n\n const softDeletedRows = await this.repo.findOne({\n where: {\n //@ts-ignore\n id, deletedAt: Not(IsNull())\n },\n withDeleted: true,\n });\n\n if (!softDeletedRows) {\n throw new Error(ERROR_MESSAGES.NO_SOFT_DELETED_RECORD_FOUND);\n }\n\n await this.repo.update(id, {\n //@ts-ignore\n deletedAt: null, deletedTracker: \"not-deleted\"\n });\n\n return { message: SUCCESS_MESSAGES.RECORD_RECOVERED, data: softDeletedRows };\n } catch (error) {\n if (error instanceof QueryFailedError) {\n if ((error as any).code === '23505') {\n throw new Error(ERROR_MESSAGES.CONFLICTING_RECORD_ON_UNARCHIVE);\n }\n }\n\n throw new Error(error);\n }\n }\n\n async recoverMany(ids: number[], solidRequestContext: any = {}) {\n try {\n const loadedmodel = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasRecoverPermissionOnModel(solidRequestContext.activeUser, loadedmodel.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n\n if (!ids || ids.length === 0) {\n throw new Error(ERROR_MESSAGES.DELETE_IDS_REQUIRED);\n }\n\n // Find soft-deleted records matching the given IDs\n const softDeletedRows = await this.repo.find({\n where: {\n //@ts-ignore\n id: In(ids),\n deletedAt: Not(IsNull()),\n },\n withDeleted: true,\n });\n\n if (softDeletedRows.length === 0) {\n throw new Error(ERROR_MESSAGES.NO_SOFT_DELETED_RECORDS_FOUND);\n }\n\n // Recover the specific records by setting deletedAt to null\n await this.repo.update(\n //@ts-ignore\n { id: In(ids) },\n { deletedAt: null, deletedTracker: \"not-deleted\" }\n );\n\n return { message: SUCCESS_MESSAGES.SELECTED_RECORDS_RECOVERED, recoveredIds: ids };\n } catch (error) {\n if (error instanceof QueryFailedError) {\n if ((error as any).code === \"23505\") {\n throw new Error(ERROR_MESSAGES.CONFLICTING_RECORD_ON_UNARCHIVE);\n }\n }\n\n throw new Error(error);\n }\n }\n\n\n async getFieldMetadataRecursively(pathParts: string[], fields: FieldMetadata[]) {\n if (!pathParts || pathParts.length === 0) {\n throw new BadRequestException(ERROR_MESSAGES.EMPTY_PATH_PARTS);\n }\n\n const [currentPart, ...remainingParts] = pathParts;\n const field = fields.find(field => field.name === currentPart);\n\n if (!field) {\n throw new BadRequestException(`Field ${currentPart} not found in model ${this.modelName}`);\n }\n\n // Base case: last part, return the field\n if (remainingParts.length === 0) {\n return field;\n }\n\n if (!field.relationCoModelSingularName) {\n throw new BadRequestException(`Field ${field.name} does not define a relationCoModelSingularName`);\n }\n\n const relationCoModel = await this.entityManager.getRepository(ModelMetadata).findOne({\n where: { singularName: field.relationCoModelSingularName },\n relations: ['fields', 'fields.mediaStorageProvider', 'fields.model'],\n });\n\n if (!relationCoModel) {\n throw new BadRequestException(`Model ${field.relationCoModelSingularName} not found`);\n }\n\n return this.getFieldMetadataRecursively(remainingParts, relationCoModel.fields);\n }\n\n async getUserKeyFieldNameForModel(modelSingularName: string): Promise<string> {\n const model = await this.modelMetadataService.findOneBySingularName(modelSingularName, ['userKeyField']);\n if (!model) {\n throw new BadRequestException(`Model ${modelSingularName} not found`);\n }\n return model.userKeyField?.name || '';\n }\n}\n\n"]}
|
|
1
|
+
{"version":3,"file":"crud.service.js","sourceRoot":"","sources":["../../src/services/crud.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { BadRequestException, Inject, NotFoundException, Optional } from \"@nestjs/common\";\nimport { ConfigService, ConfigType } from \"@nestjs/config\";\nimport { DiscoveryService, ModuleRef } from \"@nestjs/core\";\nimport { EntityManager, In, IsNull, Not, QueryFailedError, SelectQueryBuilder } from \"typeorm\";\nimport { Repository } from \"typeorm/repository/Repository\";\nimport { BasicFilterDto } from \"../dtos/basic-filters.dto\";\nimport { ComputedFieldValueType, RelationType, SelectionValueType, SolidFieldType } from \"../dtos/create-field-metadata.dto\";\nimport { MediaStorageProviderType } from \"../dtos/create-media-storage-provider-metadata.dto\";\nimport { FieldMetadata } from \"../entities/field-metadata.entity\";\nimport { ModelMetadata } from \"../entities/model-metadata.entity\";\nimport { BigIntFieldCrudManager } from \"../helpers/field-crud-managers/BigIntFieldCrudManager\";\nimport { BooleanFieldCrudManager } from \"../helpers/field-crud-managers/BooleanFieldCrudManager\";\nimport { ComputedFieldCrudManager } from \"../helpers/field-crud-managers/ComputedFieldCrudManager\";\nimport { DateFieldCrudManager } from \"../helpers/field-crud-managers/DateFieldCrudManager\";\nimport { DecimalFieldCrudManager } from \"../helpers/field-crud-managers/DecimalFieldCrudManager\";\nimport { EmailFieldCrudManager, MAX_EMAIL_LENGTH } from \"../helpers/field-crud-managers/EmailFieldCrudManager\";\nimport { IntFieldCrudManager } from \"../helpers/field-crud-managers/IntFieldCrudManager\";\nimport { JsonFieldCrudManager } from \"../helpers/field-crud-managers/JsonFieldCrudManager\";\nimport { LongTextFieldCrudManager } from \"../helpers/field-crud-managers/LongTextFieldCrudManager\";\nimport { ManyToManyRelationFieldCrudManager, ManyToManyRelationFieldOptions } from \"../helpers/field-crud-managers/ManyToManyRelationFieldCrudManager\";\nimport { ManyToOneRelationFieldCrudManager, ManyToOneRelationFieldOptions } from \"../helpers/field-crud-managers/ManyToOneRelationFieldCrudManager\";\nimport { MediaFieldCrudManager, SolidMediaType } from \"../helpers/field-crud-managers/MediaFieldCrudManager\";\nimport { NoOpsFieldCrudManager } from \"../helpers/field-crud-managers/NoOpsFieldCrudManager\";\nimport { OneToManyRelationFieldCrudManager, OneToManyRelationFieldOptions } from \"../helpers/field-crud-managers/OneToManyRelationFieldCrudManager\";\nimport { PasswordFieldCrudManager } from \"../helpers/field-crud-managers/PasswordFieldCrudManager\";\nimport { RichTextFieldCrudManager } from \"../helpers/field-crud-managers/RichTextFieldCrudManager\";\nimport { SelectionDynamicFieldCrudManager } from \"../helpers/field-crud-managers/SelectionDynamicFieldCrudManager\";\nimport { SelectionStaticFieldCrudManager } from \"../helpers/field-crud-managers/SelectionStaticFieldCrudManager\";\nimport { ShortTextFieldCrudManager } from \"../helpers/field-crud-managers/ShortTextFieldCrudManager\";\nimport { UUIDFieldCrudManager } from \"../helpers/field-crud-managers/UUIDFieldCrudManager\";\nimport { FieldCrudManager, MediaWithFullUrl } from \"../interfaces\";\nimport { CrudHelperService } from \"./crud-helper.service\";\nimport { FileService } from \"./file.service\";\nimport { getMediaStorageProvider } from \"./mediaStorageProviders\";\nimport { ModelMetadataService } from \"./model-metadata.service\";\nimport { ModuleMetadataService } from \"./module-metadata.service\";\nimport { isArray } from \"class-validator\";\nimport { ERROR_MESSAGES } from \"src/constants/error-messages\";\nimport { SUCCESS_MESSAGES } from \"src/constants/success-messages\";\nimport { RequestContextService } from \"./request-context.service\";\nimport { HashingService } from \"./hashing.service\";\n\nexport class CRUDService<T> { // Add two generic value i.e Person,CreatePersonDto, so we get the proper types in our service\n\n constructor(\n readonly modelMetadataService: ModelMetadataService,\n readonly moduleMetadataService: ModuleMetadataService,\n readonly configService: ConfigService,\n readonly fileService: FileService,\n readonly discoveryService: DiscoveryService,\n readonly crudHelperService: CrudHelperService,\n readonly entityManager: EntityManager,\n readonly repo: Repository<T>,\n readonly modelName: string,\n readonly moduleName: string,\n readonly moduleRef: ModuleRef,\n //We can just have the Model Entity here\n ) { }\n\n async create(createDto: any, files: Express.Multer.File[] = [], solidRequestContext: any = {}): Promise<T> {\n // This class will be extended by the generated service class i.e PersonService\n // The data required to identify the model and module name will be passed from the generate CrudService subclass\n //TODO: Algorithm to create the entity\n // 1. Fire a query and load all the fields in the provided model name for a particular module\n // FIXME This can be optimized to take in module name i.e (handle scenario wherein same model exists in multiple modules)\n let hasMediaFields = false;\n\n const model = await this.loadModel();\n // Check wheather user has create permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasCreatePermissionOnModel(solidRequestContext.activeUser, model.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n // const inverseRelationFields = await this.loadInverseRelationFields();\n const fieldsToProcess = [...model.fields];\n\n // 2. Loop through the fields with a switch statement\n // 3. Handle the fields based on field type\n for (const field of fieldsToProcess) {\n const transformed = await this.validateAndTransformDto(field, createDto, files, hasMediaFields);\n createDto = transformed.dto;\n hasMediaFields = transformed.hasMediaFields;\n };\n try {\n // 5. Save the entity\n // For media, we need to use a storage provider and save the media, then save the associated uri against the entity or media table\n const entity = this.repo.create(createDto);\n const savedEntity = await this.repo.save(entity) as T;\n\n // 6. Save the media\n if (hasMediaFields) {\n this.saveMedia(model, files, savedEntity);\n }\n return savedEntity;\n } catch (error) {\n if (error instanceof QueryFailedError && error.message.includes('duplicate key value violates unique constraint')) {\n throw new BadRequestException(ERROR_MESSAGES.DUPLICATE_ENTRY);\n }\n throw error;\n }\n }\n\n private async loadModel() {\n return await this.modelMetadataService.findOneBySingularName(this.modelName, {\n fields: {\n model: {\n userKeyField: true\n },\n mediaStorageProvider: true,\n },\n module: true,\n });\n }\n\n private async validateAndTransformDto(field: FieldMetadata, dto: any, files: Express.Multer.File[], hasMediaFields: boolean, isPartialUpdate: boolean = false, isUpdate: boolean = false) {\n const fieldManager: FieldCrudManager = await this.fieldCrudManager(field, this.entityManager, isPartialUpdate, isUpdate);\n const validationErrors = fieldManager.validate(dto, files);\n const errors = (validationErrors instanceof Promise) ? await validationErrors : validationErrors;\n if (errors.length > 0) {\n throw new BadRequestException(`Validation errors in ${field.name} is invalid i.e ${errors.map(e => e.error).join(', ')}`); //FIXME: Better to return a validation error object\n }\n const dtoOrPromise = fieldManager.transformForCreate(dto);\n dto = (dtoOrPromise instanceof Promise) ? await dtoOrPromise : dtoOrPromise;\n hasMediaFields = hasMediaFields || field.type === 'mediaSingle' || field.type === 'mediaMultiple';\n return { dto, hasMediaFields };\n }\n\n //FIXME: Need to make this saving media async. Use queues approach\n private saveMedia(model: ModelMetadata, files: Express.Multer.File[], savedEntity: T) {\n // Get all the media fields in the dto\n\n const mediaFields = model.fields.filter(field => field.type === 'mediaSingle' || field.type === 'mediaMultiple');\n\n // Depending upon media storage provider configured, get the appropriate storage provider\n mediaFields.forEach(async (mediaField) => {\n const media = files.filter(multerFile => multerFile.fieldname === mediaField.name);\n\n // If media is present, then save the media\n if (media.length > 0) {\n const storageProviderMetadata = mediaField.mediaStorageProvider;\n\n // Use the storage provider metadata to get the appropriate storage provider implementation\n const storageProviderType = storageProviderMetadata.type as MediaStorageProviderType;\n\n // Get the storage provider implementation\n const storageProvider = await getMediaStorageProvider(this.moduleRef, storageProviderType);\n\n //Commented the below code since we will be direclty images from server on call from ui \n // await storageProvider.delete(savedEntity, mediaField);\n await storageProvider.store(media, savedEntity, mediaField);\n }\n });\n }\n\n //TODO: Will the updates be partial i.e PATCH or full i.e PUT\n async update(id: number, updateDto: any, files: Express.Multer.File[] = [], isPartialUpdate: boolean = false, solidRequestContext: any = {}, isUpdate: boolean = false): Promise<T> {\n if (!id) {\n throw new Error(ERROR_MESSAGES.ID_REQUIRED_FOR_UPDATE);\n }\n isUpdate = true;\n const model = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasUpdatePermissionOnModel(solidRequestContext.activeUser, model.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n const entity = await this.repo.findOne({\n where: {\n //@ts-ignore\n id: id,\n }\n });\n if (!entity) {\n throw new Error(`Entity [${this.moduleName}.${this.modelName}] with id ${id} not found`);\n }\n\n updateDto.id = id;\n // This class will be extended by the generated service class i.e PersonService\n // The data required to identify the model and module name will be passed from the generate CrudService subclass\n //TODO: Algorithm to create the entity\n // 1. Fire a query and load all the fields in the provided model name for a particular module\n // FIXME This can be optimized to take in module name i.e (handle scenario wherein same model exists in multiple modules)\n let hasMediaFields = false;\n\n const fieldsToProcess = [...model.fields];\n\n // 2. Loop through the fields with a switch statement\n // 3. Handle the fields based on field type\n for (const field of fieldsToProcess) {\n const transformed = await this.validateAndTransformDto(field, updateDto, files, hasMediaFields, isPartialUpdate, isUpdate);\n updateDto = transformed.dto;\n hasMediaFields = transformed.hasMediaFields;\n }\n\n // 5. Save the entity\n // For media, we need to use a storage provider and save the media, then save the associated uri against the entity or media table\n const mergedEntity = this.repo.merge(entity, updateDto);\n const savedEntity = await this.repo.save(mergedEntity) as T;\n\n // 6. Save the media\n if (hasMediaFields) {\n this.saveMedia(model, files, savedEntity);\n }\n return savedEntity;\n }\n\n //TODO: Will the updates be partial i.e PATCH or full i.e PUT\n async delete(id: number, solidRequestContext: any = {}) {\n if (!id) {\n throw new Error(ERROR_MESSAGES.ID_REQUIRED_FOR_DELETE);\n }\n const loadedmodel = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasDeletePermissionOnModel(solidRequestContext.activeUser, loadedmodel.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n\n const model = await this.modelMetadataService.findOneBySingularName(this.modelName, {\n fields: {\n model: true,\n mediaStorageProvider: true,\n },\n module: true,\n });\n const entity = await this.repo.findOne({\n where: {\n //@ts-ignore\n id: id,\n }\n });\n if (!entity) {\n throw new Error(`Entity [${this.moduleName}.${this.modelName}] with id ${id} not found`);\n }\n\n // If the model has internationalisation enabled, delete children with defaultEntityLocaleId === this entity's id\n if (model.internationalisation) {\n // Find all child entities where defaultEntityLocaleId === this entity's id\n const childEntities = await this.repo.find({\n where: { defaultEntityLocaleId: id } as any\n });\n\n if (childEntities.length > 0) {\n if (model.enableSoftDelete === true) {\n await this.repo.softRemove(childEntities);\n } else {\n await this.repo.remove(childEntities);\n }\n }\n }\n\n if (model.enableSoftDelete === true) {\n await this.repo.softRemove(entity);\n return this.repo.save(entity);\n } else {\n return this.repo.remove(entity);\n }\n }\n\n private async fieldCrudManager(fieldMetadata: FieldMetadata, entityManager: EntityManager, isPartialUpdate: boolean = false, isUpdate: boolean = false) {\n const commonOptions = { required: fieldMetadata.required && !isPartialUpdate, fieldName: fieldMetadata.name, isUpdate };\n switch (fieldMetadata.type) {\n case SolidFieldType.shortText: {\n const options = { ...commonOptions, length: fieldMetadata.max, regexPattern: fieldMetadata.regexPattern };\n return new ShortTextFieldCrudManager(options);\n }\n case SolidFieldType.longtext: {\n const options = { ...commonOptions, regexPattern: fieldMetadata.regexPattern };\n return new LongTextFieldCrudManager(options);\n }\n case SolidFieldType.boolean: {\n const options = { ...commonOptions };\n return new BooleanFieldCrudManager(options);\n }\n case SolidFieldType.richText: {\n const options = { ...commonOptions, regexPattern: fieldMetadata.regexPattern };\n return new RichTextFieldCrudManager(options);\n }\n case SolidFieldType.json: {\n const options = { ...commonOptions };\n return new JsonFieldCrudManager(options);\n }\n case SolidFieldType.int: {\n const options = { ...commonOptions, min: fieldMetadata.min, max: fieldMetadata.max };\n return new IntFieldCrudManager(options);\n }\n case SolidFieldType.decimal: {\n const options = { ...commonOptions, min: fieldMetadata.min, max: fieldMetadata.max };\n return new DecimalFieldCrudManager(options);\n }\n case SolidFieldType.bigint: {\n const options = { ...commonOptions, min: fieldMetadata.min, max: fieldMetadata.max };\n return new BigIntFieldCrudManager(options);\n }\n case SolidFieldType.email: {\n const options = { ...commonOptions, max: fieldMetadata.max ?? MAX_EMAIL_LENGTH, regexPattern: fieldMetadata.regexPattern };\n return new EmailFieldCrudManager(options);\n }\n case SolidFieldType.date:\n case SolidFieldType.datetime: {\n const options = { ...commonOptions };\n return new DateFieldCrudManager(options);\n }\n case SolidFieldType.password: {\n const options = { ...commonOptions, min: fieldMetadata.min, max: fieldMetadata.max, regexPattern: fieldMetadata.regexPattern, hashingService: this.moduleRef.get(HashingService, { strict: false }) };\n return new PasswordFieldCrudManager(options);\n }\n case SolidFieldType.mediaSingle:\n case SolidFieldType.mediaMultiple: {\n // update will need to delete the existing media and save the new media \n // case 'mediaSingle':\n // Use the EntityController to extract uploaded content & pass to the entity service.\n // If embedded media, then the media uri will saved in the entity table,\n // else the uri will be saved in the media table\n // Plan the media table schema e.g id, uri, storageProvider, entity_id, entity_name, createdAt, updatedAt\n // break;\n // case 'mediaMultiple':\n // Use the EntityController to extract uploaded content & pass to the entity service.\n // If embedded media, then the media uri will saved in the entity table, else the uri will be saved in the media table\n // Plan the media table schema e.g id, uri, storageProvider, entity_id, entity_name, createdAt, updatedAt\n const options = { ...commonOptions, type: fieldMetadata.type as unknown as SolidMediaType };\n return new MediaFieldCrudManager(options);\n }\n case SolidFieldType.relation: {\n // Identify if the field is for the inverse side or not\n if (fieldMetadata.relationType === RelationType.manyToOne) {\n const relationCoModelUserKeyFieldName = await this.getUserKeyFieldNameForModel(fieldMetadata.relationCoModelSingularName);\n const manyToOneOptions: ManyToOneRelationFieldOptions = {\n ...commonOptions,\n relationCoModelSingularName: fieldMetadata.relationCoModelSingularName,\n // modelUserKeyFieldName: fieldMetadata.model.userKeyField?.name,\n modelSingularName: fieldMetadata.model.singularName,\n relationCoModelUserKeyFieldName: relationCoModelUserKeyFieldName,\n entityManager,\n }\n return new ManyToOneRelationFieldCrudManager(manyToOneOptions);\n }\n else if (fieldMetadata.relationType === RelationType.oneToMany) {\n const oneToManyOptions: OneToManyRelationFieldOptions = {\n ...commonOptions,\n relationCoModelSingularName: fieldMetadata.relationCoModelSingularName,\n modelSingularName: fieldMetadata.model.singularName,\n entityManager,\n inverseFieldName: fieldMetadata.relationCoModelFieldName,\n inverseRelationCoModelFieldName: fieldMetadata.name,\n }\n return new OneToManyRelationFieldCrudManager(oneToManyOptions);\n }\n else if (fieldMetadata.relationType === RelationType.manyTomany) {\n if (fieldMetadata.isRelationManyToManyOwner) {\n const manyToManyOptions: ManyToManyRelationFieldOptions = {\n ...commonOptions,\n relationCoModelSingularName: fieldMetadata.relationCoModelSingularName,\n modelSingularName: fieldMetadata.model.singularName,\n isInverseSide: false,\n entityManager,\n fieldName: fieldMetadata.name,\n }\n return new ManyToManyRelationFieldCrudManager(manyToManyOptions);\n }\n else {\n const inverseManyToManyOptions: ManyToManyRelationFieldOptions = {\n ...commonOptions,\n relationCoModelSingularName: fieldMetadata.relationCoModelSingularName,\n modelSingularName: fieldMetadata.model.singularName,\n isInverseSide: true,\n entityManager,\n fieldName: fieldMetadata.relationCoModelFieldName,\n relationCoModelFieldName: fieldMetadata.name,\n }\n return new ManyToManyRelationFieldCrudManager(inverseManyToManyOptions);\n }\n }\n else throw new Error(ERROR_MESSAGES.RELATION_TYPE_NOT_SUPPORTED);\n // return (fieldMetadata.relationType === 'many-to-one') ? new ManyToOneRelationFieldCrudManager(fieldMetadata, entityManager) : new ManyToManyRelationFieldCrudManager(fieldMetadata, entityManager); //FIXME many-to-many pending\n // ManyToOne -> fieldId. The value is saved as is. No transformation is required\n // OneToMany -> fieldIds. Get the value of the oneToMany field side. No transformation is required (While saving special provision to be made)\n // ManyToMany\n // break;\n }\n case SolidFieldType.selectionStatic: {\n\n // Validation against the selectionStatic values. No transformation is required\n // If the value is not in the selectionStatic values, then throw\n // Also validate against the selectionType\n const options = { ...commonOptions, selectionStaticValues: fieldMetadata.selectionStaticValues, selectionValueType: fieldMetadata.selectionValueType as SelectionValueType, isMultiSelect: fieldMetadata.isMultiSelect };\n return new SelectionStaticFieldCrudManager(options);\n }\n case SolidFieldType.selectionDynamic: {// [HOLD]\n // Default implementation using list of values.\n // ISelectionProvider interface to be implemented for dynamic selection\n // dataSource: string; // The name of the selection provider\n // filterSchema : json // This is a custom json object that every data source will handle accordingly. We could validate the query against the selection provider\n // values : string[]; // The values returned by the selection provider\n const options = { ...commonOptions, selectionDynamicProvider: fieldMetadata.selectionDynamicProvider, selectionDynamicProviderCtxt: fieldMetadata.selectionDynamicProviderCtxt, selectionValueType: fieldMetadata.selectionValueType as SelectionValueType, discoveryService: this.discoveryService, isMultiSelect: fieldMetadata.isMultiSelect };\n return new SelectionDynamicFieldCrudManager(options);\n }\n case SolidFieldType.uuid: {\n const options = { ...commonOptions };\n // If no value is provided, then generate a uuid. Add to the dto\n return new UUIDFieldCrudManager(options);\n }\n case SolidFieldType.computed: {\n\n // The value will be computed by the computed provider\n // Invoke the appropriate computed provider, get the value and add to the dto\n const options = { ...commonOptions, computedFieldProvider: fieldMetadata.computedFieldValueProvider, computedFieldValueProviderCtxt: fieldMetadata.computedFieldValueProviderCtxt, computedFieldValueType: fieldMetadata.computedFieldValueType as ComputedFieldValueType, discoveryService: this.discoveryService, skipComputation: this.isSkipComputation(isPartialUpdate, fieldMetadata) };\n return new ComputedFieldCrudManager(options);\n }\n default:\n return new NoOpsFieldCrudManager();\n }\n }\n\n private isSkipComputation(isPartialUpdate: boolean, computedFieldMetadata: FieldMetadata) {\n if (isPartialUpdate) return true; // If it is a partial update, then skip computation\n if (computedFieldMetadata.computedFieldTriggerConfig && computedFieldMetadata.computedFieldTriggerConfig.length > 0) {\n return true; // computedFieldTriggerConfig is a new field introduced as part of the IEntityComputedFieldProvider new interface, so this computation will be skiipped in crud service & will be called in the subscriber instead\n }\n return false; // If it is not a partial update, then do not skip computation\n }\n\n async find(basicFilterDto: BasicFilterDto, solidRequestContext: any = {}) {\n const alias = 'entity';\n // Extract the required keys from the input query\n let { limit, offset, populateMedia, populateGroup, groupFilter } = basicFilterDto;\n const { singularName, internationalisation, draftPublishWorkflow } = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasReadPermissionOnModel(solidRequestContext.activeUser, singularName);\n if (!hasPermission) {\n throw new BadRequestException('Forbidden');\n }\n }\n\n // Set the request filter in the request context service\n const requestContextService = this.moduleRef.get(RequestContextService, { strict: false });\n requestContextService.setRequestFilter(basicFilterDto);\n\n // Create above query on pincode table using query builder\n var qb: SelectQueryBuilder<T> = this.repo.createQueryBuilder(alias)\n qb = this.crudHelperService.buildFilterQuery(qb, basicFilterDto, alias);\n if (internationalisation && draftPublishWorkflow) {\n qb = this.crudHelperService.buildFilterQuery(qb, basicFilterDto, alias, internationalisation, draftPublishWorkflow, this.moduleRef);\n }\n\n if (basicFilterDto.groupBy) {\n // Get the records and the count\n const { groupMeta, groupRecords } = await this.handleGroupFind(qb, groupFilter, populateGroup, alias, populateMedia);\n const totalGroups = await this.crudHelperService.countGroupedRecords(qb, basicFilterDto, alias);\n qb = this.crudHelperService.buildFilterQuery(qb, basicFilterDto, alias);\n\n return {\n meta: {\n \"totalRecords\": totalGroups\n },\n groupMeta,\n groupRecords,\n }\n }\n else {\n // Get the records and the count\n const { meta, records } = await this.handleNonGroupFind(qb, populateMedia, offset, limit, alias);\n return {\n meta,\n records,\n }\n }\n }\n\n private async handleNonGroupFind(qb: SelectQueryBuilder<T>, populateMedia: string[], offset: number, limit: number, alias: string) {\n const [entities, count] = await qb.getManyAndCount();\n\n // Populate the entity with the media\n if (populateMedia && populateMedia.length > 0) {\n await this.handlePopulateMedia(populateMedia, entities);\n }\n\n return this.wrapFindResponse(offset, limit, count, entities);\n }\n\n private async handleGroupFind(qb: SelectQueryBuilder<T>, groupFilter: BasicFilterDto, populateGroup: boolean, alias: string, populateMedia: string[]) {\n const groupByResult = await qb.getRawMany();\n\n const groupMeta = [];\n const groupRecords = [];\n // For each group, get the records and the count\n for (const group of groupByResult) {\n if (populateGroup) {\n let groupByQb: SelectQueryBuilder<T> = this.repo.createQueryBuilder(alias);\n groupByQb = this.crudHelperService.buildFilterQuery(groupByQb, groupFilter, alias);\n groupByQb = this.crudHelperService.buildGroupByRecordsQuery(groupByQb, group, alias);\n const [entities, count] = await groupByQb.getManyAndCount();\n\n // Populate the entity with the media\n if (populateMedia && populateMedia.length > 0) {\n await this.handlePopulateMedia(populateMedia, entities);\n }\n const groupData = this.wrapFindResponse(groupFilter.offset, groupFilter.limit, count, entities);\n groupRecords.push(this.crudHelperService.createGroupRecords(group, alias, groupData));\n }\n groupMeta.push(this.crudHelperService.createGroupMeta(group, alias));\n }\n return { groupMeta, groupRecords };\n }\n\n private wrapFindResponse(offset: number, limit: number, count: number, entities: T[]) {\n const currentPage = Math.floor(offset / limit) + 1;\n const totalPages = Math.ceil(count / limit);\n\n const nextPage = currentPage < totalPages ? currentPage + 1 : null;\n const prevPage = currentPage > 1 ? currentPage - 1 : null;\n\n const r = {\n meta: {\n totalRecords: count,\n currentPage: currentPage,\n nextPage: nextPage,\n prevPage: prevPage,\n totalPages: totalPages,\n perPage: +limit,\n },\n records: entities\n };\n return r;\n }\n\n private async handlePopulateMedia(populateMedia: string[], entities: T[]) {\n const model = await this.entityManager.getRepository(ModelMetadata).findOne({\n where: {\n singularName: this.modelName,\n },\n relations: ['fields', 'fields.mediaStorageProvider', 'fields.model', 'module'],\n });\n\n // Will iterate through every entity & all populateMedia & call getMediaDetails for each field\n for (const entity of entities) {\n for (const mediaFieldPath of populateMedia) {\n await this.populateMediaObject(mediaFieldPath, model, entity);\n }\n }\n return entities;\n }\n\n // Adds the media with full URL to the entity / nested entity\n private async populateMediaObject(mediaFieldPath: string, model: ModelMetadata, entity: T) {\n if (mediaFieldPath.includes('.')) { // mediaFieldPath is a nested field\n const pathParts = mediaFieldPath.split('.');\n const mediaFieldMetadata = await this.getFieldMetadataRecursively(pathParts, model.fields);\n if (!mediaFieldMetadata) {\n throw new BadRequestException(`Media field ${mediaFieldPath} not found in model ${this.modelName}`);\n }\n\n // We can assume that the media field entity model is already populated as part of the entity data\n const mediaFieldEntities = this.getMediaFieldEntities(entity, pathParts);\n if (!mediaFieldEntities || mediaFieldEntities.length === 0) {\n return;//no need to populate data if relation not exists\n }\n // Populate the media field entities with the full URL\n for (const mediaFieldEntity of mediaFieldEntities) {\n const mediaWithFullUrl = await this.getMediaWithFullUrl(mediaFieldEntity, mediaFieldMetadata);\n this.appendMediaKey(mediaWithFullUrl, mediaFieldEntity, mediaFieldMetadata.name);\n // mediaFieldEntity['_media'][mediaFieldPath] = mediaWithFullUrl\n }\n }\n else {\n // mediaFieldPath is a single field\n const mediaFieldMetadata = model.fields.find(field => field.name === mediaFieldPath);\n if (!mediaFieldMetadata) {\n throw new BadRequestException(`Media field ${mediaFieldPath} not found in model ${this.modelName}`);\n }\n const mediaWithFullUrl = await this.getMediaWithFullUrl(entity, mediaFieldMetadata);\n this.appendMediaKey(mediaWithFullUrl, entity, mediaFieldPath);\n // entity['_media'][mediaFieldPath] = mediaWithFullUrl;\n }\n }\n\n // // Add the media with full URL to the entity\n private appendMediaKey(mediaWithFullUrl: MediaWithFullUrl[], entity: T, mediaFieldPath: string) {\n // if _media key already exists, append the new media to the existing array\n if (entity['_media']) {\n entity['_media'][mediaFieldPath] = mediaWithFullUrl;\n }\n else {\n entity['_media'] = {\n [mediaFieldPath]: mediaWithFullUrl\n };\n }\n }\n\n private getMediaFieldEntities(entity: T, mediaPathParts: string[]): T[] {\n let entityPart = entity;\n for (let i = 0; i < mediaPathParts.length - 1; i++) {\n const pathPart = mediaPathParts[i];\n if (entity[pathPart]) {\n entityPart = entity[pathPart];\n } else {\n throw new BadRequestException(`Media field ${pathPart} not found in entity ${JSON.stringify(entity)}`);\n }\n }\n return isArray(entityPart) ? entityPart : [entityPart];\n }\n\n async getMediaWithFullUrl(mediaEntity: any, mediaFieldMetadata: FieldMetadata): Promise<MediaWithFullUrl[]> {\n const storageProviderMetadata = mediaFieldMetadata.mediaStorageProvider;\n const storageProviderType = storageProviderMetadata.type as MediaStorageProviderType;\n const storageProvider = await getMediaStorageProvider(this.moduleRef, storageProviderType);\n const mediaDetails = await storageProvider.retrieve(mediaEntity, mediaFieldMetadata);\n return mediaDetails as MediaWithFullUrl[];\n }\n\n async findOne(id: number, query: any, solidRequestContext: any = {}) {\n const { populate = [], fields = [], populateMedia = [] } = query;\n\n // const normalizedFields = this.crudHelperService.normalize(fields);\n const normalizedPopulate = this.crudHelperService.normalize(populate);\n const normalizedPopulateMedia = this.crudHelperService.normalize(populateMedia);\n\n // if normalizedPopulateMedia, has any nested media paths, then add then to populate excluding the last part\n const additionalPopulate = this.crudHelperService.additionalRelationsRequiredForMediaPopulation(normalizedPopulateMedia);\n // Add the additional populate relations to the normalizedPopulate, if they are not already present\n normalizedPopulate.push(...additionalPopulate.filter((relation) => !normalizedPopulate.includes(relation)));\n\n const model = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasReadPermissionOnModel(solidRequestContext.activeUser, model.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n\n let entity = await this.repo.findOne({\n where: {\n //@ts-ignore\n id: id,\n },\n relations: normalizedPopulate,\n select: fields,\n });\n if (!entity) {\n throw new NotFoundException(`Entity [${this.moduleName}.${this.modelName}] with id ${id} not found`);\n }\n // Populate the entity with the media\n if (normalizedPopulateMedia.length > 0) {\n const populatedEntities = await this.handlePopulateMedia(normalizedPopulateMedia, [entity]);\n entity = populatedEntities[0] as Awaited<T>;\n }\n return entity;\n }\n\n async insertMany(createDtos: any[], filesArray: Express.Multer.File[][] = [], solidRequestContext: any = {}): Promise<T[]> {\n\n\n // if (createDtos.length !== filesArray.length) {\n // throw new BadRequestException('Mismatch between data objects and file arrays.');\n // }\n\n const loadedmodel = await this.loadModel();\n // Check wheather user has create permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasCreatePermissionOnModel(solidRequestContext.activeUser, loadedmodel.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n\n // Fetch model metadata once\n const model = await this.modelMetadataService.findOneBySingularName(this.modelName, {\n fields: {\n model: true,\n mediaStorageProvider: true,\n },\n module: true,\n });\n\n // Process each createDto in parallel\n const createAndSavePromises = createDtos.map(async (createDto, index) => {\n const files = []; // TODO, This is set, because we are not supporting files for insertMany currently\n let hasMediaFields = false;\n\n // Process each field\n for (const field of model.fields) {\n const fieldManager: FieldCrudManager = await this.fieldCrudManager(field, this.entityManager);\n const validationErrors = await fieldManager.validate(createDto, files);\n if (validationErrors.length > 0) {\n throw new BadRequestException(`Validation errors in ${field.name} are invalid: ${validationErrors.map(e => e.error).join(', ')}`);\n }\n createDto = await fieldManager.transformForCreate(createDto);\n hasMediaFields = hasMediaFields || field.type === 'mediaSingle' || field.type === 'mediaMultiple';\n }\n\n // Save the entity\n const entity = this.repo.create(createDto);\n const savedEntity = await this.repo.save(entity) as T;\n\n //Commented since currently Files are not supported for insertmany\n // if (hasMediaFields) {\n // await this.saveMedia(model, files, savedEntity);\n // }\n\n return savedEntity;\n });\n\n // Await all promises in parallel\n const savedEntities = await Promise.all(createAndSavePromises);\n\n return savedEntities;\n }\n\n async deleteMany(ids: number[], solidRequestContext: any = {}): Promise<any> {\n\n if (!ids || ids.length === 0) {\n throw new Error(ERROR_MESSAGES.DELETE_IDS_REQUIRED);\n }\n\n const loadedmodel = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasDeletePermissionOnModel(solidRequestContext.activeUser, loadedmodel.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n const model = await this.modelMetadataService.findOneBySingularName(this.modelName, {\n fields: {\n model: true,\n mediaStorageProvider: true,\n },\n module: true,\n });\n\n\n\n const removedEntities = [];\n for (let i = 0; i < ids.length; i++) {\n const id = ids[i]\n const entity = await this.repo.findOne({\n where: {\n //@ts-ignore\n id: id,\n }\n });\n removedEntities.push(entity);\n }\n if (model.enableSoftDelete === true) {\n await this.repo.softRemove(removedEntities);\n return this.repo.save(removedEntities);\n } else {\n return this.repo.remove(removedEntities);\n }\n // return removedEntities\n }\n\n async recover(id: number, solidRequestContext: any = {}) {\n try {\n const loadedmodel = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasRecoverPermissionOnModel(solidRequestContext.activeUser, loadedmodel.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n\n const softDeletedRows = await this.repo.findOne({\n where: {\n //@ts-ignore\n id, deletedAt: Not(IsNull())\n },\n withDeleted: true,\n });\n\n if (!softDeletedRows) {\n throw new Error(ERROR_MESSAGES.NO_SOFT_DELETED_RECORD_FOUND);\n }\n\n await this.repo.update(id, {\n //@ts-ignore\n deletedAt: null, deletedTracker: \"not-deleted\"\n });\n\n return { message: SUCCESS_MESSAGES.RECORD_RECOVERED, data: softDeletedRows };\n } catch (error) {\n if (error instanceof QueryFailedError) {\n if ((error as any).code === '23505') {\n throw new Error(ERROR_MESSAGES.CONFLICTING_RECORD_ON_UNARCHIVE);\n }\n }\n\n throw new Error(error);\n }\n }\n\n async recoverMany(ids: number[], solidRequestContext: any = {}) {\n try {\n const loadedmodel = await this.loadModel();\n // Check wheather user has update permission for model\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasRecoverPermissionOnModel(solidRequestContext.activeUser, loadedmodel.singularName);\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n\n if (!ids || ids.length === 0) {\n throw new Error(ERROR_MESSAGES.DELETE_IDS_REQUIRED);\n }\n\n // Find soft-deleted records matching the given IDs\n const softDeletedRows = await this.repo.find({\n where: {\n //@ts-ignore\n id: In(ids),\n deletedAt: Not(IsNull()),\n },\n withDeleted: true,\n });\n\n if (softDeletedRows.length === 0) {\n throw new Error(ERROR_MESSAGES.NO_SOFT_DELETED_RECORDS_FOUND);\n }\n\n // Recover the specific records by setting deletedAt to null\n await this.repo.update(\n //@ts-ignore\n { id: In(ids) },\n { deletedAt: null, deletedTracker: \"not-deleted\" }\n );\n\n return { message: SUCCESS_MESSAGES.SELECTED_RECORDS_RECOVERED, recoveredIds: ids };\n } catch (error) {\n if (error instanceof QueryFailedError) {\n if ((error as any).code === \"23505\") {\n throw new Error(ERROR_MESSAGES.CONFLICTING_RECORD_ON_UNARCHIVE);\n }\n }\n\n throw new Error(error);\n }\n }\n\n\n async getFieldMetadataRecursively(pathParts: string[], fields: FieldMetadata[]) {\n if (!pathParts || pathParts.length === 0) {\n throw new BadRequestException(ERROR_MESSAGES.EMPTY_PATH_PARTS);\n }\n\n const [currentPart, ...remainingParts] = pathParts;\n const field = fields.find(field => field.name === currentPart);\n\n if (!field) {\n throw new BadRequestException(`Field ${currentPart} not found in model ${this.modelName}`);\n }\n\n // Base case: last part, return the field\n if (remainingParts.length === 0) {\n return field;\n }\n\n if (!field.relationCoModelSingularName) {\n throw new BadRequestException(`Field ${field.name} does not define a relationCoModelSingularName`);\n }\n\n const relationCoModel = await this.entityManager.getRepository(ModelMetadata).findOne({\n where: { singularName: field.relationCoModelSingularName },\n relations: ['fields', 'fields.mediaStorageProvider', 'fields.model'],\n });\n\n if (!relationCoModel) {\n throw new BadRequestException(`Model ${field.relationCoModelSingularName} not found`);\n }\n\n return this.getFieldMetadataRecursively(remainingParts, relationCoModel.fields);\n }\n\n async getUserKeyFieldNameForModel(modelSingularName: string): Promise<string> {\n const model = await this.modelMetadataService.findOneBySingularName(modelSingularName, ['userKeyField']);\n if (!model) {\n throw new BadRequestException(`Model ${modelSingularName} not found`);\n }\n return model.userKeyField?.name || '';\n }\n}\n\n"]}
|
|
@@ -15,6 +15,7 @@ import { ExcelService } from './excel.service';
|
|
|
15
15
|
import { SolidIntrospectService } from './solid-introspect.service';
|
|
16
16
|
import { ModelMetadata } from 'src/entities/model-metadata.entity';
|
|
17
17
|
import { UpdateExportTemplateDto } from 'src/dtos/update-export-template.dto';
|
|
18
|
+
import { ModelMetadataHelperService } from 'src/helpers/model-metadata-helper.service';
|
|
18
19
|
export interface ExportTransactionFileInfo {
|
|
19
20
|
exportStream: Readable;
|
|
20
21
|
fileName: string;
|
|
@@ -36,8 +37,9 @@ export declare class ExportTransactionService extends CRUDService<ExportTransact
|
|
|
36
37
|
readonly fieldRepo: Repository<FieldMetadata>;
|
|
37
38
|
readonly ModelMetadataRepo: Repository<ModelMetadata>;
|
|
38
39
|
readonly moduleRef: ModuleRef;
|
|
40
|
+
private readonly modelMetadataHelperService;
|
|
39
41
|
private logger;
|
|
40
|
-
constructor(modelMetadataService: ModelMetadataService, moduleMetadataService: ModuleMetadataService, configService: ConfigService, fileService: FileService, discoveryService: DiscoveryService, crudHelperService: CrudHelperService, entityManager: EntityManager, repo: Repository<ExportTransaction>, introspectService: SolidIntrospectService, excelService: ExcelService, csvService: CsvService, fieldRepo: Repository<FieldMetadata>, ModelMetadataRepo: Repository<ModelMetadata>, moduleRef: ModuleRef);
|
|
42
|
+
constructor(modelMetadataService: ModelMetadataService, moduleMetadataService: ModuleMetadataService, configService: ConfigService, fileService: FileService, discoveryService: DiscoveryService, crudHelperService: CrudHelperService, entityManager: EntityManager, repo: Repository<ExportTransaction>, introspectService: SolidIntrospectService, excelService: ExcelService, csvService: CsvService, fieldRepo: Repository<FieldMetadata>, ModelMetadataRepo: Repository<ModelMetadata>, moduleRef: ModuleRef, modelMetadataHelperService: ModelMetadataHelperService);
|
|
41
43
|
triggerExportSync(id: number, exportTransactionEntity: any, updateDto: UpdateExportTemplateDto, filters: any): Promise<ExportTransactionFileInfo>;
|
|
42
44
|
triggerExportAsync(id: number, exportTransactionEntity: any, updateDto: UpdateExportTemplateDto, filters: any): Promise<void>;
|
|
43
45
|
private loadExportTransaction;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"export-transaction.service.d.ts","sourceRoot":"","sources":["../../src/services/export-transaction.service.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,gBAAgB,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AAE3D,OAAO,EAAE,aAAa,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AAEpD,OAAO,EAAE,aAAa,EAAE,MAAM,gBAAgB,CAAC;AAC/C,OAAO,EAAE,iBAAiB,EAAE,MAAM,kCAAkC,CAAC;AACrE,OAAO,EAAE,WAAW,EAAE,MAAM,2BAA2B,CAAC;AACxD,OAAO,EAAE,WAAW,EAAE,MAAM,2BAA2B,CAAC;AAGxD,OAAO,EAAE,oBAAoB,EAAE,MAAM,qCAAqC,CAAC;AAC3E,OAAO,EAAE,qBAAqB,EAAE,MAAM,sCAAsC,CAAC;AAO7E,OAAO,EAAE,0BAA0B,EAAE,MAAM,wCAAwC,CAAC;AAEpF,OAAO,EAAE,aAAa,EAAE,MAAM,oCAAoC,CAAC;AACnE,OAAO,EAAE,QAAQ,EAAE,MAAM,QAAQ,CAAC;AAClC,OAAO,EAAE,iBAAiB,EAAE,MAAM,uCAAuC,CAAC;AAC1E,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAE/C,OAAO,EAAE,sBAAsB,EAAE,MAAM,4BAA4B,CAAC;AACpE,OAAO,EAAE,aAAa,EAAE,MAAM,oCAAoC,CAAC;AACnE,OAAO,EAAE,uBAAuB,EAAE,MAAM,qCAAqC,CAAC;
|
|
1
|
+
{"version":3,"file":"export-transaction.service.d.ts","sourceRoot":"","sources":["../../src/services/export-transaction.service.ts"],"names":[],"mappings":"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"}
|
|
@@ -35,6 +35,7 @@ const mediaStorageProviders_1 = require("./mediaStorageProviders");
|
|
|
35
35
|
const solid_introspect_service_1 = require("./solid-introspect.service");
|
|
36
36
|
const model_metadata_entity_1 = require("../entities/model-metadata.entity");
|
|
37
37
|
const error_messages_1 = require("../constants/error-messages");
|
|
38
|
+
const model_metadata_helper_service_1 = require("../helpers/model-metadata-helper.service");
|
|
38
39
|
const EXPORT_CHUNK_SIZE = 100;
|
|
39
40
|
var ExportStatus;
|
|
40
41
|
(function (ExportStatus) {
|
|
@@ -48,7 +49,7 @@ var ExportFormat;
|
|
|
48
49
|
ExportFormat["EXCEL"] = "excel";
|
|
49
50
|
})(ExportFormat || (ExportFormat = {}));
|
|
50
51
|
let ExportTransactionService = ExportTransactionService_1 = class ExportTransactionService extends crud_service_1.CRUDService {
|
|
51
|
-
constructor(modelMetadataService, moduleMetadataService, configService, fileService, discoveryService, crudHelperService, entityManager, repo, introspectService, excelService, csvService, fieldRepo, ModelMetadataRepo, moduleRef) {
|
|
52
|
+
constructor(modelMetadataService, moduleMetadataService, configService, fileService, discoveryService, crudHelperService, entityManager, repo, introspectService, excelService, csvService, fieldRepo, ModelMetadataRepo, moduleRef, modelMetadataHelperService) {
|
|
52
53
|
super(modelMetadataService, moduleMetadataService, configService, fileService, discoveryService, crudHelperService, entityManager, repo, 'exportTransaction', 'solid-core', moduleRef);
|
|
53
54
|
this.modelMetadataService = modelMetadataService;
|
|
54
55
|
this.moduleMetadataService = moduleMetadataService;
|
|
@@ -64,6 +65,7 @@ let ExportTransactionService = ExportTransactionService_1 = class ExportTransact
|
|
|
64
65
|
this.fieldRepo = fieldRepo;
|
|
65
66
|
this.ModelMetadataRepo = ModelMetadataRepo;
|
|
66
67
|
this.moduleRef = moduleRef;
|
|
68
|
+
this.modelMetadataHelperService = modelMetadataHelperService;
|
|
67
69
|
this.logger = new common_1.Logger(ExportTransactionService_1.name);
|
|
68
70
|
}
|
|
69
71
|
async triggerExportSync(id, exportTransactionEntity, updateDto, filters) {
|
|
@@ -159,12 +161,11 @@ let ExportTransactionService = ExportTransactionService_1 = class ExportTransact
|
|
|
159
161
|
return (fileFormat === ExportFormat.EXCEL) ? 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' : 'text/csv';
|
|
160
162
|
}
|
|
161
163
|
async getDataRecordsFunc(fields, modelService, modelMetadata, filters) {
|
|
162
|
-
const
|
|
163
|
-
|
|
164
|
-
.map((field) => field.name);
|
|
164
|
+
const allModelFields = await this.modelMetadataHelperService.loadFieldHierarchy(modelMetadata.singularName);
|
|
165
|
+
const modelFields = allModelFields.filter((f) => fields.includes(f.name));
|
|
165
166
|
const relatedModelsUserKeyMap = new Map();
|
|
166
|
-
for (const field of
|
|
167
|
-
if (field.relationType && field.relationCoModelSingularName
|
|
167
|
+
for (const field of modelFields) {
|
|
168
|
+
if (field.relationType && field.relationCoModelSingularName) {
|
|
168
169
|
const relatedModelMetadata = await this.ModelMetadataRepo.findOne({
|
|
169
170
|
where: { singularName: field.relationCoModelSingularName },
|
|
170
171
|
relations: ['userKeyField'],
|
|
@@ -174,38 +175,90 @@ let ExportTransactionService = ExportTransactionService_1 = class ExportTransact
|
|
|
174
175
|
}
|
|
175
176
|
}
|
|
176
177
|
}
|
|
178
|
+
const fieldNameToDisplayName = new Map();
|
|
179
|
+
for (const field of modelFields || []) {
|
|
180
|
+
if (field.name) {
|
|
181
|
+
fieldNameToDisplayName.set(field.name, field.displayName ?? field.name);
|
|
182
|
+
}
|
|
183
|
+
}
|
|
177
184
|
return async (chunkIndex, chunkSize) => {
|
|
178
185
|
const offset = chunkIndex * chunkSize;
|
|
179
186
|
const recordFilterDto = {
|
|
180
187
|
limit: chunkSize,
|
|
181
188
|
offset,
|
|
182
|
-
populate:
|
|
189
|
+
populate: modelFields
|
|
190
|
+
.filter((f) => f.relationType !== null)
|
|
191
|
+
.map((f) => f.name),
|
|
183
192
|
};
|
|
184
193
|
const cleanedFilters = cleanNullsFromObject(filters);
|
|
185
194
|
if (cleanedFilters && Object.keys(cleanedFilters).length > 0) {
|
|
186
195
|
recordFilterDto.filters = cleanedFilters;
|
|
187
196
|
}
|
|
188
|
-
const nonRelationalFieldSet = new Set(modelMetadata?.fields
|
|
189
|
-
.filter((field) => fields.includes(field.name) && field.relationType === null)
|
|
190
|
-
.map((field) => field.name));
|
|
191
197
|
const data = await modelService.instance.find(recordFilterDto);
|
|
192
198
|
const records = data.records ?? [];
|
|
193
199
|
const cleanedRecords = records.map((record) => {
|
|
194
200
|
const newRecord = {};
|
|
195
|
-
for (const
|
|
196
|
-
|
|
201
|
+
for (const field of modelFields) {
|
|
202
|
+
if (!field.relationType) {
|
|
203
|
+
const displayKey = fieldNameToDisplayName.get(field.name) ?? field.name;
|
|
204
|
+
const fieldMeta = modelFields.find(f => f.name === field.name);
|
|
205
|
+
if ((fieldMeta?.type === 'datetime' || fieldMeta?.type === 'date') && record[field.name]) {
|
|
206
|
+
newRecord[displayKey] = new Date(record[field.name]).toISOString();
|
|
207
|
+
}
|
|
208
|
+
else {
|
|
209
|
+
newRecord[displayKey] = record[field.name];
|
|
210
|
+
}
|
|
211
|
+
}
|
|
197
212
|
}
|
|
198
213
|
for (const [relatedFieldName, userKeyFieldName] of relatedModelsUserKeyMap.entries()) {
|
|
199
214
|
const relatedData = record[relatedFieldName];
|
|
215
|
+
const displayKey = fieldNameToDisplayName.get(relatedFieldName) ?? relatedFieldName;
|
|
200
216
|
if (Array.isArray(relatedData)) {
|
|
201
|
-
const values = relatedData
|
|
202
|
-
|
|
217
|
+
const values = relatedData
|
|
218
|
+
.map(item => {
|
|
219
|
+
let val = item?.[userKeyFieldName];
|
|
220
|
+
const relatedFieldMeta = modelFields.find(f => f.name === relatedFieldName);
|
|
221
|
+
if ((relatedFieldMeta?.type === 'datetime' || relatedFieldMeta?.type === 'date') && val) {
|
|
222
|
+
val = new Date(val).toISOString();
|
|
223
|
+
}
|
|
224
|
+
return val;
|
|
225
|
+
})
|
|
226
|
+
.filter(Boolean);
|
|
227
|
+
newRecord[displayKey] = values.join(', ');
|
|
203
228
|
}
|
|
204
229
|
else if (relatedData && typeof relatedData === 'object') {
|
|
205
230
|
newRecord[relatedFieldName] = relatedData?.[userKeyFieldName] ?? null;
|
|
206
231
|
}
|
|
207
232
|
else {
|
|
208
|
-
newRecord[
|
|
233
|
+
newRecord[displayKey] = null;
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
for (const [relatedFieldName, userKeyFieldName] of relatedModelsUserKeyMap.entries()) {
|
|
237
|
+
const relatedData = record[relatedFieldName];
|
|
238
|
+
const displayKey = fieldNameToDisplayName.get(relatedFieldName) ?? relatedFieldName;
|
|
239
|
+
if (Array.isArray(relatedData)) {
|
|
240
|
+
const values = relatedData
|
|
241
|
+
.map(item => {
|
|
242
|
+
const val = item?.[userKeyFieldName];
|
|
243
|
+
const relatedFieldMeta = modelFields.find(f => f.name === relatedFieldName);
|
|
244
|
+
if ((relatedFieldMeta?.type === 'datetime' || relatedFieldMeta?.type === 'date') && val) {
|
|
245
|
+
return new Date(val).toISOString();
|
|
246
|
+
}
|
|
247
|
+
return val;
|
|
248
|
+
})
|
|
249
|
+
.filter(Boolean);
|
|
250
|
+
newRecord[displayKey] = values.join(', ');
|
|
251
|
+
}
|
|
252
|
+
else if (relatedData && typeof relatedData === 'object') {
|
|
253
|
+
let val = relatedData?.[userKeyFieldName] ?? null;
|
|
254
|
+
const relatedFieldMeta = modelFields.find(f => f.name === relatedFieldName);
|
|
255
|
+
if ((relatedFieldMeta?.type === 'datetime' || relatedFieldMeta?.type === 'date') && val) {
|
|
256
|
+
val = new Date(val).toISOString();
|
|
257
|
+
}
|
|
258
|
+
newRecord[displayKey] = val;
|
|
259
|
+
}
|
|
260
|
+
else {
|
|
261
|
+
newRecord[displayKey] = null;
|
|
209
262
|
}
|
|
210
263
|
}
|
|
211
264
|
return newRecord;
|
|
@@ -243,7 +296,8 @@ exports.ExportTransactionService = ExportTransactionService = ExportTransactionS
|
|
|
243
296
|
csv_service_1.CsvService,
|
|
244
297
|
typeorm_2.Repository,
|
|
245
298
|
typeorm_2.Repository,
|
|
246
|
-
core_1.ModuleRef
|
|
299
|
+
core_1.ModuleRef,
|
|
300
|
+
model_metadata_helper_service_1.ModelMetadataHelperService])
|
|
247
301
|
], ExportTransactionService);
|
|
248
302
|
function cleanNullsFromObject(obj) {
|
|
249
303
|
if (Array.isArray(obj)) {
|