rez_core 5.0.181 → 6.5.0
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/.claude/settings.local.json +26 -0
- package/.idea/250218_nodejs_core.iml +8 -11
- package/.idea/codeStyles/Project.xml +58 -58
- package/.idea/codeStyles/codeStyleConfig.xml +4 -4
- package/.idea/copilot.data.migration.agent.xml +6 -0
- package/.idea/copilot.data.migration.ask.xml +6 -0
- package/.idea/copilot.data.migration.ask2agent.xml +6 -0
- package/.idea/copilot.data.migration.edit.xml +6 -0
- package/.idea/inspectionProfiles/Project_Default.xml +1 -1
- package/.idea/misc.xml +6 -0
- package/.idea/modules.xml +7 -7
- package/.idea/prettier.xml +5 -5
- package/.idea/vcs.xml +5 -5
- package/.prettierrc +3 -3
- package/README.md +99 -99
- package/dist/module/auth/guards/role.guard.js +3 -3
- package/dist/module/enterprise/controller/enterprise.controller.d.ts +12 -0
- package/dist/module/enterprise/controller/enterprise.controller.js +57 -0
- package/dist/module/enterprise/controller/enterprise.controller.js.map +1 -0
- package/dist/module/enterprise/controller/organization.controller.d.ts +11 -1
- package/dist/module/enterprise/controller/organization.controller.js +62 -2
- package/dist/module/enterprise/controller/organization.controller.js.map +1 -1
- package/dist/module/enterprise/enterprise.module.js +2 -1
- package/dist/module/enterprise/enterprise.module.js.map +1 -1
- package/dist/module/enterprise/entity/enterprise.entity.d.ts +1 -3
- package/dist/module/enterprise/entity/enterprise.entity.js +4 -12
- package/dist/module/enterprise/entity/enterprise.entity.js.map +1 -1
- package/dist/module/enterprise/entity/organization.entity.d.ts +1 -17
- package/dist/module/enterprise/entity/organization.entity.js +4 -72
- package/dist/module/enterprise/entity/organization.entity.js.map +1 -1
- package/dist/module/enterprise/repository/enterprise.repository.d.ts +2 -2
- package/dist/module/enterprise/repository/enterprise.repository.js +9 -4
- package/dist/module/enterprise/repository/enterprise.repository.js.map +1 -1
- package/dist/module/enterprise/service/enterprise.service.d.ts +2 -2
- package/dist/module/enterprise/service/enterprise.service.js +4 -4
- package/dist/module/enterprise/service/enterprise.service.js.map +1 -1
- package/dist/module/enterprise/service/organization.service.d.ts +2 -2
- package/dist/module/enterprise/service/organization.service.js +97 -20
- package/dist/module/enterprise/service/organization.service.js.map +1 -1
- package/dist/module/filter/entity/saved-filter-master.entity.d.ts +1 -0
- package/dist/module/filter/entity/saved-filter-master.entity.js +4 -0
- package/dist/module/filter/entity/saved-filter-master.entity.js.map +1 -1
- package/dist/module/filter/repository/saved-filter.repository.js +2 -0
- package/dist/module/filter/repository/saved-filter.repository.js.map +1 -1
- package/dist/module/filter/service/filter.service.js +19 -19
- package/dist/module/integration/examples/usage.example.js +9 -9
- package/dist/module/meta/repository/attribute-master.repository.js +8 -8
- package/dist/module/meta/service/entity-dynamic.service.js +16 -16
- package/dist/module/meta/service/media-data.service.js +6 -6
- package/dist/module/meta/service/resolver.service.js +11 -11
- package/dist/module/module/repository/menu.repository.js +4 -4
- package/dist/module/notification/service/notification.service.js +6 -6
- package/dist/module/user/controller/login.controller.js +18 -18
- package/dist/module/workflow/repository/action.repository.js +2 -2
- package/dist/module/workflow/repository/stage.repository.js +8 -8
- package/dist/module/workflow/service/action-template-mapping.service.js +2 -2
- package/dist/module/workflow/service/action.service.js +5 -5
- package/dist/module/workflow/service/entity-modification.service.js +2 -2
- package/dist/module/workflow/service/task.service.js +8 -8
- package/dist/module/workflow-automation/service/schedule-handler.service.js +9 -9
- package/dist/table.config.d.ts +1 -2
- package/dist/tsconfig.build.tsbuildinfo +1 -1
- package/dist/utils/service/reflection-helper.service.js +2 -2
- package/docs/modules/event-driven-integration-design.md +91 -91
- package/docs/modules/integration.md +250 -250
- package/eslint.config.mjs +34 -34
- package/nest-cli.json +14 -14
- package/package.json +125 -125
- package/server.log +850 -0
- package/src/app.controller.ts +12 -12
- package/src/app.module.ts +68 -68
- package/src/app.service.ts +8 -8
- package/src/config/bull.config.ts +69 -69
- package/src/config/config.module.ts +17 -17
- package/src/config/database.config.ts +48 -48
- package/src/constant/global.constant.ts +67 -67
- package/src/core.module.ts +94 -94
- package/src/decorators/roles.decorator.ts +7 -7
- package/src/dtos/response.dto.ts +6 -6
- package/src/dtos/response.ts +5 -5
- package/src/index.ts +1 -1
- package/src/module/auth/auth.module.ts +49 -49
- package/src/module/auth/controller/auth.controller.ts +28 -28
- package/src/module/auth/guards/google-auth.guard.ts +9 -9
- package/src/module/auth/guards/jwt.guard.ts +22 -22
- package/src/module/auth/guards/role.guard.ts +68 -68
- package/src/module/auth/services/auth.service.ts +56 -56
- package/src/module/auth/services/jwt.service.ts +11 -11
- package/src/module/auth/strategies/google.strategy.ts +54 -54
- package/src/module/auth/strategies/jwt.strategy.ts +58 -58
- package/src/module/auth/strategies/local.strategy.ts +13 -13
- package/src/module/dashboard/controller/dashboard.controller.ts +38 -38
- package/src/module/dashboard/dashboard.module.ts +21 -21
- package/src/module/dashboard/entity/dashboard_page_data.entity.ts +27 -27
- package/src/module/dashboard/entity/widget_master.entity.ts +18 -18
- package/src/module/dashboard/repository/dashboard.repository.ts +49 -49
- package/src/module/dashboard/service/dashboard.service.ts +72 -72
- package/src/module/enterprise/controller/enterprise.controller.ts +40 -0
- package/src/module/enterprise/controller/organization.controller.ts +93 -36
- package/src/module/enterprise/enterprise.module.ts +46 -45
- package/src/module/enterprise/entity/enterprise.entity.ts +31 -37
- package/src/module/enterprise/entity/organization-app-mapping.entity.ts +13 -13
- package/src/module/enterprise/entity/organization.entity.ts +40 -92
- package/src/module/enterprise/repository/enterprise.repository.ts +39 -31
- package/src/module/enterprise/repository/organization.repository.ts +26 -26
- package/src/module/enterprise/repository/school.repository.ts +272 -272
- package/src/module/enterprise/service/brand.service.ts +5 -5
- package/src/module/enterprise/service/enterprise.service.ts +22 -16
- package/src/module/enterprise/service/organization-app-mapping.service.ts +4 -4
- package/src/module/enterprise/service/organization.service.ts +262 -145
- package/src/module/entity_json/controller/entity_json.controller.ts +47 -47
- package/src/module/entity_json/entity/entityJson.entity.ts +39 -39
- package/src/module/entity_json/entity_json.module.ts +18 -18
- package/src/module/entity_json/service/entityJson.repository.ts +37 -37
- package/src/module/entity_json/service/entity_json.service.ts +241 -241
- package/src/module/export/controller/export.controller.ts +83 -83
- package/src/module/export/export.module.ts +14 -14
- package/src/module/export/service/export.service.ts +105 -105
- package/src/module/filter/controller/filter.controller.ts +87 -87
- package/src/module/filter/dto/filter-request.dto.ts +39 -39
- package/src/module/filter/entity/saved-filter-detail.entity.ts +41 -41
- package/src/module/filter/entity/saved-filter-master.entity.ts +35 -32
- package/src/module/filter/filter.module.ts +33 -33
- package/src/module/filter/repository/saved-filter.repository.ts +247 -258
- package/src/module/filter/repository/saved.filter-detail.repository.ts +19 -19
- package/src/module/filter/service/filter-evaluator.service.ts +82 -82
- package/src/module/filter/service/filter.service.ts +1316 -1316
- package/src/module/filter/service/saved-filter.service.ts +164 -164
- package/src/module/ics/controller/ics.controller.ts +21 -21
- package/src/module/ics/dto/ics.dto.ts +55 -55
- package/src/module/ics/ics.module.ts +13 -13
- package/src/module/ics/service/ics.service.ts +57 -57
- package/src/module/integration/controller/calender-event.controller.ts +31 -31
- package/src/module/integration/controller/integration.controller.ts +662 -662
- package/src/module/integration/controller/wrapper.controller.ts +37 -37
- package/src/module/integration/dto/create-config.dto.ts +526 -526
- package/src/module/integration/entity/integration-config.entity.ts +112 -112
- package/src/module/integration/entity/integration-entity-mapper.entity.ts +14 -14
- package/src/module/integration/entity/integration-source.entity.ts +17 -17
- package/src/module/integration/entity/user-integration.entity.ts +71 -71
- package/src/module/integration/examples/usage.example.ts +338 -338
- package/src/module/integration/factories/base.factory.ts +7 -7
- package/src/module/integration/factories/email.factory.ts +49 -49
- package/src/module/integration/factories/integration.factory.ts +121 -121
- package/src/module/integration/factories/sms.factory.ts +51 -51
- package/src/module/integration/factories/telephone.factory.ts +41 -41
- package/src/module/integration/factories/whatsapp.factory.ts +56 -56
- package/src/module/integration/integration.module.ts +110 -110
- package/src/module/integration/service/calendar-event.service.ts +118 -118
- package/src/module/integration/service/integration-entity-mapper.service.ts +17 -17
- package/src/module/integration/service/integration-queue.service.ts +229 -229
- package/src/module/integration/service/integration.service.ts +2634 -2634
- package/src/module/integration/service/oauth.service.ts +224 -224
- package/src/module/integration/service/wrapper.service.ts +753 -753
- package/src/module/integration/strategies/email/gmail-api.strategy.ts +280 -280
- package/src/module/integration/strategies/email/outlook-api.strategy.ts +44 -44
- package/src/module/integration/strategies/email/outlook.strategy.ts +64 -64
- package/src/module/integration/strategies/email/sendgrid-api.strategy.ts +260 -260
- package/src/module/integration/strategies/integration.strategy.ts +97 -97
- package/src/module/integration/strategies/sms/gupshup-sms.strategy.ts +146 -146
- package/src/module/integration/strategies/sms/msg91-sms.strategy.ts +164 -164
- package/src/module/integration/strategies/sms/tubelight-sms.strategy.ts +163 -163
- package/src/module/integration/strategies/telephone/ozonetel-voice.strategy.ts +238 -238
- package/src/module/integration/strategies/telephone/tubelight-voice.strategy.ts +210 -210
- package/src/module/integration/strategies/whatsapp/gupshup-whatsapp.strategy.ts +359 -359
- package/src/module/integration/strategies/whatsapp/tubelight-whatsapp.strategy.ts +372 -372
- package/src/module/integration/strategies/whatsapp/whatsapp-cloud.strategy.ts +403 -403
- package/src/module/integration/strategies/whatsapp/whatsapp.strategy.ts +57 -57
- package/src/module/layout/controller/layout.controller.ts +47 -47
- package/src/module/layout/entity/header-items.entity.ts +28 -28
- package/src/module/layout/entity/header-section.entity.ts +19 -19
- package/src/module/layout/layout.module.ts +21 -21
- package/src/module/layout/repository/header-items.repository.ts +18 -18
- package/src/module/layout/repository/header-section.repository.ts +22 -22
- package/src/module/layout/service/header-section.service.ts +25 -25
- package/src/module/layout_preference/controller/layout_preference.controller.ts +76 -76
- package/src/module/layout_preference/entity/layout_preference.entity.ts +28 -28
- package/src/module/layout_preference/layout_preference.module.ts +22 -22
- package/src/module/layout_preference/repository/layout_preference.repository.ts +65 -65
- package/src/module/layout_preference/service/layout_preference.service.ts +191 -191
- package/src/module/lead/controller/lead.controller.ts +30 -30
- package/src/module/lead/lead.module.ts +14 -14
- package/src/module/lead/repository/lead.repository.ts +41 -41
- package/src/module/lead/service/lead.service.ts +54 -54
- package/src/module/linked_attributes/controller/linked_attributes.controller.ts +37 -37
- package/src/module/linked_attributes/entity/linked_attribute.entity.ts +51 -51
- package/src/module/linked_attributes/linked_attributes.module.ts +16 -16
- package/src/module/linked_attributes/repository/linked_attribute.repository.ts +12 -12
- package/src/module/linked_attributes/service/linked_attributes.service.ts +73 -73
- package/src/module/listmaster/controller/list-master.controller.ts +230 -230
- package/src/module/listmaster/entity/list-master-items.entity.ts +43 -43
- package/src/module/listmaster/entity/list-master.entity.ts +33 -33
- package/src/module/listmaster/listmaster.module.ts +46 -46
- package/src/module/listmaster/repository/list-master-items.repository.ts +173 -173
- package/src/module/listmaster/repository/list-master.repository.ts +56 -56
- package/src/module/listmaster/service/list-master-engine.ts +19 -19
- package/src/module/listmaster/service/list-master-extension.interface.ts +4 -4
- package/src/module/listmaster/service/list-master-item.service.ts +280 -280
- package/src/module/listmaster/service/list-master-registry.ts +15 -15
- package/src/module/listmaster/service/list-master.service.ts +527 -527
- package/src/module/mapper/controller/field-mapper.controller.ts +76 -76
- package/src/module/mapper/controller/mapper.controller.ts +20 -20
- package/src/module/mapper/dto/field-mapper.dto.ts +14 -14
- package/src/module/mapper/entity/field-lovs.entity.ts +19 -19
- package/src/module/mapper/entity/field-mapper.entity.ts +53 -53
- package/src/module/mapper/entity/mapper.entity.ts +16 -16
- package/src/module/mapper/mapper.module.ts +35 -35
- package/src/module/mapper/repository/field-lovs.repository.ts +35 -35
- package/src/module/mapper/repository/field-mapper.repository.ts +42 -42
- package/src/module/mapper/repository/mapper.repository.ts +32 -32
- package/src/module/mapper/service/field-mapper.service.ts +269 -269
- package/src/module/mapper/service/mapper.service.ts +80 -80
- package/src/module/master/controller/master.controller.ts +74 -74
- package/src/module/master/service/master.service.ts +484 -484
- package/src/module/meta/controller/app-master.controller.ts +38 -38
- package/src/module/meta/controller/attribute-master.controller.ts +84 -84
- package/src/module/meta/controller/entity-dynamic.controller.ts +125 -125
- package/src/module/meta/controller/entity-master.controller.ts +41 -41
- package/src/module/meta/controller/entity-relation.controller.ts +36 -36
- package/src/module/meta/controller/entity.controller.ts +308 -308
- package/src/module/meta/controller/entity.public.controller.ts +75 -75
- package/src/module/meta/controller/media.controller.ts +135 -135
- package/src/module/meta/controller/meta.controller.ts +101 -101
- package/src/module/meta/controller/view-master.controller.ts +79 -79
- package/src/module/meta/dto/entity-list-data.dto.ts +6 -6
- package/src/module/meta/dto/entity-tab.dto.ts +4 -4
- package/src/module/meta/dto/entity-table.dto.ts +12 -12
- package/src/module/meta/entity/app-master.entity.ts +37 -37
- package/src/module/meta/entity/attribute-master.entity.ts +92 -92
- package/src/module/meta/entity/base-entity.entity.ts +75 -75
- package/src/module/meta/entity/entity-master.entity.ts +85 -85
- package/src/module/meta/entity/entity-relation-data.entity.ts +29 -29
- package/src/module/meta/entity/entity-relation.entity.ts +23 -23
- package/src/module/meta/entity/entity-table-column.entity.ts +61 -61
- package/src/module/meta/entity/entity-table.entity.ts +50 -50
- package/src/module/meta/entity/media-data.entity.ts +32 -32
- package/src/module/meta/entity/preference.entity.ts +62 -62
- package/src/module/meta/entity/view-master.entity.ts +41 -41
- package/src/module/meta/entity.module.ts +165 -165
- package/src/module/meta/repository/app-master.repository.ts +20 -20
- package/src/module/meta/repository/attribute-master.repository.ts +164 -164
- package/src/module/meta/repository/entity-attribute-update.repository.ts +48 -48
- package/src/module/meta/repository/entity-master.repository.ts +120 -120
- package/src/module/meta/repository/entity-relation.repository.ts +22 -22
- package/src/module/meta/repository/entity-table-column.repository.ts +39 -39
- package/src/module/meta/repository/entity-table.repository.ts +53 -53
- package/src/module/meta/repository/media-data.repository.ts +50 -50
- package/src/module/meta/repository/preference.repository.ts +20 -20
- package/src/module/meta/repository/user-app-mapping.repository.ts +28 -28
- package/src/module/meta/repository/view-master.repository.ts +42 -42
- package/src/module/meta/service/app-master.service.ts +37 -37
- package/src/module/meta/service/attribute-master.service.ts +132 -132
- package/src/module/meta/service/common.service.ts +9 -9
- package/src/module/meta/service/entity-attribute-update.service.ts +26 -26
- package/src/module/meta/service/entity-dynamic.service.ts +824 -824
- package/src/module/meta/service/entity-master.service.ts +171 -171
- package/src/module/meta/service/entity-realation-data.service.ts +9 -9
- package/src/module/meta/service/entity-relation.service.ts +74 -74
- package/src/module/meta/service/entity-service-impl.service.ts +388 -388
- package/src/module/meta/service/entity-table-column.service.ts +26 -26
- package/src/module/meta/service/entity-table.service.ts +157 -157
- package/src/module/meta/service/entity-validation.service.ts +188 -188
- package/src/module/meta/service/entity.service.ts +49 -49
- package/src/module/meta/service/field-group.service.ts +103 -103
- package/src/module/meta/service/media-data.service.ts +591 -591
- package/src/module/meta/service/populate-meta.service.ts +222 -222
- package/src/module/meta/service/preference.service.ts +16 -16
- package/src/module/meta/service/resolver.service.ts +280 -280
- package/src/module/meta/service/section-master.service.ts +104 -104
- package/src/module/meta/service/update-form-json.service.ts +22 -22
- package/src/module/meta/service/user-app-mapping.service.ts +17 -17
- package/src/module/meta/service/view-master.service.ts +127 -127
- package/src/module/microservice-client/microservice-clients.module.ts +13 -13
- package/src/module/microservice-client/service/microservice-client-factory.ts +37 -37
- package/src/module/microservice-client/service/microservice-clients.ts +4 -4
- package/src/module/module/controller/menu.controller.ts +15 -15
- package/src/module/module/controller/module-access.controller.ts +133 -133
- package/src/module/module/entity/menu.entity.ts +43 -43
- package/src/module/module/entity/module-access.entity.ts +25 -25
- package/src/module/module/entity/module-action.entity.ts +17 -17
- package/src/module/module/entity/module.entity.ts +52 -52
- package/src/module/module/module.module.ts +42 -42
- package/src/module/module/repository/menu.repository.ts +186 -186
- package/src/module/module/repository/module-access.repository.ts +344 -344
- package/src/module/module/service/menu.service.ts +82 -82
- package/src/module/module/service/module-access.service.ts +189 -189
- package/src/module/notification/controller/notification.controller.ts +58 -58
- package/src/module/notification/controller/otp.controller.ts +117 -117
- package/src/module/notification/entity/notification.entity.ts +26 -26
- package/src/module/notification/entity/otp.entity.ts +28 -28
- package/src/module/notification/firebase-admin.config.ts +22 -22
- package/src/module/notification/notification.module.ts +69 -69
- package/src/module/notification/repository/otp.repository.ts +27 -27
- package/src/module/notification/service/email.service.ts +127 -127
- package/src/module/notification/service/notification.service.ts +164 -164
- package/src/module/notification/service/otp.service.ts +133 -133
- package/src/module/third-party-module/entity/third-party-api-registry.entity.ts +52 -52
- package/src/module/third-party-module/repository/third-party-api-registry.repository.ts +20 -20
- package/src/module/third-party-module/service/api-registry.service.ts +13 -13
- package/src/module/third-party-module/third-party.module.ts +12 -12
- package/src/module/user/controller/login.controller.ts +198 -198
- package/src/module/user/controller/user.controller.ts +40 -40
- package/src/module/user/dto/create-user.dto.ts +62 -62
- package/src/module/user/dto/update-user.dto.ts +4 -4
- package/src/module/user/entity/role.entity.ts +33 -33
- package/src/module/user/entity/user-role-mapping.entity.ts +38 -38
- package/src/module/user/entity/user-session.entity.ts +73 -73
- package/src/module/user/entity/user.entity.ts +62 -62
- package/src/module/user/repository/role.repository.ts +96 -96
- package/src/module/user/repository/user-role-mapping.repository.ts +126 -126
- package/src/module/user/repository/user.repository.ts +50 -50
- package/src/module/user/repository/userSession.repository.ts +33 -33
- package/src/module/user/service/login.service.ts +326 -326
- package/src/module/user/service/role.service.ts +197 -197
- package/src/module/user/service/user-role-mapping.service.ts +98 -98
- package/src/module/user/service/user-session.service.ts +200 -200
- package/src/module/user/service/user.service.ts +368 -368
- package/src/module/user/user.module.ts +65 -65
- package/src/module/workflow/controller/action-category.controller.ts +54 -54
- package/src/module/workflow/controller/action-resource-mapping.controller.ts +23 -23
- package/src/module/workflow/controller/action-template-mapping.controller.ts +35 -35
- package/src/module/workflow/controller/action.controller.ts +111 -111
- package/src/module/workflow/controller/activity-log.controller.ts +55 -55
- package/src/module/workflow/controller/comm-template.controller.ts +43 -43
- package/src/module/workflow/controller/entity-modification.controller.ts +35 -35
- package/src/module/workflow/controller/form-master.controller.ts +43 -43
- package/src/module/workflow/controller/stage-group.controller.ts +49 -49
- package/src/module/workflow/controller/stage.controller.ts +51 -51
- package/src/module/workflow/controller/task.controller.ts +77 -77
- package/src/module/workflow/controller/workflow-list-master.controller.ts +44 -44
- package/src/module/workflow/controller/workflow-meta.controller.ts +80 -80
- package/src/module/workflow/controller/workflow.controller.ts +67 -67
- package/src/module/workflow/entity/action-category.entity.ts +38 -38
- package/src/module/workflow/entity/action-data.entity.ts +55 -55
- package/src/module/workflow/entity/action-resources-mapping.entity.ts +29 -29
- package/src/module/workflow/entity/action-template-mapping.entity.ts +17 -17
- package/src/module/workflow/entity/action.entity.ts +53 -53
- package/src/module/workflow/entity/activity-log.entity.ts +43 -43
- package/src/module/workflow/entity/comm-template.entity.ts +43 -43
- package/src/module/workflow/entity/entity-modification.entity.ts +38 -38
- package/src/module/workflow/entity/form.entity.ts +25 -25
- package/src/module/workflow/entity/stage-action-mapping.entity.ts +17 -17
- package/src/module/workflow/entity/stage-group.entity.ts +23 -23
- package/src/module/workflow/entity/stage-movement-data.entity.ts +38 -38
- package/src/module/workflow/entity/stage.entity.ts +20 -20
- package/src/module/workflow/entity/task-data.entity.ts +88 -88
- package/src/module/workflow/entity/template-attach-mapper.entity.ts +30 -30
- package/src/module/workflow/entity/workflow-data.entity.ts +11 -11
- package/src/module/workflow/entity/workflow-level-mapping.entity.ts +18 -18
- package/src/module/workflow/entity/workflow.entity.ts +20 -20
- package/src/module/workflow/repository/action-category.repository.ts +79 -79
- package/src/module/workflow/repository/action-data.repository.ts +347 -347
- package/src/module/workflow/repository/action.repository.ts +339 -339
- package/src/module/workflow/repository/activity-log.repository.ts +148 -148
- package/src/module/workflow/repository/comm-template.repository.ts +157 -157
- package/src/module/workflow/repository/form-master.repository.ts +50 -50
- package/src/module/workflow/repository/stage-group.repository.ts +186 -186
- package/src/module/workflow/repository/stage-movement.repository.ts +217 -217
- package/src/module/workflow/repository/stage.repository.ts +160 -160
- package/src/module/workflow/repository/task.repository.ts +154 -154
- package/src/module/workflow/repository/workflow.repository.ts +42 -42
- package/src/module/workflow/service/action-category.service.ts +33 -33
- package/src/module/workflow/service/action-data.service.ts +62 -62
- package/src/module/workflow/service/action-resources-mapping.service.ts +10 -10
- package/src/module/workflow/service/action-template-mapping.service.ts +137 -137
- package/src/module/workflow/service/action.service.ts +302 -302
- package/src/module/workflow/service/activity-log.service.ts +107 -107
- package/src/module/workflow/service/comm-template.service.ts +181 -181
- package/src/module/workflow/service/entity-modification.service.ts +61 -61
- package/src/module/workflow/service/form-master.service.ts +35 -35
- package/src/module/workflow/service/populate-workflow.service.ts +320 -320
- package/src/module/workflow/service/stage-action-mapping.service.ts +5 -5
- package/src/module/workflow/service/stage-group.service.ts +325 -325
- package/src/module/workflow/service/stage.service.ts +197 -197
- package/src/module/workflow/service/task.service.ts +551 -551
- package/src/module/workflow/service/workflow-list-master.service.ts +68 -68
- package/src/module/workflow/service/workflow-meta.service.ts +640 -640
- package/src/module/workflow/service/workflow.service.ts +213 -213
- package/src/module/workflow/workflow.module.ts +180 -180
- package/src/module/workflow-automation/SCHEDULING_GUIDE.md +145 -145
- package/src/module/workflow-automation/controller/workflow-automation.controller.ts +43 -43
- package/src/module/workflow-automation/entity/workflow-automation-action.entity.ts +26 -26
- package/src/module/workflow-automation/entity/workflow-automation.entity.ts +40 -40
- package/src/module/workflow-automation/interface/action.decorator.ts +7 -7
- package/src/module/workflow-automation/interface/action.interface.ts +5 -5
- package/src/module/workflow-automation/service/action-registery.service.ts +35 -35
- package/src/module/workflow-automation/service/schedule-handler.service.ts +168 -168
- package/src/module/workflow-automation/service/workflow-automation-engine.service.ts +219 -219
- package/src/module/workflow-automation/service/workflow-automation.service.ts +476 -476
- package/src/module/workflow-automation/workflow-automation.module.ts +54 -54
- package/src/module/workflow-schedule/INSTALLATION.md +244 -244
- package/src/module/workflow-schedule/MULTI_PROJECT_GUIDE.md +196 -196
- package/src/module/workflow-schedule/README.md +422 -422
- package/src/module/workflow-schedule/constants/schedule.constants.ts +48 -48
- package/src/module/workflow-schedule/controller/workflow-schedule.controller.ts +253 -253
- package/src/module/workflow-schedule/docs/CLAUDE_CODE_GUIDE.md +510 -510
- package/src/module/workflow-schedule/docs/CLAUDE_CODE_PROMPT.md +362 -362
- package/src/module/workflow-schedule/docs/RUN_CLAUDE_CODE.sh +68 -68
- package/src/module/workflow-schedule/dto/create-schedule.dto.ts +147 -147
- package/src/module/workflow-schedule/dto/get-execution-logs.dto.ts +119 -119
- package/src/module/workflow-schedule/dto/update-schedule.dto.ts +96 -96
- package/src/module/workflow-schedule/entities/scheduled-workflow.entity.ts +148 -148
- package/src/module/workflow-schedule/entities/workflow-execution-log.entity.ts +154 -154
- package/src/module/workflow-schedule/interfaces/schedule-job-data.interface.ts +53 -53
- package/src/module/workflow-schedule/interfaces/workflow-schedule-options.interface.ts +12 -12
- package/src/module/workflow-schedule/processors/schedule.processor.ts +620 -620
- package/src/module/workflow-schedule/service/workflow-schedule.service.ts +597 -597
- package/src/module/workflow-schedule/workflow-schedule.module.ts +67 -67
- package/src/resources/dev.properties.yaml +31 -31
- package/src/resources/local.properties.yaml +27 -27
- package/src/resources/properties.module.ts +12 -12
- package/src/resources/properties.yaml.ts +11 -11
- package/src/resources/uat.properties.yaml +31 -31
- package/src/table.config.ts +133 -133
- package/src/utils/dto/excel-data.dto.ts +14 -14
- package/src/utils/dto/excelsheet-data.dto.ts +5 -5
- package/src/utils/service/base64util.service.ts +18 -18
- package/src/utils/service/clockIDGenUtil.service.ts +21 -21
- package/src/utils/service/codeGenerator.service.ts +22 -22
- package/src/utils/service/dateUtil.service.ts +17 -17
- package/src/utils/service/encryptUtil.service.ts +97 -97
- package/src/utils/service/excel-helper.service.ts +72 -72
- package/src/utils/service/excelUtil.service.ts +15 -15
- package/src/utils/service/file-util.service.ts +11 -11
- package/src/utils/service/json-util.service.ts +23 -23
- package/src/utils/service/loggingUtil.service.ts +88 -88
- package/src/utils/service/reflection-helper.service.ts +62 -62
- package/src/utils/service/wbsCodeGen.service.ts +8 -8
- package/src/utils/utils.module.ts +27 -27
- package/tsconfig.build.json +4 -4
- package/tsconfig.json +24 -24
|
@@ -1,527 +1,527 @@
|
|
|
1
|
-
import { BadRequestException, forwardRef, Inject, Injectable, NotFoundException } from '@nestjs/common';
|
|
2
|
-
import { ListMasterItemsRepository } from '../repository/list-master-items.repository';
|
|
3
|
-
import { ListMasterRepository } from '../repository/list-master.repository';
|
|
4
|
-
import { ApiRegistryService } from '../../third-party-module/service/api-registry.service';
|
|
5
|
-
import { firstValueFrom } from 'rxjs';
|
|
6
|
-
import { HttpService } from '@nestjs/axios';
|
|
7
|
-
import { EntityManager } from 'typeorm';
|
|
8
|
-
import { EntityMasterService } from 'src/module/meta/service/entity-master.service';
|
|
9
|
-
import { UserData } from 'src/module/user/entity/user.entity';
|
|
10
|
-
import { EntityServiceImpl } from 'src/module/meta/service/entity-service-impl.service';
|
|
11
|
-
import { STATUS_ACTIVE, STATUS_INACTIVE } from 'src/constant/global.constant';
|
|
12
|
-
import { Action } from '../../workflow-automation/interface/action.interface';
|
|
13
|
-
import axios from 'axios';
|
|
14
|
-
import { ConfigService } from '@nestjs/config';
|
|
15
|
-
import { AttributeMasterService } from '../../meta/service/attribute-master.service';
|
|
16
|
-
import { IMicroserviceClients } from '../../microservice-client/service/microservice-clients';
|
|
17
|
-
|
|
18
|
-
@Injectable()
|
|
19
|
-
export class ListMasterService {
|
|
20
|
-
constructor(
|
|
21
|
-
private readonly entityManager: EntityManager,
|
|
22
|
-
@Inject(forwardRef(() => EntityServiceImpl))
|
|
23
|
-
private readonly entityServiceImpl: EntityServiceImpl,
|
|
24
|
-
@Inject(forwardRef(() => EntityMasterService))
|
|
25
|
-
private readonly entityMasterService: EntityMasterService,
|
|
26
|
-
private readonly listMasterRepo: ListMasterRepository,
|
|
27
|
-
private readonly listItemsRepo: ListMasterItemsRepository,
|
|
28
|
-
private readonly apiRegistryService: ApiRegistryService,
|
|
29
|
-
private readonly httpService: HttpService,
|
|
30
|
-
private readonly configService: ConfigService,
|
|
31
|
-
private readonly attributeMasterService: AttributeMasterService,
|
|
32
|
-
@Inject('MICROSERVICE_CLIENT_FACTORY')
|
|
33
|
-
private readonly factory: IMicroserviceClients,
|
|
34
|
-
) {
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
schema = this.configService.get('DB_SCHEMA');
|
|
38
|
-
private readonly skipLevelFilterEntities = ['BRN', 'USR', 'UPR', 'BRNP'];
|
|
39
|
-
private readonly actions = new Map<string, Action>();
|
|
40
|
-
|
|
41
|
-
registerAction(actionName: string, actionInstance: Action) {
|
|
42
|
-
this.actions.set(actionName, actionInstance);
|
|
43
|
-
console.log(
|
|
44
|
-
`⚙️ [WorkflowAutomationEngine] Registered action: ${actionName}`,
|
|
45
|
-
);
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
async getResolvedListCode(
|
|
49
|
-
code: string,
|
|
50
|
-
enterprise_id: number,
|
|
51
|
-
): Promise<any> {
|
|
52
|
-
if (!code) return code;
|
|
53
|
-
|
|
54
|
-
// Check if it's a valid list type
|
|
55
|
-
const listMaster = await this.listItemsRepo.findItemByCode(
|
|
56
|
-
code,
|
|
57
|
-
enterprise_id,
|
|
58
|
-
);
|
|
59
|
-
if (listMaster) {
|
|
60
|
-
return listMaster;
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
async getDropdownOptions(
|
|
65
|
-
type: string,
|
|
66
|
-
params: Record<string, string>,
|
|
67
|
-
inactiveIdsArray?: number[],
|
|
68
|
-
loggedInUser?,
|
|
69
|
-
publicCall = false,
|
|
70
|
-
) {
|
|
71
|
-
const config = await this.listMasterRepo.findByType(
|
|
72
|
-
type,
|
|
73
|
-
loggedInUser?.enterprise_id,
|
|
74
|
-
);
|
|
75
|
-
|
|
76
|
-
if (!config) throw new NotFoundException(`Type ${type} not found`);
|
|
77
|
-
|
|
78
|
-
if (
|
|
79
|
-
config.appcode != null &&
|
|
80
|
-
config.appcode != loggedInUser?.appcode &&
|
|
81
|
-
!publicCall
|
|
82
|
-
) {
|
|
83
|
-
// Call internal API for appcode mismatch
|
|
84
|
-
try {
|
|
85
|
-
const baseUrl = this.configService.get<string>('REDIRECT_BE_URL');
|
|
86
|
-
|
|
87
|
-
// Prepare the query string
|
|
88
|
-
const queryParams = new URLSearchParams({
|
|
89
|
-
inactiveIds: inactiveIdsArray?.join(',') || '',
|
|
90
|
-
loggedInUser: JSON.stringify(loggedInUser),
|
|
91
|
-
...params, // Spread other params into the query string
|
|
92
|
-
}).toString();
|
|
93
|
-
|
|
94
|
-
// Make the GET request with query parameters
|
|
95
|
-
const response = await axios.get(
|
|
96
|
-
`${baseUrl}/list-master/getDropdownDataPublic/${type}?${queryParams}`,
|
|
97
|
-
{
|
|
98
|
-
headers: {
|
|
99
|
-
'Content-Type': 'application/json',
|
|
100
|
-
},
|
|
101
|
-
},
|
|
102
|
-
);
|
|
103
|
-
|
|
104
|
-
return response.data;
|
|
105
|
-
} catch (error) {
|
|
106
|
-
console.error('⚠️ Internal API call failed:', error.message);
|
|
107
|
-
throw new BadRequestException(
|
|
108
|
-
`Failed to fetch dropdown for type ${type} from internal API`,
|
|
109
|
-
);
|
|
110
|
-
}
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
// Fallback to old logic if no extension exists
|
|
114
|
-
switch (config.source) {
|
|
115
|
-
case 'entity':
|
|
116
|
-
return this.fetchFromEntity(
|
|
117
|
-
type,
|
|
118
|
-
params,
|
|
119
|
-
inactiveIdsArray,
|
|
120
|
-
loggedInUser,
|
|
121
|
-
);
|
|
122
|
-
|
|
123
|
-
case 'master':
|
|
124
|
-
return this.listItemsRepo.findItemsByType(
|
|
125
|
-
type,
|
|
126
|
-
config.sort_by,
|
|
127
|
-
inactiveIdsArray,
|
|
128
|
-
loggedInUser?.enterprise_id,
|
|
129
|
-
params,
|
|
130
|
-
);
|
|
131
|
-
|
|
132
|
-
case 'operator':
|
|
133
|
-
return this.listItemsRepo.findOperatorsByType(
|
|
134
|
-
type,
|
|
135
|
-
loggedInUser?.enterprise_id,
|
|
136
|
-
);
|
|
137
|
-
|
|
138
|
-
case 'custom':
|
|
139
|
-
// If you want Axios call here too:
|
|
140
|
-
try {
|
|
141
|
-
const response = await axios.get(
|
|
142
|
-
`https://external-source.com/${config.custom_source_id}`,
|
|
143
|
-
{ params },
|
|
144
|
-
);
|
|
145
|
-
return response.data;
|
|
146
|
-
} catch (error) {
|
|
147
|
-
console.error('⚠️ Custom source fetch failed:', error.message);
|
|
148
|
-
throw new BadRequestException('Failed to fetch custom source');
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
default:
|
|
152
|
-
throw new BadRequestException(`Unknown source: ${config.source}`);
|
|
153
|
-
}
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
private async fetchFromEntity(
|
|
157
|
-
sourceList: string,
|
|
158
|
-
params: Record<string, any>,
|
|
159
|
-
inactiveIdsArray?: number[],
|
|
160
|
-
loggedInUser?: UserData,
|
|
161
|
-
) {
|
|
162
|
-
|
|
163
|
-
let result: { label: string; value: number }[] = [];
|
|
164
|
-
if (!sourceList) return result;
|
|
165
|
-
|
|
166
|
-
const entityMaster = await this.entityMasterService.getEntityData(
|
|
167
|
-
sourceList,
|
|
168
|
-
loggedInUser,
|
|
169
|
-
);
|
|
170
|
-
const tableName = entityMaster.data_source;
|
|
171
|
-
|
|
172
|
-
const applyCommonFilters = (qb: any, status?: number) => {
|
|
173
|
-
if (status) {
|
|
174
|
-
const isView = tableName.endsWith('_vw'); // auto-detect view
|
|
175
|
-
const statusColumn = isView ? 'status_id' : 'status';
|
|
176
|
-
qb.andWhere(`${tableName}.${statusColumn} = :status`, { status });
|
|
177
|
-
}
|
|
178
|
-
|
|
179
|
-
if (loggedInUser?.level_type && loggedInUser?.level_id) {
|
|
180
|
-
// Skip level filter for certain entities
|
|
181
|
-
if (!this.skipLevelFilterEntities.includes(sourceList)) {
|
|
182
|
-
qb.andWhere(
|
|
183
|
-
`${tableName}.level_type = :levelType AND ${tableName}.level_id = :levelId AND ${tableName}.enterprise_id = :enterprise_id`,
|
|
184
|
-
{
|
|
185
|
-
levelType: loggedInUser.level_type,
|
|
186
|
-
levelId: loggedInUser.level_id,
|
|
187
|
-
enterprise_id: loggedInUser.enterprise_id,
|
|
188
|
-
},
|
|
189
|
-
);
|
|
190
|
-
}
|
|
191
|
-
}
|
|
192
|
-
|
|
193
|
-
if (sourceList == 'BRN') {
|
|
194
|
-
// IN the case of BRN, we don't filter by level
|
|
195
|
-
qb.andWhere(
|
|
196
|
-
`${tableName}.parent_id = :enterprise_id AND ${tableName}.type = 'BRN'`,
|
|
197
|
-
{ enterprise_id: loggedInUser?.enterprise_id },
|
|
198
|
-
);
|
|
199
|
-
}
|
|
200
|
-
|
|
201
|
-
if (sourceList == 'BRNP') {
|
|
202
|
-
// IN the case of BRNP, we only filter by organization_id
|
|
203
|
-
qb.andWhere(`${tableName}.enterprise_id = :enterprise_id`, {
|
|
204
|
-
enterprise_id: loggedInUser?.enterprise_id,
|
|
205
|
-
});
|
|
206
|
-
}
|
|
207
|
-
|
|
208
|
-
if (sourceList == 'USR' || sourceList == 'UPR') {
|
|
209
|
-
// IN the case of USR/UPR, we don't filter by level
|
|
210
|
-
qb.andWhere(`${tableName}.enterprise_id = :enterprise_id`, {
|
|
211
|
-
enterprise_id: loggedInUser?.enterprise_id,
|
|
212
|
-
});
|
|
213
|
-
}
|
|
214
|
-
|
|
215
|
-
if (loggedInUser?.appcode && sourceList === 'ROL') {
|
|
216
|
-
if (!this.skipLevelFilterEntities.includes(sourceList)) {
|
|
217
|
-
qb.andWhere(`${tableName}.appcode = :appcode`, {
|
|
218
|
-
appcode: loggedInUser.appcode,
|
|
219
|
-
});
|
|
220
|
-
}
|
|
221
|
-
}
|
|
222
|
-
|
|
223
|
-
// ✅ Apply dynamic params
|
|
224
|
-
for (const key in params) {
|
|
225
|
-
qb.andWhere(`${tableName}.${key} = :${key}`, {
|
|
226
|
-
[key]: params[key],
|
|
227
|
-
});
|
|
228
|
-
}
|
|
229
|
-
|
|
230
|
-
// ✅ New: Exclude customers for USR/UPR
|
|
231
|
-
if (sourceList === 'USR' || sourceList === 'UPR') {
|
|
232
|
-
qb.andWhere(`${tableName}.is_customer is NULL`);
|
|
233
|
-
}
|
|
234
|
-
|
|
235
|
-
return qb;
|
|
236
|
-
};
|
|
237
|
-
|
|
238
|
-
const resolveStatus = await this.getResolvedListCode(
|
|
239
|
-
STATUS_ACTIVE,
|
|
240
|
-
loggedInUser?.enterprise_id || 0,
|
|
241
|
-
);
|
|
242
|
-
|
|
243
|
-
// Fetch active records
|
|
244
|
-
const activeQuery = applyCommonFilters(
|
|
245
|
-
this.entityManager
|
|
246
|
-
.createQueryBuilder()
|
|
247
|
-
.select('*')
|
|
248
|
-
.from(`${this.schema}.${tableName}`, tableName),
|
|
249
|
-
resolveStatus.id,
|
|
250
|
-
);
|
|
251
|
-
|
|
252
|
-
const activeResults = await activeQuery.getRawMany();
|
|
253
|
-
const activeIds = new Set(activeResults.map((r) => r.id));
|
|
254
|
-
|
|
255
|
-
// Add active entries first
|
|
256
|
-
activeResults.forEach((r) => {
|
|
257
|
-
result.push({ label: r.name, value: r.id });
|
|
258
|
-
});
|
|
259
|
-
|
|
260
|
-
const resolveInactiveStatus = await this.getResolvedListCode(
|
|
261
|
-
STATUS_INACTIVE,
|
|
262
|
-
loggedInUser?.organization_id || 0,
|
|
263
|
-
);
|
|
264
|
-
|
|
265
|
-
// Fetch inactive records (with same filters but without status condition)
|
|
266
|
-
if (inactiveIdsArray?.length) {
|
|
267
|
-
const inactiveQuery = applyCommonFilters(
|
|
268
|
-
this.entityManager
|
|
269
|
-
.createQueryBuilder()
|
|
270
|
-
.select('*')
|
|
271
|
-
.from(`${this.schema}.${tableName}`, tableName),
|
|
272
|
-
resolveInactiveStatus.id,
|
|
273
|
-
);
|
|
274
|
-
|
|
275
|
-
inactiveQuery.andWhere(`${tableName}.id IN (:...ids)`, {
|
|
276
|
-
ids: inactiveIdsArray,
|
|
277
|
-
});
|
|
278
|
-
|
|
279
|
-
const inactiveResults = await inactiveQuery.getRawMany();
|
|
280
|
-
|
|
281
|
-
inactiveResults.forEach((item) => {
|
|
282
|
-
if (!activeIds.has(item.id)) {
|
|
283
|
-
result.push({
|
|
284
|
-
label: `${item.name} [INACTIVE]`,
|
|
285
|
-
value: item.id,
|
|
286
|
-
});
|
|
287
|
-
}
|
|
288
|
-
});
|
|
289
|
-
}
|
|
290
|
-
|
|
291
|
-
return result;
|
|
292
|
-
}
|
|
293
|
-
|
|
294
|
-
private async fetchFromExternalSource(
|
|
295
|
-
customSourceId: number,
|
|
296
|
-
params?: Record<string, string>,
|
|
297
|
-
) {
|
|
298
|
-
const apiRegistry = await this.apiRegistryService.findById(customSourceId);
|
|
299
|
-
if (!apiRegistry) return [];
|
|
300
|
-
|
|
301
|
-
const url = `${apiRegistry.base_url}${apiRegistry.endpoint.replace(/{{(.*?)}}/g, (_, key) => params?.[key.trim()] || '')}`;
|
|
302
|
-
const payload = this.injectDynamicParams(
|
|
303
|
-
apiRegistry.request_payload_schema,
|
|
304
|
-
params,
|
|
305
|
-
);
|
|
306
|
-
|
|
307
|
-
const response = await firstValueFrom(
|
|
308
|
-
this.httpService.request({
|
|
309
|
-
url,
|
|
310
|
-
method: apiRegistry.http_method || 'POST',
|
|
311
|
-
data: payload,
|
|
312
|
-
headers: apiRegistry.headers ? apiRegistry.headers : undefined,
|
|
313
|
-
}),
|
|
314
|
-
);
|
|
315
|
-
|
|
316
|
-
const data = this.extractByPath(
|
|
317
|
-
response.data,
|
|
318
|
-
apiRegistry.response_data_path,
|
|
319
|
-
);
|
|
320
|
-
|
|
321
|
-
if (!Array.isArray(data)) {
|
|
322
|
-
return [{ label: data, value: data }];
|
|
323
|
-
}
|
|
324
|
-
|
|
325
|
-
if (typeof data[0] === 'string') {
|
|
326
|
-
return data.map((item: string) => ({ label: item, value: item }));
|
|
327
|
-
}
|
|
328
|
-
|
|
329
|
-
return data.map((item: any) => ({
|
|
330
|
-
label: item[apiRegistry?.label],
|
|
331
|
-
value: item[apiRegistry?.value],
|
|
332
|
-
}));
|
|
333
|
-
}
|
|
334
|
-
|
|
335
|
-
private injectDynamicParams(
|
|
336
|
-
payload: any,
|
|
337
|
-
params?: Record<string, string>,
|
|
338
|
-
): any {
|
|
339
|
-
if (params) {
|
|
340
|
-
const stringified = JSON.stringify(payload);
|
|
341
|
-
const replaced = stringified.replace(
|
|
342
|
-
/{{(.*?)}}/g,
|
|
343
|
-
(_, key) => params[key.trim()] || '',
|
|
344
|
-
);
|
|
345
|
-
return JSON.parse(replaced);
|
|
346
|
-
}
|
|
347
|
-
}
|
|
348
|
-
|
|
349
|
-
private extractByPath(obj: any, path: string): any {
|
|
350
|
-
if (!path || !obj) return obj;
|
|
351
|
-
|
|
352
|
-
return path.split('.').reduce((acc, key) => {
|
|
353
|
-
if (acc == null) return undefined;
|
|
354
|
-
|
|
355
|
-
// Try to convert to number if it's a numeric string (for array indexes)
|
|
356
|
-
const index = Number(key);
|
|
357
|
-
if (!isNaN(index) && Array.isArray(acc)) {
|
|
358
|
-
return acc[index];
|
|
359
|
-
}
|
|
360
|
-
|
|
361
|
-
return acc[key];
|
|
362
|
-
}, obj);
|
|
363
|
-
}
|
|
364
|
-
|
|
365
|
-
// createEntity method
|
|
366
|
-
async createEntity(entityData: any, loggedInUser: UserData): Promise<any> {
|
|
367
|
-
// Trim and validate name and code
|
|
368
|
-
const name = entityData.name?.trim();
|
|
369
|
-
// const code = entityData.code?.trim();
|
|
370
|
-
|
|
371
|
-
if (!name) {
|
|
372
|
-
throw new BadRequestException('Name is required and cannot be empty');
|
|
373
|
-
}
|
|
374
|
-
|
|
375
|
-
entityData.name = name;
|
|
376
|
-
entityData.is_factory = entityData.is_factory || 0;
|
|
377
|
-
entityData.source = entityData.source || 'master';
|
|
378
|
-
entityData.status = entityData.status || 'ACTIVE';
|
|
379
|
-
entityData.type = entityData.code;
|
|
380
|
-
entityData.sort_by = entityData.sort_by || 'asc';
|
|
381
|
-
|
|
382
|
-
// Check for duplicate name
|
|
383
|
-
const nameExists = await this.listMasterRepo.findOneByCondition({
|
|
384
|
-
name,
|
|
385
|
-
enterprise_id: loggedInUser.enterprise_id,
|
|
386
|
-
});
|
|
387
|
-
|
|
388
|
-
if (nameExists) {
|
|
389
|
-
throw new BadRequestException(
|
|
390
|
-
'A List Master with the same name already exists',
|
|
391
|
-
);
|
|
392
|
-
}
|
|
393
|
-
|
|
394
|
-
const createdListMaster = await this.entityServiceImpl.createEntity(
|
|
395
|
-
entityData,
|
|
396
|
-
loggedInUser,
|
|
397
|
-
);
|
|
398
|
-
|
|
399
|
-
if (!createdListMaster) {
|
|
400
|
-
throw new BadRequestException('Failed to create entity');
|
|
401
|
-
}
|
|
402
|
-
|
|
403
|
-
return createdListMaster;
|
|
404
|
-
}
|
|
405
|
-
|
|
406
|
-
async updateEntity(entityData: any, loggedInUser: UserData): Promise<any> {
|
|
407
|
-
return await this.entityServiceImpl.updateEntity(entityData, loggedInUser);
|
|
408
|
-
}
|
|
409
|
-
|
|
410
|
-
async getEntityData(
|
|
411
|
-
entity_type: string,
|
|
412
|
-
id: number,
|
|
413
|
-
loggedInUser,
|
|
414
|
-
): Promise<any> {
|
|
415
|
-
return await this.entityServiceImpl.getEntityData(
|
|
416
|
-
entity_type,
|
|
417
|
-
id,
|
|
418
|
-
loggedInUser,
|
|
419
|
-
);
|
|
420
|
-
}
|
|
421
|
-
|
|
422
|
-
async getAllListMasterByOrganization(
|
|
423
|
-
enterprise_id: number,
|
|
424
|
-
search?: string,
|
|
425
|
-
): Promise<any[]> {
|
|
426
|
-
return await this.listMasterRepo.findAllItems(enterprise_id, search);
|
|
427
|
-
}
|
|
428
|
-
|
|
429
|
-
async getDropDownData(
|
|
430
|
-
entity_type: string,
|
|
431
|
-
attribute_key: string,
|
|
432
|
-
loggedInUser: UserData,
|
|
433
|
-
body: Record<string, string>,
|
|
434
|
-
) {
|
|
435
|
-
let entityMaster = await this.entityMasterService.getEntityData(
|
|
436
|
-
entity_type,
|
|
437
|
-
loggedInUser,
|
|
438
|
-
);
|
|
439
|
-
let appCode = entityMaster.appcode;
|
|
440
|
-
|
|
441
|
-
const { inactiveIds, ...params } = body;
|
|
442
|
-
|
|
443
|
-
const inactiveIdsArray = inactiveIds
|
|
444
|
-
? inactiveIds.split(',').map((id) => parseInt(id, 10))
|
|
445
|
-
: [];
|
|
446
|
-
|
|
447
|
-
const currentAppCode = this.configService.get<string>('appcode');
|
|
448
|
-
|
|
449
|
-
if (currentAppCode === appCode || !currentAppCode) {
|
|
450
|
-
const entityAttribute =
|
|
451
|
-
await this.attributeMasterService.findByMappedEntityTypeAndAttributeKey(
|
|
452
|
-
entity_type,
|
|
453
|
-
attribute_key,
|
|
454
|
-
loggedInUser,
|
|
455
|
-
);
|
|
456
|
-
|
|
457
|
-
if (entityAttribute && entityAttribute.data_source_type) {
|
|
458
|
-
const listMaster = await this.listMasterRepo.findByType(
|
|
459
|
-
entityAttribute.datasource_list,
|
|
460
|
-
loggedInUser?.enterprise_id,
|
|
461
|
-
);
|
|
462
|
-
|
|
463
|
-
if (!listMaster) {
|
|
464
|
-
return;
|
|
465
|
-
}
|
|
466
|
-
|
|
467
|
-
switch (listMaster.source) {
|
|
468
|
-
case 'entity':
|
|
469
|
-
return this.fetchFromEntity(
|
|
470
|
-
listMaster.type,
|
|
471
|
-
params,
|
|
472
|
-
inactiveIdsArray,
|
|
473
|
-
loggedInUser,
|
|
474
|
-
);
|
|
475
|
-
|
|
476
|
-
case 'master':
|
|
477
|
-
return this.listItemsRepo.findItemsByType(
|
|
478
|
-
listMaster.type,
|
|
479
|
-
listMaster.sort_by,
|
|
480
|
-
inactiveIdsArray,
|
|
481
|
-
loggedInUser.enterprise_id,
|
|
482
|
-
params,
|
|
483
|
-
);
|
|
484
|
-
|
|
485
|
-
case 'operator':
|
|
486
|
-
return this.listItemsRepo.findOperatorsByType(
|
|
487
|
-
listMaster?.type,
|
|
488
|
-
loggedInUser?.enterprise_id,
|
|
489
|
-
);
|
|
490
|
-
|
|
491
|
-
case 'custom':
|
|
492
|
-
// If you want Axios call here too:
|
|
493
|
-
try {
|
|
494
|
-
const response = await axios.get(
|
|
495
|
-
`https://external-source.com/${listMaster.custom_source_id}`,
|
|
496
|
-
{ params },
|
|
497
|
-
);
|
|
498
|
-
return response.data;
|
|
499
|
-
} catch (error) {
|
|
500
|
-
console.error('⚠️ Custom source fetch failed:', error.message);
|
|
501
|
-
throw new BadRequestException('Failed to fetch custom source');
|
|
502
|
-
}
|
|
503
|
-
|
|
504
|
-
default:
|
|
505
|
-
throw new BadRequestException(
|
|
506
|
-
`Unknown source: ${listMaster.source}`,
|
|
507
|
-
);
|
|
508
|
-
}
|
|
509
|
-
}
|
|
510
|
-
} else {
|
|
511
|
-
const client = this.factory.getClient(appCode);
|
|
512
|
-
|
|
513
|
-
if (!client) {
|
|
514
|
-
return;
|
|
515
|
-
}
|
|
516
|
-
|
|
517
|
-
return client
|
|
518
|
-
.send('getDropdownData', {
|
|
519
|
-
entity_type,
|
|
520
|
-
attribute_key,
|
|
521
|
-
loggedInUser,
|
|
522
|
-
body,
|
|
523
|
-
})
|
|
524
|
-
.toPromise();
|
|
525
|
-
}
|
|
526
|
-
}
|
|
527
|
-
}
|
|
1
|
+
import { BadRequestException, forwardRef, Inject, Injectable, NotFoundException } from '@nestjs/common';
|
|
2
|
+
import { ListMasterItemsRepository } from '../repository/list-master-items.repository';
|
|
3
|
+
import { ListMasterRepository } from '../repository/list-master.repository';
|
|
4
|
+
import { ApiRegistryService } from '../../third-party-module/service/api-registry.service';
|
|
5
|
+
import { firstValueFrom } from 'rxjs';
|
|
6
|
+
import { HttpService } from '@nestjs/axios';
|
|
7
|
+
import { EntityManager } from 'typeorm';
|
|
8
|
+
import { EntityMasterService } from 'src/module/meta/service/entity-master.service';
|
|
9
|
+
import { UserData } from 'src/module/user/entity/user.entity';
|
|
10
|
+
import { EntityServiceImpl } from 'src/module/meta/service/entity-service-impl.service';
|
|
11
|
+
import { STATUS_ACTIVE, STATUS_INACTIVE } from 'src/constant/global.constant';
|
|
12
|
+
import { Action } from '../../workflow-automation/interface/action.interface';
|
|
13
|
+
import axios from 'axios';
|
|
14
|
+
import { ConfigService } from '@nestjs/config';
|
|
15
|
+
import { AttributeMasterService } from '../../meta/service/attribute-master.service';
|
|
16
|
+
import { IMicroserviceClients } from '../../microservice-client/service/microservice-clients';
|
|
17
|
+
|
|
18
|
+
@Injectable()
|
|
19
|
+
export class ListMasterService {
|
|
20
|
+
constructor(
|
|
21
|
+
private readonly entityManager: EntityManager,
|
|
22
|
+
@Inject(forwardRef(() => EntityServiceImpl))
|
|
23
|
+
private readonly entityServiceImpl: EntityServiceImpl,
|
|
24
|
+
@Inject(forwardRef(() => EntityMasterService))
|
|
25
|
+
private readonly entityMasterService: EntityMasterService,
|
|
26
|
+
private readonly listMasterRepo: ListMasterRepository,
|
|
27
|
+
private readonly listItemsRepo: ListMasterItemsRepository,
|
|
28
|
+
private readonly apiRegistryService: ApiRegistryService,
|
|
29
|
+
private readonly httpService: HttpService,
|
|
30
|
+
private readonly configService: ConfigService,
|
|
31
|
+
private readonly attributeMasterService: AttributeMasterService,
|
|
32
|
+
@Inject('MICROSERVICE_CLIENT_FACTORY')
|
|
33
|
+
private readonly factory: IMicroserviceClients,
|
|
34
|
+
) {
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
schema = this.configService.get('DB_SCHEMA');
|
|
38
|
+
private readonly skipLevelFilterEntities = ['BRN', 'USR', 'UPR', 'BRNP'];
|
|
39
|
+
private readonly actions = new Map<string, Action>();
|
|
40
|
+
|
|
41
|
+
registerAction(actionName: string, actionInstance: Action) {
|
|
42
|
+
this.actions.set(actionName, actionInstance);
|
|
43
|
+
console.log(
|
|
44
|
+
`⚙️ [WorkflowAutomationEngine] Registered action: ${actionName}`,
|
|
45
|
+
);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
async getResolvedListCode(
|
|
49
|
+
code: string,
|
|
50
|
+
enterprise_id: number,
|
|
51
|
+
): Promise<any> {
|
|
52
|
+
if (!code) return code;
|
|
53
|
+
|
|
54
|
+
// Check if it's a valid list type
|
|
55
|
+
const listMaster = await this.listItemsRepo.findItemByCode(
|
|
56
|
+
code,
|
|
57
|
+
enterprise_id,
|
|
58
|
+
);
|
|
59
|
+
if (listMaster) {
|
|
60
|
+
return listMaster;
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
async getDropdownOptions(
|
|
65
|
+
type: string,
|
|
66
|
+
params: Record<string, string>,
|
|
67
|
+
inactiveIdsArray?: number[],
|
|
68
|
+
loggedInUser?,
|
|
69
|
+
publicCall = false,
|
|
70
|
+
) {
|
|
71
|
+
const config = await this.listMasterRepo.findByType(
|
|
72
|
+
type,
|
|
73
|
+
loggedInUser?.enterprise_id,
|
|
74
|
+
);
|
|
75
|
+
|
|
76
|
+
if (!config) throw new NotFoundException(`Type ${type} not found`);
|
|
77
|
+
|
|
78
|
+
if (
|
|
79
|
+
config.appcode != null &&
|
|
80
|
+
config.appcode != loggedInUser?.appcode &&
|
|
81
|
+
!publicCall
|
|
82
|
+
) {
|
|
83
|
+
// Call internal API for appcode mismatch
|
|
84
|
+
try {
|
|
85
|
+
const baseUrl = this.configService.get<string>('REDIRECT_BE_URL');
|
|
86
|
+
|
|
87
|
+
// Prepare the query string
|
|
88
|
+
const queryParams = new URLSearchParams({
|
|
89
|
+
inactiveIds: inactiveIdsArray?.join(',') || '',
|
|
90
|
+
loggedInUser: JSON.stringify(loggedInUser),
|
|
91
|
+
...params, // Spread other params into the query string
|
|
92
|
+
}).toString();
|
|
93
|
+
|
|
94
|
+
// Make the GET request with query parameters
|
|
95
|
+
const response = await axios.get(
|
|
96
|
+
`${baseUrl}/list-master/getDropdownDataPublic/${type}?${queryParams}`,
|
|
97
|
+
{
|
|
98
|
+
headers: {
|
|
99
|
+
'Content-Type': 'application/json',
|
|
100
|
+
},
|
|
101
|
+
},
|
|
102
|
+
);
|
|
103
|
+
|
|
104
|
+
return response.data;
|
|
105
|
+
} catch (error) {
|
|
106
|
+
console.error('⚠️ Internal API call failed:', error.message);
|
|
107
|
+
throw new BadRequestException(
|
|
108
|
+
`Failed to fetch dropdown for type ${type} from internal API`,
|
|
109
|
+
);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
// Fallback to old logic if no extension exists
|
|
114
|
+
switch (config.source) {
|
|
115
|
+
case 'entity':
|
|
116
|
+
return this.fetchFromEntity(
|
|
117
|
+
type,
|
|
118
|
+
params,
|
|
119
|
+
inactiveIdsArray,
|
|
120
|
+
loggedInUser,
|
|
121
|
+
);
|
|
122
|
+
|
|
123
|
+
case 'master':
|
|
124
|
+
return this.listItemsRepo.findItemsByType(
|
|
125
|
+
type,
|
|
126
|
+
config.sort_by,
|
|
127
|
+
inactiveIdsArray,
|
|
128
|
+
loggedInUser?.enterprise_id,
|
|
129
|
+
params,
|
|
130
|
+
);
|
|
131
|
+
|
|
132
|
+
case 'operator':
|
|
133
|
+
return this.listItemsRepo.findOperatorsByType(
|
|
134
|
+
type,
|
|
135
|
+
loggedInUser?.enterprise_id,
|
|
136
|
+
);
|
|
137
|
+
|
|
138
|
+
case 'custom':
|
|
139
|
+
// If you want Axios call here too:
|
|
140
|
+
try {
|
|
141
|
+
const response = await axios.get(
|
|
142
|
+
`https://external-source.com/${config.custom_source_id}`,
|
|
143
|
+
{ params },
|
|
144
|
+
);
|
|
145
|
+
return response.data;
|
|
146
|
+
} catch (error) {
|
|
147
|
+
console.error('⚠️ Custom source fetch failed:', error.message);
|
|
148
|
+
throw new BadRequestException('Failed to fetch custom source');
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
default:
|
|
152
|
+
throw new BadRequestException(`Unknown source: ${config.source}`);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
private async fetchFromEntity(
|
|
157
|
+
sourceList: string,
|
|
158
|
+
params: Record<string, any>,
|
|
159
|
+
inactiveIdsArray?: number[],
|
|
160
|
+
loggedInUser?: UserData,
|
|
161
|
+
) {
|
|
162
|
+
|
|
163
|
+
let result: { label: string; value: number }[] = [];
|
|
164
|
+
if (!sourceList) return result;
|
|
165
|
+
|
|
166
|
+
const entityMaster = await this.entityMasterService.getEntityData(
|
|
167
|
+
sourceList,
|
|
168
|
+
loggedInUser,
|
|
169
|
+
);
|
|
170
|
+
const tableName = entityMaster.data_source;
|
|
171
|
+
|
|
172
|
+
const applyCommonFilters = (qb: any, status?: number) => {
|
|
173
|
+
if (status) {
|
|
174
|
+
const isView = tableName.endsWith('_vw'); // auto-detect view
|
|
175
|
+
const statusColumn = isView ? 'status_id' : 'status';
|
|
176
|
+
qb.andWhere(`${tableName}.${statusColumn} = :status`, { status });
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
if (loggedInUser?.level_type && loggedInUser?.level_id) {
|
|
180
|
+
// Skip level filter for certain entities
|
|
181
|
+
if (!this.skipLevelFilterEntities.includes(sourceList)) {
|
|
182
|
+
qb.andWhere(
|
|
183
|
+
`${tableName}.level_type = :levelType AND ${tableName}.level_id = :levelId AND ${tableName}.enterprise_id = :enterprise_id`,
|
|
184
|
+
{
|
|
185
|
+
levelType: loggedInUser.level_type,
|
|
186
|
+
levelId: loggedInUser.level_id,
|
|
187
|
+
enterprise_id: loggedInUser.enterprise_id,
|
|
188
|
+
},
|
|
189
|
+
);
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
if (sourceList == 'BRN') {
|
|
194
|
+
// IN the case of BRN, we don't filter by level
|
|
195
|
+
qb.andWhere(
|
|
196
|
+
`${tableName}.parent_id = :enterprise_id AND ${tableName}.type = 'BRN'`,
|
|
197
|
+
{ enterprise_id: loggedInUser?.enterprise_id },
|
|
198
|
+
);
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
if (sourceList == 'BRNP') {
|
|
202
|
+
// IN the case of BRNP, we only filter by organization_id
|
|
203
|
+
qb.andWhere(`${tableName}.enterprise_id = :enterprise_id`, {
|
|
204
|
+
enterprise_id: loggedInUser?.enterprise_id,
|
|
205
|
+
});
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
if (sourceList == 'USR' || sourceList == 'UPR') {
|
|
209
|
+
// IN the case of USR/UPR, we don't filter by level
|
|
210
|
+
qb.andWhere(`${tableName}.enterprise_id = :enterprise_id`, {
|
|
211
|
+
enterprise_id: loggedInUser?.enterprise_id,
|
|
212
|
+
});
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
if (loggedInUser?.appcode && sourceList === 'ROL') {
|
|
216
|
+
if (!this.skipLevelFilterEntities.includes(sourceList)) {
|
|
217
|
+
qb.andWhere(`${tableName}.appcode = :appcode`, {
|
|
218
|
+
appcode: loggedInUser.appcode,
|
|
219
|
+
});
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
// ✅ Apply dynamic params
|
|
224
|
+
for (const key in params) {
|
|
225
|
+
qb.andWhere(`${tableName}.${key} = :${key}`, {
|
|
226
|
+
[key]: params[key],
|
|
227
|
+
});
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
// ✅ New: Exclude customers for USR/UPR
|
|
231
|
+
if (sourceList === 'USR' || sourceList === 'UPR') {
|
|
232
|
+
qb.andWhere(`${tableName}.is_customer is NULL`);
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
return qb;
|
|
236
|
+
};
|
|
237
|
+
|
|
238
|
+
const resolveStatus = await this.getResolvedListCode(
|
|
239
|
+
STATUS_ACTIVE,
|
|
240
|
+
loggedInUser?.enterprise_id || 0,
|
|
241
|
+
);
|
|
242
|
+
|
|
243
|
+
// Fetch active records
|
|
244
|
+
const activeQuery = applyCommonFilters(
|
|
245
|
+
this.entityManager
|
|
246
|
+
.createQueryBuilder()
|
|
247
|
+
.select('*')
|
|
248
|
+
.from(`${this.schema}.${tableName}`, tableName),
|
|
249
|
+
resolveStatus.id,
|
|
250
|
+
);
|
|
251
|
+
|
|
252
|
+
const activeResults = await activeQuery.getRawMany();
|
|
253
|
+
const activeIds = new Set(activeResults.map((r) => r.id));
|
|
254
|
+
|
|
255
|
+
// Add active entries first
|
|
256
|
+
activeResults.forEach((r) => {
|
|
257
|
+
result.push({ label: r.name, value: r.id });
|
|
258
|
+
});
|
|
259
|
+
|
|
260
|
+
const resolveInactiveStatus = await this.getResolvedListCode(
|
|
261
|
+
STATUS_INACTIVE,
|
|
262
|
+
loggedInUser?.organization_id || 0,
|
|
263
|
+
);
|
|
264
|
+
|
|
265
|
+
// Fetch inactive records (with same filters but without status condition)
|
|
266
|
+
if (inactiveIdsArray?.length) {
|
|
267
|
+
const inactiveQuery = applyCommonFilters(
|
|
268
|
+
this.entityManager
|
|
269
|
+
.createQueryBuilder()
|
|
270
|
+
.select('*')
|
|
271
|
+
.from(`${this.schema}.${tableName}`, tableName),
|
|
272
|
+
resolveInactiveStatus.id,
|
|
273
|
+
);
|
|
274
|
+
|
|
275
|
+
inactiveQuery.andWhere(`${tableName}.id IN (:...ids)`, {
|
|
276
|
+
ids: inactiveIdsArray,
|
|
277
|
+
});
|
|
278
|
+
|
|
279
|
+
const inactiveResults = await inactiveQuery.getRawMany();
|
|
280
|
+
|
|
281
|
+
inactiveResults.forEach((item) => {
|
|
282
|
+
if (!activeIds.has(item.id)) {
|
|
283
|
+
result.push({
|
|
284
|
+
label: `${item.name} [INACTIVE]`,
|
|
285
|
+
value: item.id,
|
|
286
|
+
});
|
|
287
|
+
}
|
|
288
|
+
});
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
return result;
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
private async fetchFromExternalSource(
|
|
295
|
+
customSourceId: number,
|
|
296
|
+
params?: Record<string, string>,
|
|
297
|
+
) {
|
|
298
|
+
const apiRegistry = await this.apiRegistryService.findById(customSourceId);
|
|
299
|
+
if (!apiRegistry) return [];
|
|
300
|
+
|
|
301
|
+
const url = `${apiRegistry.base_url}${apiRegistry.endpoint.replace(/{{(.*?)}}/g, (_, key) => params?.[key.trim()] || '')}`;
|
|
302
|
+
const payload = this.injectDynamicParams(
|
|
303
|
+
apiRegistry.request_payload_schema,
|
|
304
|
+
params,
|
|
305
|
+
);
|
|
306
|
+
|
|
307
|
+
const response = await firstValueFrom(
|
|
308
|
+
this.httpService.request({
|
|
309
|
+
url,
|
|
310
|
+
method: apiRegistry.http_method || 'POST',
|
|
311
|
+
data: payload,
|
|
312
|
+
headers: apiRegistry.headers ? apiRegistry.headers : undefined,
|
|
313
|
+
}),
|
|
314
|
+
);
|
|
315
|
+
|
|
316
|
+
const data = this.extractByPath(
|
|
317
|
+
response.data,
|
|
318
|
+
apiRegistry.response_data_path,
|
|
319
|
+
);
|
|
320
|
+
|
|
321
|
+
if (!Array.isArray(data)) {
|
|
322
|
+
return [{ label: data, value: data }];
|
|
323
|
+
}
|
|
324
|
+
|
|
325
|
+
if (typeof data[0] === 'string') {
|
|
326
|
+
return data.map((item: string) => ({ label: item, value: item }));
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
return data.map((item: any) => ({
|
|
330
|
+
label: item[apiRegistry?.label],
|
|
331
|
+
value: item[apiRegistry?.value],
|
|
332
|
+
}));
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
private injectDynamicParams(
|
|
336
|
+
payload: any,
|
|
337
|
+
params?: Record<string, string>,
|
|
338
|
+
): any {
|
|
339
|
+
if (params) {
|
|
340
|
+
const stringified = JSON.stringify(payload);
|
|
341
|
+
const replaced = stringified.replace(
|
|
342
|
+
/{{(.*?)}}/g,
|
|
343
|
+
(_, key) => params[key.trim()] || '',
|
|
344
|
+
);
|
|
345
|
+
return JSON.parse(replaced);
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
|
|
349
|
+
private extractByPath(obj: any, path: string): any {
|
|
350
|
+
if (!path || !obj) return obj;
|
|
351
|
+
|
|
352
|
+
return path.split('.').reduce((acc, key) => {
|
|
353
|
+
if (acc == null) return undefined;
|
|
354
|
+
|
|
355
|
+
// Try to convert to number if it's a numeric string (for array indexes)
|
|
356
|
+
const index = Number(key);
|
|
357
|
+
if (!isNaN(index) && Array.isArray(acc)) {
|
|
358
|
+
return acc[index];
|
|
359
|
+
}
|
|
360
|
+
|
|
361
|
+
return acc[key];
|
|
362
|
+
}, obj);
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
// createEntity method
|
|
366
|
+
async createEntity(entityData: any, loggedInUser: UserData): Promise<any> {
|
|
367
|
+
// Trim and validate name and code
|
|
368
|
+
const name = entityData.name?.trim();
|
|
369
|
+
// const code = entityData.code?.trim();
|
|
370
|
+
|
|
371
|
+
if (!name) {
|
|
372
|
+
throw new BadRequestException('Name is required and cannot be empty');
|
|
373
|
+
}
|
|
374
|
+
|
|
375
|
+
entityData.name = name;
|
|
376
|
+
entityData.is_factory = entityData.is_factory || 0;
|
|
377
|
+
entityData.source = entityData.source || 'master';
|
|
378
|
+
entityData.status = entityData.status || 'ACTIVE';
|
|
379
|
+
entityData.type = entityData.code;
|
|
380
|
+
entityData.sort_by = entityData.sort_by || 'asc';
|
|
381
|
+
|
|
382
|
+
// Check for duplicate name
|
|
383
|
+
const nameExists = await this.listMasterRepo.findOneByCondition({
|
|
384
|
+
name,
|
|
385
|
+
enterprise_id: loggedInUser.enterprise_id,
|
|
386
|
+
});
|
|
387
|
+
|
|
388
|
+
if (nameExists) {
|
|
389
|
+
throw new BadRequestException(
|
|
390
|
+
'A List Master with the same name already exists',
|
|
391
|
+
);
|
|
392
|
+
}
|
|
393
|
+
|
|
394
|
+
const createdListMaster = await this.entityServiceImpl.createEntity(
|
|
395
|
+
entityData,
|
|
396
|
+
loggedInUser,
|
|
397
|
+
);
|
|
398
|
+
|
|
399
|
+
if (!createdListMaster) {
|
|
400
|
+
throw new BadRequestException('Failed to create entity');
|
|
401
|
+
}
|
|
402
|
+
|
|
403
|
+
return createdListMaster;
|
|
404
|
+
}
|
|
405
|
+
|
|
406
|
+
async updateEntity(entityData: any, loggedInUser: UserData): Promise<any> {
|
|
407
|
+
return await this.entityServiceImpl.updateEntity(entityData, loggedInUser);
|
|
408
|
+
}
|
|
409
|
+
|
|
410
|
+
async getEntityData(
|
|
411
|
+
entity_type: string,
|
|
412
|
+
id: number,
|
|
413
|
+
loggedInUser,
|
|
414
|
+
): Promise<any> {
|
|
415
|
+
return await this.entityServiceImpl.getEntityData(
|
|
416
|
+
entity_type,
|
|
417
|
+
id,
|
|
418
|
+
loggedInUser,
|
|
419
|
+
);
|
|
420
|
+
}
|
|
421
|
+
|
|
422
|
+
async getAllListMasterByOrganization(
|
|
423
|
+
enterprise_id: number,
|
|
424
|
+
search?: string,
|
|
425
|
+
): Promise<any[]> {
|
|
426
|
+
return await this.listMasterRepo.findAllItems(enterprise_id, search);
|
|
427
|
+
}
|
|
428
|
+
|
|
429
|
+
async getDropDownData(
|
|
430
|
+
entity_type: string,
|
|
431
|
+
attribute_key: string,
|
|
432
|
+
loggedInUser: UserData,
|
|
433
|
+
body: Record<string, string>,
|
|
434
|
+
) {
|
|
435
|
+
let entityMaster = await this.entityMasterService.getEntityData(
|
|
436
|
+
entity_type,
|
|
437
|
+
loggedInUser,
|
|
438
|
+
);
|
|
439
|
+
let appCode = entityMaster.appcode;
|
|
440
|
+
|
|
441
|
+
const { inactiveIds, ...params } = body;
|
|
442
|
+
|
|
443
|
+
const inactiveIdsArray = inactiveIds
|
|
444
|
+
? inactiveIds.split(',').map((id) => parseInt(id, 10))
|
|
445
|
+
: [];
|
|
446
|
+
|
|
447
|
+
const currentAppCode = this.configService.get<string>('appcode');
|
|
448
|
+
|
|
449
|
+
if (currentAppCode === appCode || !currentAppCode) {
|
|
450
|
+
const entityAttribute =
|
|
451
|
+
await this.attributeMasterService.findByMappedEntityTypeAndAttributeKey(
|
|
452
|
+
entity_type,
|
|
453
|
+
attribute_key,
|
|
454
|
+
loggedInUser,
|
|
455
|
+
);
|
|
456
|
+
|
|
457
|
+
if (entityAttribute && entityAttribute.data_source_type) {
|
|
458
|
+
const listMaster = await this.listMasterRepo.findByType(
|
|
459
|
+
entityAttribute.datasource_list,
|
|
460
|
+
loggedInUser?.enterprise_id,
|
|
461
|
+
);
|
|
462
|
+
|
|
463
|
+
if (!listMaster) {
|
|
464
|
+
return;
|
|
465
|
+
}
|
|
466
|
+
|
|
467
|
+
switch (listMaster.source) {
|
|
468
|
+
case 'entity':
|
|
469
|
+
return this.fetchFromEntity(
|
|
470
|
+
listMaster.type,
|
|
471
|
+
params,
|
|
472
|
+
inactiveIdsArray,
|
|
473
|
+
loggedInUser,
|
|
474
|
+
);
|
|
475
|
+
|
|
476
|
+
case 'master':
|
|
477
|
+
return this.listItemsRepo.findItemsByType(
|
|
478
|
+
listMaster.type,
|
|
479
|
+
listMaster.sort_by,
|
|
480
|
+
inactiveIdsArray,
|
|
481
|
+
loggedInUser.enterprise_id,
|
|
482
|
+
params,
|
|
483
|
+
);
|
|
484
|
+
|
|
485
|
+
case 'operator':
|
|
486
|
+
return this.listItemsRepo.findOperatorsByType(
|
|
487
|
+
listMaster?.type,
|
|
488
|
+
loggedInUser?.enterprise_id,
|
|
489
|
+
);
|
|
490
|
+
|
|
491
|
+
case 'custom':
|
|
492
|
+
// If you want Axios call here too:
|
|
493
|
+
try {
|
|
494
|
+
const response = await axios.get(
|
|
495
|
+
`https://external-source.com/${listMaster.custom_source_id}`,
|
|
496
|
+
{ params },
|
|
497
|
+
);
|
|
498
|
+
return response.data;
|
|
499
|
+
} catch (error) {
|
|
500
|
+
console.error('⚠️ Custom source fetch failed:', error.message);
|
|
501
|
+
throw new BadRequestException('Failed to fetch custom source');
|
|
502
|
+
}
|
|
503
|
+
|
|
504
|
+
default:
|
|
505
|
+
throw new BadRequestException(
|
|
506
|
+
`Unknown source: ${listMaster.source}`,
|
|
507
|
+
);
|
|
508
|
+
}
|
|
509
|
+
}
|
|
510
|
+
} else {
|
|
511
|
+
const client = this.factory.getClient(appCode);
|
|
512
|
+
|
|
513
|
+
if (!client) {
|
|
514
|
+
return;
|
|
515
|
+
}
|
|
516
|
+
|
|
517
|
+
return client
|
|
518
|
+
.send('getDropdownData', {
|
|
519
|
+
entity_type,
|
|
520
|
+
attribute_key,
|
|
521
|
+
loggedInUser,
|
|
522
|
+
body,
|
|
523
|
+
})
|
|
524
|
+
.toPromise();
|
|
525
|
+
}
|
|
526
|
+
}
|
|
527
|
+
}
|