@solidstarters/solid-core 1.2.192 → 1.2.200

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.
Files changed (376) hide show
  1. package/dist/commands/mcp.command.d.ts +20 -0
  2. package/dist/commands/mcp.command.d.ts.map +1 -0
  3. package/dist/commands/mcp.command.js +208 -0
  4. package/dist/commands/mcp.command.js.map +1 -0
  5. package/dist/commands/refresh-model.command.d.ts +8 -5
  6. package/dist/commands/refresh-model.command.d.ts.map +1 -1
  7. package/dist/commands/refresh-model.command.js +32 -0
  8. package/dist/commands/refresh-model.command.js.map +1 -1
  9. package/dist/commands/remove-fields.command.js +1 -1
  10. package/dist/commands/remove-fields.command.js.map +1 -1
  11. package/dist/commands/seed.command.d.ts +2 -0
  12. package/dist/commands/seed.command.d.ts.map +1 -1
  13. package/dist/commands/seed.command.js +28 -1
  14. package/dist/commands/seed.command.js.map +1 -1
  15. package/dist/controllers/authentication.controller.js +1 -1
  16. package/dist/controllers/authentication.controller.js.map +1 -1
  17. package/dist/controllers/email-template.controller.js +1 -1
  18. package/dist/controllers/email-template.controller.js.map +1 -1
  19. package/dist/controllers/field-metadata.controller.d.ts +136 -4
  20. package/dist/controllers/field-metadata.controller.d.ts.map +1 -1
  21. package/dist/controllers/field-metadata.controller.js +14 -1
  22. package/dist/controllers/field-metadata.controller.js.map +1 -1
  23. package/dist/controllers/media-storage-provider-metadata.controller.js +1 -1
  24. package/dist/controllers/media-storage-provider-metadata.controller.js.map +1 -1
  25. package/dist/controllers/model-metadata.controller.js +1 -1
  26. package/dist/controllers/model-metadata.controller.js.map +1 -1
  27. package/dist/controllers/module-metadata.controller.js +1 -1
  28. package/dist/controllers/module-metadata.controller.js.map +1 -1
  29. package/dist/controllers/otp-authentication.controller.js +1 -1
  30. package/dist/controllers/otp-authentication.controller.js.map +1 -1
  31. package/dist/controllers/service.controller.js +1 -1
  32. package/dist/controllers/service.controller.js.map +1 -1
  33. package/dist/controllers/sms-template.controller.js +1 -1
  34. package/dist/controllers/sms-template.controller.js.map +1 -1
  35. package/dist/controllers/test-queue.controller.js +1 -1
  36. package/dist/controllers/test-queue.controller.js.map +1 -1
  37. package/dist/controllers/test.controller.js +1 -1
  38. package/dist/controllers/test.controller.js.map +1 -1
  39. package/dist/dtos/create-ai-interaction.dto.d.ts +2 -0
  40. package/dist/dtos/create-ai-interaction.dto.d.ts.map +1 -1
  41. package/dist/dtos/create-ai-interaction.dto.js +14 -1
  42. package/dist/dtos/create-ai-interaction.dto.js.map +1 -1
  43. package/dist/dtos/create-field-metadata.dto.d.ts +22 -5
  44. package/dist/dtos/create-field-metadata.dto.d.ts.map +1 -1
  45. package/dist/dtos/create-field-metadata.dto.js +33 -7
  46. package/dist/dtos/create-field-metadata.dto.js.map +1 -1
  47. package/dist/dtos/create-import-transaction.dto.d.ts +0 -1
  48. package/dist/dtos/create-import-transaction.dto.d.ts.map +1 -1
  49. package/dist/dtos/create-import-transaction.dto.js +1 -7
  50. package/dist/dtos/create-import-transaction.dto.js.map +1 -1
  51. package/dist/dtos/create-model-metadata.dto.d.ts +2 -0
  52. package/dist/dtos/create-model-metadata.dto.d.ts.map +1 -1
  53. package/dist/dtos/create-model-metadata.dto.js +13 -1
  54. package/dist/dtos/create-model-metadata.dto.js.map +1 -1
  55. package/dist/dtos/create-role-metadata.dto.d.ts.map +1 -1
  56. package/dist/dtos/create-role-metadata.dto.js +5 -1
  57. package/dist/dtos/create-role-metadata.dto.js.map +1 -1
  58. package/dist/dtos/resolve-s3-url.dto.d.ts +10 -0
  59. package/dist/dtos/resolve-s3-url.dto.d.ts.map +1 -0
  60. package/dist/dtos/resolve-s3-url.dto.js +49 -0
  61. package/dist/dtos/resolve-s3-url.dto.js.map +1 -0
  62. package/dist/dtos/update-ai-interaction.dto.d.ts +2 -0
  63. package/dist/dtos/update-ai-interaction.dto.d.ts.map +1 -1
  64. package/dist/dtos/update-ai-interaction.dto.js +13 -1
  65. package/dist/dtos/update-ai-interaction.dto.js.map +1 -1
  66. package/dist/dtos/update-field-metadata.dto.d.ts.map +1 -1
  67. package/dist/dtos/update-field-metadata.dto.js.map +1 -1
  68. package/dist/dtos/update-import-transaction.dto.d.ts +0 -1
  69. package/dist/dtos/update-import-transaction.dto.d.ts.map +1 -1
  70. package/dist/dtos/update-import-transaction.dto.js +1 -7
  71. package/dist/dtos/update-import-transaction.dto.js.map +1 -1
  72. package/dist/entities/action-metadata.entity.js +4 -4
  73. package/dist/entities/action-metadata.entity.js.map +1 -1
  74. package/dist/entities/ai-interaction.entity.d.ts +2 -0
  75. package/dist/entities/ai-interaction.entity.d.ts.map +1 -1
  76. package/dist/entities/ai-interaction.entity.js +15 -6
  77. package/dist/entities/ai-interaction.entity.js.map +1 -1
  78. package/dist/entities/chatter-message-details.entity.d.ts.map +1 -1
  79. package/dist/entities/chatter-message-details.entity.js +1 -1
  80. package/dist/entities/chatter-message-details.entity.js.map +1 -1
  81. package/dist/entities/chatter-message.entity.js +1 -1
  82. package/dist/entities/chatter-message.entity.js.map +1 -1
  83. package/dist/entities/common.entity.d.ts +2 -3
  84. package/dist/entities/common.entity.d.ts.map +1 -1
  85. package/dist/entities/common.entity.js +9 -11
  86. package/dist/entities/common.entity.js.map +1 -1
  87. package/dist/entities/dashboard-question-sql-dataset-config.entity.js +2 -2
  88. package/dist/entities/dashboard-question-sql-dataset-config.entity.js.map +1 -1
  89. package/dist/entities/dashboard-question.entity.js +3 -3
  90. package/dist/entities/dashboard-question.entity.js.map +1 -1
  91. package/dist/entities/dashboard-variable.entity.js +3 -3
  92. package/dist/entities/dashboard-variable.entity.js.map +1 -1
  93. package/dist/entities/dashboard.entity.js +1 -1
  94. package/dist/entities/dashboard.entity.js.map +1 -1
  95. package/dist/entities/email-template.entity.d.ts.map +1 -1
  96. package/dist/entities/email-template.entity.js +4 -3
  97. package/dist/entities/email-template.entity.js.map +1 -1
  98. package/dist/entities/export-template.entity.js +2 -2
  99. package/dist/entities/export-template.entity.js.map +1 -1
  100. package/dist/entities/export-transaction.entity.js +2 -2
  101. package/dist/entities/export-transaction.entity.js.map +1 -1
  102. package/dist/entities/field-metadata.entity.d.ts +1 -0
  103. package/dist/entities/field-metadata.entity.d.ts.map +1 -1
  104. package/dist/entities/field-metadata.entity.js +8 -4
  105. package/dist/entities/field-metadata.entity.js.map +1 -1
  106. package/dist/entities/import-transaction-error-log.entity.js +1 -1
  107. package/dist/entities/import-transaction-error-log.entity.js.map +1 -1
  108. package/dist/entities/import-transaction.entity.d.ts +0 -1
  109. package/dist/entities/import-transaction.entity.d.ts.map +1 -1
  110. package/dist/entities/import-transaction.entity.js +2 -7
  111. package/dist/entities/import-transaction.entity.js.map +1 -1
  112. package/dist/entities/legacy-common-with-id.entity.d.ts +5 -0
  113. package/dist/entities/legacy-common-with-id.entity.d.ts.map +1 -0
  114. package/dist/entities/legacy-common-with-id.entity.js +32 -0
  115. package/dist/entities/legacy-common-with-id.entity.js.map +1 -0
  116. package/dist/entities/legacy-common.entity.d.ts +13 -0
  117. package/dist/entities/legacy-common.entity.d.ts.map +1 -0
  118. package/dist/entities/legacy-common.entity.js +67 -0
  119. package/dist/entities/legacy-common.entity.js.map +1 -0
  120. package/dist/entities/list-of-values.entity.js +2 -2
  121. package/dist/entities/list-of-values.entity.js.map +1 -1
  122. package/dist/entities/locale.entity.js +1 -1
  123. package/dist/entities/locale.entity.js.map +1 -1
  124. package/dist/entities/media.entity.d.ts.map +1 -1
  125. package/dist/entities/media.entity.js +3 -3
  126. package/dist/entities/media.entity.js.map +1 -1
  127. package/dist/entities/menu-item-metadata.entity.js +4 -4
  128. package/dist/entities/menu-item-metadata.entity.js.map +1 -1
  129. package/dist/entities/model-metadata.entity.d.ts +2 -0
  130. package/dist/entities/model-metadata.entity.d.ts.map +1 -1
  131. package/dist/entities/model-metadata.entity.js +11 -3
  132. package/dist/entities/model-metadata.entity.js.map +1 -1
  133. package/dist/entities/mq-message-queue.entity.js +1 -1
  134. package/dist/entities/mq-message-queue.entity.js.map +1 -1
  135. package/dist/entities/mq-message.entity.d.ts +3 -3
  136. package/dist/entities/mq-message.entity.d.ts.map +1 -1
  137. package/dist/entities/mq-message.entity.js +10 -10
  138. package/dist/entities/mq-message.entity.js.map +1 -1
  139. package/dist/entities/saved-filters.entity.js +4 -4
  140. package/dist/entities/saved-filters.entity.js.map +1 -1
  141. package/dist/entities/scheduled-job.entity.js +4 -4
  142. package/dist/entities/scheduled-job.entity.js.map +1 -1
  143. package/dist/entities/security-rule.entity.js +2 -2
  144. package/dist/entities/security-rule.entity.js.map +1 -1
  145. package/dist/entities/setting.entity.js +2 -2
  146. package/dist/entities/setting.entity.js.map +1 -1
  147. package/dist/entities/sms-template.entity.js +2 -2
  148. package/dist/entities/sms-template.entity.js.map +1 -1
  149. package/dist/entities/user-activity-history.entity.d.ts.map +1 -1
  150. package/dist/entities/user-activity-history.entity.js +1 -1
  151. package/dist/entities/user-activity-history.entity.js.map +1 -1
  152. package/dist/entities/user-view-metadata.entity.js +2 -2
  153. package/dist/entities/user-view-metadata.entity.js.map +1 -1
  154. package/dist/entities/user.entity.js +13 -15
  155. package/dist/entities/user.entity.js.map +1 -1
  156. package/dist/entities/view-metadata.entity.js +3 -3
  157. package/dist/entities/view-metadata.entity.js.map +1 -1
  158. package/dist/helpers/date.helper.d.ts +1 -1
  159. package/dist/helpers/date.helper.d.ts.map +1 -1
  160. package/dist/helpers/date.helper.js +24 -2
  161. package/dist/helpers/date.helper.js.map +1 -1
  162. package/dist/helpers/model-metadata-helper.service.d.ts +4 -1
  163. package/dist/helpers/model-metadata-helper.service.d.ts.map +1 -1
  164. package/dist/helpers/model-metadata-helper.service.js +48 -30
  165. package/dist/helpers/model-metadata-helper.service.js.map +1 -1
  166. package/dist/helpers/module.helper.d.ts.map +1 -1
  167. package/dist/helpers/module.helper.js +1 -3
  168. package/dist/helpers/module.helper.js.map +1 -1
  169. package/dist/helpers/schematic.service.d.ts +9 -3
  170. package/dist/helpers/schematic.service.d.ts.map +1 -1
  171. package/dist/helpers/schematic.service.js +49 -32
  172. package/dist/helpers/schematic.service.js.map +1 -1
  173. package/dist/helpers/solid-registry.d.ts +3 -2
  174. package/dist/helpers/solid-registry.d.ts.map +1 -1
  175. package/dist/helpers/solid-registry.js +10 -0
  176. package/dist/helpers/solid-registry.js.map +1 -1
  177. package/dist/helpers/typeorm-db-helper.d.ts +3 -0
  178. package/dist/helpers/typeorm-db-helper.d.ts.map +1 -0
  179. package/dist/helpers/typeorm-db-helper.js +24 -0
  180. package/dist/helpers/typeorm-db-helper.js.map +1 -0
  181. package/dist/index.d.ts +5 -1
  182. package/dist/index.d.ts.map +1 -1
  183. package/dist/index.js +5 -1
  184. package/dist/index.js.map +1 -1
  185. package/dist/interfaces.d.ts +3 -0
  186. package/dist/interfaces.d.ts.map +1 -1
  187. package/dist/interfaces.js.map +1 -1
  188. package/dist/jobs/database/trigger-mcp-client-subscriber-database.service.d.ts.map +1 -1
  189. package/dist/jobs/database/trigger-mcp-client-subscriber-database.service.js +6 -1
  190. package/dist/jobs/database/trigger-mcp-client-subscriber-database.service.js.map +1 -1
  191. package/dist/seeders/module-metadata-seeder.service.d.ts +1 -1
  192. package/dist/seeders/module-metadata-seeder.service.d.ts.map +1 -1
  193. package/dist/seeders/module-metadata-seeder.service.js +39 -17
  194. package/dist/seeders/module-metadata-seeder.service.js.map +1 -1
  195. package/dist/seeders/seed-data/solid-core-metadata.json +53 -468
  196. package/dist/seeders/system-fields-seeder.service.d.ts.map +1 -1
  197. package/dist/seeders/system-fields-seeder.service.js +1 -1
  198. package/dist/seeders/system-fields-seeder.service.js.map +1 -1
  199. package/dist/services/ai-interaction.service.js +1 -1
  200. package/dist/services/ai-interaction.service.js.map +1 -1
  201. package/dist/services/computed-fields/entity/alpha-num-external-id-computed-field-provider.d.ts.map +1 -1
  202. package/dist/services/computed-fields/entity/alpha-num-external-id-computed-field-provider.js +6 -4
  203. package/dist/services/computed-fields/entity/alpha-num-external-id-computed-field-provider.js.map +1 -1
  204. package/dist/services/crud-helper.service.d.ts +8 -0
  205. package/dist/services/crud-helper.service.d.ts.map +1 -1
  206. package/dist/services/crud-helper.service.js +28 -5
  207. package/dist/services/crud-helper.service.js.map +1 -1
  208. package/dist/services/crud.service.d.ts +4 -1
  209. package/dist/services/crud.service.d.ts.map +1 -1
  210. package/dist/services/crud.service.js +87 -7
  211. package/dist/services/crud.service.js.map +1 -1
  212. package/dist/services/field-metadata.service.d.ts +143 -6
  213. package/dist/services/field-metadata.service.d.ts.map +1 -1
  214. package/dist/services/field-metadata.service.js +211 -56
  215. package/dist/services/field-metadata.service.js.map +1 -1
  216. package/dist/services/menu-item-metadata.service.d.ts +3 -1
  217. package/dist/services/menu-item-metadata.service.d.ts.map +1 -1
  218. package/dist/services/menu-item-metadata.service.js +31 -11
  219. package/dist/services/menu-item-metadata.service.js.map +1 -1
  220. package/dist/services/model-metadata.service.d.ts.map +1 -1
  221. package/dist/services/model-metadata.service.js +14 -3
  222. package/dist/services/model-metadata.service.js.map +1 -1
  223. package/dist/services/mq-message.service.d.ts.map +1 -1
  224. package/dist/services/mq-message.service.js +3 -4
  225. package/dist/services/mq-message.service.js.map +1 -1
  226. package/dist/services/queues/publisher-factory.service.d.ts.map +1 -1
  227. package/dist/services/queues/publisher-factory.service.js +4 -0
  228. package/dist/services/queues/publisher-factory.service.js.map +1 -1
  229. package/dist/{providers → services/selection-providers}/list-of-values-selection-providers.service.d.ts +2 -2
  230. package/dist/services/selection-providers/list-of-values-selection-providers.service.d.ts.map +1 -0
  231. package/dist/{providers → services/selection-providers}/list-of-values-selection-providers.service.js +3 -3
  232. package/dist/services/selection-providers/list-of-values-selection-providers.service.js.map +1 -0
  233. package/dist/services/selection-providers/pseudo-foreign-key-selection-provider.service.d.ts +20 -0
  234. package/dist/services/selection-providers/pseudo-foreign-key-selection-provider.service.d.ts.map +1 -0
  235. package/dist/services/selection-providers/pseudo-foreign-key-selection-provider.service.js +87 -0
  236. package/dist/services/selection-providers/pseudo-foreign-key-selection-provider.service.js.map +1 -0
  237. package/dist/services/setting.service.d.ts.map +1 -1
  238. package/dist/services/setting.service.js +2 -1
  239. package/dist/services/setting.service.js.map +1 -1
  240. package/dist/services/solid-introspect.service.d.ts +19 -3
  241. package/dist/services/solid-introspect.service.d.ts.map +1 -1
  242. package/dist/services/solid-introspect.service.js +83 -11
  243. package/dist/services/solid-introspect.service.js.map +1 -1
  244. package/dist/services/view-metadata.service.d.ts +3 -1
  245. package/dist/services/view-metadata.service.d.ts.map +1 -1
  246. package/dist/services/view-metadata.service.js +31 -5
  247. package/dist/services/view-metadata.service.js.map +1 -1
  248. package/dist/solid-core.module.d.ts.map +1 -1
  249. package/dist/solid-core.module.js +5 -1
  250. package/dist/solid-core.module.js.map +1 -1
  251. package/dist/subscribers/audit.subscriber.d.ts +3 -2
  252. package/dist/subscribers/audit.subscriber.d.ts.map +1 -1
  253. package/dist/subscribers/audit.subscriber.js +6 -12
  254. package/dist/subscribers/audit.subscriber.js.map +1 -1
  255. package/dist/subscribers/computed-entity-field.subscriber.d.ts +3 -2
  256. package/dist/subscribers/computed-entity-field.subscriber.d.ts.map +1 -1
  257. package/dist/subscribers/computed-entity-field.subscriber.js +7 -9
  258. package/dist/subscribers/computed-entity-field.subscriber.js.map +1 -1
  259. package/dist/subscribers/created-by-updated-by.subscriber.d.ts +4 -2
  260. package/dist/subscribers/created-by-updated-by.subscriber.d.ts.map +1 -1
  261. package/dist/subscribers/created-by-updated-by.subscriber.js +10 -8
  262. package/dist/subscribers/created-by-updated-by.subscriber.js.map +1 -1
  263. package/dist/subscribers/field-metadata.subscriber.d.ts.map +1 -1
  264. package/dist/subscribers/field-metadata.subscriber.js +0 -1
  265. package/dist/subscribers/field-metadata.subscriber.js.map +1 -1
  266. package/dist/subscribers/model-metadata.subscriber.d.ts.map +1 -1
  267. package/dist/subscribers/model-metadata.subscriber.js +3 -2
  268. package/dist/subscribers/model-metadata.subscriber.js.map +1 -1
  269. package/dist/subscribers/soft-delete-aware-event.subscriber.d.ts +3 -2
  270. package/dist/subscribers/soft-delete-aware-event.subscriber.d.ts.map +1 -1
  271. package/dist/subscribers/soft-delete-aware-event.subscriber.js +6 -11
  272. package/dist/subscribers/soft-delete-aware-event.subscriber.js.map +1 -1
  273. package/dist/tsconfig.tsbuildinfo +1 -1
  274. package/package.json +1 -1
  275. package/sql/mssql/proc_CleanupModelMetadata.sql +175 -0
  276. package/sql/mssql/proc_CleanupModuleMetadata.sql +78 -0
  277. package/sql/mssql/scratchpad.sql +10 -0
  278. package/sql/postgres/proc_CleanupModelMetadata.sql +148 -0
  279. package/sql/postgres/proc_CleanupModuleMetadata.sql +50 -0
  280. package/sql/postgres/scratchpad.sql +12 -0
  281. package/sql-server-changes.txt +88 -0
  282. package/src/commands/mcp.command.ts +215 -0
  283. package/src/commands/refresh-model.command.ts +37 -5
  284. package/src/commands/remove-fields.command.ts +1 -1
  285. package/src/commands/seed.command.ts +34 -15
  286. package/src/controllers/authentication.controller.ts +1 -1
  287. package/src/controllers/email-template.controller.ts +1 -1
  288. package/src/controllers/field-metadata.controller.ts +7 -1
  289. package/src/controllers/media-storage-provider-metadata.controller.ts +1 -1
  290. package/src/controllers/model-metadata.controller.ts +1 -1
  291. package/src/controllers/module-metadata.controller.ts +1 -1
  292. package/src/controllers/otp-authentication.controller.ts +1 -1
  293. package/src/controllers/service.controller.ts +1 -1
  294. package/src/controllers/sms-template.controller.ts +1 -1
  295. package/src/controllers/test-queue.controller.ts +1 -1
  296. package/src/controllers/test.controller.ts +1 -1
  297. package/src/dtos/create-ai-interaction.dto.ts +16 -9
  298. package/src/dtos/create-field-metadata.dto.ts +42 -12
  299. package/src/dtos/create-import-transaction.dto.ts +0 -4
  300. package/src/dtos/create-model-metadata.dto.ts +10 -0
  301. package/src/dtos/create-role-metadata.dto.ts +8 -2
  302. package/src/dtos/resolve-s3-url.dto.ts +33 -0
  303. package/src/dtos/update-ai-interaction.dto.ts +16 -9
  304. package/src/dtos/update-field-metadata.dto.ts +1 -2
  305. package/src/dtos/update-import-transaction.dto.ts +0 -4
  306. package/src/entities/action-metadata.entity.ts +4 -4
  307. package/src/entities/ai-interaction.entity.ts +16 -12
  308. package/src/entities/chatter-message-details.entity.ts +18 -18
  309. package/src/entities/chatter-message.entity.ts +2 -2
  310. package/src/entities/common.entity.ts +19 -10
  311. package/src/entities/dashboard-question-sql-dataset-config.entity.ts +2 -2
  312. package/src/entities/dashboard-question.entity.ts +3 -3
  313. package/src/entities/dashboard-variable.entity.ts +3 -3
  314. package/src/entities/dashboard.entity.ts +1 -1
  315. package/src/entities/email-template.entity.ts +4 -3
  316. package/src/entities/export-template.entity.ts +2 -2
  317. package/src/entities/export-transaction.entity.ts +2 -2
  318. package/src/entities/field-metadata.entity.ts +6 -3
  319. package/src/entities/import-transaction-error-log.entity.ts +2 -2
  320. package/src/entities/import-transaction.entity.ts +2 -5
  321. package/src/entities/legacy-common-with-id.entity.ts +11 -0
  322. package/src/entities/legacy-common.entity.ts +57 -0
  323. package/src/entities/list-of-values.entity.ts +2 -2
  324. package/src/entities/locale.entity.ts +1 -1
  325. package/src/entities/media.entity.ts +25 -25
  326. package/src/entities/menu-item-metadata.entity.ts +4 -4
  327. package/src/entities/model-metadata.entity.ts +9 -2
  328. package/src/entities/mq-message-queue.entity.ts +1 -1
  329. package/src/entities/mq-message.entity.ts +34 -34
  330. package/src/entities/saved-filters.entity.ts +5 -5
  331. package/src/entities/scheduled-job.entity.ts +4 -4
  332. package/src/entities/security-rule.entity.ts +2 -2
  333. package/src/entities/setting.entity.ts +2 -2
  334. package/src/entities/sms-template.entity.ts +2 -2
  335. package/src/entities/user-activity-history.entity.ts +14 -14
  336. package/src/entities/user-view-metadata.entity.ts +2 -2
  337. package/src/entities/user.entity.ts +15 -15
  338. package/src/entities/view-metadata.entity.ts +3 -3
  339. package/src/helpers/date.helper.ts +31 -10
  340. package/src/helpers/model-metadata-helper.service.ts +63 -32
  341. package/src/helpers/module.helper.ts +4 -3
  342. package/src/helpers/schematic.service.ts +72 -46
  343. package/src/helpers/solid-registry.ts +14 -2
  344. package/src/helpers/typeorm-db-helper.ts +26 -0
  345. package/src/index.ts +5 -1
  346. package/src/interfaces.ts +3 -0
  347. package/src/jobs/database/trigger-mcp-client-subscriber-database.service.ts +17 -21
  348. package/src/seeders/module-metadata-seeder.service.ts +73 -29
  349. package/src/seeders/seed-data/solid-core-metadata.json +55 -470
  350. package/src/seeders/system-fields-seeder.service.ts +34 -35
  351. package/src/services/ai-interaction.service.ts +1 -1
  352. package/src/services/computed-fields/entity/alpha-num-external-id-computed-field-provider.ts +13 -9
  353. package/src/services/crud-helper.service.ts +32 -4
  354. package/src/services/crud.service.ts +158 -10
  355. package/src/services/field-metadata.service.ts +272 -67
  356. package/src/services/menu-item-metadata.service.ts +81 -50
  357. package/src/services/model-metadata.service.ts +18 -3
  358. package/src/services/mq-message.service.ts +16 -15
  359. package/src/services/queues/publisher-factory.service.ts +2 -1
  360. package/src/{providers → services/selection-providers}/list-of-values-selection-providers.service.ts +2 -2
  361. package/src/services/selection-providers/pseudo-foreign-key-selection-provider.service.ts +94 -0
  362. package/src/services/setting.service.ts +13 -10
  363. package/src/services/solid-introspect.service.ts +110 -10
  364. package/src/services/view-metadata.service.ts +48 -18
  365. package/src/solid-core.module.ts +5 -1
  366. package/src/subscribers/audit.subscriber.ts +12 -8
  367. package/src/subscribers/computed-entity-field.subscriber.ts +13 -7
  368. package/src/subscribers/created-by-updated-by.subscriber.ts +14 -8
  369. package/src/subscribers/field-metadata.subscriber.ts +1 -1
  370. package/src/subscribers/model-metadata.subscriber.ts +4 -2
  371. package/src/subscribers/soft-delete-aware-event.subscriber.ts +12 -6
  372. package/dist/providers/list-of-values-selection-providers.service.d.ts.map +0 -1
  373. package/dist/providers/list-of-values-selection-providers.service.js.map +0 -1
  374. package/src/commands/ingest-rag-chunking-strategy-for.md +0 -224
  375. package/src/entities/user.entity.ts.bkp +0 -144
  376. package/src/services/docker exec -it mssql //" +0 -8
@@ -3,6 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.CRUDService = void 0;
4
4
  const common_1 = require("@nestjs/common");
5
5
  const class_validator_1 = require("class-validator");
6
+ const src_1 = require("..");
6
7
  const error_messages_1 = require("../constants/error-messages");
7
8
  const success_messages_1 = require("../constants/success-messages");
8
9
  const typeorm_1 = require("typeorm");
@@ -131,6 +132,9 @@ class CRUDService {
131
132
  if (!entity) {
132
133
  throw new Error(`Entity [${this.moduleName}.${this.modelName}] with id ${id} not found`);
133
134
  }
135
+ if (model.draftPublishWorkflow === true && entity.publishedAt) {
136
+ throw new common_1.BadRequestException(`Cannot update a published record for model ${this.modelName}. Unpublish it first.`);
137
+ }
134
138
  updateDto.id = id;
135
139
  let hasMediaFields = false;
136
140
  const fieldsToProcess = [...model.fields];
@@ -172,6 +176,9 @@ class CRUDService {
172
176
  if (!entity) {
173
177
  throw new Error(`Entity [${this.moduleName}.${this.modelName}] with id ${id} not found`);
174
178
  }
179
+ if (model.draftPublishWorkflow === true && entity.publishedAt) {
180
+ throw new common_1.BadRequestException(`Cannot update a published record for model ${this.modelName}, Unpublish it first.`);
181
+ }
175
182
  if (model.internationalisation) {
176
183
  const childEntities = await this.repo.find({
177
184
  where: { defaultEntityLocaleId: id }
@@ -328,6 +335,7 @@ class CRUDService {
328
335
  async find(basicFilterDto, solidRequestContext = {}) {
329
336
  const alias = 'entity';
330
337
  let { limit, offset, populateMedia, populateGroup, groupFilter } = basicFilterDto;
338
+ const populateUserIdFields = this.crudHelperService.extractUserIdFieldsFromPopulate(basicFilterDto.populate);
331
339
  const { singularName, internationalisation, draftPublishWorkflow } = await this.loadModel();
332
340
  if (solidRequestContext.activeUser) {
333
341
  const hasPermission = this.crudHelperService.hasReadPermissionOnModel(solidRequestContext.activeUser, singularName);
@@ -338,14 +346,15 @@ class CRUDService {
338
346
  const requestContextService = this.moduleRef.get(request_context_service_1.RequestContextService, { strict: false });
339
347
  requestContextService.setRequestFilter(basicFilterDto);
340
348
  var qb = await this.repo.createSecurityRuleAwareQueryBuilder(alias);
341
- qb = this.crudHelperService.buildFilterQuery(qb, basicFilterDto, alias);
342
349
  if (internationalisation && draftPublishWorkflow) {
343
350
  qb = this.crudHelperService.buildFilterQuery(qb, basicFilterDto, alias, internationalisation, draftPublishWorkflow, this.moduleRef);
344
351
  }
352
+ else {
353
+ qb = this.crudHelperService.buildFilterQuery(qb, basicFilterDto, alias);
354
+ }
345
355
  if (basicFilterDto.groupBy) {
346
- const { groupMeta, groupRecords } = await this.handleGroupFind(qb, groupFilter, populateGroup, alias, populateMedia);
356
+ const { groupMeta, groupRecords } = await this.handleGroupFind(qb, groupFilter, populateGroup, alias, populateUserIdFields, populateMedia);
347
357
  const totalGroups = await this.crudHelperService.countGroupedRecords(qb, basicFilterDto, alias);
348
- qb = this.crudHelperService.buildFilterQuery(qb, basicFilterDto, alias);
349
358
  return {
350
359
  meta: {
351
360
  "totalRecords": totalGroups
@@ -355,21 +364,24 @@ class CRUDService {
355
364
  };
356
365
  }
357
366
  else {
358
- const { meta, records } = await this.handleNonGroupFind(qb, populateMedia, offset, limit, alias);
367
+ const { meta, records } = await this.handleNonGroupFind(qb, populateUserIdFields, populateMedia, offset, limit, alias);
359
368
  return {
360
369
  meta,
361
370
  records,
362
371
  };
363
372
  }
364
373
  }
365
- async handleNonGroupFind(qb, populateMedia, offset, limit, alias) {
374
+ async handleNonGroupFind(qb, populateUserIdFields, populateMedia, offset, limit, alias) {
366
375
  const [entities, count] = await qb.getManyAndCount();
376
+ if (populateUserIdFields && populateUserIdFields.length > 0) {
377
+ await this.handlePopulateUserIdFields(populateUserIdFields, entities);
378
+ }
367
379
  if (populateMedia && populateMedia.length > 0) {
368
380
  await this.handlePopulateMedia(populateMedia, entities);
369
381
  }
370
382
  return this.wrapFindResponse(offset, limit, count, entities);
371
383
  }
372
- async handleGroupFind(qb, groupFilter, populateGroup, alias, populateMedia) {
384
+ async handleGroupFind(qb, groupFilter, populateGroup, alias, populateUserIdFields, populateMedia) {
373
385
  const groupByResult = await qb.getRawMany();
374
386
  const groupMeta = [];
375
387
  const groupRecords = [];
@@ -379,6 +391,9 @@ class CRUDService {
379
391
  groupByQb = this.crudHelperService.buildFilterQuery(groupByQb, groupFilter, alias);
380
392
  groupByQb = this.crudHelperService.buildGroupByRecordsQuery(groupByQb, group, alias);
381
393
  const [entities, count] = await groupByQb.getManyAndCount();
394
+ if (populateUserIdFields && populateUserIdFields.length > 0) {
395
+ await this.handlePopulateUserIdFields(populateUserIdFields, entities);
396
+ }
382
397
  if (populateMedia && populateMedia.length > 0) {
383
398
  await this.handlePopulateMedia(populateMedia, entities);
384
399
  }
@@ -407,6 +422,20 @@ class CRUDService {
407
422
  };
408
423
  return r;
409
424
  }
425
+ async handlePopulateUserIdFields(userIdFields, entities) {
426
+ const userRepository = this.entityManager.getRepository(src_1.User);
427
+ for (const entity of entities) {
428
+ for (const userFieldPath of userIdFields) {
429
+ const userId = entity[userFieldPath];
430
+ if (userId) {
431
+ const user = await userRepository.findOne({
432
+ where: { id: userId },
433
+ });
434
+ entity[userFieldPath] = user;
435
+ }
436
+ }
437
+ }
438
+ }
410
439
  async handlePopulateMedia(populateMedia, entities) {
411
440
  const model = await this.entityManager.getRepository(model_metadata_entity_1.ModelMetadata).findOne({
412
441
  where: {
@@ -476,7 +505,7 @@ class CRUDService {
476
505
  const mediaDetails = await storageProvider.retrieve(mediaEntity, mediaFieldMetadata);
477
506
  return mediaDetails;
478
507
  }
479
- async findOne(id, query, solidRequestContext = {}) {
508
+ async findOne(id, query = {}, solidRequestContext = {}) {
480
509
  const { populate = [], fields = [], populateMedia = [] } = query;
481
510
  const normalizedPopulate = this.crudHelperService.normalize(populate);
482
511
  const normalizedPopulateMedia = this.crudHelperService.normalize(populateMedia);
@@ -569,6 +598,15 @@ class CRUDService {
569
598
  });
570
599
  removedEntities.push(entity);
571
600
  }
601
+ const isDraftPublishEnabled = model?.draftPublishWorkflow === true;
602
+ let publishedEntitiesExists = [];
603
+ if (isDraftPublishEnabled) {
604
+ publishedEntitiesExists = removedEntities.filter((x) => !!x?.publishedAt);
605
+ }
606
+ if (publishedEntitiesExists.length > 0) {
607
+ const publishedEntitiesExistsID = publishedEntitiesExists.map(x => x.id);
608
+ throw new common_1.BadRequestException(`Cannot delete published record(s) for model ${this.modelName} with Ids ${publishedEntitiesExistsID.join(', ')}. Unpublish them first.`);
609
+ }
572
610
  if (model.enableSoftDelete === true) {
573
611
  await this.repo.softRemove(removedEntities);
574
612
  return this.repo.save(removedEntities);
@@ -674,6 +712,48 @@ class CRUDService {
674
712
  }
675
713
  return model.userKeyField?.name || '';
676
714
  }
715
+ async publishRecord(id, solidRequestContext = {}) {
716
+ const model = await this.loadModel();
717
+ if (!model.draftPublishWorkflow) {
718
+ throw new common_1.BadRequestException(`Publish workflow is not enabled for ${this.modelName}`);
719
+ }
720
+ if (solidRequestContext.activeUser) {
721
+ const hasPermission = this.crudHelperService.hasPublishPermissionOnModel(solidRequestContext.activeUser, model.singularName);
722
+ if (!hasPermission) {
723
+ throw new common_1.BadRequestException(error_messages_1.ERROR_MESSAGES.FORBIDDEN);
724
+ }
725
+ }
726
+ const entity = await this.repo.findOne({ where: { id } });
727
+ if (!entity) {
728
+ throw new common_1.NotFoundException(`${this.modelName} with id ${id} not found`);
729
+ }
730
+ if (entity.publishedAt) {
731
+ throw new common_1.BadRequestException(`${this.modelName} with id ${id} is already published`);
732
+ }
733
+ const updatedEntity = await this.repo.save({ ...entity, publishedAt: new Date() });
734
+ return updatedEntity;
735
+ }
736
+ async unpublishRecord(id, solidRequestContext = {}) {
737
+ const model = await this.loadModel();
738
+ if (!model.draftPublishWorkflow) {
739
+ throw new common_1.BadRequestException(`Publish workflow is not enabled for ${this.modelName}`);
740
+ }
741
+ if (solidRequestContext.activeUser) {
742
+ const hasPermission = this.crudHelperService.hasUnpublishPermissionOnModel(solidRequestContext.activeUser, model.singularName);
743
+ if (!hasPermission) {
744
+ throw new common_1.BadRequestException(error_messages_1.ERROR_MESSAGES.FORBIDDEN);
745
+ }
746
+ }
747
+ const entity = await this.repo.findOne({ where: { id } });
748
+ if (!entity) {
749
+ throw new common_1.NotFoundException(`${this.modelName} with id ${id} not found`);
750
+ }
751
+ if (!entity.publishedAt) {
752
+ throw new common_1.BadRequestException(`${this.modelName} with id ${id} is already unpublished`);
753
+ }
754
+ const updatedEntity = await this.repo.save({ ...entity, publishedAt: null });
755
+ return updatedEntity;
756
+ }
677
757
  }
678
758
  exports.CRUDService = CRUDService;
679
759
  //# sourceMappingURL=crud.service.js.map
@@ -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, NotFoundException } from \"@nestjs/common\";\nimport { ConfigService } from \"@nestjs/config\";\nimport { DiscoveryService, ModuleRef } from \"@nestjs/core\";\nimport { isArray } from \"class-validator\";\nimport { CommonEntity, SolidBaseRepository } from \"src\";\nimport { ERROR_MESSAGES } from \"src/constants/error-messages\";\nimport { SUCCESS_MESSAGES } from \"src/constants/success-messages\";\nimport { EntityManager, FindOptionsWhere, In, IsNull, Not, QueryFailedError, SelectQueryBuilder } from \"typeorm\";\nimport { QueryDeepPartialEntity } from \"typeorm/query-builder/QueryPartialEntity\";\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 { HashingService } from \"./hashing.service\";\nimport { getMediaStorageProvider } from \"./mediaStorageProviders\";\nimport { ModelMetadataService } from \"./model-metadata.service\";\nimport { ModuleMetadataService } from \"./module-metadata.service\";\nimport { RequestContextService } from \"./request-context.service\";\n\nexport class CRUDService<T extends CommonEntity> { // 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: SolidBaseRepository<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 unknown 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 id: id,\n } as unknown as FindOptionsWhere<T>,\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 id: id,\n } as unknown as FindOptionsWhere<T>,\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> = await this.repo.createSecurityRuleAwareQueryBuilder(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> = await this.repo.createSecurityRuleAwareQueryBuilder(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 id: id,\n } as unknown as FindOptionsWhere<T>,\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 createMany(createDtos: any[], solidRequestContext: any = {}): Promise<T[]> {\n const loadedmodel = await this.loadModel();\n\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 const entitiesForSave: T[] = [];\n for (const createDto of createDtos) {\n // Validate and transform each createDto sequentially\n let transformedDto = createDto;\n for (const field of model.fields) {\n const fieldManager: FieldCrudManager = await this.fieldCrudManager(field, this.entityManager);\n const validationErrors = await fieldManager.validate(createDto, []); // TODO, This is set, because we are not supporting files for insertMany currently\n if (validationErrors.length > 0) {\n throw new BadRequestException(`Validation errors in ${field.name} are invalid: ${validationErrors.map(e => e.error).join(', ')}`);\n }\n transformedDto = await fieldManager.transformForCreate(createDto);\n }\n const entity = this.repo.create(transformedDto);\n entitiesForSave.push(entity as unknown as T);\n }\n // Save all entities in a single batch\n const savedEntities = await this.repo.save(entitiesForSave) as T[];\n return savedEntities;\n }\n\n async insertMany(createDtos: any[], filesArray: Express.Multer.File[][] = [], solidRequestContext: any = {}): Promise<T[]> {\n const savedEntities = await this.createMany(createDtos, solidRequestContext);\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 id: id,\n } as unknown as FindOptionsWhere<T>,\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 id, deletedAt: Not(IsNull())\n } as unknown as FindOptionsWhere<T>,\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 deletedAt: null, deletedTracker: \"not-deleted\"\n } as unknown as QueryDeepPartialEntity<T>\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 id: In(ids),\n deletedAt: Not(IsNull()),\n } as unknown as FindOptionsWhere<T>,\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 { id: In(ids) } as unknown as FindOptionsWhere<T>,\n { deletedAt: null, deletedTracker: \"not-deleted\" } as unknown as QueryDeepPartialEntity<T>\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, NotFoundException } from \"@nestjs/common\";\nimport { ConfigService } from \"@nestjs/config\";\nimport { DiscoveryService, ModuleRef } from \"@nestjs/core\";\nimport { isArray } from \"class-validator\";\nimport { CommonEntity, SolidBaseRepository, User } from \"src\";\nimport { ERROR_MESSAGES } from \"src/constants/error-messages\";\nimport { SUCCESS_MESSAGES } from \"src/constants/success-messages\";\nimport { EntityManager, FindOptionsWhere, In, IsNull, Not, QueryFailedError, SelectQueryBuilder } from \"typeorm\";\nimport { QueryDeepPartialEntity } from \"typeorm/query-builder/QueryPartialEntity\";\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, UserIdFields } from \"./crud-helper.service\";\nimport { FileService } from \"./file.service\";\nimport { HashingService } from \"./hashing.service\";\nimport { getMediaStorageProvider } from \"./mediaStorageProviders\";\nimport { ModelMetadataService } from \"./model-metadata.service\";\nimport { ModuleMetadataService } from \"./module-metadata.service\";\nimport { RequestContextService } from \"./request-context.service\";\n\nexport class CRUDService<T extends CommonEntity> { // 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: SolidBaseRepository<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 unknown 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 id: id,\n } as unknown as FindOptionsWhere<T>,\n });\n if (!entity) {\n throw new Error(`Entity [${this.moduleName}.${this.modelName}] with id ${id} not found`);\n }\n\n if (model.draftPublishWorkflow === true && entity.publishedAt) {\n throw new BadRequestException(`Cannot update a published record for model ${this.modelName}. Unpublish it first.`\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 id: id,\n } as unknown as FindOptionsWhere<T>,\n }\n );\n if (!entity) {\n throw new Error(`Entity [${this.moduleName}.${this.modelName}] with id ${id} not found`);\n }\n\n if (model.draftPublishWorkflow === true && entity.publishedAt) {\n throw new BadRequestException(`Cannot update a published record for model ${this.modelName}, Unpublish it first.`);\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 populateUserIdFields = this.crudHelperService.extractUserIdFieldsFromPopulate(basicFilterDto.populate);\n\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> = await this.repo.createSecurityRuleAwareQueryBuilder(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 else {\n qb = this.crudHelperService.buildFilterQuery(qb, basicFilterDto, alias);\n }\n\n if (basicFilterDto.groupBy) {\n // Get the records and the count\n const { groupMeta, groupRecords } = await this.handleGroupFind(qb, groupFilter, populateGroup, alias, populateUserIdFields, 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, populateUserIdFields, populateMedia, offset, limit, alias);\n return {\n meta,\n records,\n }\n }\n }\n\n private async handleNonGroupFind(qb: SelectQueryBuilder<T>, populateUserIdFields: UserIdFields[], populateMedia: string[], offset: number, limit: number, alias: string) {\n const [entities, count] = await qb.getManyAndCount();\n\n // Populate the entity with the userId fields\n if (populateUserIdFields && populateUserIdFields.length > 0) {\n await this.handlePopulateUserIdFields(populateUserIdFields, entities);\n }\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, populateUserIdFields: UserIdFields[], 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> = await this.repo.createSecurityRuleAwareQueryBuilder(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 userId fields\n if (populateUserIdFields && populateUserIdFields.length > 0) {\n await this.handlePopulateUserIdFields(populateUserIdFields, entities);\n }\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 // entities is an array of T\n // T can contain createdBy and updatedBy fields\n // We need to populate the createdBy and updatedBy fields with the User entity\n private async handlePopulateUserIdFields(userIdFields: UserIdFields[], entities: T[]) {\n const userRepository = this.entityManager.getRepository(User);\n for (const entity of entities) {\n for (const userFieldPath of userIdFields) {\n const userId = entity[userFieldPath as keyof T] as unknown as number;\n if (userId) {\n const user = await userRepository.findOne({\n where: { id: userId },\n });\n // @ts-ignore\n entity[userFieldPath] = user;\n }\n }\n }\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 id: id,\n } as unknown as FindOptionsWhere<T>,\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 createMany(createDtos: any[], solidRequestContext: any = {}): Promise<T[]> {\n const loadedmodel = await this.loadModel();\n\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 const entitiesForSave: T[] = [];\n for (const createDto of createDtos) {\n // Validate and transform each createDto sequentially\n let transformedDto = createDto;\n for (const field of model.fields) {\n const fieldManager: FieldCrudManager = await this.fieldCrudManager(field, this.entityManager);\n const validationErrors = await fieldManager.validate(createDto, []); // TODO, This is set, because we are not supporting files for insertMany currently\n if (validationErrors.length > 0) {\n throw new BadRequestException(`Validation errors in ${field.name} are invalid: ${validationErrors.map(e => e.error).join(', ')}`);\n }\n transformedDto = await fieldManager.transformForCreate(createDto);\n }\n const entity = this.repo.create(transformedDto);\n entitiesForSave.push(entity as unknown as T);\n }\n // Save all entities in a single batch\n const savedEntities = await this.repo.save(entitiesForSave) as T[];\n return savedEntities;\n }\n\n async insertMany(createDtos: any[], filesArray: Express.Multer.File[][] = [], solidRequestContext: any = {}): Promise<T[]> {\n const savedEntities = await this.createMany(createDtos, solidRequestContext);\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 id: id,\n } as unknown as FindOptionsWhere<T>,\n });\n\n removedEntities.push(entity);\n }\n\n\n // entity-level flag\n const isDraftPublishEnabled = model?.draftPublishWorkflow === true;\n\n let publishedEntitiesExists: T[] = [];\n\n if (isDraftPublishEnabled) {\n publishedEntitiesExists = removedEntities.filter(\n (x) => !!x?.publishedAt\n );\n }\n\n if (publishedEntitiesExists.length > 0) {\n const publishedEntitiesExistsID = publishedEntitiesExists.map(x => x.id);\n\n throw new BadRequestException(\n `Cannot delete published record(s) for model ${this.modelName} with Ids ${publishedEntitiesExistsID.join(', ')}. Unpublish them first.`\n );\n }\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 id, deletedAt: Not(IsNull())\n } as unknown as FindOptionsWhere<T>,\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 deletedAt: null, deletedTracker: \"not-deleted\"\n } as unknown as QueryDeepPartialEntity<T>\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 id: In(ids),\n deletedAt: Not(IsNull()),\n } as unknown as FindOptionsWhere<T>,\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 { id: In(ids) } as unknown as FindOptionsWhere<T>,\n { deletedAt: null, deletedTracker: \"not-deleted\" } as unknown as QueryDeepPartialEntity<T>\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 /* Publish a record - sets publishedAt timestamp */\n async publishRecord(id: number, solidRequestContext: any = {}): Promise<T> {\n\n const model = await this.loadModel();\n\n // Check if publish workflow is enabled for this model\n if (!model.draftPublishWorkflow) {\n throw new BadRequestException(\n `Publish workflow is not enabled for ${this.modelName}`\n );\n }\n\n // Check user permissions\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasPublishPermissionOnModel(\n solidRequestContext.activeUser,\n model.singularName\n );\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n\n // Find the entity\n const entity = await this.repo.findOne({ where: { id } as any });\n if (!entity) {\n throw new NotFoundException(`${this.modelName} with id ${id} not found`);\n }\n\n // Check if already published\n if (entity.publishedAt) {\n throw new BadRequestException(\n `${this.modelName} with id ${id} is already published`\n );\n }\n\n // Update publish status\n const updatedEntity = await this.repo.save({ ...entity, publishedAt: new Date() });\n\n return updatedEntity\n }\n\n /* Unpublish a record - clears publishedAt timestamp */\n async unpublishRecord(id: number, solidRequestContext: any = {}): Promise<T> {\n\n const model = await this.loadModel();\n\n // Check if publish workflow is enabled for this model\n if (!model.draftPublishWorkflow) {\n throw new BadRequestException(\n `Publish workflow is not enabled for ${this.modelName}`\n );\n }\n\n // Check user permissions\n if (solidRequestContext.activeUser) {\n const hasPermission = this.crudHelperService.hasUnpublishPermissionOnModel(\n solidRequestContext.activeUser,\n model.singularName\n );\n if (!hasPermission) {\n throw new BadRequestException(ERROR_MESSAGES.FORBIDDEN);\n }\n }\n\n // Find the entity\n const entity = await this.repo.findOne({ where: { id } as any });\n if (!entity) {\n throw new NotFoundException(`${this.modelName} with id ${id} not found`);\n }\n\n // Check if already unpublished\n if (!entity.publishedAt) {\n throw new BadRequestException(\n `${this.modelName} with id ${id} is already unpublished`\n );\n }\n\n // Update unpublish status\n const updatedEntity = await this.repo.save({ ...entity, publishedAt: null });\n\n return updatedEntity\n }\n}\n\n"]}