@gnggln/ng-ui-system 1.0.0-alpha.14 → 1.0.0-alpha.16
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.
- package/base-layout/lib/components/blackbox/blackbox.service.d.ts +1 -13
- package/blackbox/lib/components/blackbox/blackbox.service.d.ts +1 -13
- package/button/lib/components/blackbox/blackbox.service.d.ts +1 -13
- package/crud-table/lib/components/blackbox/blackbox.service.d.ts +1 -13
- package/crud-table/lib/components/form-builder/form-builder.component.d.ts +2 -0
- package/crud-table/lib/components/form-builder/types/field.types.d.ts +9 -2
- package/crud-table/lib/components/form-builder/types/schema.types.d.ts +2 -0
- package/crud-table/lib/core/logging/logger.config.d.ts +18 -0
- package/crud-table/lib/core/logging/logger.service.d.ts +75 -0
- package/crud-table/lib/core/logging/logger.types.d.ts +70 -0
- package/esm2022/base-layout/lib/components/blackbox/blackbox.service.mjs +8 -57
- package/esm2022/blackbox/lib/components/blackbox/blackbox.service.mjs +8 -57
- package/esm2022/button/lib/components/blackbox/blackbox.service.mjs +8 -57
- package/esm2022/crud-table/lib/components/blackbox/blackbox.service.mjs +8 -57
- package/esm2022/crud-table/lib/components/form-builder/form-builder.component.mjs +37 -20
- package/esm2022/crud-table/lib/components/form-builder/types/field.types.mjs +1 -1
- package/esm2022/crud-table/lib/components/form-builder/types/schema.types.mjs +1 -1
- package/esm2022/crud-table/lib/core/logging/logger.config.mjs +18 -0
- package/esm2022/crud-table/lib/core/logging/logger.service.mjs +295 -0
- package/esm2022/crud-table/lib/core/logging/logger.types.mjs +7 -0
- package/esm2022/form-builder/lib/components/blackbox/blackbox.service.mjs +8 -57
- package/esm2022/form-builder/lib/components/form-builder/form-builder.component.mjs +37 -20
- package/esm2022/form-builder/lib/components/form-builder/form-wizard.component.mjs +464 -236
- package/esm2022/form-builder/lib/components/form-builder/types/field.types.mjs +1 -1
- package/esm2022/form-builder/lib/components/form-builder/types/schema.types.mjs +1 -1
- package/esm2022/form-builder/lib/core/logging/logger.config.mjs +18 -0
- package/esm2022/form-builder/lib/core/logging/logger.service.mjs +295 -0
- package/esm2022/form-builder/lib/core/logging/logger.types.mjs +7 -0
- package/esm2022/form-builder-editor/lib/components/blackbox/blackbox.service.mjs +8 -57
- package/esm2022/form-builder-editor/lib/components/form-builder/form-builder.component.mjs +37 -20
- package/esm2022/form-builder-editor/lib/components/form-builder/types/field.types.mjs +1 -1
- package/esm2022/form-builder-editor/lib/components/form-builder/types/schema.types.mjs +1 -1
- package/esm2022/form-builder-editor/lib/components/form-builder-editor/index.mjs +3 -1
- package/esm2022/form-builder-editor/lib/components/form-builder-editor/presets/editor-presets.mjs +395 -0
- package/esm2022/form-builder-editor/lib/core/logging/logger.config.mjs +18 -0
- package/esm2022/form-builder-editor/lib/core/logging/logger.service.mjs +295 -0
- package/esm2022/form-builder-editor/lib/core/logging/logger.types.mjs +7 -0
- package/esm2022/http/lib/components/blackbox/blackbox.service.mjs +8 -57
- package/esm2022/lib/components/blackbox/blackbox.service.mjs +8 -57
- package/esm2022/lib/components/form-builder/form-builder.component.mjs +37 -20
- package/esm2022/lib/components/form-builder/form-wizard.component.mjs +464 -236
- package/esm2022/lib/components/form-builder/types/field.types.mjs +1 -1
- package/esm2022/lib/components/form-builder/types/schema.types.mjs +1 -1
- package/esm2022/lib/components/form-builder-editor/index.mjs +3 -1
- package/esm2022/lib/components/form-builder-editor/presets/editor-presets.mjs +395 -0
- package/esm2022/lib/core/logging/index.mjs +10 -0
- package/esm2022/lib/core/logging/logger.config.mjs +18 -0
- package/esm2022/lib/core/logging/logger.service.mjs +295 -0
- package/esm2022/lib/core/logging/logger.types.mjs +7 -0
- package/esm2022/lib/version/ng-ui-system-version.mjs +12 -0
- package/esm2022/public-api.mjs +5 -2
- package/fesm2022/gnggln-ng-ui-system-base-layout.mjs +7 -56
- package/fesm2022/gnggln-ng-ui-system-base-layout.mjs.map +1 -1
- package/fesm2022/gnggln-ng-ui-system-blackbox.mjs +7 -56
- package/fesm2022/gnggln-ng-ui-system-blackbox.mjs.map +1 -1
- package/fesm2022/gnggln-ng-ui-system-button.mjs +7 -56
- package/fesm2022/gnggln-ng-ui-system-button.mjs.map +1 -1
- package/fesm2022/gnggln-ng-ui-system-crud-table.mjs +351 -75
- package/fesm2022/gnggln-ng-ui-system-crud-table.mjs.map +1 -1
- package/fesm2022/gnggln-ng-ui-system-form-builder-editor.mjs +747 -76
- package/fesm2022/gnggln-ng-ui-system-form-builder-editor.mjs.map +1 -1
- package/fesm2022/gnggln-ng-ui-system-form-builder.mjs +813 -310
- package/fesm2022/gnggln-ng-ui-system-form-builder.mjs.map +1 -1
- package/fesm2022/gnggln-ng-ui-system-http.mjs +7 -56
- package/fesm2022/gnggln-ng-ui-system-http.mjs.map +1 -1
- package/fesm2022/gnggln-ng-ui-system.mjs +1232 -315
- package/fesm2022/gnggln-ng-ui-system.mjs.map +1 -1
- package/form-builder/lib/components/blackbox/blackbox.service.d.ts +1 -13
- package/form-builder/lib/components/form-builder/form-builder.component.d.ts +2 -0
- package/form-builder/lib/components/form-builder/form-wizard.component.d.ts +34 -0
- package/form-builder/lib/components/form-builder/types/field.types.d.ts +9 -2
- package/form-builder/lib/components/form-builder/types/schema.types.d.ts +2 -0
- package/form-builder/lib/core/logging/logger.config.d.ts +18 -0
- package/form-builder/lib/core/logging/logger.service.d.ts +75 -0
- package/form-builder/lib/core/logging/logger.types.d.ts +70 -0
- package/form-builder-editor/lib/components/blackbox/blackbox.service.d.ts +1 -13
- package/form-builder-editor/lib/components/form-builder/form-builder.component.d.ts +2 -0
- package/form-builder-editor/lib/components/form-builder/types/field.types.d.ts +9 -2
- package/form-builder-editor/lib/components/form-builder/types/schema.types.d.ts +2 -0
- package/form-builder-editor/lib/components/form-builder-editor/index.d.ts +2 -0
- package/form-builder-editor/lib/components/form-builder-editor/presets/editor-presets.d.ts +25 -0
- package/form-builder-editor/lib/core/logging/logger.config.d.ts +18 -0
- package/form-builder-editor/lib/core/logging/logger.service.d.ts +75 -0
- package/form-builder-editor/lib/core/logging/logger.types.d.ts +70 -0
- package/http/lib/components/blackbox/blackbox.service.d.ts +1 -13
- package/lib/components/blackbox/blackbox.service.d.ts +1 -13
- package/lib/components/form-builder/form-builder.component.d.ts +2 -0
- package/lib/components/form-builder/form-wizard.component.d.ts +34 -0
- package/lib/components/form-builder/types/field.types.d.ts +9 -2
- package/lib/components/form-builder/types/schema.types.d.ts +2 -0
- package/lib/components/form-builder-editor/index.d.ts +2 -0
- package/lib/components/form-builder-editor/presets/editor-presets.d.ts +25 -0
- package/lib/core/logging/index.d.ts +8 -0
- package/lib/core/logging/logger.config.d.ts +18 -0
- package/lib/core/logging/logger.service.d.ts +75 -0
- package/lib/core/logging/logger.types.d.ts +70 -0
- package/lib/version/ng-ui-system-version.d.ts +9 -0
- package/package.json +13 -13
- package/public-api.d.ts +2 -0
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
* - Root singleton (`providedIn: 'root'`)
|
|
8
8
|
* - On first inject: generates fingerprint, creates session, subscribes to Router events
|
|
9
9
|
* - Centralised in-memory buffer with periodic/threshold-based flush to IndexedDB
|
|
10
|
-
* - `window:beforeunload` triggers a final
|
|
10
|
+
* - `window:beforeunload` triggers a final best-effort async flush
|
|
11
11
|
*/
|
|
12
12
|
import { OnDestroy } from '@angular/core';
|
|
13
13
|
import { UiBlackboxConfig, UiBlackboxFormTrackingConfig, UiBlackboxSession, UiBlackboxFormEvent, UiBlackboxHttpCall } from './blackbox.types';
|
|
@@ -136,18 +136,6 @@ export declare class UiBlackboxService implements OnDestroy {
|
|
|
136
136
|
* @internal
|
|
137
137
|
*/
|
|
138
138
|
private flush;
|
|
139
|
-
/**
|
|
140
|
-
* Synchronous flush for `beforeunload`.
|
|
141
|
-
* Falls back to a best-effort approach since IndexedDB is async.
|
|
142
|
-
* @internal
|
|
143
|
-
*/
|
|
144
|
-
private flushSync;
|
|
145
|
-
/**
|
|
146
|
-
* Recover sessions that were saved to localStorage during `beforeunload`
|
|
147
|
-
* and persist them to IndexedDB. Called on next service init.
|
|
148
|
-
* @internal
|
|
149
|
-
*/
|
|
150
|
-
private recoverPendingSessions;
|
|
151
139
|
/** Generate a UUID v4. */
|
|
152
140
|
private generateUuid;
|
|
153
141
|
ngOnDestroy(): void;
|
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
* - Root singleton (`providedIn: 'root'`)
|
|
8
8
|
* - On first inject: generates fingerprint, creates session, subscribes to Router events
|
|
9
9
|
* - Centralised in-memory buffer with periodic/threshold-based flush to IndexedDB
|
|
10
|
-
* - `window:beforeunload` triggers a final
|
|
10
|
+
* - `window:beforeunload` triggers a final best-effort async flush
|
|
11
11
|
*/
|
|
12
12
|
import { OnDestroy } from '@angular/core';
|
|
13
13
|
import { UiBlackboxConfig, UiBlackboxFormTrackingConfig, UiBlackboxSession, UiBlackboxFormEvent, UiBlackboxHttpCall } from './blackbox.types';
|
|
@@ -136,18 +136,6 @@ export declare class UiBlackboxService implements OnDestroy {
|
|
|
136
136
|
* @internal
|
|
137
137
|
*/
|
|
138
138
|
private flush;
|
|
139
|
-
/**
|
|
140
|
-
* Synchronous flush for `beforeunload`.
|
|
141
|
-
* Falls back to a best-effort approach since IndexedDB is async.
|
|
142
|
-
* @internal
|
|
143
|
-
*/
|
|
144
|
-
private flushSync;
|
|
145
|
-
/**
|
|
146
|
-
* Recover sessions that were saved to localStorage during `beforeunload`
|
|
147
|
-
* and persist them to IndexedDB. Called on next service init.
|
|
148
|
-
* @internal
|
|
149
|
-
*/
|
|
150
|
-
private recoverPendingSessions;
|
|
151
139
|
/** Generate a UUID v4. */
|
|
152
140
|
private generateUuid;
|
|
153
141
|
ngOnDestroy(): void;
|
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
* - Root singleton (`providedIn: 'root'`)
|
|
8
8
|
* - On first inject: generates fingerprint, creates session, subscribes to Router events
|
|
9
9
|
* - Centralised in-memory buffer with periodic/threshold-based flush to IndexedDB
|
|
10
|
-
* - `window:beforeunload` triggers a final
|
|
10
|
+
* - `window:beforeunload` triggers a final best-effort async flush
|
|
11
11
|
*/
|
|
12
12
|
import { OnDestroy } from '@angular/core';
|
|
13
13
|
import { UiBlackboxConfig, UiBlackboxFormTrackingConfig, UiBlackboxSession, UiBlackboxFormEvent, UiBlackboxHttpCall } from './blackbox.types';
|
|
@@ -136,18 +136,6 @@ export declare class UiBlackboxService implements OnDestroy {
|
|
|
136
136
|
* @internal
|
|
137
137
|
*/
|
|
138
138
|
private flush;
|
|
139
|
-
/**
|
|
140
|
-
* Synchronous flush for `beforeunload`.
|
|
141
|
-
* Falls back to a best-effort approach since IndexedDB is async.
|
|
142
|
-
* @internal
|
|
143
|
-
*/
|
|
144
|
-
private flushSync;
|
|
145
|
-
/**
|
|
146
|
-
* Recover sessions that were saved to localStorage during `beforeunload`
|
|
147
|
-
* and persist them to IndexedDB. Called on next service init.
|
|
148
|
-
* @internal
|
|
149
|
-
*/
|
|
150
|
-
private recoverPendingSessions;
|
|
151
139
|
/** Generate a UUID v4. */
|
|
152
140
|
private generateUuid;
|
|
153
141
|
ngOnDestroy(): void;
|
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
* - Root singleton (`providedIn: 'root'`)
|
|
8
8
|
* - On first inject: generates fingerprint, creates session, subscribes to Router events
|
|
9
9
|
* - Centralised in-memory buffer with periodic/threshold-based flush to IndexedDB
|
|
10
|
-
* - `window:beforeunload` triggers a final
|
|
10
|
+
* - `window:beforeunload` triggers a final best-effort async flush
|
|
11
11
|
*/
|
|
12
12
|
import { OnDestroy } from '@angular/core';
|
|
13
13
|
import { UiBlackboxConfig, UiBlackboxFormTrackingConfig, UiBlackboxSession, UiBlackboxFormEvent, UiBlackboxHttpCall } from './blackbox.types';
|
|
@@ -136,18 +136,6 @@ export declare class UiBlackboxService implements OnDestroy {
|
|
|
136
136
|
* @internal
|
|
137
137
|
*/
|
|
138
138
|
private flush;
|
|
139
|
-
/**
|
|
140
|
-
* Synchronous flush for `beforeunload`.
|
|
141
|
-
* Falls back to a best-effort approach since IndexedDB is async.
|
|
142
|
-
* @internal
|
|
143
|
-
*/
|
|
144
|
-
private flushSync;
|
|
145
|
-
/**
|
|
146
|
-
* Recover sessions that were saved to localStorage during `beforeunload`
|
|
147
|
-
* and persist them to IndexedDB. Called on next service init.
|
|
148
|
-
* @internal
|
|
149
|
-
*/
|
|
150
|
-
private recoverPendingSessions;
|
|
151
139
|
/** Generate a UUID v4. */
|
|
152
140
|
private generateUuid;
|
|
153
141
|
ngOnDestroy(): void;
|
|
@@ -44,6 +44,8 @@ export declare class UiFormBuilderComponent implements OnInit, OnChanges, OnDest
|
|
|
44
44
|
private readonly blackbox;
|
|
45
45
|
/** @internal Router inject for BlackBox route context. */
|
|
46
46
|
private readonly bbRouter;
|
|
47
|
+
/** @internal Optional Logger service — logs form actions when available. */
|
|
48
|
+
private readonly logger;
|
|
47
49
|
private readonly destroy$;
|
|
48
50
|
private valueChangeSub?;
|
|
49
51
|
/** @internal Listener cleanup for BlackBox focus/blur tracking. */
|
|
@@ -273,9 +273,16 @@ export interface UiFormFieldDescriptor {
|
|
|
273
273
|
searchable?: boolean;
|
|
274
274
|
/** Mostra pulsante "Seleziona tutto" per multiselect. */
|
|
275
275
|
allowSelectAll?: boolean;
|
|
276
|
-
/**
|
|
276
|
+
/**
|
|
277
|
+
* Con `searchable: true` su `select`/`text`, ogni `input` dell'autocomplete
|
|
278
|
+
* invoca la callback: il risultato sostituisce l'elenco opzioni mostrato.
|
|
279
|
+
*/
|
|
277
280
|
asyncOptions?: (query: string) => Promise<UiFieldOption[]>;
|
|
278
|
-
/**
|
|
281
|
+
/**
|
|
282
|
+
* Contratto per risolvere opzioni da id (es. prefill da API). Il Form Builder
|
|
283
|
+
* Angular attuale non invoca questa callback; l'app può comporre le opzioni
|
|
284
|
+
* in `options` o collegarla a logica custom.
|
|
285
|
+
*/
|
|
279
286
|
recoverAsyncValues?: (ids: any[]) => Promise<UiFieldOption[]>;
|
|
280
287
|
/** Generazione dinamica di opzioni. */
|
|
281
288
|
dynamicOptions?: UiDynamicOptionsConfig;
|
|
@@ -191,6 +191,8 @@ export interface UiWizardConfig {
|
|
|
191
191
|
onStepSave?: (stepData: Record<string, any>, stepNumber: number) => Promise<void> | void;
|
|
192
192
|
/** Mostra pulsanti di navigazione (precedente/successivo). @default true */
|
|
193
193
|
showNavigationButtons?: boolean;
|
|
194
|
+
/** Posizione dei pulsanti di navigazione. @default 'bottom' */
|
|
195
|
+
navigationPosition?: 'top' | 'bottom' | 'both';
|
|
194
196
|
/** Label personalizzate per i pulsanti del wizard. */
|
|
195
197
|
buttonLabels?: {
|
|
196
198
|
previous?: string;
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module ng-ui-system/core/logging
|
|
3
|
+
* Configurazione IndexedDB per UiLoggerService.
|
|
4
|
+
* Pattern omologato a action-logger.config.ts del progetto Siform-PAL.
|
|
5
|
+
*/
|
|
6
|
+
import type { UiLogLevel } from './logger.types';
|
|
7
|
+
/** Durata default retention log (7 giorni). */
|
|
8
|
+
export declare const UI_LOGGER_DEFAULT_TTL_MS: number;
|
|
9
|
+
/** Nome database IndexedDB. */
|
|
10
|
+
export declare const UI_LOGGER_IDB_NAME = "ng-ui-system-logs";
|
|
11
|
+
/** Versione database (incrementare su schema change). */
|
|
12
|
+
export declare const UI_LOGGER_IDB_VERSION = 1;
|
|
13
|
+
/** Nome object store. */
|
|
14
|
+
export declare const UI_LOGGER_IDB_STORE = "entries";
|
|
15
|
+
/** Max entry totali per cleanup automatico. */
|
|
16
|
+
export declare const UI_LOGGER_MAX_ENTRIES = 10000;
|
|
17
|
+
/** Livelli di log ordinati per severità crescente. */
|
|
18
|
+
export declare const UI_LOGGER_LEVELS: readonly UiLogLevel[];
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import type { UiLogCategory, UiLogEntry, UiLogFilter, UiLogLevel, UiLogSummary, UiLoggerConfig } from './logger.types';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* Logger centralizzato per componenti ng-ui-system.
|
|
5
|
+
* Persiste su IndexedDB con cleanup automatico.
|
|
6
|
+
*/
|
|
7
|
+
export declare class UiLoggerService {
|
|
8
|
+
private readonly platformId;
|
|
9
|
+
private dbPromise;
|
|
10
|
+
/** Stato toggle runtime. */
|
|
11
|
+
readonly enabled: import("@angular/core").WritableSignal<boolean>;
|
|
12
|
+
/** Configurazione runtime. */
|
|
13
|
+
private config;
|
|
14
|
+
/** True se IndexedDB disponibile e siamo in browser. */
|
|
15
|
+
private idbSupported;
|
|
16
|
+
/** Rileva se logging attivo (default: true in dev se non production). */
|
|
17
|
+
private detectEnabled;
|
|
18
|
+
/** Abilita/disabilita logging runtime. */
|
|
19
|
+
setEnabled(value: boolean): void;
|
|
20
|
+
/** Configura il logger. */
|
|
21
|
+
configure(config: Partial<UiLoggerConfig>): void;
|
|
22
|
+
/**
|
|
23
|
+
* Log generico con controllo enabled e livello.
|
|
24
|
+
*/
|
|
25
|
+
log(category: UiLogCategory, level: UiLogLevel, action: string, payload?: Record<string, unknown>, duration?: number): void;
|
|
26
|
+
/**
|
|
27
|
+
* Log azione form (submit, reset, valueChange, etc).
|
|
28
|
+
*/
|
|
29
|
+
logFormAction(action: 'init' | 'submit' | 'reset' | 'valueChange' | 'validationChange' | 'customEvent' | 'destroy', payload: {
|
|
30
|
+
schemaId: string;
|
|
31
|
+
[key: string]: unknown;
|
|
32
|
+
}): void;
|
|
33
|
+
/**
|
|
34
|
+
* Log azione wizard (stepChange, complete, validationFailed, etc).
|
|
35
|
+
*/
|
|
36
|
+
logWizardAction(action: 'init' | 'stepChange' | 'stepComplete' | 'stepValidationFailed' | 'complete' | 'navigation' | 'save' | 'valueChange' | 'keyboardNavigation' | 'destroy', payload: {
|
|
37
|
+
wizardId?: string;
|
|
38
|
+
step?: number;
|
|
39
|
+
[key: string]: unknown;
|
|
40
|
+
}): void;
|
|
41
|
+
/**
|
|
42
|
+
* Log errore.
|
|
43
|
+
*/
|
|
44
|
+
logError(error: Error | unknown, context?: Record<string, unknown>): void;
|
|
45
|
+
/**
|
|
46
|
+
* Log debug (solo se enabled e minLevel <= debug).
|
|
47
|
+
*/
|
|
48
|
+
debug(category: UiLogCategory, action: string, payload?: Record<string, unknown>): void;
|
|
49
|
+
private persist;
|
|
50
|
+
/** Recupera userId da localStorage se disponibile. */
|
|
51
|
+
private getUserId;
|
|
52
|
+
/** Verifica se il livello deve essere loggato in base a minLevel. */
|
|
53
|
+
private shouldLogLevel;
|
|
54
|
+
/**
|
|
55
|
+
* Recupera log con filtri.
|
|
56
|
+
*/
|
|
57
|
+
query(filter?: UiLogFilter): Promise<UiLogEntry[]>;
|
|
58
|
+
/**
|
|
59
|
+
* Riepilogo log per dashboard.
|
|
60
|
+
*/
|
|
61
|
+
summary(): Promise<UiLogSummary>;
|
|
62
|
+
/**
|
|
63
|
+
* Cancella log più vecchi di TTL.
|
|
64
|
+
*/
|
|
65
|
+
purgeOld(): Promise<number>;
|
|
66
|
+
/**
|
|
67
|
+
* Cancella tutti i log.
|
|
68
|
+
*/
|
|
69
|
+
clear(): Promise<void>;
|
|
70
|
+
private openDb;
|
|
71
|
+
private cleanupIfNeeded;
|
|
72
|
+
private applyFilter;
|
|
73
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<UiLoggerService, never>;
|
|
74
|
+
static ɵprov: i0.ɵɵInjectableDeclaration<UiLoggerService>;
|
|
75
|
+
}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module ng-ui-system/core/logging
|
|
3
|
+
* Tipologie per il sistema di logging centralizzato della libreria.
|
|
4
|
+
* Pattern omologato a ActionLoggerService del progetto Siform-PAL.
|
|
5
|
+
*/
|
|
6
|
+
/**
|
|
7
|
+
* Categorie di log per filtraggio e organizzazione.
|
|
8
|
+
*/
|
|
9
|
+
export type UiLogCategory = 'form' | 'wizard' | 'navigation' | 'system' | 'error';
|
|
10
|
+
/**
|
|
11
|
+
* Livelli di severità log.
|
|
12
|
+
*/
|
|
13
|
+
export type UiLogLevel = 'debug' | 'info' | 'warn' | 'error';
|
|
14
|
+
/**
|
|
15
|
+
* Entry singola nel log. Persistita su IndexedDB.
|
|
16
|
+
*/
|
|
17
|
+
export interface UiLogEntry {
|
|
18
|
+
/** ID univoco (timestamp + random). */
|
|
19
|
+
id: string;
|
|
20
|
+
/** Categoria log. */
|
|
21
|
+
category: UiLogCategory;
|
|
22
|
+
/** Livello severità. */
|
|
23
|
+
level: UiLogLevel;
|
|
24
|
+
/** Azione/etichetta descrittiva (es. 'form_submit', 'step_change'). */
|
|
25
|
+
action: string;
|
|
26
|
+
/** Payload dati (schema id, step, valori, errori). */
|
|
27
|
+
payload?: Record<string, unknown>;
|
|
28
|
+
/** Timestamp creazione (ms). */
|
|
29
|
+
timestamp: number;
|
|
30
|
+
/** Durata operazione (ms), opzionale. */
|
|
31
|
+
duration?: number;
|
|
32
|
+
/** URL corrente al momento del log. */
|
|
33
|
+
url: string;
|
|
34
|
+
/** User ID se disponibile. */
|
|
35
|
+
userId?: string;
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Filtri per query log.
|
|
39
|
+
*/
|
|
40
|
+
export interface UiLogFilter {
|
|
41
|
+
category?: UiLogCategory;
|
|
42
|
+
level?: UiLogLevel;
|
|
43
|
+
action?: string;
|
|
44
|
+
from?: number;
|
|
45
|
+
to?: number;
|
|
46
|
+
limit?: number;
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* Riepilogo log per UI admin/debug.
|
|
50
|
+
*/
|
|
51
|
+
export interface UiLogSummary {
|
|
52
|
+
total: number;
|
|
53
|
+
byCategory: Record<UiLogCategory, number>;
|
|
54
|
+
byLevel: Record<UiLogLevel, number>;
|
|
55
|
+
oldest: number;
|
|
56
|
+
newest: number;
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* Configurazione runtime del logger.
|
|
60
|
+
*/
|
|
61
|
+
export interface UiLoggerConfig {
|
|
62
|
+
/** Abilita/disabilita logging. Default: true in dev, false in production. */
|
|
63
|
+
enabled: boolean;
|
|
64
|
+
/** Livello minimo da loggare. */
|
|
65
|
+
minLevel: UiLogLevel;
|
|
66
|
+
/** Durata retention log in ms. Default: 7 giorni. */
|
|
67
|
+
ttlMs: number;
|
|
68
|
+
/** Max entry totali per cleanup automatico. */
|
|
69
|
+
maxEntries: number;
|
|
70
|
+
}
|
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
* - Root singleton (`providedIn: 'root'`)
|
|
8
8
|
* - On first inject: generates fingerprint, creates session, subscribes to Router events
|
|
9
9
|
* - Centralised in-memory buffer with periodic/threshold-based flush to IndexedDB
|
|
10
|
-
* - `window:beforeunload` triggers a final
|
|
10
|
+
* - `window:beforeunload` triggers a final best-effort async flush
|
|
11
11
|
*/
|
|
12
12
|
import { Injectable, inject, PLATFORM_ID, NgZone, } from '@angular/core';
|
|
13
13
|
import { isPlatformBrowser } from '@angular/common';
|
|
@@ -97,8 +97,11 @@ export class UiBlackboxService {
|
|
|
97
97
|
// Setup periodic flush (outside Angular zone to avoid triggering CD)
|
|
98
98
|
this.zone.runOutsideAngular(() => {
|
|
99
99
|
this.flushTimerId = setInterval(() => this.flush(), this.config.flushIntervalMs);
|
|
100
|
-
//
|
|
101
|
-
this.beforeUnloadHandler = () =>
|
|
100
|
+
// Best-effort flush on tab close (IndexedDB is async, no sync fallback)
|
|
101
|
+
this.beforeUnloadHandler = () => {
|
|
102
|
+
this.session.endedAt = Date.now();
|
|
103
|
+
void this.flush();
|
|
104
|
+
};
|
|
102
105
|
window.addEventListener('beforeunload', this.beforeUnloadHandler);
|
|
103
106
|
});
|
|
104
107
|
this.initialised = true;
|
|
@@ -284,58 +287,6 @@ export class UiBlackboxService {
|
|
|
284
287
|
console.warn('[UiBlackbox] Storage flush failed:', e);
|
|
285
288
|
}
|
|
286
289
|
}
|
|
287
|
-
/**
|
|
288
|
-
* Synchronous flush for `beforeunload`.
|
|
289
|
-
* Falls back to a best-effort approach since IndexedDB is async.
|
|
290
|
-
* @internal
|
|
291
|
-
*/
|
|
292
|
-
flushSync() {
|
|
293
|
-
if (!this.initialised)
|
|
294
|
-
return;
|
|
295
|
-
this.session.endedAt = Date.now();
|
|
296
|
-
// Use sendBeacon-like approach: serialise session to localStorage
|
|
297
|
-
// as a fallback that can be recovered on next load.
|
|
298
|
-
// The proper IndexedDB flush is async and may not complete on unload.
|
|
299
|
-
try {
|
|
300
|
-
const key = `__ui_bb_pending_${this.session.sessionId}`;
|
|
301
|
-
const data = JSON.stringify(this.session);
|
|
302
|
-
localStorage.setItem(key, data);
|
|
303
|
-
}
|
|
304
|
-
catch {
|
|
305
|
-
// Best effort — quota may be exceeded
|
|
306
|
-
}
|
|
307
|
-
}
|
|
308
|
-
// ─── Pending session recovery ──────────────────────────────────────
|
|
309
|
-
/**
|
|
310
|
-
* Recover sessions that were saved to localStorage during `beforeunload`
|
|
311
|
-
* and persist them to IndexedDB. Called on next service init.
|
|
312
|
-
* @internal
|
|
313
|
-
*/
|
|
314
|
-
async recoverPendingSessions() {
|
|
315
|
-
if (!isPlatformBrowser(this.platformId))
|
|
316
|
-
return;
|
|
317
|
-
const prefix = '__ui_bb_pending_';
|
|
318
|
-
const keys = [];
|
|
319
|
-
for (let i = 0; i < localStorage.length; i++) {
|
|
320
|
-
const key = localStorage.key(i);
|
|
321
|
-
if (key?.startsWith(prefix)) {
|
|
322
|
-
keys.push(key);
|
|
323
|
-
}
|
|
324
|
-
}
|
|
325
|
-
for (const key of keys) {
|
|
326
|
-
try {
|
|
327
|
-
const data = localStorage.getItem(key);
|
|
328
|
-
if (data) {
|
|
329
|
-
const session = JSON.parse(data);
|
|
330
|
-
await this.storage.saveSession(session);
|
|
331
|
-
}
|
|
332
|
-
localStorage.removeItem(key);
|
|
333
|
-
}
|
|
334
|
-
catch {
|
|
335
|
-
localStorage.removeItem(key);
|
|
336
|
-
}
|
|
337
|
-
}
|
|
338
|
-
}
|
|
339
290
|
// ─── Utilities ─────────────────────────────────────────────────────
|
|
340
291
|
/** Generate a UUID v4. */
|
|
341
292
|
generateUuid() {
|
|
@@ -360,7 +311,7 @@ export class UiBlackboxService {
|
|
|
360
311
|
window.removeEventListener('beforeunload', this.beforeUnloadHandler);
|
|
361
312
|
}
|
|
362
313
|
// Final flush
|
|
363
|
-
this.
|
|
314
|
+
void this.flush();
|
|
364
315
|
}
|
|
365
316
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiBlackboxService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
366
317
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiBlackboxService, providedIn: 'root' }); }
|
|
@@ -369,4 +320,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
369
320
|
type: Injectable,
|
|
370
321
|
args: [{ providedIn: 'root' }]
|
|
371
322
|
}], ctorParameters: () => [] });
|
|
372
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"blackbox.service.js","sourceRoot":"","sources":["../../../../../../../packages/ng-ui-system/src/lib/components/blackbox/blackbox.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * @module ng-ui-system/blackbox\r\n * Core BlackBox observability service — the main entry point for session\r\n * recording, event buffering, and export.\r\n *\r\n * Architecture:\r\n * - Root singleton (`providedIn: 'root'`)\r\n * - On first inject: generates fingerprint, creates session, subscribes to Router events\r\n * - Centralised in-memory buffer with periodic/threshold-based flush to IndexedDB\r\n * - `window:beforeunload` triggers a final synchronous flush\r\n */\r\n\r\nimport {\r\n  Injectable,\r\n  inject,\r\n  PLATFORM_ID,\r\n  OnDestroy,\r\n  NgZone,\r\n} from '@angular/core';\r\nimport { isPlatformBrowser } from '@angular/common';\r\nimport { Router, NavigationEnd } from '@angular/router';\r\nimport { Subject } from 'rxjs';\r\nimport { filter, takeUntil } from 'rxjs/operators';\r\n\r\nimport {\r\n  UiBlackboxConfig,\r\n  UiBlackboxFormTrackingConfig,\r\n  UiBlackboxSession,\r\n  UiBlackboxStep,\r\n  UiBlackboxAction,\r\n  UiBlackboxFormEvent,\r\n  UiBlackboxHttpCall,\r\n  UiBlackboxBufferEvent,\r\n  UI_BLACKBOX_DEFAULTS,\r\n} from './blackbox.types';\r\nimport { UiBlackboxFingerprintService } from './blackbox-fingerprint.service';\r\nimport { UiBlackboxStorageService } from './blackbox-storage.service';\r\n\r\n/**\r\n * Central BlackBox observability service.\r\n *\r\n * Records navigation, UI interactions, form events, and HTTP calls\r\n * into compressed sessions stored in IndexedDB.\r\n *\r\n * @usageNotes\r\n * The service activates automatically when injected. Provide it at root level\r\n * and it will start recording immediately.\r\n *\r\n * ```typescript\r\n * // app.config.ts — just importing the service is enough\r\n * import { UiBlackboxService } from '@gnggln/ng-ui-system';\r\n *\r\n * export const appConfig: ApplicationConfig = {\r\n *   providers: [\r\n *     // The service is providedIn: 'root', so it's auto-provided.\r\n *     // To force eager initialisation:\r\n *     { provide: APP_INITIALIZER, useFactory: () => () => inject(UiBlackboxService), multi: true },\r\n *   ],\r\n * };\r\n * ```\r\n */\r\n@Injectable({ providedIn: 'root' })\r\nexport class UiBlackboxService implements OnDestroy {\r\n  private readonly platformId = inject(PLATFORM_ID);\r\n  private readonly router = inject(Router);\r\n  private readonly zone = inject(NgZone);\r\n  private readonly fingerprint = inject(UiBlackboxFingerprintService);\r\n  private readonly storage = inject(UiBlackboxStorageService);\r\n\r\n  private readonly destroy$ = new Subject<void>();\r\n  private config: UiBlackboxConfig = { ...UI_BLACKBOX_DEFAULTS };\r\n\r\n  // ─── Session state ─────────────────────────────────────────────────\r\n  private session!: UiBlackboxSession;\r\n  private currentStep: UiBlackboxStep | null = null;\r\n  private initialised = false;\r\n\r\n  // ─── Centralised event buffer ──────────────────────────────────────\r\n  private buffer: UiBlackboxBufferEvent[] = [];\r\n  private flushTimerId: ReturnType<typeof setInterval> | null = null;\r\n  private beforeUnloadHandler: (() => void) | null = null;\r\n\r\n  constructor() {\r\n    if (isPlatformBrowser(this.platformId)) {\r\n      this.init();\r\n    }\r\n  }\r\n\r\n  // ─── Initialisation ────────────────────────────────────────────────\r\n\r\n  private async init(): Promise<void> {\r\n    // Generate session ID\r\n    const sessionId = this.generateUuid();\r\n    const fp = await this.fingerprint.getFingerprint();\r\n\r\n    this.session = {\r\n      sessionId,\r\n      fingerprint: fp,\r\n      startedAt: Date.now(),\r\n      userAgent: navigator.userAgent,\r\n      steps: [],\r\n      formEvents: [],\r\n      calls: [],\r\n    };\r\n\r\n    // Create initial step from current URL\r\n    this.currentStep = {\r\n      timestamp: Date.now(),\r\n      route: this.router.url,\r\n      actions: [],\r\n    };\r\n    this.session.steps.push(this.currentStep);\r\n\r\n    // Subscribe to router navigation\r\n    this.router.events\r\n      .pipe(\r\n        filter((e): e is NavigationEnd => e instanceof NavigationEnd),\r\n        takeUntil(this.destroy$),\r\n      )\r\n      .subscribe((event) => {\r\n        const previousRoute = this.currentStep?.route;\r\n        this.currentStep = {\r\n          timestamp: Date.now(),\r\n          route: event.urlAfterRedirects,\r\n          previousRoute,\r\n          actions: [],\r\n        };\r\n        this.pushBufferEvent({ kind: 'step', payload: this.currentStep });\r\n      });\r\n\r\n    // Setup periodic flush (outside Angular zone to avoid triggering CD)\r\n    this.zone.runOutsideAngular(() => {\r\n      this.flushTimerId = setInterval(\r\n        () => this.flush(),\r\n        this.config.flushIntervalMs,\r\n      );\r\n\r\n      // Final flush on tab close\r\n      this.beforeUnloadHandler = () => this.flushSync();\r\n      window.addEventListener('beforeunload', this.beforeUnloadHandler);\r\n    });\r\n\r\n    this.initialised = true;\r\n  }\r\n\r\n  // ─── Public API: Configuration ─────────────────────────────────────\r\n\r\n  /**\r\n   * Update the BlackBox configuration.\r\n   * Can be called at any time; changes take effect immediately.\r\n   */\r\n  configure(config: Partial<Omit<UiBlackboxConfig, 'formTracking'>> & { formTracking?: Partial<UiBlackboxFormTrackingConfig> }): void {\r\n    this.config = {\r\n      ...this.config,\r\n      ...config,\r\n      formTracking: {\r\n        ...this.config.formTracking,\r\n        ...(config.formTracking ?? {}),\r\n      },\r\n    };\r\n    this.storage.setMaxStorageMb(this.config.maxStorageMb);\r\n\r\n    // Restart flush timer with new interval\r\n    if (this.flushTimerId !== null) {\r\n      clearInterval(this.flushTimerId);\r\n      this.zone.runOutsideAngular(() => {\r\n        this.flushTimerId = setInterval(\r\n          () => this.flush(),\r\n          this.config.flushIntervalMs,\r\n        );\r\n      });\r\n    }\r\n  }\r\n\r\n  /** Returns the current configuration (readonly snapshot). */\r\n  getConfig(): Readonly<UiBlackboxConfig> {\r\n    return { ...this.config };\r\n  }\r\n\r\n  // ─── Public API: Tracking ──────────────────────────────────────────\r\n\r\n  /**\r\n   * Manually track a UI action (click).\r\n   *\r\n   * Called automatically by `UiButtonAreaComponent` and the `[uiTrack]` directive.\r\n   * Can also be called programmatically for custom interactions.\r\n   *\r\n   * @param trackId - Identifier for the action (e.g. button id, link name).\r\n   * @param meta - Optional metadata: `tag`, `text`, `id` of the element.\r\n   */\r\n  trackAction(\r\n    trackId: string,\r\n    meta?: { tag?: string; text?: string; id?: string },\r\n  ): void {\r\n    if (!this.initialised) return;\r\n\r\n    const action: UiBlackboxAction = {\r\n      timestamp: Date.now(),\r\n      type: 'click',\r\n      target: {\r\n        tag: meta?.tag ?? 'unknown',\r\n        id: meta?.id,\r\n        trackId,\r\n        text: meta?.text?.substring(0, 50),\r\n      },\r\n    };\r\n\r\n    // Add to current step immediately (for in-memory session access)\r\n    this.currentStep?.actions.push(action);\r\n\r\n    // Also push to buffer for persistence\r\n    this.pushBufferEvent({\r\n      kind: 'action',\r\n      payload: { ...action, route: this.currentStep?.route ?? this.router.url },\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Track a form interaction event.\r\n   *\r\n   * Called automatically by `UiFormBuilderComponent` when the BlackBox\r\n   * service is available.\r\n   */\r\n  trackFormEvent(event: UiBlackboxFormEvent): void {\r\n    if (!this.initialised) return;\r\n\r\n    this.session.formEvents.push(event);\r\n    this.pushBufferEvent({ kind: 'formEvent', payload: event });\r\n  }\r\n\r\n  /**\r\n   * Track an intercepted HTTP call.\r\n   * Called by `UiBlackboxInterceptor`.\r\n   * @internal\r\n   */\r\n  trackHttpCall(call: UiBlackboxHttpCall): void {\r\n    if (!this.initialised) return;\r\n\r\n    this.session.calls.push(call);\r\n    this.pushBufferEvent({ kind: 'httpCall', payload: call });\r\n  }\r\n\r\n  // ─── Public API: Session access ────────────────────────────────────\r\n\r\n  /**\r\n   * Returns the current in-memory session (live, not yet flushed).\r\n   * Useful for debugging or real-time inspection.\r\n   */\r\n  getCurrentSession(): UiBlackboxSession {\r\n    return this.session;\r\n  }\r\n\r\n  /**\r\n   * Retrieve all sessions stored in IndexedDB (decompressed).\r\n   */\r\n  async getSessionHistory(): Promise<UiBlackboxSession[]> {\r\n    return this.storage.getAllSessions();\r\n  }\r\n\r\n  /**\r\n   * Retrieve sessions for the current device fingerprint.\r\n   */\r\n  async getSessionsForDevice(): Promise<UiBlackboxSession[]> {\r\n    const fp = await this.fingerprint.getFingerprint();\r\n    return this.storage.getSessionsByFingerprint(fp);\r\n  }\r\n\r\n  // ─── Public API: Export ────────────────────────────────────────────\r\n\r\n  /**\r\n   * Export all sessions as a downloadable JSONL Blob.\r\n   *\r\n   * @returns A Blob containing JSONL data (one session per line).\r\n   *\r\n   * @example\r\n   * ```typescript\r\n   * const blob = await blackbox.exportSessions();\r\n   * const url = URL.createObjectURL(blob);\r\n   * const a = document.createElement('a');\r\n   * a.href = url;\r\n   * a.download = 'blackbox-sessions.jsonl';\r\n   * a.click();\r\n   * ```\r\n   */\r\n  async exportSessions(): Promise<Blob> {\r\n    // Flush current session first\r\n    await this.flush();\r\n    return this.storage.exportDump();\r\n  }\r\n\r\n  // ─── Public API: Lifecycle ─────────────────────────────────────────\r\n\r\n  /**\r\n   * Explicitly end the current session.\r\n   * Flushes all buffered events and marks the session as ended.\r\n   */\r\n  async endSession(): Promise<void> {\r\n    if (!this.initialised) return;\r\n    this.session.endedAt = Date.now();\r\n    await this.flush();\r\n  }\r\n\r\n  /**\r\n   * Clear all stored sessions from IndexedDB.\r\n   */\r\n  async clearHistory(): Promise<void> {\r\n    return this.storage.clearAll();\r\n  }\r\n\r\n  // ─── Buffer management ─────────────────────────────────────────────\r\n\r\n  /**\r\n   * Push an event into the centralised buffer.\r\n   * Triggers an immediate flush if the buffer exceeds the threshold.\r\n   * @internal\r\n   */\r\n  private pushBufferEvent(event: UiBlackboxBufferEvent): void {\r\n    this.buffer.push(event);\r\n    if (this.buffer.length >= this.config.flushThreshold) {\r\n      this.flush();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Flush buffered events to IndexedDB asynchronously.\r\n   * @internal\r\n   */\r\n  private async flush(): Promise<void> {\r\n    if (!this.initialised || this.buffer.length === 0) return;\r\n\r\n    // Drain buffer\r\n    const events = this.buffer.splice(0);\r\n\r\n    // Apply step events to the session\r\n    for (const event of events) {\r\n      if (event.kind === 'step') {\r\n        // Steps are already pushed in real-time; ensure they're in the session\r\n        if (!this.session.steps.includes(event.payload)) {\r\n          this.session.steps.push(event.payload);\r\n        }\r\n      }\r\n      // Actions, formEvents, httpCalls are already pushed in real-time\r\n      // to the in-memory session by trackAction/trackFormEvent/trackHttpCall\r\n    }\r\n\r\n    // Save the full session snapshot\r\n    try {\r\n      await this.storage.saveSession(this.session);\r\n    } catch (e) {\r\n      // storage error — re-enqueue events for next flush\r\n      console.warn('[UiBlackbox] Storage flush failed:', e);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Synchronous flush for `beforeunload`.\r\n   * Falls back to a best-effort approach since IndexedDB is async.\r\n   * @internal\r\n   */\r\n  private flushSync(): void {\r\n    if (!this.initialised) return;\r\n\r\n    this.session.endedAt = Date.now();\r\n\r\n    // Use sendBeacon-like approach: serialise session to localStorage\r\n    // as a fallback that can be recovered on next load.\r\n    // The proper IndexedDB flush is async and may not complete on unload.\r\n    try {\r\n      const key = `__ui_bb_pending_${this.session.sessionId}`;\r\n      const data = JSON.stringify(this.session);\r\n      localStorage.setItem(key, data);\r\n    } catch {\r\n      // Best effort — quota may be exceeded\r\n    }\r\n  }\r\n\r\n  // ─── Pending session recovery ──────────────────────────────────────\r\n\r\n  /**\r\n   * Recover sessions that were saved to localStorage during `beforeunload`\r\n   * and persist them to IndexedDB. Called on next service init.\r\n   * @internal\r\n   */\r\n  private async recoverPendingSessions(): Promise<void> {\r\n    if (!isPlatformBrowser(this.platformId)) return;\r\n\r\n    const prefix = '__ui_bb_pending_';\r\n    const keys: string[] = [];\r\n\r\n    for (let i = 0; i < localStorage.length; i++) {\r\n      const key = localStorage.key(i);\r\n      if (key?.startsWith(prefix)) {\r\n        keys.push(key);\r\n      }\r\n    }\r\n\r\n    for (const key of keys) {\r\n      try {\r\n        const data = localStorage.getItem(key);\r\n        if (data) {\r\n          const session: UiBlackboxSession = JSON.parse(data);\r\n          await this.storage.saveSession(session);\r\n        }\r\n        localStorage.removeItem(key);\r\n      } catch {\r\n        localStorage.removeItem(key);\r\n      }\r\n    }\r\n  }\r\n\r\n  // ─── Utilities ─────────────────────────────────────────────────────\r\n\r\n  /** Generate a UUID v4. */\r\n  private generateUuid(): string {\r\n    if (typeof crypto !== 'undefined' && crypto.randomUUID) {\r\n      return crypto.randomUUID();\r\n    }\r\n    // Fallback for older browsers\r\n    return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {\r\n      const r = (Math.random() * 16) | 0;\r\n      const v = c === 'x' ? r : (r & 0x3) | 0x8;\r\n      return v.toString(16);\r\n    });\r\n  }\r\n\r\n  // ─── Cleanup ───────────────────────────────────────────────────────\r\n\r\n  ngOnDestroy(): void {\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n\r\n    if (this.flushTimerId !== null) {\r\n      clearInterval(this.flushTimerId);\r\n    }\r\n\r\n    if (this.beforeUnloadHandler) {\r\n      window.removeEventListener('beforeunload', this.beforeUnloadHandler);\r\n    }\r\n\r\n    // Final flush\r\n    this.flushSync();\r\n  }\r\n}\r\n"]}
|
|
323
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"blackbox.service.js","sourceRoot":"","sources":["../../../../../../../packages/ng-ui-system/src/lib/components/blackbox/blackbox.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * @module ng-ui-system/blackbox\r\n * Core BlackBox observability service — the main entry point for session\r\n * recording, event buffering, and export.\r\n *\r\n * Architecture:\r\n * - Root singleton (`providedIn: 'root'`)\r\n * - On first inject: generates fingerprint, creates session, subscribes to Router events\r\n * - Centralised in-memory buffer with periodic/threshold-based flush to IndexedDB\r\n * - `window:beforeunload` triggers a final best-effort async flush\r\n */\r\n\r\nimport {\r\n  Injectable,\r\n  inject,\r\n  PLATFORM_ID,\r\n  OnDestroy,\r\n  NgZone,\r\n} from '@angular/core';\r\nimport { isPlatformBrowser } from '@angular/common';\r\nimport { Router, NavigationEnd } from '@angular/router';\r\nimport { Subject } from 'rxjs';\r\nimport { filter, takeUntil } from 'rxjs/operators';\r\n\r\nimport {\r\n  UiBlackboxConfig,\r\n  UiBlackboxFormTrackingConfig,\r\n  UiBlackboxSession,\r\n  UiBlackboxStep,\r\n  UiBlackboxAction,\r\n  UiBlackboxFormEvent,\r\n  UiBlackboxHttpCall,\r\n  UiBlackboxBufferEvent,\r\n  UI_BLACKBOX_DEFAULTS,\r\n} from './blackbox.types';\r\nimport { UiBlackboxFingerprintService } from './blackbox-fingerprint.service';\r\nimport { UiBlackboxStorageService } from './blackbox-storage.service';\r\n\r\n/**\r\n * Central BlackBox observability service.\r\n *\r\n * Records navigation, UI interactions, form events, and HTTP calls\r\n * into compressed sessions stored in IndexedDB.\r\n *\r\n * @usageNotes\r\n * The service activates automatically when injected. Provide it at root level\r\n * and it will start recording immediately.\r\n *\r\n * ```typescript\r\n * // app.config.ts — just importing the service is enough\r\n * import { UiBlackboxService } from '@gnggln/ng-ui-system';\r\n *\r\n * export const appConfig: ApplicationConfig = {\r\n *   providers: [\r\n *     // The service is providedIn: 'root', so it's auto-provided.\r\n *     // To force eager initialisation:\r\n *     { provide: APP_INITIALIZER, useFactory: () => () => inject(UiBlackboxService), multi: true },\r\n *   ],\r\n * };\r\n * ```\r\n */\r\n@Injectable({ providedIn: 'root' })\r\nexport class UiBlackboxService implements OnDestroy {\r\n  private readonly platformId = inject(PLATFORM_ID);\r\n  private readonly router = inject(Router);\r\n  private readonly zone = inject(NgZone);\r\n  private readonly fingerprint = inject(UiBlackboxFingerprintService);\r\n  private readonly storage = inject(UiBlackboxStorageService);\r\n\r\n  private readonly destroy$ = new Subject<void>();\r\n  private config: UiBlackboxConfig = { ...UI_BLACKBOX_DEFAULTS };\r\n\r\n  // ─── Session state ─────────────────────────────────────────────────\r\n  private session!: UiBlackboxSession;\r\n  private currentStep: UiBlackboxStep | null = null;\r\n  private initialised = false;\r\n\r\n  // ─── Centralised event buffer ──────────────────────────────────────\r\n  private buffer: UiBlackboxBufferEvent[] = [];\r\n  private flushTimerId: ReturnType<typeof setInterval> | null = null;\r\n  private beforeUnloadHandler: (() => void) | null = null;\r\n\r\n  constructor() {\r\n    if (isPlatformBrowser(this.platformId)) {\r\n      this.init();\r\n    }\r\n  }\r\n\r\n  // ─── Initialisation ────────────────────────────────────────────────\r\n\r\n  private async init(): Promise<void> {\r\n    // Generate session ID\r\n    const sessionId = this.generateUuid();\r\n    const fp = await this.fingerprint.getFingerprint();\r\n\r\n    this.session = {\r\n      sessionId,\r\n      fingerprint: fp,\r\n      startedAt: Date.now(),\r\n      userAgent: navigator.userAgent,\r\n      steps: [],\r\n      formEvents: [],\r\n      calls: [],\r\n    };\r\n\r\n    // Create initial step from current URL\r\n    this.currentStep = {\r\n      timestamp: Date.now(),\r\n      route: this.router.url,\r\n      actions: [],\r\n    };\r\n    this.session.steps.push(this.currentStep);\r\n\r\n    // Subscribe to router navigation\r\n    this.router.events\r\n      .pipe(\r\n        filter((e): e is NavigationEnd => e instanceof NavigationEnd),\r\n        takeUntil(this.destroy$),\r\n      )\r\n      .subscribe((event) => {\r\n        const previousRoute = this.currentStep?.route;\r\n        this.currentStep = {\r\n          timestamp: Date.now(),\r\n          route: event.urlAfterRedirects,\r\n          previousRoute,\r\n          actions: [],\r\n        };\r\n        this.pushBufferEvent({ kind: 'step', payload: this.currentStep });\r\n      });\r\n\r\n    // Setup periodic flush (outside Angular zone to avoid triggering CD)\r\n    this.zone.runOutsideAngular(() => {\r\n      this.flushTimerId = setInterval(\r\n        () => this.flush(),\r\n        this.config.flushIntervalMs,\r\n      );\r\n\r\n      // Best-effort flush on tab close (IndexedDB is async, no sync fallback)\r\n      this.beforeUnloadHandler = () => {\r\n        this.session.endedAt = Date.now();\r\n        void this.flush();\r\n      };\r\n      window.addEventListener('beforeunload', this.beforeUnloadHandler);\r\n    });\r\n\r\n    this.initialised = true;\r\n  }\r\n\r\n  // ─── Public API: Configuration ─────────────────────────────────────\r\n\r\n  /**\r\n   * Update the BlackBox configuration.\r\n   * Can be called at any time; changes take effect immediately.\r\n   */\r\n  configure(config: Partial<Omit<UiBlackboxConfig, 'formTracking'>> & { formTracking?: Partial<UiBlackboxFormTrackingConfig> }): void {\r\n    this.config = {\r\n      ...this.config,\r\n      ...config,\r\n      formTracking: {\r\n        ...this.config.formTracking,\r\n        ...(config.formTracking ?? {}),\r\n      },\r\n    };\r\n    this.storage.setMaxStorageMb(this.config.maxStorageMb);\r\n\r\n    // Restart flush timer with new interval\r\n    if (this.flushTimerId !== null) {\r\n      clearInterval(this.flushTimerId);\r\n      this.zone.runOutsideAngular(() => {\r\n        this.flushTimerId = setInterval(\r\n          () => this.flush(),\r\n          this.config.flushIntervalMs,\r\n        );\r\n      });\r\n    }\r\n  }\r\n\r\n  /** Returns the current configuration (readonly snapshot). */\r\n  getConfig(): Readonly<UiBlackboxConfig> {\r\n    return { ...this.config };\r\n  }\r\n\r\n  // ─── Public API: Tracking ──────────────────────────────────────────\r\n\r\n  /**\r\n   * Manually track a UI action (click).\r\n   *\r\n   * Called automatically by `UiButtonAreaComponent` and the `[uiTrack]` directive.\r\n   * Can also be called programmatically for custom interactions.\r\n   *\r\n   * @param trackId - Identifier for the action (e.g. button id, link name).\r\n   * @param meta - Optional metadata: `tag`, `text`, `id` of the element.\r\n   */\r\n  trackAction(\r\n    trackId: string,\r\n    meta?: { tag?: string; text?: string; id?: string },\r\n  ): void {\r\n    if (!this.initialised) return;\r\n\r\n    const action: UiBlackboxAction = {\r\n      timestamp: Date.now(),\r\n      type: 'click',\r\n      target: {\r\n        tag: meta?.tag ?? 'unknown',\r\n        id: meta?.id,\r\n        trackId,\r\n        text: meta?.text?.substring(0, 50),\r\n      },\r\n    };\r\n\r\n    // Add to current step immediately (for in-memory session access)\r\n    this.currentStep?.actions.push(action);\r\n\r\n    // Also push to buffer for persistence\r\n    this.pushBufferEvent({\r\n      kind: 'action',\r\n      payload: { ...action, route: this.currentStep?.route ?? this.router.url },\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Track a form interaction event.\r\n   *\r\n   * Called automatically by `UiFormBuilderComponent` when the BlackBox\r\n   * service is available.\r\n   */\r\n  trackFormEvent(event: UiBlackboxFormEvent): void {\r\n    if (!this.initialised) return;\r\n\r\n    this.session.formEvents.push(event);\r\n    this.pushBufferEvent({ kind: 'formEvent', payload: event });\r\n  }\r\n\r\n  /**\r\n   * Track an intercepted HTTP call.\r\n   * Called by `UiBlackboxInterceptor`.\r\n   * @internal\r\n   */\r\n  trackHttpCall(call: UiBlackboxHttpCall): void {\r\n    if (!this.initialised) return;\r\n\r\n    this.session.calls.push(call);\r\n    this.pushBufferEvent({ kind: 'httpCall', payload: call });\r\n  }\r\n\r\n  // ─── Public API: Session access ────────────────────────────────────\r\n\r\n  /**\r\n   * Returns the current in-memory session (live, not yet flushed).\r\n   * Useful for debugging or real-time inspection.\r\n   */\r\n  getCurrentSession(): UiBlackboxSession {\r\n    return this.session;\r\n  }\r\n\r\n  /**\r\n   * Retrieve all sessions stored in IndexedDB (decompressed).\r\n   */\r\n  async getSessionHistory(): Promise<UiBlackboxSession[]> {\r\n    return this.storage.getAllSessions();\r\n  }\r\n\r\n  /**\r\n   * Retrieve sessions for the current device fingerprint.\r\n   */\r\n  async getSessionsForDevice(): Promise<UiBlackboxSession[]> {\r\n    const fp = await this.fingerprint.getFingerprint();\r\n    return this.storage.getSessionsByFingerprint(fp);\r\n  }\r\n\r\n  // ─── Public API: Export ────────────────────────────────────────────\r\n\r\n  /**\r\n   * Export all sessions as a downloadable JSONL Blob.\r\n   *\r\n   * @returns A Blob containing JSONL data (one session per line).\r\n   *\r\n   * @example\r\n   * ```typescript\r\n   * const blob = await blackbox.exportSessions();\r\n   * const url = URL.createObjectURL(blob);\r\n   * const a = document.createElement('a');\r\n   * a.href = url;\r\n   * a.download = 'blackbox-sessions.jsonl';\r\n   * a.click();\r\n   * ```\r\n   */\r\n  async exportSessions(): Promise<Blob> {\r\n    // Flush current session first\r\n    await this.flush();\r\n    return this.storage.exportDump();\r\n  }\r\n\r\n  // ─── Public API: Lifecycle ─────────────────────────────────────────\r\n\r\n  /**\r\n   * Explicitly end the current session.\r\n   * Flushes all buffered events and marks the session as ended.\r\n   */\r\n  async endSession(): Promise<void> {\r\n    if (!this.initialised) return;\r\n    this.session.endedAt = Date.now();\r\n    await this.flush();\r\n  }\r\n\r\n  /**\r\n   * Clear all stored sessions from IndexedDB.\r\n   */\r\n  async clearHistory(): Promise<void> {\r\n    return this.storage.clearAll();\r\n  }\r\n\r\n  // ─── Buffer management ─────────────────────────────────────────────\r\n\r\n  /**\r\n   * Push an event into the centralised buffer.\r\n   * Triggers an immediate flush if the buffer exceeds the threshold.\r\n   * @internal\r\n   */\r\n  private pushBufferEvent(event: UiBlackboxBufferEvent): void {\r\n    this.buffer.push(event);\r\n    if (this.buffer.length >= this.config.flushThreshold) {\r\n      this.flush();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Flush buffered events to IndexedDB asynchronously.\r\n   * @internal\r\n   */\r\n  private async flush(): Promise<void> {\r\n    if (!this.initialised || this.buffer.length === 0) return;\r\n\r\n    // Drain buffer\r\n    const events = this.buffer.splice(0);\r\n\r\n    // Apply step events to the session\r\n    for (const event of events) {\r\n      if (event.kind === 'step') {\r\n        // Steps are already pushed in real-time; ensure they're in the session\r\n        if (!this.session.steps.includes(event.payload)) {\r\n          this.session.steps.push(event.payload);\r\n        }\r\n      }\r\n      // Actions, formEvents, httpCalls are already pushed in real-time\r\n      // to the in-memory session by trackAction/trackFormEvent/trackHttpCall\r\n    }\r\n\r\n    // Save the full session snapshot\r\n    try {\r\n      await this.storage.saveSession(this.session);\r\n    } catch (e) {\r\n      // storage error — re-enqueue events for next flush\r\n      console.warn('[UiBlackbox] Storage flush failed:', e);\r\n    }\r\n  }\r\n\r\n  // ─── Utilities ─────────────────────────────────────────────────────\r\n\r\n  /** Generate a UUID v4. */\r\n  private generateUuid(): string {\r\n    if (typeof crypto !== 'undefined' && crypto.randomUUID) {\r\n      return crypto.randomUUID();\r\n    }\r\n    // Fallback for older browsers\r\n    return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {\r\n      const r = (Math.random() * 16) | 0;\r\n      const v = c === 'x' ? r : (r & 0x3) | 0x8;\r\n      return v.toString(16);\r\n    });\r\n  }\r\n\r\n  // ─── Cleanup ───────────────────────────────────────────────────────\r\n\r\n  ngOnDestroy(): void {\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n\r\n    if (this.flushTimerId !== null) {\r\n      clearInterval(this.flushTimerId);\r\n    }\r\n\r\n    if (this.beforeUnloadHandler) {\r\n      window.removeEventListener('beforeunload', this.beforeUnloadHandler);\r\n    }\r\n\r\n    // Final flush\r\n    void this.flush();\r\n  }\r\n}\r\n"]}
|