@strapi/core 0.0.0-experimental.c5235059f5636c4549ea2118c75c43b92e2615c8 → 0.0.0-experimental.c72b48da735f44a3ccafb7a6dc9ba5213f3844db

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 (736) hide show
  1. package/dist/Strapi.d.ts.map +1 -1
  2. package/dist/Strapi.js +427 -403
  3. package/dist/Strapi.js.map +1 -1
  4. package/dist/Strapi.mjs +426 -402
  5. package/dist/Strapi.mjs.map +1 -1
  6. package/dist/compile.js +23 -14
  7. package/dist/compile.js.map +1 -1
  8. package/dist/compile.mjs +22 -14
  9. package/dist/compile.mjs.map +1 -1
  10. package/dist/configuration/config-loader.js +110 -97
  11. package/dist/configuration/config-loader.js.map +1 -1
  12. package/dist/configuration/config-loader.mjs +109 -96
  13. package/dist/configuration/config-loader.mjs.map +1 -1
  14. package/dist/configuration/get-dirs.js +31 -29
  15. package/dist/configuration/get-dirs.js.map +1 -1
  16. package/dist/configuration/get-dirs.mjs +30 -30
  17. package/dist/configuration/get-dirs.mjs.map +1 -1
  18. package/dist/configuration/index.d.ts +1 -0
  19. package/dist/configuration/index.d.ts.map +1 -1
  20. package/dist/configuration/index.js +85 -73
  21. package/dist/configuration/index.js.map +1 -1
  22. package/dist/configuration/index.mjs +84 -69
  23. package/dist/configuration/index.mjs.map +1 -1
  24. package/dist/configuration/urls.d.ts.map +1 -1
  25. package/dist/configuration/urls.js +74 -61
  26. package/dist/configuration/urls.js.map +1 -1
  27. package/dist/configuration/urls.mjs +73 -62
  28. package/dist/configuration/urls.mjs.map +1 -1
  29. package/dist/container.js +27 -23
  30. package/dist/container.js.map +1 -1
  31. package/dist/container.mjs +26 -24
  32. package/dist/container.mjs.map +1 -1
  33. package/dist/core-api/controller/collection-type.js +77 -76
  34. package/dist/core-api/controller/collection-type.js.map +1 -1
  35. package/dist/core-api/controller/collection-type.mjs +76 -77
  36. package/dist/core-api/controller/collection-type.mjs.map +1 -1
  37. package/dist/core-api/controller/index.js +63 -48
  38. package/dist/core-api/controller/index.js.map +1 -1
  39. package/dist/core-api/controller/index.mjs +62 -49
  40. package/dist/core-api/controller/index.mjs.map +1 -1
  41. package/dist/core-api/controller/single-type.js +41 -40
  42. package/dist/core-api/controller/single-type.js.map +1 -1
  43. package/dist/core-api/controller/single-type.mjs +40 -41
  44. package/dist/core-api/controller/single-type.mjs.map +1 -1
  45. package/dist/core-api/controller/transform.js +72 -63
  46. package/dist/core-api/controller/transform.js.map +1 -1
  47. package/dist/core-api/controller/transform.mjs +71 -64
  48. package/dist/core-api/controller/transform.mjs.map +1 -1
  49. package/dist/core-api/routes/index.js +64 -62
  50. package/dist/core-api/routes/index.js.map +1 -1
  51. package/dist/core-api/routes/index.mjs +62 -62
  52. package/dist/core-api/routes/index.mjs.map +1 -1
  53. package/dist/core-api/service/collection-type.d.ts +2 -2
  54. package/dist/core-api/service/collection-type.js +73 -60
  55. package/dist/core-api/service/collection-type.js.map +1 -1
  56. package/dist/core-api/service/collection-type.mjs +72 -62
  57. package/dist/core-api/service/collection-type.mjs.map +1 -1
  58. package/dist/core-api/service/core-service.js +9 -8
  59. package/dist/core-api/service/core-service.js.map +1 -1
  60. package/dist/core-api/service/core-service.mjs +8 -9
  61. package/dist/core-api/service/core-service.mjs.map +1 -1
  62. package/dist/core-api/service/index.js +13 -13
  63. package/dist/core-api/service/index.js.map +1 -1
  64. package/dist/core-api/service/index.mjs +12 -14
  65. package/dist/core-api/service/index.mjs.map +1 -1
  66. package/dist/core-api/service/pagination.js +69 -43
  67. package/dist/core-api/service/pagination.js.map +1 -1
  68. package/dist/core-api/service/pagination.mjs +68 -47
  69. package/dist/core-api/service/pagination.mjs.map +1 -1
  70. package/dist/core-api/service/single-type.js +43 -39
  71. package/dist/core-api/service/single-type.js.map +1 -1
  72. package/dist/core-api/service/single-type.mjs +42 -41
  73. package/dist/core-api/service/single-type.mjs.map +1 -1
  74. package/dist/domain/content-type/index.js +95 -100
  75. package/dist/domain/content-type/index.js.map +1 -1
  76. package/dist/domain/content-type/index.mjs +94 -100
  77. package/dist/domain/content-type/index.mjs.map +1 -1
  78. package/dist/domain/content-type/validator.js +84 -69
  79. package/dist/domain/content-type/validator.js.map +1 -1
  80. package/dist/domain/content-type/validator.mjs +83 -68
  81. package/dist/domain/content-type/validator.mjs.map +1 -1
  82. package/dist/domain/module/index.js +109 -100
  83. package/dist/domain/module/index.js.map +1 -1
  84. package/dist/domain/module/index.mjs +108 -99
  85. package/dist/domain/module/index.mjs.map +1 -1
  86. package/dist/domain/module/validation.js +25 -20
  87. package/dist/domain/module/validation.js.map +1 -1
  88. package/dist/domain/module/validation.mjs +24 -21
  89. package/dist/domain/module/validation.mjs.map +1 -1
  90. package/dist/ee/index.d.ts +1 -0
  91. package/dist/ee/index.d.ts.map +1 -1
  92. package/dist/ee/index.js +171 -132
  93. package/dist/ee/index.js.map +1 -1
  94. package/dist/ee/index.mjs +170 -134
  95. package/dist/ee/index.mjs.map +1 -1
  96. package/dist/ee/license.d.ts.map +1 -1
  97. package/dist/ee/license.js +100 -75
  98. package/dist/ee/license.js.map +1 -1
  99. package/dist/ee/license.mjs +99 -76
  100. package/dist/ee/license.mjs.map +1 -1
  101. package/dist/factories.js +72 -67
  102. package/dist/factories.js.map +1 -1
  103. package/dist/factories.mjs +71 -71
  104. package/dist/factories.mjs.map +1 -1
  105. package/dist/index.js +29 -26
  106. package/dist/index.js.map +1 -1
  107. package/dist/index.mjs +29 -29
  108. package/dist/index.mjs.map +1 -1
  109. package/dist/loaders/admin.d.ts.map +1 -1
  110. package/dist/loaders/admin.js +21 -20
  111. package/dist/loaders/admin.js.map +1 -1
  112. package/dist/loaders/admin.mjs +20 -20
  113. package/dist/loaders/admin.mjs.map +1 -1
  114. package/dist/loaders/apis.js +143 -120
  115. package/dist/loaders/apis.js.map +1 -1
  116. package/dist/loaders/apis.mjs +142 -119
  117. package/dist/loaders/apis.mjs.map +1 -1
  118. package/dist/loaders/components.js +33 -34
  119. package/dist/loaders/components.js.map +1 -1
  120. package/dist/loaders/components.mjs +32 -34
  121. package/dist/loaders/components.mjs.map +1 -1
  122. package/dist/loaders/index.js +22 -20
  123. package/dist/loaders/index.js.map +1 -1
  124. package/dist/loaders/index.mjs +21 -21
  125. package/dist/loaders/index.mjs.map +1 -1
  126. package/dist/loaders/middlewares.js +29 -25
  127. package/dist/loaders/middlewares.js.map +1 -1
  128. package/dist/loaders/middlewares.mjs +28 -25
  129. package/dist/loaders/middlewares.mjs.map +1 -1
  130. package/dist/loaders/plugins/get-enabled-plugins.d.ts.map +1 -1
  131. package/dist/loaders/plugins/get-enabled-plugins.js +125 -102
  132. package/dist/loaders/plugins/get-enabled-plugins.js.map +1 -1
  133. package/dist/loaders/plugins/get-enabled-plugins.mjs +124 -101
  134. package/dist/loaders/plugins/get-enabled-plugins.mjs.map +1 -1
  135. package/dist/loaders/plugins/get-user-plugins-config.js +25 -24
  136. package/dist/loaders/plugins/get-user-plugins-config.js.map +1 -1
  137. package/dist/loaders/plugins/get-user-plugins-config.mjs +24 -23
  138. package/dist/loaders/plugins/get-user-plugins-config.mjs.map +1 -1
  139. package/dist/loaders/plugins/index.d.ts.map +1 -1
  140. package/dist/loaders/plugins/index.js +141 -104
  141. package/dist/loaders/plugins/index.js.map +1 -1
  142. package/dist/loaders/plugins/index.mjs +121 -104
  143. package/dist/loaders/plugins/index.mjs.map +1 -1
  144. package/dist/loaders/policies.js +24 -20
  145. package/dist/loaders/policies.js.map +1 -1
  146. package/dist/loaders/policies.mjs +23 -20
  147. package/dist/loaders/policies.mjs.map +1 -1
  148. package/dist/loaders/sanitizers.js +10 -4
  149. package/dist/loaders/sanitizers.js.map +1 -1
  150. package/dist/loaders/sanitizers.mjs +9 -6
  151. package/dist/loaders/sanitizers.mjs.map +1 -1
  152. package/dist/loaders/src-index.js +35 -27
  153. package/dist/loaders/src-index.js.map +1 -1
  154. package/dist/loaders/src-index.mjs +34 -29
  155. package/dist/loaders/src-index.mjs.map +1 -1
  156. package/dist/loaders/validators.js +9 -4
  157. package/dist/loaders/validators.js.map +1 -1
  158. package/dist/loaders/validators.mjs +8 -6
  159. package/dist/loaders/validators.mjs.map +1 -1
  160. package/dist/middlewares/body.d.ts.map +1 -1
  161. package/dist/middlewares/body.js +58 -54
  162. package/dist/middlewares/body.js.map +1 -1
  163. package/dist/middlewares/body.mjs +57 -51
  164. package/dist/middlewares/body.mjs.map +1 -1
  165. package/dist/middlewares/compression.js +6 -6
  166. package/dist/middlewares/compression.js.map +1 -1
  167. package/dist/middlewares/compression.mjs +5 -5
  168. package/dist/middlewares/compression.mjs.map +1 -1
  169. package/dist/middlewares/cors.js +59 -48
  170. package/dist/middlewares/cors.js.map +1 -1
  171. package/dist/middlewares/cors.mjs +58 -47
  172. package/dist/middlewares/cors.mjs.map +1 -1
  173. package/dist/middlewares/errors.js +32 -30
  174. package/dist/middlewares/errors.js.map +1 -1
  175. package/dist/middlewares/errors.mjs +31 -31
  176. package/dist/middlewares/errors.mjs.map +1 -1
  177. package/dist/middlewares/favicon.js +27 -17
  178. package/dist/middlewares/favicon.js.map +1 -1
  179. package/dist/middlewares/favicon.mjs +26 -16
  180. package/dist/middlewares/favicon.mjs.map +1 -1
  181. package/dist/middlewares/index.js +32 -30
  182. package/dist/middlewares/index.js.map +1 -1
  183. package/dist/middlewares/index.mjs +31 -31
  184. package/dist/middlewares/index.mjs.map +1 -1
  185. package/dist/middlewares/ip.js +6 -6
  186. package/dist/middlewares/ip.js.map +1 -1
  187. package/dist/middlewares/ip.mjs +5 -5
  188. package/dist/middlewares/ip.mjs.map +1 -1
  189. package/dist/middlewares/logger.js +10 -9
  190. package/dist/middlewares/logger.js.map +1 -1
  191. package/dist/middlewares/logger.mjs +9 -10
  192. package/dist/middlewares/logger.mjs.map +1 -1
  193. package/dist/middlewares/powered-by.js +13 -9
  194. package/dist/middlewares/powered-by.js.map +1 -1
  195. package/dist/middlewares/powered-by.mjs +12 -10
  196. package/dist/middlewares/powered-by.mjs.map +1 -1
  197. package/dist/middlewares/public.js +33 -29
  198. package/dist/middlewares/public.js.map +1 -1
  199. package/dist/middlewares/public.mjs +32 -28
  200. package/dist/middlewares/public.mjs.map +1 -1
  201. package/dist/middlewares/query.js +35 -32
  202. package/dist/middlewares/query.js.map +1 -1
  203. package/dist/middlewares/query.mjs +34 -31
  204. package/dist/middlewares/query.mjs.map +1 -1
  205. package/dist/middlewares/response-time.js +10 -9
  206. package/dist/middlewares/response-time.js.map +1 -1
  207. package/dist/middlewares/response-time.mjs +9 -10
  208. package/dist/middlewares/response-time.mjs.map +1 -1
  209. package/dist/middlewares/responses.js +14 -12
  210. package/dist/middlewares/responses.js.map +1 -1
  211. package/dist/middlewares/responses.mjs +13 -13
  212. package/dist/middlewares/responses.mjs.map +1 -1
  213. package/dist/middlewares/security.d.ts.map +1 -1
  214. package/dist/middlewares/security.js +109 -71
  215. package/dist/middlewares/security.js.map +1 -1
  216. package/dist/middlewares/security.mjs +108 -70
  217. package/dist/middlewares/security.mjs.map +1 -1
  218. package/dist/middlewares/session.js +26 -25
  219. package/dist/middlewares/session.js.map +1 -1
  220. package/dist/middlewares/session.mjs +25 -24
  221. package/dist/middlewares/session.mjs.map +1 -1
  222. package/dist/migrations/database/5.0.0-discard-drafts.d.ts +14 -11
  223. package/dist/migrations/database/5.0.0-discard-drafts.d.ts.map +1 -1
  224. package/dist/migrations/database/5.0.0-discard-drafts.js +153 -35
  225. package/dist/migrations/database/5.0.0-discard-drafts.js.map +1 -1
  226. package/dist/migrations/database/5.0.0-discard-drafts.mjs +152 -37
  227. package/dist/migrations/database/5.0.0-discard-drafts.mjs.map +1 -1
  228. package/dist/migrations/draft-publish.d.ts +2 -2
  229. package/dist/migrations/draft-publish.d.ts.map +1 -1
  230. package/dist/migrations/draft-publish.js +61 -34
  231. package/dist/migrations/draft-publish.js.map +1 -1
  232. package/dist/migrations/draft-publish.mjs +60 -36
  233. package/dist/migrations/draft-publish.mjs.map +1 -1
  234. package/dist/migrations/i18n.d.ts +5 -0
  235. package/dist/migrations/i18n.d.ts.map +1 -0
  236. package/dist/migrations/i18n.js +71 -0
  237. package/dist/migrations/i18n.js.map +1 -0
  238. package/dist/migrations/i18n.mjs +68 -0
  239. package/dist/migrations/i18n.mjs.map +1 -0
  240. package/dist/migrations/index.d.ts +5 -0
  241. package/dist/migrations/index.d.ts.map +1 -0
  242. package/dist/migrations/index.js +29 -0
  243. package/dist/migrations/index.js.map +1 -0
  244. package/dist/migrations/index.mjs +26 -0
  245. package/dist/migrations/index.mjs.map +1 -0
  246. package/dist/package.json.js +182 -0
  247. package/dist/package.json.js.map +1 -0
  248. package/dist/package.json.mjs +159 -0
  249. package/dist/package.json.mjs.map +1 -0
  250. package/dist/providers/admin.d.ts.map +1 -1
  251. package/dist/providers/admin.js +27 -17
  252. package/dist/providers/admin.js.map +1 -1
  253. package/dist/providers/admin.mjs +26 -19
  254. package/dist/providers/admin.mjs.map +1 -1
  255. package/dist/providers/coreStore.js +13 -8
  256. package/dist/providers/coreStore.js.map +1 -1
  257. package/dist/providers/coreStore.mjs +12 -10
  258. package/dist/providers/coreStore.mjs.map +1 -1
  259. package/dist/providers/cron.js +19 -16
  260. package/dist/providers/cron.js.map +1 -1
  261. package/dist/providers/cron.mjs +18 -18
  262. package/dist/providers/cron.mjs.map +1 -1
  263. package/dist/providers/index.js +18 -9
  264. package/dist/providers/index.js.map +1 -1
  265. package/dist/providers/index.mjs +17 -10
  266. package/dist/providers/index.mjs.map +1 -1
  267. package/dist/providers/provider.js +4 -3
  268. package/dist/providers/provider.js.map +1 -1
  269. package/dist/providers/provider.mjs +3 -4
  270. package/dist/providers/provider.mjs.map +1 -1
  271. package/dist/providers/registries.js +37 -32
  272. package/dist/providers/registries.js.map +1 -1
  273. package/dist/providers/registries.mjs +36 -34
  274. package/dist/providers/registries.mjs.map +1 -1
  275. package/dist/providers/telemetry.js +19 -16
  276. package/dist/providers/telemetry.js.map +1 -1
  277. package/dist/providers/telemetry.mjs +18 -18
  278. package/dist/providers/telemetry.mjs.map +1 -1
  279. package/dist/providers/webhooks.js +28 -26
  280. package/dist/providers/webhooks.js.map +1 -1
  281. package/dist/providers/webhooks.mjs +27 -28
  282. package/dist/providers/webhooks.mjs.map +1 -1
  283. package/dist/registries/apis.js +23 -20
  284. package/dist/registries/apis.js.map +1 -1
  285. package/dist/registries/apis.mjs +22 -22
  286. package/dist/registries/apis.mjs.map +1 -1
  287. package/dist/registries/components.js +35 -37
  288. package/dist/registries/components.js.map +1 -1
  289. package/dist/registries/components.mjs +34 -39
  290. package/dist/registries/components.mjs.map +1 -1
  291. package/dist/registries/content-types.js +54 -59
  292. package/dist/registries/content-types.js.map +1 -1
  293. package/dist/registries/content-types.mjs +53 -61
  294. package/dist/registries/content-types.mjs.map +1 -1
  295. package/dist/registries/controllers.js +70 -71
  296. package/dist/registries/controllers.js.map +1 -1
  297. package/dist/registries/controllers.mjs +69 -73
  298. package/dist/registries/controllers.mjs.map +1 -1
  299. package/dist/registries/custom-fields.js +75 -65
  300. package/dist/registries/custom-fields.js.map +1 -1
  301. package/dist/registries/custom-fields.mjs +74 -67
  302. package/dist/registries/custom-fields.mjs.map +1 -1
  303. package/dist/registries/hooks.js +46 -49
  304. package/dist/registries/hooks.js.map +1 -1
  305. package/dist/registries/hooks.mjs +45 -51
  306. package/dist/registries/hooks.mjs.map +1 -1
  307. package/dist/registries/middlewares.js +49 -51
  308. package/dist/registries/middlewares.js.map +1 -1
  309. package/dist/registries/middlewares.mjs +48 -53
  310. package/dist/registries/middlewares.mjs.map +1 -1
  311. package/dist/registries/models.js +14 -13
  312. package/dist/registries/models.js.map +1 -1
  313. package/dist/registries/models.mjs +13 -14
  314. package/dist/registries/models.mjs.map +1 -1
  315. package/dist/registries/modules.js +39 -36
  316. package/dist/registries/modules.js.map +1 -1
  317. package/dist/registries/modules.mjs +38 -38
  318. package/dist/registries/modules.mjs.map +1 -1
  319. package/dist/registries/namespace.js +21 -20
  320. package/dist/registries/namespace.js.map +1 -1
  321. package/dist/registries/namespace.mjs +20 -23
  322. package/dist/registries/namespace.mjs.map +1 -1
  323. package/dist/registries/plugins.js +23 -20
  324. package/dist/registries/plugins.js.map +1 -1
  325. package/dist/registries/plugins.mjs +22 -22
  326. package/dist/registries/plugins.mjs.map +1 -1
  327. package/dist/registries/policies.d.ts +1 -1
  328. package/dist/registries/policies.d.ts.map +1 -1
  329. package/dist/registries/policies.js +103 -96
  330. package/dist/registries/policies.js.map +1 -1
  331. package/dist/registries/policies.mjs +102 -98
  332. package/dist/registries/policies.mjs.map +1 -1
  333. package/dist/registries/sanitizers.js +23 -22
  334. package/dist/registries/sanitizers.js.map +1 -1
  335. package/dist/registries/sanitizers.mjs +22 -22
  336. package/dist/registries/sanitizers.mjs.map +1 -1
  337. package/dist/registries/services.js +71 -71
  338. package/dist/registries/services.js.map +1 -1
  339. package/dist/registries/services.mjs +70 -73
  340. package/dist/registries/services.mjs.map +1 -1
  341. package/dist/registries/validators.js +23 -22
  342. package/dist/registries/validators.js.map +1 -1
  343. package/dist/registries/validators.mjs +22 -22
  344. package/dist/registries/validators.mjs.map +1 -1
  345. package/dist/services/auth/index.js +74 -74
  346. package/dist/services/auth/index.js.map +1 -1
  347. package/dist/services/auth/index.mjs +73 -74
  348. package/dist/services/auth/index.mjs.map +1 -1
  349. package/dist/services/config.js +47 -43
  350. package/dist/services/config.js.map +1 -1
  351. package/dist/services/config.mjs +46 -44
  352. package/dist/services/config.mjs.map +1 -1
  353. package/dist/services/content-api/index.d.ts +10 -12
  354. package/dist/services/content-api/index.d.ts.map +1 -1
  355. package/dist/services/content-api/index.js +80 -79
  356. package/dist/services/content-api/index.js.map +1 -1
  357. package/dist/services/content-api/index.mjs +79 -79
  358. package/dist/services/content-api/index.mjs.map +1 -1
  359. package/dist/services/content-api/permissions/engine.js +8 -5
  360. package/dist/services/content-api/permissions/engine.js.map +1 -1
  361. package/dist/services/content-api/permissions/engine.mjs +7 -5
  362. package/dist/services/content-api/permissions/engine.mjs.map +1 -1
  363. package/dist/services/content-api/permissions/index.d.ts +10 -12
  364. package/dist/services/content-api/permissions/index.d.ts.map +1 -1
  365. package/dist/services/content-api/permissions/index.js +101 -81
  366. package/dist/services/content-api/permissions/index.js.map +1 -1
  367. package/dist/services/content-api/permissions/index.mjs +100 -81
  368. package/dist/services/content-api/permissions/index.mjs.map +1 -1
  369. package/dist/services/content-api/permissions/providers/action.d.ts +5 -6
  370. package/dist/services/content-api/permissions/providers/action.d.ts.map +1 -1
  371. package/dist/services/content-api/permissions/providers/action.js +17 -14
  372. package/dist/services/content-api/permissions/providers/action.js.map +1 -1
  373. package/dist/services/content-api/permissions/providers/action.mjs +16 -16
  374. package/dist/services/content-api/permissions/providers/action.mjs.map +1 -1
  375. package/dist/services/content-api/permissions/providers/condition.d.ts +5 -6
  376. package/dist/services/content-api/permissions/providers/condition.d.ts.map +1 -1
  377. package/dist/services/content-api/permissions/providers/condition.js +17 -14
  378. package/dist/services/content-api/permissions/providers/condition.js.map +1 -1
  379. package/dist/services/content-api/permissions/providers/condition.mjs +16 -16
  380. package/dist/services/content-api/permissions/providers/condition.mjs.map +1 -1
  381. package/dist/services/core-store.js +115 -95
  382. package/dist/services/core-store.js.map +1 -1
  383. package/dist/services/core-store.mjs +114 -97
  384. package/dist/services/core-store.mjs.map +1 -1
  385. package/dist/services/cron.d.ts +3 -3
  386. package/dist/services/cron.d.ts.map +1 -1
  387. package/dist/services/cron.js +74 -60
  388. package/dist/services/cron.js.map +1 -1
  389. package/dist/services/cron.mjs +73 -62
  390. package/dist/services/cron.mjs.map +1 -1
  391. package/dist/services/custom-fields.js +9 -7
  392. package/dist/services/custom-fields.js.map +1 -1
  393. package/dist/services/custom-fields.mjs +8 -9
  394. package/dist/services/custom-fields.mjs.map +1 -1
  395. package/dist/services/document-service/attributes/index.js +23 -18
  396. package/dist/services/document-service/attributes/index.js.map +1 -1
  397. package/dist/services/document-service/attributes/index.mjs +22 -19
  398. package/dist/services/document-service/attributes/index.mjs.map +1 -1
  399. package/dist/services/document-service/attributes/transforms.js +16 -15
  400. package/dist/services/document-service/attributes/transforms.js.map +1 -1
  401. package/dist/services/document-service/attributes/transforms.mjs +15 -15
  402. package/dist/services/document-service/attributes/transforms.mjs.map +1 -1
  403. package/dist/services/document-service/common.d.ts +1 -1
  404. package/dist/services/document-service/common.d.ts.map +1 -1
  405. package/dist/services/document-service/common.js +5 -4
  406. package/dist/services/document-service/common.js.map +1 -1
  407. package/dist/services/document-service/common.mjs +4 -5
  408. package/dist/services/document-service/common.mjs.map +1 -1
  409. package/dist/services/document-service/components.d.ts.map +1 -1
  410. package/dist/services/document-service/components.js +255 -257
  411. package/dist/services/document-service/components.js.map +1 -1
  412. package/dist/services/document-service/components.mjs +254 -262
  413. package/dist/services/document-service/components.mjs.map +1 -1
  414. package/dist/services/document-service/draft-and-publish.d.ts +1 -1
  415. package/dist/services/document-service/draft-and-publish.d.ts.map +1 -1
  416. package/dist/services/document-service/draft-and-publish.js +88 -48
  417. package/dist/services/document-service/draft-and-publish.js.map +1 -1
  418. package/dist/services/document-service/draft-and-publish.mjs +87 -54
  419. package/dist/services/document-service/draft-and-publish.mjs.map +1 -1
  420. package/dist/services/document-service/entries.d.ts +2 -2
  421. package/dist/services/document-service/entries.d.ts.map +1 -1
  422. package/dist/services/document-service/entries.js +109 -92
  423. package/dist/services/document-service/entries.js.map +1 -1
  424. package/dist/services/document-service/entries.mjs +108 -93
  425. package/dist/services/document-service/entries.mjs.map +1 -1
  426. package/dist/services/document-service/events.d.ts +1 -1
  427. package/dist/services/document-service/events.d.ts.map +1 -1
  428. package/dist/services/document-service/events.js +52 -40
  429. package/dist/services/document-service/events.js.map +1 -1
  430. package/dist/services/document-service/events.mjs +51 -41
  431. package/dist/services/document-service/events.mjs.map +1 -1
  432. package/dist/services/document-service/index.d.ts +2 -1
  433. package/dist/services/document-service/index.d.ts.map +1 -1
  434. package/dist/services/document-service/index.js +53 -32
  435. package/dist/services/document-service/index.js.map +1 -1
  436. package/dist/services/document-service/index.mjs +52 -33
  437. package/dist/services/document-service/index.mjs.map +1 -1
  438. package/dist/services/document-service/internationalization.js +62 -46
  439. package/dist/services/document-service/internationalization.js.map +1 -1
  440. package/dist/services/document-service/internationalization.mjs +61 -50
  441. package/dist/services/document-service/internationalization.mjs.map +1 -1
  442. package/dist/services/document-service/middlewares/errors.js +23 -19
  443. package/dist/services/document-service/middlewares/errors.js.map +1 -1
  444. package/dist/services/document-service/middlewares/errors.mjs +22 -20
  445. package/dist/services/document-service/middlewares/errors.mjs.map +1 -1
  446. package/dist/services/document-service/middlewares/middleware-manager.js +46 -44
  447. package/dist/services/document-service/middlewares/middleware-manager.js.map +1 -1
  448. package/dist/services/document-service/middlewares/middleware-manager.mjs +45 -45
  449. package/dist/services/document-service/middlewares/middleware-manager.mjs.map +1 -1
  450. package/dist/services/document-service/params.js +11 -5
  451. package/dist/services/document-service/params.js.map +1 -1
  452. package/dist/services/document-service/params.mjs +10 -6
  453. package/dist/services/document-service/params.mjs.map +1 -1
  454. package/dist/services/document-service/repository.d.ts.map +1 -1
  455. package/dist/services/document-service/repository.js +355 -269
  456. package/dist/services/document-service/repository.js.map +1 -1
  457. package/dist/services/document-service/repository.mjs +354 -270
  458. package/dist/services/document-service/repository.mjs.map +1 -1
  459. package/dist/services/document-service/transform/data.js +22 -12
  460. package/dist/services/document-service/transform/data.js.map +1 -1
  461. package/dist/services/document-service/transform/data.mjs +21 -13
  462. package/dist/services/document-service/transform/data.mjs.map +1 -1
  463. package/dist/services/document-service/transform/fields.js +26 -17
  464. package/dist/services/document-service/transform/fields.js.map +1 -1
  465. package/dist/services/document-service/transform/fields.mjs +25 -18
  466. package/dist/services/document-service/transform/fields.mjs.map +1 -1
  467. package/dist/services/document-service/transform/id-map.d.ts +1 -1
  468. package/dist/services/document-service/transform/id-map.d.ts.map +1 -1
  469. package/dist/services/document-service/transform/id-map.js +116 -69
  470. package/dist/services/document-service/transform/id-map.js.map +1 -1
  471. package/dist/services/document-service/transform/id-map.mjs +115 -70
  472. package/dist/services/document-service/transform/id-map.mjs.map +1 -1
  473. package/dist/services/document-service/transform/id-transform.d.ts +1 -1
  474. package/dist/services/document-service/transform/id-transform.d.ts.map +1 -1
  475. package/dist/services/document-service/transform/id-transform.js +37 -29
  476. package/dist/services/document-service/transform/id-transform.js.map +1 -1
  477. package/dist/services/document-service/transform/id-transform.mjs +36 -30
  478. package/dist/services/document-service/transform/id-transform.mjs.map +1 -1
  479. package/dist/services/document-service/transform/populate.js +23 -18
  480. package/dist/services/document-service/transform/populate.js.map +1 -1
  481. package/dist/services/document-service/transform/populate.mjs +22 -19
  482. package/dist/services/document-service/transform/populate.mjs.map +1 -1
  483. package/dist/services/document-service/transform/query.js +11 -6
  484. package/dist/services/document-service/transform/query.js.map +1 -1
  485. package/dist/services/document-service/transform/query.mjs +10 -7
  486. package/dist/services/document-service/transform/query.mjs.map +1 -1
  487. package/dist/services/document-service/transform/relations/extract/data-ids.d.ts +1 -1
  488. package/dist/services/document-service/transform/relations/extract/data-ids.d.ts.map +1 -1
  489. package/dist/services/document-service/transform/relations/extract/data-ids.js +71 -48
  490. package/dist/services/document-service/transform/relations/extract/data-ids.js.map +1 -1
  491. package/dist/services/document-service/transform/relations/extract/data-ids.mjs +70 -49
  492. package/dist/services/document-service/transform/relations/extract/data-ids.mjs.map +1 -1
  493. package/dist/services/document-service/transform/relations/transform/data-ids.d.ts.map +1 -1
  494. package/dist/services/document-service/transform/relations/transform/data-ids.js +97 -63
  495. package/dist/services/document-service/transform/relations/transform/data-ids.js.map +1 -1
  496. package/dist/services/document-service/transform/relations/transform/data-ids.mjs +96 -64
  497. package/dist/services/document-service/transform/relations/transform/data-ids.mjs.map +1 -1
  498. package/dist/services/document-service/transform/relations/transform/default-locale.js +47 -29
  499. package/dist/services/document-service/transform/relations/transform/default-locale.js.map +1 -1
  500. package/dist/services/document-service/transform/relations/transform/default-locale.mjs +46 -30
  501. package/dist/services/document-service/transform/relations/transform/default-locale.mjs.map +1 -1
  502. package/dist/services/document-service/transform/relations/utils/dp.d.ts +1 -1
  503. package/dist/services/document-service/transform/relations/utils/dp.d.ts.map +1 -1
  504. package/dist/services/document-service/transform/relations/utils/dp.js +52 -26
  505. package/dist/services/document-service/transform/relations/utils/dp.js.map +1 -1
  506. package/dist/services/document-service/transform/relations/utils/dp.mjs +51 -27
  507. package/dist/services/document-service/transform/relations/utils/dp.mjs.map +1 -1
  508. package/dist/services/document-service/transform/relations/utils/i18n.d.ts +1 -1
  509. package/dist/services/document-service/transform/relations/utils/i18n.d.ts.map +1 -1
  510. package/dist/services/document-service/transform/relations/utils/i18n.js +20 -18
  511. package/dist/services/document-service/transform/relations/utils/i18n.js.map +1 -1
  512. package/dist/services/document-service/transform/relations/utils/i18n.mjs +19 -21
  513. package/dist/services/document-service/transform/relations/utils/i18n.mjs.map +1 -1
  514. package/dist/services/document-service/transform/relations/utils/map-relation.d.ts.map +1 -1
  515. package/dist/services/document-service/transform/relations/utils/map-relation.js +116 -77
  516. package/dist/services/document-service/transform/relations/utils/map-relation.js.map +1 -1
  517. package/dist/services/document-service/transform/relations/utils/map-relation.mjs +115 -79
  518. package/dist/services/document-service/transform/relations/utils/map-relation.mjs.map +1 -1
  519. package/dist/services/document-service/utils/bidirectional-relations.d.ts +95 -0
  520. package/dist/services/document-service/utils/bidirectional-relations.d.ts.map +1 -0
  521. package/dist/services/document-service/utils/bidirectional-relations.js +148 -0
  522. package/dist/services/document-service/utils/bidirectional-relations.js.map +1 -0
  523. package/dist/services/document-service/utils/bidirectional-relations.mjs +145 -0
  524. package/dist/services/document-service/utils/bidirectional-relations.mjs.map +1 -0
  525. package/dist/services/document-service/utils/populate.d.ts +1 -1
  526. package/dist/services/document-service/utils/populate.d.ts.map +1 -1
  527. package/dist/services/document-service/utils/populate.js +67 -41
  528. package/dist/services/document-service/utils/populate.js.map +1 -1
  529. package/dist/services/document-service/utils/populate.mjs +66 -42
  530. package/dist/services/document-service/utils/populate.mjs.map +1 -1
  531. package/dist/services/document-service/utils/unidirectional-relations.d.ts +36 -0
  532. package/dist/services/document-service/utils/unidirectional-relations.d.ts.map +1 -0
  533. package/dist/services/document-service/utils/unidirectional-relations.js +116 -0
  534. package/dist/services/document-service/utils/unidirectional-relations.js.map +1 -0
  535. package/dist/services/document-service/utils/unidirectional-relations.mjs +113 -0
  536. package/dist/services/document-service/utils/unidirectional-relations.mjs.map +1 -0
  537. package/dist/services/entity-service/index.js +230 -161
  538. package/dist/services/entity-service/index.js.map +1 -1
  539. package/dist/services/entity-service/index.mjs +229 -160
  540. package/dist/services/entity-service/index.mjs.map +1 -1
  541. package/dist/services/entity-validator/blocks-validator.js +135 -103
  542. package/dist/services/entity-validator/blocks-validator.js.map +1 -1
  543. package/dist/services/entity-validator/blocks-validator.mjs +134 -104
  544. package/dist/services/entity-validator/blocks-validator.mjs.map +1 -1
  545. package/dist/services/entity-validator/index.d.ts +1 -1
  546. package/dist/services/entity-validator/index.d.ts.map +1 -1
  547. package/dist/services/entity-validator/index.js +362 -363
  548. package/dist/services/entity-validator/index.js.map +1 -1
  549. package/dist/services/entity-validator/index.mjs +358 -360
  550. package/dist/services/entity-validator/index.mjs.map +1 -1
  551. package/dist/services/entity-validator/validators.d.ts.map +1 -1
  552. package/dist/services/entity-validator/validators.js +270 -208
  553. package/dist/services/entity-validator/validators.js.map +1 -1
  554. package/dist/services/entity-validator/validators.mjs +269 -214
  555. package/dist/services/entity-validator/validators.mjs.map +1 -1
  556. package/dist/services/errors.js +65 -65
  557. package/dist/services/errors.js.map +1 -1
  558. package/dist/services/errors.mjs +64 -66
  559. package/dist/services/errors.mjs.map +1 -1
  560. package/dist/services/event-hub.js +82 -69
  561. package/dist/services/event-hub.js.map +1 -1
  562. package/dist/services/event-hub.mjs +81 -71
  563. package/dist/services/event-hub.mjs.map +1 -1
  564. package/dist/services/features.js +19 -14
  565. package/dist/services/features.js.map +1 -1
  566. package/dist/services/features.mjs +18 -15
  567. package/dist/services/features.mjs.map +1 -1
  568. package/dist/services/fs.js +41 -40
  569. package/dist/services/fs.js.map +1 -1
  570. package/dist/services/fs.mjs +40 -39
  571. package/dist/services/fs.mjs.map +1 -1
  572. package/dist/services/metrics/admin-user-hash.d.ts.map +1 -1
  573. package/dist/services/metrics/admin-user-hash.js +13 -11
  574. package/dist/services/metrics/admin-user-hash.js.map +1 -1
  575. package/dist/services/metrics/admin-user-hash.mjs +12 -10
  576. package/dist/services/metrics/admin-user-hash.mjs.map +1 -1
  577. package/dist/services/metrics/index.js +46 -40
  578. package/dist/services/metrics/index.js.map +1 -1
  579. package/dist/services/metrics/index.mjs +45 -42
  580. package/dist/services/metrics/index.mjs.map +1 -1
  581. package/dist/services/metrics/is-truthy.js +13 -6
  582. package/dist/services/metrics/is-truthy.js.map +1 -1
  583. package/dist/services/metrics/is-truthy.mjs +12 -6
  584. package/dist/services/metrics/is-truthy.mjs.map +1 -1
  585. package/dist/services/metrics/middleware.d.ts.map +1 -1
  586. package/dist/services/metrics/middleware.js +37 -22
  587. package/dist/services/metrics/middleware.js.map +1 -1
  588. package/dist/services/metrics/middleware.mjs +36 -24
  589. package/dist/services/metrics/middleware.mjs.map +1 -1
  590. package/dist/services/metrics/rate-limiter.d.ts.map +1 -1
  591. package/dist/services/metrics/rate-limiter.js +24 -19
  592. package/dist/services/metrics/rate-limiter.js.map +1 -1
  593. package/dist/services/metrics/rate-limiter.mjs +23 -21
  594. package/dist/services/metrics/rate-limiter.mjs.map +1 -1
  595. package/dist/services/metrics/sender.d.ts.map +1 -1
  596. package/dist/services/metrics/sender.js +78 -69
  597. package/dist/services/metrics/sender.js.map +1 -1
  598. package/dist/services/metrics/sender.mjs +77 -64
  599. package/dist/services/metrics/sender.mjs.map +1 -1
  600. package/dist/services/query-params.js +13 -10
  601. package/dist/services/query-params.js.map +1 -1
  602. package/dist/services/query-params.mjs +12 -12
  603. package/dist/services/query-params.mjs.map +1 -1
  604. package/dist/services/reloader.js +35 -32
  605. package/dist/services/reloader.js.map +1 -1
  606. package/dist/services/reloader.mjs +34 -33
  607. package/dist/services/reloader.mjs.map +1 -1
  608. package/dist/services/request-context.js +11 -8
  609. package/dist/services/request-context.js.map +1 -1
  610. package/dist/services/request-context.mjs +10 -10
  611. package/dist/services/request-context.mjs.map +1 -1
  612. package/dist/services/server/admin-api.js +11 -10
  613. package/dist/services/server/admin-api.js.map +1 -1
  614. package/dist/services/server/admin-api.mjs +10 -11
  615. package/dist/services/server/admin-api.mjs.map +1 -1
  616. package/dist/services/server/api.js +33 -27
  617. package/dist/services/server/api.js.map +1 -1
  618. package/dist/services/server/api.mjs +32 -26
  619. package/dist/services/server/api.mjs.map +1 -1
  620. package/dist/services/server/compose-endpoint.js +116 -105
  621. package/dist/services/server/compose-endpoint.js.map +1 -1
  622. package/dist/services/server/compose-endpoint.mjs +115 -105
  623. package/dist/services/server/compose-endpoint.mjs.map +1 -1
  624. package/dist/services/server/content-api.js +11 -9
  625. package/dist/services/server/content-api.js.map +1 -1
  626. package/dist/services/server/content-api.mjs +10 -10
  627. package/dist/services/server/content-api.mjs.map +1 -1
  628. package/dist/services/server/http-server.js +48 -44
  629. package/dist/services/server/http-server.js.map +1 -1
  630. package/dist/services/server/http-server.mjs +47 -43
  631. package/dist/services/server/http-server.mjs.map +1 -1
  632. package/dist/services/server/index.js +85 -82
  633. package/dist/services/server/index.js.map +1 -1
  634. package/dist/services/server/index.mjs +84 -81
  635. package/dist/services/server/index.mjs.map +1 -1
  636. package/dist/services/server/koa.js +49 -47
  637. package/dist/services/server/koa.js.map +1 -1
  638. package/dist/services/server/koa.mjs +48 -44
  639. package/dist/services/server/koa.mjs.map +1 -1
  640. package/dist/services/server/middleware.js +86 -82
  641. package/dist/services/server/middleware.js.map +1 -1
  642. package/dist/services/server/middleware.mjs +85 -82
  643. package/dist/services/server/middleware.mjs.map +1 -1
  644. package/dist/services/server/policy.js +24 -17
  645. package/dist/services/server/policy.js.map +1 -1
  646. package/dist/services/server/policy.mjs +23 -18
  647. package/dist/services/server/policy.mjs.map +1 -1
  648. package/dist/services/server/register-middlewares.js +68 -61
  649. package/dist/services/server/register-middlewares.js.map +1 -1
  650. package/dist/services/server/register-middlewares.mjs +67 -63
  651. package/dist/services/server/register-middlewares.mjs.map +1 -1
  652. package/dist/services/server/register-routes.js +90 -67
  653. package/dist/services/server/register-routes.js.map +1 -1
  654. package/dist/services/server/register-routes.mjs +89 -67
  655. package/dist/services/server/register-routes.mjs.map +1 -1
  656. package/dist/services/server/routing.js +94 -81
  657. package/dist/services/server/routing.js.map +1 -1
  658. package/dist/services/server/routing.mjs +93 -81
  659. package/dist/services/server/routing.mjs.map +1 -1
  660. package/dist/services/utils/dynamic-zones.js +13 -14
  661. package/dist/services/utils/dynamic-zones.js.map +1 -1
  662. package/dist/services/utils/dynamic-zones.mjs +12 -16
  663. package/dist/services/utils/dynamic-zones.mjs.map +1 -1
  664. package/dist/services/webhook-runner.js +124 -122
  665. package/dist/services/webhook-runner.js.map +1 -1
  666. package/dist/services/webhook-runner.mjs +123 -121
  667. package/dist/services/webhook-runner.mjs.map +1 -1
  668. package/dist/services/webhook-store.js +132 -99
  669. package/dist/services/webhook-store.js.map +1 -1
  670. package/dist/services/webhook-store.mjs +131 -101
  671. package/dist/services/webhook-store.mjs.map +1 -1
  672. package/dist/services/worker-queue.js +44 -49
  673. package/dist/services/worker-queue.js.map +1 -1
  674. package/dist/services/worker-queue.mjs +43 -49
  675. package/dist/services/worker-queue.mjs.map +1 -1
  676. package/dist/utils/convert-custom-field-type.js +17 -20
  677. package/dist/utils/convert-custom-field-type.js.map +1 -1
  678. package/dist/utils/convert-custom-field-type.mjs +16 -21
  679. package/dist/utils/convert-custom-field-type.mjs.map +1 -1
  680. package/dist/utils/cron.js +64 -30
  681. package/dist/utils/cron.js.map +1 -1
  682. package/dist/utils/cron.mjs +63 -31
  683. package/dist/utils/cron.mjs.map +1 -1
  684. package/dist/utils/fetch.js +24 -18
  685. package/dist/utils/fetch.js.map +1 -1
  686. package/dist/utils/fetch.mjs +23 -19
  687. package/dist/utils/fetch.mjs.map +1 -1
  688. package/dist/utils/filepath-to-prop-path.js +20 -28
  689. package/dist/utils/filepath-to-prop-path.js.map +1 -1
  690. package/dist/utils/filepath-to-prop-path.mjs +19 -26
  691. package/dist/utils/filepath-to-prop-path.mjs.map +1 -1
  692. package/dist/utils/is-initialized.js +21 -12
  693. package/dist/utils/is-initialized.js.map +1 -1
  694. package/dist/utils/is-initialized.mjs +20 -13
  695. package/dist/utils/is-initialized.mjs.map +1 -1
  696. package/dist/utils/lifecycles.js +6 -5
  697. package/dist/utils/lifecycles.js.map +1 -1
  698. package/dist/utils/lifecycles.mjs +5 -6
  699. package/dist/utils/lifecycles.mjs.map +1 -1
  700. package/dist/utils/load-config-file.js +40 -38
  701. package/dist/utils/load-config-file.js.map +1 -1
  702. package/dist/utils/load-config-file.mjs +39 -36
  703. package/dist/utils/load-config-file.mjs.map +1 -1
  704. package/dist/utils/load-files.js +40 -35
  705. package/dist/utils/load-files.js.map +1 -1
  706. package/dist/utils/load-files.mjs +39 -32
  707. package/dist/utils/load-files.mjs.map +1 -1
  708. package/dist/utils/open-browser.js +8 -8
  709. package/dist/utils/open-browser.js.map +1 -1
  710. package/dist/utils/open-browser.mjs +7 -7
  711. package/dist/utils/open-browser.mjs.map +1 -1
  712. package/dist/utils/resolve-working-dirs.js +23 -10
  713. package/dist/utils/resolve-working-dirs.js.map +1 -1
  714. package/dist/utils/resolve-working-dirs.mjs +22 -9
  715. package/dist/utils/resolve-working-dirs.mjs.map +1 -1
  716. package/dist/utils/signals.js +20 -14
  717. package/dist/utils/signals.js.map +1 -1
  718. package/dist/utils/signals.mjs +19 -15
  719. package/dist/utils/signals.mjs.map +1 -1
  720. package/dist/utils/startup-logger.d.ts.map +1 -1
  721. package/dist/utils/startup-logger.js +107 -78
  722. package/dist/utils/startup-logger.js.map +1 -1
  723. package/dist/utils/startup-logger.mjs +106 -75
  724. package/dist/utils/startup-logger.mjs.map +1 -1
  725. package/dist/utils/transform-content-types-to-models.d.ts +355 -23
  726. package/dist/utils/transform-content-types-to-models.d.ts.map +1 -1
  727. package/dist/utils/transform-content-types-to-models.js +350 -260
  728. package/dist/utils/transform-content-types-to-models.js.map +1 -1
  729. package/dist/utils/transform-content-types-to-models.mjs +349 -268
  730. package/dist/utils/transform-content-types-to-models.mjs.map +1 -1
  731. package/dist/utils/update-notifier/index.d.ts.map +1 -1
  732. package/dist/utils/update-notifier/index.js +68 -73
  733. package/dist/utils/update-notifier/index.js.map +1 -1
  734. package/dist/utils/update-notifier/index.mjs +67 -67
  735. package/dist/utils/update-notifier/index.mjs.map +1 -1
  736. package/package.json +30 -30
@@ -1 +1 @@
1
- {"version":3,"file":"map-relation.mjs","sources":["../../../../../../src/services/document-service/transform/relations/utils/map-relation.ts"],"sourcesContent":["/* eslint-disable node/no-callback-literal */\nimport { isObject, curry, isNil } from 'lodash/fp';\n\nimport type { UID } from '@strapi/types';\nimport { traverseEntity, async } from '@strapi/utils';\n\nimport { Relation } from './types';\n\nconst isNumeric = (value: any): value is number => {\n if (Array.isArray(value)) return false; // Handle [1, 'docId'] case\n const parsed = parseInt(value, 10);\n return !Number.isNaN(parsed);\n};\n\nconst toArray = (value: any) => {\n // Keep value as it is if it's a nullish value\n if (isNil(value)) return value;\n if (Array.isArray(value)) return value;\n\n return [value];\n};\n\n/**\n * There are multiple ways to create Strapi relations.\n * This is a utility to traverse and transform relation data\n *\n *\n * For consistency and ease of use, the response will always be an object with the following shape:\n * { set: [{...}], connect: [{...}], disconnect: [{...}] }\n *\n * @example\n * transformRelationData({\n * onLongHand: (relation) => {\n * // Change the id of the relation\n * return { id: 'other' };\n * },\n * }, relation)\n */\nconst mapRelation = async (\n callback: (relation: any) => any,\n rel: Relation,\n isRecursive = false\n): Promise<Relation> => {\n let relation: Relation = rel;\n\n const wrapInSet = (value: any) => {\n // Ignore wrapping if it's a recursive call\n if (isRecursive) {\n return value;\n }\n return { set: toArray(value) };\n };\n\n // undefined | null\n if (isNil(relation)) {\n return callback(relation);\n }\n\n // LongHand[] | ShortHand[]\n if (Array.isArray(relation)) {\n return async\n .map(relation, (r: Relation) => mapRelation(callback, r, true))\n .then((result: any) => result.flat().filter(Boolean))\n .then(wrapInSet);\n }\n\n // LongHand\n if (isObject(relation)) {\n // { id: 1 } || { documentId: 1 }\n if ('id' in relation || 'documentId' in relation) {\n const result = await callback(relation);\n return wrapInSet(result);\n }\n\n // If not connecting anything, return default visitor\n if (!relation.set && !relation.disconnect && !relation.connect) {\n return callback(relation);\n }\n\n // { set }\n if (relation.set) {\n const set: any = await mapRelation(callback, relation.set, true);\n relation = { ...relation, set: toArray(set) };\n }\n\n // { disconnect}\n if (relation.disconnect) {\n const disconnect: any = await mapRelation(callback, relation.disconnect, true);\n relation = { ...relation, disconnect: toArray(disconnect) };\n }\n\n // { connect }\n if (relation.connect) {\n // Transform the relation to connect\n const connect: any = await mapRelation(callback, relation.connect, true);\n relation = { ...relation, connect: toArray(connect) };\n }\n\n return relation;\n }\n\n // ShortHand\n if (isNumeric(relation)) {\n const result = await callback({ id: relation });\n return wrapInSet(result);\n }\n\n if (typeof relation === 'string') {\n const result = await callback({ documentId: relation });\n return wrapInSet(result);\n }\n\n // Anything else\n return callback(relation);\n};\n\ntype TraverseEntity = Parameters<typeof traverseEntity>;\n\n/**\n * Utility function, same as `traverseEntity` but only for relations.\n */\nconst traverseEntityRelations = async (\n visitor: TraverseEntity[0],\n options: TraverseEntity[1],\n data: TraverseEntity[2]\n) => {\n return traverseEntity(\n async (options, utils) => {\n const { attribute } = options;\n\n if (!attribute) {\n return;\n }\n\n if (attribute.type !== 'relation') {\n return;\n }\n\n // TODO: Handle join columns\n if (attribute.useJoinTable === false) {\n return;\n }\n\n // TODO: Handle morph relations (they have multiple targets)\n const target = attribute.target as UID.Schema | undefined;\n if (!target) {\n return;\n }\n\n return visitor(options, utils);\n },\n options,\n data\n );\n};\n\nconst mapRelationCurried = curry(mapRelation);\nconst traverseEntityRelationsCurried = curry(traverseEntityRelations);\n\nexport {\n mapRelationCurried as mapRelation,\n traverseEntityRelationsCurried as traverseEntityRelations,\n};\n"],"names":["options"],"mappings":";;AAQA,MAAM,YAAY,CAAC,UAAgC;AAC7C,MAAA,MAAM,QAAQ,KAAK;AAAU,WAAA;AAC3B,QAAA,SAAS,SAAS,OAAO,EAAE;AAC1B,SAAA,CAAC,OAAO,MAAM,MAAM;AAC7B;AAEA,MAAM,UAAU,CAAC,UAAe;AAE9B,MAAI,MAAM,KAAK;AAAU,WAAA;AACrB,MAAA,MAAM,QAAQ,KAAK;AAAU,WAAA;AAEjC,SAAO,CAAC,KAAK;AACf;AAkBA,MAAM,cAAc,OAClB,UACA,KACA,cAAc,UACQ;AACtB,MAAI,WAAqB;AAEnB,QAAA,YAAY,CAAC,UAAe;AAEhC,QAAI,aAAa;AACR,aAAA;AAAA,IACT;AACA,WAAO,EAAE,KAAK,QAAQ,KAAK,EAAE;AAAA,EAAA;AAI3B,MAAA,MAAM,QAAQ,GAAG;AACnB,WAAO,SAAS,QAAQ;AAAA,EAC1B;AAGI,MAAA,MAAM,QAAQ,QAAQ,GAAG;AACpB,WAAA,MACJ,IAAI,UAAU,CAAC,MAAgB,YAAY,UAAU,GAAG,IAAI,CAAC,EAC7D,KAAK,CAAC,WAAgB,OAAO,OAAO,OAAO,OAAO,CAAC,EACnD,KAAK,SAAS;AAAA,EACnB;AAGI,MAAA,SAAS,QAAQ,GAAG;AAElB,QAAA,QAAQ,YAAY,gBAAgB,UAAU;AAC1C,YAAA,SAAS,MAAM,SAAS,QAAQ;AACtC,aAAO,UAAU,MAAM;AAAA,IACzB;AAGI,QAAA,CAAC,SAAS,OAAO,CAAC,SAAS,cAAc,CAAC,SAAS,SAAS;AAC9D,aAAO,SAAS,QAAQ;AAAA,IAC1B;AAGA,QAAI,SAAS,KAAK;AAChB,YAAM,MAAW,MAAM,YAAY,UAAU,SAAS,KAAK,IAAI;AAC/D,iBAAW,EAAE,GAAG,UAAU,KAAK,QAAQ,GAAG;IAC5C;AAGA,QAAI,SAAS,YAAY;AACvB,YAAM,aAAkB,MAAM,YAAY,UAAU,SAAS,YAAY,IAAI;AAC7E,iBAAW,EAAE,GAAG,UAAU,YAAY,QAAQ,UAAU;IAC1D;AAGA,QAAI,SAAS,SAAS;AAEpB,YAAM,UAAe,MAAM,YAAY,UAAU,SAAS,SAAS,IAAI;AACvE,iBAAW,EAAE,GAAG,UAAU,SAAS,QAAQ,OAAO;IACpD;AAEO,WAAA;AAAA,EACT;AAGI,MAAA,UAAU,QAAQ,GAAG;AACvB,UAAM,SAAS,MAAM,SAAS,EAAE,IAAI,SAAU,CAAA;AAC9C,WAAO,UAAU,MAAM;AAAA,EACzB;AAEI,MAAA,OAAO,aAAa,UAAU;AAChC,UAAM,SAAS,MAAM,SAAS,EAAE,YAAY,SAAU,CAAA;AACtD,WAAO,UAAU,MAAM;AAAA,EACzB;AAGA,SAAO,SAAS,QAAQ;AAC1B;AAOA,MAAM,0BAA0B,OAC9B,SACA,SACA,SACG;AACI,SAAA;AAAA,IACL,OAAOA,UAAS,UAAU;AAClB,YAAA,EAAE,UAAcA,IAAAA;AAEtB,UAAI,CAAC,WAAW;AACd;AAAA,MACF;AAEI,UAAA,UAAU,SAAS,YAAY;AACjC;AAAA,MACF;AAGI,UAAA,UAAU,iBAAiB,OAAO;AACpC;AAAA,MACF;AAGA,YAAM,SAAS,UAAU;AACzB,UAAI,CAAC,QAAQ;AACX;AAAA,MACF;AAEO,aAAA,QAAQA,UAAS,KAAK;AAAA,IAC/B;AAAA,IACA;AAAA,IACA;AAAA,EAAA;AAEJ;AAEM,MAAA,qBAAqB,MAAM,WAAW;AACtC,MAAA,iCAAiC,MAAM,uBAAuB;"}
1
+ {"version":3,"file":"map-relation.mjs","sources":["../../../../../../src/services/document-service/transform/relations/utils/map-relation.ts"],"sourcesContent":["/* eslint-disable node/no-callback-literal */\nimport { isObject, curry, isNil } from 'lodash/fp';\n\nimport { traverseEntity, async } from '@strapi/utils';\n\nimport { Relation } from './types';\n\nconst isNumeric = (value: any): value is number => {\n if (Array.isArray(value)) return false; // Handle [1, 'docId'] case\n const parsed = parseInt(value, 10);\n return !Number.isNaN(parsed);\n};\n\nconst toArray = (value: any) => {\n // Keep value as it is if it's a nullish value\n if (isNil(value)) return value;\n if (Array.isArray(value)) return value;\n\n return [value];\n};\n\n/**\n * There are multiple ways to create Strapi relations.\n * This is a utility to traverse and transform relation data\n *\n *\n * For consistency and ease of use, the response will always be an object with the following shape:\n * { set: [{...}], connect: [{...}], disconnect: [{...}] }\n *\n * @example\n * transformRelationData({\n * onLongHand: (relation) => {\n * // Change the id of the relation\n * return { id: 'other' };\n * },\n * }, relation)\n */\nconst mapRelation = async (\n callback: (relation: any) => any,\n rel: Relation,\n isRecursive = false\n): Promise<Relation> => {\n let relation: Relation = rel;\n\n const wrapInSet = (value: any) => {\n // Ignore wrapping if it's a recursive call\n if (isRecursive) {\n return value;\n }\n return { set: toArray(value) };\n };\n\n // undefined | null\n if (isNil(relation)) {\n return callback(relation);\n }\n\n // LongHand[] | ShortHand[]\n if (Array.isArray(relation)) {\n return async\n .map(relation, (r: Relation) => mapRelation(callback, r, true))\n .then((result: any) => result.flat().filter(Boolean))\n .then(wrapInSet);\n }\n\n // LongHand\n if (isObject(relation)) {\n // { id: 1 } || { documentId: 1 }\n if ('id' in relation || 'documentId' in relation) {\n const result = await callback(relation);\n return wrapInSet(result);\n }\n\n // If not connecting anything, return default visitor\n if (!relation.set && !relation.disconnect && !relation.connect) {\n return callback(relation);\n }\n\n // { set }\n if (relation.set) {\n const set: any = await mapRelation(callback, relation.set, true);\n relation = { ...relation, set: toArray(set) };\n }\n\n // { disconnect}\n if (relation.disconnect) {\n const disconnect: any = await mapRelation(callback, relation.disconnect, true);\n relation = { ...relation, disconnect: toArray(disconnect) };\n }\n\n // { connect }\n if (relation.connect) {\n // Transform the relation to connect\n const connect: any = await mapRelation(callback, relation.connect, true);\n relation = { ...relation, connect: toArray(connect) };\n }\n\n return relation;\n }\n\n // ShortHand\n if (isNumeric(relation)) {\n const result = await callback({ id: relation });\n return wrapInSet(result);\n }\n\n if (typeof relation === 'string') {\n const result = await callback({ documentId: relation });\n return wrapInSet(result);\n }\n\n // Anything else\n return callback(relation);\n};\n\ntype TraverseEntity = Parameters<typeof traverseEntity>;\n\n/**\n * Utility function, same as `traverseEntity` but only for relations.\n */\nconst traverseEntityRelations = async (\n visitor: TraverseEntity[0],\n options: TraverseEntity[1],\n data: TraverseEntity[2]\n) => {\n return traverseEntity(\n async (options, utils) => {\n const { attribute } = options;\n\n if (!attribute) {\n return;\n }\n\n if (attribute.type !== 'relation') {\n return;\n }\n\n // TODO: Handle join columns\n if (attribute.useJoinTable === false) {\n return;\n }\n\n return visitor(options, utils);\n },\n options,\n data\n );\n};\n\nconst mapRelationCurried = curry(mapRelation);\nconst traverseEntityRelationsCurried = curry(traverseEntityRelations);\n\nexport {\n mapRelationCurried as mapRelation,\n traverseEntityRelationsCurried as traverseEntityRelations,\n};\n"],"names":["isNumeric","value","Array","isArray","parsed","parseInt","Number","isNaN","toArray","isNil","mapRelation","callback","rel","isRecursive","relation","wrapInSet","set","async","map","r","then","result","flat","filter","Boolean","isObject","disconnect","connect","id","documentId","traverseEntityRelations","visitor","options","data","traverseEntity","utils","attribute","type","useJoinTable","mapRelationCurried","curry","traverseEntityRelationsCurried"],"mappings":"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"}
@@ -0,0 +1,95 @@
1
+ import type { UID } from '@strapi/types';
2
+ interface LoadContext {
3
+ oldVersions: {
4
+ id: string;
5
+ locale: string;
6
+ }[];
7
+ newVersions: {
8
+ id: string;
9
+ locale: string;
10
+ }[];
11
+ }
12
+ /**
13
+ * Loads all bidirectional relations that need to be synchronized when content entries change state
14
+ * (e.g., during publish/unpublish operations).
15
+ *
16
+ * In Strapi, bidirectional relations allow maintaining order from both sides of the relation.
17
+ * When an entry is published, the following occurs:
18
+ *
19
+ * 1. The old published entry is deleted
20
+ * 2. A new entry is created with all its relations
21
+ *
22
+ * This process affects relation ordering in the following way:
23
+ *
24
+ * Initial state (Entry A related to X, Y, Z):
25
+ * ```
26
+ * Entry A (draft) Entry A (published)
27
+ * │ │
28
+ * ├──(1)→ X ├──(1)→ X
29
+ * ├──(2)→ Y ├──(2)→ Y
30
+ * └──(3)→ Z └──(3)→ Z
31
+ *
32
+ * X's perspective: Y's perspective: Z's perspective:
33
+ * └──(2)→ Entry A └──(1)→ Entry A └──(3)→ Entry A
34
+ * ```
35
+ *
36
+ * After publishing Entry A (without relation order sync):
37
+ * ```
38
+ * Entry A (draft) Entry A (new published)
39
+ * │ │
40
+ * ├──(1)→ X ├──(1)→ X
41
+ * ├──(2)→ Y ├──(2)→ Y
42
+ * └──(3)→ Z └──(3)→ Z
43
+ *
44
+ * X's perspective: Y's perspective: Z's perspective:
45
+ * └──(3)→ Entry A └──(3)→ Entry A └──(3)→ Entry A
46
+ * (all relations appear last in order)
47
+ * ```
48
+ *
49
+ * This module preserves the original ordering from both perspectives by:
50
+ * 1. Capturing the relation order before the entry state changes
51
+ * 2. Restoring this order after the new relations are created
52
+ *
53
+ * @param uid - The unique identifier of the content type being processed
54
+ * @param context - Object containing arrays of old and new entry versions
55
+ * @returns Array of objects containing join table metadata and relations to be updated
56
+ */
57
+ declare const load: (uid: UID.ContentType, { oldVersions }: LoadContext) => Promise<any>;
58
+ /**
59
+ * Synchronizes the order of bidirectional relations after content entries have changed state.
60
+ *
61
+ * When entries change state (e.g., draft → published), their IDs change and all relations are recreated.
62
+ * While the order of relations from the entry's perspective is maintained (as they're created in order),
63
+ * the inverse relations (from related entries' perspective) would all appear last in order since they're new.
64
+ *
65
+ * Example:
66
+ * ```
67
+ * Before publish:
68
+ * Article(id:1) →(order:1)→ Category(id:5)
69
+ * Category(id:5) →(order:3)→ Article(id:1)
70
+ *
71
+ * After publish (without sync):
72
+ * Article(id:2) →(order:1)→ Category(id:5) [order preserved]
73
+ * Category(id:5) →(order:99)→ Article(id:2) [order lost - appears last]
74
+ *
75
+ * After sync:
76
+ * Article(id:2) →(order:1)→ Category(id:5) [order preserved]
77
+ * Category(id:5) →(order:3)→ Article(id:2) [order restored]
78
+ * ```
79
+ *
80
+ * @param oldEntries - Array of previous entry versions with their IDs and locales
81
+ * @param newEntries - Array of new entry versions with their IDs and locales
82
+ * @param existingRelations - Array of join table data containing the relations to be updated
83
+ */
84
+ declare const sync: (oldEntries: {
85
+ id: string;
86
+ locale: string;
87
+ }[], newEntries: {
88
+ id: string;
89
+ locale: string;
90
+ }[], existingRelations: {
91
+ joinTable: any;
92
+ relations: any[];
93
+ }[]) => Promise<void>;
94
+ export { load, sync };
95
+ //# sourceMappingURL=bidirectional-relations.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bidirectional-relations.d.ts","sourceRoot":"","sources":["../../../../src/services/document-service/utils/bidirectional-relations.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,GAAG,EAAU,MAAM,eAAe,CAAC;AAEjD,UAAU,WAAW;IACnB,WAAW,EAAE;QAAE,EAAE,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAA;KAAE,EAAE,CAAC;IAC9C,WAAW,EAAE;QAAE,EAAE,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAA;KAAE,EAAE,CAAC;CAC/C;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA4CG;AACH,QAAA,MAAM,IAAI,QAAe,IAAI,WAAW,mBAAmB,WAAW,iBAmDrE,CAAC;AAEF;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;AACH,QAAA,MAAM,IAAI,eACI;IAAE,EAAE,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,EAAE,cAChC;IAAE,EAAE,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,EAAE,qBACzB;IAAE,SAAS,EAAE,GAAG,CAAC;IAAC,SAAS,EAAE,GAAG,EAAE,CAAA;CAAE,EAAE,kBA6C1D,CAAC;AAEF,OAAO,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC"}
@@ -0,0 +1,148 @@
1
+ 'use strict';
2
+
3
+ var fp = require('lodash/fp');
4
+ var strapiUtils = require('@strapi/utils');
5
+
6
+ /**
7
+ * Loads all bidirectional relations that need to be synchronized when content entries change state
8
+ * (e.g., during publish/unpublish operations).
9
+ *
10
+ * In Strapi, bidirectional relations allow maintaining order from both sides of the relation.
11
+ * When an entry is published, the following occurs:
12
+ *
13
+ * 1. The old published entry is deleted
14
+ * 2. A new entry is created with all its relations
15
+ *
16
+ * This process affects relation ordering in the following way:
17
+ *
18
+ * Initial state (Entry A related to X, Y, Z):
19
+ * ```
20
+ * Entry A (draft) Entry A (published)
21
+ * │ │
22
+ * ├──(1)→ X ├──(1)→ X
23
+ * ├──(2)→ Y ├──(2)→ Y
24
+ * └──(3)→ Z └──(3)→ Z
25
+ *
26
+ * X's perspective: Y's perspective: Z's perspective:
27
+ * └──(2)→ Entry A └──(1)→ Entry A └──(3)→ Entry A
28
+ * ```
29
+ *
30
+ * After publishing Entry A (without relation order sync):
31
+ * ```
32
+ * Entry A (draft) Entry A (new published)
33
+ * │ │
34
+ * ├──(1)→ X ├──(1)→ X
35
+ * ├──(2)→ Y ├──(2)→ Y
36
+ * └──(3)→ Z └──(3)→ Z
37
+ *
38
+ * X's perspective: Y's perspective: Z's perspective:
39
+ * └──(3)→ Entry A └──(3)→ Entry A └──(3)→ Entry A
40
+ * (all relations appear last in order)
41
+ * ```
42
+ *
43
+ * This module preserves the original ordering from both perspectives by:
44
+ * 1. Capturing the relation order before the entry state changes
45
+ * 2. Restoring this order after the new relations are created
46
+ *
47
+ * @param uid - The unique identifier of the content type being processed
48
+ * @param context - Object containing arrays of old and new entry versions
49
+ * @returns Array of objects containing join table metadata and relations to be updated
50
+ */ const load = async (uid, { oldVersions })=>{
51
+ const relationsToUpdate = [];
52
+ await strapi.db.transaction(async ({ trx })=>{
53
+ const contentTypes = Object.values(strapi.contentTypes);
54
+ const components = Object.values(strapi.components);
55
+ for (const model of [
56
+ ...contentTypes,
57
+ ...components
58
+ ]){
59
+ const dbModel = strapi.db.metadata.get(model.uid);
60
+ for (const attribute of Object.values(dbModel.attributes)){
61
+ // Skip if not a bidirectional relation targeting our content type
62
+ if (attribute.type !== 'relation' || attribute.target !== uid || !(attribute.inversedBy || attribute.mappedBy)) {
63
+ continue;
64
+ }
65
+ // If it's a self referencing relation, there is no need to sync any relation
66
+ // The order will already be handled as both sides are inside the same content type
67
+ if (model.uid === uid) {
68
+ continue;
69
+ }
70
+ const joinTable = attribute.joinTable;
71
+ if (!joinTable) {
72
+ continue;
73
+ }
74
+ const { name: targetColumnName } = joinTable.inverseJoinColumn;
75
+ // Load all relations that need their order preserved
76
+ const oldEntryIds = oldVersions.map((entry)=>entry.id);
77
+ const existingRelations = await strapi.db.getConnection().select('*').from(joinTable.name).whereIn(targetColumnName, oldEntryIds).transacting(trx);
78
+ if (existingRelations.length > 0) {
79
+ relationsToUpdate.push({
80
+ joinTable,
81
+ relations: existingRelations
82
+ });
83
+ }
84
+ }
85
+ }
86
+ });
87
+ return relationsToUpdate;
88
+ };
89
+ /**
90
+ * Synchronizes the order of bidirectional relations after content entries have changed state.
91
+ *
92
+ * When entries change state (e.g., draft → published), their IDs change and all relations are recreated.
93
+ * While the order of relations from the entry's perspective is maintained (as they're created in order),
94
+ * the inverse relations (from related entries' perspective) would all appear last in order since they're new.
95
+ *
96
+ * Example:
97
+ * ```
98
+ * Before publish:
99
+ * Article(id:1) →(order:1)→ Category(id:5)
100
+ * Category(id:5) →(order:3)→ Article(id:1)
101
+ *
102
+ * After publish (without sync):
103
+ * Article(id:2) →(order:1)→ Category(id:5) [order preserved]
104
+ * Category(id:5) →(order:99)→ Article(id:2) [order lost - appears last]
105
+ *
106
+ * After sync:
107
+ * Article(id:2) →(order:1)→ Category(id:5) [order preserved]
108
+ * Category(id:5) →(order:3)→ Article(id:2) [order restored]
109
+ * ```
110
+ *
111
+ * @param oldEntries - Array of previous entry versions with their IDs and locales
112
+ * @param newEntries - Array of new entry versions with their IDs and locales
113
+ * @param existingRelations - Array of join table data containing the relations to be updated
114
+ */ const sync = async (oldEntries, newEntries, existingRelations)=>{
115
+ // Group new entries by locale for easier lookup
116
+ const newEntriesByLocale = fp.keyBy('locale', newEntries);
117
+ // Create a mapping of old entry IDs to new entry IDs based on locale
118
+ const entryIdMapping = oldEntries.reduce((acc, oldEntry)=>{
119
+ const newEntry = newEntriesByLocale[oldEntry.locale];
120
+ if (!newEntry) return acc;
121
+ acc[oldEntry.id] = newEntry.id;
122
+ return acc;
123
+ }, {});
124
+ await strapi.db.transaction(async ({ trx })=>{
125
+ for (const { joinTable, relations } of existingRelations){
126
+ const sourceColumn = joinTable.inverseJoinColumn.name;
127
+ const targetColumn = joinTable.joinColumn.name;
128
+ const orderColumn = joinTable.orderColumnName;
129
+ // Failsafe in case those don't exist
130
+ if (!sourceColumn || !targetColumn || !orderColumn) {
131
+ continue;
132
+ }
133
+ // Update order values for each relation
134
+ // TODO: Find a way to batch it more efficiently
135
+ await strapiUtils.async.map(relations, (relation)=>{
136
+ const { [sourceColumn]: oldSourceId, [targetColumn]: targetId, [orderColumn]: originalOrder } = relation;
137
+ // Update the order column for the new relation entry
138
+ return trx.from(joinTable.name).where(sourceColumn, entryIdMapping[oldSourceId]).where(targetColumn, targetId).update({
139
+ [orderColumn]: originalOrder
140
+ });
141
+ });
142
+ }
143
+ });
144
+ };
145
+
146
+ exports.load = load;
147
+ exports.sync = sync;
148
+ //# sourceMappingURL=bidirectional-relations.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bidirectional-relations.js","sources":["../../../../src/services/document-service/utils/bidirectional-relations.ts"],"sourcesContent":["/* eslint-disable no-continue */\nimport { keyBy } from 'lodash/fp';\nimport { async } from '@strapi/utils';\nimport type { UID, Schema } from '@strapi/types';\n\ninterface LoadContext {\n oldVersions: { id: string; locale: string }[];\n newVersions: { id: string; locale: string }[];\n}\n\n/**\n * Loads all bidirectional relations that need to be synchronized when content entries change state\n * (e.g., during publish/unpublish operations).\n *\n * In Strapi, bidirectional relations allow maintaining order from both sides of the relation.\n * When an entry is published, the following occurs:\n *\n * 1. The old published entry is deleted\n * 2. A new entry is created with all its relations\n *\n * This process affects relation ordering in the following way:\n *\n * Initial state (Entry A related to X, Y, Z):\n * ```\n * Entry A (draft) Entry A (published)\n * │ │\n * ├──(1)→ X ├──(1)→ X\n * ├──(2)→ Y ├──(2)→ Y\n * └──(3)→ Z └──(3)→ Z\n *\n * X's perspective: Y's perspective: Z's perspective:\n * └──(2)→ Entry A └──(1)→ Entry A └──(3)→ Entry A\n * ```\n *\n * After publishing Entry A (without relation order sync):\n * ```\n * Entry A (draft) Entry A (new published)\n * │ │\n * ├──(1)→ X ├──(1)→ X\n * ├──(2)→ Y ├──(2)→ Y\n * └──(3)→ Z └──(3)→ Z\n *\n * X's perspective: Y's perspective: Z's perspective:\n * └──(3)→ Entry A └──(3)→ Entry A └──(3)→ Entry A\n * (all relations appear last in order)\n * ```\n *\n * This module preserves the original ordering from both perspectives by:\n * 1. Capturing the relation order before the entry state changes\n * 2. Restoring this order after the new relations are created\n *\n * @param uid - The unique identifier of the content type being processed\n * @param context - Object containing arrays of old and new entry versions\n * @returns Array of objects containing join table metadata and relations to be updated\n */\nconst load = async (uid: UID.ContentType, { oldVersions }: LoadContext) => {\n const relationsToUpdate = [] as any;\n\n await strapi.db.transaction(async ({ trx }) => {\n const contentTypes = Object.values(strapi.contentTypes) as Schema.ContentType[];\n const components = Object.values(strapi.components) as Schema.Component[];\n\n for (const model of [...contentTypes, ...components]) {\n const dbModel = strapi.db.metadata.get(model.uid);\n\n for (const attribute of Object.values(dbModel.attributes) as any) {\n // Skip if not a bidirectional relation targeting our content type\n if (\n attribute.type !== 'relation' ||\n attribute.target !== uid ||\n !(attribute.inversedBy || attribute.mappedBy)\n ) {\n continue;\n }\n\n // If it's a self referencing relation, there is no need to sync any relation\n // The order will already be handled as both sides are inside the same content type\n if (model.uid === uid) {\n continue;\n }\n\n const joinTable = attribute.joinTable;\n if (!joinTable) {\n continue;\n }\n\n const { name: targetColumnName } = joinTable.inverseJoinColumn;\n\n // Load all relations that need their order preserved\n const oldEntryIds = oldVersions.map((entry) => entry.id);\n\n const existingRelations = await strapi.db\n .getConnection()\n .select('*')\n .from(joinTable.name)\n .whereIn(targetColumnName, oldEntryIds)\n .transacting(trx);\n\n if (existingRelations.length > 0) {\n relationsToUpdate.push({ joinTable, relations: existingRelations });\n }\n }\n }\n });\n\n return relationsToUpdate;\n};\n\n/**\n * Synchronizes the order of bidirectional relations after content entries have changed state.\n *\n * When entries change state (e.g., draft → published), their IDs change and all relations are recreated.\n * While the order of relations from the entry's perspective is maintained (as they're created in order),\n * the inverse relations (from related entries' perspective) would all appear last in order since they're new.\n *\n * Example:\n * ```\n * Before publish:\n * Article(id:1) →(order:1)→ Category(id:5)\n * Category(id:5) →(order:3)→ Article(id:1)\n *\n * After publish (without sync):\n * Article(id:2) →(order:1)→ Category(id:5) [order preserved]\n * Category(id:5) →(order:99)→ Article(id:2) [order lost - appears last]\n *\n * After sync:\n * Article(id:2) →(order:1)→ Category(id:5) [order preserved]\n * Category(id:5) →(order:3)→ Article(id:2) [order restored]\n * ```\n *\n * @param oldEntries - Array of previous entry versions with their IDs and locales\n * @param newEntries - Array of new entry versions with their IDs and locales\n * @param existingRelations - Array of join table data containing the relations to be updated\n */\nconst sync = async (\n oldEntries: { id: string; locale: string }[],\n newEntries: { id: string; locale: string }[],\n existingRelations: { joinTable: any; relations: any[] }[]\n) => {\n // Group new entries by locale for easier lookup\n const newEntriesByLocale = keyBy('locale', newEntries);\n\n // Create a mapping of old entry IDs to new entry IDs based on locale\n const entryIdMapping = oldEntries.reduce(\n (acc, oldEntry) => {\n const newEntry = newEntriesByLocale[oldEntry.locale];\n if (!newEntry) return acc;\n acc[oldEntry.id] = newEntry.id;\n return acc;\n },\n {} as Record<string, string>\n );\n\n await strapi.db.transaction(async ({ trx }) => {\n for (const { joinTable, relations } of existingRelations) {\n const sourceColumn = joinTable.inverseJoinColumn.name;\n const targetColumn = joinTable.joinColumn.name;\n const orderColumn = joinTable.orderColumnName;\n\n // Failsafe in case those don't exist\n if (!sourceColumn || !targetColumn || !orderColumn) {\n continue;\n }\n\n // Update order values for each relation\n // TODO: Find a way to batch it more efficiently\n await async.map(relations, (relation: any) => {\n const {\n [sourceColumn]: oldSourceId,\n [targetColumn]: targetId,\n [orderColumn]: originalOrder,\n } = relation;\n\n // Update the order column for the new relation entry\n return trx\n .from(joinTable.name)\n .where(sourceColumn, entryIdMapping[oldSourceId])\n .where(targetColumn, targetId)\n .update({ [orderColumn]: originalOrder });\n });\n }\n });\n};\n\nexport { load, sync };\n"],"names":["load","uid","oldVersions","relationsToUpdate","strapi","db","transaction","trx","contentTypes","Object","values","components","model","dbModel","metadata","get","attribute","attributes","type","target","inversedBy","mappedBy","joinTable","name","targetColumnName","inverseJoinColumn","oldEntryIds","map","entry","id","existingRelations","getConnection","select","from","whereIn","transacting","length","push","relations","sync","oldEntries","newEntries","newEntriesByLocale","keyBy","entryIdMapping","reduce","acc","oldEntry","newEntry","locale","sourceColumn","targetColumn","joinColumn","orderColumn","orderColumnName","async","relation","oldSourceId","targetId","originalOrder","where","update"],"mappings":"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"}
@@ -0,0 +1,145 @@
1
+ import { keyBy } from 'lodash/fp';
2
+ import { async } from '@strapi/utils';
3
+
4
+ /**
5
+ * Loads all bidirectional relations that need to be synchronized when content entries change state
6
+ * (e.g., during publish/unpublish operations).
7
+ *
8
+ * In Strapi, bidirectional relations allow maintaining order from both sides of the relation.
9
+ * When an entry is published, the following occurs:
10
+ *
11
+ * 1. The old published entry is deleted
12
+ * 2. A new entry is created with all its relations
13
+ *
14
+ * This process affects relation ordering in the following way:
15
+ *
16
+ * Initial state (Entry A related to X, Y, Z):
17
+ * ```
18
+ * Entry A (draft) Entry A (published)
19
+ * │ │
20
+ * ├──(1)→ X ├──(1)→ X
21
+ * ├──(2)→ Y ├──(2)→ Y
22
+ * └──(3)→ Z └──(3)→ Z
23
+ *
24
+ * X's perspective: Y's perspective: Z's perspective:
25
+ * └──(2)→ Entry A └──(1)→ Entry A └──(3)→ Entry A
26
+ * ```
27
+ *
28
+ * After publishing Entry A (without relation order sync):
29
+ * ```
30
+ * Entry A (draft) Entry A (new published)
31
+ * │ │
32
+ * ├──(1)→ X ├──(1)→ X
33
+ * ├──(2)→ Y ├──(2)→ Y
34
+ * └──(3)→ Z └──(3)→ Z
35
+ *
36
+ * X's perspective: Y's perspective: Z's perspective:
37
+ * └──(3)→ Entry A └──(3)→ Entry A └──(3)→ Entry A
38
+ * (all relations appear last in order)
39
+ * ```
40
+ *
41
+ * This module preserves the original ordering from both perspectives by:
42
+ * 1. Capturing the relation order before the entry state changes
43
+ * 2. Restoring this order after the new relations are created
44
+ *
45
+ * @param uid - The unique identifier of the content type being processed
46
+ * @param context - Object containing arrays of old and new entry versions
47
+ * @returns Array of objects containing join table metadata and relations to be updated
48
+ */ const load = async (uid, { oldVersions })=>{
49
+ const relationsToUpdate = [];
50
+ await strapi.db.transaction(async ({ trx })=>{
51
+ const contentTypes = Object.values(strapi.contentTypes);
52
+ const components = Object.values(strapi.components);
53
+ for (const model of [
54
+ ...contentTypes,
55
+ ...components
56
+ ]){
57
+ const dbModel = strapi.db.metadata.get(model.uid);
58
+ for (const attribute of Object.values(dbModel.attributes)){
59
+ // Skip if not a bidirectional relation targeting our content type
60
+ if (attribute.type !== 'relation' || attribute.target !== uid || !(attribute.inversedBy || attribute.mappedBy)) {
61
+ continue;
62
+ }
63
+ // If it's a self referencing relation, there is no need to sync any relation
64
+ // The order will already be handled as both sides are inside the same content type
65
+ if (model.uid === uid) {
66
+ continue;
67
+ }
68
+ const joinTable = attribute.joinTable;
69
+ if (!joinTable) {
70
+ continue;
71
+ }
72
+ const { name: targetColumnName } = joinTable.inverseJoinColumn;
73
+ // Load all relations that need their order preserved
74
+ const oldEntryIds = oldVersions.map((entry)=>entry.id);
75
+ const existingRelations = await strapi.db.getConnection().select('*').from(joinTable.name).whereIn(targetColumnName, oldEntryIds).transacting(trx);
76
+ if (existingRelations.length > 0) {
77
+ relationsToUpdate.push({
78
+ joinTable,
79
+ relations: existingRelations
80
+ });
81
+ }
82
+ }
83
+ }
84
+ });
85
+ return relationsToUpdate;
86
+ };
87
+ /**
88
+ * Synchronizes the order of bidirectional relations after content entries have changed state.
89
+ *
90
+ * When entries change state (e.g., draft → published), their IDs change and all relations are recreated.
91
+ * While the order of relations from the entry's perspective is maintained (as they're created in order),
92
+ * the inverse relations (from related entries' perspective) would all appear last in order since they're new.
93
+ *
94
+ * Example:
95
+ * ```
96
+ * Before publish:
97
+ * Article(id:1) →(order:1)→ Category(id:5)
98
+ * Category(id:5) →(order:3)→ Article(id:1)
99
+ *
100
+ * After publish (without sync):
101
+ * Article(id:2) →(order:1)→ Category(id:5) [order preserved]
102
+ * Category(id:5) →(order:99)→ Article(id:2) [order lost - appears last]
103
+ *
104
+ * After sync:
105
+ * Article(id:2) →(order:1)→ Category(id:5) [order preserved]
106
+ * Category(id:5) →(order:3)→ Article(id:2) [order restored]
107
+ * ```
108
+ *
109
+ * @param oldEntries - Array of previous entry versions with their IDs and locales
110
+ * @param newEntries - Array of new entry versions with their IDs and locales
111
+ * @param existingRelations - Array of join table data containing the relations to be updated
112
+ */ const sync = async (oldEntries, newEntries, existingRelations)=>{
113
+ // Group new entries by locale for easier lookup
114
+ const newEntriesByLocale = keyBy('locale', newEntries);
115
+ // Create a mapping of old entry IDs to new entry IDs based on locale
116
+ const entryIdMapping = oldEntries.reduce((acc, oldEntry)=>{
117
+ const newEntry = newEntriesByLocale[oldEntry.locale];
118
+ if (!newEntry) return acc;
119
+ acc[oldEntry.id] = newEntry.id;
120
+ return acc;
121
+ }, {});
122
+ await strapi.db.transaction(async ({ trx })=>{
123
+ for (const { joinTable, relations } of existingRelations){
124
+ const sourceColumn = joinTable.inverseJoinColumn.name;
125
+ const targetColumn = joinTable.joinColumn.name;
126
+ const orderColumn = joinTable.orderColumnName;
127
+ // Failsafe in case those don't exist
128
+ if (!sourceColumn || !targetColumn || !orderColumn) {
129
+ continue;
130
+ }
131
+ // Update order values for each relation
132
+ // TODO: Find a way to batch it more efficiently
133
+ await async.map(relations, (relation)=>{
134
+ const { [sourceColumn]: oldSourceId, [targetColumn]: targetId, [orderColumn]: originalOrder } = relation;
135
+ // Update the order column for the new relation entry
136
+ return trx.from(joinTable.name).where(sourceColumn, entryIdMapping[oldSourceId]).where(targetColumn, targetId).update({
137
+ [orderColumn]: originalOrder
138
+ });
139
+ });
140
+ }
141
+ });
142
+ };
143
+
144
+ export { load, sync };
145
+ //# sourceMappingURL=bidirectional-relations.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bidirectional-relations.mjs","sources":["../../../../src/services/document-service/utils/bidirectional-relations.ts"],"sourcesContent":["/* eslint-disable no-continue */\nimport { keyBy } from 'lodash/fp';\nimport { async } from '@strapi/utils';\nimport type { UID, Schema } from '@strapi/types';\n\ninterface LoadContext {\n oldVersions: { id: string; locale: string }[];\n newVersions: { id: string; locale: string }[];\n}\n\n/**\n * Loads all bidirectional relations that need to be synchronized when content entries change state\n * (e.g., during publish/unpublish operations).\n *\n * In Strapi, bidirectional relations allow maintaining order from both sides of the relation.\n * When an entry is published, the following occurs:\n *\n * 1. The old published entry is deleted\n * 2. A new entry is created with all its relations\n *\n * This process affects relation ordering in the following way:\n *\n * Initial state (Entry A related to X, Y, Z):\n * ```\n * Entry A (draft) Entry A (published)\n * │ │\n * ├──(1)→ X ├──(1)→ X\n * ├──(2)→ Y ├──(2)→ Y\n * └──(3)→ Z └──(3)→ Z\n *\n * X's perspective: Y's perspective: Z's perspective:\n * └──(2)→ Entry A └──(1)→ Entry A └──(3)→ Entry A\n * ```\n *\n * After publishing Entry A (without relation order sync):\n * ```\n * Entry A (draft) Entry A (new published)\n * │ │\n * ├──(1)→ X ├──(1)→ X\n * ├──(2)→ Y ├──(2)→ Y\n * └──(3)→ Z └──(3)→ Z\n *\n * X's perspective: Y's perspective: Z's perspective:\n * └──(3)→ Entry A └──(3)→ Entry A └──(3)→ Entry A\n * (all relations appear last in order)\n * ```\n *\n * This module preserves the original ordering from both perspectives by:\n * 1. Capturing the relation order before the entry state changes\n * 2. Restoring this order after the new relations are created\n *\n * @param uid - The unique identifier of the content type being processed\n * @param context - Object containing arrays of old and new entry versions\n * @returns Array of objects containing join table metadata and relations to be updated\n */\nconst load = async (uid: UID.ContentType, { oldVersions }: LoadContext) => {\n const relationsToUpdate = [] as any;\n\n await strapi.db.transaction(async ({ trx }) => {\n const contentTypes = Object.values(strapi.contentTypes) as Schema.ContentType[];\n const components = Object.values(strapi.components) as Schema.Component[];\n\n for (const model of [...contentTypes, ...components]) {\n const dbModel = strapi.db.metadata.get(model.uid);\n\n for (const attribute of Object.values(dbModel.attributes) as any) {\n // Skip if not a bidirectional relation targeting our content type\n if (\n attribute.type !== 'relation' ||\n attribute.target !== uid ||\n !(attribute.inversedBy || attribute.mappedBy)\n ) {\n continue;\n }\n\n // If it's a self referencing relation, there is no need to sync any relation\n // The order will already be handled as both sides are inside the same content type\n if (model.uid === uid) {\n continue;\n }\n\n const joinTable = attribute.joinTable;\n if (!joinTable) {\n continue;\n }\n\n const { name: targetColumnName } = joinTable.inverseJoinColumn;\n\n // Load all relations that need their order preserved\n const oldEntryIds = oldVersions.map((entry) => entry.id);\n\n const existingRelations = await strapi.db\n .getConnection()\n .select('*')\n .from(joinTable.name)\n .whereIn(targetColumnName, oldEntryIds)\n .transacting(trx);\n\n if (existingRelations.length > 0) {\n relationsToUpdate.push({ joinTable, relations: existingRelations });\n }\n }\n }\n });\n\n return relationsToUpdate;\n};\n\n/**\n * Synchronizes the order of bidirectional relations after content entries have changed state.\n *\n * When entries change state (e.g., draft → published), their IDs change and all relations are recreated.\n * While the order of relations from the entry's perspective is maintained (as they're created in order),\n * the inverse relations (from related entries' perspective) would all appear last in order since they're new.\n *\n * Example:\n * ```\n * Before publish:\n * Article(id:1) →(order:1)→ Category(id:5)\n * Category(id:5) →(order:3)→ Article(id:1)\n *\n * After publish (without sync):\n * Article(id:2) →(order:1)→ Category(id:5) [order preserved]\n * Category(id:5) →(order:99)→ Article(id:2) [order lost - appears last]\n *\n * After sync:\n * Article(id:2) →(order:1)→ Category(id:5) [order preserved]\n * Category(id:5) →(order:3)→ Article(id:2) [order restored]\n * ```\n *\n * @param oldEntries - Array of previous entry versions with their IDs and locales\n * @param newEntries - Array of new entry versions with their IDs and locales\n * @param existingRelations - Array of join table data containing the relations to be updated\n */\nconst sync = async (\n oldEntries: { id: string; locale: string }[],\n newEntries: { id: string; locale: string }[],\n existingRelations: { joinTable: any; relations: any[] }[]\n) => {\n // Group new entries by locale for easier lookup\n const newEntriesByLocale = keyBy('locale', newEntries);\n\n // Create a mapping of old entry IDs to new entry IDs based on locale\n const entryIdMapping = oldEntries.reduce(\n (acc, oldEntry) => {\n const newEntry = newEntriesByLocale[oldEntry.locale];\n if (!newEntry) return acc;\n acc[oldEntry.id] = newEntry.id;\n return acc;\n },\n {} as Record<string, string>\n );\n\n await strapi.db.transaction(async ({ trx }) => {\n for (const { joinTable, relations } of existingRelations) {\n const sourceColumn = joinTable.inverseJoinColumn.name;\n const targetColumn = joinTable.joinColumn.name;\n const orderColumn = joinTable.orderColumnName;\n\n // Failsafe in case those don't exist\n if (!sourceColumn || !targetColumn || !orderColumn) {\n continue;\n }\n\n // Update order values for each relation\n // TODO: Find a way to batch it more efficiently\n await async.map(relations, (relation: any) => {\n const {\n [sourceColumn]: oldSourceId,\n [targetColumn]: targetId,\n [orderColumn]: originalOrder,\n } = relation;\n\n // Update the order column for the new relation entry\n return trx\n .from(joinTable.name)\n .where(sourceColumn, entryIdMapping[oldSourceId])\n .where(targetColumn, targetId)\n .update({ [orderColumn]: originalOrder });\n });\n }\n });\n};\n\nexport { load, sync };\n"],"names":["load","uid","oldVersions","relationsToUpdate","strapi","db","transaction","trx","contentTypes","Object","values","components","model","dbModel","metadata","get","attribute","attributes","type","target","inversedBy","mappedBy","joinTable","name","targetColumnName","inverseJoinColumn","oldEntryIds","map","entry","id","existingRelations","getConnection","select","from","whereIn","transacting","length","push","relations","sync","oldEntries","newEntries","newEntriesByLocale","keyBy","entryIdMapping","reduce","acc","oldEntry","newEntry","locale","sourceColumn","targetColumn","joinColumn","orderColumn","orderColumnName","async","relation","oldSourceId","targetId","originalOrder","where","update"],"mappings":";;;AAUA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA4CC,UACKA,IAAO,GAAA,OAAOC,GAAsB,EAAA,EAAEC,WAAW,EAAe,GAAA;AACpE,IAAA,MAAMC,oBAAoB,EAAE;IAE5B,MAAMC,MAAAA,CAAOC,EAAE,CAACC,WAAW,CAAC,OAAO,EAAEC,GAAG,EAAE,GAAA;AACxC,QAAA,MAAMC,YAAeC,GAAAA,MAAAA,CAAOC,MAAM,CAACN,OAAOI,YAAY,CAAA;AACtD,QAAA,MAAMG,UAAaF,GAAAA,MAAAA,CAAOC,MAAM,CAACN,OAAOO,UAAU,CAAA;AAElD,QAAA,KAAK,MAAMC,KAAS,IAAA;AAAIJ,YAAAA,GAAAA,YAAAA;AAAiBG,YAAAA,GAAAA;SAAW,CAAE;YACpD,MAAME,OAAAA,GAAUT,OAAOC,EAAE,CAACS,QAAQ,CAACC,GAAG,CAACH,KAAAA,CAAMX,GAAG,CAAA;AAEhD,YAAA,KAAK,MAAMe,SAAaP,IAAAA,MAAAA,CAAOC,MAAM,CAACG,OAAAA,CAAQI,UAAU,CAAU,CAAA;;AAEhE,gBAAA,IACED,UAAUE,IAAI,KAAK,UACnBF,IAAAA,SAAAA,CAAUG,MAAM,KAAKlB,GAAAA,IACrB,EAAEe,UAAUI,UAAU,IAAIJ,SAAUK,CAAAA,QAAQ,CAC5C,EAAA;AACA,oBAAA;AACF;;;gBAIA,IAAIT,KAAAA,CAAMX,GAAG,KAAKA,GAAK,EAAA;AACrB,oBAAA;AACF;gBAEA,MAAMqB,SAAAA,GAAYN,UAAUM,SAAS;AACrC,gBAAA,IAAI,CAACA,SAAW,EAAA;AACd,oBAAA;AACF;AAEA,gBAAA,MAAM,EAAEC,IAAMC,EAAAA,gBAAgB,EAAE,GAAGF,UAAUG,iBAAiB;;AAG9D,gBAAA,MAAMC,cAAcxB,WAAYyB,CAAAA,GAAG,CAAC,CAACC,KAAAA,GAAUA,MAAMC,EAAE,CAAA;gBAEvD,MAAMC,iBAAAA,GAAoB,MAAM1B,MAAOC,CAAAA,EAAE,CACtC0B,aAAa,EAAA,CACbC,MAAM,CAAC,GAAA,CAAA,CACPC,IAAI,CAACX,SAAAA,CAAUC,IAAI,CACnBW,CAAAA,OAAO,CAACV,gBAAkBE,EAAAA,WAAAA,CAAAA,CAC1BS,WAAW,CAAC5B,GAAAA,CAAAA;gBAEf,IAAIuB,iBAAAA,CAAkBM,MAAM,GAAG,CAAG,EAAA;AAChCjC,oBAAAA,iBAAAA,CAAkBkC,IAAI,CAAC;AAAEf,wBAAAA,SAAAA;wBAAWgB,SAAWR,EAAAA;AAAkB,qBAAA,CAAA;AACnE;AACF;AACF;AACF,KAAA,CAAA;IAEA,OAAO3B,iBAAAA;AACT;AAEA;;;;;;;;;;;;;;;;;;;;;;;;;AAyBC,IACKoC,MAAAA,IAAAA,GAAO,OACXC,UAAAA,EACAC,UACAX,EAAAA,iBAAAA,GAAAA;;IAGA,MAAMY,kBAAAA,GAAqBC,MAAM,QAAUF,EAAAA,UAAAA,CAAAA;;AAG3C,IAAA,MAAMG,cAAiBJ,GAAAA,UAAAA,CAAWK,MAAM,CACtC,CAACC,GAAKC,EAAAA,QAAAA,GAAAA;AACJ,QAAA,MAAMC,QAAWN,GAAAA,kBAAkB,CAACK,QAAAA,CAASE,MAAM,CAAC;QACpD,IAAI,CAACD,UAAU,OAAOF,GAAAA;AACtBA,QAAAA,GAAG,CAACC,QAASlB,CAAAA,EAAE,CAAC,GAAGmB,SAASnB,EAAE;QAC9B,OAAOiB,GAAAA;AACT,KAAA,EACA,EAAC,CAAA;IAGH,MAAM1C,MAAAA,CAAOC,EAAE,CAACC,WAAW,CAAC,OAAO,EAAEC,GAAG,EAAE,GAAA;AACxC,QAAA,KAAK,MAAM,EAAEe,SAAS,EAAEgB,SAAS,EAAE,IAAIR,iBAAmB,CAAA;AACxD,YAAA,MAAMoB,YAAe5B,GAAAA,SAAAA,CAAUG,iBAAiB,CAACF,IAAI;AACrD,YAAA,MAAM4B,YAAe7B,GAAAA,SAAAA,CAAU8B,UAAU,CAAC7B,IAAI;YAC9C,MAAM8B,WAAAA,GAAc/B,UAAUgC,eAAe;;AAG7C,YAAA,IAAI,CAACJ,YAAAA,IAAgB,CAACC,YAAAA,IAAgB,CAACE,WAAa,EAAA;AAClD,gBAAA;AACF;;;AAIA,YAAA,MAAME,KAAM5B,CAAAA,GAAG,CAACW,SAAAA,EAAW,CAACkB,QAAAA,GAAAA;AAC1B,gBAAA,MAAM,EACJ,CAACN,YAAAA,GAAeO,WAAW,EAC3B,CAACN,YAAAA,GAAeO,QAAQ,EACxB,CAACL,WAAAA,GAAcM,aAAa,EAC7B,GAAGH,QAAAA;;AAGJ,gBAAA,OAAOjD,IACJ0B,IAAI,CAACX,UAAUC,IAAI,CAAA,CACnBqC,KAAK,CAACV,YAAAA,EAAcN,cAAc,CAACa,YAAY,CAC/CG,CAAAA,KAAK,CAACT,YAAcO,EAAAA,QAAAA,CAAAA,CACpBG,MAAM,CAAC;AAAE,oBAAA,CAACR,cAAcM;AAAc,iBAAA,CAAA;AAC3C,aAAA,CAAA;AACF;AACF,KAAA,CAAA;AACF;;;;"}
@@ -1,4 +1,4 @@
1
- import { UID } from '@strapi/types';
1
+ import type { UID } from '@strapi/types';
2
2
  interface Options {
3
3
  /**
4
4
  * Fields to select when populating relations
@@ -1 +1 @@
1
- {"version":3,"file":"populate.d.ts","sourceRoot":"","sources":["../../../../src/services/document-service/utils/populate.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,GAAG,EAAE,MAAM,eAAe,CAAC;AAGpC,UAAU,OAAO;IACf;;OAEG;IACH,gBAAgB,CAAC,EAAE,MAAM,EAAE,CAAC;CAC7B;AAGD,eAAO,MAAM,eAAe,QAAS,IAAI,MAAM,SAAQ,OAAO,QAoD7D,CAAC"}
1
+ {"version":3,"file":"populate.d.ts","sourceRoot":"","sources":["../../../../src/services/document-service/utils/populate.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,eAAe,CAAC;AAGzC,UAAU,OAAO;IACf;;OAEG;IACH,gBAAgB,CAAC,EAAE,MAAM,EAAE,CAAC;CAC7B;AAKD,eAAO,MAAM,eAAe,QAAS,IAAI,MAAM,SAAQ,OAAO,QAwD7D,CAAC"}
@@ -1,45 +1,71 @@
1
- "use strict";
2
- Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
- const strapiUtils = require("@strapi/utils");
4
- const getDeepPopulate = (uid, opts = {}) => {
5
- const model = strapi.getModel(uid);
6
- const attributes = Object.entries(model.attributes);
7
- return attributes.reduce((acc, [attributeName, attribute]) => {
8
- switch (attribute.type) {
9
- case "relation": {
10
- const isMorphRelation = attribute.relation.toLowerCase().startsWith("morph");
11
- if (isMorphRelation) {
12
- break;
1
+ 'use strict';
2
+
3
+ var strapiUtils = require('@strapi/utils');
4
+
5
+ const { CREATED_BY_ATTRIBUTE, UPDATED_BY_ATTRIBUTE } = strapiUtils.contentTypes.constants;
6
+ // We want to build a populate object based on the schema
7
+ const getDeepPopulate = (uid, opts = {})=>{
8
+ const model = strapi.getModel(uid);
9
+ const attributes = Object.entries(model.attributes);
10
+ return attributes.reduce((acc, [attributeName, attribute])=>{
11
+ switch(attribute.type){
12
+ case 'relation':
13
+ {
14
+ // TODO: Support polymorphic relations
15
+ const isMorphRelation = attribute.relation.toLowerCase().startsWith('morph');
16
+ if (isMorphRelation) {
17
+ break;
18
+ }
19
+ // Ignore not visible fields other than createdBy and updatedBy
20
+ const isVisible = strapiUtils.contentTypes.isVisibleAttribute(model, attributeName);
21
+ const isCreatorField = [
22
+ CREATED_BY_ATTRIBUTE,
23
+ UPDATED_BY_ATTRIBUTE
24
+ ].includes(attributeName);
25
+ if (isVisible || isCreatorField) {
26
+ acc[attributeName] = {
27
+ select: opts.relationalFields
28
+ };
29
+ }
30
+ break;
31
+ }
32
+ case 'media':
33
+ {
34
+ // We populate all media fields for completeness of webhook responses
35
+ // see https://github.com/strapi/strapi/issues/21546
36
+ acc[attributeName] = {
37
+ select: [
38
+ '*'
39
+ ]
40
+ };
41
+ break;
42
+ }
43
+ case 'component':
44
+ {
45
+ const populate = getDeepPopulate(attribute.component, opts);
46
+ acc[attributeName] = {
47
+ populate
48
+ };
49
+ break;
50
+ }
51
+ case 'dynamiczone':
52
+ {
53
+ // Use fragments to populate the dynamic zone components
54
+ const populatedComponents = (attribute.components || []).reduce((acc, componentUID)=>{
55
+ acc[componentUID] = {
56
+ populate: getDeepPopulate(componentUID, opts)
57
+ };
58
+ return acc;
59
+ }, {});
60
+ acc[attributeName] = {
61
+ on: populatedComponents
62
+ };
63
+ break;
64
+ }
13
65
  }
14
- const isVisible = strapiUtils.contentTypes.isVisibleAttribute(model, attributeName);
15
- if (isVisible) {
16
- acc[attributeName] = { select: opts.relationalFields };
17
- }
18
- break;
19
- }
20
- case "media": {
21
- acc[attributeName] = { select: ["id"] };
22
- break;
23
- }
24
- case "component": {
25
- const populate = getDeepPopulate(attribute.component, opts);
26
- acc[attributeName] = { populate };
27
- break;
28
- }
29
- case "dynamiczone": {
30
- const populatedComponents = (attribute.components || []).reduce(
31
- (acc2, componentUID) => {
32
- acc2[componentUID] = { populate: getDeepPopulate(componentUID, opts) };
33
- return acc2;
34
- },
35
- {}
36
- );
37
- acc[attributeName] = { on: populatedComponents };
38
- break;
39
- }
40
- }
41
- return acc;
42
- }, {});
66
+ return acc;
67
+ }, {});
43
68
  };
69
+
44
70
  exports.getDeepPopulate = getDeepPopulate;
45
71
  //# sourceMappingURL=populate.js.map