@cccsaurora/howler-ui 2.19.0-dev.842 → 2.19.0-dev.905

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 (328) hide show
  1. package/api/index.d.ts +0 -4
  2. package/api/index.js +2 -10
  3. package/api/search/facet/hit.d.ts +3 -1
  4. package/api/search/facet/index.d.ts +1 -3
  5. package/api/search/index.d.ts +1 -2
  6. package/api/search/index.js +1 -2
  7. package/commons/components/leftnav/LeftNavDrawer.js +1 -1
  8. package/components/app/App.js +8 -41
  9. package/components/app/hooks/useMatchers.d.ts +1 -1
  10. package/components/app/hooks/useMatchers.js +11 -23
  11. package/components/app/hooks/useMatchers.test.js +22 -22
  12. package/components/app/hooks/useTitle.js +5 -5
  13. package/components/app/providers/FavouritesProvider.js +2 -2
  14. package/components/app/providers/HitProvider.d.ts +22 -0
  15. package/components/app/providers/{RecordProvider.js → HitProvider.js} +41 -41
  16. package/components/app/providers/{RecordSearchProvider.d.ts → HitSearchProvider.d.ts} +6 -6
  17. package/components/app/providers/{RecordSearchProvider.js → HitSearchProvider.js} +17 -12
  18. package/components/app/providers/{RecordSearchProvider.test.js → HitSearchProvider.test.js} +71 -52
  19. package/components/app/providers/ModalProvider.d.ts +0 -1
  20. package/components/app/providers/ParameterProvider.d.ts +2 -9
  21. package/components/app/providers/ParameterProvider.js +240 -165
  22. package/components/app/providers/ParameterProvider.test.js +94 -346
  23. package/components/app/providers/SocketProvider.d.ts +2 -11
  24. package/components/app/providers/SocketProvider.js +5 -18
  25. package/components/app/providers/UserListProvider.js +8 -28
  26. package/components/elements/PluginTypography.d.ts +1 -2
  27. package/components/elements/PluginTypography.js +2 -3
  28. package/components/elements/UserList.d.ts +2 -5
  29. package/components/elements/UserList.js +8 -18
  30. package/components/elements/addons/search/phrase/Phrase.js +1 -1
  31. package/components/elements/display/ChipPopper.d.ts +1 -1
  32. package/components/elements/display/ChipPopper.js +5 -5
  33. package/components/elements/display/HowlerCard.js +1 -1
  34. package/components/elements/display/Modal.js +0 -2
  35. package/components/elements/display/icons/BundleButton.d.ts +6 -0
  36. package/components/elements/display/icons/BundleButton.js +32 -0
  37. package/components/elements/hit/HitActions.js +4 -4
  38. package/components/elements/hit/HitBanner.d.ts +0 -1
  39. package/components/elements/hit/HitBanner.js +47 -29
  40. package/components/elements/hit/HitCard.d.ts +0 -2
  41. package/components/elements/hit/HitCard.js +7 -7
  42. package/components/elements/{record/RecordComments.d.ts → hit/HitComments.d.ts} +4 -5
  43. package/components/elements/{record/RecordComments.js → hit/HitComments.js} +28 -29
  44. package/components/elements/{ObjectDetails.js → hit/HitDetails.js} +17 -17
  45. package/components/elements/hit/HitLabels.js +2 -2
  46. package/components/elements/hit/HitOutline.d.ts +0 -1
  47. package/components/elements/hit/HitOutline.js +3 -3
  48. package/components/elements/hit/{HitPreview.d.ts → HitQuickSearch.d.ts} +3 -3
  49. package/components/elements/hit/{HitPreview.js → HitQuickSearch.js} +4 -10
  50. package/components/elements/hit/HitRelated.d.ts +6 -0
  51. package/components/elements/hit/HitRelated.js +7 -0
  52. package/components/elements/hit/HitSummary.d.ts +1 -2
  53. package/components/elements/hit/HitSummary.js +5 -6
  54. package/components/elements/{record/RecordWorklog.d.ts → hit/HitWorklog.d.ts} +3 -4
  55. package/components/elements/{record/RecordWorklog.js → hit/HitWorklog.js} +13 -15
  56. package/components/elements/hit/aggregate/HitGraph.js +8 -8
  57. package/components/elements/hit/elements/Assigned.js +3 -6
  58. package/components/elements/hit/outlines/DefaultOutline.js +1 -1
  59. package/components/elements/view/ViewTitle.d.ts +0 -1
  60. package/components/elements/view/ViewTitle.js +2 -9
  61. package/components/hooks/useHitActions.d.ts +1 -1
  62. package/components/hooks/useHitActions.js +4 -4
  63. package/components/hooks/{useRecordSelection.d.ts → useHitSelection.d.ts} +2 -2
  64. package/components/hooks/{useRecordSelection.js → useHitSelection.js} +33 -12
  65. package/components/hooks/useMyPreferences.js +1 -10
  66. package/components/hooks/useMySearch.js +2 -2
  67. package/components/hooks/useMySitemap.js +1 -4
  68. package/components/hooks/useMyTheme.js +2 -9
  69. package/components/routes/action/edit/ActionEditor.js +2 -2
  70. package/components/routes/action/view/ActionSearch.js +1 -1
  71. package/components/routes/advanced/QueryBuilder.js +1 -1
  72. package/components/routes/advanced/QueryEditor.js +3 -3
  73. package/components/routes/advanced/historyCompletionProvider.js +3 -3
  74. package/components/routes/analytics/AnalyticDetails.js +2 -2
  75. package/components/routes/analytics/AnalyticSearch.js +1 -1
  76. package/components/routes/dossiers/DossierEditor.js +2 -2
  77. package/components/routes/dossiers/DossierEditor.test.js +1 -1
  78. package/components/routes/help/ApiDocumentation.js +1 -1
  79. package/components/routes/help/BundleDocumentation.d.ts +3 -0
  80. package/components/routes/help/BundleDocumentation.js +12 -0
  81. package/components/routes/help/HitBannerDocumentation.js +0 -1
  82. package/components/routes/help/HitDocumentation.js +3 -1
  83. package/components/routes/help/markdown/en/bundles.md.js +1 -0
  84. package/components/routes/help/markdown/fr/bundles.md.js +1 -0
  85. package/components/routes/hits/search/BundleParentMenu.d.ts +6 -0
  86. package/components/routes/hits/search/BundleParentMenu.js +32 -0
  87. package/components/routes/hits/search/BundleScroller.d.ts +2 -0
  88. package/components/routes/hits/search/BundleScroller.js +6 -0
  89. package/components/routes/hits/search/{RecordBrowser.js → HitBrowser.js} +9 -9
  90. package/components/{elements/record/RecordContextMenu.d.ts → routes/hits/search/HitContextMenu.d.ts} +3 -3
  91. package/components/routes/hits/search/HitContextMenu.js +239 -0
  92. package/components/{elements/record/RecordContextMenu.test.js → routes/hits/search/HitContextMenu.test.js} +43 -98
  93. package/components/routes/hits/search/{RecordQuery.d.ts → HitQuery.d.ts} +2 -2
  94. package/components/routes/hits/search/{RecordQuery.js → HitQuery.js} +6 -6
  95. package/components/routes/hits/search/InformationPane.d.ts +0 -1
  96. package/components/routes/hits/search/InformationPane.js +63 -50
  97. package/components/routes/hits/search/LayoutSettings.js +3 -3
  98. package/components/routes/hits/search/QuerySettings.js +1 -2
  99. package/components/routes/hits/search/QuerySettings.test.js +9 -14
  100. package/components/routes/hits/search/SearchPane.js +49 -26
  101. package/components/routes/hits/search/ViewLink.js +3 -3
  102. package/components/routes/hits/search/ViewLink.test.js +8 -8
  103. package/components/routes/hits/search/grid/AddColumnModal.js +4 -5
  104. package/components/routes/hits/search/grid/EnhancedCell.d.ts +1 -2
  105. package/components/routes/hits/search/grid/EnhancedCell.js +2 -2
  106. package/components/routes/hits/search/grid/HitGrid.js +18 -20
  107. package/components/routes/hits/search/grid/{RecordRow.d.ts → HitRow.d.ts} +2 -3
  108. package/components/routes/hits/search/grid/{RecordRow.js → HitRow.js} +8 -10
  109. package/components/routes/hits/view/HitViewer.js +13 -12
  110. package/components/routes/home/ViewCard.js +41 -47
  111. package/components/{elements/MarkdownEditor.js → routes/overviews/OverviewEditor.js} +3 -3
  112. package/components/routes/overviews/OverviewViewer.js +2 -2
  113. package/components/routes/views/ViewComposer.js +19 -46
  114. package/locales/en/translation.json +3 -122
  115. package/locales/fr/translation.json +3 -120
  116. package/models/WithMetadata.d.ts +1 -2
  117. package/models/entities/generated/{ThreatEnrichment.d.ts → Enrichment.d.ts} +1 -1
  118. package/models/entities/generated/Hit.d.ts +0 -1
  119. package/models/entities/generated/Howler.d.ts +5 -0
  120. package/models/entities/generated/Rule.d.ts +9 -6
  121. package/models/entities/generated/Threat.d.ts +2 -2
  122. package/models/entities/generated/View.d.ts +0 -1
  123. package/package.json +121 -141
  124. package/plugins/clue/components/ClueTypography.js +2 -2
  125. package/plugins/clue/utils.d.ts +1 -2
  126. package/tests/mocks.d.ts +1 -11
  127. package/tests/mocks.js +7 -12
  128. package/tests/server-handlers.js +1 -6
  129. package/tests/utils.d.ts +0 -4
  130. package/tests/utils.js +0 -20
  131. package/utils/constants.d.ts +3 -4
  132. package/utils/constants.js +0 -6
  133. package/utils/hitFunctions.d.ts +1 -2
  134. package/utils/hitFunctions.js +4 -4
  135. package/utils/socketUtils.d.ts +0 -14
  136. package/utils/socketUtils.js +1 -17
  137. package/utils/viewUtils.js +0 -3
  138. package/api/search/case.d.ts +0 -4
  139. package/api/search/case.js +0 -8
  140. package/api/socket/index.d.ts +0 -3
  141. package/api/socket/index.js +0 -6
  142. package/api/socket/viewers.d.ts +0 -2
  143. package/api/socket/viewers.js +0 -8
  144. package/api/socket/viewers.test.js +0 -44
  145. package/api/v2/case/index.d.ts +0 -9
  146. package/api/v2/case/index.js +0 -21
  147. package/api/v2/case/items.d.ts +0 -6
  148. package/api/v2/case/items.js +0 -18
  149. package/api/v2/case/rules.d.ts +0 -6
  150. package/api/v2/case/rules.js +0 -18
  151. package/api/v2/index.d.ts +0 -4
  152. package/api/v2/index.js +0 -6
  153. package/api/v2/search/facet.d.ts +0 -3
  154. package/api/v2/search/facet.js +0 -12
  155. package/api/v2/search/index.d.ts +0 -5
  156. package/api/v2/search/index.js +0 -24
  157. package/components/app/providers/RecordProvider.d.ts +0 -23
  158. package/components/elements/ContextMenu.d.ts +0 -56
  159. package/components/elements/ContextMenu.js +0 -109
  160. package/components/elements/ContextMenu.test.d.ts +0 -1
  161. package/components/elements/ContextMenu.test.js +0 -215
  162. package/components/elements/ObjectDetails.d.ts +0 -6
  163. package/components/elements/case/CaseCard.d.ts +0 -12
  164. package/components/elements/case/CaseCard.js +0 -42
  165. package/components/elements/case/CasePreview.d.ts +0 -6
  166. package/components/elements/case/CasePreview.js +0 -17
  167. package/components/elements/case/StatusIcon.d.ts +0 -5
  168. package/components/elements/case/StatusIcon.js +0 -13
  169. package/components/elements/hit/elements/AnalyticLink.d.ts +0 -9
  170. package/components/elements/hit/elements/AnalyticLink.js +0 -22
  171. package/components/elements/hit/elements/Assigned.test.d.ts +0 -1
  172. package/components/elements/hit/elements/Assigned.test.js +0 -65
  173. package/components/elements/hit/related/RelatedRecords.js +0 -63
  174. package/components/elements/observable/ObservableCard.d.ts +0 -6
  175. package/components/elements/observable/ObservableCard.js +0 -22
  176. package/components/elements/observable/ObservablePreview.d.ts +0 -6
  177. package/components/elements/observable/ObservablePreview.js +0 -12
  178. package/components/elements/record/RecordContextMenu.js +0 -268
  179. package/components/elements/record/RecordContextMenu.test.d.ts +0 -1
  180. package/components/elements/record/RecordRelated.d.ts +0 -7
  181. package/components/elements/record/RecordRelated.js +0 -34
  182. package/components/hooks/useRelatedRecords.d.ts +0 -13
  183. package/components/hooks/useRelatedRecords.js +0 -32
  184. package/components/routes/cases/CaseViewer.d.ts +0 -2
  185. package/components/routes/cases/CaseViewer.js +0 -44
  186. package/components/routes/cases/CaseViewer.test.d.ts +0 -1
  187. package/components/routes/cases/CaseViewer.test.js +0 -133
  188. package/components/routes/cases/Cases.d.ts +0 -2
  189. package/components/routes/cases/Cases.js +0 -148
  190. package/components/routes/cases/constants.d.ts +0 -6
  191. package/components/routes/cases/constants.js +0 -6
  192. package/components/routes/cases/detail/AlertPanel.d.ts +0 -6
  193. package/components/routes/cases/detail/AlertPanel.js +0 -33
  194. package/components/routes/cases/detail/CaseAssets.d.ts +0 -11
  195. package/components/routes/cases/detail/CaseAssets.js +0 -104
  196. package/components/routes/cases/detail/CaseAssets.test.d.ts +0 -1
  197. package/components/routes/cases/detail/CaseAssets.test.js +0 -167
  198. package/components/routes/cases/detail/CaseDashboard.d.ts +0 -7
  199. package/components/routes/cases/detail/CaseDashboard.js +0 -66
  200. package/components/routes/cases/detail/CaseDetails.d.ts +0 -6
  201. package/components/routes/cases/detail/CaseDetails.js +0 -70
  202. package/components/routes/cases/detail/CaseOverview.d.ts +0 -7
  203. package/components/routes/cases/detail/CaseOverview.js +0 -43
  204. package/components/routes/cases/detail/CaseRules.d.ts +0 -7
  205. package/components/routes/cases/detail/CaseRules.js +0 -57
  206. package/components/routes/cases/detail/CaseRules.test.d.ts +0 -1
  207. package/components/routes/cases/detail/CaseRules.test.js +0 -221
  208. package/components/routes/cases/detail/CaseSidebar.d.ts +0 -8
  209. package/components/routes/cases/detail/CaseSidebar.js +0 -107
  210. package/components/routes/cases/detail/CaseSidebar.test.d.ts +0 -1
  211. package/components/routes/cases/detail/CaseSidebar.test.js +0 -266
  212. package/components/routes/cases/detail/CaseTask.d.ts +0 -11
  213. package/components/routes/cases/detail/CaseTask.js +0 -66
  214. package/components/routes/cases/detail/CaseTimeline.d.ts +0 -12
  215. package/components/routes/cases/detail/CaseTimeline.js +0 -106
  216. package/components/routes/cases/detail/CaseTimeline.test.d.ts +0 -1
  217. package/components/routes/cases/detail/CaseTimeline.test.js +0 -320
  218. package/components/routes/cases/detail/CreateRuleDialog.d.ts +0 -9
  219. package/components/routes/cases/detail/CreateRuleDialog.js +0 -163
  220. package/components/routes/cases/detail/CreateRuleDialog.test.d.ts +0 -1
  221. package/components/routes/cases/detail/CreateRuleDialog.test.js +0 -259
  222. package/components/routes/cases/detail/ItemPage.d.ts +0 -6
  223. package/components/routes/cases/detail/ItemPage.js +0 -95
  224. package/components/routes/cases/detail/RelatedCasePanel.d.ts +0 -6
  225. package/components/routes/cases/detail/RelatedCasePanel.js +0 -34
  226. package/components/routes/cases/detail/TaskPanel.d.ts +0 -7
  227. package/components/routes/cases/detail/TaskPanel.js +0 -52
  228. package/components/routes/cases/detail/aggregates/CaseAggregate.d.ts +0 -11
  229. package/components/routes/cases/detail/aggregates/CaseAggregate.js +0 -24
  230. package/components/routes/cases/detail/aggregates/SourceAggregate.d.ts +0 -6
  231. package/components/routes/cases/detail/aggregates/SourceAggregate.js +0 -26
  232. package/components/routes/cases/detail/assets/Asset.d.ts +0 -14
  233. package/components/routes/cases/detail/assets/Asset.js +0 -12
  234. package/components/routes/cases/detail/assets/Asset.test.d.ts +0 -1
  235. package/components/routes/cases/detail/assets/Asset.test.js +0 -72
  236. package/components/routes/cases/detail/sidebar/CaseFolder.d.ts +0 -20
  237. package/components/routes/cases/detail/sidebar/CaseFolder.js +0 -83
  238. package/components/routes/cases/detail/sidebar/CaseFolder.test.d.ts +0 -1
  239. package/components/routes/cases/detail/sidebar/CaseFolder.test.js +0 -295
  240. package/components/routes/cases/detail/sidebar/CaseFolderContextMenu.d.ts +0 -34
  241. package/components/routes/cases/detail/sidebar/CaseFolderContextMenu.js +0 -103
  242. package/components/routes/cases/detail/sidebar/CaseFolderContextMenu.test.d.ts +0 -1
  243. package/components/routes/cases/detail/sidebar/CaseFolderContextMenu.test.js +0 -363
  244. package/components/routes/cases/detail/sidebar/FolderEntry.d.ts +0 -25
  245. package/components/routes/cases/detail/sidebar/FolderEntry.js +0 -88
  246. package/components/routes/cases/detail/sidebar/FolderEntry.test.d.ts +0 -1
  247. package/components/routes/cases/detail/sidebar/FolderEntry.test.js +0 -206
  248. package/components/routes/cases/detail/sidebar/RootDropZone.d.ts +0 -5
  249. package/components/routes/cases/detail/sidebar/RootDropZone.js +0 -33
  250. package/components/routes/cases/detail/sidebar/types.d.ts +0 -9
  251. package/components/routes/cases/detail/sidebar/utils.d.ts +0 -3
  252. package/components/routes/cases/detail/sidebar/utils.js +0 -29
  253. package/components/routes/cases/detail/sidebar/utils.test.d.ts +0 -1
  254. package/components/routes/cases/detail/sidebar/utils.test.js +0 -82
  255. package/components/routes/cases/hooks/useCase.d.ts +0 -13
  256. package/components/routes/cases/hooks/useCase.js +0 -69
  257. package/components/routes/cases/hooks/useCase.test.d.ts +0 -1
  258. package/components/routes/cases/hooks/useCase.test.js +0 -141
  259. package/components/routes/cases/modals/AddToCaseModal.d.ts +0 -7
  260. package/components/routes/cases/modals/AddToCaseModal.js +0 -59
  261. package/components/routes/cases/modals/AddToCaseModal.test.d.ts +0 -1
  262. package/components/routes/cases/modals/AddToCaseModal.test.js +0 -313
  263. package/components/routes/cases/modals/CaseRecordRow.d.ts +0 -9
  264. package/components/routes/cases/modals/CaseRecordRow.js +0 -15
  265. package/components/routes/cases/modals/CreateCaseModal.d.ts +0 -7
  266. package/components/routes/cases/modals/CreateCaseModal.js +0 -55
  267. package/components/routes/cases/modals/CreateCaseModal.test.d.ts +0 -1
  268. package/components/routes/cases/modals/CreateCaseModal.test.js +0 -358
  269. package/components/routes/cases/modals/RenameItemModal.d.ts +0 -9
  270. package/components/routes/cases/modals/RenameItemModal.js +0 -48
  271. package/components/routes/cases/modals/ResolveModal.d.ts +0 -7
  272. package/components/routes/cases/modals/ResolveModal.js +0 -115
  273. package/components/routes/cases/modals/ResolveModal.test.d.ts +0 -1
  274. package/components/routes/cases/modals/ResolveModal.test.js +0 -394
  275. package/components/routes/cases/modals/hooks.d.ts +0 -7
  276. package/components/routes/cases/modals/hooks.js +0 -44
  277. package/components/routes/cases/modals/types.d.ts +0 -5
  278. package/components/routes/cases/search/CaseAssigneeFilter.d.ts +0 -6
  279. package/components/routes/cases/search/CaseAssigneeFilter.js +0 -33
  280. package/components/routes/cases/search/CaseAssigneeFilter.test.d.ts +0 -1
  281. package/components/routes/cases/search/CaseAssigneeFilter.test.js +0 -127
  282. package/components/routes/cases/search/CaseDateFilter.d.ts +0 -13
  283. package/components/routes/cases/search/CaseDateFilter.js +0 -26
  284. package/components/routes/cases/search/CaseDateFilter.test.d.ts +0 -1
  285. package/components/routes/cases/search/CaseDateFilter.test.js +0 -115
  286. package/components/routes/cases/search/CaseStatusFilter.d.ts +0 -6
  287. package/components/routes/cases/search/CaseStatusFilter.js +0 -13
  288. package/components/routes/cases/search/CaseStatusFilter.test.d.ts +0 -1
  289. package/components/routes/cases/search/CaseStatusFilter.test.js +0 -86
  290. package/components/routes/hits/search/shared/IndexPicker.d.ts +0 -2
  291. package/components/routes/hits/search/shared/IndexPicker.js +0 -20
  292. package/components/routes/observables/ObservableViewer.d.ts +0 -7
  293. package/components/routes/observables/ObservableViewer.js +0 -27
  294. package/models/entities/generated/AttachmentsFile.d.ts +0 -12
  295. package/models/entities/generated/Case.d.ts +0 -28
  296. package/models/entities/generated/DestinationOriginal.d.ts +0 -19
  297. package/models/entities/generated/EmailAttachment.d.ts +0 -8
  298. package/models/entities/generated/EmailParent.d.ts +0 -19
  299. package/models/entities/generated/Enrichments.d.ts +0 -7
  300. package/models/entities/generated/EnrichmentsIndicator.d.ts +0 -21
  301. package/models/entities/generated/HttpResponse.d.ts +0 -11
  302. package/models/entities/generated/Item.d.ts +0 -9
  303. package/models/entities/generated/Observable.d.ts +0 -85
  304. package/models/entities/generated/ObservableCloud.d.ts +0 -20
  305. package/models/entities/generated/ObservableDestination.d.ts +0 -23
  306. package/models/entities/generated/ObservableEmail.d.ts +0 -30
  307. package/models/entities/generated/ObservableFile.d.ts +0 -36
  308. package/models/entities/generated/ObservableHowler.d.ts +0 -42
  309. package/models/entities/generated/ObservableHttp.d.ts +0 -11
  310. package/models/entities/generated/ObservableObserver.d.ts +0 -21
  311. package/models/entities/generated/ObservableOrganization.d.ts +0 -7
  312. package/models/entities/generated/ObservableProcess.d.ts +0 -34
  313. package/models/entities/generated/ObservableSource.d.ts +0 -23
  314. package/models/entities/generated/ObservableThreat.d.ts +0 -21
  315. package/models/entities/generated/ObservableTls.d.ts +0 -12
  316. package/models/entities/generated/ObserverIngress.d.ts +0 -9
  317. package/models/entities/generated/Task.d.ts +0 -10
  318. package/models/socket/CaseUpdate.d.ts +0 -5
  319. package/models/socket/ViewersUpdate.d.ts +0 -4
  320. package/utils/socketUtils.test.d.ts +0 -1
  321. package/utils/socketUtils.test.js +0 -59
  322. package/utils/typeUtils.d.ts +0 -7
  323. package/utils/typeUtils.js +0 -27
  324. /package/{api/socket/viewers.test.d.ts → components/app/providers/HitSearchProvider.test.d.ts} +0 -0
  325. /package/components/elements/hit/{related/RelatedRecords.d.ts → HitDetails.d.ts} +0 -0
  326. /package/components/routes/hits/search/{RecordBrowser.d.ts → HitBrowser.d.ts} +0 -0
  327. /package/components/{app/providers/RecordSearchProvider.test.d.ts → routes/hits/search/HitContextMenu.test.d.ts} +0 -0
  328. /package/components/{elements/MarkdownEditor.d.ts → routes/overviews/OverviewEditor.d.ts} +0 -0
@@ -1,259 +0,0 @@
1
- import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
2
- import { act, render, screen, waitFor } from '@testing-library/react';
3
- import userEvent from '@testing-library/user-event';
4
- import { describe, expect, it, vi } from 'vitest';
5
- globalThis.IS_REACT_ACT_ENVIRONMENT = true;
6
- const mockDispatchApi = vi.hoisted(() => vi.fn());
7
- vi.mock('react-i18next', () => ({
8
- useTranslation: () => ({ t: (key) => key, i18n: { language: 'en' } })
9
- }));
10
- vi.mock('components/hooks/useMyApi', () => ({
11
- default: () => ({ dispatchApi: mockDispatchApi })
12
- }));
13
- vi.mock('api', () => ({
14
- default: {
15
- search: {
16
- hit: {
17
- post: vi.fn()
18
- }
19
- }
20
- }
21
- }));
22
- vi.mock('components/routes/advanced/QueryEditor', () => ({
23
- default: ({ query, setQuery, id }) => (_jsx("textarea", { id: id ?? 'query-editor', value: query, onChange: e => setQuery(e.target.value) }))
24
- }));
25
- vi.mock('components/elements/display/QueryResultText', () => ({
26
- default: ({ count, query }) => (_jsxs("div", { id: "query-result-text", children: [count, " results for ", query] }))
27
- }));
28
- vi.mock('@monaco-editor/react', () => ({
29
- useMonaco: () => null
30
- }));
31
- vi.mock('@mui/x-date-pickers', () => ({
32
- LocalizationProvider: ({ children }) => _jsx(_Fragment, { children: children })
33
- }));
34
- vi.mock('@mui/x-date-pickers/AdapterDayjs', () => ({
35
- AdapterDayjs: class {
36
- }
37
- }));
38
- vi.mock('@mui/x-date-pickers/DateTimePicker', () => ({
39
- DateTimePicker: ({ label, disabled }) => (_jsx("input", { id: "rule-timeframe-input", "aria-label": label, disabled: disabled, readOnly: true }))
40
- }));
41
- import api from '@cccsaurora/howler-ui/api';
42
- import CreateRuleDialog from './CreateRuleDialog';
43
- describe('CreateRuleDialog', () => {
44
- const defaultProps = {
45
- open: true,
46
- onClose: vi.fn(),
47
- onSubmit: vi.fn().mockResolvedValue(undefined)
48
- };
49
- beforeEach(() => {
50
- vi.clearAllMocks();
51
- });
52
- it('renders nothing when closed', () => {
53
- render(_jsx(CreateRuleDialog, { ...defaultProps, open: false }));
54
- expect(screen.queryByTestId('create-rule-dialog')).not.toBeInTheDocument();
55
- });
56
- it('renders dialog fields when open', () => {
57
- render(_jsx(CreateRuleDialog, { ...defaultProps }));
58
- expect(screen.getByTestId('create-rule-dialog')).toBeInTheDocument();
59
- expect(screen.getByTestId('rule-query-editor')).toBeInTheDocument();
60
- expect(screen.getByTestId('rule-destination-input')).toBeInTheDocument();
61
- expect(screen.getByTestId('rule-timeframe-input')).toBeInTheDocument();
62
- expect(screen.getByTestId('rule-no-expiry-checkbox')).toBeInTheDocument();
63
- expect(screen.getByTestId('rule-index-hit')).toBeInTheDocument();
64
- expect(screen.getByTestId('rule-index-observable')).toBeInTheDocument();
65
- });
66
- it('disables submit when query is empty', () => {
67
- render(_jsx(CreateRuleDialog, { ...defaultProps }));
68
- expect(screen.getByTestId('rule-submit-button')).toBeDisabled();
69
- });
70
- it('disables submit when destination is empty', async () => {
71
- const user = userEvent.setup();
72
- render(_jsx(CreateRuleDialog, { ...defaultProps }));
73
- await act(async () => {
74
- await user.type(screen.getByTestId('rule-query-editor'), 'event.kind:alert');
75
- });
76
- expect(screen.getByTestId('rule-submit-button')).toBeDisabled();
77
- });
78
- it('enables submit when query, destination are filled and search has been run', async () => {
79
- const user = userEvent.setup();
80
- mockDispatchApi.mockResolvedValueOnce({ items: [], total: 5, offset: 0, rows: 0 });
81
- api.search.hit.post.mockReturnValue('search-request');
82
- render(_jsx(CreateRuleDialog, { ...defaultProps }));
83
- await act(async () => {
84
- await user.type(screen.getByTestId('rule-query-editor'), 'event.kind:alert');
85
- });
86
- await act(async () => {
87
- await user.click(screen.getByTestId('rule-search-button'));
88
- });
89
- await act(async () => {
90
- await user.type(screen.getByTestId('rule-destination-input'), 'alerts/incoming');
91
- });
92
- await waitFor(() => {
93
- expect(screen.getByTestId('rule-submit-button')).not.toBeDisabled();
94
- });
95
- });
96
- it('calls onSubmit with rule data when submitted', async () => {
97
- const user = userEvent.setup();
98
- const onSubmit = vi.fn().mockResolvedValue(undefined);
99
- mockDispatchApi.mockResolvedValueOnce({ items: [], total: 1, offset: 0, rows: 0 });
100
- api.search.hit.post.mockReturnValue('search-request');
101
- render(_jsx(CreateRuleDialog, { ...defaultProps, onSubmit: onSubmit }));
102
- await act(async () => {
103
- await user.type(screen.getByTestId('rule-query-editor'), 'event.kind:alert');
104
- });
105
- await act(async () => {
106
- await user.click(screen.getByTestId('rule-search-button'));
107
- });
108
- await act(async () => {
109
- await user.type(screen.getByTestId('rule-destination-input'), 'alerts/incoming');
110
- });
111
- await act(async () => {
112
- await user.click(screen.getByTestId('rule-submit-button'));
113
- });
114
- await waitFor(() => {
115
- expect(onSubmit).toHaveBeenCalledWith(expect.objectContaining({
116
- query: 'event.kind:alert',
117
- destination: 'alerts/incoming',
118
- indexes: ['hit']
119
- }));
120
- });
121
- });
122
- it('calls onClose when cancel is clicked', async () => {
123
- const user = userEvent.setup();
124
- const onClose = vi.fn();
125
- render(_jsx(CreateRuleDialog, { ...defaultProps, onClose: onClose }));
126
- await act(async () => {
127
- await user.click(screen.getByText('cancel'));
128
- });
129
- expect(onClose).toHaveBeenCalled();
130
- });
131
- it('shows search results after triggering search', async () => {
132
- const user = userEvent.setup();
133
- mockDispatchApi.mockResolvedValueOnce({ items: [], total: 42, offset: 0, rows: 0 });
134
- api.search.hit.post.mockReturnValue('search-request');
135
- render(_jsx(CreateRuleDialog, { ...defaultProps }));
136
- await act(async () => {
137
- await user.type(screen.getByTestId('rule-query-editor'), 'event.kind:alert');
138
- });
139
- await act(async () => {
140
- await user.click(screen.getByTestId('rule-search-button'));
141
- });
142
- await waitFor(() => {
143
- expect(screen.getByTestId('query-result-text')).toBeInTheDocument();
144
- expect(screen.getByText(/42 results/)).toBeInTheDocument();
145
- });
146
- });
147
- it('shows prompt text before search is triggered', () => {
148
- render(_jsx(CreateRuleDialog, { ...defaultProps }));
149
- expect(screen.getByText('hit.search.prompt')).toBeInTheDocument();
150
- expect(screen.queryByTestId('query-result-text')).not.toBeInTheDocument();
151
- });
152
- it('includes timeframe as ISO string when no expiry is unchecked', async () => {
153
- const user = userEvent.setup();
154
- const onSubmit = vi.fn().mockResolvedValue(undefined);
155
- mockDispatchApi.mockResolvedValueOnce({ items: [], total: 1, offset: 0, rows: 0 });
156
- api.search.hit.post.mockReturnValue('search-request');
157
- render(_jsx(CreateRuleDialog, { ...defaultProps, onSubmit: onSubmit }));
158
- await act(async () => {
159
- await user.type(screen.getByTestId('rule-query-editor'), 'event.kind:alert');
160
- });
161
- await act(async () => {
162
- await user.click(screen.getByTestId('rule-search-button'));
163
- });
164
- await act(async () => {
165
- await user.type(screen.getByTestId('rule-destination-input'), 'alerts/incoming');
166
- });
167
- await act(async () => {
168
- await user.click(screen.getByTestId('rule-submit-button'));
169
- });
170
- await waitFor(() => {
171
- expect(onSubmit).toHaveBeenCalledWith(expect.objectContaining({
172
- timeframe: expect.any(String)
173
- }));
174
- });
175
- });
176
- it('omits timeframe when no expiry is checked', async () => {
177
- const user = userEvent.setup();
178
- const onSubmit = vi.fn().mockResolvedValue(undefined);
179
- mockDispatchApi.mockResolvedValueOnce({ items: [], total: 1, offset: 0, rows: 0 });
180
- api.search.hit.post.mockReturnValue('search-request');
181
- render(_jsx(CreateRuleDialog, { ...defaultProps, onSubmit: onSubmit }));
182
- await act(async () => {
183
- await user.click(screen.getByTestId('rule-no-expiry-checkbox'));
184
- });
185
- await act(async () => {
186
- await user.type(screen.getByTestId('rule-query-editor'), 'event.kind:alert');
187
- });
188
- await act(async () => {
189
- await user.click(screen.getByTestId('rule-search-button'));
190
- });
191
- await act(async () => {
192
- await user.type(screen.getByTestId('rule-destination-input'), 'alerts/incoming');
193
- });
194
- await act(async () => {
195
- await user.click(screen.getByTestId('rule-submit-button'));
196
- });
197
- await waitFor(() => {
198
- expect(onSubmit).toHaveBeenCalledWith(expect.objectContaining({
199
- query: 'event.kind:alert',
200
- destination: 'alerts/incoming',
201
- timeframe: undefined
202
- }));
203
- });
204
- });
205
- it('defaults to hit index checked', () => {
206
- render(_jsx(CreateRuleDialog, { ...defaultProps }));
207
- const hitCheckbox = screen.getByTestId('rule-index-hit');
208
- const observableCheckbox = screen.getByTestId('rule-index-observable');
209
- expect(hitCheckbox).toBeChecked();
210
- expect(observableCheckbox).not.toBeChecked();
211
- });
212
- it('includes both indexes when both are selected', async () => {
213
- const user = userEvent.setup();
214
- const onSubmit = vi.fn().mockResolvedValue(undefined);
215
- mockDispatchApi.mockResolvedValueOnce({ items: [], total: 1, offset: 0, rows: 0 });
216
- api.search.hit.post.mockReturnValue('search-request');
217
- render(_jsx(CreateRuleDialog, { ...defaultProps, onSubmit: onSubmit }));
218
- await act(async () => {
219
- await user.click(screen.getByTestId('rule-index-observable'));
220
- });
221
- await act(async () => {
222
- await user.type(screen.getByTestId('rule-query-editor'), 'event.kind:alert');
223
- });
224
- await act(async () => {
225
- await user.click(screen.getByTestId('rule-search-button'));
226
- });
227
- await act(async () => {
228
- await user.type(screen.getByTestId('rule-destination-input'), 'alerts/incoming');
229
- });
230
- await act(async () => {
231
- await user.click(screen.getByTestId('rule-submit-button'));
232
- });
233
- await waitFor(() => {
234
- expect(onSubmit).toHaveBeenCalledWith(expect.objectContaining({
235
- indexes: expect.arrayContaining(['hit', 'observable'])
236
- }));
237
- });
238
- });
239
- it('disables submit when no indexes are selected', async () => {
240
- const user = userEvent.setup();
241
- mockDispatchApi.mockResolvedValueOnce({ items: [], total: 1, offset: 0, rows: 0 });
242
- api.search.hit.post.mockReturnValue('search-request');
243
- render(_jsx(CreateRuleDialog, { ...defaultProps }));
244
- await act(async () => {
245
- await user.type(screen.getByTestId('rule-query-editor'), 'event.kind:alert');
246
- });
247
- await act(async () => {
248
- await user.click(screen.getByTestId('rule-search-button'));
249
- });
250
- await act(async () => {
251
- await user.type(screen.getByTestId('rule-destination-input'), 'alerts/incoming');
252
- });
253
- // Uncheck the default hit index
254
- await act(async () => {
255
- await user.click(screen.getByTestId('rule-index-hit'));
256
- });
257
- expect(screen.getByTestId('rule-submit-button')).toBeDisabled();
258
- });
259
- });
@@ -1,6 +0,0 @@
1
- import type { Case } from '@cccsaurora/howler-ui/models/entities/generated/Case';
2
- import { type FC } from 'react';
3
- declare const ItemPage: FC<{
4
- case?: Case;
5
- }>;
6
- export default ItemPage;
@@ -1,95 +0,0 @@
1
- import { jsx as _jsx } from "react/jsx-runtime";
2
- import api from '@cccsaurora/howler-ui/api';
3
- import useMyApi from '@cccsaurora/howler-ui/components/hooks/useMyApi';
4
- import NotFoundPage from '@cccsaurora/howler-ui/components/routes/404';
5
- import InformationPane from '@cccsaurora/howler-ui/components/routes/hits/search/InformationPane';
6
- import { useEffect, useMemo, useState } from 'react';
7
- import { useOutletContext, useParams } from 'react-router-dom';
8
- import useCase from '../hooks/useCase';
9
- import CaseDashboard from './CaseDashboard';
10
- const ItemPage = ({ case: providedCase }) => {
11
- const params = useParams();
12
- const routeCase = useOutletContext();
13
- const { case: fetchedCase } = useCase({ caseId: !providedCase && !routeCase ? params.id : undefined });
14
- const _case = providedCase ?? routeCase ?? fetchedCase;
15
- const { dispatchApi } = useMyApi();
16
- const [item, setItem] = useState(null);
17
- const [loading, setLoading] = useState(true);
18
- // When rendered as a child route, the wildcard segment is in params['*'].
19
- // When rendered directly with a case prop, fall back to parsing the pathname.
20
- const subPath = params['*'] ?? '';
21
- const normalizedSubPath = useMemo(() => subPath.replace(/^\/+|\/+$/g, ''), [subPath]);
22
- useEffect(() => {
23
- let cancelled = false;
24
- const resolveItem = async () => {
25
- setLoading(true);
26
- if (!normalizedSubPath) {
27
- if (!cancelled) {
28
- setItem(null);
29
- setLoading(false);
30
- }
31
- return;
32
- }
33
- let currentCase = _case;
34
- let remainingPath = normalizedSubPath;
35
- while (currentCase && remainingPath) {
36
- const currentRemainingPath = remainingPath;
37
- const matchedNestedCase = currentCase.items
38
- .filter(_item => _item?.path &&
39
- _item?.type?.toLowerCase() === 'case' &&
40
- (currentRemainingPath === _item.path || currentRemainingPath.startsWith(`${_item.path}/`)))
41
- .sort((a, b) => (b.path?.length || 0) - (a.path?.length || 0))[0];
42
- if (!matchedNestedCase) {
43
- break;
44
- }
45
- if (currentRemainingPath === matchedNestedCase.path) {
46
- if (!cancelled) {
47
- setItem(matchedNestedCase);
48
- setLoading(false);
49
- }
50
- return;
51
- }
52
- if (!matchedNestedCase.value) {
53
- if (!cancelled) {
54
- setItem(null);
55
- setLoading(false);
56
- }
57
- return;
58
- }
59
- const nextCase = await dispatchApi(api.v2.case.get(matchedNestedCase.value), { throwError: false });
60
- if (!nextCase) {
61
- if (!cancelled) {
62
- setItem(null);
63
- setLoading(false);
64
- }
65
- return;
66
- }
67
- remainingPath = currentRemainingPath.slice((matchedNestedCase.path?.length || 0) + 1);
68
- currentCase = nextCase;
69
- }
70
- const resolvedItem = currentCase?.items?.find(_item => _item.path === remainingPath);
71
- if (!cancelled) {
72
- setItem(resolvedItem || null);
73
- setLoading(false);
74
- }
75
- };
76
- resolveItem();
77
- return () => {
78
- cancelled = true;
79
- };
80
- }, [_case, dispatchApi, normalizedSubPath]);
81
- if (loading) {
82
- return null;
83
- }
84
- if (!item) {
85
- return _jsx(NotFoundPage, {});
86
- }
87
- if (item.type === 'hit' || item.type === 'observable') {
88
- return _jsx(InformationPane, { selected: item.value });
89
- }
90
- if (item.type === 'case') {
91
- return _jsx(CaseDashboard, { caseId: item.value });
92
- }
93
- return _jsx("h1", { children: JSON.stringify(item) });
94
- };
95
- export default ItemPage;
@@ -1,6 +0,0 @@
1
- import type { Case } from '@cccsaurora/howler-ui/models/entities/generated/Case';
2
- import { type FC } from 'react';
3
- declare const RelatedCasePanel: FC<{
4
- case: Case;
5
- }>;
6
- export default RelatedCasePanel;
@@ -1,34 +0,0 @@
1
- import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
- import { Box, Divider, Pagination, Skeleton, Stack, Typography, useTheme } from '@mui/material';
3
- import { chunk, uniq } from 'lodash-es';
4
- import { useMemo, useState } from 'react';
5
- import { useTranslation } from 'react-i18next';
6
- import { Link } from 'react-router-dom';
7
- import CaseCard from '../../../elements/case/CaseCard';
8
- const RelatedCasePanel = ({ case: _case }) => {
9
- const { t } = useTranslation();
10
- const theme = useTheme();
11
- const [casePage, setCasePage] = useState(1);
12
- const casePages = useMemo(() => chunk(uniq((_case?.items ?? []).filter(item => item.type === 'case')), 5), [_case?.items]);
13
- if (!_case) {
14
- return _jsx(Skeleton, { height: 240 });
15
- }
16
- if (casePages.length < 1) {
17
- return null;
18
- }
19
- return (_jsxs(Stack, { spacing: 1, children: [_jsxs(Stack, { direction: "row", children: [_jsx(Typography, { flex: 1, variant: "h4", children: t('page.cases.dashboard.cases') }), _jsx(Pagination, { count: casePages.length, page: casePage, onChange: (_, page) => setCasePage(page) })] }), _jsx(Divider, {}), casePages[casePage - 1]?.map(item => (_jsxs(Box, { position: "relative", children: [_jsx(CaseCard, { caseId: item.value }), _jsx(Box, { component: Link, to: item.path, sx: {
20
- position: 'absolute',
21
- top: 0,
22
- left: 0,
23
- width: '100%',
24
- height: '100%',
25
- cursor: 'pointer',
26
- zIndex: 100,
27
- borderRadius: '4px',
28
- '&:hover': {
29
- background: theme.palette.divider,
30
- border: `thin solid ${theme.palette.primary.light}`
31
- }
32
- } })] }, item.path)))] }));
33
- };
34
- export default RelatedCasePanel;
@@ -1,7 +0,0 @@
1
- import type { Case } from '@cccsaurora/howler-ui/models/entities/generated/Case';
2
- import { type FC } from 'react';
3
- declare const TaskPanel: FC<{
4
- case: Case;
5
- updateCase: (_case: Partial<Case>) => Promise<void>;
6
- }>;
7
- export default TaskPanel;
@@ -1,52 +0,0 @@
1
- import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
- import { Add } from '@mui/icons-material';
3
- import { Divider, Skeleton, Stack, Typography } from '@mui/material';
4
- import { useState } from 'react';
5
- import { useTranslation } from 'react-i18next';
6
- import CaseTask from './CaseTask';
7
- const TaskPanel = ({ case: _case, updateCase }) => {
8
- const { t } = useTranslation();
9
- const [addingTask, setAddingTask] = useState(false);
10
- const onEdit = (task) => async (newTask) => {
11
- if (task) {
12
- await updateCase({
13
- tasks: _case.tasks.map(_task => {
14
- if (_task.id !== task.id) {
15
- return _task;
16
- }
17
- return {
18
- ..._task,
19
- ...newTask
20
- };
21
- })
22
- });
23
- }
24
- else {
25
- await updateCase({
26
- tasks: [..._case.tasks, newTask]
27
- });
28
- }
29
- };
30
- if (!_case) {
31
- return _jsx(Skeleton, { height: 240 });
32
- }
33
- return (_jsxs(Stack, { spacing: 1, children: [_jsx(Typography, { flex: 1, variant: "h4", children: t('page.cases.dashboard.tasks') }), _jsx(Divider, {}), _case.tasks.map(task => (_jsx(CaseTask, { task: task, paths: _case.items.map(item => item.path), onEdit: onEdit(task), onDelete: () => updateCase({ tasks: _case.tasks.filter(_task => _task.id !== task.id) }) }, task.id))), addingTask && (_jsx(CaseTask, { newTask: true, paths: _case.items.map(item => item.path), onEdit: async (task) => {
34
- await onEdit()(task);
35
- setAddingTask(false);
36
- }, onDelete: async () => setAddingTask(false) })), _jsxs(Stack, { onClick: () => setAddingTask(true), direction: "row", spacing: 2, sx: theme => ({
37
- borderStyle: 'dashed',
38
- borderColor: theme.palette.text.secondary,
39
- borderWidth: '0.15rem',
40
- borderRadius: '0.15rem',
41
- opacity: 0.3,
42
- justifyContent: 'center',
43
- alignItems: 'center',
44
- padding: 1,
45
- transition: theme.transitions.create('opacity'),
46
- '&:hover': {
47
- opacity: 1,
48
- cursor: 'pointer'
49
- }
50
- }), children: [_jsx(Add, {}), _jsx(Typography, { children: t('page.cases.dashboard.tasks.add') })] })] }));
51
- };
52
- export default TaskPanel;
@@ -1,11 +0,0 @@
1
- import type { Hit } from '@cccsaurora/howler-ui/models/entities/generated/Hit';
2
- import type { Observable } from '@cccsaurora/howler-ui/models/entities/generated/Observable';
3
- declare const _default: import("react").NamedExoticComponent<{
4
- icon?: string;
5
- iconColor?: string;
6
- field?: string;
7
- records?: Partial<Hit | Observable>[];
8
- title?: string;
9
- subtitle?: string;
10
- }>;
11
- export default _default;
@@ -1,24 +0,0 @@
1
- import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
- import { Icon } from '@iconify/react';
3
- import { Card, CardContent, Skeleton, Stack, styled, Tooltip, tooltipClasses, Typography, useTheme } from '@mui/material';
4
- import { get, isEmpty, uniq } from 'lodash-es';
5
- import { memo } from 'react';
6
- const NoMaxWidthTooltip = styled(({ className, ...props }) => (_jsx(Tooltip, { ...props, classes: { popper: className } })))({
7
- [`& .${tooltipClasses.tooltip}`]: {
8
- maxWidth: 'none'
9
- }
10
- });
11
- const CaseAggregate = ({ icon, iconColor, field, records, title, subtitle }) => {
12
- const theme = useTheme();
13
- if (!title && (!records || !field)) {
14
- return _jsx(Skeleton, { height: 120 });
15
- }
16
- const values = records
17
- ? uniq(records
18
- .map(_record => get(_record, field))
19
- .flat()
20
- .filter(Boolean))
21
- : [];
22
- return (_jsx(Card, { sx: { height: '100%' }, children: _jsx(CardContent, { children: _jsxs(Stack, { alignItems: "center", spacing: 1, children: [_jsxs(Stack, { direction: "row", alignItems: "center", spacing: 1, children: [icon && _jsx(Icon, { fontSize: "96px", icon: icon, color: iconColor || theme.palette.grey[700] }), _jsx(NoMaxWidthTooltip, { title: !isEmpty(values) && (_jsx(Stack, { spacing: 0.5, children: uniq(values).map(value => (_jsx("span", { children: value }, value))) })), children: _jsxs(Typography, { variant: "h3", children: [values.length, !isEmpty(values) && !!title && ' - ', title] }) })] }), _jsx(Typography, { color: "textSecondary", children: subtitle })] }) }) }));
23
- };
24
- export default memo(CaseAggregate);
@@ -1,6 +0,0 @@
1
- import type { Case } from '@cccsaurora/howler-ui/models/entities/generated/Case';
2
- import { type FC } from 'react';
3
- declare const SourceAggregate: FC<{
4
- case: Case;
5
- }>;
6
- export default SourceAggregate;
@@ -1,26 +0,0 @@
1
- import { jsx as _jsx } from "react/jsx-runtime";
2
- import { Chip, Grid, Skeleton } from '@mui/material';
3
- import { RecordContext } from '@cccsaurora/howler-ui/components/app/providers/RecordProvider';
4
- import { uniq } from 'lodash-es';
5
- import { useMemo } from 'react';
6
- import { useContextSelector } from 'use-context-selector';
7
- import useCase from '../../hooks/useCase';
8
- const SourceAggregate = ({ case: providedCase }) => {
9
- const { case: _case } = useCase({ case: providedCase });
10
- const records = useContextSelector(RecordContext, ctx => ctx.records);
11
- const analytics = useMemo(() => {
12
- if (!_case) {
13
- return [];
14
- }
15
- const hitIds = _case.items
16
- .filter(item => item.type === 'hit')
17
- .map(item => item.value)
18
- .filter(value => !!value);
19
- return uniq(hitIds.map(id => records[id]?.howler?.analytic).filter(analytic => !!analytic));
20
- }, [_case, records]);
21
- if (!_case) {
22
- return _jsx(Skeleton, { height: 12, variant: "rounded" });
23
- }
24
- return (_jsx(Grid, { container: true, spacing: 1, children: analytics.map(_analytic => (_jsx(Grid, { item: true, children: _jsx(Chip, { size: "small", label: _analytic }) }, _analytic))) }));
25
- };
26
- export default SourceAggregate;
@@ -1,14 +0,0 @@
1
- import type { Case } from '@cccsaurora/howler-ui/models/entities/generated/Case';
2
- import type { FC } from 'react';
3
- export type AssetType = 'hash' | 'hosts' | 'ip' | 'user' | 'ids' | 'id' | 'uri' | 'signature';
4
- export interface AssetEntry {
5
- type: AssetType;
6
- value: string;
7
- /** IDs of the hits/observables this asset was seen in */
8
- seenIn: string[];
9
- }
10
- declare const Asset: FC<{
11
- asset: AssetEntry;
12
- case: Case;
13
- }>;
14
- export default Asset;
@@ -1,12 +0,0 @@
1
- import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
- import { Card, CardContent, Chip, Stack, Typography } from '@mui/material';
3
- import { useTranslation } from 'react-i18next';
4
- import { Link } from 'react-router-dom';
5
- const Asset = ({ asset, case: _case }) => {
6
- const { t } = useTranslation();
7
- return (_jsx(Card, { sx: { height: '100%' }, children: _jsx(CardContent, { children: _jsxs(Stack, { spacing: 1, children: [_jsxs(Stack, { direction: "row", alignItems: "center", spacing: 1, children: [_jsx(Chip, { size: "small", label: t(`page.cases.assets.type.${asset.type}`), color: "primary", variant: "outlined" }), _jsx(Typography, { variant: "body2", sx: { wordBreak: 'break-all', fontFamily: 'monospace' }, children: asset.value })] }), asset.seenIn.length > 0 && (_jsxs(Stack, { spacing: 0.5, children: [_jsx(Typography, { variant: "caption", color: "text.secondary", children: t('page.cases.assets.seen_in') }), _jsx(Stack, { direction: "row", flexWrap: "wrap", gap: 0.5, children: asset.seenIn.map(id => {
8
- const entry = _case.items.find(item => item.value === id);
9
- return (_jsx(Chip, { clickable: true, size: "small", label: entry.path, variant: "outlined", component: Link, to: `/cases/${_case.case_id}/${entry.path}` }, id));
10
- }) })] }))] }) }) }));
11
- };
12
- export default Asset;
@@ -1 +0,0 @@
1
- export {};
@@ -1,72 +0,0 @@
1
- import { jsx as _jsx } from "react/jsx-runtime";
2
- /// <reference types="vitest" />
3
- import { render, screen } from '@testing-library/react';
4
- import { MemoryRouter } from 'react-router-dom';
5
- import { createMockCase } from '@cccsaurora/howler-ui/tests/utils';
6
- import { describe, expect, it } from 'vitest';
7
- import Asset, {} from './Asset';
8
- const makeAsset = (overrides = {}) => ({
9
- type: 'ip',
10
- value: '192.168.1.1',
11
- seenIn: [],
12
- ...overrides
13
- });
14
- describe('Asset', () => {
15
- describe('type chip', () => {
16
- it('renders the correct label for each type', () => {
17
- const cases = ['hash', 'hosts', 'ip', 'user', 'ids', 'id', 'uri', 'signature'];
18
- for (const type of cases) {
19
- const { unmount } = render(_jsx(MemoryRouter, { children: _jsx(Asset, { asset: makeAsset({ type, value: 'x' }), case: createMockCase() }) }));
20
- expect(screen.getByText(`page.cases.assets.type.${type}`)).toBeTruthy();
21
- unmount();
22
- }
23
- });
24
- });
25
- describe('value display', () => {
26
- it('renders the asset value', () => {
27
- render(_jsx(MemoryRouter, { children: _jsx(Asset, { asset: makeAsset({ value: '10.0.0.1' }), case: createMockCase() }) }));
28
- expect(screen.getByText('10.0.0.1')).toBeTruthy();
29
- });
30
- it('renders long hash values without truncation', () => {
31
- const hash = 'a'.repeat(64);
32
- render(_jsx(MemoryRouter, { children: _jsx(Asset, { asset: makeAsset({ type: 'hash', value: hash }), case: createMockCase() }) }));
33
- expect(screen.getByText(hash)).toBeTruthy();
34
- });
35
- });
36
- describe('seen-in chips', () => {
37
- it('renders nothing when seenIn is empty', () => {
38
- render(_jsx(MemoryRouter, { children: _jsx(Asset, { asset: makeAsset({ seenIn: [] }), case: createMockCase() }) }));
39
- expect(screen.queryByText('page.cases.assets.seen_in')).toBeNull();
40
- });
41
- it('renders "Seen in" label when seenIn has entries', () => {
42
- const _case = createMockCase({
43
- items: [{ path: 'alerts/test-analytic (hit-001)', type: 'hit', value: 'hit-001' }]
44
- });
45
- render(_jsx(MemoryRouter, { children: _jsx(Asset, { asset: makeAsset({ seenIn: ['hit-001'] }), case: _case }) }));
46
- expect(screen.getByText('page.cases.assets.seen_in')).toBeTruthy();
47
- });
48
- it('renders a chip labelled with entry.path for each seenIn id', () => {
49
- const _case = createMockCase({
50
- items: [
51
- { path: 'alerts/my-analytic (hit-001)', type: 'hit', value: 'hit-001' },
52
- { path: 'observables/obs-002', type: 'observable', value: 'obs-002' },
53
- { path: 'alerts/other-analytic (hit-003)', type: 'hit', value: 'hit-003' }
54
- ]
55
- });
56
- render(_jsx(MemoryRouter, { children: _jsx(Asset, { asset: makeAsset({ seenIn: ['hit-001', 'obs-002', 'hit-003'] }), case: _case }) }));
57
- expect(screen.getByText('alerts/my-analytic (hit-001)')).toBeTruthy();
58
- expect(screen.getByText('observables/obs-002')).toBeTruthy();
59
- expect(screen.getByText('alerts/other-analytic (hit-003)')).toBeTruthy();
60
- });
61
- it('links each chip to /cases/:case_id/:path', () => {
62
- const _case = createMockCase({
63
- case_id: 'case-abc',
64
- items: [{ path: 'alerts/my-analytic (hit-001)', type: 'hit', value: 'hit-001' }]
65
- });
66
- render(_jsx(MemoryRouter, { children: _jsx(Asset, { asset: makeAsset({ seenIn: ['hit-001'] }), case: _case }) }));
67
- const link = screen.getByText('alerts/my-analytic (hit-001)').closest('a');
68
- expect(link).not.toBeNull();
69
- expect(link?.getAttribute('href')).toBe('/cases/case-abc/alerts/my-analytic (hit-001)');
70
- });
71
- });
72
- });