@kalisio/kdk 2.6.4 → 2.7.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (398) hide show
  1. package/core/api/application.js +2 -4
  2. package/core/api/authentication.js +2 -3
  3. package/core/api/db.js +10 -2
  4. package/core/api/hooks/hooks.authorisations.js +4 -2
  5. package/core/api/hooks/hooks.push.js +6 -2
  6. package/core/api/hooks/hooks.query.js +29 -12
  7. package/core/api/hooks/hooks.users.js +30 -17
  8. package/core/api/models/configurations.model.mongodb.js +4 -0
  9. package/core/api/services/authorisations/authorisations.service.js +1 -1
  10. package/core/api/services/configurations/configurations.hooks.js +33 -0
  11. package/core/api/services/index.js +41 -7
  12. package/core/api/services/messages/messages.hooks.js +9 -3
  13. package/core/client/api.js +14 -1
  14. package/core/client/capabilities.js +1 -6
  15. package/core/client/components/KAvatar.vue +24 -20
  16. package/core/client/components/account/KProfile.vue +10 -71
  17. package/core/client/components/account/index.js +0 -2
  18. package/core/client/components/app/KSettings.vue +1 -0
  19. package/core/client/components/collection/KBoard.vue +4 -3
  20. package/core/client/components/collection/KCardSection.vue +1 -0
  21. package/core/client/components/collection/KGrid.vue +2 -0
  22. package/core/client/components/collection/KTable.vue +5 -1
  23. package/core/client/components/collection/KTimeLine.vue +9 -1
  24. package/core/client/components/collection/index.js +0 -2
  25. package/core/client/components/form/KChipsField.vue +2 -1
  26. package/core/client/components/form/KEmailField.vue +1 -0
  27. package/core/client/components/form/KFileField.vue +22 -1
  28. package/core/client/components/form/KForm.vue +2 -0
  29. package/core/client/components/form/KItemField.vue +1 -0
  30. package/core/client/components/form/KNumberField.vue +1 -0
  31. package/core/client/components/form/KPasswordField.vue +1 -0
  32. package/core/client/components/form/KPhoneField.vue +1 -0
  33. package/core/client/components/form/KPropertyItemField.vue +1 -0
  34. package/core/client/components/form/KResolutionField.vue +1 -0
  35. package/core/client/components/form/KSelectField.vue +31 -0
  36. package/core/client/components/form/KTagField.vue +1 -0
  37. package/core/client/components/form/KTextField.vue +1 -0
  38. package/core/client/components/form/KTokenField.vue +1 -0
  39. package/core/client/components/form/KUnitField.vue +1 -0
  40. package/core/client/components/form/KUrlField.vue +1 -0
  41. package/core/client/components/graphics/KIcon.vue +3 -4
  42. package/core/client/components/layout/KPage.vue +1 -0
  43. package/core/client/components/layout/KWindow.vue +6 -3
  44. package/core/client/components/messages/KMessageComposer.vue +2 -1
  45. package/core/client/components/messages/KMessagesTimeLine.vue +1 -1
  46. package/core/client/components/time/KDate.vue +1 -2
  47. package/core/client/components/time/KDateTime.vue +11 -11
  48. package/core/client/components/time/KTime.vue +1 -1
  49. package/core/client/composables/collection.js +33 -8
  50. package/core/client/composables/errors.js +1 -1
  51. package/core/client/composables/layout.js +9 -9
  52. package/core/client/configurations.js +50 -0
  53. package/core/client/exporter.js +1 -1
  54. package/core/client/i18n/core_en.json +6 -39
  55. package/core/client/i18n/core_fr.json +6 -39
  56. package/core/client/index.js +2 -0
  57. package/core/client/layout.js +8 -8
  58. package/core/client/mixins/mixin.base-activity.js +5 -2
  59. package/core/client/mixins/mixin.base-field.js +3 -3
  60. package/core/client/search.js +2 -1
  61. package/core/client/utils/utils.collection.js +8 -8
  62. package/core/client/utils/utils.items.js +4 -0
  63. package/core/client/utils/utils.push.js +3 -3
  64. package/core/client/utils/utils.session.js +7 -5
  65. package/core/client/utils/utils.shapes.js +38 -7
  66. package/core/client/utils/utils.time.js +21 -22
  67. package/core/common/schemas/users.update-profile.json +3 -2
  68. package/coverage/core/api/application.js.html +392 -398
  69. package/coverage/core/api/authentication.js.html +352 -187
  70. package/coverage/core/api/db.js.html +165 -126
  71. package/coverage/core/api/hooks/hooks.authentication.js.html +22 -196
  72. package/coverage/core/api/hooks/hooks.authorisations.js.html +383 -662
  73. package/coverage/core/api/hooks/hooks.logger.js.html +41 -41
  74. package/coverage/core/api/hooks/hooks.model.js.html +113 -101
  75. package/coverage/core/api/hooks/hooks.push.js.html +124 -97
  76. package/coverage/core/api/hooks/hooks.query.js.html +292 -217
  77. package/coverage/core/api/hooks/hooks.schemas.js.html +123 -123
  78. package/coverage/core/api/hooks/hooks.service.js.html +1 -1
  79. package/coverage/core/api/hooks/hooks.storage.js.html +1 -1
  80. package/coverage/core/api/hooks/{hooks.groups.js.html → hooks.tags.js.html} +100 -76
  81. package/coverage/core/api/hooks/hooks.users.js.html +255 -447
  82. package/coverage/core/api/hooks/index.html +107 -122
  83. package/coverage/core/api/hooks/index.js.html +4 -10
  84. package/coverage/core/api/index.html +46 -61
  85. package/coverage/core/api/index.js.html +9 -9
  86. package/coverage/core/api/marshall.js.html +9 -9
  87. package/coverage/core/api/models/{organisations.model.mongodb.js.html → configurations.model.mongodb.js.html} +10 -7
  88. package/coverage/core/api/models/index.html +35 -50
  89. package/coverage/core/api/models/messages.model.mongodb.js.html +39 -27
  90. package/coverage/core/api/models/tags.model.mongodb.js.html +26 -32
  91. package/coverage/core/api/models/users.model.mongodb.js.html +10 -10
  92. package/coverage/core/api/services/account/account.hooks.js.html +5 -5
  93. package/coverage/core/api/services/account/account.service.js.html +127 -127
  94. package/coverage/core/api/services/account/index.html +22 -22
  95. package/coverage/core/api/services/authorisations/authorisations.hooks.js.html +1 -1
  96. package/coverage/core/api/services/authorisations/authorisations.service.js.html +213 -222
  97. package/coverage/core/api/services/authorisations/index.html +21 -21
  98. package/coverage/core/api/services/{organisations/organisations.hooks.js.html → configurations/configurations.hooks.js.html} +16 -10
  99. package/coverage/core/api/services/{groups → configurations}/index.html +8 -8
  100. package/coverage/core/api/services/databases/databases.hooks.js.html +1 -1
  101. package/coverage/core/api/services/databases/databases.service.js.html +1 -1
  102. package/coverage/core/api/services/databases/index.html +1 -1
  103. package/coverage/core/api/services/import-export/import-export.hooks.js.html +76 -76
  104. package/coverage/core/api/services/import-export/import-export.service.js.html +32 -32
  105. package/coverage/core/api/services/import-export/index.html +32 -32
  106. package/coverage/core/api/services/index.html +21 -21
  107. package/coverage/core/api/services/index.js.html +313 -142
  108. package/coverage/core/api/services/mailer/index.html +32 -32
  109. package/coverage/core/api/services/mailer/mailer.hooks.js.html +80 -80
  110. package/coverage/core/api/services/mailer/mailer.service.js.html +32 -32
  111. package/coverage/core/api/services/messages/index.html +21 -21
  112. package/coverage/core/api/services/messages/messages.hooks.js.html +112 -76
  113. package/coverage/core/api/services/push/index.html +32 -32
  114. package/coverage/core/api/services/push/push.hooks.js.html +80 -80
  115. package/coverage/core/api/services/push/push.service.js.html +34 -34
  116. package/coverage/core/api/services/storage/index.html +29 -29
  117. package/coverage/core/api/services/storage/storage.hooks.js.html +80 -80
  118. package/coverage/core/api/services/storage/storage.service.js.html +29 -29
  119. package/coverage/core/api/services/tags/index.html +21 -21
  120. package/coverage/core/api/services/tags/tags.hooks.js.html +119 -71
  121. package/coverage/core/api/services/users/index.html +27 -12
  122. package/coverage/core/api/services/users/users.hooks.js.html +14 -11
  123. package/coverage/core/api/services/users/users.service.js.html +100 -0
  124. package/coverage/core/common/errors.js.html +1 -1
  125. package/coverage/core/common/index.html +42 -27
  126. package/coverage/core/common/index.js.html +1 -1
  127. package/coverage/core/common/permissions.js.html +166 -472
  128. package/coverage/core/common/schema.js.html +4 -4
  129. package/coverage/core/common/utils.js.html +31 -25
  130. package/coverage/core/common/utils.offline.js.html +199 -0
  131. package/coverage/index.html +192 -192
  132. package/coverage/lcov-report/core/api/application.js.html +392 -398
  133. package/coverage/lcov-report/core/api/authentication.js.html +352 -187
  134. package/coverage/lcov-report/core/api/db.js.html +165 -126
  135. package/coverage/lcov-report/core/api/hooks/hooks.authentication.js.html +22 -196
  136. package/coverage/lcov-report/core/api/hooks/hooks.authorisations.js.html +383 -662
  137. package/coverage/lcov-report/core/api/hooks/hooks.logger.js.html +41 -41
  138. package/coverage/lcov-report/core/api/hooks/hooks.model.js.html +113 -101
  139. package/coverage/lcov-report/core/api/hooks/hooks.push.js.html +124 -97
  140. package/coverage/lcov-report/core/api/hooks/hooks.query.js.html +292 -217
  141. package/coverage/lcov-report/core/api/hooks/hooks.schemas.js.html +123 -123
  142. package/coverage/lcov-report/core/api/hooks/hooks.service.js.html +1 -1
  143. package/coverage/lcov-report/core/api/hooks/hooks.storage.js.html +1 -1
  144. package/coverage/lcov-report/core/api/hooks/{hooks.groups.js.html → hooks.tags.js.html} +100 -76
  145. package/coverage/lcov-report/core/api/hooks/hooks.users.js.html +255 -447
  146. package/coverage/lcov-report/core/api/hooks/index.html +107 -122
  147. package/coverage/lcov-report/core/api/hooks/index.js.html +4 -10
  148. package/coverage/lcov-report/core/api/index.html +46 -61
  149. package/coverage/lcov-report/core/api/index.js.html +9 -9
  150. package/coverage/lcov-report/core/api/marshall.js.html +9 -9
  151. package/coverage/lcov-report/core/api/models/{organisations.model.mongodb.js.html → configurations.model.mongodb.js.html} +10 -7
  152. package/coverage/lcov-report/core/api/models/index.html +35 -50
  153. package/coverage/lcov-report/core/api/models/messages.model.mongodb.js.html +39 -27
  154. package/coverage/lcov-report/core/api/models/tags.model.mongodb.js.html +26 -32
  155. package/coverage/lcov-report/core/api/models/users.model.mongodb.js.html +10 -10
  156. package/coverage/lcov-report/core/api/services/account/account.hooks.js.html +5 -5
  157. package/coverage/lcov-report/core/api/services/account/account.service.js.html +127 -127
  158. package/coverage/lcov-report/core/api/services/account/index.html +22 -22
  159. package/coverage/lcov-report/core/api/services/authorisations/authorisations.hooks.js.html +1 -1
  160. package/coverage/lcov-report/core/api/services/authorisations/authorisations.service.js.html +213 -222
  161. package/coverage/lcov-report/core/api/services/authorisations/index.html +21 -21
  162. package/coverage/lcov-report/core/api/services/{groups/groups.hooks.js.html → configurations/configurations.hooks.js.html} +16 -10
  163. package/coverage/lcov-report/core/api/services/{groups → configurations}/index.html +8 -8
  164. package/coverage/lcov-report/core/api/services/databases/databases.hooks.js.html +1 -1
  165. package/coverage/lcov-report/core/api/services/databases/databases.service.js.html +1 -1
  166. package/coverage/lcov-report/core/api/services/databases/index.html +1 -1
  167. package/coverage/lcov-report/core/api/services/import-export/import-export.hooks.js.html +76 -76
  168. package/coverage/lcov-report/core/api/services/import-export/import-export.service.js.html +32 -32
  169. package/coverage/lcov-report/core/api/services/import-export/index.html +32 -32
  170. package/coverage/lcov-report/core/api/services/index.html +21 -21
  171. package/coverage/lcov-report/core/api/services/index.js.html +313 -142
  172. package/coverage/lcov-report/core/api/services/mailer/index.html +32 -32
  173. package/coverage/lcov-report/core/api/services/mailer/mailer.hooks.js.html +80 -80
  174. package/coverage/lcov-report/core/api/services/mailer/mailer.service.js.html +32 -32
  175. package/coverage/lcov-report/core/api/services/messages/index.html +21 -21
  176. package/coverage/lcov-report/core/api/services/messages/messages.hooks.js.html +112 -76
  177. package/coverage/lcov-report/core/api/services/push/index.html +32 -32
  178. package/coverage/lcov-report/core/api/services/push/push.hooks.js.html +80 -80
  179. package/coverage/lcov-report/core/api/services/push/push.service.js.html +34 -34
  180. package/coverage/lcov-report/core/api/services/storage/index.html +29 -29
  181. package/coverage/lcov-report/core/api/services/storage/storage.hooks.js.html +80 -80
  182. package/coverage/lcov-report/core/api/services/storage/storage.service.js.html +29 -29
  183. package/coverage/lcov-report/core/api/services/tags/index.html +21 -21
  184. package/coverage/lcov-report/core/api/services/tags/tags.hooks.js.html +119 -71
  185. package/coverage/lcov-report/core/api/services/users/index.html +27 -12
  186. package/coverage/lcov-report/core/api/services/users/users.hooks.js.html +14 -11
  187. package/coverage/lcov-report/core/api/services/users/users.service.js.html +100 -0
  188. package/coverage/lcov-report/core/common/errors.js.html +1 -1
  189. package/coverage/lcov-report/core/common/index.html +42 -27
  190. package/coverage/lcov-report/core/common/index.js.html +1 -1
  191. package/coverage/lcov-report/core/common/permissions.js.html +166 -472
  192. package/coverage/lcov-report/core/common/schema.js.html +4 -4
  193. package/coverage/lcov-report/core/common/utils.js.html +31 -25
  194. package/coverage/lcov-report/core/common/utils.offline.js.html +199 -0
  195. package/coverage/lcov-report/index.html +192 -192
  196. package/coverage/lcov-report/map/api/hooks/hooks.catalog.js.html +169 -31
  197. package/coverage/lcov-report/map/api/hooks/hooks.features.js.html +1 -1
  198. package/coverage/lcov-report/map/api/hooks/hooks.query.js.html +215 -35
  199. package/coverage/lcov-report/map/api/hooks/index.html +7 -7
  200. package/coverage/lcov-report/map/api/hooks/index.js.html +1 -1
  201. package/coverage/lcov-report/map/api/index.html +1 -1
  202. package/coverage/lcov-report/map/api/index.js.html +1 -1
  203. package/coverage/lcov-report/map/api/marshall.js.html +1 -1
  204. package/coverage/lcov-report/map/api/models/alerts.model.mongodb.js.html +1 -1
  205. package/coverage/lcov-report/map/api/models/catalog.model.mongodb.js.html +82 -7
  206. package/coverage/lcov-report/map/api/models/features.model.mongodb.js.html +1 -1
  207. package/coverage/lcov-report/map/api/models/index.html +22 -7
  208. package/coverage/lcov-report/map/api/models/projects.model.mongodb.js.html +1 -1
  209. package/coverage/lcov-report/{core/api/models/groups.model.mongodb.js.html → map/api/models/styles.model.mongodb.js.html} +10 -7
  210. package/coverage/lcov-report/map/api/services/alerts/alerts.hooks.js.html +1 -1
  211. package/coverage/lcov-report/map/api/services/alerts/alerts.service.js.html +1 -1
  212. package/coverage/lcov-report/map/api/services/alerts/index.html +1 -1
  213. package/coverage/lcov-report/map/api/services/catalog/catalog.hooks.js.html +39 -12
  214. package/coverage/lcov-report/map/api/services/catalog/index.html +5 -5
  215. package/coverage/lcov-report/map/api/services/daptiles/daptiles.service.js.html +1 -1
  216. package/coverage/lcov-report/map/api/services/daptiles/index.html +1 -1
  217. package/coverage/lcov-report/map/api/services/features/features.hooks.js.html +86 -11
  218. package/coverage/lcov-report/map/api/services/features/features.service.js.html +307 -4
  219. package/coverage/lcov-report/map/api/services/features/index.html +7 -7
  220. package/coverage/lcov-report/map/api/services/index.html +5 -5
  221. package/coverage/lcov-report/map/api/services/index.js.html +326 -50
  222. package/coverage/lcov-report/map/api/services/projects/index.html +1 -1
  223. package/coverage/lcov-report/map/api/services/projects/projects.hooks.js.html +1 -1
  224. package/coverage/{core/api/services/organisations → lcov-report/map/api/services/styles}/index.html +10 -25
  225. package/coverage/lcov-report/{core/api/services/organisations/organisations.hooks.js.html → map/api/services/styles/styles.hooks.js.html} +45 -12
  226. package/coverage/lcov-report/map/common/dynamic-grid-source.js.html +1 -1
  227. package/coverage/lcov-report/map/common/errors.js.html +1 -1
  228. package/coverage/lcov-report/map/common/geotiff-grid-source.js.html +7 -10
  229. package/coverage/lcov-report/map/common/grid.js.html +1 -1
  230. package/coverage/lcov-report/map/common/index.html +19 -19
  231. package/coverage/lcov-report/map/common/index.js.html +1 -1
  232. package/coverage/lcov-report/map/common/meteo-model-grid-source.js.html +1 -1
  233. package/coverage/lcov-report/map/common/moment-utils.js.html +1 -1
  234. package/coverage/lcov-report/map/common/opendap-grid-source.js.html +1 -1
  235. package/coverage/lcov-report/map/common/opendap-utils.js.html +4 -7
  236. package/coverage/lcov-report/map/common/permissions.js.html +10 -4
  237. package/coverage/lcov-report/map/common/time-based-grid-source.js.html +1 -1
  238. package/coverage/lcov-report/map/common/tms-utils.js.html +9 -12
  239. package/coverage/lcov-report/map/common/wcs-grid-source.js.html +3 -3
  240. package/coverage/lcov-report/map/common/wcs-utils.js.html +12 -15
  241. package/coverage/lcov-report/map/common/weacast-grid-source.js.html +2 -2
  242. package/coverage/lcov-report/map/common/wfs-utils.js.html +14 -17
  243. package/coverage/lcov-report/map/common/wms-utils.js.html +30 -12
  244. package/coverage/lcov-report/map/common/wmts-utils.js.html +10 -13
  245. package/coverage/lcov.info +4157 -3816
  246. package/coverage/map/api/hooks/hooks.catalog.js.html +169 -31
  247. package/coverage/map/api/hooks/hooks.features.js.html +1 -1
  248. package/coverage/map/api/hooks/hooks.query.js.html +215 -35
  249. package/coverage/map/api/hooks/index.html +7 -7
  250. package/coverage/map/api/hooks/index.js.html +1 -1
  251. package/coverage/map/api/index.html +1 -1
  252. package/coverage/map/api/index.js.html +1 -1
  253. package/coverage/map/api/marshall.js.html +1 -1
  254. package/coverage/map/api/models/alerts.model.mongodb.js.html +1 -1
  255. package/coverage/map/api/models/catalog.model.mongodb.js.html +82 -7
  256. package/coverage/map/api/models/features.model.mongodb.js.html +1 -1
  257. package/coverage/map/api/models/index.html +22 -7
  258. package/coverage/map/api/models/projects.model.mongodb.js.html +1 -1
  259. package/coverage/{core/api/models/groups.model.mongodb.js.html → map/api/models/styles.model.mongodb.js.html} +10 -7
  260. package/coverage/map/api/services/alerts/alerts.hooks.js.html +1 -1
  261. package/coverage/map/api/services/alerts/alerts.service.js.html +1 -1
  262. package/coverage/map/api/services/alerts/index.html +1 -1
  263. package/coverage/map/api/services/catalog/catalog.hooks.js.html +39 -12
  264. package/coverage/map/api/services/catalog/index.html +5 -5
  265. package/coverage/map/api/services/daptiles/daptiles.service.js.html +1 -1
  266. package/coverage/map/api/services/daptiles/index.html +1 -1
  267. package/coverage/map/api/services/features/features.hooks.js.html +86 -11
  268. package/coverage/map/api/services/features/features.service.js.html +307 -4
  269. package/coverage/map/api/services/features/index.html +7 -7
  270. package/coverage/map/api/services/index.html +5 -5
  271. package/coverage/map/api/services/index.js.html +326 -50
  272. package/coverage/map/api/services/projects/index.html +1 -1
  273. package/coverage/map/api/services/projects/projects.hooks.js.html +1 -1
  274. package/coverage/{lcov-report/core/api/services/organisations → map/api/services/styles}/index.html +10 -25
  275. package/coverage/{core/api/services/groups/groups.hooks.js.html → map/api/services/styles/styles.hooks.js.html} +45 -12
  276. package/coverage/map/common/dynamic-grid-source.js.html +1 -1
  277. package/coverage/map/common/errors.js.html +1 -1
  278. package/coverage/map/common/geotiff-grid-source.js.html +7 -10
  279. package/coverage/map/common/grid.js.html +1 -1
  280. package/coverage/map/common/index.html +19 -19
  281. package/coverage/map/common/index.js.html +1 -1
  282. package/coverage/map/common/meteo-model-grid-source.js.html +1 -1
  283. package/coverage/map/common/moment-utils.js.html +1 -1
  284. package/coverage/map/common/opendap-grid-source.js.html +1 -1
  285. package/coverage/map/common/opendap-utils.js.html +4 -7
  286. package/coverage/map/common/permissions.js.html +10 -4
  287. package/coverage/map/common/time-based-grid-source.js.html +1 -1
  288. package/coverage/map/common/tms-utils.js.html +9 -12
  289. package/coverage/map/common/wcs-grid-source.js.html +3 -3
  290. package/coverage/map/common/wcs-utils.js.html +12 -15
  291. package/coverage/map/common/weacast-grid-source.js.html +2 -2
  292. package/coverage/map/common/wfs-utils.js.html +14 -17
  293. package/coverage/map/common/wms-utils.js.html +30 -12
  294. package/coverage/map/common/wmts-utils.js.html +10 -13
  295. package/coverage/tmp/coverage-1028514-1773134124472-0.json +1 -0
  296. package/coverage/tmp/coverage-1028526-1773134124448-0.json +1 -0
  297. package/coverage/tmp/coverage-1028537-1773134124431-0.json +1 -0
  298. package/coverage/tmp/coverage-1028549-1773134124401-0.json +1 -0
  299. package/coverage/tmp/coverage-1028556-1773134124353-0.json +1 -0
  300. package/extras/configs/widgets.top.js +3 -3
  301. package/extras/tests/core/collection.mjs +2 -9
  302. package/extras/tours/pane.top.js +0 -9
  303. package/map/api/hooks/hooks.catalog.js +18 -4
  304. package/map/api/services/catalog/catalog.hooks.js +3 -0
  305. package/map/api/services/features/features.hooks.js +3 -1
  306. package/map/api/services/index.js +2 -6
  307. package/map/api/services/styles/styles.hooks.js +6 -1
  308. package/map/client/components/KFeatureActionButton.vue +9 -3
  309. package/map/client/components/KFeaturesFilterManager.vue +5 -5
  310. package/map/client/components/KFilterCondition.vue +17 -10
  311. package/map/client/components/KLayerEditor.vue +49 -39
  312. package/map/client/components/KMeasureTool.vue +7 -1
  313. package/map/client/components/KTimezoneMap.vue +29 -9
  314. package/map/client/components/catalog/KLayersPanel.vue +26 -16
  315. package/map/client/components/catalog/KLayersSelector.vue +13 -2
  316. package/map/client/components/catalog/KViewsPanel.vue +5 -4
  317. package/map/client/components/form/KSelectLayersField.vue +28 -17
  318. package/map/client/components/form/KSelectViewsField.vue +18 -9
  319. package/map/client/components/form/KTimezoneField.vue +1 -2
  320. package/map/client/components/legend/KVariablesLegend.vue +10 -1
  321. package/map/client/components/location/KLocationCardSection.vue +7 -2
  322. package/map/client/components/location/KLocationMap.vue +31 -7
  323. package/map/client/components/selection/KSelectedLayerFeatures.vue +2 -2
  324. package/map/client/components/stickies/KZoomControl.vue +1 -1
  325. package/map/client/components/styles/KStyleManager.vue +4 -1
  326. package/map/client/components/widget/KTimeSeries.vue +174 -497
  327. package/map/client/components/widget/KTimeSeriesSelector.vue +72 -0
  328. package/map/client/components/widget/KTimeSeriesToolbar.vue +83 -0
  329. package/map/client/composables/catalog.js +6 -10
  330. package/map/client/composables/highlight.js +12 -9
  331. package/map/client/composables/project.js +1 -1
  332. package/map/client/composables/selection.js +8 -7
  333. package/map/client/composables/weather.js +9 -2
  334. package/map/client/geolocation.js +8 -5
  335. package/map/client/i18n/map_en.json +11 -10
  336. package/map/client/i18n/map_fr.json +10 -9
  337. package/map/client/leaflet/TiledFeatureLayer.js +85 -82
  338. package/map/client/leaflet/utils/utils.geojson.js +3 -3
  339. package/map/client/mixins/globe/mixin.base-globe.js +15 -6
  340. package/map/client/mixins/globe/mixin.geojson-layers.js +27 -18
  341. package/map/client/mixins/map/mixin.edit-layers.js +9 -1
  342. package/map/client/mixins/map/mixin.pmtiles-layers.js +118 -29
  343. package/map/client/mixins/map/mixin.tiled-mesh-layers.js +12 -5
  344. package/map/client/mixins/map/mixin.tiled-wind-layers.js +19 -10
  345. package/map/client/mixins/mixin.activity.js +23 -30
  346. package/map/client/mixins/mixin.feature-selection.js +41 -5
  347. package/map/client/planets.js +1 -1
  348. package/map/client/readers/reader.kml.js +2 -3
  349. package/map/client/utils/utils.catalog.js +36 -10
  350. package/map/client/utils/utils.layers.js +39 -8
  351. package/map/client/utils/utils.project.js +4 -0
  352. package/map/client/utils/utils.style.js +37 -7
  353. package/map/client/utils/utils.time-series.js +215 -6
  354. package/map/common/schemas/catalog.update.json +1 -1
  355. package/map/common/weacast-grid-source.js +1 -1
  356. package/package.json +3 -3
  357. package/scripts/kash/CHANGELOG.md +0 -4
  358. package/scripts/kash/README.md +0 -9
  359. package/scripts/kash/kash.sh +45 -40
  360. package/scripts/kash/scripts/run_tests.sh +1 -4
  361. package/test/api/core/authentication.test.js +9 -4
  362. package/test/api/core/config/default.cjs +1 -0
  363. package/test/api/core/hooks.test.js +6 -0
  364. package/test/api/core/index.test.js +43 -18
  365. package/test/api/core/push.test.js +8 -8
  366. package/test/api/core/test-log-2026-03-10.log +60 -0
  367. package/test/api/core/users.test.js +384 -0
  368. package/test/api/map/grid-sources.test.js +1 -1
  369. package/test/api/map/test-log-2026-03-10.log +56 -0
  370. package/vite/package.json +11 -2
  371. package/vite/test/core/composables.test.js +77 -0
  372. package/vite/vitest.config.js +13 -0
  373. package/vite/yarn.lock +1096 -18
  374. package/.vscode/settings.json +0 -5
  375. package/core/client/components/account/KAccount.vue +0 -68
  376. package/core/client/components/account/KDeleteAccountManager.vue +0 -62
  377. package/core/client/components/account/KEmailManager.vue +0 -128
  378. package/core/client/components/account/KPasswordManager.vue +0 -90
  379. package/core/client/components/account/KVerifyEmailManager.vue +0 -105
  380. package/core/client/components/collection/KColumn.vue +0 -227
  381. package/core/client/components/collection/KHistory.vue +0 -113
  382. package/core/client/components/collection/KHistoryEntry.vue +0 -109
  383. package/coverage/core/api/hooks/hooks.organisations.js.html +0 -541
  384. package/coverage/core/api/services/organisations/organisations.service.js.html +0 -343
  385. package/coverage/core/api/utils.js.html +0 -118
  386. package/coverage/lcov-report/core/api/hooks/hooks.organisations.js.html +0 -541
  387. package/coverage/lcov-report/core/api/services/organisations/organisations.service.js.html +0 -343
  388. package/coverage/lcov-report/core/api/utils.js.html +0 -118
  389. package/coverage/tmp/coverage-151166-1723543324307-0.json +0 -1
  390. package/coverage/tmp/coverage-151178-1723543324283-0.json +0 -1
  391. package/coverage/tmp/coverage-151189-1723543324271-0.json +0 -1
  392. package/coverage/tmp/coverage-151201-1723543324248-0.json +0 -1
  393. package/coverage/tmp/coverage-151208-1723543324227-0.json +0 -1
  394. package/scripts/kash/LICENSE +0 -21
  395. package/test/api/core/test-log-2024-04-22.log +0 -84
  396. package/test/api/core/test-log-2024-04-23.log +0 -23
  397. package/test/api/core/test-log-2024-08-13.log +0 -3
  398. package/test/api/map/test-log-2025-03-08.log +0 -0
@@ -23,30 +23,30 @@
23
23
  <div class='clearfix'>
24
24
 
25
25
  <div class='fl pad1y space-right2'>
26
- <span class="strong">30.21% </span>
26
+ <span class="strong">66.21% </span>
27
27
  <span class="quiet">Statements</span>
28
- <span class='fraction'>97/321</span>
28
+ <span class='fraction'>145/219</span>
29
29
  </div>
30
30
 
31
31
 
32
32
  <div class='fl pad1y space-right2'>
33
- <span class="strong">100% </span>
33
+ <span class="strong">72.22% </span>
34
34
  <span class="quiet">Branches</span>
35
- <span class='fraction'>3/3</span>
35
+ <span class='fraction'>13/18</span>
36
36
  </div>
37
37
 
38
38
 
39
39
  <div class='fl pad1y space-right2'>
40
- <span class="strong">3.7% </span>
40
+ <span class="strong">35.29% </span>
41
41
  <span class="quiet">Functions</span>
42
- <span class='fraction'>1/27</span>
42
+ <span class='fraction'>6/17</span>
43
43
  </div>
44
44
 
45
45
 
46
46
  <div class='fl pad1y space-right2'>
47
- <span class="strong">30.21% </span>
47
+ <span class="strong">66.21% </span>
48
48
  <span class="quiet">Lines</span>
49
- <span class='fraction'>97/321</span>
49
+ <span class='fraction'>145/219</span>
50
50
  </div>
51
51
 
52
52
 
@@ -61,7 +61,7 @@
61
61
  </div>
62
62
  </template>
63
63
  </div>
64
- <div class='status-line low'></div>
64
+ <div class='status-line medium'></div>
65
65
  <pre><table class="coverage">
66
66
  <tr><td class="line-count quiet"><a name='L1'></a><a href='#L1'>1</a>
67
67
  <a name='L2'></a><a href='#L2'>2</a>
@@ -282,109 +282,7 @@
282
282
  <a name='L217'></a><a href='#L217'>217</a>
283
283
  <a name='L218'></a><a href='#L218'>218</a>
284
284
  <a name='L219'></a><a href='#L219'>219</a>
285
- <a name='L220'></a><a href='#L220'>220</a>
286
- <a name='L221'></a><a href='#L221'>221</a>
287
- <a name='L222'></a><a href='#L222'>222</a>
288
- <a name='L223'></a><a href='#L223'>223</a>
289
- <a name='L224'></a><a href='#L224'>224</a>
290
- <a name='L225'></a><a href='#L225'>225</a>
291
- <a name='L226'></a><a href='#L226'>226</a>
292
- <a name='L227'></a><a href='#L227'>227</a>
293
- <a name='L228'></a><a href='#L228'>228</a>
294
- <a name='L229'></a><a href='#L229'>229</a>
295
- <a name='L230'></a><a href='#L230'>230</a>
296
- <a name='L231'></a><a href='#L231'>231</a>
297
- <a name='L232'></a><a href='#L232'>232</a>
298
- <a name='L233'></a><a href='#L233'>233</a>
299
- <a name='L234'></a><a href='#L234'>234</a>
300
- <a name='L235'></a><a href='#L235'>235</a>
301
- <a name='L236'></a><a href='#L236'>236</a>
302
- <a name='L237'></a><a href='#L237'>237</a>
303
- <a name='L238'></a><a href='#L238'>238</a>
304
- <a name='L239'></a><a href='#L239'>239</a>
305
- <a name='L240'></a><a href='#L240'>240</a>
306
- <a name='L241'></a><a href='#L241'>241</a>
307
- <a name='L242'></a><a href='#L242'>242</a>
308
- <a name='L243'></a><a href='#L243'>243</a>
309
- <a name='L244'></a><a href='#L244'>244</a>
310
- <a name='L245'></a><a href='#L245'>245</a>
311
- <a name='L246'></a><a href='#L246'>246</a>
312
- <a name='L247'></a><a href='#L247'>247</a>
313
- <a name='L248'></a><a href='#L248'>248</a>
314
- <a name='L249'></a><a href='#L249'>249</a>
315
- <a name='L250'></a><a href='#L250'>250</a>
316
- <a name='L251'></a><a href='#L251'>251</a>
317
- <a name='L252'></a><a href='#L252'>252</a>
318
- <a name='L253'></a><a href='#L253'>253</a>
319
- <a name='L254'></a><a href='#L254'>254</a>
320
- <a name='L255'></a><a href='#L255'>255</a>
321
- <a name='L256'></a><a href='#L256'>256</a>
322
- <a name='L257'></a><a href='#L257'>257</a>
323
- <a name='L258'></a><a href='#L258'>258</a>
324
- <a name='L259'></a><a href='#L259'>259</a>
325
- <a name='L260'></a><a href='#L260'>260</a>
326
- <a name='L261'></a><a href='#L261'>261</a>
327
- <a name='L262'></a><a href='#L262'>262</a>
328
- <a name='L263'></a><a href='#L263'>263</a>
329
- <a name='L264'></a><a href='#L264'>264</a>
330
- <a name='L265'></a><a href='#L265'>265</a>
331
- <a name='L266'></a><a href='#L266'>266</a>
332
- <a name='L267'></a><a href='#L267'>267</a>
333
- <a name='L268'></a><a href='#L268'>268</a>
334
- <a name='L269'></a><a href='#L269'>269</a>
335
- <a name='L270'></a><a href='#L270'>270</a>
336
- <a name='L271'></a><a href='#L271'>271</a>
337
- <a name='L272'></a><a href='#L272'>272</a>
338
- <a name='L273'></a><a href='#L273'>273</a>
339
- <a name='L274'></a><a href='#L274'>274</a>
340
- <a name='L275'></a><a href='#L275'>275</a>
341
- <a name='L276'></a><a href='#L276'>276</a>
342
- <a name='L277'></a><a href='#L277'>277</a>
343
- <a name='L278'></a><a href='#L278'>278</a>
344
- <a name='L279'></a><a href='#L279'>279</a>
345
- <a name='L280'></a><a href='#L280'>280</a>
346
- <a name='L281'></a><a href='#L281'>281</a>
347
- <a name='L282'></a><a href='#L282'>282</a>
348
- <a name='L283'></a><a href='#L283'>283</a>
349
- <a name='L284'></a><a href='#L284'>284</a>
350
- <a name='L285'></a><a href='#L285'>285</a>
351
- <a name='L286'></a><a href='#L286'>286</a>
352
- <a name='L287'></a><a href='#L287'>287</a>
353
- <a name='L288'></a><a href='#L288'>288</a>
354
- <a name='L289'></a><a href='#L289'>289</a>
355
- <a name='L290'></a><a href='#L290'>290</a>
356
- <a name='L291'></a><a href='#L291'>291</a>
357
- <a name='L292'></a><a href='#L292'>292</a>
358
- <a name='L293'></a><a href='#L293'>293</a>
359
- <a name='L294'></a><a href='#L294'>294</a>
360
- <a name='L295'></a><a href='#L295'>295</a>
361
- <a name='L296'></a><a href='#L296'>296</a>
362
- <a name='L297'></a><a href='#L297'>297</a>
363
- <a name='L298'></a><a href='#L298'>298</a>
364
- <a name='L299'></a><a href='#L299'>299</a>
365
- <a name='L300'></a><a href='#L300'>300</a>
366
- <a name='L301'></a><a href='#L301'>301</a>
367
- <a name='L302'></a><a href='#L302'>302</a>
368
- <a name='L303'></a><a href='#L303'>303</a>
369
- <a name='L304'></a><a href='#L304'>304</a>
370
- <a name='L305'></a><a href='#L305'>305</a>
371
- <a name='L306'></a><a href='#L306'>306</a>
372
- <a name='L307'></a><a href='#L307'>307</a>
373
- <a name='L308'></a><a href='#L308'>308</a>
374
- <a name='L309'></a><a href='#L309'>309</a>
375
- <a name='L310'></a><a href='#L310'>310</a>
376
- <a name='L311'></a><a href='#L311'>311</a>
377
- <a name='L312'></a><a href='#L312'>312</a>
378
- <a name='L313'></a><a href='#L313'>313</a>
379
- <a name='L314'></a><a href='#L314'>314</a>
380
- <a name='L315'></a><a href='#L315'>315</a>
381
- <a name='L316'></a><a href='#L316'>316</a>
382
- <a name='L317'></a><a href='#L317'>317</a>
383
- <a name='L318'></a><a href='#L318'>318</a>
384
- <a name='L319'></a><a href='#L319'>319</a>
385
- <a name='L320'></a><a href='#L320'>320</a>
386
- <a name='L321'></a><a href='#L321'>321</a>
387
- <a name='L322'></a><a href='#L322'>322</a></td><td class="line-coverage quiet"><span class="cline-any cline-yes">1x</span>
285
+ <a name='L220'></a><a href='#L220'>220</a></td><td class="line-coverage quiet"><span class="cline-any cline-yes">1x</span>
388
286
  <span class="cline-any cline-yes">1x</span>
389
287
  <span class="cline-any cline-yes">1x</span>
390
288
  <span class="cline-any cline-yes">1x</span>
@@ -432,129 +330,102 @@
432
330
  <span class="cline-any cline-yes">1x</span>
433
331
  <span class="cline-any cline-yes">1x</span>
434
332
  <span class="cline-any cline-yes">1x</span>
435
- <span class="cline-any cline-no">&nbsp;</span>
436
- <span class="cline-any cline-no">&nbsp;</span>
437
- <span class="cline-any cline-no">&nbsp;</span>
438
- <span class="cline-any cline-no">&nbsp;</span>
439
- <span class="cline-any cline-no">&nbsp;</span>
440
- <span class="cline-any cline-no">&nbsp;</span>
441
- <span class="cline-any cline-no">&nbsp;</span>
442
- <span class="cline-any cline-no">&nbsp;</span>
443
- <span class="cline-any cline-no">&nbsp;</span>
444
- <span class="cline-any cline-no">&nbsp;</span>
445
- <span class="cline-any cline-no">&nbsp;</span>
446
- <span class="cline-any cline-no">&nbsp;</span>
447
- <span class="cline-any cline-no">&nbsp;</span>
448
- <span class="cline-any cline-no">&nbsp;</span>
449
- <span class="cline-any cline-no">&nbsp;</span>
450
- <span class="cline-any cline-no">&nbsp;</span>
451
- <span class="cline-any cline-no">&nbsp;</span>
452
- <span class="cline-any cline-no">&nbsp;</span>
453
- <span class="cline-any cline-no">&nbsp;</span>
454
- <span class="cline-any cline-no">&nbsp;</span>
455
- <span class="cline-any cline-no">&nbsp;</span>
456
- <span class="cline-any cline-no">&nbsp;</span>
457
- <span class="cline-any cline-no">&nbsp;</span>
458
- <span class="cline-any cline-no">&nbsp;</span>
459
- <span class="cline-any cline-no">&nbsp;</span>
460
- <span class="cline-any cline-no">&nbsp;</span>
461
- <span class="cline-any cline-no">&nbsp;</span>
462
- <span class="cline-any cline-no">&nbsp;</span>
463
- <span class="cline-any cline-no">&nbsp;</span>
464
- <span class="cline-any cline-no">&nbsp;</span>
465
- <span class="cline-any cline-no">&nbsp;</span>
466
- <span class="cline-any cline-no">&nbsp;</span>
467
- <span class="cline-any cline-no">&nbsp;</span>
333
+ <span class="cline-any cline-yes">23x</span>
334
+ <span class="cline-any cline-yes">23x</span>
335
+ <span class="cline-any cline-yes">23x</span>
336
+ <span class="cline-any cline-yes">23x</span>
337
+ <span class="cline-any cline-yes">23x</span>
338
+ <span class="cline-any cline-yes">23x</span>
339
+ <span class="cline-any cline-yes">23x</span>
340
+ <span class="cline-any cline-yes">23x</span>
341
+ <span class="cline-any cline-yes">23x</span>
342
+ <span class="cline-any cline-yes">23x</span>
343
+ <span class="cline-any cline-yes">23x</span>
344
+ <span class="cline-any cline-yes">23x</span>
345
+ <span class="cline-any cline-yes">23x</span>
346
+ <span class="cline-any cline-yes">23x</span>
347
+ <span class="cline-any cline-yes">23x</span>
348
+ <span class="cline-any cline-yes">23x</span>
349
+ <span class="cline-any cline-yes">22x</span>
350
+ <span class="cline-any cline-yes">22x</span>
351
+ <span class="cline-any cline-yes">22x</span>
352
+ <span class="cline-any cline-yes">22x</span>
353
+ <span class="cline-any cline-yes">22x</span>
354
+ <span class="cline-any cline-yes">22x</span>
355
+ <span class="cline-any cline-yes">22x</span>
356
+ <span class="cline-any cline-yes">22x</span>
357
+ <span class="cline-any cline-yes">22x</span>
358
+ <span class="cline-any cline-yes">22x</span>
359
+ <span class="cline-any cline-yes">22x</span>
360
+ <span class="cline-any cline-yes">22x</span>
361
+ <span class="cline-any cline-yes">22x</span>
362
+ <span class="cline-any cline-yes">22x</span>
363
+ <span class="cline-any cline-yes">22x</span>
364
+ <span class="cline-any cline-yes">22x</span>
365
+ <span class="cline-any cline-yes">22x</span>
366
+ <span class="cline-any cline-yes">22x</span>
367
+ <span class="cline-any cline-yes">22x</span>
368
+ <span class="cline-any cline-yes">23x</span>
468
369
  <span class="cline-any cline-yes">1x</span>
469
370
  <span class="cline-any cline-yes">1x</span>
470
371
  <span class="cline-any cline-yes">1x</span>
372
+ <span class="cline-any cline-yes">23x</span>
373
+ <span class="cline-any cline-yes">23x</span>
374
+ <span class="cline-any cline-yes">23x</span>
375
+ <span class="cline-any cline-yes">23x</span>
376
+ <span class="cline-any cline-yes">23x</span>
377
+ <span class="cline-any cline-yes">23x</span>
378
+ <span class="cline-any cline-yes">23x</span>
379
+ <span class="cline-any cline-yes">23x</span>
380
+ <span class="cline-any cline-yes">23x</span>
381
+ <span class="cline-any cline-yes">23x</span>
382
+ <span class="cline-any cline-yes">23x</span>
383
+ <span class="cline-any cline-yes">3x</span>
471
384
  <span class="cline-any cline-no">&nbsp;</span>
472
385
  <span class="cline-any cline-no">&nbsp;</span>
473
- <span class="cline-any cline-no">&nbsp;</span>
474
- <span class="cline-any cline-no">&nbsp;</span>
475
- <span class="cline-any cline-no">&nbsp;</span>
476
- <span class="cline-any cline-no">&nbsp;</span>
477
- <span class="cline-any cline-no">&nbsp;</span>
478
- <span class="cline-any cline-no">&nbsp;</span>
479
- <span class="cline-any cline-no">&nbsp;</span>
480
- <span class="cline-any cline-no">&nbsp;</span>
481
- <span class="cline-any cline-no">&nbsp;</span>
482
- <span class="cline-any cline-no">&nbsp;</span>
483
- <span class="cline-any cline-no">&nbsp;</span>
484
- <span class="cline-any cline-no">&nbsp;</span>
485
- <span class="cline-any cline-no">&nbsp;</span>
486
- <span class="cline-any cline-no">&nbsp;</span>
487
- <span class="cline-any cline-no">&nbsp;</span>
488
- <span class="cline-any cline-no">&nbsp;</span>
489
- <span class="cline-any cline-no">&nbsp;</span>
490
- <span class="cline-any cline-yes">1x</span>
491
- <span class="cline-any cline-yes">1x</span>
492
- <span class="cline-any cline-yes">8x</span>
493
- <span class="cline-any cline-yes">3x</span>
494
386
  <span class="cline-any cline-yes">3x</span>
495
- <span class="cline-any cline-yes">8x</span>
387
+ <span class="cline-any cline-yes">23x</span>
388
+ <span class="cline-any cline-yes">23x</span>
389
+ <span class="cline-any cline-yes">23x</span>
390
+ <span class="cline-any cline-yes">23x</span>
496
391
  <span class="cline-any cline-yes">1x</span>
497
392
  <span class="cline-any cline-yes">1x</span>
498
- <span class="cline-any cline-no">&nbsp;</span>
499
- <span class="cline-any cline-no">&nbsp;</span>
393
+ <span class="cline-any cline-yes">5x</span>
500
394
  <span class="cline-any cline-yes">1x</span>
501
395
  <span class="cline-any cline-yes">1x</span>
502
- <span class="cline-any cline-no">&nbsp;</span>
503
- <span class="cline-any cline-no">&nbsp;</span>
504
- <span class="cline-any cline-no">&nbsp;</span>
505
- <span class="cline-any cline-no">&nbsp;</span>
506
- <span class="cline-any cline-no">&nbsp;</span>
507
- <span class="cline-any cline-no">&nbsp;</span>
508
- <span class="cline-any cline-no">&nbsp;</span>
509
- <span class="cline-any cline-no">&nbsp;</span>
510
- <span class="cline-any cline-no">&nbsp;</span>
511
- <span class="cline-any cline-no">&nbsp;</span>
396
+ <span class="cline-any cline-yes">5x</span>
512
397
  <span class="cline-any cline-yes">1x</span>
513
398
  <span class="cline-any cline-yes">1x</span>
514
399
  <span class="cline-any cline-no">&nbsp;</span>
515
400
  <span class="cline-any cline-no">&nbsp;</span>
516
- <span class="cline-any cline-no">&nbsp;</span>
517
- <span class="cline-any cline-no">&nbsp;</span>
518
- <span class="cline-any cline-no">&nbsp;</span>
519
- <span class="cline-any cline-no">&nbsp;</span>
520
- <span class="cline-any cline-no">&nbsp;</span>
521
- <span class="cline-any cline-no">&nbsp;</span>
522
- <span class="cline-any cline-no">&nbsp;</span>
523
- <span class="cline-any cline-no">&nbsp;</span>
524
- <span class="cline-any cline-no">&nbsp;</span>
525
401
  <span class="cline-any cline-yes">1x</span>
526
402
  <span class="cline-any cline-yes">1x</span>
403
+ <span class="cline-any cline-yes">4x</span>
404
+ <span class="cline-any cline-yes">4x</span>
405
+ <span class="cline-any cline-yes">4x</span>
406
+ <span class="cline-any cline-yes">4x</span>
407
+ <span class="cline-any cline-yes">4x</span>
408
+ <span class="cline-any cline-yes">4x</span>
409
+ <span class="cline-any cline-yes">4x</span>
410
+ <span class="cline-any cline-yes">4x</span>
411
+ <span class="cline-any cline-yes">4x</span>
527
412
  <span class="cline-any cline-yes">1x</span>
528
- <span class="cline-any cline-no">&nbsp;</span>
529
- <span class="cline-any cline-no">&nbsp;</span>
530
- <span class="cline-any cline-no">&nbsp;</span>
531
- <span class="cline-any cline-no">&nbsp;</span>
532
- <span class="cline-any cline-no">&nbsp;</span>
533
- <span class="cline-any cline-no">&nbsp;</span>
534
- <span class="cline-any cline-no">&nbsp;</span>
535
- <span class="cline-any cline-no">&nbsp;</span>
536
- <span class="cline-any cline-no">&nbsp;</span>
537
- <span class="cline-any cline-no">&nbsp;</span>
538
- <span class="cline-any cline-no">&nbsp;</span>
539
- <span class="cline-any cline-no">&nbsp;</span>
540
- <span class="cline-any cline-no">&nbsp;</span>
541
- <span class="cline-any cline-no">&nbsp;</span>
542
- <span class="cline-any cline-no">&nbsp;</span>
543
- <span class="cline-any cline-no">&nbsp;</span>
544
- <span class="cline-any cline-no">&nbsp;</span>
545
- <span class="cline-any cline-no">&nbsp;</span>
546
- <span class="cline-any cline-no">&nbsp;</span>
413
+ <span class="cline-any cline-yes">4x</span>
547
414
  <span class="cline-any cline-yes">1x</span>
548
415
  <span class="cline-any cline-yes">1x</span>
416
+ <span class="cline-any cline-yes">3x</span>
417
+ <span class="cline-any cline-yes">3x</span>
418
+ <span class="cline-any cline-yes">3x</span>
419
+ <span class="cline-any cline-yes">3x</span>
420
+ <span class="cline-any cline-yes">3x</span>
421
+ <span class="cline-any cline-yes">3x</span>
422
+ <span class="cline-any cline-yes">3x</span>
423
+ <span class="cline-any cline-yes">3x</span>
424
+ <span class="cline-any cline-yes">3x</span>
425
+ <span class="cline-any cline-yes">3x</span>
426
+ <span class="cline-any cline-yes">3x</span>
549
427
  <span class="cline-any cline-yes">1x</span>
550
428
  <span class="cline-any cline-yes">1x</span>
551
- <span class="cline-any cline-no">&nbsp;</span>
552
- <span class="cline-any cline-no">&nbsp;</span>
553
- <span class="cline-any cline-no">&nbsp;</span>
554
- <span class="cline-any cline-no">&nbsp;</span>
555
- <span class="cline-any cline-no">&nbsp;</span>
556
- <span class="cline-any cline-no">&nbsp;</span>
557
- <span class="cline-any cline-no">&nbsp;</span>
558
429
  <span class="cline-any cline-yes">1x</span>
559
430
  <span class="cline-any cline-no">&nbsp;</span>
560
431
  <span class="cline-any cline-no">&nbsp;</span>
@@ -563,15 +434,11 @@
563
434
  <span class="cline-any cline-no">&nbsp;</span>
564
435
  <span class="cline-any cline-no">&nbsp;</span>
565
436
  <span class="cline-any cline-no">&nbsp;</span>
566
- <span class="cline-any cline-yes">1x</span>
567
- <span class="cline-any cline-yes">1x</span>
568
437
  <span class="cline-any cline-no">&nbsp;</span>
569
438
  <span class="cline-any cline-no">&nbsp;</span>
570
439
  <span class="cline-any cline-no">&nbsp;</span>
571
440
  <span class="cline-any cline-no">&nbsp;</span>
572
441
  <span class="cline-any cline-no">&nbsp;</span>
573
- <span class="cline-any cline-yes">1x</span>
574
- <span class="cline-any cline-yes">1x</span>
575
442
  <span class="cline-any cline-no">&nbsp;</span>
576
443
  <span class="cline-any cline-no">&nbsp;</span>
577
444
  <span class="cline-any cline-no">&nbsp;</span>
@@ -582,47 +449,6 @@
582
449
  <span class="cline-any cline-yes">1x</span>
583
450
  <span class="cline-any cline-yes">1x</span>
584
451
  <span class="cline-any cline-yes">1x</span>
585
- <span class="cline-any cline-no">&nbsp;</span>
586
- <span class="cline-any cline-no">&nbsp;</span>
587
- <span class="cline-any cline-no">&nbsp;</span>
588
- <span class="cline-any cline-no">&nbsp;</span>
589
- <span class="cline-any cline-no">&nbsp;</span>
590
- <span class="cline-any cline-no">&nbsp;</span>
591
- <span class="cline-any cline-no">&nbsp;</span>
592
- <span class="cline-any cline-no">&nbsp;</span>
593
- <span class="cline-any cline-no">&nbsp;</span>
594
- <span class="cline-any cline-no">&nbsp;</span>
595
- <span class="cline-any cline-no">&nbsp;</span>
596
- <span class="cline-any cline-no">&nbsp;</span>
597
- <span class="cline-any cline-no">&nbsp;</span>
598
- <span class="cline-any cline-no">&nbsp;</span>
599
- <span class="cline-any cline-no">&nbsp;</span>
600
- <span class="cline-any cline-no">&nbsp;</span>
601
- <span class="cline-any cline-no">&nbsp;</span>
602
- <span class="cline-any cline-no">&nbsp;</span>
603
- <span class="cline-any cline-no">&nbsp;</span>
604
- <span class="cline-any cline-no">&nbsp;</span>
605
- <span class="cline-any cline-no">&nbsp;</span>
606
- <span class="cline-any cline-no">&nbsp;</span>
607
- <span class="cline-any cline-no">&nbsp;</span>
608
- <span class="cline-any cline-no">&nbsp;</span>
609
- <span class="cline-any cline-no">&nbsp;</span>
610
- <span class="cline-any cline-no">&nbsp;</span>
611
- <span class="cline-any cline-no">&nbsp;</span>
612
- <span class="cline-any cline-no">&nbsp;</span>
613
- <span class="cline-any cline-no">&nbsp;</span>
614
- <span class="cline-any cline-no">&nbsp;</span>
615
- <span class="cline-any cline-no">&nbsp;</span>
616
- <span class="cline-any cline-no">&nbsp;</span>
617
- <span class="cline-any cline-no">&nbsp;</span>
618
- <span class="cline-any cline-no">&nbsp;</span>
619
- <span class="cline-any cline-no">&nbsp;</span>
620
- <span class="cline-any cline-no">&nbsp;</span>
621
- <span class="cline-any cline-no">&nbsp;</span>
622
- <span class="cline-any cline-no">&nbsp;</span>
623
- <span class="cline-any cline-no">&nbsp;</span>
624
- <span class="cline-any cline-yes">1x</span>
625
- <span class="cline-any cline-yes">1x</span>
626
452
  <span class="cline-any cline-yes">1x</span>
627
453
  <span class="cline-any cline-no">&nbsp;</span>
628
454
  <span class="cline-any cline-no">&nbsp;</span>
@@ -631,46 +457,17 @@
631
457
  <span class="cline-any cline-no">&nbsp;</span>
632
458
  <span class="cline-any cline-no">&nbsp;</span>
633
459
  <span class="cline-any cline-no">&nbsp;</span>
634
- <span class="cline-any cline-no">&nbsp;</span>
635
- <span class="cline-any cline-no">&nbsp;</span>
636
- <span class="cline-any cline-no">&nbsp;</span>
637
- <span class="cline-any cline-no">&nbsp;</span>
638
- <span class="cline-any cline-no">&nbsp;</span>
639
- <span class="cline-any cline-no">&nbsp;</span>
640
- <span class="cline-any cline-no">&nbsp;</span>
641
- <span class="cline-any cline-no">&nbsp;</span>
642
- <span class="cline-any cline-no">&nbsp;</span>
643
- <span class="cline-any cline-no">&nbsp;</span>
644
- <span class="cline-any cline-no">&nbsp;</span>
645
- <span class="cline-any cline-yes">1x</span>
646
- <span class="cline-any cline-yes">1x</span>
647
460
  <span class="cline-any cline-yes">1x</span>
648
461
  <span class="cline-any cline-no">&nbsp;</span>
649
462
  <span class="cline-any cline-no">&nbsp;</span>
650
- <span class="cline-any cline-yes">1x</span>
651
- <span class="cline-any cline-yes">1x</span>
652
463
  <span class="cline-any cline-no">&nbsp;</span>
653
464
  <span class="cline-any cline-no">&nbsp;</span>
654
- <span class="cline-any cline-yes">1x</span>
655
- <span class="cline-any cline-yes">1x</span>
656
- <span class="cline-any cline-yes">1x</span>
657
465
  <span class="cline-any cline-no">&nbsp;</span>
658
466
  <span class="cline-any cline-no">&nbsp;</span>
659
- <span class="cline-any cline-yes">1x</span>
660
- <span class="cline-any cline-yes">1x</span>
661
- <span class="cline-any cline-no">&nbsp;</span>
662
467
  <span class="cline-any cline-no">&nbsp;</span>
663
468
  <span class="cline-any cline-yes">1x</span>
664
469
  <span class="cline-any cline-yes">1x</span>
665
- <span class="cline-any cline-yes">1x</span>
666
- <span class="cline-any cline-no">&nbsp;</span>
667
470
  <span class="cline-any cline-no">&nbsp;</span>
668
- <span class="cline-any cline-yes">1x</span>
669
- <span class="cline-any cline-yes">1x</span>
670
- <span class="cline-any cline-no">&nbsp;</span>
671
- <span class="cline-any cline-no">&nbsp;</span>
672
- <span class="cline-any cline-yes">1x</span>
673
- <span class="cline-any cline-yes">1x</span>
674
471
  <span class="cline-any cline-no">&nbsp;</span>
675
472
  <span class="cline-any cline-no">&nbsp;</span>
676
473
  <span class="cline-any cline-no">&nbsp;</span>
@@ -681,8 +478,7 @@
681
478
  <span class="cline-any cline-no">&nbsp;</span>
682
479
  <span class="cline-any cline-no">&nbsp;</span>
683
480
  <span class="cline-any cline-no">&nbsp;</span>
684
- <span class="cline-any cline-yes">1x</span>
685
- <span class="cline-any cline-yes">1x</span>
481
+ <span class="cline-any cline-no">&nbsp;</span>
686
482
  <span class="cline-any cline-no">&nbsp;</span>
687
483
  <span class="cline-any cline-no">&nbsp;</span>
688
484
  <span class="cline-any cline-yes">1x</span>
@@ -752,62 +548,65 @@ export <span class="fstat-no" title="function not covered" >function defineResou
752
548
  <span class="cstat-no" title="statement not covered" >}</span>
753
549
  &nbsp;
754
550
  // Hook computing default abilities for a given user
755
- export <span class="fstat-no" title="function not covered" >function defineUserAbilities (subject, can, cannot) {</span>
756
- <span class="cstat-no" title="statement not covered" > // Allow user registration</span>
757
- <span class="cstat-no" title="statement not covered" > can('service', 'users')</span>
758
- <span class="cstat-no" title="statement not covered" > can('create', 'users')</span>
759
- <span class="cstat-no" title="statement not covered" > // Verification email, reset password, etc.</span>
760
- <span class="cstat-no" title="statement not covered" > can('service', 'account')</span>
761
- <span class="cstat-no" title="statement not covered" > can(['create', 'verifyEmail'], 'account')</span>
762
- <span class="cstat-no" title="statement not covered" > // Allow import/export</span>
763
- <span class="cstat-no" title="statement not covered" > can('service', 'import-export')</span>
764
- <span class="cstat-no" title="statement not covered" > can('create', 'import-export')</span>
765
- <span class="cstat-no" title="statement not covered" > // Allow push registration</span>
766
- <span class="cstat-no" title="statement not covered" > can('service', 'push')</span>
767
- <span class="cstat-no" title="statement not covered" > can('create', 'push')</span>
768
- <span class="cstat-no" title="statement not covered" > if (subject &amp;&amp; subject._id) {</span>
769
- <span class="cstat-no" title="statement not covered" > // Read user profiles for authorizing</span>
770
- <span class="cstat-no" title="statement not covered" > can('read', 'users')</span>
771
- <span class="cstat-no" title="statement not covered" > // Update user profile and destroy it</span>
772
- <span class="cstat-no" title="statement not covered" > can(['update', 'remove'], 'users', { _id: subject._id })</span>
773
- <span class="cstat-no" title="statement not covered" > // Access authorisation service, then rights will be granted on a per-resource basis</span>
774
- <span class="cstat-no" title="statement not covered" > can('service', 'authorisations')</span>
775
- <span class="cstat-no" title="statement not covered" > // Access storage service, then rights will be granted on a per-resource basis</span>
776
- <span class="cstat-no" title="statement not covered" > can('service', 'storage')</span>
777
- <span class="cstat-no" title="statement not covered" > // This is for the user avatar</span>
778
- <span class="cstat-no" title="statement not covered" > // take care that the storage service uses 'id' as input but produces _id as output</span>
779
- <span class="cstat-no" title="statement not covered" > can('create', 'storage', { id: 'avatars/' + subject._id.toString() })</span>
780
- <span class="cstat-no" title="statement not covered" > can('create', 'storage', { id: 'avatars/' + subject._id.toString() + '.thumbnail' })</span>
781
- <span class="cstat-no" title="statement not covered" > can(['createMultipartUpload', 'completeMultipartUpload', 'uploadPart', 'putObject'], 'storage', { id: 'avatars/' + subject._id.toString() })</span>
782
- <span class="cstat-no" title="statement not covered" > can(['createMultipartUpload', 'completeMultipartUpload', 'uploadPart', 'putObject'], 'storage', { id: 'avatars/' + subject._id.toString() + '.thumbnail' })</span>
783
- <span class="cstat-no" title="statement not covered" > can('remove', 'storage', { _id: 'avatars/' + subject._id.toString() })</span>
784
- <span class="cstat-no" title="statement not covered" > can('remove', 'storage', { _id: 'avatars/' + subject._id.toString() + '.thumbnail' })</span>
785
- <span class="cstat-no" title="statement not covered" > // Avatar is part of user profiles so that they can be read by any</span>
786
- <span class="cstat-no" title="statement not covered" > can('read', 'storage', { _id: { $regex: '^avatars/*' } })</span>
787
- <span class="cstat-no" title="statement not covered" > }</span>
788
- <span class="cstat-no" title="statement not covered" >}</span>
551
+ export function defineUserAbilities (subject, can, cannot) {
552
+ // Allow user registration
553
+ can('service', 'users')
554
+ can('create', 'users')
555
+ // Verification email, reset password, etc.
556
+ can('service', 'account')
557
+ can(['create', 'verifyEmail'], 'account')
558
+ // Allow import/export
559
+ can('service', 'import-export')
560
+ can('create', 'import-export')
561
+ // Allow push registration
562
+ can('service', 'push')
563
+ can('create', 'push')
564
+ // Allow tag management
565
+ can('service', 'tags')
566
+ can('read', 'tags')
567
+ if (subject &amp;&amp; subject._id) {
568
+ // Read user profiles for authorizing
569
+ can('read', 'users')
570
+ // Update user profile and destroy it
571
+ can(['update', 'remove'], 'users', { _id: subject._id })
572
+ // Access authorisation service, then rights will be granted on a per-resource basis
573
+ can('service', 'authorisations')
574
+ // Access storage service, then rights will be granted on a per-resource basis
575
+ can('service', 'storage')
576
+ // This is for the user avatar
577
+ // take care that the storage service uses 'id' as input but produces _id as output
578
+ can('create', 'storage', { id: 'avatars/' + subject._id.toString() })
579
+ can('create', 'storage', { id: 'avatars/' + subject._id.toString() + '.thumbnail' })
580
+ can(['createMultipartUpload', 'completeMultipartUpload', 'uploadPart', 'putObject'], 'storage', { id: 'avatars/' + subject._id.toString() })
581
+ can(['createMultipartUpload', 'completeMultipartUpload', 'uploadPart', 'putObject'], 'storage', { id: 'avatars/' + subject._id.toString() + '.thumbnail' })
582
+ can('remove', 'storage', { _id: 'avatars/' + subject._id.toString() })
583
+ can('remove', 'storage', { _id: 'avatars/' + subject._id.toString() + '.thumbnail' })
584
+ // Avatar is part of user profiles so that they can be read by any
585
+ can('read', 'storage', { _id: { $regex: '^avatars/*' } })
586
+ }
587
+ }
789
588
  &nbsp;
790
589
  // Compute abilities for a given user
791
- export <span class="fstat-no" title="function not covered" >async function defineAbilities (subject, ...args) {</span>
792
- <span class="cstat-no" title="statement not covered" > const { build, can, cannot } = new AbilityBuilder(Ability)</span>
793
- <span class="cstat-no" title="statement not covered" ></span>
794
- <span class="cstat-no" title="statement not covered" > // Run registered hooks providing any additional arguments used to handle complex use cases</span>
795
- <span class="cstat-no" title="statement not covered" > await Promise.all(hooks.map(async hook =&gt; {</span>
796
- <span class="cstat-no" title="statement not covered" > await hook(subject, can, cannot, ...args)</span>
797
- <span class="cstat-no" title="statement not covered" > }))</span>
798
- <span class="cstat-no" title="statement not covered" ></span>
799
- <span class="cstat-no" title="statement not covered" > // CASL cannot infer the object type from the object itself so we need</span>
800
- <span class="cstat-no" title="statement not covered" > // to tell it how he can find the object type, i.e. service name.</span>
801
- <span class="cstat-no" title="statement not covered" > return build({</span>
802
- <span class="cstat-no" title="statement not covered" > detectSubjectType: resource =&gt; {</span>
803
- <span class="cstat-no" title="statement not covered" > if (!resource || typeof resource === 'string') {</span>
590
+ export async function defineAbilities (subject, ...args) {
591
+ const { build, can, cannot } = new AbilityBuilder(Ability)
592
+ &nbsp;
593
+ // Run registered hooks providing any additional arguments used to handle complex use cases
594
+ await Promise.all(hooks.map(async hook =&gt; {
595
+ await hook(subject, can, cannot, ...args)
596
+ }))
597
+ &nbsp;
598
+ // CASL cannot infer the object type from the object itself so we need
599
+ // to tell it how he can find the object type, i.e. service name.
600
+ return build({
601
+ detectSubjectType: resource =&gt; {
602
+ if (!resource || typeof resource === 'string') <span class="branch-0 cbranch-no" title="branch not covered" >{</span>
804
603
  <span class="cstat-no" title="statement not covered" > return resource</span>
805
604
  <span class="cstat-no" title="statement not covered" > }</span>
806
- <span class="cstat-no" title="statement not covered" > return resource[RESOURCE_TYPE_KEY]</span>
807
- <span class="cstat-no" title="statement not covered" > },</span>
808
- <span class="cstat-no" title="statement not covered" > resolveAction</span>
809
- <span class="cstat-no" title="statement not covered" > })</span>
810
- <span class="cstat-no" title="statement not covered" >}</span>
605
+ return resource[RESOURCE_TYPE_KEY]
606
+ },
607
+ resolveAction
608
+ })
609
+ }
811
610
  &nbsp;
812
611
  defineAbilities.registerHook = function (hook) {
813
612
  if (!hooks.includes(hook)) {
@@ -819,30 +618,31 @@ defineAbilities.unregisterHook = <span class="fstat-no" title="function not cove
819
618
  <span class="cstat-no" title="statement not covered" > hooks = hooks.filter(registeredHook =&gt; registeredHook !== hook)</span>
820
619
  <span class="cstat-no" title="statement not covered" >}</span>
821
620
  &nbsp;
822
- export <span class="fstat-no" title="function not covered" >function hasServiceAbilities (abilities, service) {</span>
823
- <span class="cstat-no" title="statement not covered" > if (!abilities) return false</span>
824
- <span class="cstat-no" title="statement not covered" > // The unique identifier of a service is its path not its name.</span>
825
- <span class="cstat-no" title="statement not covered" > // Indeed we have for instance a 'groups' service in each organisation</span>
826
- <span class="cstat-no" title="statement not covered" > // Take care that in client we have the service path while on server we have the actual object</span>
827
- <span class="cstat-no" title="statement not covered" > const path = typeof service === 'string' ? service : service.getPath()</span>
828
- <span class="cstat-no" title="statement not covered" > // */groups will allow to access any groups service in any context</span>
829
- <span class="cstat-no" title="statement not covered" > return abilities.can('service', path) ||</span>
830
- <span class="cstat-no" title="statement not covered" > abilities.can('service', `*/${path}`) ||</span>
831
- <span class="cstat-no" title="statement not covered" > abilities.can('service', _.replace(path, /^.*\//, '*/'))</span>
832
- <span class="cstat-no" title="statement not covered" >}</span>
621
+ export function hasServiceAbilities (abilities, service) {
622
+ if (!abilities) <span class="branch-0 cbranch-no" title="branch not covered" >return false</span>
623
+ // The unique identifier of a service is its path not its name.
624
+ // Indeed we have for instance a 'groups' service in each context
625
+ // Take care that in client we have the service path while on server we have the actual object
626
+ const path = typeof service === 'string' <span class="branch-0 cbranch-no" title="branch not covered" >? service </span>: service.getPath()
627
+ // */groups will allow to access any groups service in any context
628
+ const allContextsPath = _.replace(path, /^.*\//, '*/')
629
+ return abilities.can('service', path) ||
630
+ abilities.can('service', `*/${path}`) ||
631
+ abilities.can('service', allContextsPath)
632
+ }
833
633
  &nbsp;
834
- export <span class="fstat-no" title="function not covered" >function hasResourceAbilities (abilities, operation, resourceType, context, resource) {</span>
835
- <span class="cstat-no" title="statement not covered" > if (!abilities) return false</span>
836
- <span class="cstat-no" title="statement not covered" > // Create a shallow copy adding context and type</span>
837
- <span class="cstat-no" title="statement not covered" > const object = Object.assign({}, resource)</span>
838
- <span class="cstat-no" title="statement not covered" > object[RESOURCE_TYPE_KEY] = resourceType</span>
839
- <span class="cstat-no" title="statement not covered" > // Add a virtual context to take it into account for object having no link to it</span>
840
- <span class="cstat-no" title="statement not covered" > if (context) object.context = (typeof context === 'object' ? context._id.toString() : context.toString())</span>
841
- <span class="cstat-no" title="statement not covered" ></span>
842
- <span class="cstat-no" title="statement not covered" > const result = abilities.can(operation, object)</span>
843
- <span class="cstat-no" title="statement not covered" ></span>
844
- <span class="cstat-no" title="statement not covered" > return result</span>
845
- <span class="cstat-no" title="statement not covered" >}</span>
634
+ export function hasResourceAbilities (abilities, operation, resourceType, context, resource) {
635
+ if (!abilities) <span class="branch-0 cbranch-no" title="branch not covered" >return false</span>
636
+ // Create a shallow copy adding context and type
637
+ const object = Object.assign({}, resource)
638
+ object[RESOURCE_TYPE_KEY] = resourceType
639
+ // Add a virtual context to take it into account for object having no link to it
640
+ if (context) <span class="branch-0 cbranch-no" title="branch not covered" >object.context = (typeof context === 'object' ? context._id.toString() : context.toString())</span>
641
+ &nbsp;
642
+ const result = abilities.can(operation, object)
643
+ &nbsp;
644
+ return result
645
+ }
846
646
  &nbsp;
847
647
  // Utility function used to remove the virtual context from query
848
648
  export <span class="fstat-no" title="function not covered" >function removeContext (query) {</span>
@@ -901,112 +701,6 @@ export <span class="fstat-no" title="function not covered" >function countSubjec
901
701
  <span class="cstat-no" title="statement not covered" > return subjectService.find({ query })</span>
902
702
  <span class="cstat-no" title="statement not covered" >}</span>
903
703
  &nbsp;
904
- // Hook computing organisation abilities for a given user
905
- export <span class="fstat-no" title="function not covered" >function defineOrganisationAbilities (subject, can, cannot) {</span>
906
- <span class="cstat-no" title="statement not covered" > if (subject) {</span>
907
- <span class="cstat-no" title="statement not covered" > // Create new organisations</span>
908
- <span class="cstat-no" title="statement not covered" > can('service', 'organisations')</span>
909
- <span class="cstat-no" title="statement not covered" > can('create', 'organisations')</span>
910
- <span class="cstat-no" title="statement not covered" ></span>
911
- <span class="cstat-no" title="statement not covered" > if (subject.organisations) {</span>
912
- <span class="cstat-no" title="statement not covered" > subject.organisations.forEach(organisation =&gt; {</span>
913
- <span class="cstat-no" title="statement not covered" > if (organisation._id) {</span>
914
- <span class="cstat-no" title="statement not covered" > // Generic rules for resources</span>
915
- <span class="cstat-no" title="statement not covered" > defineResourceRules(subject, organisation, 'organisations', can)</span>
916
- <span class="cstat-no" title="statement not covered" > // Specific rules for organisations</span>
917
- <span class="cstat-no" title="statement not covered" > const role = Roles[organisation.permissions]</span>
918
- <span class="cstat-no" title="statement not covered" > if (role &gt;= Roles.member) {</span>
919
- <span class="cstat-no" title="statement not covered" > // The unique identifier of a service is its path not its name.</span>
920
- <span class="cstat-no" title="statement not covered" > // Indeed we have for instance a 'groups' service in each organisation.</span>
921
- <span class="cstat-no" title="statement not covered" > can('service', organisation._id.toString() + '/members')</span>
922
- <span class="cstat-no" title="statement not covered" > can('read', 'members', { context: organisation._id })</span>
923
- <span class="cstat-no" title="statement not covered" > can('service', organisation._id.toString() + '/tags')</span>
924
- <span class="cstat-no" title="statement not covered" > // Tags are public</span>
925
- <span class="cstat-no" title="statement not covered" > can('read', 'tags', { context: organisation._id })</span>
926
- <span class="cstat-no" title="statement not covered" > // Groups are private</span>
927
- <span class="cstat-no" title="statement not covered" > can('service', organisation._id.toString() + '/groups')</span>
928
- <span class="cstat-no" title="statement not covered" > can('service', organisation._id.toString() + '/storage')</span>
929
- <span class="cstat-no" title="statement not covered" > can(['read', 'create', 'remove', 'createMultipartUpload', 'completeMultipartUpload', 'uploadPart', 'putObject'], 'storage', { context: organisation._id })</span>
930
- <span class="cstat-no" title="statement not covered" > }</span>
931
- <span class="cstat-no" title="statement not covered" > if (role &gt;= Roles.manager) {</span>
932
- <span class="cstat-no" title="statement not covered" > can('update', 'members', { context: organisation._id })</span>
933
- <span class="cstat-no" title="statement not covered" > // Managers can manage all groups/tags</span>
934
- <span class="cstat-no" title="statement not covered" > can('all', 'groups', { context: organisation._id })</span>
935
- <span class="cstat-no" title="statement not covered" > can(['create', 'remove'], 'authorisations', { resourcesService: organisation._id.toString() + '/groups', scope: 'groups' })</span>
936
- <span class="cstat-no" title="statement not covered" > can('all', 'tags', { context: organisation._id })</span>
937
- <span class="cstat-no" title="statement not covered" > // Remove invited members</span>
938
- <span class="cstat-no" title="statement not covered" > can(['remove'], 'users', { 'sponsor.organisationId': organisation._id })</span>
939
- <span class="cstat-no" title="statement not covered" > }</span>
940
- <span class="cstat-no" title="statement not covered" > }</span>
941
- <span class="cstat-no" title="statement not covered" > })</span>
942
- <span class="cstat-no" title="statement not covered" > }</span>
943
- <span class="cstat-no" title="statement not covered" > }</span>
944
- <span class="cstat-no" title="statement not covered" >}</span>
945
- &nbsp;
946
- // Hook computing group abilities for a given user
947
- export <span class="fstat-no" title="function not covered" >function defineGroupAbilities (subject, can, cannot) {</span>
948
- <span class="cstat-no" title="statement not covered" > if (subject) {</span>
949
- <span class="cstat-no" title="statement not covered" > if (subject.groups) {</span>
950
- <span class="cstat-no" title="statement not covered" > subject.groups.forEach(group =&gt; {</span>
951
- <span class="cstat-no" title="statement not covered" > if (group._id) {</span>
952
- <span class="cstat-no" title="statement not covered" > // Specific rules for groups</span>
953
- <span class="cstat-no" title="statement not covered" > const role = Roles[group.permissions]</span>
954
- <span class="cstat-no" title="statement not covered" ></span>
955
- <span class="cstat-no" title="statement not covered" > if (role &gt;= Roles.member) {</span>
956
- <span class="cstat-no" title="statement not covered" > can('read', 'groups', { _id: group._id })</span>
957
- <span class="cstat-no" title="statement not covered" > }</span>
958
- <span class="cstat-no" title="statement not covered" > if (role &gt;= Roles.manager) {</span>
959
- <span class="cstat-no" title="statement not covered" > can(['create', 'remove'], 'authorisations', { resource: group._id, permissions: 'member' })</span>
960
- <span class="cstat-no" title="statement not covered" > }</span>
961
- <span class="cstat-no" title="statement not covered" > }</span>
962
- <span class="cstat-no" title="statement not covered" > })</span>
963
- <span class="cstat-no" title="statement not covered" > }</span>
964
- <span class="cstat-no" title="statement not covered" > }</span>
965
- <span class="cstat-no" title="statement not covered" >}</span>
966
- &nbsp;
967
- // Helper functions to find the members of a given organisation
968
- export <span class="fstat-no" title="function not covered" >function findMembersOfOrganisation (usersService, organisationId, role) {</span>
969
- <span class="cstat-no" title="statement not covered" > return findSubjectsForResource(usersService, 'organisations', organisationId, role)</span>
970
- <span class="cstat-no" title="statement not covered" >}</span>
971
- &nbsp;
972
- export <span class="fstat-no" title="function not covered" >function countMembersOfOrganisation (usersService, organisationId, role) {</span>
973
- <span class="cstat-no" title="statement not covered" > return countSubjectsForResource(usersService, 'organisations', organisationId, role)</span>
974
- <span class="cstat-no" title="statement not covered" >}</span>
975
- &nbsp;
976
- // Helper functions to find the members of a given group
977
- export <span class="fstat-no" title="function not covered" >function findMembersOfGroup (membersService, groupId, role) {</span>
978
- <span class="cstat-no" title="statement not covered" > return findSubjectsForResource(membersService, 'groups', groupId, role)</span>
979
- <span class="cstat-no" title="statement not covered" >}</span>
980
- &nbsp;
981
- export <span class="fstat-no" title="function not covered" >function countMembersOfGroup (membersService, groupId, role) {</span>
982
- <span class="cstat-no" title="statement not covered" > return countSubjectsForResource(membersService, 'groups', groupId, role)</span>
983
- <span class="cstat-no" title="statement not covered" >}</span>
984
- &nbsp;
985
- // Helper functions to find the members with a given tag
986
- export <span class="fstat-no" title="function not covered" >function findMembersWithTag (membersService, tagId) {</span>
987
- <span class="cstat-no" title="statement not covered" > return findSubjectsForResource(membersService, 'tags', tagId)</span>
988
- <span class="cstat-no" title="statement not covered" >}</span>
989
- &nbsp;
990
- export <span class="fstat-no" title="function not covered" >function countMembersWithTag (membersService, tagId) {</span>
991
- <span class="cstat-no" title="statement not covered" > return countSubjectsForResource(membersService, 'tags', tagId)</span>
992
- <span class="cstat-no" title="statement not covered" >}</span>
993
- &nbsp;
994
- export <span class="fstat-no" title="function not covered" >function getRoleForOrganisation (user, organisationId) {</span>
995
- <span class="cstat-no" title="statement not covered" > const result = _.find(user.organisations, { _id: organisationId })</span>
996
- <span class="cstat-no" title="statement not covered" > if (!_.isUndefined(result)) return result.permissions</span>
997
- <span class="cstat-no" title="statement not covered" > return undefined</span>
998
- <span class="cstat-no" title="statement not covered" >}</span>
999
- &nbsp;
1000
- export <span class="fstat-no" title="function not covered" >function getRoleForGroup (user, organisationId, groupId) {</span>
1001
- <span class="cstat-no" title="statement not covered" > const result = _.find(user.groups, { context: organisationId, _id: groupId })</span>
1002
- <span class="cstat-no" title="statement not covered" > if (!_.isUndefined(result)) return result.permissions</span>
1003
- <span class="cstat-no" title="statement not covered" > return undefined</span>
1004
- <span class="cstat-no" title="statement not covered" >}</span>
1005
- &nbsp;
1006
- export <span class="fstat-no" title="function not covered" >function findGroupsWithRole (user, organisationId, role) {</span>
1007
- <span class="cstat-no" title="statement not covered" > return _.filter(user.groups || [], { context: organisationId, permissions: (typeof role === 'string' ? role : RoleNames[role]) })</span>
1008
- <span class="cstat-no" title="statement not covered" >}</span>
1009
- &nbsp;
1010
704
  export <span class="fstat-no" title="function not covered" >function isSeniorRole (roleName, juniorName) {</span>
1011
705
  <span class="cstat-no" title="statement not covered" > return Roles[roleName] &gt;= Roles[juniorName]</span>
1012
706
  <span class="cstat-no" title="statement not covered" >}</span>
@@ -1033,7 +727,7 @@ export <span class="fstat-no" title="function not covered" >function getJuniorRo
1033
727
  <div class='footer quiet pad2 space-top1 center small'>
1034
728
  Code coverage generated by
1035
729
  <a href="https://istanbul.js.org/" target="_blank" rel="noopener noreferrer">istanbul</a>
1036
- at 2024-08-13T10:02:04.843Z
730
+ at 2026-03-10T09:15:24.743Z
1037
731
  </div>
1038
732
  <script src="../../prettify.js"></script>
1039
733
  <script>