@sinequa/atomic-angular 0.0.146

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 (174) hide show
  1. package/assets/tailwind.css +334 -0
  2. package/esm2022/lib/assistant/index.mjs +2 -0
  3. package/esm2022/lib/assistant/signalR.web.service.mjs +81 -0
  4. package/esm2022/lib/components/dropdown-input.mjs +97 -0
  5. package/esm2022/lib/components/dropdown-list.mjs +35 -0
  6. package/esm2022/lib/components/dropdown.mjs +127 -0
  7. package/esm2022/lib/components/index.mjs +7 -0
  8. package/esm2022/lib/components/menu/index.mjs +3 -0
  9. package/esm2022/lib/components/menu/menu-item.mjs +22 -0
  10. package/esm2022/lib/components/menu/menu.mjs +99 -0
  11. package/esm2022/lib/components/metadata/index.mjs +2 -0
  12. package/esm2022/lib/components/metadata/metadata.component.mjs +65 -0
  13. package/esm2022/lib/components/theme/index.mjs +3 -0
  14. package/esm2022/lib/components/theme/theme-selector.component.mjs +67 -0
  15. package/esm2022/lib/components/theme/theme-toggle.component.mjs +67 -0
  16. package/esm2022/lib/directives/index.mjs +5 -0
  17. package/esm2022/lib/directives/infinite-scroll.directive.mjs +47 -0
  18. package/esm2022/lib/directives/select-article-on-click.directive.mjs +39 -0
  19. package/esm2022/lib/directives/show-bookmark.directive.mjs +55 -0
  20. package/esm2022/lib/directives/theme-provider.directive.mjs +31 -0
  21. package/esm2022/lib/guards/auth.guard.mjs +27 -0
  22. package/esm2022/lib/guards/index.mjs +3 -0
  23. package/esm2022/lib/guards/initialization.guard.mjs +41 -0
  24. package/esm2022/lib/interceptors/audit.interceptor.mjs +23 -0
  25. package/esm2022/lib/interceptors/auth.interceptor.mjs +49 -0
  26. package/esm2022/lib/interceptors/body.interceptor.mjs +24 -0
  27. package/esm2022/lib/interceptors/error.interceptor.mjs +35 -0
  28. package/esm2022/lib/interceptors/index.mjs +7 -0
  29. package/esm2022/lib/interceptors/toast.interceptor.mjs +27 -0
  30. package/esm2022/lib/models/aggregation.mjs +2 -0
  31. package/esm2022/lib/models/article-metadata.mjs +2 -0
  32. package/esm2022/lib/models/autocomplete.mjs +2 -0
  33. package/esm2022/lib/models/custom-json.mjs +2 -0
  34. package/esm2022/lib/models/filter-dropdown.mjs +2 -0
  35. package/esm2022/lib/models/index.mjs +7 -0
  36. package/esm2022/lib/models/user-settings.mjs +2 -0
  37. package/esm2022/lib/pipes/highlight-word.pipe.mjs +33 -0
  38. package/esm2022/lib/pipes/index.mjs +3 -0
  39. package/esm2022/lib/pipes/source-icon.pipe.mjs +43 -0
  40. package/esm2022/lib/public-api.mjs +18 -0
  41. package/esm2022/lib/resolvers/index.mjs +2 -0
  42. package/esm2022/lib/resolvers/query-name-resolver.mjs +14 -0
  43. package/esm2022/lib/resources/index.mjs +2 -0
  44. package/esm2022/lib/resources/themes.mjs +53 -0
  45. package/esm2022/lib/services/application.service.mjs +245 -0
  46. package/esm2022/lib/services/autocomplete.service.mjs +85 -0
  47. package/esm2022/lib/services/drawer/backdrop.service.mjs +23 -0
  48. package/esm2022/lib/services/drawer/drawer-stack.service.mjs +152 -0
  49. package/esm2022/lib/services/drawer/drawer.service.mjs +38 -0
  50. package/esm2022/lib/services/index.mjs +12 -0
  51. package/esm2022/lib/services/label.service.mjs +161 -0
  52. package/esm2022/lib/services/navigation.service.mjs +59 -0
  53. package/esm2022/lib/services/saved-searches.service.mjs +75 -0
  54. package/esm2022/lib/services/search.service.mjs +89 -0
  55. package/esm2022/lib/services/selection-history.service.mjs +92 -0
  56. package/esm2022/lib/services/selection.service.mjs +87 -0
  57. package/esm2022/lib/stores/aggregations.store.mjs +59 -0
  58. package/esm2022/lib/stores/app.store.mjs +270 -0
  59. package/esm2022/lib/stores/application.store.mjs +87 -0
  60. package/esm2022/lib/stores/index.mjs +9 -0
  61. package/esm2022/lib/stores/principal.store.mjs +47 -0
  62. package/esm2022/lib/stores/query-params.store.mjs +207 -0
  63. package/esm2022/lib/stores/selection.store.mjs +45 -0
  64. package/esm2022/lib/stores/theme.store.mjs +116 -0
  65. package/esm2022/lib/stores/user-settings.store.mjs +391 -0
  66. package/esm2022/lib/tokens/highlights.mjs +32 -0
  67. package/esm2022/lib/tokens/index.mjs +2 -0
  68. package/esm2022/lib/utils/debounced-signal.mjs +38 -0
  69. package/esm2022/lib/utils/index.mjs +8 -0
  70. package/esm2022/lib/utils/inline-worker.mjs +40 -0
  71. package/esm2022/lib/utils/query.mjs +58 -0
  72. package/esm2022/lib/utils/routes.mjs +28 -0
  73. package/esm2022/lib/utils/tailwind-utils.mjs +6 -0
  74. package/esm2022/lib/utils/theme-body-hook.mjs +18 -0
  75. package/esm2022/lib/utils/theme-registry.mjs +6 -0
  76. package/esm2022/lib/web-services/aggregations.service.mjs +104 -0
  77. package/esm2022/lib/web-services/app.service.mjs +48 -0
  78. package/esm2022/lib/web-services/audit.service.mjs +122 -0
  79. package/esm2022/lib/web-services/index.mjs +10 -0
  80. package/esm2022/lib/web-services/json-method-plugin.service.mjs +54 -0
  81. package/esm2022/lib/web-services/preview.service.mjs +327 -0
  82. package/esm2022/lib/web-services/principal.service.mjs +46 -0
  83. package/esm2022/lib/web-services/query.service.mjs +123 -0
  84. package/esm2022/lib/web-services/text-chunck.service.mjs +46 -0
  85. package/esm2022/public-api.mjs +5 -0
  86. package/esm2022/sinequa-atomic-angular.mjs +5 -0
  87. package/fesm2022/sinequa-atomic-angular.mjs +4420 -0
  88. package/fesm2022/sinequa-atomic-angular.mjs.map +1 -0
  89. package/index.d.ts +5 -0
  90. package/lib/assistant/index.d.ts +1 -0
  91. package/lib/assistant/signalR.web.service.d.ts +46 -0
  92. package/lib/components/dropdown-input.d.ts +18 -0
  93. package/lib/components/dropdown-list.d.ts +8 -0
  94. package/lib/components/dropdown.d.ts +50 -0
  95. package/lib/components/index.d.ts +6 -0
  96. package/lib/components/menu/index.d.ts +2 -0
  97. package/lib/components/menu/menu-item.d.ts +8 -0
  98. package/lib/components/menu/menu.d.ts +24 -0
  99. package/lib/components/metadata/index.d.ts +1 -0
  100. package/lib/components/metadata/metadata.component.d.ts +24 -0
  101. package/lib/components/theme/index.d.ts +2 -0
  102. package/lib/components/theme/theme-selector.component.d.ts +70 -0
  103. package/lib/components/theme/theme-toggle.component.d.ts +10 -0
  104. package/lib/directives/index.d.ts +4 -0
  105. package/lib/directives/infinite-scroll.directive.d.ts +30 -0
  106. package/lib/directives/select-article-on-click.directive.d.ts +14 -0
  107. package/lib/directives/show-bookmark.directive.d.ts +60 -0
  108. package/lib/directives/theme-provider.directive.d.ts +20 -0
  109. package/lib/guards/auth.guard.d.ts +7 -0
  110. package/lib/guards/index.d.ts +2 -0
  111. package/lib/guards/initialization.guard.d.ts +20 -0
  112. package/lib/interceptors/audit.interceptor.d.ts +13 -0
  113. package/lib/interceptors/auth.interceptor.d.ts +14 -0
  114. package/lib/interceptors/body.interceptor.d.ts +11 -0
  115. package/lib/interceptors/error.interceptor.d.ts +9 -0
  116. package/lib/interceptors/index.d.ts +5 -0
  117. package/lib/interceptors/toast.interceptor.d.ts +13 -0
  118. package/lib/models/aggregation.d.ts +12 -0
  119. package/lib/models/article-metadata.d.ts +5 -0
  120. package/lib/models/autocomplete.d.ts +5 -0
  121. package/lib/models/custom-json.d.ts +69 -0
  122. package/lib/models/filter-dropdown.d.ts +10 -0
  123. package/lib/models/index.d.ts +6 -0
  124. package/lib/models/user-settings.d.ts +38 -0
  125. package/lib/pipes/highlight-word.pipe.d.ts +22 -0
  126. package/lib/pipes/index.d.ts +2 -0
  127. package/lib/pipes/source-icon.pipe.d.ts +51 -0
  128. package/lib/public-api.d.ts +14 -0
  129. package/lib/resolvers/index.d.ts +1 -0
  130. package/lib/resolvers/query-name-resolver.d.ts +9 -0
  131. package/lib/resources/index.d.ts +1 -0
  132. package/lib/resources/themes.d.ts +51 -0
  133. package/lib/services/application.service.d.ts +183 -0
  134. package/lib/services/autocomplete.service.d.ts +96 -0
  135. package/lib/services/drawer/backdrop.service.d.ts +9 -0
  136. package/lib/services/drawer/drawer-stack.service.d.ts +70 -0
  137. package/lib/services/drawer/drawer.service.d.ts +15 -0
  138. package/lib/services/index.d.ts +11 -0
  139. package/lib/services/label.service.d.ts +114 -0
  140. package/lib/services/navigation.service.d.ts +33 -0
  141. package/lib/services/saved-searches.service.d.ts +149 -0
  142. package/lib/services/search.service.d.ts +159 -0
  143. package/lib/services/selection-history.service.d.ts +50 -0
  144. package/lib/services/selection.service.d.ts +123 -0
  145. package/lib/stores/aggregations.store.d.ts +25 -0
  146. package/lib/stores/app.store.d.ts +110 -0
  147. package/lib/stores/application.store.d.ts +83 -0
  148. package/lib/stores/index.d.ts +8 -0
  149. package/lib/stores/principal.store.d.ts +47 -0
  150. package/lib/stores/query-params.store.d.ts +128 -0
  151. package/lib/stores/selection.store.d.ts +52 -0
  152. package/lib/stores/theme.store.d.ts +66 -0
  153. package/lib/stores/user-settings.store.d.ts +111 -0
  154. package/lib/tokens/highlights.d.ts +8 -0
  155. package/lib/tokens/index.d.ts +1 -0
  156. package/lib/utils/debounced-signal.d.ts +25 -0
  157. package/lib/utils/index.d.ts +7 -0
  158. package/lib/utils/inline-worker.d.ts +11 -0
  159. package/lib/utils/query.d.ts +26 -0
  160. package/lib/utils/routes.d.ts +16 -0
  161. package/lib/utils/tailwind-utils.d.ts +2 -0
  162. package/lib/utils/theme-body-hook.d.ts +6 -0
  163. package/lib/utils/theme-registry.d.ts +3 -0
  164. package/lib/web-services/aggregations.service.d.ts +57 -0
  165. package/lib/web-services/app.service.d.ts +30 -0
  166. package/lib/web-services/audit.service.d.ts +72 -0
  167. package/lib/web-services/index.d.ts +9 -0
  168. package/lib/web-services/json-method-plugin.service.d.ts +41 -0
  169. package/lib/web-services/preview.service.d.ts +283 -0
  170. package/lib/web-services/principal.service.d.ts +28 -0
  171. package/lib/web-services/query.service.d.ts +29 -0
  172. package/lib/web-services/text-chunck.service.d.ts +22 -0
  173. package/package.json +28 -0
  174. package/public-api.d.ts +1 -0
@@ -0,0 +1,327 @@
1
+ import { HttpClient } from "@angular/common/http";
2
+ import { EventEmitter, Inject, Injectable, InjectionToken, Injector, inject } from "@angular/core";
3
+ import { DomSanitizer } from "@angular/platform-browser";
4
+ import { getState } from "@ngrx/signals";
5
+ import { EMPTY, Subject, catchError, tap } from "rxjs";
6
+ import { Audit, globalConfig } from "@sinequa/atomic";
7
+ import { SearchService } from "../services/search.service";
8
+ import { AppStore, ApplicationStore, QueryParamsStore, SelectionStore } from "../stores";
9
+ import { HIGHLIGHTS } from "../tokens";
10
+ import { AuditService } from "./audit.service";
11
+ import { QueryService } from "./query.service";
12
+ import * as i0 from "@angular/core";
13
+ const PREVIEW_CONFIG = new InjectionToken("Preview configuration", {
14
+ factory: () => ({
15
+ allowWorker: false
16
+ })
17
+ });
18
+ export class PreviewService {
19
+ constructor(highlights) {
20
+ this.highlights = highlights;
21
+ this.http = inject(HttpClient);
22
+ this.injector = inject(Injector);
23
+ this.API_URL = `${globalConfig.backendUrl}/${globalConfig.apiPath}`;
24
+ this.applicationStore = inject(ApplicationStore);
25
+ this.selectionStore = inject(SelectionStore);
26
+ this.queryParamsStore = inject(QueryParamsStore);
27
+ this.appStore = inject(AppStore);
28
+ this.sanitizer = inject(DomSanitizer);
29
+ this.queryService = inject(QueryService);
30
+ this.searchService = inject(SearchService);
31
+ this.auditService = inject(AuditService);
32
+ this.events = new EventEmitter();
33
+ // ! worker
34
+ this.onMessage = new Subject();
35
+ this.allowWoker = inject(PREVIEW_CONFIG).allowWorker;
36
+ this.highlightCategory = "extractslocations";
37
+ this.extracts = ["matchlocations", "extractslocations", "matchingpassages"];
38
+ this.entities = ["company", "geo", "person"];
39
+ window.addEventListener('message', this.receiveMessage.bind(this), false);
40
+ }
41
+ ngOnDestroy() {
42
+ window.removeEventListener('message', this.receiveMessage.bind(this), false);
43
+ }
44
+ /**
45
+ * Handles incoming messages from a MessageEvent.
46
+ *
47
+ * @param event - The MessageEvent containing the message data.
48
+ *
49
+ * The function processes messages of type 'ready' and 'get-html-results'.
50
+ *
51
+ * - For 'ready' messages:
52
+ * - Initializes the preview iframe with the app name and highlights.
53
+ * - If preview data is available, retrieves HTML content based on the current selection.
54
+ *
55
+ * - For 'get-html-results' messages:
56
+ * - Updates the application store with the extracted HTML results.
57
+ * - If no extracts are found, updates the application store with an empty array.
58
+ */
59
+ receiveMessage(event) {
60
+ const message = event.data;
61
+ // the webworker needs to be initialized with the app name to be able to load the correct webworker file (worker.js)
62
+ const { app } = globalConfig;
63
+ // when the iframe is ready, we can send the init message
64
+ if (message.type === 'ready') {
65
+ // Initialize the preview iframe with the app name and highlights
66
+ // app name is used to intialize the webworker
67
+ this.sendMessage({ action: "init", highlights: this.highlights, appname: app });
68
+ if (this.previewData) {
69
+ const { id } = getState(this.selectionStore);
70
+ this.events.emit('fetching');
71
+ this.retrieveHtmlContent(id, this.highlightCategory, this.previewData);
72
+ }
73
+ }
74
+ // when the iframe sends back the html content of the extracts (no worker)
75
+ if (message.type === 'get-html-results') {
76
+ const { id } = getState(this.selectionStore);
77
+ if (id) {
78
+ console.info("no worker");
79
+ const extracts = this.fetchExtracts(id, message.data, this.previewData);
80
+ this.applicationStore.updateExtracts(this.previewData?.record.id || '', extracts);
81
+ this.events.emit('fetched');
82
+ }
83
+ }
84
+ // when the iframe sends back the html content of the extracts (with worker)
85
+ // we update the extracts in the application store
86
+ if (message.type === 'get-html-results-webworker') {
87
+ console.info("with worker");
88
+ if (message.data.extracts.length === 0) {
89
+ this.events.emit('fetched');
90
+ this.applicationStore.updateExtracts(this.previewData?.record.id || '', []);
91
+ return;
92
+ }
93
+ const extracts = message.data.extracts.map((item) => ({
94
+ ...item,
95
+ text: this.sanitizer.bypassSecurityTrustHtml(item.text)
96
+ }));
97
+ this.applicationStore.updateExtracts(this.previewData?.record.id || '', extracts);
98
+ this.events.emit('fetched');
99
+ }
100
+ }
101
+ /**
102
+ * Generates a preview of the document with the given ID, applying the specified query parameters,
103
+ * custom highlights, and audit events.
104
+ *
105
+ * @param id - The unique identifier of the document to preview.
106
+ * @param q - Partial query parameters to customize the preview request.
107
+ * @param customHighlights - Optional array of custom highlights to apply to the preview.
108
+ * @param audit - Optional audit events to merge with the default audit record.
109
+ * @returns An Observable that emits the preview data.
110
+ */
111
+ preview(id, q, customHighlights, audit) {
112
+ const detail = this.getAuditPreviewDetail(id, q);
113
+ const defaultAudit = {
114
+ type: "Doc_Preview",
115
+ detail
116
+ };
117
+ // merge the default audit record with the provided audit record
118
+ const $auditRecord = audit ? { auditEvents: [{ ...defaultAudit, ...audit }] } : { auditEvents: [defaultAudit] };
119
+ const { app } = globalConfig;
120
+ const query = { ...this.queryParamsStore.getQuery(), ...q };
121
+ const browserUrl = `${window.location.origin}${window.location.pathname}`;
122
+ const body = {
123
+ app,
124
+ action: "get",
125
+ id,
126
+ query: { name: query.name, text: query.text },
127
+ browserUrl,
128
+ customHighlights,
129
+ $auditRecord
130
+ };
131
+ return this.http.post(this.API_URL + "/preview", body)
132
+ .pipe(tap((data) => this.setPreviewData(data)), catchError(error => {
133
+ console.error("queryService.preview failure - error: ", error);
134
+ return EMPTY;
135
+ }));
136
+ }
137
+ /**
138
+ * Closes the preview with the specified ID and updates the audit log.
139
+ *
140
+ * @param id - The ID of the preview to close.
141
+ * @param query - The partial query object used to retrieve the preview detail.
142
+ */
143
+ close(id, query) {
144
+ const detail = this.getAuditPreviewDetail(id, query);
145
+ const auditEvent = {
146
+ type: "Preview_Close" /* AuditEventType.Preview_Close */,
147
+ detail
148
+ };
149
+ Audit.notify(auditEvent);
150
+ }
151
+ /**
152
+ * Previews an article in a new browser's tab.
153
+ *
154
+ * @param article - The article to preview.
155
+ */
156
+ openExternal(article) {
157
+ const query = this.queryParamsStore.getQuery();
158
+ const detail = this.getAuditPreviewDetail(article.id, query);
159
+ const result = this.searchService.result;
160
+ this.auditService.notifyDocument("Click_ResultLink", article, result || detail.resultid || "", {
161
+ querytext: detail.querytext,
162
+ querylang: detail.querylang,
163
+ score: detail.score
164
+ }, {
165
+ queryhash: result ? result.rfmQueryHash : undefined,
166
+ querytext: detail.querytext,
167
+ querylang: detail.querylang
168
+ });
169
+ window.open(article.url1, '_blank', 'noopener noreferrer');
170
+ }
171
+ /**
172
+ * Sets the iframe window object.
173
+ *
174
+ * @param iframe - The window object of the iframe or null to unset.
175
+ */
176
+ setIframe(iframe) {
177
+ this.iframe = iframe;
178
+ }
179
+ /**
180
+ * Sets the preview data and updates the highlight category based on the provided data.
181
+ *
182
+ * @param data - The preview data to be set.
183
+ *
184
+ * If the provided data contains highlights per category with 'matchingpassages' having values,
185
+ * the highlight category is set to "matchingpassages". Otherwise, it is set to "extractslocations".
186
+ */
187
+ setPreviewData(data) {
188
+ this.previewData = data;
189
+ if (data) {
190
+ this.highlightCategory = data.highlightsPerCategory?.['matchingpassages']?.values.length ? "matchingpassages" : "extractslocations";
191
+ }
192
+ }
193
+ /**
194
+ * Sends a message to the iframe if it exists.
195
+ *
196
+ * @param message - The message to be sent. It can be of any type.
197
+ */
198
+ sendMessage(message) {
199
+ if (this.iframe) {
200
+ this.iframe.postMessage(message, '*');
201
+ }
202
+ }
203
+ /**
204
+ * Get preview id for an entity value for an index
205
+ * @param entity the entity type (i.e. "geo")
206
+ * @param value the entity value (i.e. "AUSTRALIA")
207
+ * @param index the entity index from its list
208
+ * @returns the html id
209
+ */
210
+ getEntityId(entity, value, index) {
211
+ if (!this.previewData)
212
+ return undefined;
213
+ return this.previewData.highlightsPerLocation
214
+ .filter(h => h.values.find(v => v === value))[index].positionInCategories[entity];
215
+ }
216
+ /**
217
+ * Send a message to the prewiew iFrame with the required data to retrieve HTML content for a specific highlight category
218
+ *
219
+ * @param id - The unique identifier for the request.
220
+ * @param highlightCategory - The category of highlights to retrieve.
221
+ * @param previewData - The data containing highlights and their locations.
222
+ */
223
+ retrieveHtmlContent(id, highlightCategory, previewData) {
224
+ // Generate the list of items we want to retrieve
225
+ const ids = previewData.highlightsPerCategory[highlightCategory]?.values[0]?.locations.map((_, i) => `${highlightCategory}_${i}`);
226
+ if (this.allowWoker) {
227
+ // if the worker is allowed, we send the message to the worker
228
+ this.sendMessage({ action: 'get-html-webworker', id, ids, previewData });
229
+ }
230
+ else {
231
+ this.sendMessage({ action: 'get-html', id, ids, previewData });
232
+ }
233
+ }
234
+ /**
235
+ * Sends a message to zoom in.
236
+ *
237
+ * This method triggers a "zoom-in" action by sending a message
238
+ * with the specified action type.
239
+ */
240
+ zoomIn() {
241
+ this.sendMessage({ action: "zoom-in" });
242
+ }
243
+ /**
244
+ * Sends a message to zoom out the preview.
245
+ *
246
+ * This method triggers a "zoom-out" action by sending a message
247
+ * to the relevant service or component.
248
+ */
249
+ zoomOut() {
250
+ this.sendMessage({ action: "zoom-out" });
251
+ }
252
+ /**
253
+ * Toggles the highlights based on the provided flags for extracts and entities.
254
+ *
255
+ * @param extracts - A boolean flag indicating whether to include extracts highlights.
256
+ * @param entities - A boolean flag indicating whether to include entities highlights.
257
+ */
258
+ toggle(extracts, entities) {
259
+ const extractsHighlights = extracts ? this.highlights.filter(h => this.extracts.includes(h.name)) : [];
260
+ const entitiesHighlights = entities ? this.highlights.filter(h => this.entities.includes(h.name)) : [];
261
+ const highlights = [...extractsHighlights, ...entitiesHighlights];
262
+ this.sendMessage({ action: "highlight", highlights });
263
+ }
264
+ getAuditPreviewDetail(id, q) {
265
+ const results = this.searchService.result;
266
+ const queryLanguage = results?.queryAnalysis?.queryLanguage
267
+ || q?.questionLanguage
268
+ || this.appStore.getQueryByName(q.name)?.questionLanguage;
269
+ const record = this.previewData?.record;
270
+ const collectionColumn = record?.collection;
271
+ const collection = !!collectionColumn ? collectionColumn[0] : id.split("|")[0];
272
+ const rank = !!record ? record.rank : 0;
273
+ const passages = record?.matchingpassages?.passages;
274
+ const score = passages && passages.length ? passages[0].score : undefined;
275
+ return {
276
+ docid: id,
277
+ rank,
278
+ collection,
279
+ source: collection.split("|")[0],
280
+ resultid: results.id,
281
+ querylang: queryLanguage,
282
+ querytext: q?.text,
283
+ filename: record?.filename,
284
+ fileext: record?.fileext,
285
+ score
286
+ };
287
+ }
288
+ fetchExtracts(id, extracts, data) {
289
+ // extracts contains the html of extracts in chronological order
290
+ // locations contains the list of start positions sorted by score
291
+ const locations = data.highlightsPerCategory[this.highlightCategory]?.values[0]?.locations || [];
292
+ // first extract all the extracts locations
293
+ let extractslocations = locations.map((l, relevanceIndex) => ({
294
+ startIndex: l.start,
295
+ relevanceIndex
296
+ }));
297
+ // sort them by start index
298
+ extractslocations.sort((a, b) => a.startIndex - b.startIndex);
299
+ // then extract the text of each extract
300
+ extractslocations = extractslocations.map((ex, textIndex) => ({
301
+ ...ex,
302
+ textIndex: textIndex,
303
+ text: extracts[textIndex] || "",
304
+ id: `${this.highlightCategory}_${textIndex}`,
305
+ }));
306
+ // then sanitize the text and remove empty extracts
307
+ const _extracts = extractslocations.filter(item => item.text.trim().length > 0).map(item => ({
308
+ ...item,
309
+ text: this.sanitizer.bypassSecurityTrustHtml(item.text)
310
+ }));
311
+ // finally sort them by relevance index
312
+ _extracts.sort((a, b) => a.relevanceIndex - b.relevanceIndex);
313
+ return _extracts;
314
+ }
315
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: PreviewService, deps: [{ token: HIGHLIGHTS }], target: i0.ɵɵFactoryTarget.Injectable }); }
316
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: PreviewService, providedIn: 'root' }); }
317
+ }
318
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: PreviewService, decorators: [{
319
+ type: Injectable,
320
+ args: [{
321
+ providedIn: 'root'
322
+ }]
323
+ }], ctorParameters: () => [{ type: undefined, decorators: [{
324
+ type: Inject,
325
+ args: [HIGHLIGHTS]
326
+ }] }] });
327
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"preview.service.js","sourceRoot":"","sources":["../../../../../projects/atomic-angular/src/lib/web-services/preview.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient } from \"@angular/common/http\";\r\nimport { EventEmitter, Inject, Injectable, InjectionToken, Injector, OnDestroy, inject } from \"@angular/core\";\r\nimport { DomSanitizer } from \"@angular/platform-browser\";\r\nimport { getState } from \"@ngrx/signals\";\r\nimport { EMPTY, Observable, Subject, catchError, tap } from \"rxjs\";\r\n\r\nimport { Article, Audit, AuditEventType, AuditEvents, CustomHighlights, PreviewData, Query, globalConfig } from \"@sinequa/atomic\";\r\n\r\nimport { SearchService } from \"../services/search.service\";\r\nimport { AppStore, ApplicationStore, Extract, QueryParamsStore, SelectionStore } from \"../stores\";\r\nimport { HIGHLIGHTS, PreviewHighlight } from \"../tokens\";\r\nimport { InlineWorker } from \"../utils\";\r\nimport { AuditService } from \"./audit.service\";\r\nimport { QueryService } from \"./query.service\";\r\n\r\n\r\nconst PREVIEW_CONFIG = new InjectionToken(\"Preview configuration\", {\r\n  factory: () => ({\r\n    allowWorker: false\r\n  })\r\n});\r\n\r\ntype ExtractsLocations = Extract & {\r\n  text: string // HTML text\r\n}\r\n\r\nexport type ExtractsEvents = 'fetching' | 'fetched' | 'error';\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class PreviewService implements OnDestroy {\r\n  protected readonly http = inject(HttpClient);\r\n  protected readonly injector = inject(Injector);\r\n\r\n  protected readonly API_URL = `${globalConfig.backendUrl}/${globalConfig.apiPath}`;\r\n\r\n  protected readonly applicationStore = inject(ApplicationStore);\r\n  protected readonly selectionStore = inject(SelectionStore);\r\n  protected readonly queryParamsStore = inject(QueryParamsStore);\r\n  protected readonly appStore = inject(AppStore);\r\n\r\n  protected readonly sanitizer = inject(DomSanitizer);\r\n\r\n  protected readonly queryService = inject(QueryService);\r\n  protected readonly searchService = inject(SearchService);\r\n  protected readonly auditService = inject(AuditService);\r\n\r\n  events = new EventEmitter<ExtractsEvents>();\r\n\r\n  // ! worker\r\n  onMessage = new Subject<unknown>();\r\n  worker: InlineWorker | undefined;\r\n\r\n  allowWoker = inject(PREVIEW_CONFIG).allowWorker;\r\n\r\n  protected previewData: PreviewData | undefined;\r\n  protected iframe: Window | null | undefined;\r\n\r\n  protected highlightCategory = \"extractslocations\";\r\n\r\n  extracts = [\"matchlocations\", \"extractslocations\", \"matchingpassages\"];\r\n  entities = [\"company\", \"geo\", \"person\"];\r\n\r\n  constructor(\r\n    @Inject(HIGHLIGHTS) public highlights: PreviewHighlight[]) {\r\n    window.addEventListener('message', this.receiveMessage.bind(this), false);\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    window.removeEventListener('message', this.receiveMessage.bind(this), false);\r\n  }\r\n\r\n  /**\r\n   * Handles incoming messages from a MessageEvent.\r\n   *\r\n   * @param event - The MessageEvent containing the message data.\r\n   *\r\n   * The function processes messages of type 'ready' and 'get-html-results'.\r\n   *\r\n   * - For 'ready' messages:\r\n   *   - Initializes the preview iframe with the app name and highlights.\r\n   *   - If preview data is available, retrieves HTML content based on the current selection.\r\n   *\r\n   * - For 'get-html-results' messages:\r\n   *   - Updates the application store with the extracted HTML results.\r\n   *   - If no extracts are found, updates the application store with an empty array.\r\n   */\r\n  receiveMessage(event: MessageEvent) {\r\n    const message = event.data;\r\n    // the webworker needs to be initialized with the app name to be able to load the correct webworker file (worker.js)\r\n    const { app } = globalConfig;\r\n\r\n    // when the iframe is ready, we can send the init message\r\n    if (message.type === 'ready') {\r\n      // Initialize the preview iframe with the app name and highlights\r\n      // app name is used to intialize the webworker\r\n      this.sendMessage({ action: \"init\", highlights: this.highlights, appname: app });\r\n\r\n      if (this.previewData) {\r\n        const { id } = getState(this.selectionStore);\r\n        this.events.emit('fetching');\r\n        this.retrieveHtmlContent(id, this.highlightCategory, this.previewData);\r\n      }\r\n    }\r\n\r\n    // when the iframe sends back the html content of the extracts (no worker)\r\n    if (message.type === 'get-html-results') {\r\n      const { id } = getState(this.selectionStore)\r\n\r\n      if (id) {\r\n        console.info(\"no worker\");\r\n        const extracts = this.fetchExtracts(id, message.data, this.previewData!);\r\n        this.applicationStore.updateExtracts(this.previewData?.record.id || '', extracts);\r\n        this.events.emit('fetched');\r\n      }\r\n    }\r\n\r\n    // when the iframe sends back the html content of the extracts (with worker)\r\n    // we update the extracts in the application store\r\n    if (message.type === 'get-html-results-webworker') {\r\n      console.info(\"with worker\");\r\n\r\n      if (message.data.extracts.length === 0) {\r\n        this.events.emit('fetched');\r\n        this.applicationStore.updateExtracts(this.previewData?.record.id || '', []);\r\n        return;\r\n      }\r\n\r\n      const extracts: Extract[] = message.data.extracts.map((item: ExtractsLocations) => ({\r\n        ...item,\r\n        text: this.sanitizer.bypassSecurityTrustHtml(item.text)\r\n      }));\r\n\r\n      this.applicationStore.updateExtracts(this.previewData?.record.id || '', extracts);\r\n      this.events.emit('fetched');\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Generates a preview of the document with the given ID, applying the specified query parameters,\r\n   * custom highlights, and audit events.\r\n   *\r\n   * @param id - The unique identifier of the document to preview.\r\n   * @param q - Partial query parameters to customize the preview request.\r\n   * @param customHighlights - Optional array of custom highlights to apply to the preview.\r\n   * @param audit - Optional audit events to merge with the default audit record.\r\n   * @returns An Observable that emits the preview data.\r\n   */\r\n  preview(id: string, q: Partial<Query>, customHighlights?: CustomHighlights[], audit?: AuditEvents): Observable<PreviewData> {\r\n    const detail = this.getAuditPreviewDetail(id, q);\r\n\r\n    const defaultAudit: AuditEvents = {\r\n      type: \"Doc_Preview\",\r\n      detail\r\n    };\r\n\r\n    // merge the default audit record with the provided audit record\r\n    const $auditRecord = audit ? { auditEvents: [{ ...defaultAudit, ...audit }] } : { auditEvents: [defaultAudit] };\r\n\r\n    const { app } = globalConfig;\r\n    const query = { ...this.queryParamsStore.getQuery(), ...q };\r\n\r\n    const browserUrl = `${window.location.origin}${window.location.pathname}`;\r\n\r\n    const body = {\r\n      app,\r\n      action: \"get\",\r\n      id,\r\n      query: { name: query.name, text: query.text },\r\n      browserUrl,\r\n      customHighlights,\r\n      $auditRecord\r\n    };\r\n\r\n    return this.http.post<PreviewData>(this.API_URL + \"/preview\", body)\r\n      .pipe(\r\n        tap((data: PreviewData) => this.setPreviewData(data)),\r\n        catchError(error => {\r\n          console.error(\"queryService.preview failure - error: \", error);\r\n          return EMPTY;\r\n        })\r\n      );\r\n  }\r\n\r\n  /**\r\n   * Closes the preview with the specified ID and updates the audit log.\r\n   *\r\n   * @param id - The ID of the preview to close.\r\n   * @param query - The partial query object used to retrieve the preview detail.\r\n   */\r\n  close(id: string, query: Partial<Query>) {\r\n    const detail = this.getAuditPreviewDetail(id, query);\r\n    const auditEvent = {\r\n      type: AuditEventType.Preview_Close,\r\n      detail\r\n    }\r\n    Audit.notify(auditEvent);\r\n  }\r\n\r\n  /**\r\n   * Previews an article in a new browser's tab.\r\n   *\r\n   * @param article - The article to preview.\r\n   */\r\n  openExternal(article: Article) {\r\n    const query = this.queryParamsStore.getQuery();\r\n    const detail = this.getAuditPreviewDetail(article.id, query);\r\n    const result = this.searchService.result;\r\n    this.auditService.notifyDocument(\r\n      \"Click_ResultLink\",\r\n      article,\r\n      result || detail.resultid || \"\",\r\n      {\r\n        querytext: detail.querytext,\r\n        querylang: detail.querylang,\r\n        score: detail.score\r\n      },\r\n      {\r\n        queryhash: result ? result.rfmQueryHash : undefined,\r\n        querytext: detail.querytext,\r\n        querylang: detail.querylang\r\n      }\r\n    )\r\n    window.open(article.url1, '_blank', 'noopener noreferrer');\r\n  }\r\n\r\n\r\n  /**\r\n   * Sets the iframe window object.\r\n   *\r\n   * @param iframe - The window object of the iframe or null to unset.\r\n   */\r\n  setIframe(iframe: Window | null) {\r\n    this.iframe = iframe;\r\n  }\r\n\r\n  /**\r\n   * Sets the preview data and updates the highlight category based on the provided data.\r\n   *\r\n   * @param data - The preview data to be set.\r\n   *\r\n   * If the provided data contains highlights per category with 'matchingpassages' having values,\r\n   * the highlight category is set to \"matchingpassages\". Otherwise, it is set to \"extractslocations\".\r\n   */\r\n  setPreviewData(data: PreviewData) {\r\n    this.previewData = data;\r\n\r\n    if (data) {\r\n      this.highlightCategory = data.highlightsPerCategory?.['matchingpassages']?.values.length ? \"matchingpassages\" : \"extractslocations\";\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Sends a message to the iframe if it exists.\r\n   *\r\n   * @param message - The message to be sent. It can be of any type.\r\n   */\r\n  sendMessage(message: unknown) {\r\n    if (this.iframe) {\r\n      this.iframe!.postMessage(message, '*');\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Get preview id for an entity value for an index\r\n   * @param entity the entity type (i.e. \"geo\")\r\n   * @param value the entity value (i.e. \"AUSTRALIA\")\r\n   * @param index the entity index from its list\r\n   * @returns the html id\r\n   */\r\n  getEntityId(entity: string, value: string, index: number): number | undefined {\r\n    if (!this.previewData) return undefined;\r\n\r\n    return this.previewData.highlightsPerLocation\r\n      .filter(h => h.values.find(v => v === value))[index].positionInCategories[entity];\r\n  }\r\n\r\n  /**\r\n   * Send a message to the prewiew iFrame with the required data to retrieve HTML content for a specific highlight category\r\n   *\r\n   * @param id - The unique identifier for the request.\r\n   * @param highlightCategory - The category of highlights to retrieve.\r\n   * @param previewData - The data containing highlights and their locations.\r\n   */\r\n  retrieveHtmlContent(id: string, highlightCategory: string, previewData: PreviewData) {\r\n    // Generate the list of items we want to retrieve\r\n    const ids = previewData.highlightsPerCategory[highlightCategory]?.values[0]?.locations.map(\r\n      (_, i) => `${highlightCategory}_${i}`\r\n    );\r\n\r\n    if (this.allowWoker) {\r\n      // if the worker is allowed, we send the message to the worker\r\n      this.sendMessage({ action: 'get-html-webworker', id, ids, previewData });\r\n    } else {\r\n      this.sendMessage({ action: 'get-html', id, ids, previewData });\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Sends a message to zoom in.\r\n   *\r\n   * This method triggers a \"zoom-in\" action by sending a message\r\n   * with the specified action type.\r\n   */\r\n  zoomIn() {\r\n    this.sendMessage({ action: \"zoom-in\" });\r\n  }\r\n\r\n  /**\r\n   * Sends a message to zoom out the preview.\r\n   *\r\n   * This method triggers a \"zoom-out\" action by sending a message\r\n   * to the relevant service or component.\r\n   */\r\n  zoomOut() {\r\n    this.sendMessage({ action: \"zoom-out\" });\r\n  }\r\n\r\n  /**\r\n   * Toggles the highlights based on the provided flags for extracts and entities.\r\n   *\r\n   * @param extracts - A boolean flag indicating whether to include extracts highlights.\r\n   * @param entities - A boolean flag indicating whether to include entities highlights.\r\n   */\r\n  toggle(extracts: boolean, entities: boolean) {\r\n    const extractsHighlights = extracts ? this.highlights.filter(h => this.extracts.includes(h.name)) : [];\r\n    const entitiesHighlights = entities ? this.highlights.filter(h => this.entities.includes(h.name)) : [];\r\n    const highlights = [...extractsHighlights, ...entitiesHighlights];\r\n\r\n    this.sendMessage({ action: \"highlight\", highlights });\r\n  }\r\n\r\n  getAuditPreviewDetail(id: string, q: Partial<Query>) {\r\n    const results = this.searchService.result;\r\n    const queryLanguage = results?.queryAnalysis?.queryLanguage\r\n      || q?.questionLanguage\r\n      || this.appStore.getQueryByName(q.name!)?.questionLanguage;\r\n\r\n    const record = this.previewData?.record;\r\n    const collectionColumn = record?.collection;\r\n    const collection = !!collectionColumn ? collectionColumn[0] : id.split(\"|\")[0];\r\n    const rank = !!record ? record.rank : 0;\r\n    const passages = record?.matchingpassages?.passages;\r\n    const score = passages && passages.length ? passages[0].score : undefined;\r\n    return {\r\n      docid: id,\r\n      rank,\r\n      collection,\r\n      source: collection.split(\"|\")[0],\r\n      resultid: results.id,\r\n      querylang: queryLanguage,\r\n      querytext: q?.text,\r\n      filename: record?.filename,\r\n      fileext: record?.fileext,\r\n      score\r\n    }\r\n  }\r\n\r\n  private fetchExtracts(id: string, extracts: string[], data: PreviewData): Extract[] {\r\n    // extracts contains the html of extracts in chronological order\r\n    // locations contains the list of start positions sorted by score\r\n    const locations = data.highlightsPerCategory[this.highlightCategory]?.values[0]?.locations || [];\r\n\r\n    // first extract all the extracts locations\r\n    let extractslocations = locations.map((l, relevanceIndex) => ({\r\n      startIndex: l.start,\r\n      relevanceIndex\r\n    })) as ExtractsLocations[];\r\n\r\n    // sort them by start index\r\n    extractslocations.sort((a, b) => a.startIndex - b.startIndex);\r\n\r\n    // then extract the text of each extract\r\n    extractslocations = extractslocations.map((ex, textIndex) => ({\r\n      ...ex,\r\n      textIndex: textIndex,\r\n      text: extracts[textIndex] || \"\",\r\n      id: `${this.highlightCategory}_${textIndex}`,\r\n    }));\r\n\r\n    // then sanitize the text and remove empty extracts\r\n    const _extracts = extractslocations.filter(item => item.text.trim().length > 0).map(item => ({\r\n      ...item,\r\n      text: this.sanitizer.bypassSecurityTrustHtml(item.text)\r\n    }));\r\n\r\n    // finally sort them by relevance index\r\n    _extracts.sort((a, b) => a.relevanceIndex - b.relevanceIndex);\r\n\r\n    return _extracts;\r\n  }\r\n}"]}
@@ -0,0 +1,46 @@
1
+ import { HttpClient, HttpParams } from "@angular/common/http";
2
+ import { Injectable, inject } from "@angular/core";
3
+ import { EMPTY, catchError } from "rxjs";
4
+ import { globalConfig } from "@sinequa/atomic";
5
+ import * as i0 from "@angular/core";
6
+ export class PrincipalService {
7
+ constructor() {
8
+ this.http = inject(HttpClient);
9
+ this.API_URL = `${globalConfig.backendUrl}/${globalConfig.apiPath}`;
10
+ }
11
+ /**
12
+ * Retrieves the principal information from the server.
13
+ *
14
+ * @returns Observable<Principal> An observable that emits the principal information.
15
+ *
16
+ * @remarks
17
+ * This method sends a GET request to the API endpoint to fetch the principal data.
18
+ * It includes query parameters to specify the action and to indicate that no authentication is required.
19
+ * In case of an error, it logs the error to the console and returns an empty observable.
20
+ *
21
+ * @example
22
+ * ```typescript
23
+ * principalService.getPrincipal().subscribe(principal => {
24
+ * console.log(principal);
25
+ * });
26
+ * ```
27
+ */
28
+ getPrincipal() {
29
+ const params = new HttpParams().set("action", "get");
30
+ params.append("noAuthentication", "true");
31
+ return this.http.get(this.API_URL + "/principal", { params })
32
+ .pipe(catchError(error => {
33
+ console.error("PrincipalService.getPrincipal failure - error: ", error);
34
+ return EMPTY;
35
+ }));
36
+ }
37
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: PrincipalService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
38
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: PrincipalService, providedIn: 'root' }); }
39
+ }
40
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: PrincipalService, decorators: [{
41
+ type: Injectable,
42
+ args: [{
43
+ providedIn: 'root'
44
+ }]
45
+ }] });
46
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJpbmNpcGFsLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hdG9taWMtYW5ndWxhci9zcmMvbGliL3dlYi1zZXJ2aWNlcy9wcmluY2lwYWwuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLFVBQVUsRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBQzlELE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ25ELE9BQU8sRUFBRSxLQUFLLEVBQWMsVUFBVSxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBRXJELE9BQU8sRUFBYSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQzs7QUFLMUQsTUFBTSxPQUFPLGdCQUFnQjtJQUg3QjtRQUlxQixTQUFJLEdBQUcsTUFBTSxDQUFDLFVBQVUsQ0FBQyxDQUFDO1FBRTFCLFlBQU8sR0FBRyxHQUFHLFlBQVksQ0FBQyxVQUFVLElBQUksWUFBWSxDQUFDLE9BQU8sRUFBRSxDQUFDO0tBK0JuRjtJQTdCQzs7Ozs7Ozs7Ozs7Ozs7OztPQWdCRztJQUNILFlBQVk7UUFDVixNQUFNLE1BQU0sR0FBRyxJQUFJLFVBQVUsRUFBRSxDQUFDLEdBQUcsQ0FBQyxRQUFRLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFDckQsTUFBTSxDQUFDLE1BQU0sQ0FBQyxrQkFBa0IsRUFBRSxNQUFNLENBQUMsQ0FBQztRQUUxQyxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFZLElBQUksQ0FBQyxPQUFPLEdBQUcsWUFBWSxFQUFFLEVBQUUsTUFBTSxFQUFFLENBQUM7YUFDdkUsSUFBSSxDQUNILFVBQVUsQ0FBQyxLQUFLLENBQUMsRUFBRTtZQUNqQixPQUFPLENBQUMsS0FBSyxDQUFDLGlEQUFpRCxFQUFFLEtBQUssQ0FBQyxDQUFDO1lBQ3hFLE9BQU8sS0FBSyxDQUFDO1FBQ2YsQ0FBQyxDQUFDLENBQ0gsQ0FBQztJQUNKLENBQUM7OEdBakNVLGdCQUFnQjtrSEFBaEIsZ0JBQWdCLGNBRmYsTUFBTTs7MkZBRVAsZ0JBQWdCO2tCQUg1QixVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEh0dHBDbGllbnQsIEh0dHBQYXJhbXMgfSBmcm9tIFwiQGFuZ3VsYXIvY29tbW9uL2h0dHBcIjtcclxuaW1wb3J0IHsgSW5qZWN0YWJsZSwgaW5qZWN0IH0gZnJvbSBcIkBhbmd1bGFyL2NvcmVcIjtcclxuaW1wb3J0IHsgRU1QVFksIE9ic2VydmFibGUsIGNhdGNoRXJyb3IgfSBmcm9tIFwicnhqc1wiO1xyXG5cclxuaW1wb3J0IHsgUHJpbmNpcGFsLCBnbG9iYWxDb25maWcgfSBmcm9tIFwiQHNpbmVxdWEvYXRvbWljXCI7XHJcblxyXG5ASW5qZWN0YWJsZSh7XHJcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBQcmluY2lwYWxTZXJ2aWNlIHtcclxuICBwcm90ZWN0ZWQgcmVhZG9ubHkgaHR0cCA9IGluamVjdChIdHRwQ2xpZW50KTtcclxuXHJcbiAgcHJvdGVjdGVkIHJlYWRvbmx5IEFQSV9VUkwgPSBgJHtnbG9iYWxDb25maWcuYmFja2VuZFVybH0vJHtnbG9iYWxDb25maWcuYXBpUGF0aH1gO1xyXG5cclxuICAvKipcclxuICAgKiBSZXRyaWV2ZXMgdGhlIHByaW5jaXBhbCBpbmZvcm1hdGlvbiBmcm9tIHRoZSBzZXJ2ZXIuXHJcbiAgICpcclxuICAgKiBAcmV0dXJucyBPYnNlcnZhYmxlPFByaW5jaXBhbD4gQW4gb2JzZXJ2YWJsZSB0aGF0IGVtaXRzIHRoZSBwcmluY2lwYWwgaW5mb3JtYXRpb24uXHJcbiAgICpcclxuICAgKiBAcmVtYXJrc1xyXG4gICAqIFRoaXMgbWV0aG9kIHNlbmRzIGEgR0VUIHJlcXVlc3QgdG8gdGhlIEFQSSBlbmRwb2ludCB0byBmZXRjaCB0aGUgcHJpbmNpcGFsIGRhdGEuXHJcbiAgICogSXQgaW5jbHVkZXMgcXVlcnkgcGFyYW1ldGVycyB0byBzcGVjaWZ5IHRoZSBhY3Rpb24gYW5kIHRvIGluZGljYXRlIHRoYXQgbm8gYXV0aGVudGljYXRpb24gaXMgcmVxdWlyZWQuXHJcbiAgICogSW4gY2FzZSBvZiBhbiBlcnJvciwgaXQgbG9ncyB0aGUgZXJyb3IgdG8gdGhlIGNvbnNvbGUgYW5kIHJldHVybnMgYW4gZW1wdHkgb2JzZXJ2YWJsZS5cclxuICAgKlxyXG4gICAqIEBleGFtcGxlXHJcbiAgICogYGBgdHlwZXNjcmlwdFxyXG4gICAqIHByaW5jaXBhbFNlcnZpY2UuZ2V0UHJpbmNpcGFsKCkuc3Vic2NyaWJlKHByaW5jaXBhbCA9PiB7XHJcbiAgICogICBjb25zb2xlLmxvZyhwcmluY2lwYWwpO1xyXG4gICAqIH0pO1xyXG4gICAqIGBgYFxyXG4gICAqL1xyXG4gIGdldFByaW5jaXBhbCgpOiBPYnNlcnZhYmxlPFByaW5jaXBhbD4ge1xyXG4gICAgY29uc3QgcGFyYW1zID0gbmV3IEh0dHBQYXJhbXMoKS5zZXQoXCJhY3Rpb25cIiwgXCJnZXRcIik7XHJcbiAgICBwYXJhbXMuYXBwZW5kKFwibm9BdXRoZW50aWNhdGlvblwiLCBcInRydWVcIik7XHJcblxyXG4gICAgcmV0dXJuIHRoaXMuaHR0cC5nZXQ8UHJpbmNpcGFsPih0aGlzLkFQSV9VUkwgKyBcIi9wcmluY2lwYWxcIiwgeyBwYXJhbXMgfSlcclxuICAgIC5waXBlKFxyXG4gICAgICBjYXRjaEVycm9yKGVycm9yID0+IHtcclxuICAgICAgICBjb25zb2xlLmVycm9yKFwiUHJpbmNpcGFsU2VydmljZS5nZXRQcmluY2lwYWwgZmFpbHVyZSAtIGVycm9yOiBcIiwgZXJyb3IpO1xyXG4gICAgICAgIHJldHVybiBFTVBUWTtcclxuICAgICAgfSlcclxuICAgICk7XHJcbiAgfVxyXG59Il19
@@ -0,0 +1,123 @@
1
+ import { HttpClient } from '@angular/common/http';
2
+ import { Injectable, inject } from '@angular/core';
3
+ import { EMPTY, catchError, map, of, tap } from 'rxjs';
4
+ import { buildPathsAndLevels, globalConfig } from '@sinequa/atomic';
5
+ import { AppStore, QueryParamsStore } from '../stores';
6
+ import * as i0 from "@angular/core";
7
+ const AGGREGATION_MAX_COUNT = 11;
8
+ export class QueryService {
9
+ constructor() {
10
+ this.http = inject(HttpClient);
11
+ this.appStore = inject(AppStore);
12
+ this.queryParamsStore = inject(QueryParamsStore);
13
+ this.API_URL = `${globalConfig.backendUrl}/${globalConfig.apiPath}`;
14
+ }
15
+ /**
16
+ * Performs a search query.
17
+ *
18
+ * @param q - The partial query object.
19
+ * @param includeQueryParams - Indicates whether to include query parameters automatically in the request.
20
+ * @param audit - The audit events object.
21
+ * @returns An observable that emits the search results.
22
+ */
23
+ search(q, includeQueryParams = true, audit) {
24
+ const $auditRecord = audit ? { auditEvents: [audit] } : undefined;
25
+ const { app } = globalConfig;
26
+ const currentQuery = q ?? this.queryParamsStore.getQuery();
27
+ const query = includeQueryParams ? { ...this.queryParamsStore.getQuery(), ...currentQuery } : currentQuery;
28
+ // Check if the search query is empty and if empty searches are allowed
29
+ // If not allowed, return an empty result
30
+ const allowEmptySearch = this.appStore.allowEmptySearch(query?.name || '');
31
+ if (allowEmptySearch === false && query?.text === '') {
32
+ return of({ records: [] });
33
+ }
34
+ const body = {
35
+ app,
36
+ query,
37
+ $auditRecord
38
+ };
39
+ return this.http.post(this.API_URL + "/query", body)
40
+ .pipe(catchError(error => {
41
+ console.error("queryService.getResults failure - error: ", error);
42
+ return of({});
43
+ }), map((result) => {
44
+ // update $hasMore flag
45
+ result.aggregations.forEach((agg) => {
46
+ agg.$hasMore = false;
47
+ if (!agg.isDistribution && !agg.isTree && agg.items) {
48
+ agg.$hasMore = agg.items.length === AGGREGATION_MAX_COUNT;
49
+ }
50
+ if (agg.isTree) {
51
+ buildPathsAndLevels(agg);
52
+ }
53
+ });
54
+ return result;
55
+ }), map(result => {
56
+ result.records?.map((article) => (Object.assign(article, { value: article.title, type: 'default' })));
57
+ return result;
58
+ }), map(result => {
59
+ const r = ({ ...result, nextPage: result.page < Math.ceil(result.rowCount / result.pageSize) ? result.page + 1 : undefined, previousPage: result.page > 1 ? result.page - 1 : undefined });
60
+ return r;
61
+ }), tap(response => console.log("queryService.getResults success - data: ", response)));
62
+ }
63
+ /**
64
+ * Performs a bulk search operation.
65
+ *
66
+ * @param q An array of Query objects representing the search queries.
67
+ * @param audit An optional AuditEvents object for auditing purposes.
68
+ * @returns An Observable that emits an array of Result objects.
69
+ */
70
+ bulkSearch(q, audit) {
71
+ const { app = '' } = globalConfig;
72
+ const body = {
73
+ methods: [],
74
+ propagateErrors: true,
75
+ $auditRecord: { auditEvents: [audit] }
76
+ };
77
+ for (const query of q) {
78
+ body.methods.push({
79
+ method: "query",
80
+ app,
81
+ query
82
+ });
83
+ }
84
+ return this.http.post(this.API_URL + "/multi", body)
85
+ .pipe(catchError(error => {
86
+ console.error("queryService.bulkSearch failure - error: ", error);
87
+ return EMPTY;
88
+ }), map(response => {
89
+ // update $hasMore flag
90
+ response.results.forEach((result) => {
91
+ result.aggregations.forEach((agg) => {
92
+ agg.$hasMore = false;
93
+ if (!agg.isDistribution && !agg.isTree && agg.items) {
94
+ agg.$hasMore = agg.items.length === AGGREGATION_MAX_COUNT;
95
+ }
96
+ if (agg.isTree) {
97
+ buildPathsAndLevels(agg);
98
+ }
99
+ });
100
+ });
101
+ return response.results;
102
+ }), map(results => {
103
+ results.forEach(result => result.records?.map((article) => (Object.assign(article, { value: article.title, type: 'default' }))));
104
+ return results;
105
+ }), map(results => {
106
+ return results.map(result => {
107
+ const r = ({ ...result, nextPage: result.page < Math.ceil(result.rowCount / result.pageSize) ? result.page + 1 : undefined, previousPage: result.page > 1 ? result.page - 1 : undefined });
108
+ return r;
109
+ });
110
+ }),
111
+ // map(response => ResultsSchema.parse(response) as T),
112
+ tap(response => console.log("queryService.bulkSearch success - data: ", response)));
113
+ }
114
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: QueryService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
115
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: QueryService, providedIn: 'root' }); }
116
+ }
117
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: QueryService, decorators: [{
118
+ type: Injectable,
119
+ args: [{
120
+ providedIn: 'root'
121
+ }]
122
+ }] });
123
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query.service.js","sourceRoot":"","sources":["../../../../../projects/atomic-angular/src/lib/web-services/query.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient } from '@angular/common/http';\r\nimport { Injectable, inject } from '@angular/core';\r\nimport { EMPTY, Observable, catchError, map, of, tap } from 'rxjs';\r\n\r\nimport { Aggregation, Article, AuditEvents, CustomHighlights, PreviewData, Query, Result, TreeAggregation, TreeAggregationNode, buildPathsAndLevels, globalConfig } from '@sinequa/atomic';\r\n\r\nimport { AppStore, QueryParamsStore } from '../stores';\r\n\r\nconst AGGREGATION_MAX_COUNT = 11;\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class QueryService {\r\n  protected readonly http = inject(HttpClient);\r\n\r\n  private readonly appStore = inject(AppStore);\r\n  private readonly queryParamsStore = inject(QueryParamsStore)\r\n\r\n  protected readonly API_URL = `${globalConfig.backendUrl}/${globalConfig.apiPath}`;\r\n\r\n  /**\r\n   * Performs a search query.\r\n   *\r\n   * @param q - The partial query object.\r\n   * @param includeQueryParams - Indicates whether to include query parameters automatically in the request.\r\n   * @param audit - The audit events object.\r\n   * @returns An observable that emits the search results.\r\n   */\r\n  search(q?: Partial<Query>, includeQueryParams = true, audit?: AuditEvents): Observable<Result> {\r\n    const $auditRecord = audit ? { auditEvents: [audit] } : undefined;\r\n\r\n    const { app } = globalConfig;\r\n    const currentQuery = q ?? this.queryParamsStore.getQuery();\r\n    const query = includeQueryParams ? { ...this.queryParamsStore.getQuery(), ...currentQuery } : currentQuery;\r\n\r\n    // Check if the search query is empty and if empty searches are allowed\r\n    // If not allowed, return an empty result\r\n    const allowEmptySearch = this.appStore.allowEmptySearch(query?.name || '');\r\n    if (allowEmptySearch === false && query?.text === '') {\r\n      return of({ records: [] as Article[] } as Result);\r\n    }\r\n\r\n    const body = {\r\n      app,\r\n      query,\r\n      $auditRecord\r\n    };\r\n\r\n    return this.http.post<Result>(this.API_URL + \"/query\", body)\r\n      .pipe(\r\n        catchError(error => {\r\n          console.error(\"queryService.getResults failure - error: \", error);\r\n          return of({} as Result);\r\n        }),\r\n        map((result: Result) => {\r\n          // update $hasMore flag\r\n          result.aggregations.forEach((agg: Aggregation) => {\r\n            agg.$hasMore = false;\r\n            if (!agg.isDistribution && !agg.isTree && agg.items) {\r\n              agg.$hasMore = agg.items.length === AGGREGATION_MAX_COUNT;\r\n            }\r\n            if (agg.isTree) {\r\n              buildPathsAndLevels(agg as TreeAggregation);\r\n            }\r\n          });\r\n          return result;\r\n        }),\r\n        map(result => {\r\n          result.records?.map((article: Article) => (Object.assign(article, { value: article.title, type: 'default' })));\r\n          return result;\r\n        }),\r\n        map(result => {\r\n          const r = ({ ...result, nextPage: result.page < Math.ceil(result.rowCount / result.pageSize) ? result.page + 1 : undefined, previousPage: result.page > 1 ? result.page - 1 : undefined })\r\n          return r;\r\n        }),\r\n        tap(response => console.log(\"queryService.getResults success - data: \", response)),\r\n      );\r\n  }\r\n\r\n  /**\r\n   * Performs a bulk search operation.\r\n   *\r\n   * @param q An array of Query objects representing the search queries.\r\n   * @param audit An optional AuditEvents object for auditing purposes.\r\n   * @returns An Observable that emits an array of Result objects.\r\n   */\r\n  bulkSearch(q: Query[], audit?: AuditEvents): Observable<Result[]> {\r\n\r\n    const { app = '' } = globalConfig;\r\n\r\n    const body: {\r\n      methods: {\r\n        method: string,\r\n        app: string,\r\n        query: Query\r\n      }[],\r\n      propagateErrors: true,\r\n      $auditRecord?: { auditEvents: [AuditEvents] }\r\n    } = {\r\n      methods: [],\r\n      propagateErrors: true,\r\n      $auditRecord: { auditEvents: [audit!] }\r\n    };\r\n\r\n    for (const query of q) {\r\n      body.methods.push({\r\n        method: \"query\",\r\n        app,\r\n        query\r\n      });\r\n    }\r\n\r\n    return this.http.post<{ results: Result[] }>(this.API_URL + \"/multi\", body)\r\n      .pipe(\r\n        catchError(error => {\r\n          console.error(\"queryService.bulkSearch failure - error: \", error);\r\n          return EMPTY;\r\n        }),\r\n        map(response => {\r\n          // update $hasMore flag\r\n          response.results.forEach((result: Result) => {\r\n            result.aggregations.forEach((agg: Aggregation) => {\r\n              agg.$hasMore = false;\r\n              if (!agg.isDistribution && !agg.isTree && agg.items) {\r\n                agg.$hasMore = agg.items.length === AGGREGATION_MAX_COUNT;\r\n              }\r\n              if (agg.isTree) {\r\n                buildPathsAndLevels(agg as TreeAggregation);\r\n              }\r\n            });\r\n          })\r\n\r\n          return response.results;\r\n        }),\r\n        map(results => {\r\n          results.forEach(result =>  result.records?.map((article: Article) => (Object.assign(article, { value: article.title, type: 'default' }))));\r\n          return results;\r\n        }),\r\n        map(results => {\r\n          return results.map(result => {\r\n            const r = ({ ...result, nextPage: result.page < Math.ceil(result.rowCount / result.pageSize) ? result.page + 1 : undefined, previousPage: result.page > 1 ? result.page - 1 : undefined })\r\n            return r;\r\n          });\r\n        }),\r\n        // map(response => ResultsSchema.parse(response) as T),\r\n        tap(response => console.log(\"queryService.bulkSearch success - data: \", response)),\r\n      );\r\n\r\n  }\r\n}\r\n"]}