@o3r/rules-engine 11.7.0-prerelease.41 → 11.7.0-prerelease.43

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.
@@ -37,14 +37,14 @@ export class RulesetHistoryPresComponent {
37
37
  this.cd.detectChanges();
38
38
  }
39
39
  /** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RulesetHistoryPresComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
40
- /** @nocollapse */ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RulesetHistoryPresComponent, selector: "o3r-ruleset-history-pres", inputs: { rulesetExecutions: "rulesetExecutions", executionDurationFormat: "executionDurationFormat" }, ngImport: i0, template: "<section>\n <h4 class=\"mb-4\">Ruleset Execution History</h4>\n <ng-template #noRulesEngine>\n <div class=\"alert alert-danger m-2\" role=\"alert\">\n The Rules Engine is not configured on this page.\n </div>\n </ng-template>\n <ul *ngIf=\"rulesetExecutions; else noRulesEngine\" class=\"rulesets\">\n <li *ngFor=\"let execution of rulesetExecutions\" class=\"ruleset\">\n <!-- eslint-disable-next-line @angular-eslint/template/click-events-have-key-events, @angular-eslint/template/interactive-supports-focus -- need to refactor the div to accordion from DF #1518 -->\n <div class=\"ruleset-panel-title ruleset-expansion-action\"\n [class.error]=\"execution.type === 'RulesetExecutionError'\"\n (click)=\"toggleExpansion(execution.executionId, 'ruleset')\">\n <div><span [title]=\"'This ruleset has been evaluated ' + execution.executionCounter + ' time(s)'\">{{execution.executionCounter}}</span> - {{execution.rulesetName | titlecase }}\n <div class=\"ruleset-panel-subtitle\" *ngIf=\"execution.rulesetInformation?.linkedComponents?.or || execution.rulesetInformation?.linkedComponent\">\n <ng-container *ngFor=\"let lc of (execution.rulesetInformation?.linkedComponents?.or || [execution.rulesetInformation.linkedComponent]); last as isLast\">\n <div>{{lc.name}} {{lc.library}} <span *ngIf=\"!isLast\"> OR </span></div>\n </ng-container>\n </div>\n <div class=\"ruleset-panel-subtitle\" *ngIf=\"execution.rulesetInformation?.validityRange as validityRange\">\n Date range: {{validityRange.from}} - {{validityRange.to}}\n </div>\n </div>\n <div class=\"ruleset-panel-title-aside\">\n <span class=\"error capsule\" *ngIf=\"execution.status === 'Error'\">Error</span>\n <span class=\"success capsule\" *ngIf=\"execution.status === 'Active'\">Applied</span>\n <span class=\"inactive capsule\" *ngIf=\"execution.status === 'Deactivated'\">Deactivated</span>\n <span class=\"warn capsule\" *ngIf=\"execution.status === 'NoEffect'\">No effect</span>\n <span class=\"time capsule\">\n <span>{{execution.timestamp | date: 'HH:mm:ss SSS'}}</span>\n <span>({{execution.duration | number: executionDurationFormat}}ms)</span>\n </span>\n <button\n class=\"icon\"\n [class.icon-caret-down]=\"!expansionStatus[execution.executionId]?.ruleset\"\n [class.icon-caret-up]=\"expansionStatus[execution.executionId]?.ruleset\">\n </button>\n </div>\n </div>\n <div class=\"ruleset-panel-description\" *ngIf=\"expansionStatus[execution.executionId]?.ruleset\">\n <ng-container [ngTemplateOutlet]=\"rules\"\n [ngTemplateOutletContext]=\"{\n rules: execution.rulesetInformation.rules,\n expansionID: execution.executionId\n }\"></ng-container>\n <ng-container [ngTemplateOutlet]=\"inputs\"\n [ngTemplateOutletContext]=\"{\n inputs: execution.inputFacts\n }\"></ng-container>\n <ng-container *ngIf=\"execution.type === 'RulesetExecutionError'; else success\">\n <div class=\"ruleset-panel-category-title\">Rules:</div>\n <ul class=\"ruleset-panel-category-body rule-description\">\n <li *ngFor=\"let ruleEvaluation of execution.rulesEvaluations; let index=index;\">\n <ng-container>\n <div class=\"ruleset-panel-title\" [class.error]=\"ruleEvaluation.error\">\n <span>{{ruleEvaluation.rule.name | titlecase}} </span>\n <span class=\"capsule error\" *ngIf=\"ruleEvaluation.error\">Error</span>\n </div>\n <div>\n <ng-container *ngIf=\"ruleEvaluation.error\">\n <span class=\"ruleset-panel-category-title\">Error:</span>\n <pre class=\"ruleset-panel-category-body error\">{{ruleEvaluation.error | o3rJsonOrString}}</pre>\n </ng-container>\n <ng-container [ngTemplateOutlet]=\"inputs\"\n [ngTemplateOutletContext]=\"{\n inputs: execution.inputFacts,\n runtimeInputs: execution.rulesetInformation?.rules[index]?.inputRuntimeFacts\n }\"></ng-container>\n <o3r-rule-actions-pres *ngIf=\"!ruleEvaluation.error\"\n [temporaryFacts]=\"ruleEvaluation.temporaryFacts\"\n [runtimeOutputs]=\"execution.rulesetInformation?.rules[index]?.outputRuntimeFacts\"\n ></o3r-rule-actions-pres>\n </div>\n </ng-container>\n </li>\n </ul>\n </ng-container>\n <ng-template #success>\n <o3r-rule-actions-pres [actions]=\"execution.outputActions\"></o3r-rule-actions-pres>\n <div class=\"ruleset-panel-category-title\">Executed Rules</div>\n <ul class=\"rule-description ruleset-panel-category-body\">\n <li *ngFor=\"let ruleEvaluation of execution.rulesEvaluations; let index=index;\">\n <div class=\"ruleset-panel-title\">\n <span>{{ruleEvaluation.rule.name | titlecase}}</span>\n <span class=\"capsule inactive\" *ngIf=\"ruleEvaluation.cached\">Cached</span>\n <span class=\"capsule\">({{ruleEvaluation.duration | number: executionDurationFormat}}ms)</span>\n </div>\n <div>\n <ng-container [ngTemplateOutlet]=\"triggers\"\n [ngTemplateOutletContext]=\"{triggers: (ruleEvaluation.triggers[ruleEvaluation.rule.id])}\"></ng-container>\n <o3r-rule-actions-pres\n [actions]=\"ruleEvaluation.outputActions\"\n [temporaryFacts]=\"ruleEvaluation.temporaryFacts\"\n [runtimeOutputs]=\"execution.rulesetInformation?.rules[index]?.outputRuntimeFacts\">\n </o3r-rule-actions-pres>\n </div>\n </li>\n </ul>\n </ng-template>\n </div>\n </li>\n </ul>\n</section>\n\n<ng-template let-triggers=\"triggers\" #triggers>\n <div class=\"ruleset-panel-category-title\">Basefacts Triggers</div>\n <ul class=\"ruleset-panel-category-body triggers\">\n <ng-container *ngFor=\"let trigger of (triggers | keyvalue)\">\n <li *ngIf=\"trigger.value?.factName\">\n <o3r-rule-key-value-pres\n [key]=\"trigger.value.factName\"\n [oldValue]=\"trigger.value.oldValue | o3rFallbackTo\"\n [value]=\"trigger.value.newValue | o3rFallbackTo\"\n [type]=\"'state'\"></o3r-rule-key-value-pres>\n </li>\n </ng-container>\n </ul>\n</ng-template>\n\n<ng-template let-rules=\"rules\" let-expansionID=\"expansionID\" #rules>\n <!-- eslint-disable-next-line @angular-eslint/template/click-events-have-key-events, @angular-eslint/template/interactive-supports-focus -- need to refactor the div to accordion from DF #1518 -->\n <div class=\"ruleset-panel-category-title ruleset-expansion-action\"\n (click)=\"toggleExpansion(expansionID, 'rulesOverview')\">\n <span>Rules Overview</span>\n <button class=\"icon\"\n [class.icon-caret-down]=\"!expansionStatus[expansionID]?.rulesOverview\"\n [class.icon-caret-up]=\"expansionStatus[expansionID]?.rulesOverview\">\n </button>\n </div>\n <ng-container *ngIf=\"expansionStatus[expansionID]?.rulesOverview\">\n <div *ngIf=\"rules?.length === 0\" class=\"ruleset-panel-category-body empty\">No rule</div>\n <ul class=\"ruleset-panel-category-body\" *ngIf=\"rules?.length > 0\">\n <li *ngFor=\"let rule of rules\">\n <o3r-rule-tree-pres [name]=\"rule.name\"\n [condition]=\"rule?.rootElement?.condition\"\n [blockType]=\"rule?.rootElement?.blockType\"\n [successElements]=\"rule?.rootElement?.successElements\"\n [failureElements]=\"rule?.rootElement?.failureElements\">\n </o3r-rule-tree-pres>\n </li>\n </ul>\n </ng-container>\n</ng-template>\n\n<ng-template let-inputs=\"inputs\" let-runtimeInputs=\"runtimeInputs\" #inputs>\n <div class=\"ruleset-panel-category-title\">Inputs snapshot</div>\n <div *ngIf=\"inputs?.length === 0\" class=\"ruleset-panel-category-body empty\">No inputs</div>\n <ul class=\"ruleset-panel-category-body\" *ngIf=\"inputs?.length > 0\">\n <li *ngFor=\"let input of inputs\">\n <o3r-rule-key-value-pres\n [key]=\"input.factName\"\n [value]=\"input.value | o3rFallbackTo\"\n [type]=\"'state'\"></o3r-rule-key-value-pres>\n </li>\n <li *ngFor=\"let input of runtimeInputs\">{{input}} (scope limited to ruleset)</li>\n </ul>\n</ng-template>\n", styles: ["o3r-ruleset-history-pres .ruleset-panel-title,o3r-ruleset-history-pres .ruleset-panel-category-title{display:flex;justify-content:space-between;align-items:center}o3r-ruleset-history-pres .ruleset-panel-title{font-size:1rem;padding:.5rem 0 .1rem}o3r-ruleset-history-pres .ruleset-expansion-action,o3r-ruleset-history-pres .icon-caret-down,o3r-ruleset-history-pres .icon-caret-up{cursor:pointer}o3r-ruleset-history-pres .ruleset-panel-subtitle{font-size:.75rem}o3r-ruleset-history-pres .ruleset-panel-title-aside{display:flex;flex-wrap:wrap;justify-content:flex-end;align-items:center;min-width:fit-content}o3r-ruleset-history-pres .ruleset-panel-category-title{font-size:.95rem;background:#eee;padding:.5rem;margin-bottom:.5rem;margin-top:1rem}o3r-ruleset-history-pres .rule-description .ruleset-panel-category-title{font-size:.893rem;cursor:default}o3r-ruleset-history-pres .rule-description .ruleset-panel-title{font-size:.94rem}o3r-ruleset-history-pres .rule-description .ruleset-panel-category-body{padding-bottom:.5rem;padding-left:1.5rem}o3r-ruleset-history-pres .rule-description .ruleset-panel-category-body:empty{margin:0;padding:0 0 0 1.5rem}o3r-ruleset-history-pres .rulesets{margin:0;padding:0;list-style:none}o3r-ruleset-history-pres .rulesets ul{margin:0;padding-left:2rem}o3r-ruleset-history-pres .rulesets li.ruleset:nth-child(odd){background:#fbfbfb}o3r-ruleset-history-pres .ruleset-panel-category-body li{list-style:disc}o3r-ruleset-history-pres li:empty{display:none}o3r-ruleset-history-pres .ruleset{border-bottom:1px solid #dddddd}o3r-ruleset-history-pres .ruleset .ruleset-panel-description{padding:0 1rem 0 2rem;margin-bottom:2em}o3r-ruleset-history-pres .ruleset:first-child{border-top:1px solid #dddddd}o3r-ruleset-history-pres .ruleset-panel-title.ruleset-expansion-action{padding:.5rem 1rem}o3r-ruleset-history-pres .empty{font-style:italic;padding-left:1.5rem}o3r-ruleset-history-pres .ruleset-panel-category-body:empty.triggers:after{content:\"No trigger for this rule\";display:block;font-style:italic;padding-bottom:.5rem}o3r-ruleset-history-pres .ruleset-panel-title-aside{padding-left:1rem}o3r-ruleset-history-pres .rule-description .capsule{font-size:.8em;padding:.1rem .5rem;margin:0 .5rem}o3r-ruleset-history-pres .icon{background:none;border:none;font-size:1em}o3r-ruleset-history-pres .capsule{padding:.3rem;margin:.5rem;font-size:.875em;min-width:6rem;text-align:center}o3r-ruleset-history-pres .capsule.time{display:flex;flex-direction:column}o3r-ruleset-history-pres .success{background-color:#16aa32;color:#fff}o3r-ruleset-history-pres .inactive{background-color:#aaa;color:#fff}o3r-ruleset-history-pres .error{color:#c02020}o3r-ruleset-history-pres .error .capsule.error{background-color:#c02020;color:#fff}o3r-ruleset-history-pres .warn{background-color:#b92;color:#fff}o3r-ruleset-history-pres .input-key{color:#26c}o3r-ruleset-history-pres .input-value{color:#c29}o3r-ruleset-history-pres .input-key,o3r-ruleset-history-pres .input-value{font-family:monospace}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.RuleTreePresComponent, selector: "o3r-rule-tree-pres", inputs: ["name", "blockType", "condition", "successElements", "failureElements"] }, { kind: "component", type: i3.RuleActionsPresComponent, selector: "o3r-rule-actions-pres", inputs: ["actions", "temporaryFacts", "runtimeOutputs"] }, { kind: "component", type: i4.RuleKeyValuePresComponent, selector: "o3r-rule-key-value-pres", inputs: ["key", "value", "oldValue", "type"] }, { kind: "pipe", type: i5.O3rFallbackToPipe, name: "o3rFallbackTo" }, { kind: "pipe", type: i1.DecimalPipe, name: "number" }, { kind: "pipe", type: i1.TitleCasePipe, name: "titlecase" }, { kind: "pipe", type: i1.DatePipe, name: "date" }, { kind: "pipe", type: i1.KeyValuePipe, name: "keyvalue" }, { kind: "pipe", type: i6.O3rJsonOrStringPipe, name: "o3rJsonOrString" }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
40
+ /** @nocollapse */ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RulesetHistoryPresComponent, selector: "o3r-ruleset-history-pres", inputs: { rulesetExecutions: "rulesetExecutions", executionDurationFormat: "executionDurationFormat" }, ngImport: i0, template: "<section>\n <ng-template #noRulesEngine>\n <div class=\"alert alert-danger m-2\" role=\"alert\">\n The Rules Engine is not configured on this page.\n </div>\n </ng-template>\n <ul *ngIf=\"rulesetExecutions; else noRulesEngine\" class=\"rulesets\">\n <li *ngFor=\"let execution of rulesetExecutions\" class=\"ruleset\">\n <!-- eslint-disable-next-line @angular-eslint/template/click-events-have-key-events, @angular-eslint/template/interactive-supports-focus -- need to refactor the div to accordion from DF #1518 -->\n <div class=\"ruleset-panel-title ruleset-expansion-action\"\n [class.error]=\"execution.type === 'RulesetExecutionError'\"\n (click)=\"toggleExpansion(execution.executionId, 'ruleset')\">\n <div><span [title]=\"'This ruleset has been evaluated ' + execution.executionCounter + ' time(s)'\">{{execution.executionCounter}}</span> - {{execution.rulesetName | titlecase }}\n <div class=\"ruleset-panel-subtitle\" *ngIf=\"execution.rulesetInformation?.linkedComponents?.or || execution.rulesetInformation?.linkedComponent\">\n <ng-container *ngFor=\"let lc of (execution.rulesetInformation?.linkedComponents?.or || [execution.rulesetInformation.linkedComponent]); last as isLast\">\n <div>{{lc.name}} {{lc.library}} <span *ngIf=\"!isLast\"> OR </span></div>\n </ng-container>\n </div>\n <div class=\"ruleset-panel-subtitle\" *ngIf=\"execution.rulesetInformation?.validityRange as validityRange\">\n Date range: {{validityRange.from}} - {{validityRange.to}}\n </div>\n </div>\n <div class=\"ruleset-panel-title-aside\">\n <span class=\"error capsule\" *ngIf=\"execution.status === 'Error'\">Error</span>\n <span class=\"success capsule\" *ngIf=\"execution.status === 'Active'\">Applied</span>\n <span class=\"inactive capsule\" *ngIf=\"execution.status === 'Deactivated'\">Deactivated</span>\n <span class=\"warn capsule\" *ngIf=\"execution.status === 'NoEffect'\">No effect</span>\n <span class=\"time capsule\">\n <span>{{execution.timestamp | date: 'HH:mm:ss SSS'}}</span>\n <span>({{execution.duration | number: executionDurationFormat}}ms)</span>\n </span>\n <button\n class=\"icon\"\n [class.icon-caret-down]=\"!expansionStatus[execution.executionId]?.ruleset\"\n [class.icon-caret-up]=\"expansionStatus[execution.executionId]?.ruleset\">\n </button>\n </div>\n </div>\n <div class=\"ruleset-panel-description\" *ngIf=\"expansionStatus[execution.executionId]?.ruleset\">\n <ng-container [ngTemplateOutlet]=\"rules\"\n [ngTemplateOutletContext]=\"{\n rules: execution.rulesetInformation.rules,\n expansionID: execution.executionId\n }\"></ng-container>\n <ng-container [ngTemplateOutlet]=\"inputs\"\n [ngTemplateOutletContext]=\"{\n inputs: execution.inputFacts\n }\"></ng-container>\n <ng-container *ngIf=\"execution.type === 'RulesetExecutionError'; else success\">\n <div class=\"ruleset-panel-category-title\">Rules:</div>\n <ul class=\"ruleset-panel-category-body rule-description\">\n <li *ngFor=\"let ruleEvaluation of execution.rulesEvaluations; let index=index;\">\n <ng-container>\n <div class=\"ruleset-panel-title\" [class.error]=\"ruleEvaluation.error\">\n <span>{{ruleEvaluation.rule.name | titlecase}} </span>\n <span class=\"capsule error\" *ngIf=\"ruleEvaluation.error\">Error</span>\n </div>\n <div>\n <ng-container *ngIf=\"ruleEvaluation.error\">\n <span class=\"ruleset-panel-category-title\">Error:</span>\n <pre class=\"ruleset-panel-category-body error\">{{ruleEvaluation.error | o3rJsonOrString}}</pre>\n </ng-container>\n <ng-container [ngTemplateOutlet]=\"inputs\"\n [ngTemplateOutletContext]=\"{\n inputs: execution.inputFacts,\n runtimeInputs: execution.rulesetInformation?.rules[index]?.inputRuntimeFacts\n }\"></ng-container>\n <o3r-rule-actions-pres *ngIf=\"!ruleEvaluation.error\"\n [temporaryFacts]=\"ruleEvaluation.temporaryFacts\"\n [runtimeOutputs]=\"execution.rulesetInformation?.rules[index]?.outputRuntimeFacts\"\n ></o3r-rule-actions-pres>\n </div>\n </ng-container>\n </li>\n </ul>\n </ng-container>\n <ng-template #success>\n <o3r-rule-actions-pres [actions]=\"execution.outputActions\"></o3r-rule-actions-pres>\n <div class=\"ruleset-panel-category-title\">Executed Rules</div>\n <ul class=\"rule-description ruleset-panel-category-body\">\n <li *ngFor=\"let ruleEvaluation of execution.rulesEvaluations; let index=index;\">\n <div class=\"ruleset-panel-title\">\n <span>{{ruleEvaluation.rule.name | titlecase}}</span>\n <span class=\"capsule inactive\" *ngIf=\"ruleEvaluation.cached\">Cached</span>\n <span class=\"capsule\">({{ruleEvaluation.duration | number: executionDurationFormat}}ms)</span>\n </div>\n <div>\n <ng-container [ngTemplateOutlet]=\"triggers\"\n [ngTemplateOutletContext]=\"{triggers: (ruleEvaluation.triggers[ruleEvaluation.rule.id])}\"></ng-container>\n <o3r-rule-actions-pres\n [actions]=\"ruleEvaluation.outputActions\"\n [temporaryFacts]=\"ruleEvaluation.temporaryFacts\"\n [runtimeOutputs]=\"execution.rulesetInformation?.rules[index]?.outputRuntimeFacts\">\n </o3r-rule-actions-pres>\n </div>\n </li>\n </ul>\n </ng-template>\n </div>\n </li>\n </ul>\n</section>\n\n<ng-template let-triggers=\"triggers\" #triggers>\n <div class=\"ruleset-panel-category-title\">Basefacts Triggers</div>\n <ul class=\"ruleset-panel-category-body triggers\">\n <ng-container *ngFor=\"let trigger of (triggers | keyvalue)\">\n <li *ngIf=\"trigger.value?.factName\">\n <o3r-rule-key-value-pres\n [key]=\"trigger.value.factName\"\n [oldValue]=\"trigger.value.oldValue | o3rFallbackTo\"\n [value]=\"trigger.value.newValue | o3rFallbackTo\"\n [type]=\"'state'\"></o3r-rule-key-value-pres>\n </li>\n </ng-container>\n </ul>\n</ng-template>\n\n<ng-template let-rules=\"rules\" let-expansionID=\"expansionID\" #rules>\n <!-- eslint-disable-next-line @angular-eslint/template/click-events-have-key-events, @angular-eslint/template/interactive-supports-focus -- need to refactor the div to accordion from DF #1518 -->\n <div class=\"ruleset-panel-category-title ruleset-expansion-action\"\n (click)=\"toggleExpansion(expansionID, 'rulesOverview')\">\n <span>Rules Overview</span>\n <button class=\"icon\"\n [class.icon-caret-down]=\"!expansionStatus[expansionID]?.rulesOverview\"\n [class.icon-caret-up]=\"expansionStatus[expansionID]?.rulesOverview\">\n </button>\n </div>\n <ng-container *ngIf=\"expansionStatus[expansionID]?.rulesOverview\">\n <div *ngIf=\"rules?.length === 0\" class=\"ruleset-panel-category-body empty\">No rule</div>\n <ul class=\"ruleset-panel-category-body\" *ngIf=\"rules?.length > 0\">\n <li *ngFor=\"let rule of rules\">\n <o3r-rule-tree-pres [name]=\"rule.name\"\n [condition]=\"rule?.rootElement?.condition\"\n [blockType]=\"rule?.rootElement?.blockType\"\n [successElements]=\"rule?.rootElement?.successElements\"\n [failureElements]=\"rule?.rootElement?.failureElements\">\n </o3r-rule-tree-pres>\n </li>\n </ul>\n </ng-container>\n</ng-template>\n\n<ng-template let-inputs=\"inputs\" let-runtimeInputs=\"runtimeInputs\" #inputs>\n <div class=\"ruleset-panel-category-title\">Inputs snapshot</div>\n <div *ngIf=\"inputs?.length === 0\" class=\"ruleset-panel-category-body empty\">No inputs</div>\n <ul class=\"ruleset-panel-category-body\" *ngIf=\"inputs?.length > 0\">\n <li *ngFor=\"let input of inputs\">\n <o3r-rule-key-value-pres\n [key]=\"input.factName\"\n [value]=\"input.value | o3rFallbackTo\"\n [type]=\"'state'\"></o3r-rule-key-value-pres>\n </li>\n <li *ngFor=\"let input of runtimeInputs\">{{input}} (scope limited to ruleset)</li>\n </ul>\n</ng-template>\n", styles: ["o3r-ruleset-history-pres .ruleset-panel-title,o3r-ruleset-history-pres .ruleset-panel-category-title{display:flex;justify-content:space-between;align-items:center}o3r-ruleset-history-pres .ruleset-panel-title{font-size:1rem;padding:.5rem 0 .1rem}o3r-ruleset-history-pres .ruleset-expansion-action,o3r-ruleset-history-pres .icon-caret-down,o3r-ruleset-history-pres .icon-caret-up{cursor:pointer}o3r-ruleset-history-pres .ruleset-panel-subtitle{font-size:.75rem}o3r-ruleset-history-pres .ruleset-panel-title-aside{display:flex;flex-wrap:wrap;justify-content:flex-end;align-items:center;min-width:fit-content}o3r-ruleset-history-pres .ruleset-panel-category-title{font-size:.95rem;background:#eee;padding:.5rem;margin-bottom:.5rem;margin-top:1rem}o3r-ruleset-history-pres .rule-description .ruleset-panel-category-title{font-size:.893rem;cursor:default}o3r-ruleset-history-pres .rule-description .ruleset-panel-title{font-size:.94rem}o3r-ruleset-history-pres .rule-description .ruleset-panel-category-body{padding-bottom:.5rem;padding-left:1.5rem}o3r-ruleset-history-pres .rule-description .ruleset-panel-category-body:empty{margin:0;padding:0 0 0 1.5rem}o3r-ruleset-history-pres .rulesets{margin:0;padding:0;list-style:none}o3r-ruleset-history-pres .rulesets ul{margin:0;padding-left:2rem}o3r-ruleset-history-pres .rulesets li.ruleset:nth-child(odd){background:#fbfbfb}o3r-ruleset-history-pres .ruleset-panel-category-body li{list-style:disc}o3r-ruleset-history-pres li:empty{display:none}o3r-ruleset-history-pres .ruleset{border-bottom:1px solid #dddddd}o3r-ruleset-history-pres .ruleset .ruleset-panel-description{padding:0 1rem 0 2rem;margin-bottom:2em}o3r-ruleset-history-pres .ruleset:first-child{border-top:1px solid #dddddd}o3r-ruleset-history-pres .ruleset-panel-title.ruleset-expansion-action{padding:.5rem 1rem}o3r-ruleset-history-pres .empty{font-style:italic;padding-left:1.5rem}o3r-ruleset-history-pres .ruleset-panel-category-body:empty.triggers:after{content:\"No trigger for this rule\";display:block;font-style:italic;padding-bottom:.5rem}o3r-ruleset-history-pres .ruleset-panel-title-aside{padding-left:1rem}o3r-ruleset-history-pres .rule-description .capsule{font-size:.8em;padding:.1rem .5rem;margin:0 .5rem}o3r-ruleset-history-pres .icon{background:none;border:none;font-size:1em}o3r-ruleset-history-pres .capsule{padding:.3rem;margin:.5rem;font-size:.875em;min-width:6rem;text-align:center}o3r-ruleset-history-pres .capsule.time{display:flex;flex-direction:column}o3r-ruleset-history-pres .success{background-color:#16aa32;color:#fff}o3r-ruleset-history-pres .inactive{background-color:#aaa;color:#fff}o3r-ruleset-history-pres .error{color:#c02020}o3r-ruleset-history-pres .error .capsule.error{background-color:#c02020;color:#fff}o3r-ruleset-history-pres .warn{background-color:#b92;color:#fff}o3r-ruleset-history-pres .input-key{color:#26c}o3r-ruleset-history-pres .input-value{color:#c29}o3r-ruleset-history-pres .input-key,o3r-ruleset-history-pres .input-value{font-family:monospace}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.RuleTreePresComponent, selector: "o3r-rule-tree-pres", inputs: ["name", "blockType", "condition", "successElements", "failureElements"] }, { kind: "component", type: i3.RuleActionsPresComponent, selector: "o3r-rule-actions-pres", inputs: ["actions", "temporaryFacts", "runtimeOutputs"] }, { kind: "component", type: i4.RuleKeyValuePresComponent, selector: "o3r-rule-key-value-pres", inputs: ["key", "value", "oldValue", "type"] }, { kind: "pipe", type: i5.O3rFallbackToPipe, name: "o3rFallbackTo" }, { kind: "pipe", type: i1.DecimalPipe, name: "number" }, { kind: "pipe", type: i1.TitleCasePipe, name: "titlecase" }, { kind: "pipe", type: i1.DatePipe, name: "date" }, { kind: "pipe", type: i1.KeyValuePipe, name: "keyvalue" }, { kind: "pipe", type: i6.O3rJsonOrStringPipe, name: "o3rJsonOrString" }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
41
41
  }
42
42
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RulesetHistoryPresComponent, decorators: [{
43
43
  type: Component,
44
- args: [{ selector: 'o3r-ruleset-history-pres', changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, template: "<section>\n <h4 class=\"mb-4\">Ruleset Execution History</h4>\n <ng-template #noRulesEngine>\n <div class=\"alert alert-danger m-2\" role=\"alert\">\n The Rules Engine is not configured on this page.\n </div>\n </ng-template>\n <ul *ngIf=\"rulesetExecutions; else noRulesEngine\" class=\"rulesets\">\n <li *ngFor=\"let execution of rulesetExecutions\" class=\"ruleset\">\n <!-- eslint-disable-next-line @angular-eslint/template/click-events-have-key-events, @angular-eslint/template/interactive-supports-focus -- need to refactor the div to accordion from DF #1518 -->\n <div class=\"ruleset-panel-title ruleset-expansion-action\"\n [class.error]=\"execution.type === 'RulesetExecutionError'\"\n (click)=\"toggleExpansion(execution.executionId, 'ruleset')\">\n <div><span [title]=\"'This ruleset has been evaluated ' + execution.executionCounter + ' time(s)'\">{{execution.executionCounter}}</span> - {{execution.rulesetName | titlecase }}\n <div class=\"ruleset-panel-subtitle\" *ngIf=\"execution.rulesetInformation?.linkedComponents?.or || execution.rulesetInformation?.linkedComponent\">\n <ng-container *ngFor=\"let lc of (execution.rulesetInformation?.linkedComponents?.or || [execution.rulesetInformation.linkedComponent]); last as isLast\">\n <div>{{lc.name}} {{lc.library}} <span *ngIf=\"!isLast\"> OR </span></div>\n </ng-container>\n </div>\n <div class=\"ruleset-panel-subtitle\" *ngIf=\"execution.rulesetInformation?.validityRange as validityRange\">\n Date range: {{validityRange.from}} - {{validityRange.to}}\n </div>\n </div>\n <div class=\"ruleset-panel-title-aside\">\n <span class=\"error capsule\" *ngIf=\"execution.status === 'Error'\">Error</span>\n <span class=\"success capsule\" *ngIf=\"execution.status === 'Active'\">Applied</span>\n <span class=\"inactive capsule\" *ngIf=\"execution.status === 'Deactivated'\">Deactivated</span>\n <span class=\"warn capsule\" *ngIf=\"execution.status === 'NoEffect'\">No effect</span>\n <span class=\"time capsule\">\n <span>{{execution.timestamp | date: 'HH:mm:ss SSS'}}</span>\n <span>({{execution.duration | number: executionDurationFormat}}ms)</span>\n </span>\n <button\n class=\"icon\"\n [class.icon-caret-down]=\"!expansionStatus[execution.executionId]?.ruleset\"\n [class.icon-caret-up]=\"expansionStatus[execution.executionId]?.ruleset\">\n </button>\n </div>\n </div>\n <div class=\"ruleset-panel-description\" *ngIf=\"expansionStatus[execution.executionId]?.ruleset\">\n <ng-container [ngTemplateOutlet]=\"rules\"\n [ngTemplateOutletContext]=\"{\n rules: execution.rulesetInformation.rules,\n expansionID: execution.executionId\n }\"></ng-container>\n <ng-container [ngTemplateOutlet]=\"inputs\"\n [ngTemplateOutletContext]=\"{\n inputs: execution.inputFacts\n }\"></ng-container>\n <ng-container *ngIf=\"execution.type === 'RulesetExecutionError'; else success\">\n <div class=\"ruleset-panel-category-title\">Rules:</div>\n <ul class=\"ruleset-panel-category-body rule-description\">\n <li *ngFor=\"let ruleEvaluation of execution.rulesEvaluations; let index=index;\">\n <ng-container>\n <div class=\"ruleset-panel-title\" [class.error]=\"ruleEvaluation.error\">\n <span>{{ruleEvaluation.rule.name | titlecase}} </span>\n <span class=\"capsule error\" *ngIf=\"ruleEvaluation.error\">Error</span>\n </div>\n <div>\n <ng-container *ngIf=\"ruleEvaluation.error\">\n <span class=\"ruleset-panel-category-title\">Error:</span>\n <pre class=\"ruleset-panel-category-body error\">{{ruleEvaluation.error | o3rJsonOrString}}</pre>\n </ng-container>\n <ng-container [ngTemplateOutlet]=\"inputs\"\n [ngTemplateOutletContext]=\"{\n inputs: execution.inputFacts,\n runtimeInputs: execution.rulesetInformation?.rules[index]?.inputRuntimeFacts\n }\"></ng-container>\n <o3r-rule-actions-pres *ngIf=\"!ruleEvaluation.error\"\n [temporaryFacts]=\"ruleEvaluation.temporaryFacts\"\n [runtimeOutputs]=\"execution.rulesetInformation?.rules[index]?.outputRuntimeFacts\"\n ></o3r-rule-actions-pres>\n </div>\n </ng-container>\n </li>\n </ul>\n </ng-container>\n <ng-template #success>\n <o3r-rule-actions-pres [actions]=\"execution.outputActions\"></o3r-rule-actions-pres>\n <div class=\"ruleset-panel-category-title\">Executed Rules</div>\n <ul class=\"rule-description ruleset-panel-category-body\">\n <li *ngFor=\"let ruleEvaluation of execution.rulesEvaluations; let index=index;\">\n <div class=\"ruleset-panel-title\">\n <span>{{ruleEvaluation.rule.name | titlecase}}</span>\n <span class=\"capsule inactive\" *ngIf=\"ruleEvaluation.cached\">Cached</span>\n <span class=\"capsule\">({{ruleEvaluation.duration | number: executionDurationFormat}}ms)</span>\n </div>\n <div>\n <ng-container [ngTemplateOutlet]=\"triggers\"\n [ngTemplateOutletContext]=\"{triggers: (ruleEvaluation.triggers[ruleEvaluation.rule.id])}\"></ng-container>\n <o3r-rule-actions-pres\n [actions]=\"ruleEvaluation.outputActions\"\n [temporaryFacts]=\"ruleEvaluation.temporaryFacts\"\n [runtimeOutputs]=\"execution.rulesetInformation?.rules[index]?.outputRuntimeFacts\">\n </o3r-rule-actions-pres>\n </div>\n </li>\n </ul>\n </ng-template>\n </div>\n </li>\n </ul>\n</section>\n\n<ng-template let-triggers=\"triggers\" #triggers>\n <div class=\"ruleset-panel-category-title\">Basefacts Triggers</div>\n <ul class=\"ruleset-panel-category-body triggers\">\n <ng-container *ngFor=\"let trigger of (triggers | keyvalue)\">\n <li *ngIf=\"trigger.value?.factName\">\n <o3r-rule-key-value-pres\n [key]=\"trigger.value.factName\"\n [oldValue]=\"trigger.value.oldValue | o3rFallbackTo\"\n [value]=\"trigger.value.newValue | o3rFallbackTo\"\n [type]=\"'state'\"></o3r-rule-key-value-pres>\n </li>\n </ng-container>\n </ul>\n</ng-template>\n\n<ng-template let-rules=\"rules\" let-expansionID=\"expansionID\" #rules>\n <!-- eslint-disable-next-line @angular-eslint/template/click-events-have-key-events, @angular-eslint/template/interactive-supports-focus -- need to refactor the div to accordion from DF #1518 -->\n <div class=\"ruleset-panel-category-title ruleset-expansion-action\"\n (click)=\"toggleExpansion(expansionID, 'rulesOverview')\">\n <span>Rules Overview</span>\n <button class=\"icon\"\n [class.icon-caret-down]=\"!expansionStatus[expansionID]?.rulesOverview\"\n [class.icon-caret-up]=\"expansionStatus[expansionID]?.rulesOverview\">\n </button>\n </div>\n <ng-container *ngIf=\"expansionStatus[expansionID]?.rulesOverview\">\n <div *ngIf=\"rules?.length === 0\" class=\"ruleset-panel-category-body empty\">No rule</div>\n <ul class=\"ruleset-panel-category-body\" *ngIf=\"rules?.length > 0\">\n <li *ngFor=\"let rule of rules\">\n <o3r-rule-tree-pres [name]=\"rule.name\"\n [condition]=\"rule?.rootElement?.condition\"\n [blockType]=\"rule?.rootElement?.blockType\"\n [successElements]=\"rule?.rootElement?.successElements\"\n [failureElements]=\"rule?.rootElement?.failureElements\">\n </o3r-rule-tree-pres>\n </li>\n </ul>\n </ng-container>\n</ng-template>\n\n<ng-template let-inputs=\"inputs\" let-runtimeInputs=\"runtimeInputs\" #inputs>\n <div class=\"ruleset-panel-category-title\">Inputs snapshot</div>\n <div *ngIf=\"inputs?.length === 0\" class=\"ruleset-panel-category-body empty\">No inputs</div>\n <ul class=\"ruleset-panel-category-body\" *ngIf=\"inputs?.length > 0\">\n <li *ngFor=\"let input of inputs\">\n <o3r-rule-key-value-pres\n [key]=\"input.factName\"\n [value]=\"input.value | o3rFallbackTo\"\n [type]=\"'state'\"></o3r-rule-key-value-pres>\n </li>\n <li *ngFor=\"let input of runtimeInputs\">{{input}} (scope limited to ruleset)</li>\n </ul>\n</ng-template>\n", styles: ["o3r-ruleset-history-pres .ruleset-panel-title,o3r-ruleset-history-pres .ruleset-panel-category-title{display:flex;justify-content:space-between;align-items:center}o3r-ruleset-history-pres .ruleset-panel-title{font-size:1rem;padding:.5rem 0 .1rem}o3r-ruleset-history-pres .ruleset-expansion-action,o3r-ruleset-history-pres .icon-caret-down,o3r-ruleset-history-pres .icon-caret-up{cursor:pointer}o3r-ruleset-history-pres .ruleset-panel-subtitle{font-size:.75rem}o3r-ruleset-history-pres .ruleset-panel-title-aside{display:flex;flex-wrap:wrap;justify-content:flex-end;align-items:center;min-width:fit-content}o3r-ruleset-history-pres .ruleset-panel-category-title{font-size:.95rem;background:#eee;padding:.5rem;margin-bottom:.5rem;margin-top:1rem}o3r-ruleset-history-pres .rule-description .ruleset-panel-category-title{font-size:.893rem;cursor:default}o3r-ruleset-history-pres .rule-description .ruleset-panel-title{font-size:.94rem}o3r-ruleset-history-pres .rule-description .ruleset-panel-category-body{padding-bottom:.5rem;padding-left:1.5rem}o3r-ruleset-history-pres .rule-description .ruleset-panel-category-body:empty{margin:0;padding:0 0 0 1.5rem}o3r-ruleset-history-pres .rulesets{margin:0;padding:0;list-style:none}o3r-ruleset-history-pres .rulesets ul{margin:0;padding-left:2rem}o3r-ruleset-history-pres .rulesets li.ruleset:nth-child(odd){background:#fbfbfb}o3r-ruleset-history-pres .ruleset-panel-category-body li{list-style:disc}o3r-ruleset-history-pres li:empty{display:none}o3r-ruleset-history-pres .ruleset{border-bottom:1px solid #dddddd}o3r-ruleset-history-pres .ruleset .ruleset-panel-description{padding:0 1rem 0 2rem;margin-bottom:2em}o3r-ruleset-history-pres .ruleset:first-child{border-top:1px solid #dddddd}o3r-ruleset-history-pres .ruleset-panel-title.ruleset-expansion-action{padding:.5rem 1rem}o3r-ruleset-history-pres .empty{font-style:italic;padding-left:1.5rem}o3r-ruleset-history-pres .ruleset-panel-category-body:empty.triggers:after{content:\"No trigger for this rule\";display:block;font-style:italic;padding-bottom:.5rem}o3r-ruleset-history-pres .ruleset-panel-title-aside{padding-left:1rem}o3r-ruleset-history-pres .rule-description .capsule{font-size:.8em;padding:.1rem .5rem;margin:0 .5rem}o3r-ruleset-history-pres .icon{background:none;border:none;font-size:1em}o3r-ruleset-history-pres .capsule{padding:.3rem;margin:.5rem;font-size:.875em;min-width:6rem;text-align:center}o3r-ruleset-history-pres .capsule.time{display:flex;flex-direction:column}o3r-ruleset-history-pres .success{background-color:#16aa32;color:#fff}o3r-ruleset-history-pres .inactive{background-color:#aaa;color:#fff}o3r-ruleset-history-pres .error{color:#c02020}o3r-ruleset-history-pres .error .capsule.error{background-color:#c02020;color:#fff}o3r-ruleset-history-pres .warn{background-color:#b92;color:#fff}o3r-ruleset-history-pres .input-key{color:#26c}o3r-ruleset-history-pres .input-value{color:#c29}o3r-ruleset-history-pres .input-key,o3r-ruleset-history-pres .input-value{font-family:monospace}\n"] }]
44
+ args: [{ selector: 'o3r-ruleset-history-pres', changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, template: "<section>\n <ng-template #noRulesEngine>\n <div class=\"alert alert-danger m-2\" role=\"alert\">\n The Rules Engine is not configured on this page.\n </div>\n </ng-template>\n <ul *ngIf=\"rulesetExecutions; else noRulesEngine\" class=\"rulesets\">\n <li *ngFor=\"let execution of rulesetExecutions\" class=\"ruleset\">\n <!-- eslint-disable-next-line @angular-eslint/template/click-events-have-key-events, @angular-eslint/template/interactive-supports-focus -- need to refactor the div to accordion from DF #1518 -->\n <div class=\"ruleset-panel-title ruleset-expansion-action\"\n [class.error]=\"execution.type === 'RulesetExecutionError'\"\n (click)=\"toggleExpansion(execution.executionId, 'ruleset')\">\n <div><span [title]=\"'This ruleset has been evaluated ' + execution.executionCounter + ' time(s)'\">{{execution.executionCounter}}</span> - {{execution.rulesetName | titlecase }}\n <div class=\"ruleset-panel-subtitle\" *ngIf=\"execution.rulesetInformation?.linkedComponents?.or || execution.rulesetInformation?.linkedComponent\">\n <ng-container *ngFor=\"let lc of (execution.rulesetInformation?.linkedComponents?.or || [execution.rulesetInformation.linkedComponent]); last as isLast\">\n <div>{{lc.name}} {{lc.library}} <span *ngIf=\"!isLast\"> OR </span></div>\n </ng-container>\n </div>\n <div class=\"ruleset-panel-subtitle\" *ngIf=\"execution.rulesetInformation?.validityRange as validityRange\">\n Date range: {{validityRange.from}} - {{validityRange.to}}\n </div>\n </div>\n <div class=\"ruleset-panel-title-aside\">\n <span class=\"error capsule\" *ngIf=\"execution.status === 'Error'\">Error</span>\n <span class=\"success capsule\" *ngIf=\"execution.status === 'Active'\">Applied</span>\n <span class=\"inactive capsule\" *ngIf=\"execution.status === 'Deactivated'\">Deactivated</span>\n <span class=\"warn capsule\" *ngIf=\"execution.status === 'NoEffect'\">No effect</span>\n <span class=\"time capsule\">\n <span>{{execution.timestamp | date: 'HH:mm:ss SSS'}}</span>\n <span>({{execution.duration | number: executionDurationFormat}}ms)</span>\n </span>\n <button\n class=\"icon\"\n [class.icon-caret-down]=\"!expansionStatus[execution.executionId]?.ruleset\"\n [class.icon-caret-up]=\"expansionStatus[execution.executionId]?.ruleset\">\n </button>\n </div>\n </div>\n <div class=\"ruleset-panel-description\" *ngIf=\"expansionStatus[execution.executionId]?.ruleset\">\n <ng-container [ngTemplateOutlet]=\"rules\"\n [ngTemplateOutletContext]=\"{\n rules: execution.rulesetInformation.rules,\n expansionID: execution.executionId\n }\"></ng-container>\n <ng-container [ngTemplateOutlet]=\"inputs\"\n [ngTemplateOutletContext]=\"{\n inputs: execution.inputFacts\n }\"></ng-container>\n <ng-container *ngIf=\"execution.type === 'RulesetExecutionError'; else success\">\n <div class=\"ruleset-panel-category-title\">Rules:</div>\n <ul class=\"ruleset-panel-category-body rule-description\">\n <li *ngFor=\"let ruleEvaluation of execution.rulesEvaluations; let index=index;\">\n <ng-container>\n <div class=\"ruleset-panel-title\" [class.error]=\"ruleEvaluation.error\">\n <span>{{ruleEvaluation.rule.name | titlecase}} </span>\n <span class=\"capsule error\" *ngIf=\"ruleEvaluation.error\">Error</span>\n </div>\n <div>\n <ng-container *ngIf=\"ruleEvaluation.error\">\n <span class=\"ruleset-panel-category-title\">Error:</span>\n <pre class=\"ruleset-panel-category-body error\">{{ruleEvaluation.error | o3rJsonOrString}}</pre>\n </ng-container>\n <ng-container [ngTemplateOutlet]=\"inputs\"\n [ngTemplateOutletContext]=\"{\n inputs: execution.inputFacts,\n runtimeInputs: execution.rulesetInformation?.rules[index]?.inputRuntimeFacts\n }\"></ng-container>\n <o3r-rule-actions-pres *ngIf=\"!ruleEvaluation.error\"\n [temporaryFacts]=\"ruleEvaluation.temporaryFacts\"\n [runtimeOutputs]=\"execution.rulesetInformation?.rules[index]?.outputRuntimeFacts\"\n ></o3r-rule-actions-pres>\n </div>\n </ng-container>\n </li>\n </ul>\n </ng-container>\n <ng-template #success>\n <o3r-rule-actions-pres [actions]=\"execution.outputActions\"></o3r-rule-actions-pres>\n <div class=\"ruleset-panel-category-title\">Executed Rules</div>\n <ul class=\"rule-description ruleset-panel-category-body\">\n <li *ngFor=\"let ruleEvaluation of execution.rulesEvaluations; let index=index;\">\n <div class=\"ruleset-panel-title\">\n <span>{{ruleEvaluation.rule.name | titlecase}}</span>\n <span class=\"capsule inactive\" *ngIf=\"ruleEvaluation.cached\">Cached</span>\n <span class=\"capsule\">({{ruleEvaluation.duration | number: executionDurationFormat}}ms)</span>\n </div>\n <div>\n <ng-container [ngTemplateOutlet]=\"triggers\"\n [ngTemplateOutletContext]=\"{triggers: (ruleEvaluation.triggers[ruleEvaluation.rule.id])}\"></ng-container>\n <o3r-rule-actions-pres\n [actions]=\"ruleEvaluation.outputActions\"\n [temporaryFacts]=\"ruleEvaluation.temporaryFacts\"\n [runtimeOutputs]=\"execution.rulesetInformation?.rules[index]?.outputRuntimeFacts\">\n </o3r-rule-actions-pres>\n </div>\n </li>\n </ul>\n </ng-template>\n </div>\n </li>\n </ul>\n</section>\n\n<ng-template let-triggers=\"triggers\" #triggers>\n <div class=\"ruleset-panel-category-title\">Basefacts Triggers</div>\n <ul class=\"ruleset-panel-category-body triggers\">\n <ng-container *ngFor=\"let trigger of (triggers | keyvalue)\">\n <li *ngIf=\"trigger.value?.factName\">\n <o3r-rule-key-value-pres\n [key]=\"trigger.value.factName\"\n [oldValue]=\"trigger.value.oldValue | o3rFallbackTo\"\n [value]=\"trigger.value.newValue | o3rFallbackTo\"\n [type]=\"'state'\"></o3r-rule-key-value-pres>\n </li>\n </ng-container>\n </ul>\n</ng-template>\n\n<ng-template let-rules=\"rules\" let-expansionID=\"expansionID\" #rules>\n <!-- eslint-disable-next-line @angular-eslint/template/click-events-have-key-events, @angular-eslint/template/interactive-supports-focus -- need to refactor the div to accordion from DF #1518 -->\n <div class=\"ruleset-panel-category-title ruleset-expansion-action\"\n (click)=\"toggleExpansion(expansionID, 'rulesOverview')\">\n <span>Rules Overview</span>\n <button class=\"icon\"\n [class.icon-caret-down]=\"!expansionStatus[expansionID]?.rulesOverview\"\n [class.icon-caret-up]=\"expansionStatus[expansionID]?.rulesOverview\">\n </button>\n </div>\n <ng-container *ngIf=\"expansionStatus[expansionID]?.rulesOverview\">\n <div *ngIf=\"rules?.length === 0\" class=\"ruleset-panel-category-body empty\">No rule</div>\n <ul class=\"ruleset-panel-category-body\" *ngIf=\"rules?.length > 0\">\n <li *ngFor=\"let rule of rules\">\n <o3r-rule-tree-pres [name]=\"rule.name\"\n [condition]=\"rule?.rootElement?.condition\"\n [blockType]=\"rule?.rootElement?.blockType\"\n [successElements]=\"rule?.rootElement?.successElements\"\n [failureElements]=\"rule?.rootElement?.failureElements\">\n </o3r-rule-tree-pres>\n </li>\n </ul>\n </ng-container>\n</ng-template>\n\n<ng-template let-inputs=\"inputs\" let-runtimeInputs=\"runtimeInputs\" #inputs>\n <div class=\"ruleset-panel-category-title\">Inputs snapshot</div>\n <div *ngIf=\"inputs?.length === 0\" class=\"ruleset-panel-category-body empty\">No inputs</div>\n <ul class=\"ruleset-panel-category-body\" *ngIf=\"inputs?.length > 0\">\n <li *ngFor=\"let input of inputs\">\n <o3r-rule-key-value-pres\n [key]=\"input.factName\"\n [value]=\"input.value | o3rFallbackTo\"\n [type]=\"'state'\"></o3r-rule-key-value-pres>\n </li>\n <li *ngFor=\"let input of runtimeInputs\">{{input}} (scope limited to ruleset)</li>\n </ul>\n</ng-template>\n", styles: ["o3r-ruleset-history-pres .ruleset-panel-title,o3r-ruleset-history-pres .ruleset-panel-category-title{display:flex;justify-content:space-between;align-items:center}o3r-ruleset-history-pres .ruleset-panel-title{font-size:1rem;padding:.5rem 0 .1rem}o3r-ruleset-history-pres .ruleset-expansion-action,o3r-ruleset-history-pres .icon-caret-down,o3r-ruleset-history-pres .icon-caret-up{cursor:pointer}o3r-ruleset-history-pres .ruleset-panel-subtitle{font-size:.75rem}o3r-ruleset-history-pres .ruleset-panel-title-aside{display:flex;flex-wrap:wrap;justify-content:flex-end;align-items:center;min-width:fit-content}o3r-ruleset-history-pres .ruleset-panel-category-title{font-size:.95rem;background:#eee;padding:.5rem;margin-bottom:.5rem;margin-top:1rem}o3r-ruleset-history-pres .rule-description .ruleset-panel-category-title{font-size:.893rem;cursor:default}o3r-ruleset-history-pres .rule-description .ruleset-panel-title{font-size:.94rem}o3r-ruleset-history-pres .rule-description .ruleset-panel-category-body{padding-bottom:.5rem;padding-left:1.5rem}o3r-ruleset-history-pres .rule-description .ruleset-panel-category-body:empty{margin:0;padding:0 0 0 1.5rem}o3r-ruleset-history-pres .rulesets{margin:0;padding:0;list-style:none}o3r-ruleset-history-pres .rulesets ul{margin:0;padding-left:2rem}o3r-ruleset-history-pres .rulesets li.ruleset:nth-child(odd){background:#fbfbfb}o3r-ruleset-history-pres .ruleset-panel-category-body li{list-style:disc}o3r-ruleset-history-pres li:empty{display:none}o3r-ruleset-history-pres .ruleset{border-bottom:1px solid #dddddd}o3r-ruleset-history-pres .ruleset .ruleset-panel-description{padding:0 1rem 0 2rem;margin-bottom:2em}o3r-ruleset-history-pres .ruleset:first-child{border-top:1px solid #dddddd}o3r-ruleset-history-pres .ruleset-panel-title.ruleset-expansion-action{padding:.5rem 1rem}o3r-ruleset-history-pres .empty{font-style:italic;padding-left:1.5rem}o3r-ruleset-history-pres .ruleset-panel-category-body:empty.triggers:after{content:\"No trigger for this rule\";display:block;font-style:italic;padding-bottom:.5rem}o3r-ruleset-history-pres .ruleset-panel-title-aside{padding-left:1rem}o3r-ruleset-history-pres .rule-description .capsule{font-size:.8em;padding:.1rem .5rem;margin:0 .5rem}o3r-ruleset-history-pres .icon{background:none;border:none;font-size:1em}o3r-ruleset-history-pres .capsule{padding:.3rem;margin:.5rem;font-size:.875em;min-width:6rem;text-align:center}o3r-ruleset-history-pres .capsule.time{display:flex;flex-direction:column}o3r-ruleset-history-pres .success{background-color:#16aa32;color:#fff}o3r-ruleset-history-pres .inactive{background-color:#aaa;color:#fff}o3r-ruleset-history-pres .error{color:#c02020}o3r-ruleset-history-pres .error .capsule.error{background-color:#c02020;color:#fff}o3r-ruleset-history-pres .warn{background-color:#b92;color:#fff}o3r-ruleset-history-pres .input-key{color:#26c}o3r-ruleset-history-pres .input-value{color:#c29}o3r-ruleset-history-pres .input-key,o3r-ruleset-history-pres .input-value{font-family:monospace}\n"] }]
45
45
  }], ctorParameters: () => [{ type: i0.ChangeDetectorRef }], propDecorators: { rulesetExecutions: [{
46
46
  type: Input
47
47
  }], executionDurationFormat: [{
48
48
  type: Input
49
49
  }] } });
50
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ruleset-history-pres.component.js","sourceRoot":"","sources":["../../../../../src/components/rules-engine/ruleset-history/ruleset-history-pres.component.ts","../../../../../src/components/rules-engine/ruleset-history/ruleset-history-pres.template.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,iBAAiB,EACjB,SAAS,EACT,KAAK,EACL,iBAAiB,GAClB,MAAM,eAAe,CAAC;;;;;;;;AAwBvB,MAAM,OAAO,2BAA2B;IAkBtC,YAA6B,EAAqB;QAArB,OAAE,GAAF,EAAE,CAAmB;QAjBlD;;;;;;;;WAQG;QACI,oBAAe,GAAuD,EAAE,CAAC;QAGzE,sBAAiB,GAA4B,EAAE,CAAC;QAGhD,4BAAuB,GAAG,OAAO,CAAC;IAEY,CAAC;IAEtD;;;;OAIG;IACI,eAAe,CAAC,MAAc,EAAE,QAAgB;QACrD,IAAI,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,EAAE,CAAC;YACjC,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,CAAC;QACnF,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,QAAQ,CAAC,EAAE,IAAI,EAAE,CAAC;QACtD,CAAC;QACD,IAAI,CAAC,EAAE,CAAC,aAAa,EAAE,CAAC;IAC1B,CAAC;kIAhCU,2BAA2B;sHAA3B,2BAA2B,wKC9BxC,q0RA6JA;;4FD/Ha,2BAA2B;kBAPvC,SAAS;+BACE,0BAA0B,mBAGnB,uBAAuB,CAAC,MAAM,iBAChC,iBAAiB,CAAC,IAAI;sFAe9B,iBAAiB;sBADvB,KAAK;gBAIC,uBAAuB;sBAD7B,KAAK","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  Input,\n  ViewEncapsulation,\n} from '@angular/core';\nimport type {\n  Ruleset,\n  RulesetExecutionErrorEvent,\n  RulesetExecutionEvent,\n} from '../../../engine';\n\nexport type RulesetExecutionStatus = 'Error' | 'Active' | 'Deactivated' | 'NoEffect';\n/**\n * Model of a RulesetExecution with more information for debug purpose\n */\nexport type RulesetExecutionDebug = (RulesetExecutionEvent | RulesetExecutionErrorEvent) & {\n  isActive: boolean;\n  status: RulesetExecutionStatus;\n  rulesetInformation: Ruleset | undefined;\n};\n\n@Component({\n  selector: 'o3r-ruleset-history-pres',\n  styleUrls: ['./ruleset-history-pres.style.scss'],\n  templateUrl: './ruleset-history-pres.template.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None\n})\nexport class RulesetHistoryPresComponent {\n  /**\n   * Reflects the state of each ruleset expanded elements.\n   * Each ruleset entry contains a list of subpanel that can be collapsed or expanded.\n   * Ruleset whole panel status is store the 'ruleset' entry.\n   * @example\n   * Expanded ruleset with rule overview collapsed:\n   * {'rulesetId': {'ruleset' : true, 'ruleOverview': false}}\n   * @note Collapsing a ruleset will not reset the subpanel expansion status\n   */\n  public expansionStatus: { [key: string]: { [subpanel: string]: boolean } } = {};\n\n  @Input()\n  public rulesetExecutions: RulesetExecutionDebug[] = [];\n\n  @Input()\n  public executionDurationFormat = '1.3-3';\n\n  constructor(private readonly cd: ChangeDetectorRef) {}\n\n  /**\n   * Toggle a ruleset subpanel\n   * @param ruleId\n   * @param subpanel element to collapse. 'ruleset' will toggle the whole panel but won't reset the subpanels states.\n   */\n  public toggleExpansion(ruleId: string, subpanel: string) {\n    if (this.expansionStatus[ruleId]) {\n      this.expansionStatus[ruleId][subpanel] = !this.expansionStatus[ruleId][subpanel];\n    } else {\n      this.expansionStatus[ruleId] = { [subpanel]: true };\n    }\n    this.cd.detectChanges();\n  }\n}\n","<section>\n  <h4 class=\"mb-4\">Ruleset Execution History</h4>\n  <ng-template #noRulesEngine>\n    <div class=\"alert alert-danger m-2\" role=\"alert\">\n      The Rules Engine is not configured on this page.\n    </div>\n  </ng-template>\n  <ul *ngIf=\"rulesetExecutions; else noRulesEngine\" class=\"rulesets\">\n    <li *ngFor=\"let execution of rulesetExecutions\" class=\"ruleset\">\n      <!-- eslint-disable-next-line @angular-eslint/template/click-events-have-key-events, @angular-eslint/template/interactive-supports-focus -- need to refactor the div to accordion from DF #1518 -->\n      <div class=\"ruleset-panel-title ruleset-expansion-action\"\n        [class.error]=\"execution.type === 'RulesetExecutionError'\"\n        (click)=\"toggleExpansion(execution.executionId, 'ruleset')\">\n        <div><span [title]=\"'This ruleset has been evaluated ' + execution.executionCounter + ' time(s)'\">{{execution.executionCounter}}</span> - {{execution.rulesetName | titlecase }}\n          <div class=\"ruleset-panel-subtitle\" *ngIf=\"execution.rulesetInformation?.linkedComponents?.or || execution.rulesetInformation?.linkedComponent\">\n            <ng-container *ngFor=\"let lc of (execution.rulesetInformation?.linkedComponents?.or || [execution.rulesetInformation.linkedComponent]); last as isLast\">\n              <div>{{lc.name}} {{lc.library}} <span *ngIf=\"!isLast\"> OR </span></div>\n            </ng-container>\n          </div>\n          <div class=\"ruleset-panel-subtitle\" *ngIf=\"execution.rulesetInformation?.validityRange as validityRange\">\n            Date range: {{validityRange.from}} - {{validityRange.to}}\n          </div>\n        </div>\n        <div class=\"ruleset-panel-title-aside\">\n          <span class=\"error capsule\" *ngIf=\"execution.status === 'Error'\">Error</span>\n          <span class=\"success capsule\" *ngIf=\"execution.status === 'Active'\">Applied</span>\n          <span class=\"inactive capsule\" *ngIf=\"execution.status === 'Deactivated'\">Deactivated</span>\n          <span class=\"warn capsule\" *ngIf=\"execution.status === 'NoEffect'\">No effect</span>\n          <span class=\"time capsule\">\n            <span>{{execution.timestamp | date: 'HH:mm:ss SSS'}}</span>\n            <span>({{execution.duration | number: executionDurationFormat}}ms)</span>\n          </span>\n          <button\n            class=\"icon\"\n            [class.icon-caret-down]=\"!expansionStatus[execution.executionId]?.ruleset\"\n            [class.icon-caret-up]=\"expansionStatus[execution.executionId]?.ruleset\">\n          </button>\n        </div>\n      </div>\n      <div class=\"ruleset-panel-description\" *ngIf=\"expansionStatus[execution.executionId]?.ruleset\">\n        <ng-container [ngTemplateOutlet]=\"rules\"\n                      [ngTemplateOutletContext]=\"{\n                        rules: execution.rulesetInformation.rules,\n                        expansionID: execution.executionId\n                      }\"></ng-container>\n        <ng-container [ngTemplateOutlet]=\"inputs\"\n                      [ngTemplateOutletContext]=\"{\n                        inputs: execution.inputFacts\n                      }\"></ng-container>\n        <ng-container *ngIf=\"execution.type === 'RulesetExecutionError'; else success\">\n          <div class=\"ruleset-panel-category-title\">Rules:</div>\n          <ul class=\"ruleset-panel-category-body rule-description\">\n            <li *ngFor=\"let ruleEvaluation of execution.rulesEvaluations; let index=index;\">\n              <ng-container>\n                <div class=\"ruleset-panel-title\" [class.error]=\"ruleEvaluation.error\">\n                  <span>{{ruleEvaluation.rule.name | titlecase}} </span>\n                  <span class=\"capsule error\" *ngIf=\"ruleEvaluation.error\">Error</span>\n                </div>\n                <div>\n                  <ng-container *ngIf=\"ruleEvaluation.error\">\n                    <span class=\"ruleset-panel-category-title\">Error:</span>\n                    <pre class=\"ruleset-panel-category-body error\">{{ruleEvaluation.error | o3rJsonOrString}}</pre>\n                  </ng-container>\n                  <ng-container [ngTemplateOutlet]=\"inputs\"\n                                [ngTemplateOutletContext]=\"{\n                                  inputs: execution.inputFacts,\n                                  runtimeInputs: execution.rulesetInformation?.rules[index]?.inputRuntimeFacts\n                                }\"></ng-container>\n                  <o3r-rule-actions-pres *ngIf=\"!ruleEvaluation.error\"\n                    [temporaryFacts]=\"ruleEvaluation.temporaryFacts\"\n                    [runtimeOutputs]=\"execution.rulesetInformation?.rules[index]?.outputRuntimeFacts\"\n                  ></o3r-rule-actions-pres>\n                </div>\n              </ng-container>\n            </li>\n          </ul>\n        </ng-container>\n        <ng-template #success>\n          <o3r-rule-actions-pres [actions]=\"execution.outputActions\"></o3r-rule-actions-pres>\n          <div class=\"ruleset-panel-category-title\">Executed Rules</div>\n          <ul class=\"rule-description ruleset-panel-category-body\">\n            <li *ngFor=\"let ruleEvaluation of execution.rulesEvaluations; let index=index;\">\n              <div class=\"ruleset-panel-title\">\n                <span>{{ruleEvaluation.rule.name | titlecase}}</span>\n                <span class=\"capsule inactive\" *ngIf=\"ruleEvaluation.cached\">Cached</span>\n                <span class=\"capsule\">({{ruleEvaluation.duration | number: executionDurationFormat}}ms)</span>\n              </div>\n              <div>\n                <ng-container [ngTemplateOutlet]=\"triggers\"\n                              [ngTemplateOutletContext]=\"{triggers: (ruleEvaluation.triggers[ruleEvaluation.rule.id])}\"></ng-container>\n                <o3r-rule-actions-pres\n                  [actions]=\"ruleEvaluation.outputActions\"\n                  [temporaryFacts]=\"ruleEvaluation.temporaryFacts\"\n                  [runtimeOutputs]=\"execution.rulesetInformation?.rules[index]?.outputRuntimeFacts\">\n                </o3r-rule-actions-pres>\n              </div>\n            </li>\n          </ul>\n        </ng-template>\n      </div>\n    </li>\n  </ul>\n</section>\n\n<ng-template let-triggers=\"triggers\" #triggers>\n  <div class=\"ruleset-panel-category-title\">Basefacts Triggers</div>\n  <ul class=\"ruleset-panel-category-body triggers\">\n    <ng-container *ngFor=\"let trigger of (triggers | keyvalue)\">\n      <li *ngIf=\"trigger.value?.factName\">\n        <o3r-rule-key-value-pres\n          [key]=\"trigger.value.factName\"\n          [oldValue]=\"trigger.value.oldValue | o3rFallbackTo\"\n          [value]=\"trigger.value.newValue | o3rFallbackTo\"\n          [type]=\"'state'\"></o3r-rule-key-value-pres>\n      </li>\n    </ng-container>\n  </ul>\n</ng-template>\n\n<ng-template let-rules=\"rules\" let-expansionID=\"expansionID\" #rules>\n  <!-- eslint-disable-next-line @angular-eslint/template/click-events-have-key-events, @angular-eslint/template/interactive-supports-focus -- need to refactor the div to accordion from DF #1518 -->\n  <div class=\"ruleset-panel-category-title ruleset-expansion-action\"\n    (click)=\"toggleExpansion(expansionID, 'rulesOverview')\">\n    <span>Rules Overview</span>\n    <button class=\"icon\"\n            [class.icon-caret-down]=\"!expansionStatus[expansionID]?.rulesOverview\"\n            [class.icon-caret-up]=\"expansionStatus[expansionID]?.rulesOverview\">\n    </button>\n  </div>\n  <ng-container *ngIf=\"expansionStatus[expansionID]?.rulesOverview\">\n    <div *ngIf=\"rules?.length === 0\" class=\"ruleset-panel-category-body empty\">No rule</div>\n    <ul class=\"ruleset-panel-category-body\" *ngIf=\"rules?.length > 0\">\n      <li *ngFor=\"let rule of rules\">\n        <o3r-rule-tree-pres [name]=\"rule.name\"\n                            [condition]=\"rule?.rootElement?.condition\"\n                            [blockType]=\"rule?.rootElement?.blockType\"\n                            [successElements]=\"rule?.rootElement?.successElements\"\n                            [failureElements]=\"rule?.rootElement?.failureElements\">\n        </o3r-rule-tree-pres>\n      </li>\n    </ul>\n  </ng-container>\n</ng-template>\n\n<ng-template let-inputs=\"inputs\" let-runtimeInputs=\"runtimeInputs\" #inputs>\n  <div class=\"ruleset-panel-category-title\">Inputs snapshot</div>\n  <div *ngIf=\"inputs?.length === 0\" class=\"ruleset-panel-category-body empty\">No inputs</div>\n  <ul class=\"ruleset-panel-category-body\" *ngIf=\"inputs?.length > 0\">\n    <li *ngFor=\"let input of inputs\">\n      <o3r-rule-key-value-pres\n        [key]=\"input.factName\"\n        [value]=\"input.value | o3rFallbackTo\"\n        [type]=\"'state'\"></o3r-rule-key-value-pres>\n    </li>\n    <li *ngFor=\"let input of runtimeInputs\">{{input}} (scope limited to ruleset)</li>\n  </ul>\n</ng-template>\n"]}
50
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ruleset-history-pres.component.js","sourceRoot":"","sources":["../../../../../src/components/rules-engine/ruleset-history/ruleset-history-pres.component.ts","../../../../../src/components/rules-engine/ruleset-history/ruleset-history-pres.template.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,iBAAiB,EACjB,SAAS,EACT,KAAK,EACL,iBAAiB,GAClB,MAAM,eAAe,CAAC;;;;;;;;AAwBvB,MAAM,OAAO,2BAA2B;IAkBtC,YAA6B,EAAqB;QAArB,OAAE,GAAF,EAAE,CAAmB;QAjBlD;;;;;;;;WAQG;QACI,oBAAe,GAAuD,EAAE,CAAC;QAGzE,sBAAiB,GAA4B,EAAE,CAAC;QAGhD,4BAAuB,GAAG,OAAO,CAAC;IAEY,CAAC;IAEtD;;;;OAIG;IACI,eAAe,CAAC,MAAc,EAAE,QAAgB;QACrD,IAAI,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,EAAE,CAAC;YACjC,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,CAAC;QACnF,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,QAAQ,CAAC,EAAE,IAAI,EAAE,CAAC;QACtD,CAAC;QACD,IAAI,CAAC,EAAE,CAAC,aAAa,EAAE,CAAC;IAC1B,CAAC;kIAhCU,2BAA2B;sHAA3B,2BAA2B,wKC9BxC,gxRA4JA;;4FD9Ha,2BAA2B;kBAPvC,SAAS;+BACE,0BAA0B,mBAGnB,uBAAuB,CAAC,MAAM,iBAChC,iBAAiB,CAAC,IAAI;sFAe9B,iBAAiB;sBADvB,KAAK;gBAIC,uBAAuB;sBAD7B,KAAK","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  Input,\n  ViewEncapsulation,\n} from '@angular/core';\nimport type {\n  Ruleset,\n  RulesetExecutionErrorEvent,\n  RulesetExecutionEvent,\n} from '../../../engine';\n\nexport type RulesetExecutionStatus = 'Error' | 'Active' | 'Deactivated' | 'NoEffect';\n/**\n * Model of a RulesetExecution with more information for debug purpose\n */\nexport type RulesetExecutionDebug = (RulesetExecutionEvent | RulesetExecutionErrorEvent) & {\n  isActive: boolean;\n  status: RulesetExecutionStatus;\n  rulesetInformation: Ruleset | undefined;\n};\n\n@Component({\n  selector: 'o3r-ruleset-history-pres',\n  styleUrls: ['./ruleset-history-pres.style.scss'],\n  templateUrl: './ruleset-history-pres.template.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None\n})\nexport class RulesetHistoryPresComponent {\n  /**\n   * Reflects the state of each ruleset expanded elements.\n   * Each ruleset entry contains a list of subpanel that can be collapsed or expanded.\n   * Ruleset whole panel status is store the 'ruleset' entry.\n   * @example\n   * Expanded ruleset with rule overview collapsed:\n   * {'rulesetId': {'ruleset' : true, 'ruleOverview': false}}\n   * @note Collapsing a ruleset will not reset the subpanel expansion status\n   */\n  public expansionStatus: { [key: string]: { [subpanel: string]: boolean } } = {};\n\n  @Input()\n  public rulesetExecutions: RulesetExecutionDebug[] = [];\n\n  @Input()\n  public executionDurationFormat = '1.3-3';\n\n  constructor(private readonly cd: ChangeDetectorRef) {}\n\n  /**\n   * Toggle a ruleset subpanel\n   * @param ruleId\n   * @param subpanel element to collapse. 'ruleset' will toggle the whole panel but won't reset the subpanels states.\n   */\n  public toggleExpansion(ruleId: string, subpanel: string) {\n    if (this.expansionStatus[ruleId]) {\n      this.expansionStatus[ruleId][subpanel] = !this.expansionStatus[ruleId][subpanel];\n    } else {\n      this.expansionStatus[ruleId] = { [subpanel]: true };\n    }\n    this.cd.detectChanges();\n  }\n}\n","<section>\n  <ng-template #noRulesEngine>\n    <div class=\"alert alert-danger m-2\" role=\"alert\">\n      The Rules Engine is not configured on this page.\n    </div>\n  </ng-template>\n  <ul *ngIf=\"rulesetExecutions; else noRulesEngine\" class=\"rulesets\">\n    <li *ngFor=\"let execution of rulesetExecutions\" class=\"ruleset\">\n      <!-- eslint-disable-next-line @angular-eslint/template/click-events-have-key-events, @angular-eslint/template/interactive-supports-focus -- need to refactor the div to accordion from DF #1518 -->\n      <div class=\"ruleset-panel-title ruleset-expansion-action\"\n        [class.error]=\"execution.type === 'RulesetExecutionError'\"\n        (click)=\"toggleExpansion(execution.executionId, 'ruleset')\">\n        <div><span [title]=\"'This ruleset has been evaluated ' + execution.executionCounter + ' time(s)'\">{{execution.executionCounter}}</span> - {{execution.rulesetName | titlecase }}\n          <div class=\"ruleset-panel-subtitle\" *ngIf=\"execution.rulesetInformation?.linkedComponents?.or || execution.rulesetInformation?.linkedComponent\">\n            <ng-container *ngFor=\"let lc of (execution.rulesetInformation?.linkedComponents?.or || [execution.rulesetInformation.linkedComponent]); last as isLast\">\n              <div>{{lc.name}} {{lc.library}} <span *ngIf=\"!isLast\"> OR </span></div>\n            </ng-container>\n          </div>\n          <div class=\"ruleset-panel-subtitle\" *ngIf=\"execution.rulesetInformation?.validityRange as validityRange\">\n            Date range: {{validityRange.from}} - {{validityRange.to}}\n          </div>\n        </div>\n        <div class=\"ruleset-panel-title-aside\">\n          <span class=\"error capsule\" *ngIf=\"execution.status === 'Error'\">Error</span>\n          <span class=\"success capsule\" *ngIf=\"execution.status === 'Active'\">Applied</span>\n          <span class=\"inactive capsule\" *ngIf=\"execution.status === 'Deactivated'\">Deactivated</span>\n          <span class=\"warn capsule\" *ngIf=\"execution.status === 'NoEffect'\">No effect</span>\n          <span class=\"time capsule\">\n            <span>{{execution.timestamp | date: 'HH:mm:ss SSS'}}</span>\n            <span>({{execution.duration | number: executionDurationFormat}}ms)</span>\n          </span>\n          <button\n            class=\"icon\"\n            [class.icon-caret-down]=\"!expansionStatus[execution.executionId]?.ruleset\"\n            [class.icon-caret-up]=\"expansionStatus[execution.executionId]?.ruleset\">\n          </button>\n        </div>\n      </div>\n      <div class=\"ruleset-panel-description\" *ngIf=\"expansionStatus[execution.executionId]?.ruleset\">\n        <ng-container [ngTemplateOutlet]=\"rules\"\n                      [ngTemplateOutletContext]=\"{\n                        rules: execution.rulesetInformation.rules,\n                        expansionID: execution.executionId\n                      }\"></ng-container>\n        <ng-container [ngTemplateOutlet]=\"inputs\"\n                      [ngTemplateOutletContext]=\"{\n                        inputs: execution.inputFacts\n                      }\"></ng-container>\n        <ng-container *ngIf=\"execution.type === 'RulesetExecutionError'; else success\">\n          <div class=\"ruleset-panel-category-title\">Rules:</div>\n          <ul class=\"ruleset-panel-category-body rule-description\">\n            <li *ngFor=\"let ruleEvaluation of execution.rulesEvaluations; let index=index;\">\n              <ng-container>\n                <div class=\"ruleset-panel-title\" [class.error]=\"ruleEvaluation.error\">\n                  <span>{{ruleEvaluation.rule.name | titlecase}} </span>\n                  <span class=\"capsule error\" *ngIf=\"ruleEvaluation.error\">Error</span>\n                </div>\n                <div>\n                  <ng-container *ngIf=\"ruleEvaluation.error\">\n                    <span class=\"ruleset-panel-category-title\">Error:</span>\n                    <pre class=\"ruleset-panel-category-body error\">{{ruleEvaluation.error | o3rJsonOrString}}</pre>\n                  </ng-container>\n                  <ng-container [ngTemplateOutlet]=\"inputs\"\n                                [ngTemplateOutletContext]=\"{\n                                  inputs: execution.inputFacts,\n                                  runtimeInputs: execution.rulesetInformation?.rules[index]?.inputRuntimeFacts\n                                }\"></ng-container>\n                  <o3r-rule-actions-pres *ngIf=\"!ruleEvaluation.error\"\n                    [temporaryFacts]=\"ruleEvaluation.temporaryFacts\"\n                    [runtimeOutputs]=\"execution.rulesetInformation?.rules[index]?.outputRuntimeFacts\"\n                  ></o3r-rule-actions-pres>\n                </div>\n              </ng-container>\n            </li>\n          </ul>\n        </ng-container>\n        <ng-template #success>\n          <o3r-rule-actions-pres [actions]=\"execution.outputActions\"></o3r-rule-actions-pres>\n          <div class=\"ruleset-panel-category-title\">Executed Rules</div>\n          <ul class=\"rule-description ruleset-panel-category-body\">\n            <li *ngFor=\"let ruleEvaluation of execution.rulesEvaluations; let index=index;\">\n              <div class=\"ruleset-panel-title\">\n                <span>{{ruleEvaluation.rule.name | titlecase}}</span>\n                <span class=\"capsule inactive\" *ngIf=\"ruleEvaluation.cached\">Cached</span>\n                <span class=\"capsule\">({{ruleEvaluation.duration | number: executionDurationFormat}}ms)</span>\n              </div>\n              <div>\n                <ng-container [ngTemplateOutlet]=\"triggers\"\n                              [ngTemplateOutletContext]=\"{triggers: (ruleEvaluation.triggers[ruleEvaluation.rule.id])}\"></ng-container>\n                <o3r-rule-actions-pres\n                  [actions]=\"ruleEvaluation.outputActions\"\n                  [temporaryFacts]=\"ruleEvaluation.temporaryFacts\"\n                  [runtimeOutputs]=\"execution.rulesetInformation?.rules[index]?.outputRuntimeFacts\">\n                </o3r-rule-actions-pres>\n              </div>\n            </li>\n          </ul>\n        </ng-template>\n      </div>\n    </li>\n  </ul>\n</section>\n\n<ng-template let-triggers=\"triggers\" #triggers>\n  <div class=\"ruleset-panel-category-title\">Basefacts Triggers</div>\n  <ul class=\"ruleset-panel-category-body triggers\">\n    <ng-container *ngFor=\"let trigger of (triggers | keyvalue)\">\n      <li *ngIf=\"trigger.value?.factName\">\n        <o3r-rule-key-value-pres\n          [key]=\"trigger.value.factName\"\n          [oldValue]=\"trigger.value.oldValue | o3rFallbackTo\"\n          [value]=\"trigger.value.newValue | o3rFallbackTo\"\n          [type]=\"'state'\"></o3r-rule-key-value-pres>\n      </li>\n    </ng-container>\n  </ul>\n</ng-template>\n\n<ng-template let-rules=\"rules\" let-expansionID=\"expansionID\" #rules>\n  <!-- eslint-disable-next-line @angular-eslint/template/click-events-have-key-events, @angular-eslint/template/interactive-supports-focus -- need to refactor the div to accordion from DF #1518 -->\n  <div class=\"ruleset-panel-category-title ruleset-expansion-action\"\n    (click)=\"toggleExpansion(expansionID, 'rulesOverview')\">\n    <span>Rules Overview</span>\n    <button class=\"icon\"\n            [class.icon-caret-down]=\"!expansionStatus[expansionID]?.rulesOverview\"\n            [class.icon-caret-up]=\"expansionStatus[expansionID]?.rulesOverview\">\n    </button>\n  </div>\n  <ng-container *ngIf=\"expansionStatus[expansionID]?.rulesOverview\">\n    <div *ngIf=\"rules?.length === 0\" class=\"ruleset-panel-category-body empty\">No rule</div>\n    <ul class=\"ruleset-panel-category-body\" *ngIf=\"rules?.length > 0\">\n      <li *ngFor=\"let rule of rules\">\n        <o3r-rule-tree-pres [name]=\"rule.name\"\n                            [condition]=\"rule?.rootElement?.condition\"\n                            [blockType]=\"rule?.rootElement?.blockType\"\n                            [successElements]=\"rule?.rootElement?.successElements\"\n                            [failureElements]=\"rule?.rootElement?.failureElements\">\n        </o3r-rule-tree-pres>\n      </li>\n    </ul>\n  </ng-container>\n</ng-template>\n\n<ng-template let-inputs=\"inputs\" let-runtimeInputs=\"runtimeInputs\" #inputs>\n  <div class=\"ruleset-panel-category-title\">Inputs snapshot</div>\n  <div *ngIf=\"inputs?.length === 0\" class=\"ruleset-panel-category-body empty\">No inputs</div>\n  <ul class=\"ruleset-panel-category-body\" *ngIf=\"inputs?.length > 0\">\n    <li *ngFor=\"let input of inputs\">\n      <o3r-rule-key-value-pres\n        [key]=\"input.factName\"\n        [value]=\"input.value | o3rFallbackTo\"\n        [type]=\"'state'\"></o3r-rule-key-value-pres>\n    </li>\n    <li *ngFor=\"let input of runtimeInputs\">{{input}} (scope limited to ruleset)</li>\n  </ul>\n</ng-template>\n"]}
@@ -43,7 +43,7 @@ export class EngineDebugger {
43
43
  subscriber.next(list.getEntries());
44
44
  });
45
45
  performanceObserver.observe({ entryTypes: ['measure'] });
46
- return performanceObserver.disconnect();
46
+ return () => performanceObserver.disconnect();
47
47
  }).pipe(startWith([]), shareReplay(1));
48
48
  }
49
49
  async createBaseExecutionOutputObject(ruleset, executionCounter, rulesetInputFacts, runtimeFactValues, rulesetTriggers, rulesExecutions) {
@@ -144,14 +144,14 @@ export class EngineDebugger {
144
144
  }
145
145
  /**
146
146
  * Emits an 'AllActions' debug event each time the rules engine outputs the list of actions
147
- * @param actions list of outputed actions
147
+ * @param actions list of outputted actions
148
148
  */
149
149
  allActionsChange(actions) {
150
150
  const timestamp = Date.now();
151
151
  this.debugEventsSubject$.next(() => ({ timestamp, type: 'AllActions', actions }));
152
152
  }
153
153
  /**
154
- * Emits a 'RulesetExecution' debug event at the ouput of a successful ruleset execution
154
+ * Emits a 'RulesetExecution' debug event at the output of a successful ruleset execution
155
155
  * @param ruleset
156
156
  * @param executionCounter
157
157
  * @param rulesetInputFacts
@@ -165,7 +165,7 @@ export class EngineDebugger {
165
165
  this.debugEventsSubject$.next(() => this.rulesetExecution(timestamp, ruleset, executionCounter, rulesetInputFacts, allOutputActions, runtimeFactValues, rulesetTriggers, rulesExecutions));
166
166
  }
167
167
  /**
168
- * Emits a 'RulesetExecutionError' debug event at the ouput of a failing ruleset execution
168
+ * Emits a 'RulesetExecutionError' debug event at the output of a failing ruleset execution
169
169
  * @param ruleset
170
170
  * @param rulesetInputFacts
171
171
  * @param executionCounter
@@ -177,6 +177,22 @@ export class EngineDebugger {
177
177
  const timestamp = Date.now();
178
178
  this.debugEventsSubject$.next(() => this.rulesetExecutionError(timestamp, ruleset, rulesetInputFacts, executionCounter, runtimeFactValues, rulesetTriggers, rulesExecutions));
179
179
  }
180
+ /**
181
+ * Emits a 'AvailableFactsSnapshot' debug event when a fact value is updated
182
+ * @param _id
183
+ * @param factValue$
184
+ */
185
+ addAvailableFactsSnapshotEvent(_id, factValue$) {
186
+ factValue$.subscribe(async () => {
187
+ const timestamp = Date.now();
188
+ const facts = await this.getFactsSnapshot(this.registeredRuleEngine.getRegisteredFactsNames());
189
+ this.debugEventsSubject$.next(() => ({
190
+ timestamp,
191
+ type: 'AvailableFactsSnapshot',
192
+ facts
193
+ }));
194
+ });
195
+ }
180
196
  /**
181
197
  * Returns a list of fact name and value pairs
182
198
  * @param factsNames List of facts names to get the value for
@@ -192,4 +208,4 @@ export class EngineDebugger {
192
208
  return facts;
193
209
  }
194
210
  }
195
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"engine.debug.js","sourceRoot":"","sources":["../../../../src/engine/debug/engine.debug.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Observable,\n  ReplaySubject,\n} from 'rxjs';\nimport {\n  concatMap,\n  share,\n  shareReplay,\n  startWith,\n  tap,\n  withLatestFrom,\n} from 'rxjs/operators';\nimport type {\n  RulesEngine,\n} from '../engine';\nimport {\n  BaseRulesetExecution,\n  DebugEvent,\n  EvaluationReason,\n  RuleEvaluation,\n  RuleEvaluationOutput,\n  RulesetExecutionErrorEvent,\n  RulesetExecutionEvent,\n} from '../engine.interface';\nimport type {\n  Facts,\n} from '../fact';\nimport {\n  RulesetExecutor,\n} from '../ruleset-executor';\nimport type {\n  ActionBlock,\n  Ruleset,\n} from '../structure';\nimport {\n  flagCachedRules,\n  retrieveRulesetTriggers,\n} from './helpers';\n\nexport interface EngineDebuggerOptions {\n  /**\n   * Limit of events to keep in the stack before subscribing to the debugEvents$ stream.\n   * @default undefined no limit\n   */\n  eventsStackLimit?: number;\n}\n\n/**\n * Rules engine debugger object to emit debug events\n */\nexport class EngineDebugger {\n  private registeredRuleEngine?: RulesEngine;\n\n  private registeredRulesets: Pick<Ruleset, 'name' | 'id'>[] = [];\n\n  // Keep a small history in case the events$ stream from the engine is subscribed after rules engine initialization\n  private readonly debugEventsSubject$: ReplaySubject<() => (Promise<DebugEvent> | DebugEvent)>;\n\n  private performanceMeasures$!: Observable<PerformanceMeasure[]>;\n\n  /** Stream emitting a debug event when is fired; timeline is kept */\n  public readonly debugEvents$: Observable<DebugEvent>;\n\n  /** Retrieved the rules engine plugged to the debugger */\n  public get rulesEngine(): RulesEngine | undefined {\n    return this.registeredRuleEngine;\n  }\n\n  /**\n   * Instantiate a rules engine debugger\n   * @param options Options to configure the debugger\n   */\n  constructor(options?: EngineDebuggerOptions) {\n    this.debugEventsSubject$ = new ReplaySubject<() => (Promise<DebugEvent> | DebugEvent)>(options?.eventsStackLimit);\n    this.initializePerformanceObserver();\n    this.debugEvents$ = this.debugEventsSubject$.pipe(\n      withLatestFrom(this.performanceMeasures$),\n      concatMap(async ([eventFunc, performanceMeasures]) => {\n        const debugEvent: DebugEvent = await eventFunc();\n        if (debugEvent.type === 'RulesetExecution' || debugEvent.type === 'RulesetExecutionError') {\n          let rulesetDuration = 0;\n          debugEvent.rulesEvaluations.forEach((rule) => {\n            const mark = `rules-engine:${\n              this.registeredRuleEngine?.rulesEngineInstanceName || ''\n            }:${debugEvent.rulesetName}:${rule.rule.name}`;\n            const measures = performanceMeasures.filter((m) => m.name === mark);\n            const duration = measures.at(-1)?.duration || 0;\n            rule.duration = duration;\n            rulesetDuration += duration;\n          });\n          debugEvent.duration = rulesetDuration;\n        }\n        return debugEvent;\n      }),\n      tap((debugEvent) => {\n        if (debugEvent.type === 'RulesetExecution') {\n          this.rulesEngine?.logger?.debug?.(`${debugEvent.rulesetName} has been triggered and resulted in ${JSON.stringify(debugEvent.outputActions)}`);\n        }\n      }),\n      share()\n    );\n  }\n\n  private initializePerformanceObserver() {\n    this.performanceMeasures$ = new Observable<PerformanceMeasure[]>((subscriber) => {\n      const performanceObserver = new PerformanceObserver((list) => {\n        subscriber.next(list.getEntries() as PerformanceMeasure[]);\n      });\n      performanceObserver.observe({ entryTypes: ['measure'] });\n      return performanceObserver.disconnect();\n    }).pipe(startWith([]), shareReplay(1));\n  }\n\n  private async createBaseExecutionOutputObject(ruleset: Ruleset, executionCounter: number,\n    rulesetInputFacts: string[], runtimeFactValues: Record<string, Facts>,\n    rulesetTriggers: Record<string, Record<string, EvaluationReason>>, rulesExecutions: RuleEvaluation[]) {\n    const inputFacts = await this.getFactsSnapshot(rulesetInputFacts);\n\n    const baseRulesetOutputExecution: BaseRulesetExecution = {\n      executionCounter,\n      executionId: `${ruleset.id}-${executionCounter}`,\n      rulesetId: ruleset.id,\n      rulesetName: ruleset.name,\n      inputFacts,\n      triggers: rulesetTriggers,\n      rulesEvaluations: flagCachedRules(rulesExecutions?.sort((a, b) => a.timestamp - b.timestamp) || [], rulesetTriggers),\n      temporaryFacts: runtimeFactValues\n    };\n    return baseRulesetOutputExecution;\n  }\n\n  private async rulesetExecution(\n    timestamp: number,\n    ruleset: Ruleset,\n    executionCounter: number,\n    rulesetInputFacts: string[],\n    allOutputActions: ActionBlock[],\n    runtimeFactValues: Record<string, Facts>,\n    rulesetTriggers: Record<string, Record<string, EvaluationReason>>,\n    rulesExecutions: RuleEvaluation[]) {\n    const baseRulesetOutputExecution = await this.createBaseExecutionOutputObject(ruleset, executionCounter, rulesetInputFacts, runtimeFactValues, rulesetTriggers, rulesExecutions);\n\n    const rulesetOutputExecution: RulesetExecutionEvent = {\n      timestamp,\n      type: 'RulesetExecution',\n      outputActions: allOutputActions,\n      ...baseRulesetOutputExecution\n    };\n\n    return rulesetOutputExecution;\n  }\n\n  private async rulesetExecutionError(\n    timestamp: number,\n    ruleset: Ruleset,\n    rulesetInputFacts: string[],\n    executionCounter: number,\n    runtimeFactValues: Record<string, Facts>,\n    rulesetTriggers: Record<string, Record<string, EvaluationReason>>,\n    rulesExecutions: RuleEvaluation[]\n  ) {\n    const baseRulesetOutputExecution = await this.createBaseExecutionOutputObject(ruleset, executionCounter, rulesetInputFacts, runtimeFactValues, rulesetTriggers, rulesExecutions);\n\n    const rulesExecWithErrors = rulesExecutions.filter((ex) => !!ex && !!ex.error);\n    const rulesetOutputExecutionSkip: RulesetExecutionErrorEvent = {\n      timestamp,\n      type: 'RulesetExecutionError',\n      rulesCausingTheError: rulesExecWithErrors.map((e) => e.rule) || [],\n      errors: rulesExecWithErrors.map((e) => e.error),\n      ...baseRulesetOutputExecution\n    };\n    return rulesetOutputExecutionSkip;\n  }\n\n  /**\n   * Plug the debugger to a Rule Engine\n   * @param rulesEngine\n   */\n  public registerRuleEngine(rulesEngine: RulesEngine) {\n    this.registeredRuleEngine = rulesEngine;\n  }\n\n  /**\n   * Handle ruleset execution debug info\n   * @param currRes\n   * @param prevRes\n   * @param allExecutionsValid\n   * @param rulesetInputFacts\n   * @param runtimeFactValues\n   * @param executionCounter\n   * @param ruleset\n   */\n  public handleDebugRulesetExecutionInfo(\n    currRes: RuleEvaluationOutput[],\n    prevRes: RuleEvaluationOutput[] | undefined,\n    allExecutionsValid: boolean,\n    rulesetInputFacts: string[],\n    runtimeFactValues: Record<string, Facts>,\n    executionCounter: number,\n    ruleset: Ruleset) {\n    const rulesetTriggers = retrieveRulesetTriggers(currRes, prevRes);\n    const rulesetOutputExecution = currRes.map((r) => r.evaluation!);\n\n    if (!allExecutionsValid) {\n      this.addRulesetExecutionErrorEvent(ruleset, rulesetInputFacts, executionCounter, runtimeFactValues, rulesetTriggers, rulesetOutputExecution);\n    }\n\n    return {\n      executionCounter,\n      rulesetOutputExecution,\n      allExecutionsValid,\n      rulesetTriggers\n    };\n  }\n\n  /**\n   * Emits an 'AvailableRulesets' debug event when rulesets are registered to the rules engine\n   * @param rulesets\n   */\n  public addAvailableRulesets(rulesets: Ruleset[]) {\n    const timestamp = Date.now();\n    this.registeredRulesets = [...this.registeredRulesets, ...rulesets.map((r) => ({ name: r.name, id: r.id }))];\n    this.debugEventsSubject$.next(() => ({\n      timestamp,\n      type: 'AvailableRulesets',\n      availableRulesets: this.registeredRulesets\n    }));\n  }\n\n  /**\n   * Computes and emits an 'ActiveRulesets' debug event when the active rulesets are changing\n   * @param ruleSetExecutorMap map off all rulesets executors\n   * @param restrictiveRuleSets ids of the rulesets to activate; if not provided all registered rulesets will be considered as active\n   */\n  public activeRulesetsChange(ruleSetExecutorMap: Record<string, RulesetExecutor>, restrictiveRuleSets?: string[]) {\n    const timestamp = Date.now();\n    const rulesets = Object.keys(ruleSetExecutorMap).map((rulesetId) => ruleSetExecutorMap[rulesetId].engineRuleset);\n    const activeRulesets = restrictiveRuleSets\n      ? Object.values(rulesets).filter((ruleSet) => restrictiveRuleSets.includes(ruleSet.id))\n      : Object.values(rulesets);\n\n    this.debugEventsSubject$.next(() => ({\n      timestamp,\n      type: 'ActiveRulesets',\n      rulesets: activeRulesets.map((a) => ({ name: ruleSetExecutorMap[a.id].ruleset.name, id: ruleSetExecutorMap[a.id].ruleset.id }))\n    }));\n  }\n\n  /**\n   * Emits an 'AllActions' debug event each time the rules engine outputs the list of actions\n   * @param actions list of outputed actions\n   */\n  public allActionsChange(actions: ActionBlock[]) {\n    const timestamp = Date.now();\n    this.debugEventsSubject$.next(() => ({ timestamp, type: 'AllActions', actions }));\n  }\n\n  /**\n   * Emits a 'RulesetExecution' debug event at the ouput of a successful ruleset execution\n   * @param ruleset\n   * @param executionCounter\n   * @param rulesetInputFacts\n   * @param allOutputActions\n   * @param runtimeFactValues\n   * @param rulesetTriggers\n   * @param rulesExecutions\n   */\n  public addRulesetExecutionEvent(ruleset: Ruleset, executionCounter: number,\n    rulesetInputFacts: string[], allOutputActions: ActionBlock[], runtimeFactValues: Record<string, Facts>,\n    rulesetTriggers: Record<string, Record<string, EvaluationReason>>, rulesExecutions: RuleEvaluation[]) {\n    const timestamp = Date.now();\n    this.debugEventsSubject$.next(() => this.rulesetExecution(timestamp, ruleset, executionCounter,\n      rulesetInputFacts, allOutputActions, runtimeFactValues, rulesetTriggers, rulesExecutions));\n  }\n\n  /**\n   * Emits a 'RulesetExecutionError' debug event at the ouput of a failing ruleset execution\n   * @param ruleset\n   * @param rulesetInputFacts\n   * @param executionCounter\n   * @param runtimeFactValues\n   * @param rulesetTriggers\n   * @param rulesExecutions\n   */\n  public addRulesetExecutionErrorEvent(\n    ruleset: Ruleset,\n    rulesetInputFacts: string[],\n    executionCounter: number,\n    runtimeFactValues: Record<string, Facts>,\n    rulesetTriggers: Record<string, Record<string, EvaluationReason>>,\n    rulesExecutions: RuleEvaluation[]\n  ) {\n    const timestamp = Date.now();\n    this.debugEventsSubject$.next(() => this.rulesetExecutionError(timestamp, ruleset, rulesetInputFacts, executionCounter, runtimeFactValues, rulesetTriggers, rulesExecutions));\n  }\n\n  /**\n   * Returns a list of fact name and value pairs\n   * @param factsNames List of facts names to get the value for\n   */\n  public async getFactsSnapshot(factsNames: string[]) {\n    const facts: { factName: string; value: any }[] = [];\n    if (!this.registeredRuleEngine) {\n      throw new Error('Rule engine not plugged to the debugger');\n    }\n    for (const factName of factsNames) {\n      facts.push({ factName, value: await this.registeredRuleEngine.retrieveFactValue(factName) });\n    }\n    return facts;\n  }\n}\n"]}
211
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"engine.debug.js","sourceRoot":"","sources":["../../../../src/engine/debug/engine.debug.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Observable,\n  ReplaySubject,\n} from 'rxjs';\nimport {\n  concatMap,\n  share,\n  shareReplay,\n  startWith,\n  tap,\n  withLatestFrom,\n} from 'rxjs/operators';\nimport type {\n  RulesEngine,\n} from '../engine';\nimport {\n  BaseRulesetExecution,\n  DebugEvent,\n  EvaluationReason,\n  RuleEvaluation,\n  RuleEvaluationOutput,\n  RulesetExecutionErrorEvent,\n  RulesetExecutionEvent,\n} from '../engine.interface';\nimport type {\n  Facts,\n} from '../fact';\nimport {\n  RulesetExecutor,\n} from '../ruleset-executor';\nimport type {\n  ActionBlock,\n  Ruleset,\n} from '../structure';\nimport {\n  flagCachedRules,\n  retrieveRulesetTriggers,\n} from './helpers';\n\nexport interface EngineDebuggerOptions {\n  /**\n   * Limit of events to keep in the stack before subscribing to the debugEvents$ stream.\n   * @default undefined no limit\n   */\n  eventsStackLimit?: number;\n}\n\n/**\n * Rules engine debugger object to emit debug events\n */\nexport class EngineDebugger {\n  private registeredRuleEngine?: RulesEngine;\n\n  private registeredRulesets: Pick<Ruleset, 'name' | 'id'>[] = [];\n\n  // Keep a small history in case the events$ stream from the engine is subscribed after rules engine initialization\n  private readonly debugEventsSubject$: ReplaySubject<() => (Promise<DebugEvent> | DebugEvent)>;\n\n  private performanceMeasures$!: Observable<PerformanceMeasure[]>;\n\n  /** Stream emitting a debug event when is fired; timeline is kept */\n  public readonly debugEvents$: Observable<DebugEvent>;\n\n  /** Retrieved the rules engine plugged to the debugger */\n  public get rulesEngine(): RulesEngine | undefined {\n    return this.registeredRuleEngine;\n  }\n\n  /**\n   * Instantiate a rules engine debugger\n   * @param options Options to configure the debugger\n   */\n  constructor(options?: EngineDebuggerOptions) {\n    this.debugEventsSubject$ = new ReplaySubject<() => (Promise<DebugEvent> | DebugEvent)>(options?.eventsStackLimit);\n    this.initializePerformanceObserver();\n    this.debugEvents$ = this.debugEventsSubject$.pipe(\n      withLatestFrom(this.performanceMeasures$),\n      concatMap(async ([eventFunc, performanceMeasures]) => {\n        const debugEvent: DebugEvent = await eventFunc();\n        if (debugEvent.type === 'RulesetExecution' || debugEvent.type === 'RulesetExecutionError') {\n          let rulesetDuration = 0;\n          debugEvent.rulesEvaluations.forEach((rule) => {\n            const mark = `rules-engine:${\n              this.registeredRuleEngine?.rulesEngineInstanceName || ''\n            }:${debugEvent.rulesetName}:${rule.rule.name}`;\n            const measures = performanceMeasures.filter((m) => m.name === mark);\n            const duration = measures.at(-1)?.duration || 0;\n            rule.duration = duration;\n            rulesetDuration += duration;\n          });\n          debugEvent.duration = rulesetDuration;\n        }\n        return debugEvent;\n      }),\n      tap((debugEvent) => {\n        if (debugEvent.type === 'RulesetExecution') {\n          this.rulesEngine?.logger?.debug?.(`${debugEvent.rulesetName} has been triggered and resulted in ${JSON.stringify(debugEvent.outputActions)}`);\n        }\n      }),\n      share()\n    );\n  }\n\n  private initializePerformanceObserver() {\n    this.performanceMeasures$ = new Observable<PerformanceMeasure[]>((subscriber) => {\n      const performanceObserver = new PerformanceObserver((list) => {\n        subscriber.next(list.getEntries() as PerformanceMeasure[]);\n      });\n      performanceObserver.observe({ entryTypes: ['measure'] });\n      return () => performanceObserver.disconnect();\n    }).pipe(startWith([]), shareReplay(1));\n  }\n\n  private async createBaseExecutionOutputObject(ruleset: Ruleset, executionCounter: number,\n    rulesetInputFacts: string[], runtimeFactValues: Record<string, Facts>,\n    rulesetTriggers: Record<string, Record<string, EvaluationReason>>, rulesExecutions: RuleEvaluation[]) {\n    const inputFacts = await this.getFactsSnapshot(rulesetInputFacts);\n\n    const baseRulesetOutputExecution: BaseRulesetExecution = {\n      executionCounter,\n      executionId: `${ruleset.id}-${executionCounter}`,\n      rulesetId: ruleset.id,\n      rulesetName: ruleset.name,\n      inputFacts,\n      triggers: rulesetTriggers,\n      rulesEvaluations: flagCachedRules(rulesExecutions?.sort((a, b) => a.timestamp - b.timestamp) || [], rulesetTriggers),\n      temporaryFacts: runtimeFactValues\n    };\n    return baseRulesetOutputExecution;\n  }\n\n  private async rulesetExecution(\n    timestamp: number,\n    ruleset: Ruleset,\n    executionCounter: number,\n    rulesetInputFacts: string[],\n    allOutputActions: ActionBlock[],\n    runtimeFactValues: Record<string, Facts>,\n    rulesetTriggers: Record<string, Record<string, EvaluationReason>>,\n    rulesExecutions: RuleEvaluation[]) {\n    const baseRulesetOutputExecution = await this.createBaseExecutionOutputObject(ruleset, executionCounter, rulesetInputFacts, runtimeFactValues, rulesetTriggers, rulesExecutions);\n\n    const rulesetOutputExecution: RulesetExecutionEvent = {\n      timestamp,\n      type: 'RulesetExecution',\n      outputActions: allOutputActions,\n      ...baseRulesetOutputExecution\n    };\n\n    return rulesetOutputExecution;\n  }\n\n  private async rulesetExecutionError(\n    timestamp: number,\n    ruleset: Ruleset,\n    rulesetInputFacts: string[],\n    executionCounter: number,\n    runtimeFactValues: Record<string, Facts>,\n    rulesetTriggers: Record<string, Record<string, EvaluationReason>>,\n    rulesExecutions: RuleEvaluation[]\n  ) {\n    const baseRulesetOutputExecution = await this.createBaseExecutionOutputObject(ruleset, executionCounter, rulesetInputFacts, runtimeFactValues, rulesetTriggers, rulesExecutions);\n\n    const rulesExecWithErrors = rulesExecutions.filter((ex) => !!ex && !!ex.error);\n    const rulesetOutputExecutionSkip: RulesetExecutionErrorEvent = {\n      timestamp,\n      type: 'RulesetExecutionError',\n      rulesCausingTheError: rulesExecWithErrors.map((e) => e.rule) || [],\n      errors: rulesExecWithErrors.map((e) => e.error),\n      ...baseRulesetOutputExecution\n    };\n    return rulesetOutputExecutionSkip;\n  }\n\n  /**\n   * Plug the debugger to a Rule Engine\n   * @param rulesEngine\n   */\n  public registerRuleEngine(rulesEngine: RulesEngine) {\n    this.registeredRuleEngine = rulesEngine;\n  }\n\n  /**\n   * Handle ruleset execution debug info\n   * @param currRes\n   * @param prevRes\n   * @param allExecutionsValid\n   * @param rulesetInputFacts\n   * @param runtimeFactValues\n   * @param executionCounter\n   * @param ruleset\n   */\n  public handleDebugRulesetExecutionInfo(\n    currRes: RuleEvaluationOutput[],\n    prevRes: RuleEvaluationOutput[] | undefined,\n    allExecutionsValid: boolean,\n    rulesetInputFacts: string[],\n    runtimeFactValues: Record<string, Facts>,\n    executionCounter: number,\n    ruleset: Ruleset) {\n    const rulesetTriggers = retrieveRulesetTriggers(currRes, prevRes);\n    const rulesetOutputExecution = currRes.map((r) => r.evaluation!);\n\n    if (!allExecutionsValid) {\n      this.addRulesetExecutionErrorEvent(ruleset, rulesetInputFacts, executionCounter, runtimeFactValues, rulesetTriggers, rulesetOutputExecution);\n    }\n\n    return {\n      executionCounter,\n      rulesetOutputExecution,\n      allExecutionsValid,\n      rulesetTriggers\n    };\n  }\n\n  /**\n   * Emits an 'AvailableRulesets' debug event when rulesets are registered to the rules engine\n   * @param rulesets\n   */\n  public addAvailableRulesets(rulesets: Ruleset[]) {\n    const timestamp = Date.now();\n    this.registeredRulesets = [...this.registeredRulesets, ...rulesets.map((r) => ({ name: r.name, id: r.id }))];\n    this.debugEventsSubject$.next(() => ({\n      timestamp,\n      type: 'AvailableRulesets',\n      availableRulesets: this.registeredRulesets\n    }));\n  }\n\n  /**\n   * Computes and emits an 'ActiveRulesets' debug event when the active rulesets are changing\n   * @param ruleSetExecutorMap map off all rulesets executors\n   * @param restrictiveRuleSets ids of the rulesets to activate; if not provided all registered rulesets will be considered as active\n   */\n  public activeRulesetsChange(ruleSetExecutorMap: Record<string, RulesetExecutor>, restrictiveRuleSets?: string[]) {\n    const timestamp = Date.now();\n    const rulesets = Object.keys(ruleSetExecutorMap).map((rulesetId) => ruleSetExecutorMap[rulesetId].engineRuleset);\n    const activeRulesets = restrictiveRuleSets\n      ? Object.values(rulesets).filter((ruleSet) => restrictiveRuleSets.includes(ruleSet.id))\n      : Object.values(rulesets);\n\n    this.debugEventsSubject$.next(() => ({\n      timestamp,\n      type: 'ActiveRulesets',\n      rulesets: activeRulesets.map((a) => ({ name: ruleSetExecutorMap[a.id].ruleset.name, id: ruleSetExecutorMap[a.id].ruleset.id }))\n    }));\n  }\n\n  /**\n   * Emits an 'AllActions' debug event each time the rules engine outputs the list of actions\n   * @param actions list of outputted actions\n   */\n  public allActionsChange(actions: ActionBlock[]) {\n    const timestamp = Date.now();\n    this.debugEventsSubject$.next(() => ({ timestamp, type: 'AllActions', actions }));\n  }\n\n  /**\n   * Emits a 'RulesetExecution' debug event at the output of a successful ruleset execution\n   * @param ruleset\n   * @param executionCounter\n   * @param rulesetInputFacts\n   * @param allOutputActions\n   * @param runtimeFactValues\n   * @param rulesetTriggers\n   * @param rulesExecutions\n   */\n  public addRulesetExecutionEvent(ruleset: Ruleset, executionCounter: number,\n    rulesetInputFacts: string[], allOutputActions: ActionBlock[], runtimeFactValues: Record<string, Facts>,\n    rulesetTriggers: Record<string, Record<string, EvaluationReason>>, rulesExecutions: RuleEvaluation[]) {\n    const timestamp = Date.now();\n    this.debugEventsSubject$.next(() => this.rulesetExecution(timestamp, ruleset, executionCounter,\n      rulesetInputFacts, allOutputActions, runtimeFactValues, rulesetTriggers, rulesExecutions));\n  }\n\n  /**\n   * Emits a 'RulesetExecutionError' debug event at the output of a failing ruleset execution\n   * @param ruleset\n   * @param rulesetInputFacts\n   * @param executionCounter\n   * @param runtimeFactValues\n   * @param rulesetTriggers\n   * @param rulesExecutions\n   */\n  public addRulesetExecutionErrorEvent(\n    ruleset: Ruleset,\n    rulesetInputFacts: string[],\n    executionCounter: number,\n    runtimeFactValues: Record<string, Facts>,\n    rulesetTriggers: Record<string, Record<string, EvaluationReason>>,\n    rulesExecutions: RuleEvaluation[]\n  ) {\n    const timestamp = Date.now();\n    this.debugEventsSubject$.next(() => this.rulesetExecutionError(timestamp, ruleset, rulesetInputFacts, executionCounter, runtimeFactValues, rulesetTriggers, rulesExecutions));\n  }\n\n  /**\n   * Emits a 'AvailableFactsSnapshot' debug event when a fact value is updated\n   * @param _id\n   * @param factValue$\n   */\n  public addAvailableFactsSnapshotEvent(\n    _id: string,\n    factValue$: Observable<any>\n  ) {\n    factValue$.subscribe(async () => {\n      const timestamp = Date.now();\n      const facts = await this.getFactsSnapshot(this.registeredRuleEngine!.getRegisteredFactsNames());\n      this.debugEventsSubject$.next(() => ({\n        timestamp,\n        type: 'AvailableFactsSnapshot',\n        facts\n      }));\n    });\n  }\n\n  /**\n   * Returns a list of fact name and value pairs\n   * @param factsNames List of facts names to get the value for\n   */\n  public async getFactsSnapshot(factsNames: string[]) {\n    const facts: { factName: string; value: any }[] = [];\n    if (!this.registeredRuleEngine) {\n      throw new Error('Rule engine not plugged to the debugger');\n    }\n    for (const factName of factsNames) {\n      facts.push({ factName, value: await this.registeredRuleEngine.retrieveFactValue(factName) });\n    }\n    return facts;\n  }\n}\n"]}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"engine.interface.js","sourceRoot":"","sources":["../../../src/engine/engine.interface.ts"],"names":[],"mappings":"","sourcesContent":["import type {\n  ItemIdentifier,\n  Logger,\n} from '@o3r/core';\nimport {\n  BehaviorSubject,\n  Observable,\n} from 'rxjs';\nimport type {\n  EngineDebugger,\n} from './debug/engine.debug';\nimport type {\n  Fact,\n  Facts,\n} from './fact';\nimport type {\n  Operator,\n} from './operator';\nimport {\n  ActionBlock,\n  Rule,\n  Ruleset,\n} from './structure';\n\n/** Performance object supporting NodeJs Performance and Web Performance reporting  */\nexport type CrossPlatformPerformance = {\n  /** @see Performance.mark */\n  mark: (...x: Parameters<Performance['mark']>) => ReturnType<Performance['mark']> | void;\n  /** @see Performance.measure */\n  measure: (measureName: string, startOrMeasureOptions?: string, endMark?: string) => ReturnType<Performance['measure']> | void;\n};\n\n/** Fact stream object to handle fact reference change */\nexport interface FactObject<T> {\n  /** Subject of fact stream */\n  subject: BehaviorSubject<Observable<T> | undefined>;\n  /** Stream of the fact value */\n  value$: Observable<T | undefined>;\n}\n\n/** Rule Engine constructor options */\nexport interface RulesEngineOptions {\n  /** List of facts */\n  facts?: Fact<Facts>[];\n\n  /** List of rules */\n  rules?: Ruleset[];\n\n  /** List of custom operators */\n  operators?: Operator<any, any>[];\n\n  /** Delay before fact stream defaulting value */\n  factDefaultDelay?: number;\n\n  /**\n   * Skip the rule and fact circular dependency checks\n   * Turn to true to increase the speed of the upsertion of a rule\n   */\n  skipCircularDependencyChecks?: boolean;\n\n  /**\n   * Provide debugger instance to the rules engine\n   */\n  debugger?: EngineDebugger;\n\n  /**\n   * Instance of the performance reporter to use for performance measurements.\n   * @default window.performance on browser only, undefined on node\n   */\n  performance?: CrossPlatformPerformance;\n\n  /**\n   * Name of the rules engine instance\n   * @default RulesEngine\n   */\n  rulesEngineInstanceName?: string;\n\n  /**\n   * Client to log the warning and error message\n   */\n  logger?: Logger;\n}\n\n/** Rule as stored in the rules engine */\nexport interface EngineRule extends Rule {\n  /** stream of the rule conditions result */\n  result$: BehaviorSubject<ActionBlock[]>;\n}\n\n/** Rule as stored in the rules engine */\nexport interface EngineRuleset {\n  /** Optional date range where the ruleset will be executed, it supports a dateString or a timestamp as number, more info on javascript Date() documentation */\n  validityRange?: { from?: string | number; to?: string | number };\n  /**\n   * Components linked to the ruleset. If present the ruleset will not be active by default.\n   * 'or' condition: If at least one component has subscribed, the ruleset will become active.\n   * If present, the {@link linkedComponent} property will not be taken into consideration\n   */\n  linkedComponents?: { or: ItemIdentifier[] };\n  /**\n   * Component linked to the ruleset, if set it will disable the ruleset execution per default, waiting to a subscription\n   * @deprecated It will be removed in v12, use {@link linkedComponents} instead\n   */\n  linkedComponent?: ItemIdentifier;\n  /** Unique id of the ruleset*/\n  id: string;\n  /** Stores the result of each rules from the ruleset */\n  rulesResultsSubject$: Observable<ActionBlock[]>;\n}\n/** Timestamp of a rules engine output event */\nexport interface TimedEvent {\n  /** Timestamp value when the event occurs */\n  timestamp: number;\n  /** Duration of the execution */\n  duration?: number;\n}\n\n/** Fact change triggering the evaluation of a rule/execution of a ruleset */\nexport interface EvaluationReason {\n  /** Name of the fact that changed */\n  factName: string;\n  /** New value of the fact */\n  newValue?: Facts;\n  /** Old value of the fact */\n  oldValue?: Facts;\n}\n\n/** Result object resulted at the end of a rule evaluation */\nexport interface RuleEvaluation extends TimedEvent {\n  /** Identifier of the evaluation (ruleset name + rule name) */\n  id: string;\n  /** Evaluated rule identifier */\n  rule: Pick<Rule, 'id' | 'name'>;\n  /** Actions outputed by the rule evaluation */\n  outputActions: ActionBlock[] | undefined;\n  /** Map containing the facts changes triggering the rule evaluation */\n  triggers: Record<string, Record<string, EvaluationReason>>;\n  /** Error object in case of rule evaluation failure */\n  error?: any;\n  /** Runtime facts with values at the end of rule evaluation  */\n  temporaryFacts?: Record<string, Facts>;\n  /** Flag to notify if the rules evaluation comes from an old ruleset execution */\n  cached?: boolean;\n}\n\n/** Wrapped rule evaluation output */\nexport interface RuleEvaluationOutput {\n  /** Actions emitted at the end of rule evaluation */\n  actions: ActionBlock[] | undefined;\n  /** Rule evaluation output object */\n  evaluation?: RuleEvaluation;\n  /** Error object emitted at the end of rule evaluation, if any */\n  error?: any;\n}\n\n/** Base obeject resulted at the end of a ruleset execution */\nexport interface BaseRulesetExecution {\n  /** Id of the ruleset execution */\n  executionId: string;\n  /** Id of the ruleset which was executed */\n  rulesetId: string;\n  /** Name of the executed ruleset */\n  rulesetName: string;\n  /** Counter of executions for the ruleset */\n  executionCounter: number;\n  /** All input facts affecting the ruleset */\n  inputFacts: { factName: string; value: Facts }[];\n  /** Runtime facts used accros the ruleset */\n  temporaryFacts?: Record<string, Facts>;\n  /** Facts changes that triggered the execution of the ruleset */\n  triggers: Record<string, Record<string, EvaluationReason>>;\n  /** List of evaluated rules accros ruleset execution */\n  rulesEvaluations: RuleEvaluation[];\n}\n\n/** Debug event emitted in case of successful ruleset execution */\nexport interface RulesetExecutionEvent extends BaseRulesetExecution, TimedEvent {\n  /** Event type */\n  type: 'RulesetExecution';\n  /** List of the actions emitted at the end of ruleset execution */\n  outputActions: ActionBlock[];\n}\n\n/** Debug event emitted in case of ruleset execution failure */\nexport interface RulesetExecutionErrorEvent extends BaseRulesetExecution, TimedEvent {\n  /** Event type */\n  type: 'RulesetExecutionError';\n  /** List of rules causing the execution error*/\n  rulesCausingTheError: Pick<Rule, 'name' | 'id'>[];\n  /** List of outputted errors */\n  errors: any[];\n}\n\n/** Debug event emitted when active rulesets are changing */\nexport interface ActiveRulesetsEvent extends TimedEvent {\n  /** Event type */\n  type: 'ActiveRulesets';\n  /** List of active rulesets */\n  rulesets: Pick<Ruleset, 'name' | 'id'>[];\n}\n\n/** Debug event emitted each time the Rules Engine outputs a list of actions */\nexport interface AllActionsEvent extends TimedEvent {\n  /** event type */\n  type: 'AllActions';\n  /** List of emitted actions */\n  actions: ActionBlock[];\n}\n\n/**  Debug event emitted when rulesets are registered to the rules engine */\nexport interface AvailableRulesets extends TimedEvent {\n  /** Event type */\n  type: 'AvailableRulesets';\n  /** Registered rulesets list */\n  availableRulesets: Pick<Ruleset, 'name' | 'id'>[];\n}\n\n/** Type of possible debug events emited by Rules Engine */\nexport type DebugEvent = RulesetExecutionEvent | RulesetExecutionErrorEvent | ActiveRulesetsEvent | AllActionsEvent | AvailableRulesets;\n"]}
2
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"engine.interface.js","sourceRoot":"","sources":["../../../src/engine/engine.interface.ts"],"names":[],"mappings":"","sourcesContent":["import type {\n  ItemIdentifier,\n  Logger,\n} from '@o3r/core';\nimport {\n  BehaviorSubject,\n  Observable,\n} from 'rxjs';\nimport type {\n  EngineDebugger,\n} from './debug/engine.debug';\nimport type {\n  Fact,\n  Facts,\n} from './fact';\nimport type {\n  Operator,\n} from './operator';\nimport {\n  ActionBlock,\n  Rule,\n  Ruleset,\n} from './structure';\n\n/** Performance object supporting NodeJs Performance and Web Performance reporting  */\nexport type CrossPlatformPerformance = {\n  /** @see Performance.mark */\n  mark: (...x: Parameters<Performance['mark']>) => ReturnType<Performance['mark']> | void;\n  /** @see Performance.measure */\n  measure: (measureName: string, startOrMeasureOptions?: string, endMark?: string) => ReturnType<Performance['measure']> | void;\n};\n\n/** Fact stream object to handle fact reference change */\nexport interface FactObject<T> {\n  /** Subject of fact stream */\n  subject: BehaviorSubject<Observable<T> | undefined>;\n  /** Stream of the fact value */\n  value$: Observable<T | undefined>;\n}\n\n/** Rule Engine constructor options */\nexport interface RulesEngineOptions {\n  /** List of facts */\n  facts?: Fact<Facts>[];\n\n  /** List of rules */\n  rules?: Ruleset[];\n\n  /** List of custom operators */\n  operators?: Operator<any, any>[];\n\n  /** Delay before fact stream defaulting value */\n  factDefaultDelay?: number;\n\n  /**\n   * Skip the rule and fact circular dependency checks\n   * Turn to true to increase the speed of the upsert of a rule\n   */\n  skipCircularDependencyChecks?: boolean;\n\n  /**\n   * Provide debugger instance to the rules engine\n   */\n  debugger?: EngineDebugger;\n\n  /**\n   * Instance of the performance reporter to use for performance measurements.\n   * @default window.performance on browser only, undefined on node\n   */\n  performance?: CrossPlatformPerformance;\n\n  /**\n   * Name of the rules engine instance\n   * @default RulesEngine\n   */\n  rulesEngineInstanceName?: string;\n\n  /**\n   * Client to log the warning and error message\n   */\n  logger?: Logger;\n}\n\n/** Rule as stored in the rules engine */\nexport interface EngineRule extends Rule {\n  /** stream of the rule conditions result */\n  result$: BehaviorSubject<ActionBlock[]>;\n}\n\n/** Rule as stored in the rules engine */\nexport interface EngineRuleset {\n  /** Optional date range where the ruleset will be executed, it supports a dateString or a timestamp as number, more info on javascript Date() documentation */\n  validityRange?: { from?: string | number; to?: string | number };\n  /**\n   * Components linked to the ruleset. If present the ruleset will not be active by default.\n   * 'or' condition: If at least one component has subscribed, the ruleset will become active.\n   * If present, the {@link linkedComponent} property will not be taken into consideration\n   */\n  linkedComponents?: { or: ItemIdentifier[] };\n  /**\n   * Component linked to the ruleset, if set it will disable the ruleset execution per default, waiting to a subscription\n   * @deprecated It will be removed in v12, use {@link linkedComponents} instead\n   */\n  linkedComponent?: ItemIdentifier;\n  /** Unique id of the ruleset*/\n  id: string;\n  /** Stores the result of each rules from the ruleset */\n  rulesResultsSubject$: Observable<ActionBlock[]>;\n}\n/** Timestamp of a rules engine output event */\nexport interface TimedEvent {\n  /** Timestamp value when the event occurs */\n  timestamp: number;\n  /** Duration of the execution */\n  duration?: number;\n}\n\n/** Fact change triggering the evaluation of a rule/execution of a ruleset */\nexport interface EvaluationReason {\n  /** Name of the fact that changed */\n  factName: string;\n  /** New value of the fact */\n  newValue?: Facts;\n  /** Old value of the fact */\n  oldValue?: Facts;\n}\n\n/** Result object resulted at the end of a rule evaluation */\nexport interface RuleEvaluation extends TimedEvent {\n  /** Identifier of the evaluation (ruleset name + rule name) */\n  id: string;\n  /** Evaluated rule identifier */\n  rule: Pick<Rule, 'id' | 'name'>;\n  /** Actions outputted by the rule evaluation */\n  outputActions: ActionBlock[] | undefined;\n  /** Map containing the facts changes triggering the rule evaluation */\n  triggers: Record<string, Record<string, EvaluationReason>>;\n  /** Error object in case of rule evaluation failure */\n  error?: any;\n  /** Runtime facts with values at the end of rule evaluation  */\n  temporaryFacts?: Record<string, Facts>;\n  /** Flag to notify if the rules evaluation comes from an old ruleset execution */\n  cached?: boolean;\n}\n\n/** Wrapped rule evaluation output */\nexport interface RuleEvaluationOutput {\n  /** Actions emitted at the end of rule evaluation */\n  actions: ActionBlock[] | undefined;\n  /** Rule evaluation output object */\n  evaluation?: RuleEvaluation;\n  /** Error object emitted at the end of rule evaluation, if any */\n  error?: any;\n}\n\n/** Base object resulted at the end of a ruleset execution */\nexport interface BaseRulesetExecution {\n  /** Id of the ruleset execution */\n  executionId: string;\n  /** Id of the ruleset which was executed */\n  rulesetId: string;\n  /** Name of the executed ruleset */\n  rulesetName: string;\n  /** Counter of executions for the ruleset */\n  executionCounter: number;\n  /** All input facts affecting the ruleset */\n  inputFacts: { factName: string; value: Facts }[];\n  /** Runtime facts used accros the ruleset */\n  temporaryFacts?: Record<string, Facts>;\n  /** Facts changes that triggered the execution of the ruleset */\n  triggers: Record<string, Record<string, EvaluationReason>>;\n  /** List of evaluated rules accros ruleset execution */\n  rulesEvaluations: RuleEvaluation[];\n}\n\n/** Debug event emitted in case of successful ruleset execution */\nexport interface RulesetExecutionEvent extends BaseRulesetExecution, TimedEvent {\n  /** Event type */\n  type: 'RulesetExecution';\n  /** List of the actions emitted at the end of ruleset execution */\n  outputActions: ActionBlock[];\n}\n\n/** Debug event emitted in case of ruleset execution failure */\nexport interface RulesetExecutionErrorEvent extends BaseRulesetExecution, TimedEvent {\n  /** Event type */\n  type: 'RulesetExecutionError';\n  /** List of rules causing the execution error*/\n  rulesCausingTheError: Pick<Rule, 'name' | 'id'>[];\n  /** List of outputted errors */\n  errors: any[];\n}\n\n/** Debug event emitted when active rulesets are changing */\nexport interface ActiveRulesetsEvent extends TimedEvent {\n  /** Event type */\n  type: 'ActiveRulesets';\n  /** List of active rulesets */\n  rulesets: Pick<Ruleset, 'name' | 'id'>[];\n}\n\n/** Debug event emitted each time the Rules Engine outputs a list of actions */\nexport interface AllActionsEvent extends TimedEvent {\n  /** event type */\n  type: 'AllActions';\n  /** List of emitted actions */\n  actions: ActionBlock[];\n}\n\n/**  Debug event emitted when rulesets are registered to the rules engine */\nexport interface AvailableRulesets extends TimedEvent {\n  /** Event type */\n  type: 'AvailableRulesets';\n  /** Registered rulesets list */\n  availableRulesets: Pick<Ruleset, 'name' | 'id'>[];\n}\n\n/**  Debug event emitted when facts are updated */\nexport interface AvailableFactsSnapshot extends TimedEvent {\n  /** Event type */\n  type: 'AvailableFactsSnapshot';\n  /** List of all facts */\n  facts: { factName: string; value: Facts }[];\n}\n\n/** Type of possible debug events emitted by Rules Engine */\nexport type DebugEvent = RulesetExecutionEvent | RulesetExecutionErrorEvent | ActiveRulesetsEvent | AllActionsEvent | AvailableRulesets | AvailableFactsSnapshot;\n"]}
@@ -96,7 +96,10 @@ export class RulesEngine {
96
96
  * @param facts fact list to add / update
97
97
  */
98
98
  upsertFacts(facts) {
99
- (Array.isArray(facts) ? facts : [facts]).forEach(({ id, value$ }) => this.retrieveOrCreateFactStream(id, value$));
99
+ (Array.isArray(facts) ? facts : [facts]).forEach(({ id, value$ }) => {
100
+ this.engineDebug?.addAvailableFactsSnapshotEvent(id, value$);
101
+ this.retrieveOrCreateFactStream(id, value$);
102
+ });
100
103
  }
101
104
  /**
102
105
  * Update or insert rule in rules engine
@@ -131,4 +134,4 @@ export class RulesEngine {
131
134
  return Object.keys(this.factMap);
132
135
  }
133
136
  }
134
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"engine.js","sourceRoot":"","sources":["../../../src/engine/engine.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  Logger,\n} from '@o3r/logger';\nimport {\n  BehaviorSubject,\n  firstValueFrom,\n  merge,\n  Observable,\n  of,\n} from 'rxjs';\nimport {\n  delay,\n  distinctUntilChanged,\n  shareReplay,\n  switchMap,\n  takeUntil,\n  tap,\n} from 'rxjs/operators';\nimport {\n  EngineDebugger,\n} from './debug/engine.debug';\nimport {\n  FactObject,\n  RulesEngineOptions,\n} from './engine.interface';\nimport type {\n  Fact,\n  Facts,\n} from './fact/index';\nimport {\n  filterRulesetsEventStream,\n} from './helpers/filter-ruleset-event.operator';\nimport {\n  Operator,\n  operatorList,\n  UnaryOperator,\n} from './operator/index';\nimport {\n  RulesetExecutor,\n} from './ruleset-executor';\nimport {\n  ActionBlock,\n  Ruleset,\n} from './structure';\n\n/** Rules engine */\nexport class RulesEngine {\n  /** Map of registered fact stream, this map is mutated by the ruleset executors */\n  private readonly factMap: Record<string, FactObject<any>> = {};\n\n  /** Subject containing the rulesets and the results stream*/\n  private readonly rulesetMapSubject = new BehaviorSubject<Record<string, RulesetExecutor>>({});\n\n  /** Map of available operators */\n  public operators: Record<string, Operator<unknown, unknown>>;\n\n  /** List of events for the current state of the rules engine */\n  public readonly events$: Observable<ActionBlock[]>;\n\n  /** Delay before fact stream defaulting value */\n  public factDefaultDelay?: number;\n\n  /**\n   * Instance of engine debug object; Undefined if debugMode is not active\n   */\n  public readonly engineDebug?: EngineDebugger;\n\n  /** Name of the rules engine instance */\n  public readonly rulesEngineInstanceName: string;\n\n  /**\n   * Performance reporter to use for performance measurements.\n   * @default window.performance on browser only, undefined on node\n   */\n  public readonly performance;\n\n  /**\n   * Log the engine errors\n   */\n  public readonly logger?: Logger;\n  /**\n   * Flag to check if the run is in debug mode or not\n   */\n  public get debugMode(): boolean {\n    return !!this.engineDebug;\n  }\n\n  /**\n   * Rules engine\n   * @param options rules engine options\n   */\n  constructor(options?: RulesEngineOptions) {\n    this.performance = options?.performance || (typeof window === 'undefined' ? undefined : window.performance);\n    this.engineDebug = options?.debugger;\n    this.engineDebug?.registerRuleEngine(this);\n    this.logger = options?.logger;\n    this.rulesEngineInstanceName = options?.rulesEngineInstanceName || 'RulesEngine';\n    this.factDefaultDelay = options?.factDefaultDelay;\n    // Load default operators\n    this.operators = operatorList.reduce<Record<string, Operator<any, any>>>((acc, operator) => {\n      acc[operator.name] = operator;\n      return acc;\n    }, {});\n    this.events$ = this.rulesetMapSubject.pipe(\n      this.prepareActionsStream(),\n      this.handleActionsStreamOutput()\n    );\n    if (options?.facts) {\n      this.upsertFacts(options.facts);\n    }\n    if (options?.rules) {\n      this.upsertRulesets(options.rules);\n    }\n    if (options?.operators) {\n      this.upsertOperators(options.operators);\n    }\n  }\n\n  /**\n   * Attach debug events to actions stream if debug engine is activated\n   */\n  private handleActionsStreamOutput<T extends ActionBlock = ActionBlock>(): (actionsStream$: Observable<T[]>) => Observable<T[]> {\n    return (actionsStream$: Observable<T[]>) => this.engineDebug ? actionsStream$.pipe(tap((allActions) => this.engineDebug!.allActionsChange(allActions))) : actionsStream$;\n  }\n\n  /**\n   * Create the actions stream event based on provided active rulesets ids; Handle debug too\n   * @param ruleSets\n   */\n  private prepareActionsStream<T extends ActionBlock = ActionBlock>(ruleSets?: string[]): (rulesetMapSubject$: Observable<Record<string, RulesetExecutor>>) => Observable<T[]> {\n    return (rulesetMapSubject$: Observable<Record<string, RulesetExecutor>>) => (this.engineDebug\n      ? rulesetMapSubject$.pipe(\n        tap((ruleSetExecutorMap) => this.engineDebug!.activeRulesetsChange(ruleSetExecutorMap, ruleSets)),\n        filterRulesetsEventStream(ruleSets))\n      : rulesetMapSubject$.pipe(filterRulesetsEventStream(ruleSets))) as Observable<T[]>;\n  }\n\n  /**\n   * Create or retrieve a fact stream\n   * The fact stream created will be registered in the engine\n   * @param id ID of the fact to retrieve\n   * @param factValue$ Value stream for the fact\n   */\n  public retrieveOrCreateFactStream<T = Facts>(id: string, factValue$?: Observable<T>): Observable<T | undefined> {\n    // trick to emit undefined if the observable is not immediately emitting (to not bloc execution)\n    const obs$ = factValue$\n      ? merge(factValue$, of(undefined).pipe(delay(this.factDefaultDelay || 0), takeUntil(factValue$)))\n      : factValue$;\n    const factObj = this.factMap[id];\n    if (factObj) {\n      if (factValue$) {\n        factObj.subject.next(obs$);\n      }\n      return factObj.value$;\n    }\n\n    const subject = new BehaviorSubject<Observable<T | undefined> | undefined>(obs$);\n    const value$ = subject.pipe(\n      switchMap((value) => value || of(undefined)),\n      distinctUntilChanged(),\n      shareReplay(1)\n    );\n\n    this.factMap[id] = {\n      subject,\n      value$\n    };\n\n    return value$;\n  }\n\n  /**\n   * Retrieve the promise of the latest value of a fact.\n   * Return undefined if the fact is not defined.\n   * @param id ID of the fact to retrieve\n   */\n  public retrieveFactValue<T = unknown>(id: string): Promise<T | undefined> | undefined {\n    return this.factMap[id].value$ && firstValueFrom(this.retrieveOrCreateFactStream<T>(id), { defaultValue: undefined });\n  }\n\n  /**\n   * Update or insert fact in rules engine\n   * @param facts fact list to add / update\n   */\n  public upsertFacts<T = unknown>(facts: Fact<T> | Fact<T>[]) {\n    (Array.isArray(facts) ? facts : [facts]).forEach(({ id, value$ }) =>\n      this.retrieveOrCreateFactStream(id, value$)\n    );\n  }\n\n  /**\n   * Update or insert rule in rules engine\n   * @param rulesets\n   */\n  public upsertRulesets(rulesets: Ruleset[]) {\n    this.engineDebug?.addAvailableRulesets(rulesets);\n\n    this.rulesetMapSubject.next(\n      rulesets.reduce((accRuleset, ruleset) => {\n        accRuleset[ruleset.id] = new RulesetExecutor(ruleset, this);\n        return accRuleset;\n      }, { ...this.rulesetMapSubject.value })\n    );\n  }\n\n  /**\n   * Update or insert operator in rules engine\n   * @param operators operator list to add / update\n   */\n  public upsertOperators(operators: (Operator<any, any> | UnaryOperator<any>)[]) {\n    this.operators = operators.reduce((acc, operator) => {\n      acc[operator.name] = operator;\n      return acc;\n    }, { ...this.operators });\n  }\n\n  /**\n   * Operator to apply on a stream of rulesets ids\n   * Returns a stream of actions outputted by the rules engine, corresponding to the rulesetsIds\n   */\n  public getEventStream<T extends ActionBlock = ActionBlock>(): (rulesetsIds$: Observable<string[] | undefined>) => Observable<T[]> {\n    return (rulesetsIds$: Observable<string[] | undefined>) =>\n      rulesetsIds$.pipe(\n        switchMap((ruleSets) => this.rulesetMapSubject.pipe(this.prepareActionsStream<T>(ruleSets))),\n        this.handleActionsStreamOutput()\n      );\n  }\n\n  /** Get the list of registered facts names */\n  public getRegisteredFactsNames() {\n    return Object.keys(this.factMap);\n  }\n}\n"]}
137
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"engine.js","sourceRoot":"","sources":["../../../src/engine/engine.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  Logger,\n} from '@o3r/logger';\nimport {\n  BehaviorSubject,\n  firstValueFrom,\n  merge,\n  Observable,\n  of,\n} from 'rxjs';\nimport {\n  delay,\n  distinctUntilChanged,\n  shareReplay,\n  switchMap,\n  takeUntil,\n  tap,\n} from 'rxjs/operators';\nimport {\n  EngineDebugger,\n} from './debug/engine.debug';\nimport {\n  FactObject,\n  RulesEngineOptions,\n} from './engine.interface';\nimport type {\n  Fact,\n  Facts,\n} from './fact/index';\nimport {\n  filterRulesetsEventStream,\n} from './helpers/filter-ruleset-event.operator';\nimport {\n  Operator,\n  operatorList,\n  UnaryOperator,\n} from './operator/index';\nimport {\n  RulesetExecutor,\n} from './ruleset-executor';\nimport {\n  ActionBlock,\n  Ruleset,\n} from './structure';\n\n/** Rules engine */\nexport class RulesEngine {\n  /** Map of registered fact stream, this map is mutated by the ruleset executors */\n  private readonly factMap: Record<string, FactObject<any>> = {};\n\n  /** Subject containing the rulesets and the results stream*/\n  private readonly rulesetMapSubject = new BehaviorSubject<Record<string, RulesetExecutor>>({});\n\n  /** Map of available operators */\n  public operators: Record<string, Operator<unknown, unknown>>;\n\n  /** List of events for the current state of the rules engine */\n  public readonly events$: Observable<ActionBlock[]>;\n\n  /** Delay before fact stream defaulting value */\n  public factDefaultDelay?: number;\n\n  /**\n   * Instance of engine debug object; Undefined if debugMode is not active\n   */\n  public readonly engineDebug?: EngineDebugger;\n\n  /** Name of the rules engine instance */\n  public readonly rulesEngineInstanceName: string;\n\n  /**\n   * Performance reporter to use for performance measurements.\n   * @default window.performance on browser only, undefined on node\n   */\n  public readonly performance;\n\n  /**\n   * Log the engine errors\n   */\n  public readonly logger?: Logger;\n  /**\n   * Flag to check if the run is in debug mode or not\n   */\n  public get debugMode(): boolean {\n    return !!this.engineDebug;\n  }\n\n  /**\n   * Rules engine\n   * @param options rules engine options\n   */\n  constructor(options?: RulesEngineOptions) {\n    this.performance = options?.performance || (typeof window === 'undefined' ? undefined : window.performance);\n    this.engineDebug = options?.debugger;\n    this.engineDebug?.registerRuleEngine(this);\n    this.logger = options?.logger;\n    this.rulesEngineInstanceName = options?.rulesEngineInstanceName || 'RulesEngine';\n    this.factDefaultDelay = options?.factDefaultDelay;\n    // Load default operators\n    this.operators = operatorList.reduce<Record<string, Operator<any, any>>>((acc, operator) => {\n      acc[operator.name] = operator;\n      return acc;\n    }, {});\n    this.events$ = this.rulesetMapSubject.pipe(\n      this.prepareActionsStream(),\n      this.handleActionsStreamOutput()\n    );\n    if (options?.facts) {\n      this.upsertFacts(options.facts);\n    }\n    if (options?.rules) {\n      this.upsertRulesets(options.rules);\n    }\n    if (options?.operators) {\n      this.upsertOperators(options.operators);\n    }\n  }\n\n  /**\n   * Attach debug events to actions stream if debug engine is activated\n   */\n  private handleActionsStreamOutput<T extends ActionBlock = ActionBlock>(): (actionsStream$: Observable<T[]>) => Observable<T[]> {\n    return (actionsStream$: Observable<T[]>) => this.engineDebug ? actionsStream$.pipe(tap((allActions) => this.engineDebug!.allActionsChange(allActions))) : actionsStream$;\n  }\n\n  /**\n   * Create the actions stream event based on provided active rulesets ids; Handle debug too\n   * @param ruleSets\n   */\n  private prepareActionsStream<T extends ActionBlock = ActionBlock>(ruleSets?: string[]): (rulesetMapSubject$: Observable<Record<string, RulesetExecutor>>) => Observable<T[]> {\n    return (rulesetMapSubject$: Observable<Record<string, RulesetExecutor>>) => (this.engineDebug\n      ? rulesetMapSubject$.pipe(\n        tap((ruleSetExecutorMap) => this.engineDebug!.activeRulesetsChange(ruleSetExecutorMap, ruleSets)),\n        filterRulesetsEventStream(ruleSets))\n      : rulesetMapSubject$.pipe(filterRulesetsEventStream(ruleSets))) as Observable<T[]>;\n  }\n\n  /**\n   * Create or retrieve a fact stream\n   * The fact stream created will be registered in the engine\n   * @param id ID of the fact to retrieve\n   * @param factValue$ Value stream for the fact\n   */\n  public retrieveOrCreateFactStream<T = Facts>(id: string, factValue$?: Observable<T>): Observable<T | undefined> {\n    // trick to emit undefined if the observable is not immediately emitting (to not bloc execution)\n    const obs$ = factValue$\n      ? merge(factValue$, of(undefined).pipe(delay(this.factDefaultDelay || 0), takeUntil(factValue$)))\n      : factValue$;\n    const factObj = this.factMap[id];\n    if (factObj) {\n      if (factValue$) {\n        factObj.subject.next(obs$);\n      }\n      return factObj.value$;\n    }\n\n    const subject = new BehaviorSubject<Observable<T | undefined> | undefined>(obs$);\n    const value$ = subject.pipe(\n      switchMap((value) => value || of(undefined)),\n      distinctUntilChanged(),\n      shareReplay(1)\n    );\n\n    this.factMap[id] = {\n      subject,\n      value$\n    };\n\n    return value$;\n  }\n\n  /**\n   * Retrieve the promise of the latest value of a fact.\n   * Return undefined if the fact is not defined.\n   * @param id ID of the fact to retrieve\n   */\n  public retrieveFactValue<T = unknown>(id: string): Promise<T | undefined> | undefined {\n    return this.factMap[id].value$ && firstValueFrom(this.retrieveOrCreateFactStream<T>(id), { defaultValue: undefined });\n  }\n\n  /**\n   * Update or insert fact in rules engine\n   * @param facts fact list to add / update\n   */\n  public upsertFacts<T = unknown>(facts: Fact<T> | Fact<T>[]) {\n    (Array.isArray(facts) ? facts : [facts]).forEach(({ id, value$ }) => {\n      this.engineDebug?.addAvailableFactsSnapshotEvent(id, value$);\n      this.retrieveOrCreateFactStream(id, value$);\n    });\n  }\n\n  /**\n   * Update or insert rule in rules engine\n   * @param rulesets\n   */\n  public upsertRulesets(rulesets: Ruleset[]) {\n    this.engineDebug?.addAvailableRulesets(rulesets);\n\n    this.rulesetMapSubject.next(\n      rulesets.reduce((accRuleset, ruleset) => {\n        accRuleset[ruleset.id] = new RulesetExecutor(ruleset, this);\n        return accRuleset;\n      }, { ...this.rulesetMapSubject.value })\n    );\n  }\n\n  /**\n   * Update or insert operator in rules engine\n   * @param operators operator list to add / update\n   */\n  public upsertOperators(operators: (Operator<any, any> | UnaryOperator<any>)[]) {\n    this.operators = operators.reduce((acc, operator) => {\n      acc[operator.name] = operator;\n      return acc;\n    }, { ...this.operators });\n  }\n\n  /**\n   * Operator to apply on a stream of rulesets ids\n   * Returns a stream of actions outputted by the rules engine, corresponding to the rulesetsIds\n   */\n  public getEventStream<T extends ActionBlock = ActionBlock>(): (rulesetsIds$: Observable<string[] | undefined>) => Observable<T[]> {\n    return (rulesetsIds$: Observable<string[] | undefined>) =>\n      rulesetsIds$.pipe(\n        switchMap((ruleSets) => this.rulesetMapSubject.pipe(this.prepareActionsStream<T>(ruleSets))),\n        this.handleActionsStreamOutput()\n      );\n  }\n\n  /** Get the list of registered facts names */\n  public getRegisteredFactsNames() {\n    return Object.keys(this.factMap);\n  }\n}\n"]}