raain-app 1.6.21 → 1.6.24
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +17 -1
- package/cumulative-selector/cumulative-selector.component.d.ts +45 -0
- package/esm2020/cumulative-selector/cumulative-selector.component.mjs +199 -0
- package/esm2020/index.mjs +2 -1
- package/esm2020/profile.service.mjs +55 -1
- package/esm2020/raain-compare-stack/raain-compare-stack.component.mjs +3 -3
- package/esm2020/raain-details/raain-details.component.mjs +73 -7
- package/esm2020/shared.module.mjs +8 -3
- package/esm2020/tools/CompareManager.mjs +31 -8
- package/esm2020/tools/RefreshManager.mjs +2 -2
- package/fesm2015/raain-app.mjs +1622 -1267
- package/fesm2015/raain-app.mjs.map +1 -1
- package/fesm2020/raain-app.mjs +1515 -1175
- package/fesm2020/raain-app.mjs.map +1 -1
- package/index.d.ts +1 -0
- package/package.json +2 -2
- package/profile.service.d.ts +14 -1
- package/raain-details/raain-details.component.d.ts +7 -1
- package/shared.module.d.ts +6 -5
- package/tools/CompareManager.d.ts +8 -2
package/CHANGELOG.md
CHANGED
|
@@ -7,7 +7,23 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|
|
7
7
|
|
|
8
8
|
## [Unreleased]
|
|
9
9
|
|
|
10
|
-
## [1.6.
|
|
10
|
+
## [1.6.23] - 2026-01-19
|
|
11
|
+
|
|
12
|
+
### Added
|
|
13
|
+
|
|
14
|
+
- New `CumulativeSelectorComponent` for selecting and creating cumulative periods
|
|
15
|
+
- Lists available cumulative periods with window duration display
|
|
16
|
+
- Validates base cumulative coverage before allowing new period creation
|
|
17
|
+
- Polls for computation completion with progress feedback
|
|
18
|
+
- ProfileService: `getCumulativePeriods()` to list available cumulative periods
|
|
19
|
+
- ProfileService: `createCumulativePeriod()` to trigger cumulative computation
|
|
20
|
+
|
|
21
|
+
### Changed
|
|
22
|
+
|
|
23
|
+
- Cumulative toggle now shows selector popup when enabling (instead of direct toggle)
|
|
24
|
+
- RaainDetails: cumulative period selection flow with `onCumulativePeriodSelected()`
|
|
25
|
+
|
|
26
|
+
## [1.6.22] - 2025-12-16
|
|
11
27
|
|
|
12
28
|
### Added
|
|
13
29
|
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { ChangeDetectorRef, EventEmitter, OnInit } from '@angular/core';
|
|
2
|
+
import { ProfileService } from '../profile.service';
|
|
3
|
+
import { CumulativePeriod } from 'raain-model';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
export interface CumulativeSelection {
|
|
6
|
+
periodBegin: Date;
|
|
7
|
+
periodEnd: Date;
|
|
8
|
+
windowInMinutes: number;
|
|
9
|
+
}
|
|
10
|
+
export declare class CumulativeSelectorComponent implements OnInit {
|
|
11
|
+
private profileService;
|
|
12
|
+
private cdr;
|
|
13
|
+
rainId: string;
|
|
14
|
+
currentPeriodBegin: Date;
|
|
15
|
+
currentPeriodEnd: Date;
|
|
16
|
+
provider: string;
|
|
17
|
+
timeStepInMinutes: number;
|
|
18
|
+
isAdmin: boolean;
|
|
19
|
+
periodSelected: EventEmitter<CumulativeSelection>;
|
|
20
|
+
cancelled: EventEmitter<void>;
|
|
21
|
+
availablePeriods: CumulativePeriod[];
|
|
22
|
+
baseCumulatives: CumulativePeriod;
|
|
23
|
+
loading: boolean;
|
|
24
|
+
creating: boolean;
|
|
25
|
+
creationProgress: string;
|
|
26
|
+
errorMessage: string;
|
|
27
|
+
coveragePercent: number;
|
|
28
|
+
canCreateNew: boolean;
|
|
29
|
+
currentWindowMinutes: number;
|
|
30
|
+
private readonly POLL_TIMEOUT_MS;
|
|
31
|
+
private readonly POLL_INTERVAL_MS;
|
|
32
|
+
constructor(profileService: ProfileService, cdr: ChangeDetectorRef);
|
|
33
|
+
ngOnInit(): Promise<void>;
|
|
34
|
+
loadAvailablePeriods(): Promise<void>;
|
|
35
|
+
calculateCoverage(): void;
|
|
36
|
+
selectPeriod(period: CumulativePeriod): void;
|
|
37
|
+
createNewPeriod(): Promise<void>;
|
|
38
|
+
private pollForCompletion;
|
|
39
|
+
private sleep;
|
|
40
|
+
cancel(): void;
|
|
41
|
+
formatPeriod(period: CumulativePeriod): string;
|
|
42
|
+
formatWindow(minutes: number): string;
|
|
43
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<CumulativeSelectorComponent, never>;
|
|
44
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<CumulativeSelectorComponent, "cumulative-selector", never, { "rainId": "rainId"; "currentPeriodBegin": "currentPeriodBegin"; "currentPeriodEnd": "currentPeriodEnd"; "provider": "provider"; "timeStepInMinutes": "timeStepInMinutes"; "isAdmin": "isAdmin"; }, { "periodSelected": "periodSelected"; "cancelled": "cancelled"; }, never, never, false, never>;
|
|
45
|
+
}
|
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "../profile.service";
|
|
4
|
+
import * as i2 from "@angular/common";
|
|
5
|
+
import * as i3 from "@ionic/angular";
|
|
6
|
+
export class CumulativeSelectorComponent {
|
|
7
|
+
constructor(profileService, cdr) {
|
|
8
|
+
this.profileService = profileService;
|
|
9
|
+
this.cdr = cdr;
|
|
10
|
+
this.timeStepInMinutes = 5;
|
|
11
|
+
this.isAdmin = false;
|
|
12
|
+
this.periodSelected = new EventEmitter();
|
|
13
|
+
this.cancelled = new EventEmitter();
|
|
14
|
+
this.availablePeriods = [];
|
|
15
|
+
this.baseCumulatives = null;
|
|
16
|
+
this.loading = true;
|
|
17
|
+
this.creating = false;
|
|
18
|
+
this.creationProgress = '';
|
|
19
|
+
this.errorMessage = '';
|
|
20
|
+
this.coveragePercent = 0;
|
|
21
|
+
this.canCreateNew = false;
|
|
22
|
+
this.currentWindowMinutes = 0;
|
|
23
|
+
this.POLL_TIMEOUT_MS = 900000; // 900 seconds
|
|
24
|
+
this.POLL_INTERVAL_MS = 3000;
|
|
25
|
+
}
|
|
26
|
+
async ngOnInit() {
|
|
27
|
+
this.currentWindowMinutes = Math.round((this.currentPeriodEnd.getTime() - this.currentPeriodBegin.getTime()) / 60000);
|
|
28
|
+
await this.loadAvailablePeriods();
|
|
29
|
+
}
|
|
30
|
+
async loadAvailablePeriods() {
|
|
31
|
+
this.loading = true;
|
|
32
|
+
this.errorMessage = '';
|
|
33
|
+
this.cdr.markForCheck();
|
|
34
|
+
try {
|
|
35
|
+
// Fetch all cumulative periods (admin sees all, non-admin sees only customer-launched)
|
|
36
|
+
const response = await this.profileService.getCumulativePeriods(this.rainId, {
|
|
37
|
+
provider: this.provider,
|
|
38
|
+
forced: this.isAdmin,
|
|
39
|
+
});
|
|
40
|
+
// Filter for existing custom cumulatives (window > 0)
|
|
41
|
+
this.availablePeriods = response.periods.filter((p) => p.windowInMinutes > 0);
|
|
42
|
+
// Get base cumulatives (window = 0) to check coverage
|
|
43
|
+
this.baseCumulatives = response.periods.find((p) => p.windowInMinutes === 0);
|
|
44
|
+
this.calculateCoverage();
|
|
45
|
+
}
|
|
46
|
+
catch (e) {
|
|
47
|
+
this.errorMessage = 'Failed to load cumulative periods';
|
|
48
|
+
console.error('Error loading cumulative periods:', e);
|
|
49
|
+
}
|
|
50
|
+
this.loading = false;
|
|
51
|
+
this.cdr.markForCheck();
|
|
52
|
+
}
|
|
53
|
+
calculateCoverage() {
|
|
54
|
+
if (!this.baseCumulatives) {
|
|
55
|
+
this.coveragePercent = 0;
|
|
56
|
+
this.canCreateNew = false;
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
const baseBegin = new Date(this.baseCumulatives.periodBegin);
|
|
60
|
+
const baseEnd = new Date(this.baseCumulatives.periodEnd);
|
|
61
|
+
// Check if current period is within base coverage
|
|
62
|
+
const currentInRange = this.currentPeriodBegin >= baseBegin && this.currentPeriodEnd <= baseEnd;
|
|
63
|
+
if (!currentInRange) {
|
|
64
|
+
this.coveragePercent = 0;
|
|
65
|
+
this.canCreateNew = false;
|
|
66
|
+
return;
|
|
67
|
+
}
|
|
68
|
+
// Calculate expected number of base cumulatives needed
|
|
69
|
+
const expectedCount = Math.ceil(this.currentWindowMinutes / this.timeStepInMinutes);
|
|
70
|
+
// Check if enough base cumulatives exist
|
|
71
|
+
// We assume coverage is complete if count >= expected (simplified check)
|
|
72
|
+
if (this.baseCumulatives.count >= expectedCount) {
|
|
73
|
+
this.coveragePercent = 100;
|
|
74
|
+
this.canCreateNew = true;
|
|
75
|
+
}
|
|
76
|
+
else {
|
|
77
|
+
this.coveragePercent = Math.round((this.baseCumulatives.count / expectedCount) * 100);
|
|
78
|
+
this.canCreateNew = this.coveragePercent >= 100;
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
selectPeriod(period) {
|
|
82
|
+
this.periodSelected.emit({
|
|
83
|
+
periodBegin: new Date(period.periodBegin),
|
|
84
|
+
periodEnd: new Date(period.periodEnd),
|
|
85
|
+
windowInMinutes: period.windowInMinutes,
|
|
86
|
+
});
|
|
87
|
+
}
|
|
88
|
+
async createNewPeriod() {
|
|
89
|
+
if (!this.canCreateNew || this.creating) {
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
this.creating = true;
|
|
93
|
+
this.creationProgress = 'Starting cumulative computation...';
|
|
94
|
+
this.errorMessage = '';
|
|
95
|
+
this.cdr.markForCheck();
|
|
96
|
+
try {
|
|
97
|
+
// Trigger cumulative creation
|
|
98
|
+
const result = await this.profileService.createCumulativePeriod(this.rainId, {
|
|
99
|
+
periodBegin: this.currentPeriodBegin,
|
|
100
|
+
periodEnd: this.currentPeriodEnd,
|
|
101
|
+
provider: this.provider,
|
|
102
|
+
timeStepInMinutes: this.timeStepInMinutes,
|
|
103
|
+
});
|
|
104
|
+
if (!result) {
|
|
105
|
+
throw new Error('Failed to trigger cumulative computation');
|
|
106
|
+
}
|
|
107
|
+
this.creationProgress = `Jobs queued. Polling for completion...`;
|
|
108
|
+
this.cdr.markForCheck();
|
|
109
|
+
// Poll for completion
|
|
110
|
+
const success = await this.pollForCompletion();
|
|
111
|
+
if (success) {
|
|
112
|
+
this.periodSelected.emit({
|
|
113
|
+
periodBegin: this.currentPeriodBegin,
|
|
114
|
+
periodEnd: this.currentPeriodEnd,
|
|
115
|
+
windowInMinutes: this.currentWindowMinutes,
|
|
116
|
+
});
|
|
117
|
+
}
|
|
118
|
+
else {
|
|
119
|
+
this.errorMessage = 'Timeout waiting for cumulative computation';
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
catch (e) {
|
|
123
|
+
this.errorMessage = `Error: ${e.message || 'Unknown error'}`;
|
|
124
|
+
console.error('Error creating cumulative:', e);
|
|
125
|
+
}
|
|
126
|
+
this.creating = false;
|
|
127
|
+
this.cdr.markForCheck();
|
|
128
|
+
}
|
|
129
|
+
async pollForCompletion() {
|
|
130
|
+
const startTime = Date.now();
|
|
131
|
+
while (Date.now() - startTime < this.POLL_TIMEOUT_MS) {
|
|
132
|
+
try {
|
|
133
|
+
const progress = await this.profileService.getRainProgress(this.rainId);
|
|
134
|
+
if (progress === 0) {
|
|
135
|
+
// Queue is empty, check if cumulative exists (admin sees all cumulatives)
|
|
136
|
+
const response = await this.profileService.getCumulativePeriods(this.rainId, {
|
|
137
|
+
provider: this.provider,
|
|
138
|
+
windowInMinutes: this.currentWindowMinutes,
|
|
139
|
+
forced: this.isAdmin,
|
|
140
|
+
});
|
|
141
|
+
const found = response.periods.find((p) => p.windowInMinutes === this.currentWindowMinutes);
|
|
142
|
+
if (found) {
|
|
143
|
+
return true;
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
const elapsed = Math.round((Date.now() - startTime) / 1000);
|
|
147
|
+
this.creationProgress = `Computing... (${elapsed}s, queue: ${progress})`;
|
|
148
|
+
this.cdr.markForCheck();
|
|
149
|
+
await this.sleep(this.POLL_INTERVAL_MS);
|
|
150
|
+
}
|
|
151
|
+
catch (e) {
|
|
152
|
+
console.warn('Poll error:', e);
|
|
153
|
+
await this.sleep(this.POLL_INTERVAL_MS);
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
return false;
|
|
157
|
+
}
|
|
158
|
+
sleep(ms) {
|
|
159
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
160
|
+
}
|
|
161
|
+
cancel() {
|
|
162
|
+
this.cancelled.emit();
|
|
163
|
+
}
|
|
164
|
+
formatPeriod(period) {
|
|
165
|
+
const begin = new Date(period.periodBegin);
|
|
166
|
+
const end = new Date(period.periodEnd);
|
|
167
|
+
return `${begin.toLocaleString()} → ${end.toLocaleString()}`;
|
|
168
|
+
}
|
|
169
|
+
formatWindow(minutes) {
|
|
170
|
+
if (minutes < 60) {
|
|
171
|
+
return `${minutes} min`;
|
|
172
|
+
}
|
|
173
|
+
const hours = minutes / 60;
|
|
174
|
+
return hours === 1 ? '1 hour' : `${hours} hours`;
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
CumulativeSelectorComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CumulativeSelectorComponent, deps: [{ token: i1.ProfileService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
178
|
+
CumulativeSelectorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: CumulativeSelectorComponent, selector: "cumulative-selector", inputs: { rainId: "rainId", currentPeriodBegin: "currentPeriodBegin", currentPeriodEnd: "currentPeriodEnd", provider: "provider", timeStepInMinutes: "timeStepInMinutes", isAdmin: "isAdmin" }, outputs: { periodSelected: "periodSelected", cancelled: "cancelled" }, ngImport: i0, template: "<div class=\"cumulative-selector-overlay\">\n <div class=\"cumulative-selector-modal\">\n <div class=\"modal-header\">\n <h2>Select Cumulative Period</h2>\n <ion-button fill=\"clear\" (click)=\"cancel()\">\n <ion-icon name=\"close\"></ion-icon>\n </ion-button>\n </div>\n\n <div class=\"modal-content\">\n <!-- Loading state -->\n <div *ngIf=\"loading\" class=\"loading-state\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <p>Loading available periods...</p>\n </div>\n\n <!-- Error message -->\n <div *ngIf=\"errorMessage\" class=\"error-message\">\n <ion-icon name=\"warning-outline\"></ion-icon>\n <span>{{ errorMessage }}</span>\n </div>\n\n <!-- Available periods list -->\n <div *ngIf=\"!loading && availablePeriods.length > 0\" class=\"periods-section\">\n <h3>Available Cumulative Periods</h3>\n <ion-list>\n <ion-item *ngFor=\"let period of availablePeriods\"\n button\n (click)=\"selectPeriod(period)\"\n [disabled]=\"creating\">\n <ion-icon name=\"layers-outline\" slot=\"start\"></ion-icon>\n <ion-label>\n <h2>{{ formatWindow(period.windowInMinutes) }}</h2>\n <p>{{ formatPeriod(period) }}</p>\n <p class=\"count-info\">{{ period.count }} cumulative(s)</p>\n </ion-label>\n <ion-icon name=\"chevron-forward\" slot=\"end\"></ion-icon>\n </ion-item>\n </ion-list>\n </div>\n\n <!-- No periods available -->\n <div *ngIf=\"!loading && availablePeriods.length === 0\" class=\"no-periods\">\n <ion-icon name=\"information-circle-outline\"></ion-icon>\n <p>No cumulative periods available yet.</p>\n </div>\n\n <!-- Create new section (admin only) -->\n <div *ngIf=\"!loading && isAdmin\" class=\"create-section\">\n <h3>Create New Cumulative</h3>\n <div class=\"create-info\">\n <p>\n <strong>Period:</strong>\n {{ currentPeriodBegin?.toLocaleString() }} \u2192 {{ currentPeriodEnd?.toLocaleString() }}\n </p>\n <p>\n <strong>Window:</strong> {{ formatWindow(currentWindowMinutes) }}\n </p>\n <p *ngIf=\"baseCumulatives\" class=\"coverage-info\"\n [class.coverage-ok]=\"coveragePercent >= 100\"\n [class.coverage-warn]=\"coveragePercent > 0 && coveragePercent < 100\"\n [class.coverage-error]=\"coveragePercent === 0\">\n <ion-icon [name]=\"coveragePercent >= 100 ? 'checkmark-circle' : 'alert-circle'\"></ion-icon>\n Base data coverage: {{ coveragePercent }}%\n </p>\n <p *ngIf=\"!baseCumulatives\" class=\"coverage-error\">\n <ion-icon name=\"alert-circle\"></ion-icon>\n No base cumulatives available\n </p>\n </div>\n\n <!-- Creation progress -->\n <div *ngIf=\"creating\" class=\"creation-progress\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <span>{{ creationProgress }}</span>\n </div>\n\n <ion-button [disabled]=\"!canCreateNew || creating\"\n expand=\"block\"\n (click)=\"createNewPeriod()\">\n <ion-icon name=\"add-circle-outline\" slot=\"start\"></ion-icon>\n Create {{ formatWindow(currentWindowMinutes) }} Cumulative\n </ion-button>\n\n <p *ngIf=\"!canCreateNew && !creating\" class=\"create-hint\">\n Create is disabled because base 5-min cumulatives don't fully cover the selected period.\n </p>\n </div>\n </div>\n\n <div class=\"modal-footer\">\n <ion-button fill=\"outline\" (click)=\"cancel()\" [disabled]=\"creating\">\n Cancel\n </ion-button>\n </div>\n </div>\n</div>\n", styles: [".cumulative-selector-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,.5);display:flex;align-items:center;justify-content:center;z-index:1000}.cumulative-selector-modal{background:var(--ion-background-color, #fff);border-radius:12px;max-width:500px;width:90%;max-height:80vh;display:flex;flex-direction:column;box-shadow:0 4px 20px #0000004d}.modal-header{display:flex;justify-content:space-between;align-items:center;padding:16px 20px;border-bottom:1px solid var(--ion-border-color, #ddd)}.modal-header h2{margin:0;font-size:1.25rem;font-weight:600}.modal-header ion-button{--padding-start: 8px;--padding-end: 8px}.modal-content{flex:1;overflow-y:auto;padding:16px 20px}.loading-state{display:flex;flex-direction:column;align-items:center;padding:40px 0}.loading-state ion-spinner{margin-bottom:16px}.loading-state p{color:var(--ion-color-medium)}.error-message{display:flex;align-items:center;gap:8px;padding:12px;background:var(--ion-color-danger-tint);color:var(--ion-color-danger);border-radius:8px;margin-bottom:16px}.error-message ion-icon{font-size:1.25rem}.periods-section{margin-bottom:24px}.periods-section h3{font-size:1rem;font-weight:600;margin-bottom:12px;color:var(--ion-color-dark)}.periods-section ion-list{border-radius:8px;overflow:hidden}.periods-section ion-item{--padding-start: 12px;--padding-end: 12px}.periods-section ion-item ion-label h2{font-weight:500}.periods-section ion-item ion-label p{font-size:.85rem;color:var(--ion-color-medium)}.periods-section ion-item ion-label .count-info{font-size:.75rem;color:var(--ion-color-primary)}.no-periods{display:flex;flex-direction:column;align-items:center;padding:24px;text-align:center;color:var(--ion-color-medium)}.no-periods ion-icon{font-size:2rem;margin-bottom:8px}.create-section{border-top:1px solid var(--ion-border-color, #ddd);padding-top:16px}.create-section h3{font-size:1rem;font-weight:600;margin-bottom:12px;color:var(--ion-color-dark)}.create-section .create-info{background:var(--ion-color-light);padding:12px;border-radius:8px;margin-bottom:16px}.create-section .create-info p{margin:4px 0;font-size:.9rem}.create-section .coverage-info{display:flex;align-items:center;gap:6px}.create-section .coverage-info ion-icon{font-size:1.1rem}.create-section .coverage-ok{color:var(--ion-color-success)}.create-section .coverage-warn{color:var(--ion-color-warning)}.create-section .coverage-error{color:var(--ion-color-danger)}.create-section .creation-progress{display:flex;align-items:center;gap:12px;padding:16px;background:var(--ion-color-primary-tint);border-radius:8px;margin-bottom:16px}.create-section .creation-progress ion-spinner{--color: var(--ion-color-primary)}.create-section .creation-progress span{color:var(--ion-color-primary);font-size:.9rem}.create-section .create-hint{font-size:.8rem;color:var(--ion-color-medium);text-align:center;margin-top:8px}.modal-footer{padding:16px 20px;border-top:1px solid var(--ion-border-color, #ddd);display:flex;justify-content:flex-end}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: i3.IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: i3.IonItem, selector: "ion-item", inputs: ["button", "color", "counter", "counterFormatter", "detail", "detailIcon", "disabled", "download", "fill", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "shape", "target", "type"] }, { kind: "component", type: i3.IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: i3.IonList, selector: "ion-list", inputs: ["inset", "lines", "mode"] }, { kind: "component", type: i3.IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
179
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CumulativeSelectorComponent, decorators: [{
|
|
180
|
+
type: Component,
|
|
181
|
+
args: [{ selector: 'cumulative-selector', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"cumulative-selector-overlay\">\n <div class=\"cumulative-selector-modal\">\n <div class=\"modal-header\">\n <h2>Select Cumulative Period</h2>\n <ion-button fill=\"clear\" (click)=\"cancel()\">\n <ion-icon name=\"close\"></ion-icon>\n </ion-button>\n </div>\n\n <div class=\"modal-content\">\n <!-- Loading state -->\n <div *ngIf=\"loading\" class=\"loading-state\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <p>Loading available periods...</p>\n </div>\n\n <!-- Error message -->\n <div *ngIf=\"errorMessage\" class=\"error-message\">\n <ion-icon name=\"warning-outline\"></ion-icon>\n <span>{{ errorMessage }}</span>\n </div>\n\n <!-- Available periods list -->\n <div *ngIf=\"!loading && availablePeriods.length > 0\" class=\"periods-section\">\n <h3>Available Cumulative Periods</h3>\n <ion-list>\n <ion-item *ngFor=\"let period of availablePeriods\"\n button\n (click)=\"selectPeriod(period)\"\n [disabled]=\"creating\">\n <ion-icon name=\"layers-outline\" slot=\"start\"></ion-icon>\n <ion-label>\n <h2>{{ formatWindow(period.windowInMinutes) }}</h2>\n <p>{{ formatPeriod(period) }}</p>\n <p class=\"count-info\">{{ period.count }} cumulative(s)</p>\n </ion-label>\n <ion-icon name=\"chevron-forward\" slot=\"end\"></ion-icon>\n </ion-item>\n </ion-list>\n </div>\n\n <!-- No periods available -->\n <div *ngIf=\"!loading && availablePeriods.length === 0\" class=\"no-periods\">\n <ion-icon name=\"information-circle-outline\"></ion-icon>\n <p>No cumulative periods available yet.</p>\n </div>\n\n <!-- Create new section (admin only) -->\n <div *ngIf=\"!loading && isAdmin\" class=\"create-section\">\n <h3>Create New Cumulative</h3>\n <div class=\"create-info\">\n <p>\n <strong>Period:</strong>\n {{ currentPeriodBegin?.toLocaleString() }} \u2192 {{ currentPeriodEnd?.toLocaleString() }}\n </p>\n <p>\n <strong>Window:</strong> {{ formatWindow(currentWindowMinutes) }}\n </p>\n <p *ngIf=\"baseCumulatives\" class=\"coverage-info\"\n [class.coverage-ok]=\"coveragePercent >= 100\"\n [class.coverage-warn]=\"coveragePercent > 0 && coveragePercent < 100\"\n [class.coverage-error]=\"coveragePercent === 0\">\n <ion-icon [name]=\"coveragePercent >= 100 ? 'checkmark-circle' : 'alert-circle'\"></ion-icon>\n Base data coverage: {{ coveragePercent }}%\n </p>\n <p *ngIf=\"!baseCumulatives\" class=\"coverage-error\">\n <ion-icon name=\"alert-circle\"></ion-icon>\n No base cumulatives available\n </p>\n </div>\n\n <!-- Creation progress -->\n <div *ngIf=\"creating\" class=\"creation-progress\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <span>{{ creationProgress }}</span>\n </div>\n\n <ion-button [disabled]=\"!canCreateNew || creating\"\n expand=\"block\"\n (click)=\"createNewPeriod()\">\n <ion-icon name=\"add-circle-outline\" slot=\"start\"></ion-icon>\n Create {{ formatWindow(currentWindowMinutes) }} Cumulative\n </ion-button>\n\n <p *ngIf=\"!canCreateNew && !creating\" class=\"create-hint\">\n Create is disabled because base 5-min cumulatives don't fully cover the selected period.\n </p>\n </div>\n </div>\n\n <div class=\"modal-footer\">\n <ion-button fill=\"outline\" (click)=\"cancel()\" [disabled]=\"creating\">\n Cancel\n </ion-button>\n </div>\n </div>\n</div>\n", styles: [".cumulative-selector-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,.5);display:flex;align-items:center;justify-content:center;z-index:1000}.cumulative-selector-modal{background:var(--ion-background-color, #fff);border-radius:12px;max-width:500px;width:90%;max-height:80vh;display:flex;flex-direction:column;box-shadow:0 4px 20px #0000004d}.modal-header{display:flex;justify-content:space-between;align-items:center;padding:16px 20px;border-bottom:1px solid var(--ion-border-color, #ddd)}.modal-header h2{margin:0;font-size:1.25rem;font-weight:600}.modal-header ion-button{--padding-start: 8px;--padding-end: 8px}.modal-content{flex:1;overflow-y:auto;padding:16px 20px}.loading-state{display:flex;flex-direction:column;align-items:center;padding:40px 0}.loading-state ion-spinner{margin-bottom:16px}.loading-state p{color:var(--ion-color-medium)}.error-message{display:flex;align-items:center;gap:8px;padding:12px;background:var(--ion-color-danger-tint);color:var(--ion-color-danger);border-radius:8px;margin-bottom:16px}.error-message ion-icon{font-size:1.25rem}.periods-section{margin-bottom:24px}.periods-section h3{font-size:1rem;font-weight:600;margin-bottom:12px;color:var(--ion-color-dark)}.periods-section ion-list{border-radius:8px;overflow:hidden}.periods-section ion-item{--padding-start: 12px;--padding-end: 12px}.periods-section ion-item ion-label h2{font-weight:500}.periods-section ion-item ion-label p{font-size:.85rem;color:var(--ion-color-medium)}.periods-section ion-item ion-label .count-info{font-size:.75rem;color:var(--ion-color-primary)}.no-periods{display:flex;flex-direction:column;align-items:center;padding:24px;text-align:center;color:var(--ion-color-medium)}.no-periods ion-icon{font-size:2rem;margin-bottom:8px}.create-section{border-top:1px solid var(--ion-border-color, #ddd);padding-top:16px}.create-section h3{font-size:1rem;font-weight:600;margin-bottom:12px;color:var(--ion-color-dark)}.create-section .create-info{background:var(--ion-color-light);padding:12px;border-radius:8px;margin-bottom:16px}.create-section .create-info p{margin:4px 0;font-size:.9rem}.create-section .coverage-info{display:flex;align-items:center;gap:6px}.create-section .coverage-info ion-icon{font-size:1.1rem}.create-section .coverage-ok{color:var(--ion-color-success)}.create-section .coverage-warn{color:var(--ion-color-warning)}.create-section .coverage-error{color:var(--ion-color-danger)}.create-section .creation-progress{display:flex;align-items:center;gap:12px;padding:16px;background:var(--ion-color-primary-tint);border-radius:8px;margin-bottom:16px}.create-section .creation-progress ion-spinner{--color: var(--ion-color-primary)}.create-section .creation-progress span{color:var(--ion-color-primary);font-size:.9rem}.create-section .create-hint{font-size:.8rem;color:var(--ion-color-medium);text-align:center;margin-top:8px}.modal-footer{padding:16px 20px;border-top:1px solid var(--ion-border-color, #ddd);display:flex;justify-content:flex-end}\n"] }]
|
|
182
|
+
}], ctorParameters: function () { return [{ type: i1.ProfileService }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { rainId: [{
|
|
183
|
+
type: Input
|
|
184
|
+
}], currentPeriodBegin: [{
|
|
185
|
+
type: Input
|
|
186
|
+
}], currentPeriodEnd: [{
|
|
187
|
+
type: Input
|
|
188
|
+
}], provider: [{
|
|
189
|
+
type: Input
|
|
190
|
+
}], timeStepInMinutes: [{
|
|
191
|
+
type: Input
|
|
192
|
+
}], isAdmin: [{
|
|
193
|
+
type: Input
|
|
194
|
+
}], periodSelected: [{
|
|
195
|
+
type: Output
|
|
196
|
+
}], cancelled: [{
|
|
197
|
+
type: Output
|
|
198
|
+
}] } });
|
|
199
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cumulative-selector.component.js","sourceRoot":"","sources":["../../../src/core/shared/cumulative-selector/cumulative-selector.component.ts","../../../src/core/shared/cumulative-selector/cumulative-selector.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    EventEmitter,\n    Input,\n    OnInit,\n    Output,\n} from '@angular/core';\nimport {ProfileService} from '../profile.service';\nimport {CumulativePeriod} from 'raain-model';\n\nexport interface CumulativeSelection {\n    periodBegin: Date;\n    periodEnd: Date;\n    windowInMinutes: number;\n}\n\n@Component({\n    selector: 'cumulative-selector',\n    templateUrl: './cumulative-selector.component.html',\n    styleUrls: ['./cumulative-selector.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class CumulativeSelectorComponent implements OnInit {\n    @Input() rainId: string;\n    @Input() currentPeriodBegin: Date;\n    @Input() currentPeriodEnd: Date;\n    @Input() provider: string;\n    @Input() timeStepInMinutes: number = 5;\n    @Input() isAdmin: boolean = false;\n\n    @Output() periodSelected = new EventEmitter<CumulativeSelection>();\n    @Output() cancelled = new EventEmitter<void>();\n\n    availablePeriods: CumulativePeriod[] = [];\n    baseCumulatives: CumulativePeriod = null;\n    loading = true;\n    creating = false;\n    creationProgress = '';\n    errorMessage = '';\n\n    coveragePercent = 0;\n    canCreateNew = false;\n    currentWindowMinutes = 0;\n\n    private readonly POLL_TIMEOUT_MS = 900000; // 900 seconds\n    private readonly POLL_INTERVAL_MS = 3000;\n\n    constructor(\n        private profileService: ProfileService,\n        private cdr: ChangeDetectorRef\n    ) {}\n\n    async ngOnInit() {\n        this.currentWindowMinutes = Math.round(\n            (this.currentPeriodEnd.getTime() - this.currentPeriodBegin.getTime()) / 60000\n        );\n        await this.loadAvailablePeriods();\n    }\n\n    async loadAvailablePeriods() {\n        this.loading = true;\n        this.errorMessage = '';\n        this.cdr.markForCheck();\n\n        try {\n            // Fetch all cumulative periods (admin sees all, non-admin sees only customer-launched)\n            const response = await this.profileService.getCumulativePeriods(this.rainId, {\n                provider: this.provider,\n                forced: this.isAdmin,\n            });\n\n            // Filter for existing custom cumulatives (window > 0)\n            this.availablePeriods = response.periods.filter((p) => p.windowInMinutes > 0);\n\n            // Get base cumulatives (window = 0) to check coverage\n            this.baseCumulatives = response.periods.find((p) => p.windowInMinutes === 0);\n\n            this.calculateCoverage();\n        } catch (e) {\n            this.errorMessage = 'Failed to load cumulative periods';\n            console.error('Error loading cumulative periods:', e);\n        }\n\n        this.loading = false;\n        this.cdr.markForCheck();\n    }\n\n    calculateCoverage() {\n        if (!this.baseCumulatives) {\n            this.coveragePercent = 0;\n            this.canCreateNew = false;\n            return;\n        }\n\n        const baseBegin = new Date(this.baseCumulatives.periodBegin);\n        const baseEnd = new Date(this.baseCumulatives.periodEnd);\n\n        // Check if current period is within base coverage\n        const currentInRange =\n            this.currentPeriodBegin >= baseBegin && this.currentPeriodEnd <= baseEnd;\n\n        if (!currentInRange) {\n            this.coveragePercent = 0;\n            this.canCreateNew = false;\n            return;\n        }\n\n        // Calculate expected number of base cumulatives needed\n        const expectedCount = Math.ceil(this.currentWindowMinutes / this.timeStepInMinutes);\n\n        // Check if enough base cumulatives exist\n        // We assume coverage is complete if count >= expected (simplified check)\n        if (this.baseCumulatives.count >= expectedCount) {\n            this.coveragePercent = 100;\n            this.canCreateNew = true;\n        } else {\n            this.coveragePercent = Math.round((this.baseCumulatives.count / expectedCount) * 100);\n            this.canCreateNew = this.coveragePercent >= 100;\n        }\n    }\n\n    selectPeriod(period: CumulativePeriod) {\n        this.periodSelected.emit({\n            periodBegin: new Date(period.periodBegin),\n            periodEnd: new Date(period.periodEnd),\n            windowInMinutes: period.windowInMinutes,\n        });\n    }\n\n    async createNewPeriod() {\n        if (!this.canCreateNew || this.creating) {\n            return;\n        }\n\n        this.creating = true;\n        this.creationProgress = 'Starting cumulative computation...';\n        this.errorMessage = '';\n        this.cdr.markForCheck();\n\n        try {\n            // Trigger cumulative creation\n            const result = await this.profileService.createCumulativePeriod(this.rainId, {\n                periodBegin: this.currentPeriodBegin,\n                periodEnd: this.currentPeriodEnd,\n                provider: this.provider,\n                timeStepInMinutes: this.timeStepInMinutes,\n            });\n\n            if (!result) {\n                throw new Error('Failed to trigger cumulative computation');\n            }\n\n            this.creationProgress = `Jobs queued. Polling for completion...`;\n            this.cdr.markForCheck();\n\n            // Poll for completion\n            const success = await this.pollForCompletion();\n\n            if (success) {\n                this.periodSelected.emit({\n                    periodBegin: this.currentPeriodBegin,\n                    periodEnd: this.currentPeriodEnd,\n                    windowInMinutes: this.currentWindowMinutes,\n                });\n            } else {\n                this.errorMessage = 'Timeout waiting for cumulative computation';\n            }\n        } catch (e) {\n            this.errorMessage = `Error: ${e.message || 'Unknown error'}`;\n            console.error('Error creating cumulative:', e);\n        }\n\n        this.creating = false;\n        this.cdr.markForCheck();\n    }\n\n    private async pollForCompletion(): Promise<boolean> {\n        const startTime = Date.now();\n\n        while (Date.now() - startTime < this.POLL_TIMEOUT_MS) {\n            try {\n                const progress = await this.profileService.getRainProgress(this.rainId);\n\n                if (progress === 0) {\n                    // Queue is empty, check if cumulative exists (admin sees all cumulatives)\n                    const response = await this.profileService.getCumulativePeriods(this.rainId, {\n                        provider: this.provider,\n                        windowInMinutes: this.currentWindowMinutes,\n                        forced: this.isAdmin,\n                    });\n\n                    const found = response.periods.find(\n                        (p) => p.windowInMinutes === this.currentWindowMinutes\n                    );\n\n                    if (found) {\n                        return true;\n                    }\n                }\n\n                const elapsed = Math.round((Date.now() - startTime) / 1000);\n                this.creationProgress = `Computing... (${elapsed}s, queue: ${progress})`;\n                this.cdr.markForCheck();\n\n                await this.sleep(this.POLL_INTERVAL_MS);\n            } catch (e) {\n                console.warn('Poll error:', e);\n                await this.sleep(this.POLL_INTERVAL_MS);\n            }\n        }\n\n        return false;\n    }\n\n    private sleep(ms: number): Promise<void> {\n        return new Promise((resolve) => setTimeout(resolve, ms));\n    }\n\n    cancel() {\n        this.cancelled.emit();\n    }\n\n    formatPeriod(period: CumulativePeriod): string {\n        const begin = new Date(period.periodBegin);\n        const end = new Date(period.periodEnd);\n        return `${begin.toLocaleString()} → ${end.toLocaleString()}`;\n    }\n\n    formatWindow(minutes: number): string {\n        if (minutes < 60) {\n            return `${minutes} min`;\n        }\n        const hours = minutes / 60;\n        return hours === 1 ? '1 hour' : `${hours} hours`;\n    }\n}\n","<div class=\"cumulative-selector-overlay\">\n    <div class=\"cumulative-selector-modal\">\n        <div class=\"modal-header\">\n            <h2>Select Cumulative Period</h2>\n            <ion-button fill=\"clear\" (click)=\"cancel()\">\n                <ion-icon name=\"close\"></ion-icon>\n            </ion-button>\n        </div>\n\n        <div class=\"modal-content\">\n            <!-- Loading state -->\n            <div *ngIf=\"loading\" class=\"loading-state\">\n                <ion-spinner name=\"crescent\"></ion-spinner>\n                <p>Loading available periods...</p>\n            </div>\n\n            <!-- Error message -->\n            <div *ngIf=\"errorMessage\" class=\"error-message\">\n                <ion-icon name=\"warning-outline\"></ion-icon>\n                <span>{{ errorMessage }}</span>\n            </div>\n\n            <!-- Available periods list -->\n            <div *ngIf=\"!loading && availablePeriods.length > 0\" class=\"periods-section\">\n                <h3>Available Cumulative Periods</h3>\n                <ion-list>\n                    <ion-item *ngFor=\"let period of availablePeriods\"\n                              button\n                              (click)=\"selectPeriod(period)\"\n                              [disabled]=\"creating\">\n                        <ion-icon name=\"layers-outline\" slot=\"start\"></ion-icon>\n                        <ion-label>\n                            <h2>{{ formatWindow(period.windowInMinutes) }}</h2>\n                            <p>{{ formatPeriod(period) }}</p>\n                            <p class=\"count-info\">{{ period.count }} cumulative(s)</p>\n                        </ion-label>\n                        <ion-icon name=\"chevron-forward\" slot=\"end\"></ion-icon>\n                    </ion-item>\n                </ion-list>\n            </div>\n\n            <!-- No periods available -->\n            <div *ngIf=\"!loading && availablePeriods.length === 0\" class=\"no-periods\">\n                <ion-icon name=\"information-circle-outline\"></ion-icon>\n                <p>No cumulative periods available yet.</p>\n            </div>\n\n            <!-- Create new section (admin only) -->\n            <div *ngIf=\"!loading && isAdmin\" class=\"create-section\">\n                <h3>Create New Cumulative</h3>\n                <div class=\"create-info\">\n                    <p>\n                        <strong>Period:</strong>\n                        {{ currentPeriodBegin?.toLocaleString() }} → {{ currentPeriodEnd?.toLocaleString() }}\n                    </p>\n                    <p>\n                        <strong>Window:</strong> {{ formatWindow(currentWindowMinutes) }}\n                    </p>\n                    <p *ngIf=\"baseCumulatives\" class=\"coverage-info\"\n                       [class.coverage-ok]=\"coveragePercent >= 100\"\n                       [class.coverage-warn]=\"coveragePercent > 0 && coveragePercent < 100\"\n                       [class.coverage-error]=\"coveragePercent === 0\">\n                        <ion-icon [name]=\"coveragePercent >= 100 ? 'checkmark-circle' : 'alert-circle'\"></ion-icon>\n                        Base data coverage: {{ coveragePercent }}%\n                    </p>\n                    <p *ngIf=\"!baseCumulatives\" class=\"coverage-error\">\n                        <ion-icon name=\"alert-circle\"></ion-icon>\n                        No base cumulatives available\n                    </p>\n                </div>\n\n                <!-- Creation progress -->\n                <div *ngIf=\"creating\" class=\"creation-progress\">\n                    <ion-spinner name=\"crescent\"></ion-spinner>\n                    <span>{{ creationProgress }}</span>\n                </div>\n\n                <ion-button [disabled]=\"!canCreateNew || creating\"\n                            expand=\"block\"\n                            (click)=\"createNewPeriod()\">\n                    <ion-icon name=\"add-circle-outline\" slot=\"start\"></ion-icon>\n                    Create {{ formatWindow(currentWindowMinutes) }} Cumulative\n                </ion-button>\n\n                <p *ngIf=\"!canCreateNew && !creating\" class=\"create-hint\">\n                    Create is disabled because base 5-min cumulatives don't fully cover the selected period.\n                </p>\n            </div>\n        </div>\n\n        <div class=\"modal-footer\">\n            <ion-button fill=\"outline\" (click)=\"cancel()\" [disabled]=\"creating\">\n                Cancel\n            </ion-button>\n        </div>\n    </div>\n</div>\n"]}
|
package/esm2020/index.mjs
CHANGED
|
@@ -7,6 +7,7 @@ export * from './raain-speed/raain-speed.component';
|
|
|
7
7
|
export * from './raain-compare-stack/raain-compare-stack.component';
|
|
8
8
|
export * from './raain-globe/raain-globe.component';
|
|
9
9
|
export * from './raain-details/raain-details.component';
|
|
10
|
+
export * from './cumulative-selector/cumulative-selector.component';
|
|
10
11
|
export * from './tools';
|
|
11
12
|
export * from './cache.service';
|
|
12
13
|
export * from './fidj-storage.model';
|
|
@@ -19,4 +20,4 @@ export * from './storage.service';
|
|
|
19
20
|
export * from './xytype';
|
|
20
21
|
export * from './profile-icon.directive';
|
|
21
22
|
export * from './pipes.module';
|
|
22
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29yZS9zaGFyZWQvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxpQ0FBaUMsQ0FBQztBQUNoRCxjQUFjLHlDQUF5QyxDQUFDO0FBQ3hELGNBQWMscURBQXFELENBQUM7QUFDcEUsY0FBYywrQ0FBK0MsQ0FBQztBQUM5RCxjQUFjLG1EQUFtRCxDQUFDO0FBQ2xFLGNBQWMscUNBQXFDLENBQUM7QUFDcEQsY0FBYyxxREFBcUQsQ0FBQztBQUNwRSxjQUFjLHFDQUFxQyxDQUFDO0FBQ3BELGNBQWMseUNBQXlDLENBQUM7QUFDeEQsY0FBYyxxREFBcUQsQ0FBQztBQUVwRSxjQUFjLFNBQVMsQ0FBQztBQUV4QixjQUFjLGlCQUFpQixDQUFDO0FBQ2hDLGNBQWMsc0JBQXNCLENBQUM7QUFDckMsY0FBYyxtQkFBbUIsQ0FBQztBQUNsQyxjQUFjLGlCQUFpQixDQUFDO0FBQ2hDLGNBQWMsZ0JBQWdCLENBQUM7QUFDL0IsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxpQkFBaUIsQ0FBQztBQUNoQyxjQUFjLG1CQUFtQixDQUFDO0FBQ2xDLGNBQWMsVUFBVSxDQUFDO0FBQ3pCLGNBQWMsMEJBQTBCLENBQUM7QUFDekMsY0FBYyxnQkFBZ0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vcmFhaW4tbWFwL3JhYWluLW1hcC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9yYWFpbi1jb21wYXJlL3JhYWluLWNvbXBhcmUuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vcmFhaW4tY29uZmlndXJhdGlvbi9yYWFpbi1jb25maWd1cmF0aW9uLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3JhYWluLWRhdGUtZm9jdXMvcmFhaW4tZGF0ZS1mb2N1cy5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9yYWFpbi1kYXRlLWR5bmFtaWMvcmFhaW4tZGF0ZS1keW5hbWljLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3JhYWluLXNwZWVkL3JhYWluLXNwZWVkLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3JhYWluLWNvbXBhcmUtc3RhY2svcmFhaW4tY29tcGFyZS1zdGFjay5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9yYWFpbi1nbG9iZS9yYWFpbi1nbG9iZS5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9yYWFpbi1kZXRhaWxzL3JhYWluLWRldGFpbHMuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vY3VtdWxhdGl2ZS1zZWxlY3Rvci9jdW11bGF0aXZlLXNlbGVjdG9yLmNvbXBvbmVudCc7XG5cbmV4cG9ydCAqIGZyb20gJy4vdG9vbHMnO1xuXG5leHBvcnQgKiBmcm9tICcuL2NhY2hlLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9maWRqLXN0b3JhZ2UubW9kZWwnO1xuZXhwb3J0ICogZnJvbSAnLi9wcm9maWxlLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9yYWRhci5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vc2hhcmVkLmNvbnN0JztcbmV4cG9ydCAqIGZyb20gJy4vc2hhcmVkLnBpcGUnO1xuZXhwb3J0ICogZnJvbSAnLi9zaGFyZWQubW9kdWxlJztcbmV4cG9ydCAqIGZyb20gJy4vc3RvcmFnZS5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4veHl0eXBlJztcbmV4cG9ydCAqIGZyb20gJy4vcHJvZmlsZS1pY29uLmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL3BpcGVzLm1vZHVsZSc7XG4iXX0=
|