valtech-components 2.0.833 → 2.0.835
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/esm2022/lib/components/organisms/change-password-modal/change-password-modal.component.mjs +156 -58
- package/esm2022/lib/components/organisms/mfa-modal/mfa-modal.component.mjs +415 -0
- package/esm2022/lib/services/auth/interceptor.mjs +10 -6
- package/esm2022/lib/services/i18n/default-content.mjs +129 -1
- package/esm2022/lib/version.mjs +2 -2
- package/esm2022/public-api.mjs +2 -1
- package/fesm2022/valtech-components.mjs +694 -63
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/organisms/change-password-modal/change-password-modal.component.d.ts +38 -12
- package/lib/components/organisms/mfa-modal/mfa-modal.component.d.ts +114 -0
- package/lib/version.d.ts +1 -1
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
|
@@ -0,0 +1,415 @@
|
|
|
1
|
+
import { Component, EventEmitter, inject, Input, Output, signal } from '@angular/core';
|
|
2
|
+
import { FormControl, ReactiveFormsModule, Validators } from '@angular/forms';
|
|
3
|
+
import { IonButton, IonButtons, IonContent, IonHeader, IonIcon, IonInput, IonItem, IonLabel, IonModal, IonRadio, IonRadioGroup, IonSpinner, IonToolbar, } from '@ionic/angular/standalone';
|
|
4
|
+
import { addIcons } from 'ionicons';
|
|
5
|
+
import { closeOutline } from 'ionicons/icons';
|
|
6
|
+
import { QrCodeComponent } from '../../atoms/qr-code/qr-code.component';
|
|
7
|
+
import { PinInputComponent } from '../../molecules/pin-input/pin-input.component';
|
|
8
|
+
import { AuthService } from '../../../services/auth';
|
|
9
|
+
import { I18nService } from '../../../services/i18n';
|
|
10
|
+
import { QrGeneratorService } from '../../../services/qr-generator/qr-generator.service';
|
|
11
|
+
import { ToastService } from '../../../services/toast.service';
|
|
12
|
+
import * as i0 from "@angular/core";
|
|
13
|
+
import * as i1 from "@angular/forms";
|
|
14
|
+
/** Segundos de espera antes de poder reenviar el código EMAIL/SMS. */
|
|
15
|
+
const RESEND_COOLDOWN_SECONDS = 30;
|
|
16
|
+
/**
|
|
17
|
+
* `val-mfa-modal` — modal de gestión de autenticación de dos factores (MFA)
|
|
18
|
+
* para un usuario autenticado. Mismo patrón que `val-change-password-modal`.
|
|
19
|
+
*
|
|
20
|
+
* Flujo (máquina de estados interna):
|
|
21
|
+
* - `loading` → `getProfile()` para conocer el estado MFA.
|
|
22
|
+
* - `status` → muestra MFA habilitado/deshabilitado. Si está habilitado:
|
|
23
|
+
* gestión de backup codes (TOTP) + deshabilitar. Si no: botón habilitar.
|
|
24
|
+
* - `method-select` → elegir TOTP / EMAIL / SMS.
|
|
25
|
+
* - `totp-setup` → QR + secreto manual + backup codes → verificar código.
|
|
26
|
+
* - `code-confirm` → (EMAIL/SMS) ingresar código recibido, con reenvío.
|
|
27
|
+
* - `disable` → contraseña para deshabilitar MFA.
|
|
28
|
+
*
|
|
29
|
+
* El QR se genera **client-side** (`QrGeneratorService`) — el secreto TOTP
|
|
30
|
+
* nunca sale del navegador.
|
|
31
|
+
*
|
|
32
|
+
* Self-contained: inyecta `AuthService` y llama los endpoints directo. La app
|
|
33
|
+
* controla `[isOpen]` y reacciona a `(changed)` / `(dismissed)`.
|
|
34
|
+
*
|
|
35
|
+
* i18n: namespace compartido `_auth`.
|
|
36
|
+
*
|
|
37
|
+
* @example
|
|
38
|
+
* ```html
|
|
39
|
+
* <val-mfa-modal
|
|
40
|
+
* [isOpen]="isModalOpen()"
|
|
41
|
+
* (dismissed)="isModalOpen.set(false)"
|
|
42
|
+
* />
|
|
43
|
+
* ```
|
|
44
|
+
*/
|
|
45
|
+
export class MfaModalComponent {
|
|
46
|
+
/** Controla la visibilidad. Cada apertura re-resuelve el estado MFA. */
|
|
47
|
+
set isOpen(value) {
|
|
48
|
+
const opening = value && !this._isOpen;
|
|
49
|
+
this._isOpen = value;
|
|
50
|
+
if (opening) {
|
|
51
|
+
this.resolveStatus();
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
get isOpen() {
|
|
55
|
+
return this._isOpen;
|
|
56
|
+
}
|
|
57
|
+
constructor() {
|
|
58
|
+
this._isOpen = false;
|
|
59
|
+
/** Emite cuando el estado MFA cambia (habilitado / deshabilitado). */
|
|
60
|
+
this.changed = new EventEmitter();
|
|
61
|
+
/** Emite cuando el user cierra el modal (botón X o backdrop). */
|
|
62
|
+
this.dismissed = new EventEmitter();
|
|
63
|
+
this.auth = inject(AuthService);
|
|
64
|
+
this.toast = inject(ToastService);
|
|
65
|
+
this.i18n = inject(I18nService);
|
|
66
|
+
this.qrGen = inject(QrGeneratorService);
|
|
67
|
+
this._step = signal('loading');
|
|
68
|
+
/** Paso actual del flujo. */
|
|
69
|
+
this.step = this._step.asReadonly();
|
|
70
|
+
/** `true` mientras una llamada al backend está en curso. */
|
|
71
|
+
this.working = signal(false);
|
|
72
|
+
// Estado MFA actual.
|
|
73
|
+
this.mfaEnabled = signal(false);
|
|
74
|
+
this.mfaMethod = signal(null);
|
|
75
|
+
this.userPhone = signal(null);
|
|
76
|
+
this.backupCodesCount = signal(0);
|
|
77
|
+
// Estado del flujo de habilitación.
|
|
78
|
+
this.selectedMethod = signal('TOTP');
|
|
79
|
+
this.totpSetup = signal(null);
|
|
80
|
+
this.totpQr = signal(null);
|
|
81
|
+
/** Códigos de respaldo recién regenerados — se muestran una sola vez. */
|
|
82
|
+
this.regeneratedCodes = signal(null);
|
|
83
|
+
this.resendCooldown = signal(0);
|
|
84
|
+
this.pinControl = new FormControl('', [Validators.required, Validators.minLength(6), Validators.maxLength(6)]);
|
|
85
|
+
this.passwordControl = new FormControl('', [Validators.required]);
|
|
86
|
+
this.phoneControl = new FormControl('', [Validators.required, Validators.pattern(/^\+[1-9]\d{6,14}$/)]);
|
|
87
|
+
this.pinInputProps = {
|
|
88
|
+
control: this.pinControl,
|
|
89
|
+
token: 'mfa-code',
|
|
90
|
+
length: 6,
|
|
91
|
+
allowNumbersOnly: true,
|
|
92
|
+
autoFocus: true,
|
|
93
|
+
};
|
|
94
|
+
this.resendTimer = null;
|
|
95
|
+
addIcons({ closeOutline });
|
|
96
|
+
}
|
|
97
|
+
ngOnDestroy() {
|
|
98
|
+
this.stopCooldown();
|
|
99
|
+
}
|
|
100
|
+
/** Traduce una clave del namespace `_auth`. */
|
|
101
|
+
t(key) {
|
|
102
|
+
return this.i18n.t(key, '_auth');
|
|
103
|
+
}
|
|
104
|
+
/** Cierre iniciado por el user (X / backdrop). */
|
|
105
|
+
close() {
|
|
106
|
+
this.dismissed.emit();
|
|
107
|
+
}
|
|
108
|
+
// ===========================================================================
|
|
109
|
+
// Carga de estado
|
|
110
|
+
// ===========================================================================
|
|
111
|
+
/** Consulta el perfil para conocer el estado MFA y posicionar el flujo. */
|
|
112
|
+
resolveStatus() {
|
|
113
|
+
this._step.set('loading');
|
|
114
|
+
this.resetFlow();
|
|
115
|
+
this.auth.getProfile().subscribe({
|
|
116
|
+
next: profile => {
|
|
117
|
+
this.mfaEnabled.set(profile.mfaEnabled);
|
|
118
|
+
this.mfaMethod.set(profile.mfaMethod ?? null);
|
|
119
|
+
this.userPhone.set(profile.phone ?? null);
|
|
120
|
+
if (profile.mfaEnabled && profile.mfaMethod === 'TOTP') {
|
|
121
|
+
this.loadBackupCount();
|
|
122
|
+
}
|
|
123
|
+
this._step.set('status');
|
|
124
|
+
},
|
|
125
|
+
error: () => {
|
|
126
|
+
// Fallback: usar el signal de usuario en sesión.
|
|
127
|
+
const user = this.auth.user();
|
|
128
|
+
this.mfaEnabled.set(user?.mfaEnabled ?? false);
|
|
129
|
+
this.mfaMethod.set(user?.mfaMethod ?? null);
|
|
130
|
+
this._step.set('status');
|
|
131
|
+
},
|
|
132
|
+
});
|
|
133
|
+
}
|
|
134
|
+
loadBackupCount() {
|
|
135
|
+
this.auth.getBackupCodesCount().subscribe({
|
|
136
|
+
next: res => this.backupCodesCount.set(res.count),
|
|
137
|
+
error: () => this.backupCodesCount.set(0),
|
|
138
|
+
});
|
|
139
|
+
}
|
|
140
|
+
// ===========================================================================
|
|
141
|
+
// Navegación entre pasos
|
|
142
|
+
// ===========================================================================
|
|
143
|
+
goToMethodSelect() {
|
|
144
|
+
this.regeneratedCodes.set(null);
|
|
145
|
+
this._step.set('method-select');
|
|
146
|
+
}
|
|
147
|
+
goToDisable() {
|
|
148
|
+
this.passwordControl.reset();
|
|
149
|
+
this._step.set('disable');
|
|
150
|
+
}
|
|
151
|
+
backToStatus() {
|
|
152
|
+
this.stopCooldown();
|
|
153
|
+
this.resolveStatus();
|
|
154
|
+
}
|
|
155
|
+
// ===========================================================================
|
|
156
|
+
// Habilitar MFA
|
|
157
|
+
// ===========================================================================
|
|
158
|
+
/** Continúa desde el selector de método al setup correspondiente. */
|
|
159
|
+
proceedWithMethod() {
|
|
160
|
+
const method = this.selectedMethod();
|
|
161
|
+
if (method === 'TOTP') {
|
|
162
|
+
this.setupTotp();
|
|
163
|
+
return;
|
|
164
|
+
}
|
|
165
|
+
let phone;
|
|
166
|
+
if (method === 'SMS' && !this.userPhone()) {
|
|
167
|
+
if (this.phoneControl.invalid) {
|
|
168
|
+
this.phoneControl.markAsTouched();
|
|
169
|
+
this.showToast(this.t('mfaPhoneInvalid'));
|
|
170
|
+
return;
|
|
171
|
+
}
|
|
172
|
+
phone = this.phoneControl.value ?? undefined;
|
|
173
|
+
}
|
|
174
|
+
this.working.set(true);
|
|
175
|
+
this.auth.setupMFA(method, phone).subscribe({
|
|
176
|
+
next: res => {
|
|
177
|
+
this.working.set(false);
|
|
178
|
+
if (res.codeSent) {
|
|
179
|
+
this.pinControl.reset();
|
|
180
|
+
this._step.set('code-confirm');
|
|
181
|
+
this.startCooldown();
|
|
182
|
+
}
|
|
183
|
+
},
|
|
184
|
+
error: err => {
|
|
185
|
+
this.working.set(false);
|
|
186
|
+
this.showToast(this.resolveError(err));
|
|
187
|
+
},
|
|
188
|
+
});
|
|
189
|
+
}
|
|
190
|
+
setupTotp() {
|
|
191
|
+
this.working.set(true);
|
|
192
|
+
this.auth.setupTOTP().subscribe({
|
|
193
|
+
next: res => {
|
|
194
|
+
this.totpSetup.set(res);
|
|
195
|
+
this.pinControl.reset();
|
|
196
|
+
this.qrGen
|
|
197
|
+
.generate({ data: res.qrCodeUrl, width: 220 })
|
|
198
|
+
.then(qr => this.totpQr.set(qr))
|
|
199
|
+
.catch(() => this.totpQr.set(null));
|
|
200
|
+
this.working.set(false);
|
|
201
|
+
this._step.set('totp-setup');
|
|
202
|
+
},
|
|
203
|
+
error: err => {
|
|
204
|
+
this.working.set(false);
|
|
205
|
+
this.showToast(this.resolveError(err));
|
|
206
|
+
},
|
|
207
|
+
});
|
|
208
|
+
}
|
|
209
|
+
/** Verifica el código TOTP de la app de autenticación y activa MFA. */
|
|
210
|
+
verifyTotp() {
|
|
211
|
+
const code = this.pinControl.value;
|
|
212
|
+
if (!code || code.length !== 6) {
|
|
213
|
+
this.showToast(this.t('mfaCodeInvalid'));
|
|
214
|
+
return;
|
|
215
|
+
}
|
|
216
|
+
this.working.set(true);
|
|
217
|
+
this.auth.verifyTOTPSetup(code).subscribe({
|
|
218
|
+
next: res => {
|
|
219
|
+
this.working.set(false);
|
|
220
|
+
if (res.enabled) {
|
|
221
|
+
this.showToast(this.t('mfaEnabledOk'));
|
|
222
|
+
this.changed.emit();
|
|
223
|
+
this.resolveStatus();
|
|
224
|
+
}
|
|
225
|
+
},
|
|
226
|
+
error: err => {
|
|
227
|
+
this.working.set(false);
|
|
228
|
+
this.showToast(this.resolveError(err));
|
|
229
|
+
},
|
|
230
|
+
});
|
|
231
|
+
}
|
|
232
|
+
/** Confirma el código EMAIL/SMS y activa MFA. */
|
|
233
|
+
confirmCode() {
|
|
234
|
+
const code = this.pinControl.value;
|
|
235
|
+
if (!code || code.length !== 6) {
|
|
236
|
+
this.showToast(this.t('mfaCodeInvalid'));
|
|
237
|
+
return;
|
|
238
|
+
}
|
|
239
|
+
this.working.set(true);
|
|
240
|
+
this.auth.confirmMFA(code).subscribe({
|
|
241
|
+
next: res => {
|
|
242
|
+
this.working.set(false);
|
|
243
|
+
if (res.mfaEnabled) {
|
|
244
|
+
this.showToast(this.t('mfaEnabledOk'));
|
|
245
|
+
this.changed.emit();
|
|
246
|
+
this.resolveStatus();
|
|
247
|
+
}
|
|
248
|
+
},
|
|
249
|
+
error: err => {
|
|
250
|
+
this.working.set(false);
|
|
251
|
+
this.showToast(this.resolveError(err));
|
|
252
|
+
},
|
|
253
|
+
});
|
|
254
|
+
}
|
|
255
|
+
/** Reenvía el código EMAIL/SMS (re-ejecuta el setup). */
|
|
256
|
+
resendCode() {
|
|
257
|
+
if (this.resendCooldown() > 0) {
|
|
258
|
+
return;
|
|
259
|
+
}
|
|
260
|
+
this.proceedWithMethod();
|
|
261
|
+
}
|
|
262
|
+
// ===========================================================================
|
|
263
|
+
// Gestión (MFA habilitado)
|
|
264
|
+
// ===========================================================================
|
|
265
|
+
/** Regenera los códigos de respaldo TOTP y los muestra una vez. */
|
|
266
|
+
regenerateBackupCodes() {
|
|
267
|
+
this.working.set(true);
|
|
268
|
+
this.auth.regenerateBackupCodes().subscribe({
|
|
269
|
+
next: res => {
|
|
270
|
+
this.working.set(false);
|
|
271
|
+
this.backupCodesCount.set(res.backupCodes.length);
|
|
272
|
+
this.regeneratedCodes.set(res.backupCodes);
|
|
273
|
+
},
|
|
274
|
+
error: err => {
|
|
275
|
+
this.working.set(false);
|
|
276
|
+
this.showToast(this.resolveError(err));
|
|
277
|
+
},
|
|
278
|
+
});
|
|
279
|
+
}
|
|
280
|
+
/** Deshabilita MFA — requiere la contraseña de la cuenta. */
|
|
281
|
+
confirmDisable() {
|
|
282
|
+
const password = this.passwordControl.value;
|
|
283
|
+
if (!password) {
|
|
284
|
+
this.showToast(this.t('mfaPasswordRequired'));
|
|
285
|
+
return;
|
|
286
|
+
}
|
|
287
|
+
this.working.set(true);
|
|
288
|
+
this.auth.disableMFA(password).subscribe({
|
|
289
|
+
next: res => {
|
|
290
|
+
this.working.set(false);
|
|
291
|
+
if (res.mfaDisabled) {
|
|
292
|
+
this.showToast(this.t('mfaDisabledOk'));
|
|
293
|
+
this.changed.emit();
|
|
294
|
+
this.resolveStatus();
|
|
295
|
+
}
|
|
296
|
+
},
|
|
297
|
+
error: err => {
|
|
298
|
+
this.working.set(false);
|
|
299
|
+
this.showToast(this.resolveError(err));
|
|
300
|
+
},
|
|
301
|
+
});
|
|
302
|
+
}
|
|
303
|
+
/** Copia una lista de códigos de respaldo al portapapeles. */
|
|
304
|
+
async copyCodes(codes) {
|
|
305
|
+
try {
|
|
306
|
+
await navigator.clipboard.writeText(codes.join('\n'));
|
|
307
|
+
this.showToast(this.t('mfaCodesCopied'));
|
|
308
|
+
}
|
|
309
|
+
catch {
|
|
310
|
+
/* sin recurso de copia disponible */
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
/** Etiqueta i18n legible para un método MFA. */
|
|
314
|
+
methodLabel(method) {
|
|
315
|
+
switch (method) {
|
|
316
|
+
case 'TOTP':
|
|
317
|
+
return this.t('mfaMethodTotp');
|
|
318
|
+
case 'EMAIL':
|
|
319
|
+
return this.t('mfaMethodEmail');
|
|
320
|
+
case 'SMS':
|
|
321
|
+
return this.t('mfaMethodSms');
|
|
322
|
+
default:
|
|
323
|
+
return '';
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
// ===========================================================================
|
|
327
|
+
// Helpers
|
|
328
|
+
// ===========================================================================
|
|
329
|
+
resetFlow() {
|
|
330
|
+
this.pinControl.reset();
|
|
331
|
+
this.passwordControl.reset();
|
|
332
|
+
this.phoneControl.reset();
|
|
333
|
+
this.totpSetup.set(null);
|
|
334
|
+
this.totpQr.set(null);
|
|
335
|
+
this.regeneratedCodes.set(null);
|
|
336
|
+
this.selectedMethod.set('TOTP');
|
|
337
|
+
this.stopCooldown();
|
|
338
|
+
}
|
|
339
|
+
startCooldown() {
|
|
340
|
+
this.stopCooldown();
|
|
341
|
+
this.resendCooldown.set(RESEND_COOLDOWN_SECONDS);
|
|
342
|
+
this.resendTimer = setInterval(() => {
|
|
343
|
+
this.resendCooldown.update(v => v - 1);
|
|
344
|
+
if (this.resendCooldown() <= 0) {
|
|
345
|
+
this.stopCooldown();
|
|
346
|
+
}
|
|
347
|
+
}, 1000);
|
|
348
|
+
}
|
|
349
|
+
stopCooldown() {
|
|
350
|
+
if (this.resendTimer) {
|
|
351
|
+
clearInterval(this.resendTimer);
|
|
352
|
+
this.resendTimer = null;
|
|
353
|
+
}
|
|
354
|
+
this.resendCooldown.set(0);
|
|
355
|
+
}
|
|
356
|
+
/** Mapea los códigos de error MFA del backend a mensajes del namespace `_auth`. */
|
|
357
|
+
resolveError(err) {
|
|
358
|
+
const code = err?.code;
|
|
359
|
+
switch (code) {
|
|
360
|
+
case 'AUTHV2_MFA_INVALID_CODE':
|
|
361
|
+
return this.t('mfaErrorInvalidCode');
|
|
362
|
+
case 'AUTHV2_EXPIRED_CODE':
|
|
363
|
+
return this.t('mfaErrorExpiredCode');
|
|
364
|
+
case 'AUTHV2_CODE_USED':
|
|
365
|
+
return this.t('mfaErrorCodeUsed');
|
|
366
|
+
case 'AUTHV2_MFA_ALREADY_ACTIVE':
|
|
367
|
+
return this.t('mfaErrorAlreadyActive');
|
|
368
|
+
case 'AUTHV2_MFA_NOT_ENABLED':
|
|
369
|
+
return this.t('mfaErrorNotEnabled');
|
|
370
|
+
case 'AUTHV2_PHONE_REQUIRED':
|
|
371
|
+
return this.t('mfaErrorPhoneRequired');
|
|
372
|
+
case 'AUTHV2_PHONE_EXISTS':
|
|
373
|
+
return this.t('mfaErrorPhoneExists');
|
|
374
|
+
case 'AUTHV2_TOO_MANY_ATTEMPTS':
|
|
375
|
+
return this.t('errorTooManyAttempts');
|
|
376
|
+
case 'AUTHV2_INVALID_CURRENT_PASSWORD':
|
|
377
|
+
return this.t('errorCurrentPasswordWrong');
|
|
378
|
+
default:
|
|
379
|
+
return this.t('errorGeneric');
|
|
380
|
+
}
|
|
381
|
+
}
|
|
382
|
+
showToast(message) {
|
|
383
|
+
this.toast.show({ message, duration: 3500 });
|
|
384
|
+
}
|
|
385
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: MfaModalComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
386
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: MfaModalComponent, isStandalone: true, selector: "val-mfa-modal", inputs: { isOpen: "isOpen" }, outputs: { changed: "changed", dismissed: "dismissed" }, ngImport: i0, template: "<ion-modal [isOpen]=\"isOpen\" (didDismiss)=\"close()\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"close()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n\n <ion-content class=\"ion-padding\">\n <section class=\"mfa-modal\">\n @switch (step()) { @case ('loading') {\n <div class=\"mfa-loading\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n </div>\n } @case ('status') {\n <h2 class=\"mfa-title\">{{ t('mfaManageTitle') }}</h2>\n\n @if (mfaEnabled()) {\n <p class=\"mfa-status mfa-status--on\">{{ t('mfaEnabledLabel') }} \u00B7 {{ methodLabel(mfaMethod()) }}</p>\n\n @if (mfaMethod() === 'TOTP') {\n <div class=\"mfa-block\">\n <h3>{{ t('mfaBackupCodesTitle') }}</h3>\n <p class=\"mfa-muted\">{{ t('mfaBackupCodesAvailable') }}: {{ backupCodesCount() }}</p>\n @if (backupCodesCount() < 3) {\n <p class=\"mfa-warn\">{{ t('mfaBackupCodesLow') }}</p>\n } @if (regeneratedCodes(); as codes) {\n <p class=\"mfa-warn\">{{ t('mfaBackupCodesSaveWarning') }}</p>\n <div class=\"mfa-codes\">\n @for (code of codes; track code) {\n <code class=\"mfa-code\">{{ code }}</code>\n }\n </div>\n <ion-button expand=\"block\" fill=\"outline\" (click)=\"copyCodes(codes)\"> {{ t('mfaCopyCodes') }} </ion-button>\n } @else {\n <ion-button expand=\"block\" fill=\"outline\" [disabled]=\"working()\" (click)=\"regenerateBackupCodes()\">\n @if (working()) {\n <ion-spinner name=\"crescent\"></ion-spinner>\n } @else { {{ t('mfaRegenerateCodes') }} }\n </ion-button>\n }\n </div>\n }\n\n <ion-button expand=\"block\" color=\"danger\" fill=\"outline\" (click)=\"goToDisable()\">\n {{ t('mfaDisableButton') }}\n </ion-button>\n } @else {\n <p class=\"mfa-status mfa-status--off\">{{ t('mfaDisabledLabel') }}</p>\n <p class=\"mfa-muted\">{{ t('mfaDisabledHint') }}</p>\n <ion-button expand=\"block\" (click)=\"goToMethodSelect()\"> {{ t('mfaEnableButton') }} </ion-button>\n } } @case ('method-select') {\n <h2 class=\"mfa-title\">{{ t('mfaEnableTitle') }}</h2>\n <p class=\"mfa-muted\">{{ t('mfaMethodPrompt') }}</p>\n\n <ion-radio-group [value]=\"selectedMethod()\" (ionChange)=\"selectedMethod.set($event.detail.value)\">\n <ion-item>\n <ion-radio slot=\"start\" value=\"TOTP\"></ion-radio>\n <ion-label>\n <strong>{{ t('mfaMethodTotp') }}</strong>\n <p>{{ t('mfaMethodTotpHint') }}</p>\n </ion-label>\n </ion-item>\n <ion-item>\n <ion-radio slot=\"start\" value=\"EMAIL\"></ion-radio>\n <ion-label>\n <strong>{{ t('mfaMethodEmail') }}</strong>\n <p>{{ t('mfaMethodEmailHint') }}</p>\n </ion-label>\n </ion-item>\n <ion-item>\n <ion-radio slot=\"start\" value=\"SMS\"></ion-radio>\n <ion-label>\n <strong>{{ t('mfaMethodSms') }}</strong>\n <p>{{ t('mfaMethodSmsHint') }}</p>\n </ion-label>\n </ion-item>\n </ion-radio-group>\n\n @if (selectedMethod() === 'SMS' && !userPhone()) {\n <ion-input\n label=\"{{ t('mfaPhoneLabel') }}\"\n labelPlacement=\"floating\"\n type=\"tel\"\n placeholder=\"+56912345678\"\n [formControl]=\"phoneControl\"\n ></ion-input>\n @if (phoneControl.invalid && phoneControl.touched) {\n <p class=\"mfa-error\">{{ t('mfaPhoneInvalid') }}</p>\n } } @if (selectedMethod() === 'SMS' && userPhone(); as phone) {\n <p class=\"mfa-muted\">{{ t('mfaPhoneRegistered') }}: {{ phone }}</p>\n }\n\n <ion-button expand=\"block\" [disabled]=\"working()\" (click)=\"proceedWithMethod()\">\n @if (working()) {\n <ion-spinner name=\"crescent\"></ion-spinner>\n } @else { {{ t('mfaContinue') }} }\n </ion-button>\n <ion-button expand=\"block\" fill=\"clear\" color=\"medium\" (click)=\"backToStatus()\">\n {{ t('mfaCancel') }}\n </ion-button>\n } @case ('totp-setup') {\n <h2 class=\"mfa-title\">{{ t('mfaTotpSetupTitle') }}</h2>\n <p class=\"mfa-muted\">{{ t('mfaTotpStep1') }}</p>\n\n @if (totpQr(); as qr) {\n <div class=\"mfa-qr\">\n <val-qr-code [props]=\"{ qr: qr }\" />\n </div>\n } @if (totpSetup(); as setup) {\n <p class=\"mfa-muted\">{{ t('mfaTotpManualEntry') }}</p>\n <code class=\"mfa-secret\">{{ setup.secret }}</code>\n\n <p class=\"mfa-muted\">{{ t('mfaTotpStep2') }}</p>\n <div class=\"mfa-pin\">\n <val-pin-input [props]=\"pinInputProps\" />\n </div>\n\n <ion-button expand=\"block\" [disabled]=\"working()\" (click)=\"verifyTotp()\">\n @if (working()) {\n <ion-spinner name=\"crescent\"></ion-spinner>\n } @else { {{ t('mfaTotpVerify') }} }\n </ion-button>\n\n <div class=\"mfa-block mfa-block--warn\">\n <p class=\"mfa-warn\">{{ t('mfaBackupCodesSaveWarning') }}</p>\n <p class=\"mfa-muted\">{{ t('mfaBackupCodesExplain') }}</p>\n <div class=\"mfa-codes\">\n @for (code of setup.backupCodes; track code) {\n <code class=\"mfa-code\">{{ code }}</code>\n }\n </div>\n <ion-button expand=\"block\" fill=\"outline\" (click)=\"copyCodes(setup.backupCodes)\">\n {{ t('mfaCopyCodes') }}\n </ion-button>\n </div>\n }\n\n <ion-button expand=\"block\" fill=\"clear\" color=\"medium\" (click)=\"backToStatus()\">\n {{ t('mfaCancel') }}\n </ion-button>\n } @case ('code-confirm') {\n <h2 class=\"mfa-title\">{{ t('mfaConfirmTitle') }}</h2>\n <p class=\"mfa-muted\">\n {{ selectedMethod() === 'EMAIL' ? t('mfaConfirmPromptEmail') : t('mfaConfirmPromptSms') }}\n </p>\n\n <div class=\"mfa-pin\">\n <val-pin-input [props]=\"pinInputProps\" />\n </div>\n\n <ion-button expand=\"block\" [disabled]=\"working()\" (click)=\"confirmCode()\">\n @if (working()) {\n <ion-spinner name=\"crescent\"></ion-spinner>\n } @else { {{ t('mfaConfirmButton') }} }\n </ion-button>\n\n <p class=\"mfa-resend\">\n {{ t('mfaNoCode') }} @if (resendCooldown() > 0) {\n <span class=\"mfa-muted\">{{ t('mfaResendIn') }} {{ resendCooldown() }}s</span>\n } @else {\n <a (click)=\"resendCode()\">{{ t('mfaResend') }}</a>\n }\n </p>\n\n <ion-button expand=\"block\" fill=\"clear\" color=\"medium\" (click)=\"backToStatus()\">\n {{ t('mfaCancel') }}\n </ion-button>\n } @case ('disable') {\n <h2 class=\"mfa-title\">{{ t('mfaDisableTitle') }}</h2>\n <p class=\"mfa-muted\">{{ t('mfaDisablePrompt') }}</p>\n\n <ion-input\n label=\"{{ t('mfaPasswordLabel') }}\"\n labelPlacement=\"floating\"\n type=\"password\"\n [formControl]=\"passwordControl\"\n ></ion-input>\n\n <ion-button\n expand=\"block\"\n color=\"danger\"\n [disabled]=\"working() || passwordControl.invalid\"\n (click)=\"confirmDisable()\"\n >\n @if (working()) {\n <ion-spinner name=\"crescent\"></ion-spinner>\n } @else { {{ t('mfaDisableButton') }} }\n </ion-button>\n <ion-button expand=\"block\" fill=\"clear\" color=\"medium\" (click)=\"backToStatus()\">\n {{ t('mfaCancel') }}\n </ion-button>\n } }\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n", styles: [".mfa-modal{max-width:460px;margin:0 auto;display:flex;flex-direction:column;gap:12px;padding-top:4px}.mfa-loading{display:flex;justify-content:center;padding:40px 0}.mfa-title{font-size:18px;font-weight:700;margin:0;color:var(--ion-color-dark)}.mfa-status{font-weight:600;margin:0}.mfa-status--on{color:var(--ion-color-success-shade)}.mfa-status--off{color:var(--ion-color-dark)}.mfa-muted{color:var(--ion-color-medium);font-size:14px;margin:0}.mfa-warn{color:var(--ion-color-warning-shade);font-size:14px;font-weight:600;margin:0}.mfa-error{color:var(--ion-color-danger);font-size:13px;margin:4px 0 0}.mfa-block{display:flex;flex-direction:column;gap:10px;padding:14px;border-radius:12px;background:var(--ion-color-light)}.mfa-block--warn{background:var(--ion-color-warning-tint, rgba(255, 196, 9, .12));border:1px solid var(--ion-color-warning)}.mfa-block h3{font-size:15px;font-weight:600;margin:0;color:var(--ion-color-dark)}.mfa-qr{display:flex;justify-content:center;padding:12px 0}.mfa-secret{display:block;padding:10px 12px;border-radius:8px;background:var(--ion-color-light);font-family:ui-monospace,SFMono-Regular,Menlo,monospace;font-size:15px;letter-spacing:.04em;word-break:break-all;text-align:center}.mfa-pin{display:flex;justify-content:center;padding:8px 0}.mfa-codes{display:grid;grid-template-columns:repeat(2,1fr);gap:8px}.mfa-code{padding:8px;border-radius:6px;background:var(--ion-background-color, #fff);font-family:ui-monospace,SFMono-Regular,Menlo,monospace;font-size:13px;text-align:center}.mfa-resend{text-align:center;font-size:14px;color:var(--ion-color-medium);margin:4px 0}.mfa-resend a{color:var(--ion-color-primary);cursor:pointer;font-weight:600}\n"], dependencies: [{ kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: IonButtons, selector: "ion-buttons", inputs: ["collapse"] }, { kind: "component", type: IonContent, selector: "ion-content", inputs: ["color", "fixedSlotPlacement", "forceOverscroll", "fullscreen", "scrollEvents", "scrollX", "scrollY"] }, { kind: "component", type: IonHeader, selector: "ion-header", inputs: ["collapse", "mode", "translucent"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonInput, selector: "ion-input", inputs: ["accept", "autocapitalize", "autocomplete", "autocorrect", "autofocus", "clearInput", "clearOnEdit", "color", "counter", "counterFormatter", "debounce", "disabled", "enterkeyhint", "errorText", "fill", "helperText", "inputmode", "label", "labelPlacement", "max", "maxlength", "min", "minlength", "mode", "multiple", "name", "pattern", "placeholder", "readonly", "required", "shape", "size", "spellcheck", "step", "type", "value"] }, { kind: "component", type: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: IonModal, selector: "ion-modal" }, { kind: "component", type: IonRadio, selector: "ion-radio", inputs: ["alignment", "color", "disabled", "justify", "labelPlacement", "mode", "name", "value"] }, { kind: "component", type: IonRadioGroup, selector: "ion-radio-group", inputs: ["allowEmptySelection", "compareWith", "errorText", "helperText", "name", "value"] }, { kind: "component", type: IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }, { kind: "component", type: IonToolbar, selector: "ion-toolbar", inputs: ["color", "mode"] }, { kind: "component", type: QrCodeComponent, selector: "val-qr-code", inputs: ["props"], outputs: ["actionComplete", "imageLoad", "imageError"] }, { kind: "component", type: PinInputComponent, selector: "val-pin-input", inputs: ["props"] }] }); }
|
|
387
|
+
}
|
|
388
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: MfaModalComponent, decorators: [{
|
|
389
|
+
type: Component,
|
|
390
|
+
args: [{ selector: 'val-mfa-modal', standalone: true, imports: [
|
|
391
|
+
ReactiveFormsModule,
|
|
392
|
+
IonButton,
|
|
393
|
+
IonButtons,
|
|
394
|
+
IonContent,
|
|
395
|
+
IonHeader,
|
|
396
|
+
IonIcon,
|
|
397
|
+
IonInput,
|
|
398
|
+
IonItem,
|
|
399
|
+
IonLabel,
|
|
400
|
+
IonModal,
|
|
401
|
+
IonRadio,
|
|
402
|
+
IonRadioGroup,
|
|
403
|
+
IonSpinner,
|
|
404
|
+
IonToolbar,
|
|
405
|
+
QrCodeComponent,
|
|
406
|
+
PinInputComponent,
|
|
407
|
+
], template: "<ion-modal [isOpen]=\"isOpen\" (didDismiss)=\"close()\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"close()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n\n <ion-content class=\"ion-padding\">\n <section class=\"mfa-modal\">\n @switch (step()) { @case ('loading') {\n <div class=\"mfa-loading\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n </div>\n } @case ('status') {\n <h2 class=\"mfa-title\">{{ t('mfaManageTitle') }}</h2>\n\n @if (mfaEnabled()) {\n <p class=\"mfa-status mfa-status--on\">{{ t('mfaEnabledLabel') }} \u00B7 {{ methodLabel(mfaMethod()) }}</p>\n\n @if (mfaMethod() === 'TOTP') {\n <div class=\"mfa-block\">\n <h3>{{ t('mfaBackupCodesTitle') }}</h3>\n <p class=\"mfa-muted\">{{ t('mfaBackupCodesAvailable') }}: {{ backupCodesCount() }}</p>\n @if (backupCodesCount() < 3) {\n <p class=\"mfa-warn\">{{ t('mfaBackupCodesLow') }}</p>\n } @if (regeneratedCodes(); as codes) {\n <p class=\"mfa-warn\">{{ t('mfaBackupCodesSaveWarning') }}</p>\n <div class=\"mfa-codes\">\n @for (code of codes; track code) {\n <code class=\"mfa-code\">{{ code }}</code>\n }\n </div>\n <ion-button expand=\"block\" fill=\"outline\" (click)=\"copyCodes(codes)\"> {{ t('mfaCopyCodes') }} </ion-button>\n } @else {\n <ion-button expand=\"block\" fill=\"outline\" [disabled]=\"working()\" (click)=\"regenerateBackupCodes()\">\n @if (working()) {\n <ion-spinner name=\"crescent\"></ion-spinner>\n } @else { {{ t('mfaRegenerateCodes') }} }\n </ion-button>\n }\n </div>\n }\n\n <ion-button expand=\"block\" color=\"danger\" fill=\"outline\" (click)=\"goToDisable()\">\n {{ t('mfaDisableButton') }}\n </ion-button>\n } @else {\n <p class=\"mfa-status mfa-status--off\">{{ t('mfaDisabledLabel') }}</p>\n <p class=\"mfa-muted\">{{ t('mfaDisabledHint') }}</p>\n <ion-button expand=\"block\" (click)=\"goToMethodSelect()\"> {{ t('mfaEnableButton') }} </ion-button>\n } } @case ('method-select') {\n <h2 class=\"mfa-title\">{{ t('mfaEnableTitle') }}</h2>\n <p class=\"mfa-muted\">{{ t('mfaMethodPrompt') }}</p>\n\n <ion-radio-group [value]=\"selectedMethod()\" (ionChange)=\"selectedMethod.set($event.detail.value)\">\n <ion-item>\n <ion-radio slot=\"start\" value=\"TOTP\"></ion-radio>\n <ion-label>\n <strong>{{ t('mfaMethodTotp') }}</strong>\n <p>{{ t('mfaMethodTotpHint') }}</p>\n </ion-label>\n </ion-item>\n <ion-item>\n <ion-radio slot=\"start\" value=\"EMAIL\"></ion-radio>\n <ion-label>\n <strong>{{ t('mfaMethodEmail') }}</strong>\n <p>{{ t('mfaMethodEmailHint') }}</p>\n </ion-label>\n </ion-item>\n <ion-item>\n <ion-radio slot=\"start\" value=\"SMS\"></ion-radio>\n <ion-label>\n <strong>{{ t('mfaMethodSms') }}</strong>\n <p>{{ t('mfaMethodSmsHint') }}</p>\n </ion-label>\n </ion-item>\n </ion-radio-group>\n\n @if (selectedMethod() === 'SMS' && !userPhone()) {\n <ion-input\n label=\"{{ t('mfaPhoneLabel') }}\"\n labelPlacement=\"floating\"\n type=\"tel\"\n placeholder=\"+56912345678\"\n [formControl]=\"phoneControl\"\n ></ion-input>\n @if (phoneControl.invalid && phoneControl.touched) {\n <p class=\"mfa-error\">{{ t('mfaPhoneInvalid') }}</p>\n } } @if (selectedMethod() === 'SMS' && userPhone(); as phone) {\n <p class=\"mfa-muted\">{{ t('mfaPhoneRegistered') }}: {{ phone }}</p>\n }\n\n <ion-button expand=\"block\" [disabled]=\"working()\" (click)=\"proceedWithMethod()\">\n @if (working()) {\n <ion-spinner name=\"crescent\"></ion-spinner>\n } @else { {{ t('mfaContinue') }} }\n </ion-button>\n <ion-button expand=\"block\" fill=\"clear\" color=\"medium\" (click)=\"backToStatus()\">\n {{ t('mfaCancel') }}\n </ion-button>\n } @case ('totp-setup') {\n <h2 class=\"mfa-title\">{{ t('mfaTotpSetupTitle') }}</h2>\n <p class=\"mfa-muted\">{{ t('mfaTotpStep1') }}</p>\n\n @if (totpQr(); as qr) {\n <div class=\"mfa-qr\">\n <val-qr-code [props]=\"{ qr: qr }\" />\n </div>\n } @if (totpSetup(); as setup) {\n <p class=\"mfa-muted\">{{ t('mfaTotpManualEntry') }}</p>\n <code class=\"mfa-secret\">{{ setup.secret }}</code>\n\n <p class=\"mfa-muted\">{{ t('mfaTotpStep2') }}</p>\n <div class=\"mfa-pin\">\n <val-pin-input [props]=\"pinInputProps\" />\n </div>\n\n <ion-button expand=\"block\" [disabled]=\"working()\" (click)=\"verifyTotp()\">\n @if (working()) {\n <ion-spinner name=\"crescent\"></ion-spinner>\n } @else { {{ t('mfaTotpVerify') }} }\n </ion-button>\n\n <div class=\"mfa-block mfa-block--warn\">\n <p class=\"mfa-warn\">{{ t('mfaBackupCodesSaveWarning') }}</p>\n <p class=\"mfa-muted\">{{ t('mfaBackupCodesExplain') }}</p>\n <div class=\"mfa-codes\">\n @for (code of setup.backupCodes; track code) {\n <code class=\"mfa-code\">{{ code }}</code>\n }\n </div>\n <ion-button expand=\"block\" fill=\"outline\" (click)=\"copyCodes(setup.backupCodes)\">\n {{ t('mfaCopyCodes') }}\n </ion-button>\n </div>\n }\n\n <ion-button expand=\"block\" fill=\"clear\" color=\"medium\" (click)=\"backToStatus()\">\n {{ t('mfaCancel') }}\n </ion-button>\n } @case ('code-confirm') {\n <h2 class=\"mfa-title\">{{ t('mfaConfirmTitle') }}</h2>\n <p class=\"mfa-muted\">\n {{ selectedMethod() === 'EMAIL' ? t('mfaConfirmPromptEmail') : t('mfaConfirmPromptSms') }}\n </p>\n\n <div class=\"mfa-pin\">\n <val-pin-input [props]=\"pinInputProps\" />\n </div>\n\n <ion-button expand=\"block\" [disabled]=\"working()\" (click)=\"confirmCode()\">\n @if (working()) {\n <ion-spinner name=\"crescent\"></ion-spinner>\n } @else { {{ t('mfaConfirmButton') }} }\n </ion-button>\n\n <p class=\"mfa-resend\">\n {{ t('mfaNoCode') }} @if (resendCooldown() > 0) {\n <span class=\"mfa-muted\">{{ t('mfaResendIn') }} {{ resendCooldown() }}s</span>\n } @else {\n <a (click)=\"resendCode()\">{{ t('mfaResend') }}</a>\n }\n </p>\n\n <ion-button expand=\"block\" fill=\"clear\" color=\"medium\" (click)=\"backToStatus()\">\n {{ t('mfaCancel') }}\n </ion-button>\n } @case ('disable') {\n <h2 class=\"mfa-title\">{{ t('mfaDisableTitle') }}</h2>\n <p class=\"mfa-muted\">{{ t('mfaDisablePrompt') }}</p>\n\n <ion-input\n label=\"{{ t('mfaPasswordLabel') }}\"\n labelPlacement=\"floating\"\n type=\"password\"\n [formControl]=\"passwordControl\"\n ></ion-input>\n\n <ion-button\n expand=\"block\"\n color=\"danger\"\n [disabled]=\"working() || passwordControl.invalid\"\n (click)=\"confirmDisable()\"\n >\n @if (working()) {\n <ion-spinner name=\"crescent\"></ion-spinner>\n } @else { {{ t('mfaDisableButton') }} }\n </ion-button>\n <ion-button expand=\"block\" fill=\"clear\" color=\"medium\" (click)=\"backToStatus()\">\n {{ t('mfaCancel') }}\n </ion-button>\n } }\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n", styles: [".mfa-modal{max-width:460px;margin:0 auto;display:flex;flex-direction:column;gap:12px;padding-top:4px}.mfa-loading{display:flex;justify-content:center;padding:40px 0}.mfa-title{font-size:18px;font-weight:700;margin:0;color:var(--ion-color-dark)}.mfa-status{font-weight:600;margin:0}.mfa-status--on{color:var(--ion-color-success-shade)}.mfa-status--off{color:var(--ion-color-dark)}.mfa-muted{color:var(--ion-color-medium);font-size:14px;margin:0}.mfa-warn{color:var(--ion-color-warning-shade);font-size:14px;font-weight:600;margin:0}.mfa-error{color:var(--ion-color-danger);font-size:13px;margin:4px 0 0}.mfa-block{display:flex;flex-direction:column;gap:10px;padding:14px;border-radius:12px;background:var(--ion-color-light)}.mfa-block--warn{background:var(--ion-color-warning-tint, rgba(255, 196, 9, .12));border:1px solid var(--ion-color-warning)}.mfa-block h3{font-size:15px;font-weight:600;margin:0;color:var(--ion-color-dark)}.mfa-qr{display:flex;justify-content:center;padding:12px 0}.mfa-secret{display:block;padding:10px 12px;border-radius:8px;background:var(--ion-color-light);font-family:ui-monospace,SFMono-Regular,Menlo,monospace;font-size:15px;letter-spacing:.04em;word-break:break-all;text-align:center}.mfa-pin{display:flex;justify-content:center;padding:8px 0}.mfa-codes{display:grid;grid-template-columns:repeat(2,1fr);gap:8px}.mfa-code{padding:8px;border-radius:6px;background:var(--ion-background-color, #fff);font-family:ui-monospace,SFMono-Regular,Menlo,monospace;font-size:13px;text-align:center}.mfa-resend{text-align:center;font-size:14px;color:var(--ion-color-medium);margin:4px 0}.mfa-resend a{color:var(--ion-color-primary);cursor:pointer;font-weight:600}\n"] }]
|
|
408
|
+
}], ctorParameters: () => [], propDecorators: { isOpen: [{
|
|
409
|
+
type: Input
|
|
410
|
+
}], changed: [{
|
|
411
|
+
type: Output
|
|
412
|
+
}], dismissed: [{
|
|
413
|
+
type: Output
|
|
414
|
+
}] } });
|
|
415
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWZhLW1vZGFsLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3NyYy9saWIvY29tcG9uZW50cy9vcmdhbmlzbXMvbWZhLW1vZGFsL21mYS1tb2RhbC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvb3JnYW5pc21zL21mYS1tb2RhbC9tZmEtbW9kYWwuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxZQUFZLEVBQUUsTUFBTSxFQUFFLEtBQUssRUFBYSxNQUFNLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2xHLE9BQU8sRUFBRSxXQUFXLEVBQUUsbUJBQW1CLEVBQUUsVUFBVSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDOUUsT0FBTyxFQUNMLFNBQVMsRUFDVCxVQUFVLEVBQ1YsVUFBVSxFQUNWLFNBQVMsRUFDVCxPQUFPLEVBQ1AsUUFBUSxFQUNSLE9BQU8sRUFDUCxRQUFRLEVBQ1IsUUFBUSxFQUNSLFFBQVEsRUFDUixhQUFhLEVBQ2IsVUFBVSxFQUNWLFVBQVUsR0FDWCxNQUFNLDJCQUEyQixDQUFDO0FBQ25DLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxVQUFVLENBQUM7QUFDcEMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBRTlDLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQztBQUN4RSxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSwrQ0FBK0MsQ0FBQztBQUNsRixPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFFckQsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLHdCQUF3QixDQUFDO0FBQ3JELE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLHFEQUFxRCxDQUFDO0FBRXpGLE9BQU8s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LENBQUMsQ0FBQywyQkFBMkIsQ0FBQyxDQUFDO1lBQzdDO2dCQUNFLE9BQU8sSUFBSSxDQUFDLENBQUMsQ0FBQyxjQUFjLENBQUMsQ0FBQztRQUNsQyxDQUFDO0lBQ0gsQ0FBQztJQUVPLFNBQVMsQ0FBQyxPQUFlO1FBQy9CLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLEVBQUUsT0FBTyxFQUFFLFFBQVEsRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDO0lBQy9DLENBQUM7K0dBNVhVLGlCQUFpQjttR0FBakIsaUJBQWlCLGdLQ3hGOUIsdTdQQXlNQSwyc0REcklJLG1CQUFtQiwwVEFDbkIsU0FBUyxvUEFDVCxVQUFVLDhFQUNWLFVBQVUsd0tBQ1YsU0FBUyxvR0FDVCxPQUFPLDJKQUNQLFFBQVEsOGVBQ1IsT0FBTywwTkFDUCxRQUFRLDZGQUNSLFFBQVEsc0RBQ1IsUUFBUSx3SkFDUixhQUFhLHdKQUNiLFVBQVUseUdBQ1YsVUFBVSxtRkFDVixlQUFlLG1JQUNmLGlCQUFpQjs7NEZBS1IsaUJBQWlCO2tCQXhCN0IsU0FBUzsrQkFDRSxlQUFlLGNBQ2IsSUFBSSxXQUNQO3dCQUNQLG1CQUFtQjt3QkFDbkIsU0FBUzt3QkFDVCxVQUFVO3dCQUNWLFVBQVU7d0JBQ1YsU0FBUzt3QkFDVCxPQUFPO3dCQUNQLFFBQVE7d0JBQ1IsT0FBTzt3QkFDUCxRQUFRO3dCQUNSLFFBQVE7d0JBQ1IsUUFBUTt3QkFDUixhQUFhO3dCQUNiLFVBQVU7d0JBQ1YsVUFBVTt3QkFDVixlQUFlO3dCQUNmLGlCQUFpQjtxQkFDbEI7d0RBU0csTUFBTTtzQkFEVCxLQUFLO2dCQWFJLE9BQU87c0JBQWhCLE1BQU07Z0JBR0csU0FBUztzQkFBbEIsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgRXZlbnRFbWl0dGVyLCBpbmplY3QsIElucHV0LCBPbkRlc3Ryb3ksIE91dHB1dCwgc2lnbmFsIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBGb3JtQ29udHJvbCwgUmVhY3RpdmVGb3Jtc01vZHVsZSwgVmFsaWRhdG9ycyB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcbmltcG9ydCB7XG4gIElvbkJ1dHRvbixcbiAgSW9uQnV0dG9ucyxcbiAgSW9uQ29udGVudCxcbiAgSW9uSGVhZGVyLFxuICBJb25JY29uLFxuICBJb25JbnB1dCxcbiAgSW9uSXRlbSxcbiAgSW9uTGFiZWwsXG4gIElvbk1vZGFsLFxuICBJb25SYWRpbyxcbiAgSW9uUmFkaW9Hcm91cCxcbiAgSW9uU3Bpbm5lcixcbiAgSW9uVG9vbGJhcixcbn0gZnJvbSAnQGlvbmljL2FuZ3VsYXIvc3RhbmRhbG9uZSc7XG5pbXBvcnQgeyBhZGRJY29ucyB9IGZyb20gJ2lvbmljb25zJztcbmltcG9ydCB7IGNsb3NlT3V0bGluZSB9IGZyb20gJ2lvbmljb25zL2ljb25zJztcblxuaW1wb3J0IHsgUXJDb2RlQ29tcG9uZW50IH0gZnJvbSAnLi4vLi4vYXRvbXMvcXItY29kZS9xci1jb2RlLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBQaW5JbnB1dENvbXBvbmVudCB9IGZyb20gJy4uLy4uL21vbGVjdWxlcy9waW4taW5wdXQvcGluLWlucHV0LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBBdXRoU2VydmljZSB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL2F1dGgnO1xuaW1wb3J0IHsgTUZBTWV0aG9kLCBUT1RQU2V0dXBSZXNwb25zZSB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL2F1dGgvdHlwZXMnO1xuaW1wb3J0IHsgSTE4blNlcnZpY2UgfSBmcm9tICcuLi8uLi8uLi9zZXJ2aWNlcy9pMThuJztcbmltcG9ydCB7IFFyR2VuZXJhdG9yU2VydmljZSB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL3FyLWdlbmVyYXRvci9xci1nZW5lcmF0b3Iuc2VydmljZSc7XG5pbXBvcnQgeyBRclJlc3VsdCB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL3FyLWdlbmVyYXRvci90eXBlcyc7XG5pbXBvcnQgeyBUb2FzdFNlcnZpY2UgfSBmcm9tICcuLi8uLi8uLi9zZXJ2aWNlcy90b2FzdC5zZXJ2aWNlJztcblxuLyoqIFBhc28gZGVsIGZsdWpvIGRlbCBtb2RhbC4gKi9cbnR5cGUgTWZhU3RlcCA9ICdsb2FkaW5nJyB8ICdzdGF0dXMnIHwgJ21ldGhvZC1zZWxlY3QnIHwgJ3RvdHAtc2V0dXAnIHwgJ2NvZGUtY29uZmlybScgfCAnZGlzYWJsZSc7XG5cbi8qKiBTZWd1bmRvcyBkZSBlc3BlcmEgYW50ZXMgZGUgcG9kZXIgcmVlbnZpYXIgZWwgY8OzZGlnbyBFTUFJTC9TTVMuICovXG5jb25zdCBSRVNFTkRfQ09PTERPV05fU0VDT05EUyA9IDMwO1xuXG4vKipcbiAqIGB2YWwtbWZhLW1vZGFsYCDigJQgbW9kYWwgZGUgZ2VzdGnDs24gZGUgYXV0ZW50aWNhY2nDs24gZGUgZG9zIGZhY3RvcmVzIChNRkEpXG4gKiBwYXJhIHVuIHVzdWFyaW8gYXV0ZW50aWNhZG8uIE1pc21vIHBhdHLDs24gcXVlIGB2YWwtY2hhbmdlLXBhc3N3b3JkLW1vZGFsYC5cbiAqXG4gKiBGbHVqbyAobcOhcXVpbmEgZGUgZXN0YWRvcyBpbnRlcm5hKTpcbiAqICAtIGBsb2FkaW5nYCDihpIgYGdldFByb2ZpbGUoKWAgcGFyYSBjb25vY2VyIGVsIGVzdGFkbyBNRkEuXG4gKiAgLSBgc3RhdHVzYCDihpIgbXVlc3RyYSBNRkEgaGFiaWxpdGFkby9kZXNoYWJpbGl0YWRvLiBTaSBlc3TDoSBoYWJpbGl0YWRvOlxuICogICAgZ2VzdGnDs24gZGUgYmFja3VwIGNvZGVzIChUT1RQKSArIGRlc2hhYmlsaXRhci4gU2kgbm86IGJvdMOzbiBoYWJpbGl0YXIuXG4gKiAgLSBgbWV0aG9kLXNlbGVjdGAg4oaSIGVsZWdpciBUT1RQIC8gRU1BSUwgLyBTTVMuXG4gKiAgLSBgdG90cC1zZXR1cGAg4oaSIFFSICsgc2VjcmV0byBtYW51YWwgKyBiYWNrdXAgY29kZXMg4oaSIHZlcmlmaWNhciBjw7NkaWdvLlxuICogIC0gYGNvZGUtY29uZmlybWAg4oaSIChFTUFJTC9TTVMpIGluZ3Jlc2FyIGPDs2RpZ28gcmVjaWJpZG8sIGNvbiByZWVudsOtby5cbiAqICAtIGBkaXNhYmxlYCDihpIgY29udHJhc2XDsWEgcGFyYSBkZXNoYWJpbGl0YXIgTUZBLlxuICpcbiAqIEVsIFFSIHNlIGdlbmVyYSAqKmNsaWVudC1zaWRlKiogKGBRckdlbmVyYXRvclNlcnZpY2VgKSDigJQgZWwgc2VjcmV0byBUT1RQXG4gKiBudW5jYSBzYWxlIGRlbCBuYXZlZ2Fkb3IuXG4gKlxuICogU2VsZi1jb250YWluZWQ6IGlueWVjdGEgYEF1dGhTZXJ2aWNlYCB5IGxsYW1hIGxvcyBlbmRwb2ludHMgZGlyZWN0by4gTGEgYXBwXG4gKiBjb250cm9sYSBgW2lzT3Blbl1gIHkgcmVhY2Npb25hIGEgYChjaGFuZ2VkKWAgLyBgKGRpc21pc3NlZClgLlxuICpcbiAqIGkxOG46IG5hbWVzcGFjZSBjb21wYXJ0aWRvIGBfYXV0aGAuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYGh0bWxcbiAqIDx2YWwtbWZhLW1vZGFsXG4gKiAgIFtpc09wZW5dPVwiaXNNb2RhbE9wZW4oKVwiXG4gKiAgIChkaXNtaXNzZWQpPVwiaXNNb2RhbE9wZW4uc2V0KGZhbHNlKVwiXG4gKiAvPlxuICogYGBgXG4gKi9cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3ZhbC1tZmEtbW9kYWwnLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbXG4gICAgUmVhY3RpdmVGb3Jtc01vZHVsZSxcbiAgICBJb25CdXR0b24sXG4gICAgSW9uQnV0dG9ucyxcbiAgICBJb25Db250ZW50LFxuICAgIElvbkhlYWRlcixcbiAgICBJb25JY29uLFxuICAgIElvbklucHV0LFxuICAgIElvbkl0ZW0sXG4gICAgSW9uTGFiZWwsXG4gICAgSW9uTW9kYWwsXG4gICAgSW9uUmFkaW8sXG4gICAgSW9uUmFkaW9Hcm91cCxcbiAgICBJb25TcGlubmVyLFxuICAgIElvblRvb2xiYXIsXG4gICAgUXJDb2RlQ29tcG9uZW50LFxuICAgIFBpbklucHV0Q29tcG9uZW50LFxuICBdLFxuICB0ZW1wbGF0ZVVybDogJy4vbWZhLW1vZGFsLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vbWZhLW1vZGFsLmNvbXBvbmVudC5zY3NzJ10sXG59KVxuZXhwb3J0IGNsYXNzIE1mYU1vZGFsQ29tcG9uZW50IGltcGxlbWVudHMgT25EZXN0cm95IHtcbiAgcHJpdmF0ZSBfaXNPcGVuID0gZmFsc2U7XG5cbiAgLyoqIENvbnRyb2xhIGxhIHZpc2liaWxpZGFkLiBDYWRhIGFwZXJ0dXJhIHJlLXJlc3VlbHZlIGVsIGVzdGFkbyBNRkEuICovXG4gIEBJbnB1dCgpXG4gIHNldCBpc09wZW4odmFsdWU6IGJvb2xlYW4pIHtcbiAgICBjb25zdCBvcGVuaW5nID0gdmFsdWUgJiYgIXRoaXMuX2lzT3BlbjtcbiAgICB0aGlzLl9pc09wZW4gPSB2YWx1ZTtcbiAgICBpZiAob3BlbmluZykge1xuICAgICAgdGhpcy5yZXNvbHZlU3RhdHVzKCk7XG4gICAgfVxuICB9XG4gIGdldCBpc09wZW4oKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMuX2lzT3BlbjtcbiAgfVxuXG4gIC8qKiBFbWl0ZSBjdWFuZG8gZWwgZXN0YWRvIE1GQSBjYW1iaWEgKGhhYmlsaXRhZG8gLyBkZXNoYWJpbGl0YWRvKS4gKi9cbiAgQE91dHB1dCgpIGNoYW5nZWQgPSBuZXcgRXZlbnRFbWl0dGVyPHZvaWQ+KCk7XG5cbiAgLyoqIEVtaXRlIGN1YW5kbyBlbCB1c2VyIGNpZXJyYSBlbCBtb2RhbCAoYm90w7NuIFggbyBiYWNrZHJvcCkuICovXG4gIEBPdXRwdXQoKSBkaXNtaXNzZWQgPSBuZXcgRXZlbnRFbWl0dGVyPHZvaWQ+KCk7XG5cbiAgcHJpdmF0ZSBhdXRoID0gaW5qZWN0KEF1dGhTZXJ2aWNlKTtcbiAgcHJpdmF0ZSB0b2FzdCA9IGluamVjdChUb2FzdFNlcnZpY2UpO1xuICBwcml2YXRlIGkxOG4gPSBpbmplY3QoSTE4blNlcnZpY2UpO1xuICBwcml2YXRlIHFyR2VuID0gaW5qZWN0KFFyR2VuZXJhdG9yU2VydmljZSk7XG5cbiAgcHJpdmF0ZSByZWFkb25seSBfc3RlcCA9IHNpZ25hbDxNZmFTdGVwPignbG9hZGluZycpO1xuICAvKiogUGFzbyBhY3R1YWwgZGVsIGZsdWpvLiAqL1xuICByZWFkb25seSBzdGVwID0gdGhpcy5fc3RlcC5hc1JlYWRvbmx5KCk7XG5cbiAgLyoqIGB0cnVlYCBtaWVudHJhcyB1bmEgbGxhbWFkYSBhbCBiYWNrZW5kIGVzdMOhIGVuIGN1cnNvLiAqL1xuICByZWFkb25seSB3b3JraW5nID0gc2lnbmFsKGZhbHNlKTtcblxuICAvLyBFc3RhZG8gTUZBIGFjdHVhbC5cbiAgcmVhZG9ubHkgbWZhRW5hYmxlZCA9IHNpZ25hbChmYWxzZSk7XG4gIHJlYWRvbmx5IG1mYU1ldGhvZCA9IHNpZ25hbDxNRkFNZXRob2QgfCBudWxsPihudWxsKTtcbiAgcmVhZG9ubHkgdXNlclBob25lID0gc2lnbmFsPHN0cmluZyB8IG51bGw+KG51bGwpO1xuICByZWFkb25seSBiYWNrdXBDb2Rlc0NvdW50ID0gc2lnbmFsKDApO1xuXG4gIC8vIEVzdGFkbyBkZWwgZmx1am8gZGUgaGFiaWxpdGFjacOzbi5cbiAgcmVhZG9ubHkgc2VsZWN0ZWRNZXRob2QgPSBzaWduYWw8TUZBTWV0aG9kPignVE9UUCcpO1xuICByZWFkb25seSB0b3RwU2V0dXAgPSBzaWduYWw8VE9UUFNldHVwUmVzcG9uc2UgfCBudWxsPihudWxsKTtcbiAgcmVhZG9ubHkgdG90cFFyID0gc2lnbmFsPFFyUmVzdWx0IHwgbnVsbD4obnVsbCk7XG4gIC8qKiBDw7NkaWdvcyBkZSByZXNwYWxkbyByZWNpw6luIHJlZ2VuZXJhZG9zIOKAlCBzZSBtdWVzdHJhbiB1bmEgc29sYSB2ZXouICovXG4gIHJlYWRvbmx5IHJlZ2VuZXJhdGVkQ29kZXMgPSBzaWduYWw8c3RyaW5nW10gfCBudWxsPihudWxsKTtcbiAgcmVhZG9ubHkgcmVzZW5kQ29vbGRvd24gPSBzaWduYWwoMCk7XG5cbiAgcmVhZG9ubHkgcGluQ29udHJvbCA9IG5ldyBGb3JtQ29udHJvbCgnJywgW1ZhbGlkYXRvcnMucmVxdWlyZWQsIFZhbGlkYXRvcnMubWluTGVuZ3RoKDYpLCBWYWxpZGF0b3JzLm1heExlbmd0aCg2KV0pO1xuICByZWFkb25seSBwYXNzd29yZENvbnRyb2wgPSBuZXcgRm9ybUNvbnRyb2woJycsIFtWYWxpZGF0b3JzLnJlcXVpcmVkXSk7XG4gIHJlYWRvbmx5IHBob25lQ29udHJvbCA9IG5ldyBGb3JtQ29udHJvbCgnJywgW1ZhbGlkYXRvcnMucmVxdWlyZWQsIFZhbGlkYXRvcnMucGF0dGVybigvXlxcK1sxLTldXFxkezYsMTR9JC8pXSk7XG5cbiAgcmVhZG9ubHkgcGluSW5wdXRQcm9wcyA9IHtcbiAgICBjb250cm9sOiB0aGlzLnBpbkNvbnRyb2wsXG4gICAgdG9rZW46ICdtZmEtY29kZScsXG4gICAgbGVuZ3RoOiA2LFxuICAgIGFsbG93TnVtYmVyc09ubHk6IHRydWUsXG4gICAgYXV0b0ZvY3VzOiB0cnVlLFxuICB9O1xuXG4gIHByaXZhdGUgcmVzZW5kVGltZXI6IFJldHVyblR5cGU8dHlwZW9mIHNldEludGVydmFsPiB8IG51bGwgPSBudWxsO1xuXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIGFkZEljb25zKHsgY2xvc2VPdXRsaW5lIH0pO1xuICB9XG5cbiAgbmdPbkRlc3Ryb3koKTogdm9pZCB7XG4gICAgdGhpcy5zdG9wQ29vbGRvd24oKTtcbiAgfVxuXG4gIC8qKiBUcmFkdWNlIHVuYSBjbGF2ZSBkZWwgbmFtZXNwYWNlIGBfYXV0aGAuICovXG4gIHQoa2V5OiBzdHJpbmcpOiBzdHJpbmcge1xuICAgIHJldHVybiB0aGlzLmkxOG4udChrZXksICdfYXV0aCcpO1xuICB9XG5cbiAgLyoqIENpZXJyZSBpbmljaWFkbyBwb3IgZWwgdXNlciAoWCAvIGJhY2tkcm9wKS4gKi9cbiAgY2xvc2UoKTogdm9pZCB7XG4gICAgdGhpcy5kaXNtaXNzZWQuZW1pdCgpO1xuICB9XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIENhcmdhIGRlIGVzdGFkb1xuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuICAvKiogQ29uc3VsdGEgZWwgcGVyZmlsIHBhcmEgY29ub2NlciBlbCBlc3RhZG8gTUZBIHkgcG9zaWNpb25hciBlbCBmbHVqby4gKi9cbiAgcHJpdmF0ZSByZXNvbHZlU3RhdHVzKCk6IHZvaWQge1xuICAgIHRoaXMuX3N0ZXAuc2V0KCdsb2FkaW5nJyk7XG4gICAgdGhpcy5yZXNldEZsb3coKTtcbiAgICB0aGlzLmF1dGguZ2V0UHJvZmlsZSgpLnN1YnNjcmliZSh7XG4gICAgICBuZXh0OiBwcm9maWxlID0+IHtcbiAgICAgICAgdGhpcy5tZmFFbmFibGVkLnNldChwcm9maWxlLm1mYUVuYWJsZWQpO1xuICAgICAgICB0aGlzLm1mYU1ldGhvZC5zZXQocHJvZmlsZS5tZmFNZXRob2QgPz8gbnVsbCk7XG4gICAgICAgIHRoaXMudXNlclBob25lLnNldChwcm9maWxlLnBob25lID8/IG51bGwpO1xuICAgICAgICBpZiAocHJvZmlsZS5tZmFFbmFibGVkICYmIHByb2ZpbGUubWZhTWV0aG9kID09PSAnVE9UUCcpIHtcbiAgICAgICAgICB0aGlzLmxvYWRCYWNrdXBDb3VudCgpO1xuICAgICAgICB9XG4gICAgICAgIHRoaXMuX3N0ZXAuc2V0KCdzdGF0dXMnKTtcbiAgICAgIH0sXG4gICAgICBlcnJvcjogKCkgPT4ge1xuICAgICAgICAvLyBGYWxsYmFjazogdXNhciBlbCBzaWduYWwgZGUgdXN1YXJpbyBlbiBzZXNpw7NuLlxuICAgICAgICBjb25zdCB1c2VyID0gdGhpcy5hdXRoLnVzZXIoKTtcbiAgICAgICAgdGhpcy5tZmFFbmFibGVkLnNldCh1c2VyPy5tZmFFbmFibGVkID8/IGZhbHNlKTtcbiAgICAgICAgdGhpcy5tZmFNZXRob2Quc2V0KHVzZXI/Lm1mYU1ldGhvZCA/PyBudWxsKTtcbiAgICAgICAgdGhpcy5fc3RlcC5zZXQoJ3N0YXR1cycpO1xuICAgICAgfSxcbiAgICB9KTtcbiAgfVxuXG4gIHByaXZhdGUgbG9hZEJhY2t1cENvdW50KCk6IHZvaWQge1xuICAgIHRoaXMuYXV0aC5nZXRCYWNrdXBDb2Rlc0NvdW50KCkuc3Vic2NyaWJlKHtcbiAgICAgIG5leHQ6IHJlcyA9PiB0aGlzLmJhY2t1cENvZGVzQ291bnQuc2V0KHJlcy5jb3VudCksXG4gICAgICBlcnJvcjogKCkgPT4gdGhpcy5iYWNrdXBDb2Rlc0NvdW50LnNldCgwKSxcbiAgICB9KTtcbiAgfVxuXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuICAvLyBOYXZlZ2FjacOzbiBlbnRyZSBwYXNvc1xuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuICBnb1RvTWV0aG9kU2VsZWN0KCk6IHZvaWQge1xuICAgIHRoaXMucmVnZW5lcmF0ZWRDb2Rlcy5zZXQobnVsbCk7XG4gICAgdGhpcy5fc3RlcC5zZXQoJ21ldGhvZC1zZWxlY3QnKTtcbiAgfVxuXG4gIGdvVG9EaXNhYmxlKCk6IHZvaWQge1xuICAgIHRoaXMucGFzc3dvcmRDb250cm9sLnJlc2V0KCk7XG4gICAgdGhpcy5fc3RlcC5zZXQoJ2Rpc2FibGUnKTtcbiAgfVxuXG4gIGJhY2tUb1N0YXR1cygpOiB2b2lkIHtcbiAgICB0aGlzLnN0b3BDb29sZG93bigpO1xuICAgIHRoaXMucmVzb2x2ZVN0YXR1cygpO1xuICB9XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIEhhYmlsaXRhciBNRkFcbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgLyoqIENvbnRpbsO6YSBkZXNkZSBlbCBzZWxlY3RvciBkZSBtw6l0b2RvIGFsIHNldHVwIGNvcnJlc3BvbmRpZW50ZS4gKi9cbiAgcHJvY2VlZFdpdGhNZXRob2QoKTogdm9pZCB7XG4gICAgY29uc3QgbWV0aG9kID0gdGhpcy5zZWxlY3RlZE1ldGhvZCgpO1xuICAgIGlmIChtZXRob2QgPT09ICdUT1RQJykge1xuICAgICAgdGhpcy5zZXR1cFRvdHAoKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBsZXQgcGhvbmU6IHN0cmluZyB8IHVuZGVmaW5lZDtcbiAgICBpZiAobWV0aG9kID09PSAnU01TJyAmJiAhdGhpcy51c2VyUGhvbmUoKSkge1xuICAgICAgaWYgKHRoaXMucGhvbmVDb250cm9sLmludmFsaWQpIHtcbiAgICAgICAgdGhpcy5waG9uZUNvbnRyb2wubWFya0FzVG91Y2hlZCgpO1xuICAgICAgICB0aGlzLnNob3dUb2FzdCh0aGlzLnQoJ21mYVBob25lSW52YWxpZCcpKTtcbiAgICAgICAgcmV0dXJuO1xuICAgICAgfVxuICAgICAgcGhvbmUgPSB0aGlzLnBob25lQ29udHJvbC52YWx1ZSA/PyB1bmRlZmluZWQ7XG4gICAgfVxuXG4gICAgdGhpcy53b3JraW5nLnNldCh0cnVlKTtcbiAgICB0aGlzLmF1dGguc2V0dXBNRkEobWV0aG9kLCBwaG9uZSkuc3Vic2NyaWJlKHtcbiAgICAgIG5leHQ6IHJlcyA9PiB7XG4gICAgICAgIHRoaXMud29ya2luZy5zZXQoZmFsc2UpO1xuICAgICAgICBpZiAocmVzLmNvZGVTZW50KSB7XG4gICAgICAgICAgdGhpcy5waW5Db250cm9sLnJlc2V0KCk7XG4gICAgICAgICAgdGhpcy5fc3RlcC5zZXQoJ2NvZGUtY29uZmlybScpO1xuICAgICAgICAgIHRoaXMuc3RhcnRDb29sZG93bigpO1xuICAgICAgICB9XG4gICAgICB9LFxuICAgICAgZXJyb3I6IGVyciA9PiB7XG4gICAgICAgIHRoaXMud29ya2luZy5zZXQoZmFsc2UpO1xuICAgICAgICB0aGlzLnNob3dUb2FzdCh0aGlzLnJlc29sdmVFcnJvcihlcnIpKTtcbiAgICAgIH0sXG4gICAgfSk7XG4gIH1cblxuICBwcml2YXRlIHNldHVwVG90cCgpOiB2b2lkIHtcbiAgICB0aGlzLndvcmtpbmcuc2V0KHRydWUpO1xuICAgIHRoaXMuYXV0aC5zZXR1cFRPVFAoKS5zdWJzY3JpYmUoe1xuICAgICAgbmV4dDogcmVzID0+IHtcbiAgICAgICAgdGhpcy50b3RwU2V0dXAuc2V0KHJlcyk7XG4gICAgICAgIHRoaXMucGluQ29udHJvbC5yZXNldCgpO1xuICAgICAgICB0aGlzLnFyR2VuXG4gICAgICAgICAgLmdlbmVyYXRlKHsgZGF0YTogcmVzLnFyQ29kZVVybCwgd2lkdGg6IDIyMCB9KVxuICAgICAgICAgIC50aGVuKHFyID0+IHRoaXMudG90cFFyLnNldChxcikpXG4gICAgICAgICAgLmNhdGNoKCgpID0+IHRoaXMudG90cFFyLnNldChudWxsKSk7XG4gICAgICAgIHRoaXMud29ya2luZy5zZXQoZmFsc2UpO1xuICAgICAgICB0aGlzLl9zdGVwLnNldCgndG90cC1zZXR1cCcpO1xuICAgICAgfSxcbiAgICAgIGVycm9yOiBlcnIgPT4ge1xuICAgICAgICB0aGlzLndvcmtpbmcuc2V0KGZhbHNlKTtcbiAgICAgICAgdGhpcy5zaG93VG9hc3QodGhpcy5yZXNvbHZlRXJyb3IoZXJyKSk7XG4gICAgICB9LFxuICAgIH0pO1xuICB9XG5cbiAgLyoqIFZlcmlmaWNhIGVsIGPDs2RpZ28gVE9UUCBkZSBsYSBhcHAgZGUgYXV0ZW50aWNhY2nDs24geSBhY3RpdmEgTUZBLiAqL1xuICB2ZXJpZnlUb3RwKCk6IHZvaWQge1xuICAgIGNvbnN0IGNvZGUgPSB0aGlzLnBpbkNvbnRyb2wudmFsdWU7XG4gICAgaWYgKCFjb2RlIHx8IGNvZGUubGVuZ3RoICE9PSA2KSB7XG4gICAgICB0aGlzLnNob3dUb2FzdCh0aGlzLnQoJ21mYUNvZGVJbnZhbGlkJykpO1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aGlzLndvcmtpbmcuc2V0KHRydWUpO1xuICAgIHRoaXMuYXV0aC52ZXJpZnlUT1RQU2V0dXAoY29kZSkuc3Vic2NyaWJlKHtcbiAgICAgIG5leHQ6IHJlcyA9PiB7XG4gICAgICAgIHRoaXMud29ya2luZy5zZXQoZmFsc2UpO1xuICAgICAgICBpZiAocmVzLmVuYWJsZWQpIHtcbiAgICAgICAgICB0aGlzLnNob3dUb2FzdCh0aGlzLnQoJ21mYUVuYWJsZWRPaycpKTtcbiAgICAgICAgICB0aGlzLmNoYW5nZWQuZW1pdCgpO1xuICAgICAgICAgIHRoaXMucmVzb2x2ZVN0YXR1cygpO1xuICAgICAgICB9XG4gICAgICB9LFxuICAgICAgZXJyb3I6IGVyciA9PiB7XG4gICAgICAgIHRoaXMud29ya2luZy5zZXQoZmFsc2UpO1xuICAgICAgICB0aGlzLnNob3dUb2FzdCh0aGlzLnJlc29sdmVFcnJvcihlcnIpKTtcbiAgICAgIH0sXG4gICAgfSk7XG4gIH1cblxuICAvKiogQ29uZmlybWEgZWwgY8OzZGlnbyBFTUFJTC9TTVMgeSBhY3RpdmEgTUZBLiAqL1xuICBjb25maXJtQ29kZSgpOiB2b2lkIHtcbiAgICBjb25zdCBjb2RlID0gdGhpcy5waW5Db250cm9sLnZhbHVlO1xuICAgIGlmICghY29kZSB8fCBjb2RlLmxlbmd0aCAhPT0gNikge1xuICAgICAgdGhpcy5zaG93VG9hc3QodGhpcy50KCdtZmFDb2RlSW52YWxpZCcpKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhpcy53b3JraW5nLnNldCh0cnVlKTtcbiAgICB0aGlzLmF1dGguY29uZmlybU1GQShjb2RlKS5zdWJzY3JpYmUoe1xuICAgICAgbmV4dDogcmVzID0+IHtcbiAgICAgICAgdGhpcy53b3JraW5nLnNldChmYWxzZSk7XG4gICAgICAgIGlmIChyZXMubWZhRW5hYmxlZCkge1xuICAgICAgICAgIHRoaXMuc2hvd1RvYXN0KHRoaXMudCgnbWZhRW5hYmxlZE9rJykpO1xuICAgICAgICAgIHRoaXMuY2hhbmdlZC5lbWl0KCk7XG4gICAgICAgICAgdGhpcy5yZXNvbHZlU3RhdHVzKCk7XG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgICBlcnJvcjogZXJyID0+IHtcbiAgICAgICAgdGhpcy53b3JraW5nLnNldChmYWxzZSk7XG4gICAgICAgIHRoaXMuc2hvd1RvYXN0KHRoaXMucmVzb2x2ZUVycm9yKGVycikpO1xuICAgICAgfSxcbiAgICB9KTtcbiAgfVxuXG4gIC8qKiBSZWVudsOtYSBlbCBjw7NkaWdvIEVNQUlML1NNUyAocmUtZWplY3V0YSBlbCBzZXR1cCkuICovXG4gIHJlc2VuZENvZGUoKTogdm9pZCB7XG4gICAgaWYgKHRoaXMucmVzZW5kQ29vbGRvd24oKSA+IDApIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhpcy5wcm9jZWVkV2l0aE1ldGhvZCgpO1xuICB9XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIEdlc3Rpw7NuIChNRkEgaGFiaWxpdGFkbylcbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgLyoqIFJlZ2VuZXJhIGxvcyBjw7NkaWdvcyBkZSByZXNwYWxkbyBUT1RQIHkgbG9zIG11ZXN0cmEgdW5hIHZlei4gKi9cbiAgcmVnZW5lcmF0ZUJhY2t1cENvZGVzKCk6IHZvaWQge1xuICAgIHRoaXMud29ya2luZy5zZXQodHJ1ZSk7XG4gICAgdGhpcy5hdXRoLnJlZ2VuZXJhdGVCYWNrdXBDb2RlcygpLnN1YnNjcmliZSh7XG4gICAgICBuZXh0OiByZXMgPT4ge1xuICAgICAgICB0aGlzLndvcmtpbmcuc2V0KGZhbHNlKTtcbiAgICAgICAgdGhpcy5iYWNrdXBDb2Rlc0NvdW50LnNldChyZXMuYmFja3VwQ29kZXMubGVuZ3RoKTtcbiAgICAgICAgdGhpcy5yZWdlbmVyYXRlZENvZGVzLnNldChyZXMuYmFja3VwQ29kZXMpO1xuICAgICAgfSxcbiAgICAgIGVycm9yOiBlcnIgPT4ge1xuICAgICAgICB0aGlzLndvcmtpbmcuc2V0KGZhbHNlKTtcbiAgICAgICAgdGhpcy5zaG93VG9hc3QodGhpcy5yZXNvbHZlRXJyb3IoZXJyKSk7XG4gICAgICB9LFxuICAgIH0pO1xuICB9XG5cbiAgLyoqIERlc2hhYmlsaXRhIE1GQSDigJQgcmVxdWllcmUgbGEgY29udHJhc2XDsWEgZGUgbGEgY3VlbnRhLiAqL1xuICBjb25maXJtRGlzYWJsZSgpOiB2b2lkIHtcbiAgICBjb25zdCBwYXNzd29yZCA9IHRoaXMucGFzc3dvcmRDb250cm9sLnZhbHVlO1xuICAgIGlmICghcGFzc3dvcmQpIHtcbiAgICAgIHRoaXMuc2hvd1RvYXN0KHRoaXMudCgnbWZhUGFzc3dvcmRSZXF1aXJlZCcpKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhpcy53b3JraW5nLnNldCh0cnVlKTtcbiAgICB0aGlzLmF1dGguZGlzYWJsZU1GQShwYXNzd29yZCkuc3Vic2NyaWJlKHtcbiAgICAgIG5leHQ6IHJlcyA9PiB7XG4gICAgICAgIHRoaXMud29ya2luZy5zZXQoZmFsc2UpO1xuICAgICAgICBpZiAocmVzLm1mYURpc2FibGVkKSB7XG4gICAgICAgICAgdGhpcy5zaG93VG9hc3QodGhpcy50KCdtZmFEaXNhYmxlZE9rJykpO1xuICAgICAgICAgIHRoaXMuY2hhbmdlZC5lbWl0KCk7XG4gICAgICAgICAgdGhpcy5yZXNvbHZlU3RhdHVzKCk7XG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgICBlcnJvcjogZXJyID0+IHtcbiAgICAgICAgdGhpcy53b3JraW5nLnNldChmYWxzZSk7XG4gICAgICAgIHRoaXMuc2hvd1RvYXN0KHRoaXMucmVzb2x2ZUVycm9yKGVycikpO1xuICAgICAgfSxcbiAgICB9KTtcbiAgfVxuXG4gIC8qKiBDb3BpYSB1bmEgbGlzdGEgZGUgY8OzZGlnb3MgZGUgcmVzcGFsZG8gYWwgcG9ydGFwYXBlbGVzLiAqL1xuICBhc3luYyBjb3B5Q29kZXMoY29kZXM6IHN0cmluZ1tdKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgdHJ5IHtcbiAgICAgIGF3YWl0IG5hdmlnYXRvci5jbGlwYm9hcmQud3JpdGVUZXh0KGNvZGVzLmpvaW4oJ1xcbicpKTtcbiAgICAgIHRoaXMuc2hvd1RvYXN0KHRoaXMudCgnbWZhQ29kZXNDb3BpZWQnKSk7XG4gICAgfSBjYXRjaCB7XG4gICAgICAvKiBzaW4gcmVjdXJzbyBkZSBjb3BpYSBkaXNwb25pYmxlICovXG4gICAgfVxuICB9XG5cbiAgLyoqIEV0aXF1ZXRhIGkxOG4gbGVnaWJsZSBwYXJhIHVuIG3DqXRvZG8gTUZBLiAqL1xuICBtZXRob2RMYWJlbChtZXRob2Q6IE1GQU1ldGhvZCB8IG51bGwpOiBzdHJpbmcge1xuICAgIHN3aXRjaCAobWV0aG9kKSB7XG4gICAgICBjYXNlICdUT1RQJzpcbiAgICAgICAgcmV0dXJuIHRoaXMudCgnbWZhTWV0aG9kVG90cCcpO1xuICAgICAgY2FzZSAnRU1BSUwnOlxuICAgICAgICByZXR1cm4gdGhpcy50KCdtZmFNZXRob2RFbWFpbCcpO1xuICAgICAgY2FzZSAnU01TJzpcbiAgICAgICAgcmV0dXJuIHRoaXMudCgnbWZhTWV0aG9kU21zJyk7XG4gICAgICBkZWZhdWx0OlxuICAgICAgICByZXR1cm4gJyc7XG4gICAgfVxuICB9XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIEhlbHBlcnNcbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgcHJpdmF0ZSByZXNldEZsb3coKTogdm9pZCB7XG4gICAgdGhpcy5waW5Db250cm9sLnJlc2V0KCk7XG4gICAgdGhpcy5wYXNzd29yZENvbnRyb2wucmVzZXQoKTtcbiAgICB0aGlzLnBob25lQ29udHJvbC5yZXNldCgpO1xuICAgIHRoaXMudG90cFNldHVwLnNldChudWxsKTtcbiAgICB0aGlzLnRvdHBRci5zZXQobnVsbCk7XG4gICAgdGhpcy5yZWdlbmVyYXRlZENvZGVzLnNldChudWxsKTtcbiAgICB0aGlzLnNlbGVjdGVkTWV0aG9kLnNldCgnVE9UUCcpO1xuICAgIHRoaXMuc3RvcENvb2xkb3duKCk7XG4gIH1cblxuICBwcml2YXRlIHN0YXJ0Q29vbGRvd24oKTogdm9pZCB7XG4gICAgdGhpcy5zdG9wQ29vbGRvd24oKTtcbiAgICB0aGlzLnJlc2VuZENvb2xkb3duLnNldChSRVNFTkRfQ09PTERPV05fU0VDT05EUyk7XG4gICAgdGhpcy5yZXNlbmRUaW1lciA9IHNldEludGVydmFsKCgpID0+IHtcbiAgICAgIHRoaXMucmVzZW5kQ29vbGRvd24udXBkYXRlKHYgPT4gdiAtIDEpO1xuICAgICAgaWYgKHRoaXMucmVzZW5kQ29vbGRvd24oKSA8PSAwKSB7XG4gICAgICAgIHRoaXMuc3RvcENvb2xkb3duKCk7XG4gICAgICB9XG4gICAgfSwgMTAwMCk7XG4gIH1cblxuICBwcml2YXRlIHN0b3BDb29sZG93bigpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5yZXNlbmRUaW1lcikge1xuICAgICAgY2xlYXJJbnRlcnZhbCh0aGlzLnJlc2VuZFRpbWVyKTtcbiAgICAgIHRoaXMucmVzZW5kVGltZXIgPSBudWxsO1xuICAgIH1cbiAgICB0aGlzLnJlc2VuZENvb2xkb3duLnNldCgwKTtcbiAgfVxuXG4gIC8qKiBNYXBlYSBsb3MgY8OzZGlnb3MgZGUgZXJyb3IgTUZBIGRlbCBiYWNrZW5kIGEgbWVuc2FqZXMgZGVsIG5hbWVzcGFjZSBgX2F1dGhgLiAqL1xuICBwcml2YXRlIHJlc29sdmVFcnJvcihlcnI6IHVua25vd24pOiBzdHJpbmcge1xuICAgIGNvbnN0IGNvZGUgPSAoZXJyIGFzIHsgY29kZT86IHN0cmluZyB9KT8uY29kZTtcbiAgICBzd2l0Y2ggKGNvZGUpIHtcbiAgICAgIGNhc2UgJ0FVVEhWMl9NRkFfSU5WQUxJRF9DT0RFJzpcbiAgICAgICAgcmV0dXJuIHRoaXMudCgnbWZhRXJyb3JJbnZhbGlkQ29kZScpO1xuICAgICAgY2FzZSAnQVVUSFYyX0VYUElSRURfQ09ERSc6XG4gICAgICAgIHJldHVybiB0aGlzLnQoJ21mYUVycm9yRXhwaXJlZENvZGUnKTtcbiAgICAgIGNhc2UgJ0FVVEhWMl9DT0RFX1VTRUQnOlxuICAgICAgICByZXR1cm4gdGhpcy50KCdtZmFFcnJvckNvZGVVc2VkJyk7XG4gICAgICBjYXNlICdBVVRIVjJfTUZBX0FMUkVBRFlfQUNUSVZFJzpcbiAgICAgICAgcmV0dXJuIHRoaXMudCgnbWZhRXJyb3JBbHJlYWR5QWN0aXZlJyk7XG4gICAgICBjYXNlICdBVVRIVjJfTUZBX05PVF9FTkFCTEVEJzpcbiAgICAgICAgcmV0dXJuIHRoaXMudCgnbWZhRXJyb3JOb3RFbmFibGVkJyk7XG4gICAgICBjYXNlICdBVVRIVjJfUEhPTkVfUkVRVUlSRUQnOlxuICAgICAgICByZXR1cm4gdGhpcy50KCdtZmFFcnJvclBob25lUmVxdWlyZWQnKTtcbiAgICAgIGNhc2UgJ0FVVEhWMl9QSE9ORV9FWElTVFMnOlxuICAgICAgICByZXR1cm4gdGhpcy50KCdtZmFFcnJvclBob25lRXhpc3RzJyk7XG4gICAgICBjYXNlICdBVVRIVjJfVE9PX01BTllfQVRURU1QVFMnOlxuICAgICAgICByZXR1cm4gdGhpcy50KCdlcnJvclRvb01hbnlBdHRlbXB0cycpO1xuICAgICAgY2FzZSAnQVVUSFYyX0lOVkFMSURfQ1VSUkVOVF9QQVNTV09SRCc6XG4gICAgICAgIHJldHVybiB0aGlzLnQoJ2Vycm9yQ3VycmVudFBhc3N3b3JkV3JvbmcnKTtcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHJldHVybiB0aGlzLnQoJ2Vycm9yR2VuZXJpYycpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgc2hvd1RvYXN0KG1lc3NhZ2U6IHN0cmluZyk6IHZvaWQge1xuICAgIHRoaXMudG9hc3Quc2hvdyh7IG1lc3NhZ2UsIGR1cmF0aW9uOiAzNTAwIH0pO1xuICB9XG59XG4iLCI8aW9uLW1vZGFsIFtpc09wZW5dPVwiaXNPcGVuXCIgKGRpZERpc21pc3MpPVwiY2xvc2UoKVwiPlxuICA8bmctdGVtcGxhdGU+XG4gICAgPGlvbi1oZWFkZXI+XG4gICAgICA8aW9uLXRvb2xiYXI+XG4gICAgICAgIDxpb24tYnV0dG9ucyBzbG90PVwiZW5kXCI+XG4gICAgICAgICAgPGlvbi1idXR0b24gZmlsbD1cImNsZWFyXCIgKGNsaWNrKT1cImNsb3NlKClcIj5cbiAgICAgICAgICAgIDxpb24taWNvbiBuYW1lPVwiY2xvc2Utb3V0bGluZVwiPjwvaW9uLWljb24+XG4gICAgICAgICAgPC9pb24tYnV0dG9uPlxuICAgICAgICA8L2lvbi1idXR0b25zPlxuICAgICAgPC9pb24tdG9vbGJhcj5cbiAgICA8L2lvbi1oZWFkZXI+XG5cbiAgICA8aW9uLWNvbnRlbnQgY2xhc3M9XCJpb24tcGFkZGluZ1wiPlxuICAgICAgPHNlY3Rpb24gY2xhc3M9XCJtZmEtbW9kYWxcIj5cbiAgICAgICAgQHN3aXRjaCAoc3RlcCgpKSB7IEBjYXNlICgnbG9hZGluZycpIHtcbiAgICAgICAgPGRpdiBjbGFzcz1cIm1mYS1sb2FkaW5nXCI+XG4gICAgICAgICAgPGlvbi1zcGlubmVyIG5hbWU9XCJjcmVzY2VudFwiPjwvaW9uLXNwaW5uZXI+XG4gICAgICAgIDwvZGl2PlxuICAgICAgICB9IEBjYXNlICgnc3RhdHVzJykge1xuICAgICAgICA8aDIgY2xhc3M9XCJtZmEtdGl0bGVcIj57eyB0KCdtZmFNYW5hZ2VUaXRsZScpIH19PC9oMj5cblxuICAgICAgICBAaWYgKG1mYUVuYWJsZWQoKSkge1xuICAgICAgICA8cCBjbGFzcz1cIm1mYS1zdGF0dXMgbWZhLXN0YXR1cy0tb25cIj57eyB0KCdtZmFFbmFibGVkTGFiZWwnKSB9fSDCtyB7eyBtZXRob2RMYWJlbChtZmFNZXRob2QoKSkgfX08L3A+XG5cbiAgICAgICAgQGlmIChtZmFNZXRob2QoKSA9PT0gJ1RPVFAnKSB7XG4gICAgICAgIDxkaXYgY2xhc3M9XCJtZmEtYmxvY2tcIj5cbiAgICAgICAgICA8aDM+e3sgdCgnbWZhQmFja3VwQ29kZXNUaXRsZScpIH19PC9oMz5cbiAgICAgICAgICA8cCBjbGFzcz1cIm1mYS1tdXRlZFwiPnt7IHQoJ21mYUJhY2t1cENvZGVzQXZhaWxhYmxlJykgfX06IHt7IGJhY2t1cENvZGVzQ291bnQoKSB9fTwvcD5cbiAgICAgICAgICBAaWYgKGJhY2t1cENvZGVzQ291bnQoKSA8IDMpIHtcbiAgICAgICAgICA8cCBjbGFzcz1cIm1mYS13YXJuXCI+e3sgdCgnbWZhQmFja3VwQ29kZXNMb3cnKSB9fTwvcD5cbiAgICAgICAgICB9IEBpZiAocmVnZW5lcmF0ZWRDb2RlcygpOyBhcyBjb2Rlcykge1xuICAgICAgICAgIDxwIGNsYXNzPVwibWZhLXdhcm5cIj57eyB0KCdtZmFCYWNrdXBDb2Rlc1NhdmVXYXJuaW5nJykgfX08L3A+XG4gICAgICAgICAgPGRpdiBjbGFzcz1cIm1mYS1jb2Rlc1wiPlxuICAgICAgICAgICAgQGZvciAoY29kZSBvZiBjb2RlczsgdHJhY2sgY29kZSkge1xuICAgICAgICAgICAgPGNvZGUgY2xhc3M9XCJtZmEtY29kZVwiPnt7IGNvZGUgfX08L2NvZGU+XG4gICAgICAgICAgICB9XG4gICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgPGlvbi1idXR0b24gZXhwYW5kPVwiYmxvY2tcIiBmaWxsPVwib3V0bGluZVwiIChjbGljayk9XCJjb3B5Q29kZXMoY29kZXMpXCI+IHt7IHQoJ21mYUNvcHlDb2RlcycpIH19IDwvaW9uLWJ1dHRvbj5cbiAgICAgICAgICB9IEBlbHNlIHtcbiAgICAgICAgICA8aW9uLWJ1dHRvbiBleHBhbmQ9XCJibG9ja1wiIGZpbGw9XCJvdXRsaW5lXCIgW2Rpc2FibGVkXT1cIndvcmtpbmcoKVwiIChjbGljayk9XCJyZWdlbmVyYXRlQmFja3VwQ29kZXMoKVwiPlxuICAgICAgICAgICAgQGlmICh3b3JraW5nKCkpIHtcbiAgICAgICAgICAgIDxpb24tc3Bpbm5lciBuYW1lPVwiY3Jlc2NlbnRcIj48L2lvbi1zcGlubmVyPlxuICAgICAgICAgICAgfSBAZWxzZSB7IHt7IHQoJ21mYVJlZ2VuZXJhdGVDb2RlcycpIH19IH1cbiAgICAgICAgICA8L2lvbi1idXR0b24+XG4gICAgICAgICAgfVxuICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuXG4gICAgICAgIDxpb24tYnV0dG9uIGV4cGFuZD1cImJsb2NrXCIgY29sb3I9XCJkYW5nZXJcIiBmaWxsPVwib3V0bGluZVwiIChjbGljayk9XCJnb1RvRGlzYWJsZSgpXCI+XG4gICAgICAgICAge3sgdCgnbWZhRGlzYWJsZUJ1dHRvbicpIH19XG4gICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgICAgfSBAZWxzZSB7XG4gICAgICAgIDxwIGNsYXNzPVwibWZhLXN0YXR1cyBtZmEtc3RhdHVzLS1vZmZcIj57eyB0KCdtZmFEaXNhYmxlZExhYmVsJykgfX08L3A+XG4gICAgICAgIDxwIGNsYXNzPVwibWZhLW11dGVkXCI+e3sgdCgnbWZhRGlzYWJsZWRIaW50JykgfX08L3A+XG4gICAgICAgIDxpb24tYnV0dG9uIGV4cGFuZD1cImJsb2NrXCIgKGNsaWNrKT1cImdvVG9NZXRob2RTZWxlY3QoKVwiPiB7eyB0KCdtZmFFbmFibGVCdXR0b24nKSB9fSA8L2lvbi1idXR0b24+XG4gICAgICAgIH0gfSBAY2FzZSAoJ21ldGhvZC1zZWxlY3QnKSB7XG4gICAgICAgIDxoMiBjbGFzcz1cIm1mYS10aXRsZVwiPnt7IHQoJ21mYUVuYWJsZVRpdGxlJykgfX08L2gyPlxuICAgICAgICA8cCBjbGFzcz1cIm1mYS1tdXRlZFwiPnt7IHQoJ21mYU1ldGhvZFByb21wdCcpIH19PC9wPlxuXG4gICAgICAgIDxpb24tcmFkaW8tZ3JvdXAgW3ZhbHVlXT1cInNlbGVjdGVkTWV0aG9kKClcIiAoaW9uQ2hhbmdlKT1cInNlbGVjdGVkTWV0aG9kLnNldCgkZXZlbnQuZGV0YWlsLnZhbHVlKVwiPlxuICAgICAgICAgIDxpb24taXRlbT5cbiAgICAgICAgICAgIDxpb24tcmFkaW8gc2xvdD1cInN0YXJ0XCIgdmFsdWU9XCJUT1RQXCI+PC9pb24tcmFkaW8+XG4gICAgICAgICAgICA8aW9uLWxhYmVsPlxuICAgICAgICAgICAgICA8c3Ryb25nPnt7IHQoJ21mYU1ldGhvZFRvdHAnKSB9fTwvc3Ryb25nPlxuICAgICAgICAgICAgICA8cD57eyB0KCdtZmFNZXRob2RUb3RwSGludCcpIH19PC9wPlxuICAgICAgICAgICAgPC9pb24tbGFiZWw+XG4gICAgICAgICAgPC9pb24taXRlbT5cbiAgICAgICAgICA8aW9uLWl0ZW0+XG4gICAgICAgICAgICA8aW9uLXJhZGlvIHNsb3Q9XCJzdGFydFwiIHZhbHVlPVwiRU1BSUxcIj48L2lvbi1yYWRpbz5cbiAgICAgICAgICAgIDxpb24tbGFiZWw+XG4gICAgICAgICAgICAgIDxzdHJvbmc+e3sgdCgnbWZhTWV0aG9kRW1haWwnKSB9fTwvc3Ryb25nPlxuICAgICAgICAgICAgICA8cD57eyB0KCdtZmFNZXRob2RFbWFpbEhpbnQnKSB9fTwvcD5cbiAgICAgICAgICAgIDwvaW9uLWxhYmVsPlxuICAgICAgICAgIDwvaW9uLWl0ZW0+XG4gICAgICAgICAgPGlvbi1pdGVtPlxuICAgICAgICAgICAgPGlvbi1yYWRpbyBzbG90PVwic3RhcnRcIiB2YWx1ZT1cIlNNU1wiPjwvaW9uLXJhZGlvPlxuICAgICAgICAgICAgPGlvbi1sYWJlbD5cbiAgICAgICAgICAgICAgPHN0cm9uZz57eyB0KCdtZmFNZXRob2RTbXMnKSB9fTwvc3Ryb25nPlxuICAgICAgICAgICAgICA8cD57eyB0KCdtZmFNZXRob2RTbXNIaW50JykgfX08L3A+XG4gICAgICAgICAgICA8L2lvbi1sYWJlbD5cbiAgICAgICAgICA8L2lvbi1pdGVtPlxuICAgICAgICA8L2lvbi1yYWRpby1ncm91cD5cblxuICAgICAgICBAaWYgKHNlbGVjdGVkTWV0aG9kKCkgPT09ICdTTVMnICYmICF1c2VyUGhvbmUoKSkge1xuICAgICAgICA8aW9uLWlucHV0XG4gICAgICAgICAgbGFiZWw9XCJ7eyB0KCdtZmFQaG9uZUxhYmVsJykgfX1cIlxuICAgICAgICAgIGxhYmVsUGxhY2VtZW50PVwiZmxvYXRpbmdcIlxuICAgICAgICAgIHR5cGU9XCJ0ZWxcIlxuICAgICAgICAgIHBsYWNlaG9sZGVyPVwiKzU2OTEyMzQ1Njc4XCJcbiAgICAgICAgICBbZm9ybUNvbnRyb2xdPVwicGhvbmVDb250cm9sXCJcbiAgICAgICAgPjwvaW9uLWlucHV0PlxuICAgICAgICBAaWYgKHBob25lQ29udHJvbC5pbnZhbGlkICYmIHBob25lQ29udHJvbC50b3VjaGVkKSB7XG4gICAgICAgIDxwIGNsYXNzPVwibWZhLWVycm9yXCI+e3sgdCgnbWZhUGhvbmVJbnZhbGlkJykgfX08L3A+XG4gICAgICAgIH0gfSBAaWYgKHNlbGVjdGVkTWV0aG9kKCkgPT09ICdTTVMnICYmIHVzZXJQaG9uZSgpOyBhcyBwaG9uZSkge1xuICAgICAgICA8cCBjbGFzcz1cIm1mYS1tdXRlZFwiPnt7IHQoJ21mYVBob25lUmVnaXN0ZXJlZCcpIH19OiB7eyBwaG9uZSB9fTwvcD5cbiAgICAgICAgfVxuXG4gICAgICAgIDxpb24tYnV0dG9uIGV4cGFuZD1cImJsb2NrXCIgW2Rpc2FibGVkXT1cIndvcmtpbmcoKVwiIChjbGljayk9XCJwcm9jZWVkV2l0aE1ldGhvZCgpXCI+XG4gICAgICAgICAgQGlmICh3b3JraW5nKCkpIHtcbiAgICAgICAgICA8aW9uLXNwaW5uZXIgbmFtZT1cImNyZXNjZW50XCI+PC9pb24tc3Bpbm5lcj5cbiAgICAgICAgICB9IEBlbHNlIHsge3sgdCgnbWZhQ29udGludWUnKSB9fSB9XG4gICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgICAgPGlvbi1idXR0b24gZXhwYW5kPVwiYmxvY2tcIiBmaWxsPVwiY2xlYXJcIiBjb2xvcj1cIm1lZGl1bVwiIChjbGljayk9XCJiYWNrVG9TdGF0dXMoKVwiPlxuICAgICAgICAgIHt7IHQoJ21mYUNhbmNlbCcpIH19XG4gICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgICAgfSBAY2FzZSAoJ3RvdHAtc2V0dXAnKSB7XG4gICAgICAgIDxoMiBjbGFzcz1cIm1mYS10aXRsZVwiPnt7IHQoJ21mYVRvdHBTZXR1cFRpdGxlJykgfX08L2gyPlxuICAgICAgICA8cCBjbGFzcz1cIm1mYS1tdXRlZFwiPnt7IHQoJ21mYVRvdHBTdGVwMScpIH19PC9wPlxuXG4gICAgICAgIEBpZiAodG90cFFyKCk7IGFzIHFyKSB7XG4gICAgICAgIDxkaXYgY2xhc3M9XCJtZmEtcXJcIj5cbiAgICAgICAgICA8dmFsLXFyLWNvZGUgW3Byb3BzXT1cInsgcXI6IHFyIH1cIiAvPlxuICAgICAgICA8L2Rpdj5cbiAgICAgICAgfSBAaWYgKHRvdHBTZXR1cCgpOyBhcyBzZXR1cCkge1xuICAgICAgICA8cCBjbGFzcz1cIm1mYS1tdXRlZFwiPnt7IHQoJ21mYVRvdHBNYW51YWxFbnRyeScpIH19PC9wPlxuICAgICAgICA8Y29kZSBjbGFzcz1cIm1mYS1zZWNyZXRcIj57eyBzZXR1cC5zZWNyZXQgfX08L2NvZGU+XG5cbiAgICAgICAgPHAgY2xhc3M9XCJtZmEtbXV0ZWRcIj57eyB0KCdtZmFUb3RwU3RlcDInKSB9fTwvcD5cbiAgICAgICAgPGRpdiBjbGFzcz1cIm1mYS1waW5cIj5cbiAgICAgICAgICA8dmFsLXBpbi1pbnB1dCBbcHJvcHNdPVwicGluSW5wdXRQcm9wc1wiIC8+XG4gICAgICAgIDwvZGl2PlxuXG4gICAgICAgIDxpb24tYnV0dG9uIGV4cGFuZD1cImJsb2NrXCIgW2Rpc2FibGVkXT1cIndvcmtpbmcoKVwiIChjbGljayk9XCJ2ZXJpZnlUb3RwKClcIj5cbiAgICAgICAgICBAaWYgKHdvcmtpbmcoKSkge1xuICAgICAgICAgIDxpb24tc3Bpbm5lciBuYW1lPVwiY3Jlc2NlbnRcIj48L2lvbi1zcGlubmVyPlxuICAgICAgICAgIH0gQGVsc2UgeyB7eyB0KCdtZmFUb3RwVmVyaWZ5JykgfX0gfVxuICAgICAgICA8L2lvbi1idXR0b24+XG5cbiAgICAgICAgPGRpdiBjbGFzcz1cIm1mYS1ibG9jayBtZmEtYmxvY2stLXdhcm5cIj5cbiAgICAgICAgICA8cCBjbGFzcz1cIm1mYS13YXJuXCI+e3sgdCgnbWZhQmFja3VwQ29kZXNTYXZlV2FybmluZycpIH19PC9wPlxuICAgICAgICAgIDxwIGNsYXNzPVwibWZhLW11dGVkXCI+e3sgdCgnbWZhQmFja3VwQ29kZXNFeHBsYWluJykgfX08L3A+XG4gICAgICAgICAgPGRpdiBjbGFzcz1cIm1mYS1jb2Rlc1wiPlxuICAgICAgICAgICAgQGZvciAoY29kZSBvZiBzZXR1cC5iYWNrdXBDb2RlczsgdHJhY2sgY29kZSkge1xuICAgICAgICAgICAgPGNvZGUgY2xhc3M9XCJtZmEtY29kZVwiPnt7IGNvZGUgfX08L2NvZGU+XG4gICAgICAgICAgICB9XG4gICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgPGlvbi1idXR0b24gZXhwYW5kPVwiYmxvY2tcIiBmaWxsPVwib3V0bGluZVwiIChjbGljayk9XCJjb3B5Q29kZXMoc2V0dXAuYmFja3VwQ29kZXMpXCI+XG4gICAgICAgICAgICB7eyB0KCdtZmFDb3B5Q29kZXMnKSB9fVxuICAgICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgICAgPC9kaXY+XG4gICAgICAgIH1cblxuICAgICAgICA8aW9uLWJ1dHRvbiBleHBhbmQ9XCJibG9ja1wiIGZpbGw9XCJjbGVhclwiIGNvbG9yPVwibWVkaXVtXCIgKGNsaWNrKT1cImJhY2tUb1N0YXR1cygpXCI+XG4gICAgICAgICAge3sgdCgnbWZhQ2FuY2VsJykgfX1cbiAgICAgICAgPC9pb24tYnV0dG9uPlxuICAgICAgICB9IEBjYXNlICgnY29kZS1jb25maXJtJykge1xuICAgICAgICA8aDIgY2xhc3M9XCJtZmEtdGl0bGVcIj57eyB0KCdtZmFDb25maXJtVGl0bGUnKSB9fTwvaDI+XG4gICAgICAgIDxwIGNsYXNzPVwibWZhLW11dGVkXCI+XG4gICAgICAgICAge3sgc2VsZWN0ZWRNZXRob2QoKSA9PT0gJ0VNQUlMJyA/IHQoJ21mYUNvbmZpcm1Qcm9tcHRFbWFpbCcpIDogdCgnbWZhQ29uZmlybVByb21wdFNtcycpIH19XG4gICAgICAgIDwvcD5cblxuICAgICAgICA8ZGl2IGNsYXNzPVwibWZhLXBpblwiPlxuICAgICAgICAgIDx2YWwtcGluLWlucHV0IFtwcm9wc109XCJwaW5JbnB1dFByb3BzXCIgLz5cbiAgICAgICAgPC9kaXY+XG5cbiAgICAgICAgPGlvbi1idXR0b24gZXhwYW5kPVwiYmxvY2tcIiBbZGlzYWJsZWRdPVwid29ya2luZygpXCIgKGNsaWNrKT1cImNvbmZpcm1Db2RlKClcIj5cbiAgICAgICAgICBAaWYgKHdvcmtpbmcoKSkge1xuICAgICAgICAgIDxpb24tc3Bpbm5lciBuYW1lPVwiY3Jlc2NlbnRcIj48L2lvbi1zcGlubmVyPlxuICAgICAgICAgIH0gQGVsc2UgeyB7eyB0KCdtZmFDb25maXJtQnV0dG9uJykgfX0gfVxuICAgICAgICA8L2lvbi1idXR0b24+XG5cbiAgICAgICAgPHAgY2xhc3M9XCJtZmEtcmVzZW5kXCI+XG4gICAgICAgICAge3sgdCgnbWZhTm9Db2RlJykgfX0gQGlmIChyZXNlbmRDb29sZG93bigpID4gMCkge1xuICAgICAgICAgIDxzcGFuIGNsYXNzPVwibWZhLW11dGVkXCI+e3sgdCgnbWZhUmVzZW5kSW4nKSB9fSB7eyByZXNlbmRDb29sZG93bigpIH19czwvc3Bhbj5cbiAgICAgICAgICB9IEBlbHNlIHtcbiAgICAgICAgICA8YSAoY2xpY2spPVwicmVzZW5kQ29kZSgpXCI+e3sgdCgnbWZhUmVzZW5kJykgfX08L2E+XG4gICAgICAgICAgfVxuICAgICAgICA8L3A+XG5cbiAgICAgICAgPGlvbi1idXR0b24gZXhwYW5kPVwiYmxvY2tcIiBmaWxsPVwiY2xlYXJcIiBjb2xvcj1cIm1lZGl1bVwiIChjbGljayk9XCJiYWNrVG9TdGF0dXMoKVwiPlxuICAgICAgICAgIHt7IHQoJ21mYUNhbmNlbCcpIH19XG4gICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgICAgfSBAY2FzZSAoJ2Rpc2FibGUnKSB7XG4gICAgICAgIDxoMiBjbGFzcz1cIm1mYS10aXRsZVwiPnt7IHQoJ21mYURpc2FibGVUaXRsZScpIH19PC9oMj5cbiAgICAgICAgPHAgY2xhc3M9XCJtZmEtbXV0ZWRcIj57eyB0KCdtZmFEaXNhYmxlUHJvbXB0JykgfX08L3A+XG5cbiAgICAgICAgPGlvbi1pbnB1dFxuICAgICAgICAgIGxhYmVsPVwie3sgdCgnbWZhUGFzc3dvcmRMYWJlbCcpIH19XCJcbiAgICAgICAgICBsYWJlbFBsYWNlbWVudD1cImZsb2F0aW5nXCJcbiAgICAgICAgICB0eXBlPVwicGFzc3dvcmRcIlxuICAgICAgICAgIFtmb3JtQ29udHJvbF09XCJwYXNzd29yZENvbnRyb2xcIlxuICAgICAgICA+PC9pb24taW5wdXQ+XG5cbiAgICAgICAgPGlvbi1idXR0b25cbiAgICAgICAgICBleHBhbmQ9XCJibG9ja1wiXG4gICAgICAgICAgY29sb3I9XCJkYW5nZXJcIlxuICAgICAgICAgIFtkaXNhYmxlZF09XCJ3b3JraW5nKCkgfHwgcGFzc3dvcmRDb250cm9sLmludmFsaWRcIlxuICAgICAgICAgIChjbGljayk9XCJjb25maXJtRGlzYWJsZSgpXCJcbiAgICAgICAgPlxuICAgICAgICAgIEBpZiAod29ya2luZygpKSB7XG4gICAgICAgICAgPGlvbi1zcGlubmVyIG5hbWU9XCJjcmVzY2VudFwiPjwvaW9uLXNwaW5uZXI+XG4gICAgICAgICAgfSBAZWxzZSB7IHt7IHQoJ21mYURpc2FibGVCdXR0b24nKSB9fSB9XG4gICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgICAgPGlvbi1idXR0b24gZXhwYW5kPVwiYmxvY2tcIiBmaWxsPVwiY2xlYXJcIiBjb2xvcj1cIm1lZGl1bVwiIChjbGljayk9XCJiYWNrVG9TdGF0dXMoKVwiPlxuICAgICAgICAgIHt7IHQoJ21mYUNhbmNlbCcpIH19XG4gICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgICAgfSB9XG4gICAgICA8L3NlY3Rpb24+XG4gICAgPC9pb24tY29udGVudD5cbiAgPC9uZy10ZW1wbGF0ZT5cbjwvaW9uLW1vZGFsPlxuIl19
|