valtech-components 2.0.554 → 2.0.557
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/molecules/docs-callout/docs-callout.component.mjs +60 -0
- package/esm2022/lib/components/organisms/login/login.component.mjs +721 -0
- package/esm2022/lib/components/organisms/login/types.mjs +10 -0
- package/esm2022/public-api.mjs +4 -1
- package/fesm2022/valtech-components.mjs +4482 -3707
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/docs-callout/docs-callout.component.d.ts +17 -0
- package/lib/components/organisms/login/login.component.d.ts +73 -0
- package/lib/components/organisms/login/types.d.ts +68 -0
- package/package.json +1 -1
- package/public-api.d.ts +3 -0
|
@@ -0,0 +1,721 @@
|
|
|
1
|
+
import { Component, EventEmitter, inject, Input, Output, } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { Validators } from '@angular/forms';
|
|
4
|
+
import { Router } from '@angular/router';
|
|
5
|
+
import { IonButton, IonButtons, IonContent, IonHeader, IonIcon, IonModal, IonText, IonToolbar, } from '@ionic/angular/standalone';
|
|
6
|
+
import { AuthService } from '../../../services/auth';
|
|
7
|
+
import { ToastService } from '../../../services/toast.service';
|
|
8
|
+
import { FormComponent } from '../form/form.component';
|
|
9
|
+
import { ImageComponent } from '../../atoms/image/image.component';
|
|
10
|
+
import { InputType, ComponentStates, } from '../../types';
|
|
11
|
+
import { SolidDefaultBlock } from '../../atoms/button/factory';
|
|
12
|
+
import { LOGIN_DEFAULTS, } from './types';
|
|
13
|
+
import * as i0 from "@angular/core";
|
|
14
|
+
export class LoginComponent {
|
|
15
|
+
constructor() {
|
|
16
|
+
this.props = {};
|
|
17
|
+
this.onSuccess = new EventEmitter();
|
|
18
|
+
this.onError = new EventEmitter();
|
|
19
|
+
this.onMFARequired = new EventEmitter();
|
|
20
|
+
// Services
|
|
21
|
+
this.authService = inject(AuthService);
|
|
22
|
+
this.toastService = inject(ToastService);
|
|
23
|
+
this.router = inject(Router);
|
|
24
|
+
// Timers
|
|
25
|
+
this.resendTimer = null;
|
|
26
|
+
this.resetResendTimer = null;
|
|
27
|
+
// State
|
|
28
|
+
this.isOAuthLoading = false;
|
|
29
|
+
this.isRegisterModalOpen = false;
|
|
30
|
+
this.isVerifyModalOpen = false;
|
|
31
|
+
this.isForgotPasswordModalOpen = false;
|
|
32
|
+
this.isResetPasswordModalOpen = false;
|
|
33
|
+
this.isMFAVerifyModalOpen = false;
|
|
34
|
+
this.pendingVerificationEmail = '';
|
|
35
|
+
this.pendingResetEmail = '';
|
|
36
|
+
this.resendCooldown = 0;
|
|
37
|
+
this.resetResendCooldown = 0;
|
|
38
|
+
// ==========================================
|
|
39
|
+
// LOGIN FORM
|
|
40
|
+
// ==========================================
|
|
41
|
+
this.emailInput = {
|
|
42
|
+
type: InputType.EMAIL,
|
|
43
|
+
label: 'Correo electrónico',
|
|
44
|
+
name: 'email',
|
|
45
|
+
token: 'login-email',
|
|
46
|
+
hint: '',
|
|
47
|
+
placeholder: 'tu@email.com',
|
|
48
|
+
errors: {
|
|
49
|
+
required: 'El correo es requerido',
|
|
50
|
+
email: 'Ingresa un correo válido',
|
|
51
|
+
},
|
|
52
|
+
validators: [Validators.required, Validators.email],
|
|
53
|
+
order: 0,
|
|
54
|
+
state: ComponentStates.ENABLED,
|
|
55
|
+
control: undefined,
|
|
56
|
+
};
|
|
57
|
+
this.passwordInput = {
|
|
58
|
+
type: InputType.PASSWORD,
|
|
59
|
+
label: 'Contraseña',
|
|
60
|
+
name: 'password',
|
|
61
|
+
token: 'login-password',
|
|
62
|
+
hint: '',
|
|
63
|
+
placeholder: '••••••••',
|
|
64
|
+
errors: {
|
|
65
|
+
required: 'La contraseña es requerida',
|
|
66
|
+
},
|
|
67
|
+
validators: [Validators.required],
|
|
68
|
+
order: 1,
|
|
69
|
+
state: ComponentStates.ENABLED,
|
|
70
|
+
control: undefined,
|
|
71
|
+
};
|
|
72
|
+
this.loginFormProps = {
|
|
73
|
+
name: 'Iniciar sesión',
|
|
74
|
+
sections: [
|
|
75
|
+
{
|
|
76
|
+
name: '',
|
|
77
|
+
order: 0,
|
|
78
|
+
fields: [this.emailInput, this.passwordInput],
|
|
79
|
+
},
|
|
80
|
+
],
|
|
81
|
+
actions: {
|
|
82
|
+
...SolidDefaultBlock('Iniciar sesión', 'submit'),
|
|
83
|
+
token: 'login-submit',
|
|
84
|
+
},
|
|
85
|
+
state: ComponentStates.ENABLED,
|
|
86
|
+
};
|
|
87
|
+
// ==========================================
|
|
88
|
+
// REGISTER FORM
|
|
89
|
+
// ==========================================
|
|
90
|
+
this.nameInput = {
|
|
91
|
+
type: InputType.TEXT,
|
|
92
|
+
label: 'Nombre completo',
|
|
93
|
+
name: 'name',
|
|
94
|
+
token: 'register-name',
|
|
95
|
+
hint: '',
|
|
96
|
+
placeholder: 'Tu nombre',
|
|
97
|
+
errors: {
|
|
98
|
+
required: 'El nombre es requerido',
|
|
99
|
+
minlength: 'Mínimo 2 caracteres',
|
|
100
|
+
},
|
|
101
|
+
validators: [Validators.required, Validators.minLength(2)],
|
|
102
|
+
order: 0,
|
|
103
|
+
state: ComponentStates.ENABLED,
|
|
104
|
+
control: undefined,
|
|
105
|
+
};
|
|
106
|
+
this.registerEmailInput = {
|
|
107
|
+
type: InputType.EMAIL,
|
|
108
|
+
label: 'Correo electrónico',
|
|
109
|
+
name: 'email',
|
|
110
|
+
token: 'register-email',
|
|
111
|
+
hint: '',
|
|
112
|
+
placeholder: 'tu@email.com',
|
|
113
|
+
errors: {
|
|
114
|
+
required: 'El correo es requerido',
|
|
115
|
+
email: 'Ingresa un correo válido',
|
|
116
|
+
},
|
|
117
|
+
validators: [Validators.required, Validators.email],
|
|
118
|
+
order: 1,
|
|
119
|
+
state: ComponentStates.ENABLED,
|
|
120
|
+
control: undefined,
|
|
121
|
+
};
|
|
122
|
+
this.registerPasswordInput = {
|
|
123
|
+
type: InputType.PASSWORD,
|
|
124
|
+
label: 'Contraseña',
|
|
125
|
+
name: 'password',
|
|
126
|
+
token: 'register-password',
|
|
127
|
+
hint: '',
|
|
128
|
+
placeholder: '••••••••',
|
|
129
|
+
errors: {
|
|
130
|
+
required: 'La contraseña es requerida',
|
|
131
|
+
minlength: 'Mínimo 8 caracteres',
|
|
132
|
+
},
|
|
133
|
+
validators: [Validators.required, Validators.minLength(8)],
|
|
134
|
+
order: 2,
|
|
135
|
+
state: ComponentStates.ENABLED,
|
|
136
|
+
control: undefined,
|
|
137
|
+
};
|
|
138
|
+
this.registerFormProps = {
|
|
139
|
+
name: 'Crear cuenta',
|
|
140
|
+
sections: [
|
|
141
|
+
{
|
|
142
|
+
name: '',
|
|
143
|
+
order: 0,
|
|
144
|
+
fields: [this.nameInput, this.registerEmailInput, this.registerPasswordInput],
|
|
145
|
+
},
|
|
146
|
+
],
|
|
147
|
+
actions: {
|
|
148
|
+
...SolidDefaultBlock('Registrarse', 'submit'),
|
|
149
|
+
token: 'register-submit',
|
|
150
|
+
},
|
|
151
|
+
state: ComponentStates.ENABLED,
|
|
152
|
+
};
|
|
153
|
+
// ==========================================
|
|
154
|
+
// VERIFY EMAIL FORM
|
|
155
|
+
// ==========================================
|
|
156
|
+
this.verifyPinInput = {
|
|
157
|
+
type: InputType.PIN_CODE,
|
|
158
|
+
label: '',
|
|
159
|
+
name: 'code',
|
|
160
|
+
token: 'verify-pin',
|
|
161
|
+
hint: '',
|
|
162
|
+
placeholder: '',
|
|
163
|
+
errors: {
|
|
164
|
+
required: 'El código es requerido',
|
|
165
|
+
minlength: 'Ingresa los 6 dígitos',
|
|
166
|
+
},
|
|
167
|
+
validators: [Validators.required, Validators.minLength(6)],
|
|
168
|
+
order: 0,
|
|
169
|
+
state: ComponentStates.ENABLED,
|
|
170
|
+
control: undefined,
|
|
171
|
+
length: 6,
|
|
172
|
+
allowNumbersOnly: true,
|
|
173
|
+
autoFocus: true,
|
|
174
|
+
};
|
|
175
|
+
this.verifyFormProps = {
|
|
176
|
+
name: 'Verificar correo',
|
|
177
|
+
sections: [
|
|
178
|
+
{
|
|
179
|
+
name: 'Ingresa el código de verificación enviado a tu correo.',
|
|
180
|
+
order: 0,
|
|
181
|
+
fields: [this.verifyPinInput],
|
|
182
|
+
},
|
|
183
|
+
],
|
|
184
|
+
actions: {
|
|
185
|
+
...SolidDefaultBlock('Verificar', 'submit'),
|
|
186
|
+
token: 'verify-submit',
|
|
187
|
+
},
|
|
188
|
+
state: ComponentStates.ENABLED,
|
|
189
|
+
};
|
|
190
|
+
// ==========================================
|
|
191
|
+
// MFA VERIFY FORM
|
|
192
|
+
// ==========================================
|
|
193
|
+
this.mfaPinInput = {
|
|
194
|
+
type: InputType.PIN_CODE,
|
|
195
|
+
label: '',
|
|
196
|
+
name: 'code',
|
|
197
|
+
token: 'mfa-pin',
|
|
198
|
+
hint: '',
|
|
199
|
+
placeholder: '',
|
|
200
|
+
errors: {
|
|
201
|
+
required: 'El código es requerido',
|
|
202
|
+
minlength: 'Ingresa los 6 dígitos',
|
|
203
|
+
},
|
|
204
|
+
validators: [Validators.required, Validators.minLength(6)],
|
|
205
|
+
order: 0,
|
|
206
|
+
state: ComponentStates.ENABLED,
|
|
207
|
+
control: undefined,
|
|
208
|
+
length: 6,
|
|
209
|
+
allowNumbersOnly: true,
|
|
210
|
+
autoFocus: true,
|
|
211
|
+
};
|
|
212
|
+
this.mfaVerifyFormProps = {
|
|
213
|
+
name: 'Verificación MFA',
|
|
214
|
+
sections: [
|
|
215
|
+
{
|
|
216
|
+
name: '',
|
|
217
|
+
order: 0,
|
|
218
|
+
fields: [this.mfaPinInput],
|
|
219
|
+
},
|
|
220
|
+
],
|
|
221
|
+
actions: {
|
|
222
|
+
...SolidDefaultBlock('Verificar', 'submit'),
|
|
223
|
+
token: 'mfa-verify-submit',
|
|
224
|
+
},
|
|
225
|
+
state: ComponentStates.ENABLED,
|
|
226
|
+
};
|
|
227
|
+
// ==========================================
|
|
228
|
+
// FORGOT PASSWORD FORM
|
|
229
|
+
// ==========================================
|
|
230
|
+
this.forgotEmailInput = {
|
|
231
|
+
type: InputType.EMAIL,
|
|
232
|
+
label: '',
|
|
233
|
+
name: 'email',
|
|
234
|
+
token: 'forgot-email',
|
|
235
|
+
hint: '',
|
|
236
|
+
placeholder: 'tu@email.com',
|
|
237
|
+
errors: {
|
|
238
|
+
required: 'El correo es requerido',
|
|
239
|
+
email: 'Ingresa un correo válido',
|
|
240
|
+
},
|
|
241
|
+
validators: [Validators.required, Validators.email],
|
|
242
|
+
order: 0,
|
|
243
|
+
state: ComponentStates.ENABLED,
|
|
244
|
+
control: undefined,
|
|
245
|
+
};
|
|
246
|
+
this.forgotPasswordFormProps = {
|
|
247
|
+
name: 'Recuperar contraseña',
|
|
248
|
+
sections: [
|
|
249
|
+
{
|
|
250
|
+
name: 'Ingresa tu correo electrónico y te enviaremos un código para restablecer tu contraseña.',
|
|
251
|
+
order: 0,
|
|
252
|
+
fields: [this.forgotEmailInput],
|
|
253
|
+
},
|
|
254
|
+
],
|
|
255
|
+
actions: {
|
|
256
|
+
...SolidDefaultBlock('Enviar código', 'submit'),
|
|
257
|
+
token: 'forgot-submit',
|
|
258
|
+
},
|
|
259
|
+
state: ComponentStates.ENABLED,
|
|
260
|
+
};
|
|
261
|
+
// ==========================================
|
|
262
|
+
// RESET PASSWORD FORM
|
|
263
|
+
// ==========================================
|
|
264
|
+
this.resetPinInput = {
|
|
265
|
+
type: InputType.PIN_CODE,
|
|
266
|
+
label: '',
|
|
267
|
+
name: 'code',
|
|
268
|
+
token: 'reset-pin',
|
|
269
|
+
hint: '',
|
|
270
|
+
placeholder: '',
|
|
271
|
+
errors: {
|
|
272
|
+
required: 'El código es requerido',
|
|
273
|
+
minlength: 'Ingresa los 6 dígitos',
|
|
274
|
+
},
|
|
275
|
+
validators: [Validators.required, Validators.minLength(6)],
|
|
276
|
+
order: 0,
|
|
277
|
+
state: ComponentStates.ENABLED,
|
|
278
|
+
control: undefined,
|
|
279
|
+
length: 6,
|
|
280
|
+
allowNumbersOnly: true,
|
|
281
|
+
autoFocus: true,
|
|
282
|
+
};
|
|
283
|
+
this.newPasswordInput = {
|
|
284
|
+
type: InputType.PASSWORD,
|
|
285
|
+
label: 'Nueva contraseña',
|
|
286
|
+
name: 'newPassword',
|
|
287
|
+
token: 'reset-new-password',
|
|
288
|
+
hint: 'Mínimo 8 caracteres',
|
|
289
|
+
placeholder: '••••••••',
|
|
290
|
+
errors: {
|
|
291
|
+
required: 'La contraseña es requerida',
|
|
292
|
+
minlength: 'Mínimo 8 caracteres',
|
|
293
|
+
},
|
|
294
|
+
validators: [Validators.required, Validators.minLength(8)],
|
|
295
|
+
order: 1,
|
|
296
|
+
state: ComponentStates.ENABLED,
|
|
297
|
+
control: undefined,
|
|
298
|
+
};
|
|
299
|
+
this.resetPasswordFormProps = {
|
|
300
|
+
name: 'Restablecer contraseña',
|
|
301
|
+
sections: [
|
|
302
|
+
{
|
|
303
|
+
name: 'Hemos enviado un código de verificación a tu correo.',
|
|
304
|
+
order: 0,
|
|
305
|
+
fields: [this.resetPinInput, this.newPasswordInput],
|
|
306
|
+
},
|
|
307
|
+
],
|
|
308
|
+
actions: {
|
|
309
|
+
...SolidDefaultBlock('Cambiar contraseña', 'submit'),
|
|
310
|
+
token: 'reset-submit',
|
|
311
|
+
},
|
|
312
|
+
state: ComponentStates.ENABLED,
|
|
313
|
+
};
|
|
314
|
+
}
|
|
315
|
+
// Resolved props with defaults
|
|
316
|
+
get config() {
|
|
317
|
+
return { ...LOGIN_DEFAULTS, ...this.props };
|
|
318
|
+
}
|
|
319
|
+
// ==========================================
|
|
320
|
+
// HANDLERS
|
|
321
|
+
// ==========================================
|
|
322
|
+
loginHandler(event) {
|
|
323
|
+
const email = event.fields['email'];
|
|
324
|
+
const password = event.fields['password'];
|
|
325
|
+
if (!email || !password) {
|
|
326
|
+
this.showToast('Completa todos los campos.');
|
|
327
|
+
return;
|
|
328
|
+
}
|
|
329
|
+
this.loginFormProps.state = ComponentStates.WORKING;
|
|
330
|
+
this.authService.signin({ email, password }).subscribe({
|
|
331
|
+
next: () => {
|
|
332
|
+
this.loginFormProps.state = ComponentStates.ENABLED;
|
|
333
|
+
if (this.authService.mfaPending().required) {
|
|
334
|
+
this.openMFAVerifyModal();
|
|
335
|
+
return;
|
|
336
|
+
}
|
|
337
|
+
this.handleLoginSuccess();
|
|
338
|
+
},
|
|
339
|
+
error: (err) => {
|
|
340
|
+
this.loginFormProps.state = ComponentStates.ENABLED;
|
|
341
|
+
const errorCode = err?.code;
|
|
342
|
+
if (errorCode === 'AUTHV2_EMAIL_NOT_VERIFIED') {
|
|
343
|
+
this.openVerifyModal(email);
|
|
344
|
+
return;
|
|
345
|
+
}
|
|
346
|
+
this.handleError(err, 'signin');
|
|
347
|
+
},
|
|
348
|
+
});
|
|
349
|
+
}
|
|
350
|
+
loginWithOAuth(provider) {
|
|
351
|
+
this.isOAuthLoading = true;
|
|
352
|
+
this.authService.signinWithOAuth(provider).subscribe({
|
|
353
|
+
next: () => {
|
|
354
|
+
this.isOAuthLoading = false;
|
|
355
|
+
if (this.authService.mfaPending().required) {
|
|
356
|
+
this.openMFAVerifyModal();
|
|
357
|
+
return;
|
|
358
|
+
}
|
|
359
|
+
this.handleLoginSuccess(provider);
|
|
360
|
+
},
|
|
361
|
+
error: (err) => {
|
|
362
|
+
this.isOAuthLoading = false;
|
|
363
|
+
const errorCode = err?.code;
|
|
364
|
+
if (errorCode === 'POPUP_CLOSED') {
|
|
365
|
+
return;
|
|
366
|
+
}
|
|
367
|
+
this.handleError(err, 'oauth');
|
|
368
|
+
},
|
|
369
|
+
});
|
|
370
|
+
}
|
|
371
|
+
// ==========================================
|
|
372
|
+
// REGISTER HANDLERS
|
|
373
|
+
// ==========================================
|
|
374
|
+
openRegisterModal() {
|
|
375
|
+
this.isRegisterModalOpen = true;
|
|
376
|
+
}
|
|
377
|
+
closeRegisterModal() {
|
|
378
|
+
this.isRegisterModalOpen = false;
|
|
379
|
+
}
|
|
380
|
+
registerHandler(event) {
|
|
381
|
+
const name = event.fields['name'];
|
|
382
|
+
const email = event.fields['email'];
|
|
383
|
+
const password = event.fields['password'];
|
|
384
|
+
if (!name || !email || !password) {
|
|
385
|
+
this.showToast('Completa todos los campos.');
|
|
386
|
+
return;
|
|
387
|
+
}
|
|
388
|
+
this.registerFormProps.state = ComponentStates.WORKING;
|
|
389
|
+
this.authService.signup({ name, email, password }).subscribe({
|
|
390
|
+
next: () => {
|
|
391
|
+
this.registerFormProps.state = ComponentStates.ENABLED;
|
|
392
|
+
this.closeRegisterModal();
|
|
393
|
+
this.openVerifyModal(email);
|
|
394
|
+
},
|
|
395
|
+
error: (err) => {
|
|
396
|
+
this.registerFormProps.state = ComponentStates.ENABLED;
|
|
397
|
+
this.handleError(err, 'signup');
|
|
398
|
+
},
|
|
399
|
+
});
|
|
400
|
+
}
|
|
401
|
+
// ==========================================
|
|
402
|
+
// VERIFY EMAIL HANDLERS
|
|
403
|
+
// ==========================================
|
|
404
|
+
openVerifyModal(email) {
|
|
405
|
+
this.pendingVerificationEmail = email;
|
|
406
|
+
this.verifyFormProps.sections[0].name = `Ingresa el código enviado a ${email}`;
|
|
407
|
+
this.verifyFormProps.state = ComponentStates.ENABLED;
|
|
408
|
+
this.isVerifyModalOpen = true;
|
|
409
|
+
this.startResendCooldown();
|
|
410
|
+
}
|
|
411
|
+
closeVerifyModal() {
|
|
412
|
+
this.isVerifyModalOpen = false;
|
|
413
|
+
this.pendingVerificationEmail = '';
|
|
414
|
+
this.stopResendCooldown();
|
|
415
|
+
}
|
|
416
|
+
verifyHandler(event) {
|
|
417
|
+
const code = event.fields['code'];
|
|
418
|
+
this.verifyFormProps.state = ComponentStates.WORKING;
|
|
419
|
+
this.authService
|
|
420
|
+
.verifyEmail({
|
|
421
|
+
email: this.pendingVerificationEmail,
|
|
422
|
+
code,
|
|
423
|
+
})
|
|
424
|
+
.subscribe({
|
|
425
|
+
next: () => {
|
|
426
|
+
this.verifyFormProps.state = ComponentStates.ENABLED;
|
|
427
|
+
this.showToast('¡Email verificado! Bienvenido.');
|
|
428
|
+
this.closeVerifyModal();
|
|
429
|
+
this.handleLoginSuccess();
|
|
430
|
+
},
|
|
431
|
+
error: (err) => {
|
|
432
|
+
this.verifyFormProps.state = ComponentStates.ENABLED;
|
|
433
|
+
this.handleError(err, 'verify');
|
|
434
|
+
},
|
|
435
|
+
});
|
|
436
|
+
}
|
|
437
|
+
resendCode() {
|
|
438
|
+
if (this.resendCooldown > 0)
|
|
439
|
+
return;
|
|
440
|
+
this.authService
|
|
441
|
+
.resendCode({
|
|
442
|
+
email: this.pendingVerificationEmail,
|
|
443
|
+
type: 'EMAIL_VERIFY',
|
|
444
|
+
})
|
|
445
|
+
.subscribe({
|
|
446
|
+
next: () => {
|
|
447
|
+
this.showToast('Código reenviado. Revisa tu email.');
|
|
448
|
+
this.startResendCooldown();
|
|
449
|
+
},
|
|
450
|
+
error: (err) => {
|
|
451
|
+
this.handleError(err, 'verify');
|
|
452
|
+
},
|
|
453
|
+
});
|
|
454
|
+
}
|
|
455
|
+
// ==========================================
|
|
456
|
+
// MFA HANDLERS
|
|
457
|
+
// ==========================================
|
|
458
|
+
openMFAVerifyModal() {
|
|
459
|
+
const method = this.authService.mfaPending().method;
|
|
460
|
+
this.mfaVerifyFormProps.sections[0].name =
|
|
461
|
+
method === 'TOTP'
|
|
462
|
+
? 'Ingresa el código de tu app de autenticación'
|
|
463
|
+
: method === 'EMAIL'
|
|
464
|
+
? 'Ingresa el código enviado a tu correo'
|
|
465
|
+
: 'Ingresa el código enviado a tu teléfono';
|
|
466
|
+
this.mfaVerifyFormProps.state = ComponentStates.ENABLED;
|
|
467
|
+
this.isMFAVerifyModalOpen = true;
|
|
468
|
+
this.onMFARequired.emit({ method: method });
|
|
469
|
+
}
|
|
470
|
+
closeMFAVerifyModal() {
|
|
471
|
+
this.isMFAVerifyModalOpen = false;
|
|
472
|
+
}
|
|
473
|
+
verifyMFAHandler(event) {
|
|
474
|
+
const code = event.fields['code'];
|
|
475
|
+
this.mfaVerifyFormProps.state = ComponentStates.WORKING;
|
|
476
|
+
this.authService.verifyMFA(code).subscribe({
|
|
477
|
+
next: () => {
|
|
478
|
+
this.mfaVerifyFormProps.state = ComponentStates.ENABLED;
|
|
479
|
+
this.closeMFAVerifyModal();
|
|
480
|
+
this.handleLoginSuccess(undefined, true);
|
|
481
|
+
},
|
|
482
|
+
error: (err) => {
|
|
483
|
+
this.mfaVerifyFormProps.state = ComponentStates.ENABLED;
|
|
484
|
+
this.handleError(err, 'mfa');
|
|
485
|
+
},
|
|
486
|
+
});
|
|
487
|
+
}
|
|
488
|
+
// ==========================================
|
|
489
|
+
// FORGOT PASSWORD HANDLERS
|
|
490
|
+
// ==========================================
|
|
491
|
+
openForgotPasswordModal() {
|
|
492
|
+
this.isForgotPasswordModalOpen = true;
|
|
493
|
+
}
|
|
494
|
+
closeForgotPasswordModal() {
|
|
495
|
+
this.isForgotPasswordModalOpen = false;
|
|
496
|
+
}
|
|
497
|
+
forgotPasswordHandler(event) {
|
|
498
|
+
const email = event.fields['email'];
|
|
499
|
+
if (!email) {
|
|
500
|
+
this.showToast('Ingresa tu correo electrónico.');
|
|
501
|
+
return;
|
|
502
|
+
}
|
|
503
|
+
this.forgotPasswordFormProps.state = ComponentStates.WORKING;
|
|
504
|
+
this.authService.forgotPassword({ email }).subscribe({
|
|
505
|
+
next: () => {
|
|
506
|
+
this.forgotPasswordFormProps.state = ComponentStates.ENABLED;
|
|
507
|
+
this.closeForgotPasswordModal();
|
|
508
|
+
this.openResetPasswordModal(email);
|
|
509
|
+
this.showToast('Código enviado. Revisa tu email.');
|
|
510
|
+
},
|
|
511
|
+
error: (err) => {
|
|
512
|
+
this.forgotPasswordFormProps.state = ComponentStates.ENABLED;
|
|
513
|
+
this.handleError(err, 'forgot');
|
|
514
|
+
},
|
|
515
|
+
});
|
|
516
|
+
}
|
|
517
|
+
// ==========================================
|
|
518
|
+
// RESET PASSWORD HANDLERS
|
|
519
|
+
// ==========================================
|
|
520
|
+
openResetPasswordModal(email) {
|
|
521
|
+
this.pendingResetEmail = email;
|
|
522
|
+
this.resetPasswordFormProps.sections[0].name = `Ingresa el código enviado a ${email}`;
|
|
523
|
+
this.resetPasswordFormProps.state = ComponentStates.ENABLED;
|
|
524
|
+
this.isResetPasswordModalOpen = true;
|
|
525
|
+
this.startResetResendCooldown();
|
|
526
|
+
}
|
|
527
|
+
closeResetPasswordModal() {
|
|
528
|
+
this.isResetPasswordModalOpen = false;
|
|
529
|
+
this.pendingResetEmail = '';
|
|
530
|
+
this.stopResetResendCooldown();
|
|
531
|
+
}
|
|
532
|
+
resetPasswordHandler(event) {
|
|
533
|
+
const code = event.fields['code'];
|
|
534
|
+
const newPassword = event.fields['newPassword'];
|
|
535
|
+
this.resetPasswordFormProps.state = ComponentStates.WORKING;
|
|
536
|
+
this.authService
|
|
537
|
+
.resetPassword({
|
|
538
|
+
email: this.pendingResetEmail,
|
|
539
|
+
code,
|
|
540
|
+
newPassword,
|
|
541
|
+
})
|
|
542
|
+
.subscribe({
|
|
543
|
+
next: () => {
|
|
544
|
+
this.resetPasswordFormProps.state = ComponentStates.ENABLED;
|
|
545
|
+
this.showToast('¡Contraseña actualizada! Ya puedes iniciar sesión.');
|
|
546
|
+
this.closeResetPasswordModal();
|
|
547
|
+
},
|
|
548
|
+
error: (err) => {
|
|
549
|
+
this.resetPasswordFormProps.state = ComponentStates.ENABLED;
|
|
550
|
+
this.handleError(err, 'reset');
|
|
551
|
+
},
|
|
552
|
+
});
|
|
553
|
+
}
|
|
554
|
+
resendResetCode() {
|
|
555
|
+
if (this.resetResendCooldown > 0)
|
|
556
|
+
return;
|
|
557
|
+
this.authService
|
|
558
|
+
.resendCode({
|
|
559
|
+
email: this.pendingResetEmail,
|
|
560
|
+
type: 'PASSWORD_RESET',
|
|
561
|
+
})
|
|
562
|
+
.subscribe({
|
|
563
|
+
next: () => {
|
|
564
|
+
this.showToast('Código reenviado. Revisa tu email.');
|
|
565
|
+
this.startResetResendCooldown();
|
|
566
|
+
},
|
|
567
|
+
error: (err) => {
|
|
568
|
+
this.handleError(err, 'reset');
|
|
569
|
+
},
|
|
570
|
+
});
|
|
571
|
+
}
|
|
572
|
+
// ==========================================
|
|
573
|
+
// HELPERS
|
|
574
|
+
// ==========================================
|
|
575
|
+
handleLoginSuccess(oauthProvider, mfaCompleted) {
|
|
576
|
+
this.showToast('¡Bienvenido!');
|
|
577
|
+
this.onSuccess.emit({
|
|
578
|
+
user: this.authService.user(),
|
|
579
|
+
mfaCompleted,
|
|
580
|
+
oauthProvider,
|
|
581
|
+
});
|
|
582
|
+
if (this.props.redirectOnSuccess) {
|
|
583
|
+
this.router.navigate([this.props.redirectOnSuccess]);
|
|
584
|
+
}
|
|
585
|
+
}
|
|
586
|
+
handleError(err, operation) {
|
|
587
|
+
const message = this.getErrorMessage(err);
|
|
588
|
+
this.showToast(message);
|
|
589
|
+
this.onError.emit({
|
|
590
|
+
code: err?.code,
|
|
591
|
+
message,
|
|
592
|
+
operation,
|
|
593
|
+
});
|
|
594
|
+
}
|
|
595
|
+
showToast(message) {
|
|
596
|
+
this.toastService.show({
|
|
597
|
+
message,
|
|
598
|
+
duration: 3500,
|
|
599
|
+
position: 'top',
|
|
600
|
+
color: 'dark',
|
|
601
|
+
});
|
|
602
|
+
}
|
|
603
|
+
getErrorMessage(err) {
|
|
604
|
+
const error = err;
|
|
605
|
+
const errorMessages = {
|
|
606
|
+
// Validation
|
|
607
|
+
VALIDATION_MISSING_REQUIRED_FIELDS: 'Faltan campos requeridos',
|
|
608
|
+
VALIDATION_INVALID_REQUEST: 'Solicitud inválida',
|
|
609
|
+
// Signin
|
|
610
|
+
AUTHV2_INVALID_CREDENTIALS: 'Correo o contraseña incorrectos',
|
|
611
|
+
AUTHV2_EMAIL_NOT_VERIFIED: 'Debes verificar tu correo electrónico',
|
|
612
|
+
AUTHV2_ACCOUNT_SUSPENDED: 'Tu cuenta ha sido suspendida',
|
|
613
|
+
AUTHV2_SIGNIN_FAILED: 'Error al iniciar sesión',
|
|
614
|
+
// Signup
|
|
615
|
+
AUTHV2_EMAIL_EXISTS: 'Este correo ya está registrado',
|
|
616
|
+
AUTHV2_PHONE_EXISTS: 'Este teléfono ya está registrado',
|
|
617
|
+
AUTHV2_WEAK_PASSWORD: 'La contraseña es muy débil',
|
|
618
|
+
AUTHV2_SIGNUP_FAILED: 'Error al crear la cuenta',
|
|
619
|
+
// Verification codes
|
|
620
|
+
AUTHV2_INVALID_CODE: 'Código incorrecto',
|
|
621
|
+
AUTHV2_EXPIRED_CODE: 'El código ha expirado. Solicita uno nuevo.',
|
|
622
|
+
AUTHV2_CODE_EXPIRED: 'El código ha expirado. Solicita uno nuevo.',
|
|
623
|
+
AUTHV2_CODE_ALREADY_USED: 'Este código ya fue utilizado',
|
|
624
|
+
AUTHV2_TOO_MANY_ATTEMPTS: 'Demasiados intentos, intenta más tarde',
|
|
625
|
+
AUTHV2_SEND_CODE_FAILED: 'Error al enviar el código',
|
|
626
|
+
AUTHV2_ALREADY_VERIFIED: 'Este email ya está verificado',
|
|
627
|
+
// MFA
|
|
628
|
+
AUTHV2_MFA_INVALID_CODE: 'Código de verificación incorrecto',
|
|
629
|
+
// Password reset
|
|
630
|
+
AUTHV2_PASSWORD_RESET_FAILED: 'Error al restablecer la contraseña',
|
|
631
|
+
AUTHV2_INVALID_CURRENT_PASSWORD: 'Contraseña actual incorrecta',
|
|
632
|
+
AUTHV2_SAME_PASSWORD: 'La nueva contraseña debe ser diferente',
|
|
633
|
+
// Session/Tokens
|
|
634
|
+
AUTHV2_INVALID_TOKEN: 'Sesión inválida',
|
|
635
|
+
AUTHV2_EXPIRED_TOKEN: 'Tu sesión ha expirado',
|
|
636
|
+
AUTHV2_SESSION_EXPIRED: 'Tu sesión ha expirado',
|
|
637
|
+
// User
|
|
638
|
+
AUTHV2_USER_NOT_FOUND: 'Usuario no encontrado',
|
|
639
|
+
// OAuth
|
|
640
|
+
POPUP_BLOCKED: 'Por favor, permite ventanas emergentes para este sitio',
|
|
641
|
+
POPUP_CLOSED: 'Se canceló la autenticación',
|
|
642
|
+
INVALID_RESPONSE: 'Error en la respuesta del servidor',
|
|
643
|
+
OAUTH_FAILED: 'Error de autenticación',
|
|
644
|
+
OAUTH_EMAIL_EXISTS: 'Este correo ya está registrado con otro método',
|
|
645
|
+
};
|
|
646
|
+
const errorCode = error?.error?.code || error?.code;
|
|
647
|
+
if (errorCode && errorMessages[errorCode]) {
|
|
648
|
+
return errorMessages[errorCode];
|
|
649
|
+
}
|
|
650
|
+
if (error?.error?.message) {
|
|
651
|
+
return error.error.message;
|
|
652
|
+
}
|
|
653
|
+
return 'Ha ocurrido un error. Intenta de nuevo.';
|
|
654
|
+
}
|
|
655
|
+
// ==========================================
|
|
656
|
+
// COOLDOWN TIMERS
|
|
657
|
+
// ==========================================
|
|
658
|
+
startResendCooldown() {
|
|
659
|
+
this.resendCooldown = 30;
|
|
660
|
+
this.resendTimer = setInterval(() => {
|
|
661
|
+
this.resendCooldown--;
|
|
662
|
+
if (this.resendCooldown <= 0) {
|
|
663
|
+
this.stopResendCooldown();
|
|
664
|
+
}
|
|
665
|
+
}, 1000);
|
|
666
|
+
}
|
|
667
|
+
stopResendCooldown() {
|
|
668
|
+
if (this.resendTimer) {
|
|
669
|
+
clearInterval(this.resendTimer);
|
|
670
|
+
this.resendTimer = null;
|
|
671
|
+
}
|
|
672
|
+
this.resendCooldown = 0;
|
|
673
|
+
}
|
|
674
|
+
startResetResendCooldown() {
|
|
675
|
+
this.resetResendCooldown = 30;
|
|
676
|
+
this.resetResendTimer = setInterval(() => {
|
|
677
|
+
this.resetResendCooldown--;
|
|
678
|
+
if (this.resetResendCooldown <= 0) {
|
|
679
|
+
this.stopResetResendCooldown();
|
|
680
|
+
}
|
|
681
|
+
}, 1000);
|
|
682
|
+
}
|
|
683
|
+
stopResetResendCooldown() {
|
|
684
|
+
if (this.resetResendTimer) {
|
|
685
|
+
clearInterval(this.resetResendTimer);
|
|
686
|
+
this.resetResendTimer = null;
|
|
687
|
+
}
|
|
688
|
+
this.resetResendCooldown = 0;
|
|
689
|
+
}
|
|
690
|
+
ngOnDestroy() {
|
|
691
|
+
this.stopResendCooldown();
|
|
692
|
+
this.stopResetResendCooldown();
|
|
693
|
+
}
|
|
694
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LoginComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
695
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: LoginComponent, isStandalone: true, selector: "val-login", inputs: { props: "props" }, outputs: { onSuccess: "onSuccess", onError: "onError", onMFARequired: "onMFARequired" }, ngImport: i0, template: "<div class=\"val-login\">\n <!-- Logo -->\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n\n <!-- Login Form -->\n <val-form [props]=\"loginFormProps\" (onSubmit)=\"loginHandler($event)\" />\n\n <!-- OAuth Section -->\n @if (config.showOAuth && config.oauthProviders.length > 0) {\n <div class=\"oauth-separator\">\n <span>o contin\u00FAa con</span>\n </div>\n\n <div class=\"oauth-buttons\">\n @for (provider of config.oauthProviders; track provider) {\n <ion-button\n expand=\"block\"\n fill=\"outline\"\n color=\"dark\"\n (click)=\"loginWithOAuth(provider)\"\n [disabled]=\"isOAuthLoading\"\n >\n @switch (provider) {\n @case ('google') {\n <ion-icon slot=\"start\" name=\"logo-google\"></ion-icon>\n {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Google' }}\n }\n @case ('apple') {\n <ion-icon slot=\"start\" name=\"logo-apple\"></ion-icon>\n {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Apple' }}\n }\n @case ('microsoft') {\n <ion-icon slot=\"start\" name=\"logo-microsoft\"></ion-icon>\n {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Microsoft' }}\n }\n }\n </ion-button>\n }\n </div>\n }\n\n <!-- Register Link -->\n @if (config.showRegister) {\n <div class=\"auth-link\">\n <ion-text color=\"dark\">\n \u00BFNo tienes cuenta?\n <a (click)=\"openRegisterModal()\">Registrarse</a>\n </ion-text>\n </div>\n }\n\n <!-- Forgot Password Link -->\n @if (config.showForgotPassword) {\n <div class=\"auth-link forgot-password\">\n <ion-text color=\"dark\">\n \u00BFOlvidaste tu contrase\u00F1a?\n <a (click)=\"openForgotPasswordModal()\">Recuperar contrase\u00F1a</a>\n </ion-text>\n </div>\n }\n\n <!-- Legal Notice -->\n @if (props.legal) {\n <div class=\"legal-notice\">\n <ion-text color=\"medium\">\n <p>\n Utilizamos los servicios de\n @if (props.legal.companyLink) {\n <a [href]=\"props.legal.companyLink\"><strong>{{ props.legal.companyName }}</strong></a>\n } @else {\n <strong>{{ props.legal.companyName }}</strong>\n }\n para ofrecerte una experiencia segura. Al iniciar sesi\u00F3n, aceptas\n nuestros\n @if (props.legal.termsLink) {\n <a [href]=\"props.legal.termsLink\">T\u00E9rminos y Condiciones</a>\n } @else {\n <span>T\u00E9rminos y Condiciones</span>\n }\n y\n @if (props.legal.privacyLink) {\n <a [href]=\"props.legal.privacyLink\">Pol\u00EDtica de Privacidad</a>\n } @else {\n <span>Pol\u00EDtica de Privacidad</span>\n }.\n </p>\n </ion-text>\n </div>\n }\n</div>\n\n<!-- Register Modal -->\n<ion-modal [isOpen]=\"isRegisterModalOpen\" (didDismiss)=\"closeRegisterModal()\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeRegisterModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"registerFormProps\" (onSubmit)=\"registerHandler($event)\" />\n <div class=\"auth-link\">\n <ion-text color=\"dark\">\n \u00BFYa tienes cuenta?\n <a (click)=\"closeRegisterModal()\">Iniciar sesi\u00F3n</a>\n </ion-text>\n </div>\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- Verify Email Modal -->\n<ion-modal [isOpen]=\"isVerifyModalOpen\" [backdropDismiss]=\"false\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeVerifyModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"verifyFormProps\" (onSubmit)=\"verifyHandler($event)\" />\n <div class=\"resend-link\">\n <ion-text color=\"dark\">\n \u00BFNo has recibido tu c\u00F3digo?\n @if (resendCooldown > 0) {\n <span class=\"cooldown\">Reenviar en {{ resendCooldown }}s</span>\n } @else {\n <a (click)=\"resendCode()\">Reenviar</a>\n }\n </ion-text>\n </div>\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- Forgot Password Modal -->\n<ion-modal [isOpen]=\"isForgotPasswordModalOpen\" (didDismiss)=\"closeForgotPasswordModal()\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeForgotPasswordModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"forgotPasswordFormProps\" (onSubmit)=\"forgotPasswordHandler($event)\" />\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- Reset Password Modal -->\n<ion-modal [isOpen]=\"isResetPasswordModalOpen\" [backdropDismiss]=\"false\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeResetPasswordModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"resetPasswordFormProps\" (onSubmit)=\"resetPasswordHandler($event)\" />\n <div class=\"resend-link\">\n <ion-text color=\"dark\">\n \u00BFNo has recibido tu c\u00F3digo?\n @if (resetResendCooldown > 0) {\n <span class=\"cooldown\">Reenviar en {{ resetResendCooldown }}s</span>\n } @else {\n <a (click)=\"resendResetCode()\">Reenviar</a>\n }\n </ion-text>\n </div>\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- MFA Verify Modal -->\n<ion-modal [isOpen]=\"isMFAVerifyModalOpen\" [backdropDismiss]=\"false\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeMFAVerifyModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"mfaVerifyFormProps\" (onSubmit)=\"verifyMFAHandler($event)\" />\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n", styles: [".val-login{width:100%}.logo-container{max-width:130px;margin-bottom:1.5rem}.auth-link{text-align:center;margin-top:1rem;font-size:.9rem}.auth-link a{color:var(--ion-color-primary);text-decoration:none;font-weight:500;cursor:pointer}.auth-link a:hover{text-decoration:underline}.oauth-separator{display:flex;align-items:center;margin:1.5rem 0}.oauth-separator:before,.oauth-separator:after{content:\"\";flex:1;height:1px;background:var(--ion-color-medium-tint)}.oauth-separator span{padding:0 1rem;color:var(--ion-color-dark);font-size:.85rem}.oauth-buttons{margin-bottom:1rem}.oauth-buttons ion-button{--border-radius: 8px;--border-width: 1px}.oauth-buttons ion-icon{font-size:1.2rem;margin-right:.5rem}.legal-notice{margin-top:1.5rem;padding-top:1rem;border-top:1px solid var(--ion-color-medium-tint)}.legal-notice p{font-size:.75rem;line-height:1.5;text-align:center;margin:0}.legal-notice a{color:var(--ion-color-primary);text-decoration:none}.legal-notice a:hover{text-decoration:underline}.modal-form-section{padding:1rem}.resend-link{text-align:center;margin-top:1rem;font-size:.9rem}.resend-link a{color:var(--ion-color-primary);cursor:pointer;font-weight:500}.resend-link a:hover{text-decoration:underline}.resend-link .cooldown{color:var(--ion-color-medium)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { 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: IonModal, selector: "ion-modal" }, { kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "component", type: IonToolbar, selector: "ion-toolbar", inputs: ["color", "mode"] }, { kind: "component", type: FormComponent, selector: "val-form", inputs: ["props"], outputs: ["onSubmit", "onInvalid", "onSelectChange"] }, { kind: "component", type: ImageComponent, selector: "val-image", inputs: ["props"] }] }); }
|
|
696
|
+
}
|
|
697
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LoginComponent, decorators: [{
|
|
698
|
+
type: Component,
|
|
699
|
+
args: [{ selector: 'val-login', standalone: true, imports: [
|
|
700
|
+
CommonModule,
|
|
701
|
+
IonButton,
|
|
702
|
+
IonButtons,
|
|
703
|
+
IonContent,
|
|
704
|
+
IonHeader,
|
|
705
|
+
IonIcon,
|
|
706
|
+
IonModal,
|
|
707
|
+
IonText,
|
|
708
|
+
IonToolbar,
|
|
709
|
+
FormComponent,
|
|
710
|
+
ImageComponent,
|
|
711
|
+
], template: "<div class=\"val-login\">\n <!-- Logo -->\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n\n <!-- Login Form -->\n <val-form [props]=\"loginFormProps\" (onSubmit)=\"loginHandler($event)\" />\n\n <!-- OAuth Section -->\n @if (config.showOAuth && config.oauthProviders.length > 0) {\n <div class=\"oauth-separator\">\n <span>o contin\u00FAa con</span>\n </div>\n\n <div class=\"oauth-buttons\">\n @for (provider of config.oauthProviders; track provider) {\n <ion-button\n expand=\"block\"\n fill=\"outline\"\n color=\"dark\"\n (click)=\"loginWithOAuth(provider)\"\n [disabled]=\"isOAuthLoading\"\n >\n @switch (provider) {\n @case ('google') {\n <ion-icon slot=\"start\" name=\"logo-google\"></ion-icon>\n {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Google' }}\n }\n @case ('apple') {\n <ion-icon slot=\"start\" name=\"logo-apple\"></ion-icon>\n {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Apple' }}\n }\n @case ('microsoft') {\n <ion-icon slot=\"start\" name=\"logo-microsoft\"></ion-icon>\n {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Microsoft' }}\n }\n }\n </ion-button>\n }\n </div>\n }\n\n <!-- Register Link -->\n @if (config.showRegister) {\n <div class=\"auth-link\">\n <ion-text color=\"dark\">\n \u00BFNo tienes cuenta?\n <a (click)=\"openRegisterModal()\">Registrarse</a>\n </ion-text>\n </div>\n }\n\n <!-- Forgot Password Link -->\n @if (config.showForgotPassword) {\n <div class=\"auth-link forgot-password\">\n <ion-text color=\"dark\">\n \u00BFOlvidaste tu contrase\u00F1a?\n <a (click)=\"openForgotPasswordModal()\">Recuperar contrase\u00F1a</a>\n </ion-text>\n </div>\n }\n\n <!-- Legal Notice -->\n @if (props.legal) {\n <div class=\"legal-notice\">\n <ion-text color=\"medium\">\n <p>\n Utilizamos los servicios de\n @if (props.legal.companyLink) {\n <a [href]=\"props.legal.companyLink\"><strong>{{ props.legal.companyName }}</strong></a>\n } @else {\n <strong>{{ props.legal.companyName }}</strong>\n }\n para ofrecerte una experiencia segura. Al iniciar sesi\u00F3n, aceptas\n nuestros\n @if (props.legal.termsLink) {\n <a [href]=\"props.legal.termsLink\">T\u00E9rminos y Condiciones</a>\n } @else {\n <span>T\u00E9rminos y Condiciones</span>\n }\n y\n @if (props.legal.privacyLink) {\n <a [href]=\"props.legal.privacyLink\">Pol\u00EDtica de Privacidad</a>\n } @else {\n <span>Pol\u00EDtica de Privacidad</span>\n }.\n </p>\n </ion-text>\n </div>\n }\n</div>\n\n<!-- Register Modal -->\n<ion-modal [isOpen]=\"isRegisterModalOpen\" (didDismiss)=\"closeRegisterModal()\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeRegisterModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"registerFormProps\" (onSubmit)=\"registerHandler($event)\" />\n <div class=\"auth-link\">\n <ion-text color=\"dark\">\n \u00BFYa tienes cuenta?\n <a (click)=\"closeRegisterModal()\">Iniciar sesi\u00F3n</a>\n </ion-text>\n </div>\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- Verify Email Modal -->\n<ion-modal [isOpen]=\"isVerifyModalOpen\" [backdropDismiss]=\"false\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeVerifyModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"verifyFormProps\" (onSubmit)=\"verifyHandler($event)\" />\n <div class=\"resend-link\">\n <ion-text color=\"dark\">\n \u00BFNo has recibido tu c\u00F3digo?\n @if (resendCooldown > 0) {\n <span class=\"cooldown\">Reenviar en {{ resendCooldown }}s</span>\n } @else {\n <a (click)=\"resendCode()\">Reenviar</a>\n }\n </ion-text>\n </div>\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- Forgot Password Modal -->\n<ion-modal [isOpen]=\"isForgotPasswordModalOpen\" (didDismiss)=\"closeForgotPasswordModal()\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeForgotPasswordModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"forgotPasswordFormProps\" (onSubmit)=\"forgotPasswordHandler($event)\" />\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- Reset Password Modal -->\n<ion-modal [isOpen]=\"isResetPasswordModalOpen\" [backdropDismiss]=\"false\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeResetPasswordModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"resetPasswordFormProps\" (onSubmit)=\"resetPasswordHandler($event)\" />\n <div class=\"resend-link\">\n <ion-text color=\"dark\">\n \u00BFNo has recibido tu c\u00F3digo?\n @if (resetResendCooldown > 0) {\n <span class=\"cooldown\">Reenviar en {{ resetResendCooldown }}s</span>\n } @else {\n <a (click)=\"resendResetCode()\">Reenviar</a>\n }\n </ion-text>\n </div>\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- MFA Verify Modal -->\n<ion-modal [isOpen]=\"isMFAVerifyModalOpen\" [backdropDismiss]=\"false\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeMFAVerifyModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"mfaVerifyFormProps\" (onSubmit)=\"verifyMFAHandler($event)\" />\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n", styles: [".val-login{width:100%}.logo-container{max-width:130px;margin-bottom:1.5rem}.auth-link{text-align:center;margin-top:1rem;font-size:.9rem}.auth-link a{color:var(--ion-color-primary);text-decoration:none;font-weight:500;cursor:pointer}.auth-link a:hover{text-decoration:underline}.oauth-separator{display:flex;align-items:center;margin:1.5rem 0}.oauth-separator:before,.oauth-separator:after{content:\"\";flex:1;height:1px;background:var(--ion-color-medium-tint)}.oauth-separator span{padding:0 1rem;color:var(--ion-color-dark);font-size:.85rem}.oauth-buttons{margin-bottom:1rem}.oauth-buttons ion-button{--border-radius: 8px;--border-width: 1px}.oauth-buttons ion-icon{font-size:1.2rem;margin-right:.5rem}.legal-notice{margin-top:1.5rem;padding-top:1rem;border-top:1px solid var(--ion-color-medium-tint)}.legal-notice p{font-size:.75rem;line-height:1.5;text-align:center;margin:0}.legal-notice a{color:var(--ion-color-primary);text-decoration:none}.legal-notice a:hover{text-decoration:underline}.modal-form-section{padding:1rem}.resend-link{text-align:center;margin-top:1rem;font-size:.9rem}.resend-link a{color:var(--ion-color-primary);cursor:pointer;font-weight:500}.resend-link a:hover{text-decoration:underline}.resend-link .cooldown{color:var(--ion-color-medium)}\n"] }]
|
|
712
|
+
}], propDecorators: { props: [{
|
|
713
|
+
type: Input
|
|
714
|
+
}], onSuccess: [{
|
|
715
|
+
type: Output
|
|
716
|
+
}], onError: [{
|
|
717
|
+
type: Output
|
|
718
|
+
}], onMFARequired: [{
|
|
719
|
+
type: Output
|
|
720
|
+
}] } });
|
|
721
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9naW4uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9jb21wb25lbnRzL29yZ2FuaXNtcy9sb2dpbi9sb2dpbi5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvb3JnYW5pc21zL2xvZ2luL2xvZ2luLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFDTCxTQUFTLEVBQ1QsWUFBWSxFQUNaLE1BQU0sRUFDTixLQUFLLEVBRUwsTUFBTSxHQUNQLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDNUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQ3pDLE9BQU8sRUFDTCxTQUFTLEVBQ1QsVUFBVSxFQUNWLFVBQVUsRUFDVixTQUFTLEVBQ1QsT0FBTyxFQUNQLFFBQVEsRUFDUixPQUFPLEVBQ1AsVUFBVSxHQUNYLE1BQU0sMkJBQTJCLENBQUM7QUFFbkMsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLHdCQUF3QixDQUFDO0FBQ3JELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUMvRCxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFDdkQsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLG1DQUFtQyxDQUFDO0FBQ25FLE9BQU8sRUFJTCxTQUFTLEVBQ1QsZUFBZSxHQUNoQixNQUFNLGFBQWEsQ0FBQztBQUNyQixPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSw0QkFBNEIsQ0FBQztBQUMvRCxPQUFPLEVBS0wsY0FBYyxHQUNmLE1BQU0sU0FBUyxDQUFDOztBQXFCakIsTUFBTSxPQUFPLGNB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LENBQUMsbUJBQW1CLEdBQUcsQ0FBQyxDQUFDO0lBQy9CLENBQUM7SUFFRCxXQUFXO1FBQ1QsSUFBSSxDQUFDLGtCQUFrQixFQUFFLENBQUM7UUFDMUIsSUFBSSxDQUFDLHVCQUF1QixFQUFFLENBQUM7SUFDakMsQ0FBQzsrR0FoeEJVLGNBQWM7bUdBQWQsY0FBYywwTEM3RDNCLDI5UEFxUEEsOHlDRHZNSSxZQUFZLCtCQUNaLFNBQVMsb1BBQ1QsVUFBVSw4RUFDVixVQUFVLHdLQUNWLFNBQVMsb0dBQ1QsT0FBTywySkFDUCxRQUFRLHNEQUNSLE9BQU8sZ0ZBQ1AsVUFBVSxtRkFDVixhQUFhLDhIQUNiLGNBQWM7OzRGQUtMLGNBQWM7a0JBbkIxQixTQUFTOytCQUNFLFdBQVcsY0FDVCxJQUFJLFdBQ1A7d0JBQ1AsWUFBWTt3QkFDWixTQUFTO3dCQUNULFVBQVU7d0JBQ1YsVUFBVTt3QkFDVixTQUFTO3dCQUNULE9BQU87d0JBQ1AsUUFBUTt3QkFDUixPQUFPO3dCQUNQLFVBQVU7d0JBQ1YsYUFBYTt3QkFDYixjQUFjO3FCQUNmOzhCQUtRLEtBQUs7c0JBQWIsS0FBSztnQkFFSSxTQUFTO3NCQUFsQixNQUFNO2dCQUNHLE9BQU87c0JBQWhCLE1BQU07Z0JBQ0csYUFBYTtzQkFBdEIsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIENvbXBvbmVudCxcbiAgRXZlbnRFbWl0dGVyLFxuICBpbmplY3QsXG4gIElucHV0LFxuICBPbkRlc3Ryb3ksXG4gIE91dHB1dCxcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgVmFsaWRhdG9ycyB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcbmltcG9ydCB7IFJvdXRlciB9IGZyb20gJ0Bhbmd1bGFyL3JvdXRlcic7XG5pbXBvcnQge1xuICBJb25CdXR0b24sXG4gIElvbkJ1dHRvbnMsXG4gIElvbkNvbnRlbnQsXG4gIElvbkhlYWRlcixcbiAgSW9uSWNvbixcbiAgSW9uTW9kYWwsXG4gIElvblRleHQsXG4gIElvblRvb2xiYXIsXG59IGZyb20gJ0Bpb25pYy9hbmd1bGFyL3N0YW5kYWxvbmUnO1xuXG5pbXBvcnQgeyBBdXRoU2VydmljZSB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL2F1dGgnO1xuaW1wb3J0IHsgVG9hc3RTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vLi4vc2VydmljZXMvdG9hc3Quc2VydmljZSc7XG5pbXBvcnQgeyBGb3JtQ29tcG9uZW50IH0gZnJvbSAnLi4vZm9ybS9mb3JtLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBJbWFnZUNvbXBvbmVudCB9IGZyb20gJy4uLy4uL2F0b21zL2ltYWdlL2ltYWdlLmNvbXBvbmVudCc7XG5pbXBvcnQge1xuICBGb3JtTWV0YWRhdGEsXG4gIEZvcm1TdWJtaXQsXG4gIElucHV0TWV0YWRhdGEsXG4gIElucHV0VHlwZSxcbiAgQ29tcG9uZW50U3RhdGVzLFxufSBmcm9tICcuLi8uLi90eXBlcyc7XG5pbXBvcnQgeyBTb2xpZERlZmF1bHRCbG9jayB9IGZyb20gJy4uLy4uL2F0b21zL2J1dHRvbi9mYWN0b3J5JztcbmltcG9ydCB7XG4gIExvZ2luTWV0YWRhdGEsXG4gIExvZ2luU3VjY2Vzc0V2ZW50LFxuICBMb2dpbkVycm9yRXZlbnQsXG4gIE1GQVJlcXVpcmVkRXZlbnQsXG4gIExPR0lOX0RFRkFVTFRTLFxufSBmcm9tICcuL3R5cGVzJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAndmFsLWxvZ2luJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW1xuICAgIENvbW1vbk1vZHVsZSxcbiAgICBJb25CdXR0b24sXG4gICAgSW9uQnV0dG9ucyxcbiAgICBJb25Db250ZW50LFxuICAgIElvbkhlYWRlcixcbiAgICBJb25JY29uLFxuICAgIElvbk1vZGFsLFxuICAgIElvblRleHQsXG4gICAgSW9uVG9vbGJhcixcbiAgICBGb3JtQ29tcG9uZW50LFxuICAgIEltYWdlQ29tcG9uZW50LFxuICBdLFxuICB0ZW1wbGF0ZVVybDogJy4vbG9naW4uY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9sb2dpbi5jb21wb25lbnQuc2NzcyddLFxufSlcbmV4cG9ydCBjbGFzcyBMb2dpbkNvbXBvbmVudCBpbXBsZW1lbnRzIE9uRGVzdHJveSB7XG4gIEBJbnB1dCgpIHByb3BzOiBMb2dpbk1ldGFkYXRhID0ge307XG5cbiAgQE91dHB1dCgpIG9uU3VjY2VzcyA9IG5ldyBFdmVudEVtaXR0ZXI8TG9naW5TdWNjZXNzRXZlbnQ+KCk7XG4gIEBPdXRwdXQoKSBvbkVycm9yID0gbmV3IEV2ZW50RW1pdHRlcjxMb2dpbkVycm9yRXZlbnQ+KCk7XG4gIEBPdXRwdXQoKSBvbk1GQVJlcXVpcmVkID0gbmV3IEV2ZW50RW1pdHRlcjxNRkFSZXF1aXJlZEV2ZW50PigpO1xuXG4gIC8vIFNlcnZpY2VzXG4gIGF1dGhTZXJ2aWNlID0gaW5qZWN0KEF1dGhTZXJ2aWNlKTtcbiAgcHJpdmF0ZSB0b2FzdFNlcnZpY2UgPSBpbmplY3QoVG9hc3RTZXJ2aWNlKTtcbiAgcHJpdmF0ZSByb3V0ZXIgPSBpbmplY3QoUm91dGVyKTtcblxuICAvLyBUaW1lcnNcbiAgcHJpdmF0ZSByZXNlbmRUaW1lcjogUmV0dXJuVHlwZTx0eXBlb2Ygc2V0SW50ZXJ2YWw+IHwgbnVsbCA9IG51bGw7XG4gIHByaXZhdGUgcmVzZXRSZXNlbmRUaW1lcjogUmV0dXJuVHlwZTx0eXBlb2Ygc2V0SW50ZXJ2YWw+IHwgbnVsbCA9IG51bGw7XG5cbiAgLy8gU3RhdGVcbiAgaXNPQXV0aExvYWRpbmcgPSBmYWxzZTtcbiAgaXNSZWdpc3Rlck1vZGFsT3BlbiA9IGZhbHNlO1xuICBpc1ZlcmlmeU1vZGFsT3BlbiA9IGZhbHNlO1xuICBpc0ZvcmdvdFBhc3N3b3JkTW9kYWxPcGVuID0gZmFsc2U7XG4gIGlzUmVzZXRQYXNzd29yZE1vZGFsT3BlbiA9IGZhbHNlO1xuICBpc01GQVZlcmlmeU1vZGFsT3BlbiA9IGZhbHNlO1xuXG4gIHBlbmRpbmdWZXJpZmljYXRpb25FbWFpbCA9ICcnO1xuICBwZW5kaW5nUmVzZXRFbWFpbCA9ICcnO1xuICByZXNlbmRDb29sZG93biA9IDA7XG4gIHJlc2V0UmVzZW5kQ29vbGRvd24gPSAwO1xuXG4gIC8vIFJlc29sdmVkIHByb3BzIHdpdGggZGVmYXVsdHNcbiAgZ2V0IGNvbmZpZygpOiBSZXF1aXJlZDxPbWl0PExvZ2luTWV0YWRhdGEsICdsb2dvJyB8ICdsZWdhbCcgfCAncmVkaXJlY3RPblN1Y2Nlc3MnPj4gJiBMb2dpbk1ldGFkYXRhIHtcbiAgICByZXR1cm4geyAuLi5MT0dJTl9ERUZBVUxUUywgLi4udGhpcy5wcm9wcyB9O1xuICB9XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIExPR0lOIEZPUk1cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgZW1haWxJbnB1dDogSW5wdXRNZXRhZGF0YSA9IHtcbiAgICB0eXBlOiBJbnB1dFR5cGUuRU1BSUwsXG4gICAgbGFiZWw6ICdDb3JyZW8gZWxlY3Ryw7NuaWNvJyxcbiAgICBuYW1lOiAnZW1haWwnLFxuICAgIHRva2VuOiAnbG9naW4tZW1haWwnLFxuICAgIGhpbnQ6ICcnLFxuICAgIHBsYWNlaG9sZGVyOiAndHVAZW1haWwuY29tJyxcbiAgICBlcnJvcnM6IHtcbiAgICAgIHJlcXVpcmVkOiAnRWwgY29ycmVvIGVzIHJlcXVlcmlkbycsXG4gICAgICBlbWFpbDogJ0luZ3Jlc2EgdW4gY29ycmVvIHbDoWxpZG8nLFxuICAgIH0sXG4gICAgdmFsaWRhdG9yczogW1ZhbGlkYXRvcnMucmVxdWlyZWQsIFZhbGlkYXRvcnMuZW1haWxdLFxuICAgIG9yZGVyOiAwLFxuICAgIHN0YXRlOiBDb21wb25lbnRTdGF0ZXMuRU5BQkxFRCxcbiAgICBjb250cm9sOiB1bmRlZmluZWQsXG4gIH07XG5cbiAgcGFzc3dvcmRJbnB1dDogSW5wdXRNZXRhZGF0YSA9IHtcbiAgICB0eXBlOiBJbnB1dFR5cGUuUEFTU1dPUkQsXG4gICAgbGFiZWw6ICdDb250cmFzZcOxYScsXG4gICAgbmFtZTogJ3Bhc3N3b3JkJyxcbiAgICB0b2tlbjogJ2xvZ2luLXBhc3N3b3JkJyxcbiAgICBoaW50OiAnJyxcbiAgICBwbGFjZWhvbGRlcjogJ+KAouKAouKAouKAouKAouKAouKAouKAoicsXG4gICAgZXJyb3JzOiB7XG4gICAgICByZXF1aXJlZDogJ0xhIGNvbnRyYXNlw7FhIGVzIHJlcXVlcmlkYScsXG4gICAgfSxcbiAgICB2YWxpZGF0b3JzOiBbVmFsaWRhdG9ycy5yZXF1aXJlZF0sXG4gICAgb3JkZXI6IDEsXG4gICAgc3RhdGU6IENvbXBvbmVudFN0YXRlcy5FTkFCTEVELFxuICAgIGNvbnRyb2w6IHVuZGVmaW5lZCxcbiAgfTtcblxuICBsb2dpbkZvcm1Qcm9wczogRm9ybU1ldGFkYXRhID0ge1xuICAgIG5hbWU6ICdJbmljaWFyIHNlc2nDs24nLFxuICAgIHNlY3Rpb25zOiBbXG4gICAgICB7XG4gICAgICAgIG5hbWU6ICcnLFxuICAgICAgICBvcmRlcjogMCxcbiAgICAgICAgZmllbGRzOiBbdGhpcy5lbWFpbElucHV0LCB0aGlzLnBhc3N3b3JkSW5wdXRdLFxuICAgICAgfSxcbiAgICBdLFxuICAgIGFjdGlvbnM6IHtcbiAgICAgIC4uLlNvbGlkRGVmYXVsdEJsb2NrKCdJbmljaWFyIHNlc2nDs24nLCAnc3VibWl0JyksXG4gICAgICB0b2tlbjogJ2xvZ2luLXN1Ym1pdCcsXG4gICAgfSxcbiAgICBzdGF0ZTogQ29tcG9uZW50U3RhdGVzLkVOQUJMRUQsXG4gIH07XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIFJFR0lTVEVSIEZPUk1cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgbmFtZUlucHV0OiBJbnB1dE1ldGFkYXRhID0ge1xuICAgIHR5cGU6IElucHV0VHlwZS5URVhULFxuICAgIGxhYmVsOiAnTm9tYnJlIGNvbXBsZXRvJyxcbiAgICBuYW1lOiAnbmFtZScsXG4gICAgdG9rZW46ICdyZWdpc3Rlci1uYW1lJyxcbiAgICBoaW50OiAnJyxcbiAgICBwbGFjZWhvbGRlcjogJ1R1IG5vbWJyZScsXG4gICAgZXJyb3JzOiB7XG4gICAgICByZXF1aXJlZDogJ0VsIG5vbWJyZSBlcyByZXF1ZXJpZG8nLFxuICAgICAgbWlubGVuZ3RoOiAnTcOtbmltbyAyIGNhcmFjdGVyZXMnLFxuICAgIH0sXG4gICAgdmFsaWRhdG9yczogW1ZhbGlkYXRvcnMucmVxdWlyZWQsIFZhbGlkYXRvcnMubWluTGVuZ3RoKDIpXSxcbiAgICBvcmRlcjogMCxcbiAgICBzdGF0ZTogQ29tcG9uZW50U3RhdGVzLkVOQUJMRUQsXG4gICAgY29udHJvbDogdW5kZWZpbmVkLFxuICB9O1xuXG4gIHJlZ2lzdGVyRW1haWxJbnB1dDogSW5wdXRNZXRhZGF0YSA9IHtcbiAgICB0eXBlOiBJbnB1dFR5cGUuRU1BSUwsXG4gICAgbGFiZWw6ICdDb3JyZW8gZWxlY3Ryw7NuaWNvJyxcbiAgICBuYW1lOiAnZW1haWwnLFxuICAgIHRva2VuOiAncmVnaXN0ZXItZW1haWwnLFxuICAgIGhpbnQ6ICcnLFxuICAgIHBsYWNlaG9sZGVyOiAndHVAZW1haWwuY29tJyxcbiAgICBlcnJvcnM6IHtcbiAgICAgIHJlcXVpcmVkOiAnRWwgY29ycmVvIGVzIHJlcXVlcmlkbycsXG4gICAgICBlbWFpbDogJ0luZ3Jlc2EgdW4gY29ycmVvIHbDoWxpZG8nLFxuICAgIH0sXG4gICAgdmFsaWRhdG9yczogW1ZhbGlkYXRvcnMucmVxdWlyZWQsIFZhbGlkYXRvcnMuZW1haWxdLFxuICAgIG9yZGVyOiAxLFxuICAgIHN0YXRlOiBDb21wb25lbnRTdGF0ZXMuRU5BQkxFRCxcbiAgICBjb250cm9sOiB1bmRlZmluZWQsXG4gIH07XG5cbiAgcmVnaXN0ZXJQYXNzd29yZElucHV0OiBJbnB1dE1ldGFkYXRhID0ge1xuICAgIHR5cGU6IElucHV0VHlwZS5QQVNTV09SRCxcbiAgICBsYWJlbDogJ0NvbnRyYXNlw7FhJyxcbiAgICBuYW1lOiAncGFzc3dvcmQnLFxuICAgIHRva2VuOiAncmVnaXN0ZXItcGFzc3dvcmQnLFxuICAgIGhpbnQ6ICcnLFxuICAgIHBsYWNlaG9sZGVyOiAn4oCi4oCi4oCi4oCi4oCi4oCi4oCi4oCiJyxcbiAgICBlcnJvcnM6IHtcbiAgICAgIHJlcXVpcmVkOiAnTGEgY29udHJhc2XDsWEgZXMgcmVxdWVyaWRhJyxcbiAgICAgIG1pbmxlbmd0aDogJ03DrW5pbW8gOCBjYXJhY3RlcmVzJyxcbiAgICB9LFxuICAgIHZhbGlkYXRvcnM6IFtWYWxpZGF0b3JzLnJlcXVpcmVkLCBWYWxpZGF0b3JzLm1pbkxlbmd0aCg4KV0sXG4gICAgb3JkZXI6IDIsXG4gICAgc3RhdGU6IENvbXBvbmVudFN0YXRlcy5FTkFCTEVELFxuICAgIGNvbnRyb2w6IHVuZGVmaW5lZCxcbiAgfTtcblxuICByZWdpc3RlckZvcm1Qcm9wczogRm9ybU1ldGFkYXRhID0ge1xuICAgIG5hbWU6ICdDcmVhciBjdWVudGEnLFxuICAgIHNlY3Rpb25zOiBbXG4gICAgICB7XG4gICAgICAgIG5hbWU6ICcnLFxuICAgICAgICBvcmRlcjogMCxcbiAgICAgICAgZmllbGRzOiBbdGhpcy5uYW1lSW5wdXQsIHRoaXMucmVnaXN0ZXJFbWFpbElucHV0LCB0aGlzLnJlZ2lzdGVyUGFzc3dvcmRJbnB1dF0sXG4gICAgICB9LFxuICAgIF0sXG4gICAgYWN0aW9uczoge1xuICAgICAgLi4uU29saWREZWZhdWx0QmxvY2soJ1JlZ2lzdHJhcnNlJywgJ3N1Ym1pdCcpLFxuICAgICAgdG9rZW46ICdyZWdpc3Rlci1zdWJtaXQnLFxuICAgIH0sXG4gICAgc3RhdGU6IENvbXBvbmVudFN0YXRlcy5FTkFCTEVELFxuICB9O1xuXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuICAvLyBWRVJJRlkgRU1BSUwgRk9STVxuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuICB2ZXJpZnlQaW5JbnB1dDogSW5wdXRNZXRhZGF0YSA9IHtcbiAgICB0eXBlOiBJbnB1dFR5cGUuUElOX0NPREUsXG4gICAgbGFiZWw6ICcnLFxuICAgIG5hbWU6ICdjb2RlJyxcbiAgICB0b2tlbjogJ3ZlcmlmeS1waW4nLFxuICAgIGhpbnQ6ICcnLFxuICAgIHBsYWNlaG9sZGVyOiAnJyxcbiAgICBlcnJvcnM6IHtcbiAgICAgIHJlcXVpcmVkOiAnRWwgY8OzZGlnbyBlcyByZXF1ZXJpZG8nLFxuICAgICAgbWlubGVuZ3RoOiAnSW5ncmVzYSBsb3MgNiBkw61naXRvcycsXG4gICAgfSxcbiAgICB2YWxpZGF0b3JzOiBbVmFsaWRhdG9ycy5yZXF1aXJlZCwgVmFsaWRhdG9ycy5taW5MZW5ndGgoNildLFxuICAgIG9yZGVyOiAwLFxuICAgIHN0YXRlOiBDb21wb25lbnRTdGF0ZXMuRU5BQkxFRCxcbiAgICBjb250cm9sOiB1bmRlZmluZWQsXG4gICAgbGVuZ3RoOiA2LFxuICAgIGFsbG93TnVtYmVyc09ubHk6IHRydWUsXG4gICAgYXV0b0ZvY3VzOiB0cnVlLFxuICB9O1xuXG4gIHZlcmlmeUZvcm1Qcm9wczogRm9ybU1ldGFkYXRhID0ge1xuICAgIG5hbWU6ICdWZXJpZmljYXIgY29ycmVvJyxcbiAgICBzZWN0aW9uczogW1xuICAgICAge1xuICAgICAgICBuYW1lOiAnSW5ncmVzYSBlbCBjw7NkaWdvIGRlIHZlcmlmaWNhY2nDs24gZW52aWFkbyBhIHR1IGNvcnJlby4nLFxuICAgICAgICBvcmRlcjogMCxcbiAgICAgICAgZmllbGRzOiBbdGhpcy52ZXJpZnlQaW5JbnB1dF0sXG4gICAgICB9LFxuICAgIF0sXG4gICAgYWN0aW9uczoge1xuICAgICAgLi4uU29saWREZWZhdWx0QmxvY2soJ1ZlcmlmaWNhcicsICdzdWJtaXQnKSxcbiAgICAgIHRva2VuOiAndmVyaWZ5LXN1Ym1pdCcsXG4gICAgfSxcbiAgICBzdGF0ZTogQ29tcG9uZW50U3RhdGVzLkVOQUJMRUQsXG4gIH07XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIE1GQSBWRVJJRlkgRk9STVxuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuICBtZmFQaW5JbnB1dDogSW5wdXRNZXRhZGF0YSA9IHtcbiAgICB0eXBlOiBJbnB1dFR5cGUuUElOX0NPREUsXG4gICAgbGFiZWw6ICcnLFxuICAgIG5hbWU6ICdjb2RlJyxcbiAgICB0b2tlbjogJ21mYS1waW4nLFxuICAgIGhpbnQ6ICcnLFxuICAgIHBsYWNlaG9sZGVyOiAnJyxcbiAgICBlcnJvcnM6IHtcbiAgICAgIHJlcXVpcmVkOiAnRWwgY8OzZGlnbyBlcyByZXF1ZXJpZG8nLFxuICAgICAgbWlubGVuZ3RoOiAnSW5ncmVzYSBsb3MgNiBkw61naXRvcycsXG4gICAgfSxcbiAgICB2YWxpZGF0b3JzOiBbVmFsaWRhdG9ycy5yZXF1aXJlZCwgVmFsaWRhdG9ycy5taW5MZW5ndGgoNildLFxuICAgIG9yZGVyOiAwLFxuICAgIHN0YXRlOiBDb21wb25lbnRTdGF0ZXMuRU5BQkxFRCxcbiAgICBjb250cm9sOiB1bmRlZmluZWQsXG4gICAgbGVuZ3RoOiA2LFxuICAgIGFsbG93TnVtYmVyc09ubHk6IHRydWUsXG4gICAgYXV0b0ZvY3VzOiB0cnVlLFxuICB9O1xuXG4gIG1mYVZlcmlmeUZvcm1Qcm9wczogRm9ybU1ldGFkYXRhID0ge1xuICAgIG5hbWU6ICdWZXJpZmljYWNpw7NuIE1GQScsXG4gICAgc2VjdGlvbnM6IFtcbiAgICAgIHtcbiAgICAgICAgbmFtZTogJycsXG4gICAgICAgIG9yZGVyOiAwLFxuICAgICAgICBmaWVsZHM6IFt0aGlzLm1mYVBpbklucHV0XSxcbiAgICAgIH0sXG4gICAgXSxcbiAgICBhY3Rpb25zOiB7XG4gICAgICAuLi5Tb2xpZERlZmF1bHRCbG9jaygnVmVyaWZpY2FyJywgJ3N1Ym1pdCcpLFxuICAgICAgdG9rZW46ICdtZmEtdmVyaWZ5LXN1Ym1pdCcsXG4gICAgfSxcbiAgICBzdGF0ZTogQ29tcG9uZW50U3RhdGVzLkVOQUJMRUQsXG4gIH07XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIEZPUkdPVCBQQVNTV09SRCBGT1JNXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4gIGZvcmdvdEVtYWlsSW5wdXQ6IElucHV0TWV0YWRhdGEgPSB7XG4gICAgdHlwZTogSW5wdXRUeXBlLkVNQUlMLFxuICAgIGxhYmVsOiAnJyxcbiAgICBuYW1lOiAnZW1haWwnLFxuICAgIHRva2VuOiAnZm9yZ290LWVtYWlsJyxcbiAgICBoaW50OiAnJyxcbiAgICBwbGFjZWhvbGRlcjogJ3R1QGVtYWlsLmNvbScsXG4gICAgZXJyb3JzOiB7XG4gICAgICByZXF1aXJlZDogJ0VsIGNvcnJlbyBlcyByZXF1ZXJpZG8nLFxuICAgICAgZW1haWw6ICdJbmdyZXNhIHVuIGNvcnJlbyB2w6FsaWRvJyxcbiAgICB9LFxuICAgIHZhbGlkYXRvcnM6IFtWYWxpZGF0b3JzLnJlcXVpcmVkLCBWYWxpZGF0b3JzLmVtYWlsXSxcbiAgICBvcmRlcjogMCxcbiAgICBzdGF0ZTogQ29tcG9uZW50U3RhdGVzLkVOQUJMRUQsXG4gICAgY29udHJvbDogdW5kZWZpbmVkLFxuICB9O1xuXG4gIGZvcmdvdFBhc3N3b3JkRm9ybVByb3BzOiBGb3JtTWV0YWRhdGEgPSB7XG4gICAgbmFtZTogJ1JlY3VwZXJhciBjb250cmFzZcOxYScsXG4gICAgc2VjdGlvbnM6IFtcbiAgICAgIHtcbiAgICAgICAgbmFtZTogJ0luZ3Jlc2EgdHUgY29ycmVvIGVsZWN0csOzbmljbyB5IHRlIGVudmlhcmVtb3MgdW4gY8OzZGlnbyBwYXJhIHJlc3RhYmxlY2VyIHR1IGNvbnRyYXNlw7FhLicsXG4gICAgICAgIG9yZGVyOiAwLFxuICAgICAgICBmaWVsZHM6IFt0aGlzLmZvcmdvdEVtYWlsSW5wdXRdLFxuICAgICAgfSxcbiAgICBdLFxuICAgIGFjdGlvbnM6IHtcbiAgICAgIC4uLlNvbGlkRGVmYXVsdEJsb2NrKCdFbnZpYXIgY8OzZGlnbycsICdzdWJtaXQnKSxcbiAgICAgIHRva2VuOiAnZm9yZ290LXN1Ym1pdCcsXG4gICAgfSxcbiAgICBzdGF0ZTogQ29tcG9uZW50U3RhdGVzLkVOQUJMRUQsXG4gIH07XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIFJFU0VUIFBBU1NXT1JEIEZPUk1cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgcmVzZXRQaW5JbnB1dDogSW5wdXRNZXRhZGF0YSA9IHtcbiAgICB0eXBlOiBJbnB1dFR5cGUuUElOX0NPREUsXG4gICAgbGFiZWw6ICcnLFxuICAgIG5hbWU6ICdjb2RlJyxcbiAgICB0b2tlbjogJ3Jlc2V0LXBpbicsXG4gICAgaGludDogJycsXG4gICAgcGxhY2Vob2xkZXI6ICcnLFxuICAgIGVycm9yczoge1xuICAgICAgcmVxdWlyZWQ6ICdFbCBjw7NkaWdvIGVzIHJlcXVlcmlkbycsXG4gICAgICBtaW5sZW5ndGg6ICdJbmdyZXNhIGxvcyA2IGTDrWdpdG9zJyxcbiAgICB9LFxuICAgIHZhbGlkYXRvcnM6IFtWYWxpZGF0b3JzLnJlcXVpcmVkLCBWYWxpZGF0b3JzLm1pbkxlbmd0aCg2KV0sXG4gICAgb3JkZXI6IDAsXG4gICAgc3RhdGU6IENvbXBvbmVudFN0YXRlcy5FTkFCTEVELFxuICAgIGNvbnRyb2w6IHVuZGVmaW5lZCxcbiAgICBsZW5ndGg6IDYsXG4gICAgYWxsb3dOdW1iZXJzT25seTogdHJ1ZSxcbiAgICBhdXRvRm9jdXM6IHRydWUsXG4gIH07XG5cbiAgbmV3UGFzc3dvcmRJbnB1dDogSW5wdXRNZXRhZGF0YSA9IHtcbiAgICB0eXBlOiBJbnB1dFR5cGUuUEFTU1dPUkQsXG4gICAgbGFiZWw6ICdOdWV2YSBjb250cmFzZcOxYScsXG4gICAgbmFtZTogJ25ld1Bhc3N3b3JkJyxcbiAgICB0b2tlbjogJ3Jlc2V0LW5ldy1wYXNzd29yZCcsXG4gICAgaGludDogJ03DrW5pbW8gOCBjYXJhY3RlcmVzJyxcbiAgICBwbGFjZWhvbGRlcjogJ+KAouKAouKAouKAouKAouKAouKAouKAoicsXG4gICAgZXJyb3JzOiB7XG4gICAgICByZXF1aXJlZDogJ0xhIGNvbnRyYXNlw7FhIGVzIHJlcXVlcmlkYScsXG4gICAgICBtaW5sZW5ndGg6ICdNw61uaW1vIDggY2FyYWN0ZXJlcycsXG4gICAgfSxcbiAgICB2YWxpZGF0b3JzOiBbVmFsaWRhdG9ycy5yZXF1aXJlZCwgVmFsaWRhdG9ycy5taW5MZW5ndGgoOCldLFxuICAgIG9yZGVyOiAxLFxuICAgIHN0YXRlOiBDb21wb25lbnRTdGF0ZXMuRU5BQkxFRCxcbiAgICBjb250cm9sOiB1bmRlZmluZWQsXG4gIH07XG5cbiAgcmVzZXRQYXNzd29yZEZvcm1Qcm9wczogRm9ybU1ldGFkYXRhID0ge1xuICAgIG5hbWU6ICdSZXN0YWJsZWNlciBjb250cmFzZcOxYScsXG4gICAgc2VjdGlvbnM6IFtcbiAgICAgIHtcbiAgICAgICAgbmFtZTogJ0hlbW9zIGVudmlhZG8gdW4gY8OzZGlnbyBkZSB2ZXJpZmljYWNpw7NuIGEgdHUgY29ycmVvLicsXG4gICAgICAgIG9yZGVyOiAwLFxuICAgICAgICBmaWVsZHM6IFt0aGlzLnJlc2V0UGluSW5wdXQsIHRoaXMubmV3UGFzc3dvcmRJbnB1dF0sXG4gICAgICB9LFxuICAgIF0sXG4gICAgYWN0aW9uczoge1xuICAgICAgLi4uU29saWREZWZhdWx0QmxvY2soJ0NhbWJpYXIgY29udHJhc2XDsWEnLCAnc3VibWl0JyksXG4gICAgICB0b2tlbjogJ3Jlc2V0LXN1Ym1pdCcsXG4gICAgfSxcbiAgICBzdGF0ZTogQ29tcG9uZW50U3RhdGVzLkVOQUJMRUQsXG4gIH07XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIEhBTkRMRVJTXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4gIGxvZ2luSGFuZGxlcihldmVudDogRm9ybVN1Ym1pdCk6IHZvaWQge1xuICAgIGNvbnN0IGVtYWlsID0gZXZlbnQuZmllbGRzWydlbWFpbCddO1xuICAgIGNvbnN0IHBhc3N3b3JkID0gZXZlbnQuZmllbGRzWydwYXNzd29yZCddO1xuXG4gICAgaWYgKCFlbWFpbCB8fCAhcGFzc3dvcmQpIHtcbiAgICAgIHRoaXMuc2hvd1RvYXN0KCdDb21wbGV0YSB0b2RvcyBsb3MgY2FtcG9zLicpO1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIHRoaXMubG9naW5Gb3JtUHJvcHMuc3RhdGUgPSBDb21wb25lbnRTdGF0ZXMuV09SS0lORztcblxuICAgIHRoaXMuYXV0aFNlcnZpY2Uuc2lnbmluKHsgZW1haWwsIHBhc3N3b3JkIH0pLnN1YnNjcmliZSh7XG4gICAgICBuZXh0OiAoKSA9PiB7XG4gICAgICAgIHRoaXMubG9naW5Gb3JtUHJvcHMuc3RhdGUgPSBDb21wb25lbnRTdGF0ZXMuRU5BQkxFRDtcblxuICAgICAgICBpZiAodGhpcy5hdXRoU2VydmljZS5tZmFQZW5kaW5nKCkucmVxdWlyZWQpIHtcbiAgICAgICAgICB0aGlzLm9wZW5NRkFWZXJpZnlNb2RhbCgpO1xuICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIHRoaXMuaGFuZGxlTG9naW5TdWNjZXNzKCk7XG4gICAgICB9LFxuICAgICAgZXJyb3I6IChlcnIpID0+IHtcbiAgICAgICAgdGhpcy5sb2dpbkZvcm1Qcm9wcy5zdGF0ZSA9IENvbXBvbmVudFN0YXRlcy5FTkFCTEVEO1xuXG4gICAgICAgIGNvbnN0IGVycm9yQ29kZSA9IChlcnIgYXMgeyBjb2RlPzogc3RyaW5nIH0pPy5jb2RlO1xuICAgICAgICBpZiAoZXJyb3JDb2RlID09PSAnQVVUSFYyX0VNQUlMX05PVF9WRVJJRklFRCcpIHtcbiAgICAgICAgICB0aGlzLm9wZW5WZXJpZnlNb2RhbChlbWFpbCk7XG4gICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG5cbiAgICAgICAgdGhpcy5oYW5kbGVFcnJvcihlcnIsICdzaWduaW4nKTtcbiAgICAgIH0sXG4gICAgfSk7XG4gIH1cblxuICBsb2dpbldpdGhPQXV0aChwcm92aWRlcjogJ2dvb2dsZScgfCAnYXBwbGUnIHwgJ21pY3Jvc29mdCcpOiB2b2lkIHtcbiAgICB0aGlzLmlzT0F1dGhMb2FkaW5nID0gdHJ1ZTtcblxuICAgIHRoaXMuYXV0aFNlcnZpY2Uuc2lnbmluV2l0aE9BdXRoKHByb3ZpZGVyKS5zdWJzY3JpYmUoe1xuICAgICAgbmV4dDogKCkgPT4ge1xuICAgICAgICB0aGlzLmlzT0F1dGhMb2FkaW5nID0gZmFsc2U7XG5cbiAgICAgICAgaWYgKHRoaXMuYXV0aFNlcnZpY2UubWZhUGVuZGluZygpLnJlcXVpcmVkKSB7XG4gICAgICAgICAgdGhpcy5vcGVuTUZBVmVyaWZ5TW9kYWwoKTtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICB0aGlzLmhhbmRsZUxvZ2luU3VjY2Vzcyhwcm92aWRlcik7XG4gICAgICB9LFxuICAgICAgZXJyb3I6IChlcnIpID0+IHtcbiAgICAgICAgdGhpcy5pc09BdXRoTG9hZGluZyA9IGZhbHNlO1xuXG4gICAgICAgIGNvbnN0IGVycm9yQ29kZSA9IChlcnIgYXMgeyBjb2RlPzogc3RyaW5nIH0pPy5jb2RlO1xuICAgICAgICBpZiAoZXJyb3JDb2RlID09PSAnUE9QVVBfQ0xPU0VEJykge1xuICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIHRoaXMuaGFuZGxlRXJyb3IoZXJyLCAnb2F1dGgnKTtcbiAgICAgIH0sXG4gICAgfSk7XG4gIH1cblxuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbiAgLy8gUkVHSVNURVIgSEFORExFUlNcbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgb3BlblJlZ2lzdGVyTW9kYWwoKTogdm9pZCB7XG4gICAgdGhpcy5pc1JlZ2lzdGVyTW9kYWxPcGVuID0gdHJ1ZTtcbiAgfVxuXG4gIGNsb3NlUmVnaXN0ZXJNb2RhbCgpOiB2b2lkIHtcbiAgICB0aGlzLmlzUmVnaXN0ZXJNb2RhbE9wZW4gPSBmYWxzZTtcbiAgfVxuXG4gIHJlZ2lzdGVySGFuZGxlcihldmVudDogRm9ybVN1Ym1pdCk6IHZvaWQge1xuICAgIGNvbnN0IG5hbWUgPSBldmVudC5maWVsZHNbJ25hbWUnXTtcbiAgICBjb25zdCBlbWFpbCA9IGV2ZW50LmZpZWxkc1snZW1haWwnXTtcbiAgICBjb25zdCBwYXNzd29yZCA9IGV2ZW50LmZpZWxkc1sncGFzc3dvcmQnXTtcblxuICAgIGlmICghbmFtZSB8fCAhZW1haWwgfHwgIXBhc3N3b3JkKSB7XG4gICAgICB0aGlzLnNob3dUb2FzdCgnQ29tcGxldGEgdG9kb3MgbG9zIGNhbXBvcy4nKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICB0aGlzLnJlZ2lzdGVyRm9ybVByb3BzLnN0YXRlID0gQ29tcG9uZW50U3RhdGVzLldPUktJTkc7XG5cbiAgICB0aGlzLmF1dGhTZXJ2aWNlLnNpZ251cCh7IG5hbWUsIGVtYWlsLCBwYXNzd29yZCB9KS5zdWJzY3JpYmUoe1xuICAgICAgbmV4dDogKCkgPT4ge1xuICAgICAgICB0aGlzLnJlZ2lzdGVyRm9ybVByb3BzLnN0YXRlID0gQ29tcG9uZW50U3RhdGVzLkVOQUJMRUQ7XG4gICAgICAgIHRoaXMuY2xvc2VSZWdpc3Rlck1vZGFsKCk7XG4gICAgICAgIHRoaXMub3BlblZlcmlmeU1vZGFsKGVtYWlsKTtcbiAgICAgIH0sXG4gICAgICBlcnJvcjogKGVycikgPT4ge1xuICAgICAgICB0aGlzLnJlZ2lzdGVyRm9ybVByb3BzLnN0YXRlID0gQ29tcG9uZW50U3RhdGVzLkVOQUJMRUQ7XG4gICAgICAgIHRoaXMuaGFuZGxlRXJyb3IoZXJyLCAnc2lnbnVwJyk7XG4gICAgICB9LFxuICAgIH0pO1xuICB9XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIFZFUklGWSBFTUFJTCBIQU5ETEVSU1xuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuICBvcGVuVmVyaWZ5TW9kYWwoZW1haWw6IHN0cmluZyk6IHZvaWQge1xuICAgIHRoaXMucGVuZGluZ1ZlcmlmaWNhdGlvbkVtYWlsID0gZW1haWw7XG4gICAgdGhpcy52ZXJpZnlGb3JtUHJvcHMuc2VjdGlvbnNbMF0ubmFtZSA9IGBJbmdyZXNhIGVsIGPDs2RpZ28gZW52aWFkbyBhICR7ZW1haWx9YDtcbiAgICB0aGlzLnZlcmlmeUZvcm1Qcm9wcy5zdGF0ZSA9IENvbXBvbmVudFN0YXRlcy5FTkFCTEVEO1xuICAgIHRoaXMuaXNWZXJpZnlNb2RhbE9wZW4gPSB0cnVlO1xuICAgIHRoaXMuc3RhcnRSZXNlbmRDb29sZG93bigpO1xuICB9XG5cbiAgY2xvc2VWZXJpZnlNb2RhbCgpOiB2b2lkIHtcbiAgICB0aGlzLmlzVmVyaWZ5TW9kYWxPcGVuID0gZmFsc2U7XG4gICAgdGhpcy5wZW5kaW5nVmVyaWZpY2F0aW9uRW1haWwgPSAnJztcbiAgICB0aGlzLnN0b3BSZXNlbmRDb29sZG93bigpO1xuICB9XG5cbiAgdmVyaWZ5SGFuZGxlcihldmVudDogRm9ybVN1Ym1pdCk6IHZvaWQge1xuICAgIGNvbnN0IGNvZGUgPSBldmVudC5maWVsZHNbJ2NvZGUnXTtcblxuICAgIHRoaXMudmVyaWZ5Rm9ybVByb3BzLnN0YXRlID0gQ29tcG9uZW50U3RhdGVzLldPUktJTkc7XG5cbiAgICB0aGlzLmF1dGhTZXJ2aWNlXG4gICAgICAudmVyaWZ5RW1haWwoe1xuICAgICAgICBlbWFpbDogdGhpcy5wZW5kaW5nVmVyaWZpY2F0aW9uRW1haWwsXG4gICAgICAgIGNvZGUsXG4gICAgICB9KVxuICAgICAgLnN1YnNjcmliZSh7XG4gICAgICAgIG5leHQ6ICgpID0+IHtcbiAgICAgICAgICB0aGlzLnZlcmlmeUZvcm1Qcm9wcy5zdGF0ZSA9IENvbXBvbmVudFN0YXRlcy5FTkFCTEVEO1xuICAgICAgICAgIHRoaXMuc2hvd1RvYXN0KCfCoUVtYWlsIHZlcmlmaWNhZG8hIEJpZW52ZW5pZG8uJyk7XG4gICAgICAgICAgdGhpcy5jbG9zZVZlcmlmeU1vZGFsKCk7XG4gICAgICAgICAgdGhpcy5oYW5kbGVMb2dpblN1Y2Nlc3MoKTtcbiAgICAgICAgfSxcbiAgICAgICAgZXJyb3I6IChlcnIpID0+IHtcbiAgICAgICAgICB0aGlzLnZlcmlmeUZvcm1Qcm9wcy5zdGF0ZSA9IENvbXBvbmVudFN0YXRlcy5FTkFCTEVEO1xuICAgICAgICAgIHRoaXMuaGFuZGxlRXJyb3IoZXJyLCAndmVyaWZ5Jyk7XG4gICAgICAgIH0sXG4gICAgICB9KTtcbiAgfVxuXG4gIHJlc2VuZENvZGUoKTogdm9pZCB7XG4gICAgaWYgKHRoaXMucmVzZW5kQ29vbGRvd24gPiAwKSByZXR1cm47XG5cbiAgICB0aGlzLmF1dGhTZXJ2aWNlXG4gICAgICAucmVzZW5kQ29kZSh7XG4gICAgICAgIGVtYWlsOiB0aGlzLnBlbmRpbmdWZXJpZmljYXRpb25FbWFpbCxcbiAgICAgICAgdHlwZTogJ0VNQUlMX1ZFUklGWScsXG4gICAgICB9KVxuICAgICAgLnN1YnNjcmliZSh7XG4gICAgICAgIG5leHQ6ICgpID0+IHtcbiAgICAgICAgICB0aGlzLnNob3dUb2FzdCgnQ8OzZGlnbyByZWVudmlhZG8uIFJldmlzYSB0dSBlbWFpbC4nKTtcbiAgICAgICAgICB0aGlzLnN0YXJ0UmVzZW5kQ29vbGRvd24oKTtcbiAgICAgICAgfSxcbiAgICAgICAgZXJyb3I6IChlcnIpID0+IHtcbiAgICAgICAgICB0aGlzLmhhbmRsZUVycm9yKGVyciwgJ3ZlcmlmeScpO1xuICAgICAgICB9LFxuICAgICAgfSk7XG4gIH1cblxuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbiAgLy8gTUZBIEhBTkRMRVJTXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4gIG9wZW5NRkFWZXJpZnlNb2RhbCgpOiB2b2lkIHtcbiAgICBjb25zdCBtZXRob2QgPSB0aGlzLmF1dGhTZXJ2aWNlLm1mYVBlbmRpbmcoKS5tZXRob2Q7XG4gICAgdGhpcy5tZmFWZXJpZnlGb3JtUHJvcHMuc2VjdGlvbnNbMF0ubmFtZSA9XG4gICAgICBtZXRob2QgPT09ICdUT1RQJ1xuICAgICAgICA/ICdJbmdyZXNhIGVsIGPDs2RpZ28gZGUgdHUgYXBwIGRlIGF1dGVudGljYWNpw7NuJ1xuICAgICAgICA6IG1ldGhvZCA9PT0gJ0VNQUlMJ1xuICAgICAgICAgID8gJ0luZ3Jlc2EgZWwgY8OzZGlnbyBlbnZpYWRvIGEgdHUgY29ycmVvJ1xuICAgICAgICAgIDogJ0luZ3Jlc2EgZWwgY8OzZGlnbyBlbnZpYWRvIGEgdHUgdGVsw6lmb25vJztcbiAgICB0aGlzLm1mYVZlcmlmeUZvcm1Qcm9wcy5zdGF0ZSA9IENvbXBvbmVudFN0YXRlcy5FTkFCTEVEO1xuICAgIHRoaXMuaXNNRkFWZXJpZnlNb2RhbE9wZW4gPSB0cnVlO1xuXG4gICAgdGhpcy5vbk1GQVJlcXVpcmVkLmVtaXQoeyBtZXRob2Q6IG1ldGhvZCBhcyAnVE9UUCcgfCAnRU1BSUwnIHwgJ1NNUycgfSk7XG4gIH1cblxuICBjbG9zZU1GQVZlcmlmeU1vZGFsKCk6IHZvaWQge1xuICAgIHRoaXMuaXNNRkFWZXJpZnlNb2RhbE9wZW4gPSBmYWxzZTtcbiAgfVxuXG4gIHZlcmlmeU1GQUhhbmRsZXIoZXZlbnQ6IEZvcm1TdWJtaXQpOiB2b2lkIHtcbiAgICBjb25zdCBjb2RlID0gZXZlbnQuZmllbGRzWydjb2RlJ107XG5cbiAgICB0aGlzLm1mYVZlcmlmeUZvcm1Qcm9wcy5zdGF0ZSA9IENvbXBvbmVudFN0YXRlcy5XT1JLSU5HO1xuXG4gICAgdGhpcy5hdXRoU2VydmljZS52ZXJpZnlNRkEoY29kZSkuc3Vic2NyaWJlKHtcbiAgICAgIG5leHQ6ICgpID0+IHtcbiAgICAgICAgdGhpcy5tZmFWZXJpZnlGb3JtUHJvcHMuc3RhdGUgPSBDb21wb25lbnRTdGF0ZXMuRU5BQkxFRDtcbiAgICAgICAgdGhpcy5jbG9zZU1GQVZlcmlmeU1vZGFsKCk7XG4gICAgICAgIHRoaXMuaGFuZGxlTG9naW5TdWNjZXNzKHVuZGVmaW5lZCwgdHJ1ZSk7XG4gICAgICB9LFxuICAgICAgZXJyb3I6IChlcnIpID0+IHtcbiAgICAgICAgdGhpcy5tZmFWZXJpZnlGb3JtUHJvcHMuc3RhdGUgPSBDb21wb25lbnRTdGF0ZXMuRU5BQkxFRDtcbiAgICAgICAgdGhpcy5oYW5kbGVFcnJvcihlcnIsICdtZmEnKTtcbiAgICAgIH0sXG4gICAgfSk7XG4gIH1cblxuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbiAgLy8gRk9SR09UIFBBU1NXT1JEIEhBTkRMRVJTXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4gIG9wZW5Gb3Jnb3RQYXNzd29yZE1vZGFsKCk6IHZvaWQge1xuICAgIHRoaXMuaXNGb3Jnb3RQYXNzd29yZE1vZGFsT3BlbiA9IHRydWU7XG4gIH1cblxuICBjbG9zZUZvcmdvdFBhc3N3b3JkTW9kYWwoKTogdm9pZCB7XG4gICAgdGhpcy5pc0ZvcmdvdFBhc3N3b3JkTW9kYWxPcGVuID0gZmFsc2U7XG4gIH1cblxuICBmb3Jnb3RQYXNzd29yZEhhbmRsZXIoZXZlbnQ6IEZvcm1TdWJtaXQpOiB2b2lkIHtcbiAgICBjb25zdCBlbWFpbCA9IGV2ZW50LmZpZWxkc1snZW1haWwnXTtcblxuICAgIGlmICghZW1haWwpIHtcbiAgICAgIHRoaXMuc2hvd1RvYXN0KCdJbmdyZXNhIHR1IGNvcnJlbyBlbGVjdHLDs25pY28uJyk7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgdGhpcy5mb3Jnb3RQYXNzd29yZEZvcm1Qcm9wcy5zdGF0ZSA9IENvbXBvbmVudFN0YXRlcy5XT1JLSU5HO1xuXG4gICAgdGhpcy5hdXRoU2VydmljZS5mb3Jnb3RQYXNzd29yZCh7IGVtYWlsIH0pLnN1YnNjcmliZSh7XG4gICAgICBuZXh0OiAoKSA9PiB7XG4gICAgICAgIHRoaXMuZm9yZ290UGFzc3dvcmRGb3JtUHJvcHMuc3RhdGUgPSBDb21wb25lbnRTdGF0ZXMuRU5BQkxFRDtcbiAgICAgICAgdGhpcy5jbG9zZUZvcmdvdFBhc3N3b3JkTW9kYWwoKTtcbiAgICAgICAgdGhpcy5vcGVuUmVzZXRQYXNzd29yZE1vZGFsKGVtYWlsKTtcbiAgICAgICAgdGhpcy5zaG93VG9hc3QoJ0PDs2RpZ28gZW52aWFkby4gUmV2aXNhIHR1IGVtYWlsLicpO1xuICAgICAgfSxcbiAgICAgIGVycm9yOiAoZXJyKSA9PiB7XG4gICAgICAgIHRoaXMuZm9yZ290UGFzc3dvcmRGb3JtUHJvcHMuc3RhdGUgPSBDb21wb25lbnRTdGF0ZXMuRU5BQkxFRDtcbiAgICAgICAgdGhpcy5oYW5kbGVFcnJvcihlcnIsICdmb3Jnb3QnKTtcbiAgICAgIH0sXG4gICAgfSk7XG4gIH1cblxuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbiAgLy8gUkVTRVQgUEFTU1dPUkQgSEFORExFUlNcbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgb3BlblJlc2V0UGFzc3dvcmRNb2RhbChlbWFpbDogc3RyaW5nKTogdm9pZCB7XG4gICAgdGhpcy5wZW5kaW5nUmVzZXRFbWFpbCA9IGVtYWlsO1xuICAgIHRoaXMucmVzZXRQYXNzd29yZEZvcm1Qcm9wcy5zZWN0aW9uc1swXS5uYW1lID0gYEluZ3Jlc2EgZWwgY8OzZGlnbyBlbnZpYWRvIGEgJHtlbWFpbH1gO1xuICAgIHRoaXMucmVzZXRQYXNzd29yZEZvcm1Qcm9wcy5zdGF0ZSA9IENvbXBvbmVudFN0YXRlcy5FTkFCTEVEO1xuICAgIHRoaXMuaXNSZXNldFBhc3N3b3JkTW9kYWxPcGVuID0gdHJ1ZTtcbiAgICB0aGlzLnN0YXJ0UmVzZXRSZXNlbmRDb29sZG93bigpO1xuICB9XG5cbiAgY2xvc2VSZXNldFBhc3N3b3JkTW9kYWwoKTogdm9pZCB7XG4gICAgdGhpcy5pc1Jlc2V0UGFzc3dvcmRNb2RhbE9wZW4gPSBmYWxzZTtcbiAgICB0aGlzLnBlbmRpbmdSZXNldEVtYWlsID0gJyc7XG4gICAgdGhpcy5zdG9wUmVzZXRSZXNlbmRDb29sZG93bigpO1xuICB9XG5cbiAgcmVzZXRQYXNzd29yZEhhbmRsZXIoZXZlbnQ6IEZvcm1TdWJtaXQpOiB2b2lkIHtcbiAgICBjb25zdCBjb2RlID0gZXZlbnQuZmllbGRzWydjb2RlJ107XG4gICAgY29uc3QgbmV3UGFzc3dvcmQgPSBldmVudC5maWVsZHNbJ25ld1Bhc3N3b3JkJ107XG5cbiAgICB0aGlzLnJlc2V0UGFzc3dvcmRGb3JtUHJvcHMuc3RhdGUgPSBDb21wb25lbnRTdGF0ZXMuV09SS0lORztcblxuICAgIHRoaXMuYXV0aFNlcnZpY2VcbiAgICAgIC5yZXNldFBhc3N3b3JkKHtcbiAgICAgICAgZW1haWw6IHRoaXMucGVuZGluZ1Jlc2V0RW1haWwsXG4gICAgICAgIGNvZGUsXG4gICAgICAgIG5ld1Bhc3N3b3JkLFxuICAgICAgfSlcbiAgICAgIC5zdWJzY3JpYmUoe1xuICAgICAgICBuZXh0OiAoKSA9PiB7XG4gICAgICAgICAgdGhpcy5yZXNldFBhc3N3b3JkRm9ybVByb3BzLnN0YXRlID0gQ29tcG9uZW50U3RhdGVzLkVOQUJMRUQ7XG4gICAgICAgICAgdGhpcy5zaG93VG9hc3QoJ8KhQ29udHJhc2XDsWEgYWN0dWFsaXphZGEhIFlhIHB1ZWRlcyBpbmljaWFyIHNlc2nDs24uJyk7XG4gICAgICAgICAgdGhpcy5jbG9zZVJlc2V0UGFzc3dvcmRNb2RhbCgpO1xuICAgICAgICB9LFxuICAgICAgICBlcnJvcjogKGVycikgPT4ge1xuICAgICAgICAgIHRoaXMucmVzZXRQYXNzd29yZEZvcm1Qcm9wcy5zdGF0ZSA9IENvbXBvbmVudFN0YXRlcy5FTkFCTEVEO1xuICAgICAgICAgIHRoaXMuaGFuZGxlRXJyb3IoZXJyLCAncmVzZXQnKTtcbiAgICAgICAgfSxcbiAgICAgIH0pO1xuICB9XG5cbiAgcmVzZW5kUmVzZXRDb2RlKCk6IHZvaWQge1xuICAgIGlmICh0aGlzLnJlc2V0UmVzZW5kQ29vbGRvd24gPiAwKSByZXR1cm47XG5cbiAgICB0aGlzLmF1dGhTZXJ2aWNlXG4gICAgICAucmVzZW5kQ29kZSh7XG4gICAgICAgIGVtYWlsOiB0aGlzLnBlbmRpbmdSZXNldEVtYWlsLFxuICAgICAgICB0eXBlOiAnUEFTU1dPUkRfUkVTRVQnLFxuICAgICAgfSlcbiAgICAgIC5zdWJzY3JpYmUoe1xuICAgICAgICBuZXh0OiAoKSA9PiB7XG4gICAgICAgICAgdGhpcy5zaG93VG9hc3QoJ0PDs2RpZ28gcmVlbnZpYWRvLiBSZXZpc2EgdHUgZW1haWwuJyk7XG4gICAgICAgICAgdGhpcy5zdGFydFJlc2V0UmVzZW5kQ29vbGRvd24oKTtcbiAgICAgICAgfSxcbiAgICAgICAgZXJyb3I6IChlcnIpID0+IHtcbiAgICAgICAgICB0aGlzLmhhbmRsZUVycm9yKGVyciwgJ3Jlc2V0Jyk7XG4gICAgICAgIH0sXG4gICAgICB9KTtcbiAgfVxuXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuICAvLyBIRUxQRVJTXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4gIHByaXZhdGUgaGFuZGxlTG9naW5TdWNjZXNzKG9hdXRoUHJvdmlkZXI/OiAnZ29vZ2xlJyB8ICdhcHBsZScgfCAnbWljcm9zb2Z0JywgbWZhQ29tcGxldGVkPzogYm9vbGVhbik6IHZvaWQge1xuICAgIHRoaXMuc2hvd1RvYXN0KCfCoUJpZW52ZW5pZG8hJyk7XG5cbiAgICB0aGlzLm9uU3VjY2Vzcy5lbWl0KHtcbiAgICAgIHVzZXI6IHRoaXMuYXV0aFNlcnZpY2UudXNlcigpLFxuICAgICAgbWZhQ29tcGxldGVkLFxuICAgICAgb2F1dGhQcm92aWRlcixcbiAgICB9KTtcblxuICAgIGlmICh0aGlzLnByb3BzLnJlZGlyZWN0T25TdWNjZXNzKSB7XG4gICAgICB0aGlzLnJvdXRlci5uYXZpZ2F0ZShbdGhpcy5wcm9wcy5yZWRpcmVjdE9uU3VjY2Vzc10pO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgaGFuZGxlRXJyb3IoZXJyOiB1bmtub3duLCBvcGVyYXRpb246IExvZ2luRXJyb3JFdmVudFsnb3BlcmF0aW9uJ10pOiB2b2lkIHtcbiAgICBjb25zdCBtZXNzYWdlID0gdGhpcy5nZXRFcnJvck1lc3NhZ2UoZXJyKTtcbiAgICB0aGlzLnNob3dUb2FzdChtZXNzYWdlKTtcblxuICAgIHRoaXMub25FcnJvci5lbWl0KHtcbiAgICAgIGNvZGU6IChlcnIgYXMgeyBjb2RlPzogc3RyaW5nIH0pPy5jb2RlLFxuICAgICAgbWVzc2FnZSxcbiAgICAgIG9wZXJhdGlvbixcbiAgICB9KTtcbiAgfVxuXG4gIHByaXZhdGUgc2hvd1RvYXN0KG1lc3NhZ2U6IHN0cmluZyk6IHZvaWQge1xuICAgIHRoaXMudG9hc3RTZXJ2aWNlLnNob3coe1xuICAgICAgbWVzc2FnZSxcbiAgICAgIGR1cmF0aW9uOiAzNTAwLFxuICAgICAgcG9zaXRpb246ICd0b3AnLFxuICAgICAgY29sb3I6ICdkYXJrJyxcbiAgICB9KTtcbiAgfVxuXG4gIHByaXZhdGUgZ2V0RXJyb3JNZXNzYWdlKGVycjogdW5rbm93bik6IHN0cmluZyB7XG4gICAgY29uc3QgZXJyb3IgPSBlcnIgYXMge1xuICAgICAgZXJyb3I/OiB7IG1lc3NhZ2U/OiBzdHJpbmc7IGNvZGU/OiBzdHJpbmcgfTtcbiAgICAgIGNvZGU/OiBzdHJpbmc7XG4gICAgfTtcblxuICAgIGNvbnN0IGVycm9yTWVzc2FnZXM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4gPSB7XG4gICAgICAvLyBWYWxpZGF0aW9uXG4gICAgICBWQUxJREFUSU9OX01JU1NJTkdfUkVRVUlSRURfRklFTERTOiAnRmFsdGFuIGNhbXBvcyByZXF1ZXJpZG9zJyxcbiAgICAgIFZBTElEQVRJT05fSU5WQUxJRF9SRVFVRVNUOiAnU29saWNpdHVkIGludsOhbGlkYScsXG5cbiAgICAgIC8vIFNpZ25pblxuICAgICAgQVVUSFYyX0lOVkFMSURfQ1JFREVOVElBTFM6ICdDb3JyZW8gbyBjb250cmFzZcOxYSBpbmNvcnJlY3RvcycsXG4gICAgICBBVVRIVjJfRU1BSUxfTk9UX1ZFUklGSUVEOiAnRGViZXMgdmVyaWZpY2FyIHR1IGNvcnJlbyBlbGVjdHLDs25pY28nLFxuICAgICAgQVVUSFYyX0FDQ09VTlRfU1VTUEVOREVEOiAnVHUgY3VlbnRhIGhhIHNpZG8gc3VzcGVuZGlkYScsXG4gICAgICBBVVRIVjJfU0lHTklOX0ZBSUxFRDogJ0Vycm9yIGFsIGluaWNpYXIgc2VzacOzbicsXG5cbiAgICAgIC8vIFNpZ251cFxuICAgICAgQVVUSFYyX0VNQUlMX0VYSVNUUzogJ0VzdGUgY29ycmVvIHlhIGVzdMOhIHJlZ2lzdHJhZG8nLFxuICAgICAgQVVUSFYyX1BIT05FX0VYSVNUUzogJ0VzdGUgdGVsw6lmb25vIHlhIGVzdMOhIHJlZ2lzdHJhZG8nLFxuICAgICAgQVVUSFYyX1dFQUtfUEFTU1dPUkQ6ICdMYSBjb250cmFzZcOxYSBlcyBtdXkgZMOpYmlsJyxcbiAgICAgIEFVVEhWMl9TSUdOVVBfRkFJTEVEOiAnRXJyb3IgYWwgY3JlYXIgbGEgY3VlbnRhJyxcblxuICAgICAgLy8gVmVyaWZpY2F0aW9uIGNvZGVzXG4gICAgICBBVVRIVjJfSU5WQUxJRF9DT0RFOiAnQ8OzZGlnbyBpbmNvcnJlY3RvJyxcbiAgICAgIEFVVEhWMl9FWFBJUkVEX0NPREU6ICdFbCBjw7NkaWdvIGhhIGV4cGlyYWRvLiBTb2xpY2l0YSB1bm8gbnVldm8uJyxcbiAgICAgIEFVVEhWMl9DT0RFX0VYUElSRUQ6ICdFbCBjw7NkaWdvIGhhIGV4cGlyYWRvLiBTb2xpY2l0YSB1bm8gbnVldm8uJyxcbiAgICAgIEFVVEhWMl9DT0RFX0FMUkVBRFlfVVNFRDogJ0VzdGUgY8OzZGlnbyB5YSBmdWUgdXRpbGl6YWRvJyxcbiAgICAgIEFVVEhWMl9UT09fTUFOWV9BVFRFTVBUUzogJ0RlbWFzaWFkb3MgaW50ZW50b3MsIGludGVudGEgbcOhcyB0YXJkZScsXG4gICAgICBBVVRIVjJfU0VORF9DT0RFX0ZBSUxFRDogJ0Vycm9yIGFsIGVudmlhciBlbCBjw7NkaWdvJyxcbiAgICAgIEFVVEhWMl9BTFJFQURZX1ZFUklGSUVEOiAnRXN0ZSBlbWFpbCB5YSBlc3TDoSB2ZXJpZmljYWRvJyxcblxuICAgICAgLy8gTUZBXG4gICAgICBBVVRIVjJfTUZBX0lOVkFMSURfQ09ERTogJ0PDs2RpZ28gZGUgdmVyaWZpY2FjacOzbiBpbmNvcnJlY3RvJyxcblxuICAgICAgLy8gUGFzc3dvcmQgcmVzZXRcbiAgICAgIEFVVEhWMl9QQVNTV09SRF9SRVNFVF9GQUlMRUQ6ICdFcnJvciBhbCByZXN0YWJsZWNlciBsYSBjb250cmFzZcOxYScsXG4gICAgICBBVVRIVjJfSU5WQUxJRF9DVVJSRU5UX1BBU1NXT1JEOiAnQ29udHJhc2XDsWEgYWN0dWFsIGluY29ycmVjdGEnLFxuICAgICAgQVVUSFYyX1NBTUVfUEFTU1dPUkQ6ICdMYSBudWV2YSBjb250cmFzZcOxYSBkZWJlIHNlciBkaWZlcmVudGUnLFxuXG4gICAgICAvLyBTZXNzaW9uL1Rva2Vuc1xuICAgICAgQVVUSFYyX0lOVkFMSURfVE9LRU46ICdTZXNpw7NuIGludsOhbGlkYScsXG4gICAgICBBVVRIVjJfRVhQSVJFRF9UT0tFTjogJ1R1IHNlc2nDs24gaGEgZXhwaXJhZG8nLFxuICAgICAgQVVUSFYyX1NFU1NJT05fRVhQSVJFRDogJ1R1IHNlc2nDs24gaGEgZXhwaXJhZG8nLFxuXG4gICAgICAvLyBVc2VyXG4gICAgICBBVVRIVjJfVVNFUl9OT1RfRk9VTkQ6ICdVc3VhcmlvIG5vIGVuY29udHJhZG8nLFxuXG4gICAgICAvLyBPQXV0aFxuICAgICAgUE9QVVBfQkxPQ0tFRDogJ1BvciBmYXZvciwgcGVybWl0ZSB2ZW50YW5hcyBlbWVyZ2VudGVzIHBhcmEgZXN0ZSBzaXRpbycsXG4gICAgICBQT1BVUF9DTE9TRUQ6ICdTZSBjYW5jZWzDsyBsYSBhdXRlbnRpY2FjacOzbicsXG4gICAgICBJTlZBTElEX1JFU1BPTlNFOiAnRXJyb3IgZW4gbGEgcmVzcHVlc3RhIGRlbCBzZXJ2aWRvcicsXG4gICAgICBPQVVUSF9GQUlMRUQ6ICdFcnJvciBkZSBhdXRlbnRpY2FjacOzbicsXG4gICAgICBPQVVUSF9FTUFJTF9FWElTVFM6ICdFc3RlIGNvcnJlbyB5YSBlc3TDoSByZWdpc3RyYWRvIGNvbiBvdHJvIG3DqXRvZG8nLFxuICAgIH07XG5cbiAgICBjb25zdCBlcnJvckNvZGUgPSBlcnJvcj8uZXJyb3I/LmNvZGUgfHwgZXJyb3I/LmNvZGU7XG4gICAgaWYgKGVycm9yQ29kZSAmJiBlcnJvck1lc3NhZ2VzW2Vycm9yQ29kZV0pIHtcbiAgICAgIHJldHVybiBlcnJvck1lc3NhZ2VzW2Vycm9yQ29kZV07XG4gICAgfVxuXG4gICAgaWYgKGVycm9yPy5lcnJvcj8ubWVzc2FnZSkge1xuICAgICAgcmV0dXJuIGVycm9yLmVycm9yLm1lc3NhZ2U7XG4gICAgfVxuXG4gICAgcmV0dXJuICdIYSBvY3VycmlkbyB1biBlcnJvci4gSW50ZW50YSBkZSBudWV2by4nO1xuICB9XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4gIC8vIENPT0xET1dOIFRJTUVSU1xuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuICBwcml2YXRlIHN0YXJ0UmVzZW5kQ29vbGRvd24oKTogdm9pZCB7XG4gICAgdGhpcy5yZXNlbmRDb29sZG93biA9IDMwO1xuICAgIHRoaXMucmVzZW5kVGltZXIgPSBzZXRJbnRlcnZhbCgoKSA9PiB7XG4gICAgICB0aGlzLnJlc2VuZENvb2xkb3duLS07XG4gICAgICBpZiAodGhpcy5yZXNlbmRDb29sZG93biA8PSAwKSB7XG4gICAgICAgIHRoaXMuc3RvcFJlc2VuZENvb2xkb3duKCk7XG4gICAgICB9XG4gICAgfSwgMTAwMCk7XG4gIH1cblxuICBwcml2YXRlIHN0b3BSZXNlbmRDb29sZG93bigpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5yZXNlbmRUaW1lcikge1xuICAgICAgY2xlYXJJbnRlcnZhbCh0aGlzLnJlc2VuZFRpbWVyKTtcbiAgICAgIHRoaXMucmVzZW5kVGltZXIgPSBudWxsO1xuICAgIH1cbiAgICB0aGlzLnJlc2VuZENvb2xkb3duID0gMDtcbiAgfVxuXG4gIHByaXZhdGUgc3RhcnRSZXNldFJlc2VuZENvb2xkb3duKCk6IHZvaWQge1xuICAgIHRoaXMucmVzZXRSZXNlbmRDb29sZG93biA9IDMwO1xuICAgIHRoaXMucmVzZXRSZXNlbmRUaW1lciA9IHNldEludGVydmFsKCgpID0+IHtcbiAgICAgIHRoaXMucmVzZXRSZXNlbmRDb29sZG93bi0tO1xuICAgICAgaWYgKHRoaXMucmVzZXRSZXNlbmRDb29sZG93biA8PSAwKSB7XG4gICAgICAgIHRoaXMuc3RvcFJlc2V0UmVzZW5kQ29vbGRvd24oKTtcbiAgICAgIH1cbiAgICB9LCAxMDAwKTtcbiAgfVxuXG4gIHByaXZhdGUgc3RvcFJlc2V0UmVzZW5kQ29vbGRvd24oKTogdm9pZCB7XG4gICAgaWYgKHRoaXMucmVzZXRSZXNlbmRUaW1lcikge1xuICAgICAgY2xlYXJJbnRlcnZhbCh0aGlzLnJlc2V0UmVzZW5kVGltZXIpO1xuICAgICAgdGhpcy5yZXNldFJlc2VuZFRpbWVyID0gbnVsbDtcbiAgICB9XG4gICAgdGhpcy5yZXNldFJlc2VuZENvb2xkb3duID0gMDtcbiAgfVxuXG4gIG5nT25EZXN0cm95KCk6IHZvaWQge1xuICAgIHRoaXMuc3RvcFJlc2VuZENvb2xkb3duKCk7XG4gICAgdGhpcy5zdG9wUmVzZXRSZXNlbmRDb29sZG93bigpO1xuICB9XG59XG4iLCI8ZGl2IGNsYXNzPVwidmFsLWxvZ2luXCI+XG4gIDwhLS0gTG9nbyAtLT5cbiAgQGlmIChwcm9wcy5sb2dvKSB7XG4gICAgPGRpdiBjbGFzcz1cImxvZ28tY29udGFpbmVyXCI+XG4gICAgICA8dmFsLWltYWdlIFtwcm9wc109XCJwcm9wcy5sb2dvXCIgLz5cbiAgICA8L2Rpdj5cbiAgfVxuXG4gIDwhLS0gTG9naW4gRm9ybSAtLT5cbiAgPHZhbC1mb3JtIFtwcm9wc109XCJsb2dpbkZvcm1Qcm9wc1wiIChvblN1Ym1pdCk9XCJsb2dpbkhhbmRsZXIoJGV2ZW50KVwiIC8+XG5cbiAgPCEtLSBPQXV0aCBTZWN0aW9uIC0tPlxuICBAaWYgKGNvbmZpZy5zaG93T0F1dGggJiYgY29uZmlnLm9hdXRoUHJvdmlkZXJzLmxlbmd0aCA+IDApIHtcbiAgICA8ZGl2IGNsYXNzPVwib2F1dGgtc2VwYXJhdG9yXCI+XG4gICAgICA8c3Bhbj5vIGNvbnRpbsO6YSBjb248L3NwYW4+XG4gICAgPC9kaXY+XG5cbiAgICA8ZGl2IGNsYXNzPVwib2F1dGgtYnV0dG9uc1wiPlxuICAgICAgQGZvciAocHJvdmlkZXIgb2YgY29uZmlnLm9hdXRoUHJvdmlkZXJzOyB0cmFjayBwcm92aWRlcikge1xuICAgICAgICA8aW9uLWJ1dHRvblxuICAgICAgICAgIGV4cGFuZD1cImJsb2NrXCJcbiAgICAgICAgICBmaWxsPVwib3V0bGluZVwiXG4gICAgICAgICAgY29sb3I9XCJkYXJrXCJcbiAgICAgICAgICAoY2xpY2spPVwibG9naW5XaXRoT0F1dGgocHJvdmlkZXIpXCJcbiAgICAgICAgICBbZGlzYWJsZWRdPVwiaXNPQXV0aExvYWRpbmdcIlxuICAgICAgICA+XG4gICAgICAgICAgQHN3aXRjaCAocHJvdmlkZXIpIHtcbiAgICAgICAgICAgIEBjYXNlICgnZ29vZ2xlJykge1xuICAgICAgICAgICAgICA8aW9uLWljb24gc2xvdD1cInN0YXJ0XCIgbmFtZT1cImxvZ28tZ29vZ2xlXCI+PC9pb24taWNvbj5cbiAgICAgICAgICAgICAge3sgaXNPQXV0aExvYWRpbmcgPyAnQ29uZWN0YW5kby4uLicgOiAnQ29udGludWFyIGNvbiBHb29nbGUnIH19XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBAY2FzZSAoJ2FwcGxlJykge1xuICAgICAgICAgICAgICA8aW9uLWljb24gc2xvdD1cInN0YXJ0XCIgbmFtZT1cImxvZ28tYXBwbGVcIj48L2lvbi1pY29uPlxuICAgICAgICAgICAgICB7eyBpc09BdXRoTG9hZGluZyA/ICdDb25lY3RhbmRvLi4uJyA6ICdDb250aW51YXIgY29uIEFwcGxlJyB9fVxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgQGNhc2UgKCdtaWNyb3NvZnQnKSB7XG4gICAgICAgICAgICAgIDxpb24taWNvbiBzbG90PVwic3RhcnRcIiBuYW1lPVwibG9nby1taWNyb3NvZnRcIj48L2lvbi1pY29uPlxuICAgICAgICAgICAgICB7eyBpc09BdXRoTG9hZGluZyA/ICdDb25lY3RhbmRvLi4uJyA6ICdDb250aW51YXIgY29uIE1pY3Jvc29mdCcgfX1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgIH1cbiAgICA8L2Rpdj5cbiAgfVxuXG4gIDwhLS0gUmVnaXN0ZXIgTGluayAtLT5cbiAgQGlmIChjb25maWcuc2hvd1JlZ2lzdGVyKSB7XG4gICAgPGRpdiBjbGFzcz1cImF1dGgtbGlua1wiPlxuICAgICAgPGlvbi10ZXh0IGNvbG9yPVwiZGFya1wiPlxuICAgICAgICDCv05vIHRpZW5lcyBjdWVudGE/XG4gICAgICAgIDxhIChjbGljayk9XCJvcGVuUmVnaXN0ZXJNb2RhbCgpXCI+UmVnaXN0cmFyc2U8L2E+XG4gICAgICA8L2lvbi10ZXh0PlxuICAgIDwvZGl2PlxuICB9XG5cbiAgPCEtLSBGb3Jnb3QgUGFzc3dvcmQgTGluayAtLT5cbiAgQGlmIChjb25maWcuc2hvd0ZvcmdvdFBhc3N3b3JkKSB7XG4gICAgPGRpdiBjbGFzcz1cImF1dGgtbGluayBmb3Jnb3QtcGFzc3dvcmRcIj5cbiAgICAgIDxpb24tdGV4dCBjb2xvcj1cImRhcmtcIj5cbiAgICAgICAgwr9PbHZpZGFzdGUgdHUgY29udHJhc2XDsWE/XG4gICAgICAgIDxhIChjbGljayk9XCJvcGVuRm9yZ290UGFzc3dvcmRNb2RhbCgpXCI+UmVjdXBlcmFyIGNvbnRyYXNlw7FhPC9hPlxuICAgICAgPC9pb24tdGV4dD5cbiAgICA8L2Rpdj5cbiAgfVxuXG4gIDwhLS0gTGVnYWwgTm90aWNlIC0tPlxuICBAaWYgKHByb3BzLmxlZ2FsKSB7XG4gICAgPGRpdiBjbGFzcz1cImxlZ2FsLW5vdGljZVwiPlxuICAgICAgPGlvbi10ZXh0IGNvbG9yPVwibWVkaXVtXCI+XG4gICAgICAgIDxwPlxuICAgICAgICAgIFV0aWxpemFtb3MgbG9zIHNlcnZpY2lvcyBkZVxuICAgICAgICAgIEBpZiAocHJvcHMubGVnYWwuY29tcGFueUxpbmspIHtcbiAgICAgICAgICAgIDxhIFtocmVmXT1cInByb3BzLmxlZ2FsLmNvbXBhbnlMaW5rXCI+PHN0cm9uZz57eyBwcm9wcy5sZWdhbC5jb21wYW55TmFtZSB9fTwvc3Ryb25nPjwvYT5cbiAgICAgICAgICB9IEBlbHNlIHtcbiAgICAgICAgICAgIDxzdHJvbmc+e3sgcHJvcHMubGVnYWwuY29tcGFueU5hbWUgfX08L3N0cm9uZz5cbiAgICAgICAgICB9XG4gICAgICAgICAgcGFyYSBvZnJlY2VydGUgdW5hIGV4cGVyaWVuY2lhIHNlZ3VyYS4gQWwgaW5pY2lhciBzZXNpw7NuLCBhY2VwdGFzXG4gICAgICAgICAgbnVlc3Ryb3NcbiAgICAgICAgICBAaWYgKHByb3BzLmxlZ2FsLnRlcm1zTGluaykge1xuICAgICAgICAgICAgPGEgW2hyZWZdPVwicHJvcHMubGVnYWwudGVybXNMaW5rXCI+VMOpcm1pbm9zIHkgQ29uZGljaW9uZXM8L2E+XG4gICAgICAgICAgfSBAZWxzZSB7XG4gICAgICAgICAgICA8c3Bhbj5Uw6lybWlub3MgeSBDb25kaWNpb25lczwvc3Bhbj5cbiAgICAgICAgICB9XG4gICAgICAgICAgeVxuICAgICAgICAgIEBpZiAocHJvcHMubGVnYWwucHJpdmFjeUxpbmspIHtcbiAgICAgICAgICAgIDxhIFtocmVmXT1cInByb3BzLmxlZ2FsLnByaXZhY3lMaW5rXCI+UG9sw610aWNhIGRlIFByaXZhY2lkYWQ8L2E+XG4gICAgICAgICAgfSBAZWxzZSB7XG4gICAgICAgICAgICA8c3Bhbj5Qb2zDrXRpY2EgZGUgUHJpdmFjaWRhZDwvc3Bhbj5cbiAgICAgICAgICB9LlxuICAgICAgICA8L3A+XG4gICAgICA8L2lvbi10ZXh0PlxuICAgIDwvZGl2PlxuICB9XG48L2Rpdj5cblxuPCEtLSBSZWdpc3RlciBNb2RhbCAtLT5cbjxpb24tbW9kYWwgW2lzT3Blbl09XCJpc1JlZ2lzdGVyTW9kYWxPcGVuXCIgKGRpZERpc21pc3MpPVwiY2xvc2VSZWdpc3Rlck1vZGFsKClcIj5cbiAgPG5nLXRlbXBsYXRlPlxuICAgIDxpb24taGVhZGVyPlxuICAgICAgPGlvbi10b29sYmFyPlxuICAgICAgICA8aW9uLWJ1dHRvbnMgc2xvdD1cImVuZFwiPlxuICAgICAgICAgIDxpb24tYnV0dG9uIGZpbGw9XCJjbGVhclwiIChjbGljayk9XCJjbG9zZVJlZ2lzdGVyTW9kYWwoKVwiPlxuICAgICAgICAgICAgPGlvbi1pY29uIG5hbWU9XCJjbG9zZS1vdXRsaW5lXCI+PC9pb24taWNvbj5cbiAgICAgICAgICA8L2lvbi1idXR0b24+XG4gICAgICAgIDwvaW9uLWJ1dHRvbnM+XG4gICAgICA8L2lvbi10b29sYmFyPlxuICAgIDwvaW9uLWhlYWRlcj5cbiAgICA8aW9uLWNvbnRlbnQgY2xhc3M9XCJpb24tcGFkZGluZ1wiPlxuICAgICAgPHNlY3Rpb24gY2xhc3M9XCJtb2RhbC1mb3JtLXNlY3Rpb25cIj5cbiAgICAgICAgQGlmIChwcm9wcy5sb2dvKSB7XG4gICAgICAgICAgPGRpdiBjbGFzcz1cImxvZ28tY29udGFpbmVyXCI+XG4gICAgICAgICAgICA8dmFsLWltYWdlIFtwcm9wc109XCJwcm9wcy5sb2dvXCIgLz5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuICAgICAgICA8dmFsLWZvcm0gW3Byb3BzXT1cInJlZ2lzdGVyRm9ybVByb3BzXCIgKG9uU3VibWl0KT1cInJlZ2lzdGVySGFuZGxlcigkZXZlbnQpXCIgLz5cbiAgICAgICAgPGRpdiBjbGFzcz1cImF1dGgtbGlua1wiPlxuICAgICAgICAgIDxpb24tdGV4dCBjb2xvcj1cImRhcmtcIj5cbiAgICAgICAgICAgIMK/WWEgdGllbmVzIGN1ZW50YT9cbiAgICAgICAgICAgIDxhIChjbGljayk9XCJjbG9zZVJlZ2lzdGVyTW9kYWwoKVwiPkluaWNpYXIgc2VzacOzbjwvYT5cbiAgICAgICAgICA8L2lvbi10ZXh0PlxuICAgICAgICA8L2Rpdj5cbiAgICAgIDwvc2VjdGlvbj5cbiAgICA8L2lvbi1jb250ZW50PlxuICA8L25nLXRlbXBsYXRlPlxuPC9pb24tbW9kYWw+XG5cbjwhLS0gVmVyaWZ5IEVtYWlsIE1vZGFsIC0tPlxuPGlvbi1tb2RhbCBbaXNPcGVuXT1cImlzVmVyaWZ5TW9kYWxPcGVuXCIgW2JhY2tkcm9wRGlzbWlzc109XCJmYWxzZVwiPlxuICA8bmctdGVtcGxhdGU+XG4gICAgPGlvbi1oZWFkZXI+XG4gICAgICA8aW9uLXRvb2xiYXI+XG4gICAgICAgIDxpb24tYnV0dG9ucyBzbG90PVwiZW5kXCI+XG4gICAgICAgICAgPGlvbi1idXR0b24gZmlsbD1cImNsZWFyXCIgKGNsaWNrKT1cImNsb3NlVmVyaWZ5TW9kYWwoKVwiPlxuICAgICAgICAgICAgPGlvbi1pY29uIG5hbWU9XCJjbG9zZS1vdXRsaW5lXCI+PC9pb24taWNvbj5cbiAgICAgICAgICA8L2lvbi1idXR0b24+XG4gICAgICAgIDwvaW9uLWJ1dHRvbnM+XG4gICAgICA8L2lvbi10b29sYmFyPlxuICAgIDwvaW9uLWhlYWRlcj5cbiAgICA8aW9uLWNvbnRlbnQgY2xhc3M9XCJpb24tcGFkZGluZ1wiPlxuICAgICAgPHNlY3Rpb24gY2xhc3M9XCJtb2RhbC1mb3JtLXNlY3Rpb25cIj5cbiAgICAgICAgQGlmIChwcm9wcy5sb2dvKSB7XG4gICAgICAgICAgPGRpdiBjbGFzcz1cImxvZ28tY29udGFpbmVyXCI+XG4gICAgICAgICAgICA8dmFsLWltYWdlIFtwcm9wc109XCJwcm9wcy5sb2dvXCIgLz5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuICAgICAgICA8dmFsLWZvcm0gW3Byb3BzXT1cInZlcmlmeUZvcm1Qcm9wc1wiIChvblN1Ym1pdCk9XCJ2ZXJpZnlIYW5kbGVyKCRldmVudClcIiAvPlxuICAgICAgICA8ZGl2IGNsYXNzPVwicmVzZW5kLWxpbmtcIj5cbiAgICAgICAgICA8aW9uLXRleHQgY29sb3I9XCJkYXJrXCI+XG4gICAgICAgICAgICDCv05vIGhhcyByZWNpYmlkbyB0dSBjw7NkaWdvP1xuICAgICAgICAgICAgQGlmIChyZXNlbmRDb29sZG93biA+IDApIHtcbiAgICAgICAgICAgICAgPHNwYW4gY2xhc3M9XCJjb29sZG93blwiPlJlZW52aWFyIGVuIHt7IHJlc2VuZENvb2xkb3duIH19czwvc3Bhbj5cbiAgICAgICAgICAgIH0gQGVsc2Uge1xuICAgICAgICAgICAgICA8YSAoY2xpY2spPVwicmVzZW5kQ29kZSgpXCI+UmVlbnZpYXI8L2E+XG4gICAgICAgICAgICB9XG4gICAgICAgICAgPC9pb24tdGV4dD5cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L3NlY3Rpb24+XG4gICAgPC9pb24tY29udGVudD5cbiAgPC9uZy10ZW1wbGF0ZT5cbjwvaW9uLW1vZGFsPlxuXG48IS0tIEZvcmdvdCBQYXNzd29yZCBNb2RhbCAtLT5cbjxpb24tbW9kYWwgW2lzT3Blbl09XCJpc0ZvcmdvdFBhc3N3b3JkTW9kYWxPcGVuXCIgKGRpZERpc21pc3MpPVwiY2xvc2VGb3Jnb3RQYXNzd29yZE1vZGFsKClcIj5cbiAgPG5nLXRlbXBsYXRlPlxuICAgIDxpb24taGVhZGVyPlxuICAgICAgPGlvbi10b29sYmFyPlxuICAgICAgICA8aW9uLWJ1dHRvbnMgc2xvdD1cImVuZFwiPlxuICAgICAgICAgIDxpb24tYnV0dG9uIGZpbGw9XCJjbGVhclwiIChjbGljayk9XCJjbG9zZUZvcmdvdFBhc3N3b3JkTW9kYWwoKVwiPlxuICAgICAgICAgICAgPGlvbi1pY29uIG5hbWU9XCJjbG9zZS1vdXRsaW5lXCI+PC9pb24taWNvbj5cbiAgICAgICAgICA8L2lvbi1idXR0b24+XG4gICAgICAgIDwvaW9uLWJ1dHRvbnM+XG4gICAgICA8L2lvbi10b29sYmFyPlxuICAgIDwvaW9uLWhlYWRlcj5cbiAgICA8aW9uLWNvbnRlbnQgY2xhc3M9XCJpb24tcGFkZGluZ1wiPlxuICAgICAgPHNlY3Rpb24gY2xhc3M9XCJtb2RhbC1mb3JtLXNlY3Rpb25cIj5cbiAgICAgICAgQGlmIChwcm9wcy5sb2dvKSB7XG4gICAgICAgICAgPGRpdiBjbGFzcz1cImxvZ28tY29udGFpbmVyXCI+XG4gICAgICAgICAgICA8dmFsLWltYWdlIFtwcm9wc109XCJwcm9wcy5sb2dvXCIgLz5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuICAgICAgICA8dmFsLWZvcm0gW3Byb3BzXT1cImZvcmdvdFBhc3N3b3JkRm9ybVByb3BzXCIgKG9uU3VibWl0KT1cImZvcmdvdFBhc3N3b3JkSGFuZGxlcigkZXZlbnQpXCIgLz5cbiAgICAgIDwvc2VjdGlvbj5cbiAgICA8L2lvbi1jb250ZW50PlxuICA8L25nLXRlbXBsYXRlPlxuPC9pb24tbW9kYWw+XG5cbjwhLS0gUmVzZXQgUGFzc3dvcmQgTW9kYWwgLS0+XG48aW9uLW1vZGFsIFtpc09wZW5dPVwiaXNSZXNldFBhc3N3b3JkTW9kYWxPcGVuXCIgW2JhY2tkcm9wRGlzbWlzc109XCJmYWxzZVwiPlxuICA8bmctdGVtcGxhdGU+XG4gICAgPGlvbi1oZWFkZXI+XG4gICAgICA8aW9uLXRvb2xiYXI+XG4gICAgICAgIDxpb24tYnV0dG9ucyBzbG90PVwiZW5kXCI+XG4gICAgICAgICAgPGlvbi1idXR0b24gZmlsbD1cImNsZWFyXCIgKGNsaWNrKT1cImNsb3NlUmVzZXRQYXNzd29yZE1vZGFsKClcIj5cbiAgICAgICAgICAgIDxpb24taWNvbiBuYW1lPVwiY2xvc2Utb3V0bGluZVwiPjwvaW9uLWljb24+XG4gICAgICAgICAgPC9pb24tYnV0dG9uPlxuICAgICAgICA8L2lvbi1idXR0b25zPlxuICAgICAgPC9pb24tdG9vbGJhcj5cbiAgICA8L2lvbi1oZWFkZXI+XG4gICAgPGlvbi1jb250ZW50IGNsYXNzPVwiaW9uLXBhZGRpbmdcIj5cbiAgICAgIDxzZWN0aW9uIGNsYXNzPVwibW9kYWwtZm9ybS1zZWN0aW9uXCI+XG4gICAgICAgIEBpZiAocHJvcHMubG9nbykge1xuICAgICAgICAgIDxkaXYgY2xhc3M9XCJsb2dvLWNvbnRhaW5lclwiPlxuICAgICAgICAgICAgPHZhbC1pbWFnZSBbcHJvcHNdPVwicHJvcHMubG9nb1wiIC8+XG4gICAgICAgICAgPC9kaXY+XG4gICAgICAgIH1cbiAgICAgICAgPHZhbC1mb3JtIFtwcm9wc109XCJyZXNldFBhc3N3b3JkRm9ybVByb3BzXCIgKG9uU3VibWl0KT1cInJlc2V0UGFzc3dvcmRIYW5kbGVyKCRldmVudClcIiAvPlxuICAgICAgICA8ZGl2IGNsYXNzPVwicmVzZW5kLWxpbmtcIj5cbiAgICAgICAgICA8aW9uLXRleHQgY29sb3I9XCJkYXJrXCI+XG4gICAgICAgICAgICDCv05vIGhhcyByZWNpYmlkbyB0dSBjw7NkaWdvP1xuICAgICAgICAgICAgQGlmIChyZXNldFJlc2VuZENvb2xkb3duID4gMCkge1xuICAgICAgICAgICAgICA8c3BhbiBjbGFzcz1cImNvb2xkb3duXCI+UmVlbnZpYXIgZW4ge3sgcmVzZXRSZXNlbmRDb29sZG93biB9fXM8L3NwYW4+XG4gICAgICAgICAgICB9IEBlbHNlIHtcbiAgICAgICAgICAgICAgPGEgKGNsaWNrKT1cInJlc2VuZFJlc2V0Q29kZSgpXCI+UmVlbnZpYXI8L2E+XG4gICAgICAgICAgICB9XG4gICAgICAgICAgPC9pb24tdGV4dD5cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L3NlY3Rpb24+XG4gICAgPC9pb24tY29udGVudD5cbiAgPC9uZy10ZW1wbGF0ZT5cbjwvaW9uLW1vZGFsPlxuXG48IS0tIE1GQSBWZXJpZnkgTW9kYWwgLS0+XG48aW9uLW1vZGFsIFtpc09wZW5dPVwiaXNNRkFWZXJpZnlNb2RhbE9wZW5cIiBbYmFja2Ryb3BEaXNtaXNzXT1cImZhbHNlXCI+XG4gIDxuZy10ZW1wbGF0ZT5cbiAgICA8aW9uLWhlYWRlcj5cbiAgICAgIDxpb24tdG9vbGJhcj5cbiAgICAgICAgPGlvbi1idXR0b25zIHNsb3Q9XCJlbmRcIj5cbiAgICAgICAgICA8aW9uLWJ1dHRvbiBmaWxsPVwiY2xlYXJcIiAoY2xpY2spPVwiY2xvc2VNRkFWZXJpZnlNb2RhbCgpXCI+XG4gICAgICAgICAgICA8aW9uLWljb24gbmFtZT1cImNsb3NlLW91dGxpbmVcIj48L2lvbi1pY29uPlxuICAgICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgICAgPC9pb24tYnV0dG9ucz5cbiAgICAgIDwvaW9uLXRvb2xiYXI+XG4gICAgPC9pb24taGVhZGVyPlxuICAgIDxpb24tY29udGVudCBjbGFzcz1cImlvbi1wYWRkaW5nXCI+XG4gICAgICA8c2VjdGlvbiBjbGFzcz1cIm1vZGFsLWZvcm0tc2VjdGlvblwiPlxuICAgICAgICBAaWYgKHByb3BzLmxvZ28pIHtcbiAgICAgICAgICA8ZGl2IGNsYXNzPVwibG9nby1jb250YWluZXJcIj5cbiAgICAgICAgICAgIDx2YWwtaW1hZ2UgW3Byb3BzXT1cInByb3BzLmxvZ29cIiAvPlxuICAgICAgICAgIDwvZGl2PlxuICAgICAgICB9XG4gICAgICAgIDx2YWwtZm9ybSBbcHJvcHNdPVwibWZhVmVyaWZ5Rm9ybVByb3BzXCIgKG9uU3VibWl0KT1cInZlcmlmeU1GQUhhbmRsZXIoJGV2ZW50KVwiIC8+XG4gICAgICA8L3NlY3Rpb24+XG4gICAgPC9pb24tY29udGVudD5cbiAgPC9uZy10ZW1wbGF0ZT5cbjwvaW9uLW1vZGFsPlxuIl19
|