valtech-components 2.0.428 → 2.0.430
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/data-table/data-table.component.mjs +17 -3
- package/esm2022/lib/components/organisms/data-table/types.mjs +1 -1
- package/esm2022/lib/services/auth/auth-state.service.mjs +173 -0
- package/esm2022/lib/services/auth/auth.service.mjs +432 -0
- package/esm2022/lib/services/auth/config.mjs +76 -0
- package/esm2022/lib/services/auth/guards.mjs +194 -0
- package/esm2022/lib/services/auth/index.mjs +70 -0
- package/esm2022/lib/services/auth/interceptor.mjs +98 -0
- package/esm2022/lib/services/auth/storage.service.mjs +138 -0
- package/esm2022/lib/services/auth/sync.service.mjs +146 -0
- package/esm2022/lib/services/auth/token.service.mjs +113 -0
- package/esm2022/lib/services/auth/types.mjs +29 -0
- package/esm2022/public-api.mjs +4 -1
- package/fesm2022/valtech-components.mjs +1465 -8
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/organisms/data-table/types.d.ts +8 -0
- package/lib/services/auth/auth-state.service.d.ts +85 -0
- package/lib/services/auth/auth.service.d.ts +123 -0
- package/lib/services/auth/config.d.ts +38 -0
- package/lib/services/auth/guards.d.ts +123 -0
- package/lib/services/auth/index.d.ts +63 -0
- package/lib/services/auth/interceptor.d.ts +22 -0
- package/lib/services/auth/storage.service.d.ts +48 -0
- package/lib/services/auth/sync.service.d.ts +49 -0
- package/lib/services/auth/token.service.d.ts +51 -0
- package/lib/services/auth/types.d.ts +264 -0
- package/package.json +1 -9
- package/public-api.d.ts +1 -0
|
@@ -0,0 +1,432 @@
|
|
|
1
|
+
import { Injectable, inject } from '@angular/core';
|
|
2
|
+
import { HttpClient } from '@angular/common/http';
|
|
3
|
+
import { Router } from '@angular/router';
|
|
4
|
+
import { throwError, of, firstValueFrom } from 'rxjs';
|
|
5
|
+
import { tap, catchError } from 'rxjs/operators';
|
|
6
|
+
import { VALTECH_AUTH_CONFIG } from './config';
|
|
7
|
+
import { AuthStateService } from './auth-state.service';
|
|
8
|
+
import { TokenService } from './token.service';
|
|
9
|
+
import { AuthStorageService } from './storage.service';
|
|
10
|
+
import { AuthSyncService } from './sync.service';
|
|
11
|
+
import * as i0 from "@angular/core";
|
|
12
|
+
// Importación opcional de FirebaseService
|
|
13
|
+
let FirebaseService = null;
|
|
14
|
+
try {
|
|
15
|
+
// Intenta importar FirebaseService si está disponible
|
|
16
|
+
import('../firebase').then((m) => {
|
|
17
|
+
FirebaseService = m.FirebaseService;
|
|
18
|
+
});
|
|
19
|
+
}
|
|
20
|
+
catch {
|
|
21
|
+
// FirebaseService no disponible
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Servicio principal de autenticación.
|
|
25
|
+
*
|
|
26
|
+
* @example
|
|
27
|
+
* ```typescript
|
|
28
|
+
* import { AuthService } from 'valtech-components';
|
|
29
|
+
*
|
|
30
|
+
* @Component({...})
|
|
31
|
+
* export class LoginComponent {
|
|
32
|
+
* private auth = inject(AuthService);
|
|
33
|
+
*
|
|
34
|
+
* async login() {
|
|
35
|
+
* await firstValueFrom(this.auth.signin({ email, password }));
|
|
36
|
+
* if (this.auth.mfaPending().required) {
|
|
37
|
+
* // Mostrar UI de MFA
|
|
38
|
+
* } else {
|
|
39
|
+
* this.router.navigate(['/']);
|
|
40
|
+
* }
|
|
41
|
+
* }
|
|
42
|
+
* }
|
|
43
|
+
* ```
|
|
44
|
+
*/
|
|
45
|
+
export class AuthService {
|
|
46
|
+
constructor() {
|
|
47
|
+
this.config = inject(VALTECH_AUTH_CONFIG);
|
|
48
|
+
this.http = inject(HttpClient);
|
|
49
|
+
this.router = inject(Router);
|
|
50
|
+
this.stateService = inject(AuthStateService);
|
|
51
|
+
this.tokenService = inject(TokenService);
|
|
52
|
+
this.storageService = inject(AuthStorageService);
|
|
53
|
+
this.syncService = inject(AuthSyncService);
|
|
54
|
+
// Timer para refresh proactivo
|
|
55
|
+
this.refreshTimerId = null;
|
|
56
|
+
this.syncSubscription = null;
|
|
57
|
+
// =============================================
|
|
58
|
+
// ESTADO PÚBLICO (Signals readonly)
|
|
59
|
+
// =============================================
|
|
60
|
+
/** Estado completo de autenticación */
|
|
61
|
+
this.state = this.stateService.state;
|
|
62
|
+
/** Usuario está autenticado */
|
|
63
|
+
this.isAuthenticated = this.stateService.isAuthenticated;
|
|
64
|
+
/** Estado de carga */
|
|
65
|
+
this.isLoading = this.stateService.isLoading;
|
|
66
|
+
/** Información del usuario */
|
|
67
|
+
this.user = this.stateService.user;
|
|
68
|
+
/** Token de acceso */
|
|
69
|
+
this.accessToken = this.stateService.accessToken;
|
|
70
|
+
/** Roles del usuario */
|
|
71
|
+
this.roles = this.stateService.roles;
|
|
72
|
+
/** Permisos del usuario */
|
|
73
|
+
this.permissions = this.stateService.permissions;
|
|
74
|
+
/** Usuario es super admin */
|
|
75
|
+
this.isSuperAdmin = this.stateService.isSuperAdmin;
|
|
76
|
+
/** Estado de MFA pendiente */
|
|
77
|
+
this.mfaPending = this.stateService.mfaPending;
|
|
78
|
+
/** Error actual */
|
|
79
|
+
this.error = this.stateService.error;
|
|
80
|
+
}
|
|
81
|
+
// =============================================
|
|
82
|
+
// INICIALIZACIÓN
|
|
83
|
+
// =============================================
|
|
84
|
+
/**
|
|
85
|
+
* Inicializa el servicio de autenticación.
|
|
86
|
+
* Llamado automáticamente por provideValtechAuth.
|
|
87
|
+
*/
|
|
88
|
+
async initialize() {
|
|
89
|
+
// 1. Cargar estado desde storage
|
|
90
|
+
const storedState = this.storageService.loadState();
|
|
91
|
+
if (storedState.accessToken) {
|
|
92
|
+
// 2. Verificar si token es válido
|
|
93
|
+
if (this.tokenService.isTokenValid(storedState.accessToken)) {
|
|
94
|
+
this.stateService.restoreFromStorage(storedState);
|
|
95
|
+
// Extraer info del token
|
|
96
|
+
const claims = this.tokenService.parseToken(storedState.accessToken);
|
|
97
|
+
if (claims) {
|
|
98
|
+
this.stateService.updateUserInfo(claims.uid, claims.email);
|
|
99
|
+
}
|
|
100
|
+
// 3. Iniciar timer de refresco proactivo
|
|
101
|
+
this.startRefreshTimer();
|
|
102
|
+
}
|
|
103
|
+
else if (storedState.refreshToken) {
|
|
104
|
+
// 4. Token expirado pero hay refresh token - intentar refrescar
|
|
105
|
+
try {
|
|
106
|
+
await firstValueFrom(this.refreshAccessToken());
|
|
107
|
+
}
|
|
108
|
+
catch {
|
|
109
|
+
this.clearState();
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
else {
|
|
113
|
+
this.clearState();
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
// 5. Iniciar sincronización entre pestañas
|
|
117
|
+
if (this.config.enableTabSync) {
|
|
118
|
+
this.syncService.start();
|
|
119
|
+
this.syncSubscription = this.syncService.onEvent$.subscribe((event) => this.handleSyncEvent(event));
|
|
120
|
+
}
|
|
121
|
+
this.stateService.setLoading(false);
|
|
122
|
+
}
|
|
123
|
+
ngOnDestroy() {
|
|
124
|
+
this.stopRefreshTimer();
|
|
125
|
+
this.syncSubscription?.unsubscribe();
|
|
126
|
+
}
|
|
127
|
+
// =============================================
|
|
128
|
+
// AUTENTICACIÓN
|
|
129
|
+
// =============================================
|
|
130
|
+
/**
|
|
131
|
+
* Inicia sesión con email y contraseña.
|
|
132
|
+
*/
|
|
133
|
+
signin(request) {
|
|
134
|
+
this.stateService.clearError();
|
|
135
|
+
return this.http
|
|
136
|
+
.post(`${this.baseUrl}/signin`, request)
|
|
137
|
+
.pipe(tap((response) => {
|
|
138
|
+
if (response.mfaRequired) {
|
|
139
|
+
// MFA requerido - guardar estado temporal
|
|
140
|
+
this.stateService.setMFAPending({
|
|
141
|
+
required: true,
|
|
142
|
+
mfaToken: response.mfaToken,
|
|
143
|
+
method: response.mfaMethod,
|
|
144
|
+
});
|
|
145
|
+
}
|
|
146
|
+
else if (response.accessToken) {
|
|
147
|
+
// Login exitoso sin MFA
|
|
148
|
+
this.handleSuccessfulAuth(response);
|
|
149
|
+
}
|
|
150
|
+
}), catchError((error) => this.handleAuthError(error)));
|
|
151
|
+
}
|
|
152
|
+
/**
|
|
153
|
+
* Verifica código MFA.
|
|
154
|
+
*/
|
|
155
|
+
verifyMFA(code) {
|
|
156
|
+
const mfaState = this.mfaPending();
|
|
157
|
+
if (!mfaState.mfaToken) {
|
|
158
|
+
return throwError(() => ({
|
|
159
|
+
code: 'MFA_NOT_PENDING',
|
|
160
|
+
message: 'No hay verificación MFA pendiente',
|
|
161
|
+
}));
|
|
162
|
+
}
|
|
163
|
+
return this.http
|
|
164
|
+
.post(`${this.baseUrl}/mfa/verify`, {
|
|
165
|
+
mfaToken: mfaState.mfaToken,
|
|
166
|
+
code,
|
|
167
|
+
})
|
|
168
|
+
.pipe(tap((response) => {
|
|
169
|
+
this.stateService.clearMFAPending();
|
|
170
|
+
this.handleSuccessfulAuth(response);
|
|
171
|
+
}), catchError((error) => this.handleAuthError(error)));
|
|
172
|
+
}
|
|
173
|
+
/**
|
|
174
|
+
* Refresca el token de acceso.
|
|
175
|
+
*/
|
|
176
|
+
refreshAccessToken() {
|
|
177
|
+
const refreshToken = this.state().refreshToken;
|
|
178
|
+
if (!refreshToken) {
|
|
179
|
+
return throwError(() => ({
|
|
180
|
+
code: 'NO_REFRESH_TOKEN',
|
|
181
|
+
message: 'No hay token de refresco',
|
|
182
|
+
}));
|
|
183
|
+
}
|
|
184
|
+
return this.http
|
|
185
|
+
.post(`${this.baseUrl}/refresh`, { refreshToken })
|
|
186
|
+
.pipe(tap((response) => {
|
|
187
|
+
const expiresAt = Date.now() + response.expiresIn * 1000;
|
|
188
|
+
this.stateService.updateAccessToken(response.accessToken, response.expiresIn);
|
|
189
|
+
this.storageService.saveAccessToken(response.accessToken, expiresAt);
|
|
190
|
+
this.startRefreshTimer();
|
|
191
|
+
this.syncService.broadcast({
|
|
192
|
+
type: 'TOKEN_REFRESH',
|
|
193
|
+
payload: { accessToken: response.accessToken, expiresAt },
|
|
194
|
+
});
|
|
195
|
+
}), catchError((error) => {
|
|
196
|
+
this.logout();
|
|
197
|
+
return throwError(() => error);
|
|
198
|
+
}));
|
|
199
|
+
}
|
|
200
|
+
/**
|
|
201
|
+
* Cierra sesión.
|
|
202
|
+
*/
|
|
203
|
+
logout() {
|
|
204
|
+
const refreshToken = this.state().refreshToken;
|
|
205
|
+
// Notificar al backend (fire and forget)
|
|
206
|
+
if (refreshToken) {
|
|
207
|
+
this.http
|
|
208
|
+
.post(`${this.baseUrl}/logout`, { refreshToken })
|
|
209
|
+
.pipe(catchError(() => of(null)))
|
|
210
|
+
.subscribe();
|
|
211
|
+
}
|
|
212
|
+
// Cerrar sesión de Firebase si está integrado
|
|
213
|
+
this.signOutFirebase();
|
|
214
|
+
this.clearState();
|
|
215
|
+
this.syncService.broadcast({ type: 'LOGOUT' });
|
|
216
|
+
this.router.navigate([this.config.loginRoute]);
|
|
217
|
+
}
|
|
218
|
+
// =============================================
|
|
219
|
+
// MFA SETUP (usuario autenticado)
|
|
220
|
+
// =============================================
|
|
221
|
+
/**
|
|
222
|
+
* Configura MFA para el usuario.
|
|
223
|
+
*/
|
|
224
|
+
setupMFA(method, phone) {
|
|
225
|
+
return this.http
|
|
226
|
+
.post(`${this.baseUrl}/mfa/setup`, { method, phone })
|
|
227
|
+
.pipe(catchError((error) => this.handleAuthError(error)));
|
|
228
|
+
}
|
|
229
|
+
/**
|
|
230
|
+
* Confirma la configuración de MFA.
|
|
231
|
+
*/
|
|
232
|
+
confirmMFA(code) {
|
|
233
|
+
return this.http
|
|
234
|
+
.post(`${this.baseUrl}/mfa/confirm`, { code })
|
|
235
|
+
.pipe(catchError((error) => this.handleAuthError(error)));
|
|
236
|
+
}
|
|
237
|
+
/**
|
|
238
|
+
* Deshabilita MFA.
|
|
239
|
+
*/
|
|
240
|
+
disableMFA(password) {
|
|
241
|
+
return this.http
|
|
242
|
+
.post(`${this.baseUrl}/mfa/disable`, { password })
|
|
243
|
+
.pipe(catchError((error) => this.handleAuthError(error)));
|
|
244
|
+
}
|
|
245
|
+
// =============================================
|
|
246
|
+
// PERMISOS
|
|
247
|
+
// =============================================
|
|
248
|
+
/**
|
|
249
|
+
* Obtiene los permisos actualizados del backend.
|
|
250
|
+
*/
|
|
251
|
+
fetchPermissions() {
|
|
252
|
+
return this.http
|
|
253
|
+
.get(`${this.baseUrl}/permissions`)
|
|
254
|
+
.pipe(tap((response) => {
|
|
255
|
+
this.stateService.updatePermissions(response.roles, response.permissions, response.isSuperAdmin);
|
|
256
|
+
this.storageService.savePermissions(response);
|
|
257
|
+
this.syncService.broadcast({ type: 'PERMISSIONS_UPDATE' });
|
|
258
|
+
}), catchError((error) => this.handleAuthError(error)));
|
|
259
|
+
}
|
|
260
|
+
/**
|
|
261
|
+
* Verifica si el usuario tiene un permiso específico.
|
|
262
|
+
* Formato: "resource:action" (ej: "templates:edit")
|
|
263
|
+
*/
|
|
264
|
+
hasPermission(permission) {
|
|
265
|
+
if (this.isSuperAdmin())
|
|
266
|
+
return true;
|
|
267
|
+
const [resource, action] = permission.split(':');
|
|
268
|
+
return this.permissions().some((p) => {
|
|
269
|
+
const [pResource, pAction] = p.split(':');
|
|
270
|
+
return ((pResource === '*' || pResource === resource) &&
|
|
271
|
+
(pAction === '*' || pAction === action));
|
|
272
|
+
});
|
|
273
|
+
}
|
|
274
|
+
/**
|
|
275
|
+
* Verifica si el usuario tiene alguno de los permisos dados.
|
|
276
|
+
*/
|
|
277
|
+
hasAnyPermission(permissions) {
|
|
278
|
+
return permissions.some((p) => this.hasPermission(p));
|
|
279
|
+
}
|
|
280
|
+
/**
|
|
281
|
+
* Verifica si el usuario tiene todos los permisos dados.
|
|
282
|
+
*/
|
|
283
|
+
hasAllPermissions(permissions) {
|
|
284
|
+
return permissions.every((p) => this.hasPermission(p));
|
|
285
|
+
}
|
|
286
|
+
/**
|
|
287
|
+
* Verifica si el usuario tiene un rol específico.
|
|
288
|
+
*/
|
|
289
|
+
hasRole(role) {
|
|
290
|
+
return this.roles().some((r) => r.toLowerCase() === role.toLowerCase());
|
|
291
|
+
}
|
|
292
|
+
// =============================================
|
|
293
|
+
// PRIVATE METHODS
|
|
294
|
+
// =============================================
|
|
295
|
+
get baseUrl() {
|
|
296
|
+
return `${this.config.apiUrl}${this.config.authPrefix}`;
|
|
297
|
+
}
|
|
298
|
+
handleSuccessfulAuth(response) {
|
|
299
|
+
const expiresAt = Date.now() + (response.expiresIn * 1000);
|
|
300
|
+
const tokenData = this.tokenService.parseToken(response.accessToken);
|
|
301
|
+
this.stateService.setAuthenticated({
|
|
302
|
+
accessToken: response.accessToken,
|
|
303
|
+
refreshToken: response.refreshToken,
|
|
304
|
+
userId: tokenData?.uid,
|
|
305
|
+
email: tokenData?.email,
|
|
306
|
+
roles: response.roles || [],
|
|
307
|
+
permissions: response.permissions || [],
|
|
308
|
+
isSuperAdmin: response.permissions?.includes('*:*') || false,
|
|
309
|
+
expiresAt,
|
|
310
|
+
});
|
|
311
|
+
this.storageService.saveState({
|
|
312
|
+
accessToken: response.accessToken,
|
|
313
|
+
refreshToken: response.refreshToken,
|
|
314
|
+
roles: response.roles || [],
|
|
315
|
+
permissions: response.permissions || [],
|
|
316
|
+
isSuperAdmin: response.permissions?.includes('*:*') || false,
|
|
317
|
+
expiresAt,
|
|
318
|
+
});
|
|
319
|
+
this.startRefreshTimer();
|
|
320
|
+
this.syncService.broadcast({ type: 'LOGIN' });
|
|
321
|
+
// Integración con Firebase
|
|
322
|
+
if (this.config.enableFirebaseIntegration &&
|
|
323
|
+
'firebaseToken' in response &&
|
|
324
|
+
response.firebaseToken) {
|
|
325
|
+
this.signInWithFirebase(response.firebaseToken);
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
clearState() {
|
|
329
|
+
this.stopRefreshTimer();
|
|
330
|
+
this.stateService.reset();
|
|
331
|
+
this.storageService.clear();
|
|
332
|
+
}
|
|
333
|
+
startRefreshTimer() {
|
|
334
|
+
this.stopRefreshTimer();
|
|
335
|
+
const state = this.stateService.state();
|
|
336
|
+
if (!state.expiresAt)
|
|
337
|
+
return;
|
|
338
|
+
const refreshBeforeMs = (this.config.refreshBeforeExpiry || 60) * 1000;
|
|
339
|
+
const refreshAt = state.expiresAt - refreshBeforeMs;
|
|
340
|
+
const delay = refreshAt - Date.now();
|
|
341
|
+
if (delay > 0) {
|
|
342
|
+
this.refreshTimerId = setTimeout(() => {
|
|
343
|
+
this.refreshAccessToken().subscribe({
|
|
344
|
+
error: () => this.logout(),
|
|
345
|
+
});
|
|
346
|
+
}, delay);
|
|
347
|
+
}
|
|
348
|
+
else if (state.refreshToken) {
|
|
349
|
+
// Token ya debería refrescarse, intentar ahora
|
|
350
|
+
this.refreshAccessToken().subscribe({
|
|
351
|
+
error: () => this.logout(),
|
|
352
|
+
});
|
|
353
|
+
}
|
|
354
|
+
}
|
|
355
|
+
stopRefreshTimer() {
|
|
356
|
+
if (this.refreshTimerId) {
|
|
357
|
+
clearTimeout(this.refreshTimerId);
|
|
358
|
+
this.refreshTimerId = null;
|
|
359
|
+
}
|
|
360
|
+
}
|
|
361
|
+
handleSyncEvent(event) {
|
|
362
|
+
switch (event.type) {
|
|
363
|
+
case 'LOGIN':
|
|
364
|
+
case 'TOKEN_REFRESH': {
|
|
365
|
+
// Recargar estado desde storage
|
|
366
|
+
const state = this.storageService.loadState();
|
|
367
|
+
if (state.accessToken) {
|
|
368
|
+
this.stateService.restoreFromStorage(state);
|
|
369
|
+
const claims = this.tokenService.parseToken(state.accessToken);
|
|
370
|
+
if (claims) {
|
|
371
|
+
this.stateService.updateUserInfo(claims.uid, claims.email);
|
|
372
|
+
}
|
|
373
|
+
this.startRefreshTimer();
|
|
374
|
+
}
|
|
375
|
+
break;
|
|
376
|
+
}
|
|
377
|
+
case 'LOGOUT':
|
|
378
|
+
this.stateService.reset();
|
|
379
|
+
this.stopRefreshTimer();
|
|
380
|
+
this.router.navigate([this.config.loginRoute]);
|
|
381
|
+
break;
|
|
382
|
+
case 'PERMISSIONS_UPDATE': {
|
|
383
|
+
const perms = this.storageService.loadPermissions();
|
|
384
|
+
this.stateService.updatePermissions(perms.roles, perms.permissions, perms.isSuperAdmin);
|
|
385
|
+
break;
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
handleAuthError(error) {
|
|
390
|
+
const authError = {
|
|
391
|
+
code: error.error?.code || 'UNKNOWN_ERROR',
|
|
392
|
+
message: error.error?.message || 'Error de autenticación desconocido',
|
|
393
|
+
};
|
|
394
|
+
this.stateService.setError(authError);
|
|
395
|
+
return throwError(() => authError);
|
|
396
|
+
}
|
|
397
|
+
// =============================================
|
|
398
|
+
// FIREBASE INTEGRATION
|
|
399
|
+
// =============================================
|
|
400
|
+
async signInWithFirebase(firebaseToken) {
|
|
401
|
+
try {
|
|
402
|
+
// Importar FirebaseService dinámicamente
|
|
403
|
+
const firebase = await import('../firebase');
|
|
404
|
+
const injector = (await import('@angular/core')).inject;
|
|
405
|
+
// Esto es un workaround - en producción se usaría un patrón más robusto
|
|
406
|
+
console.log('[ValtechAuth] Firebase integration: token received, attempting signin...');
|
|
407
|
+
// Por ahora, solo loguear que se recibió el token
|
|
408
|
+
// La integración real requiere inyectar FirebaseService
|
|
409
|
+
}
|
|
410
|
+
catch {
|
|
411
|
+
console.warn('[ValtechAuth] FirebaseService no disponible');
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
async signOutFirebase() {
|
|
415
|
+
if (!this.config.enableFirebaseIntegration)
|
|
416
|
+
return;
|
|
417
|
+
try {
|
|
418
|
+
// Similar al signin, la integración real requiere inyección
|
|
419
|
+
console.log('[ValtechAuth] Firebase signout triggered');
|
|
420
|
+
}
|
|
421
|
+
catch {
|
|
422
|
+
// Ignorar errores de Firebase
|
|
423
|
+
}
|
|
424
|
+
}
|
|
425
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
426
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthService, providedIn: 'root' }); }
|
|
427
|
+
}
|
|
428
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthService, decorators: [{
|
|
429
|
+
type: Injectable,
|
|
430
|
+
args: [{ providedIn: 'root' }]
|
|
431
|
+
}] });
|
|
432
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/auth.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject, OnDestroy } from '@angular/core';\nimport { HttpClient, HttpErrorResponse } from '@angular/common/http';\nimport { Router } from '@angular/router';\nimport { Observable, throwError, of, firstValueFrom, Subscription } from 'rxjs';\nimport { tap, catchError } from 'rxjs/operators';\n\nimport { VALTECH_AUTH_CONFIG } from './config';\nimport { AuthStateService } from './auth-state.service';\nimport { TokenService } from './token.service';\nimport { AuthStorageService } from './storage.service';\nimport { AuthSyncService } from './sync.service';\nimport {\n  SigninRequest,\n  SigninResponse,\n  MFAVerifyResponse,\n  RefreshResponse,\n  GetPermissionsResponse,\n  MFASetupResponse,\n  MFAConfirmResponse,\n  MFADisableResponse,\n  MFAMethod,\n  AuthError,\n  AuthSyncEvent,\n} from './types';\n\n// Importación opcional de FirebaseService\nlet FirebaseService: any = null;\ntry {\n  // Intenta importar FirebaseService si está disponible\n  import('../firebase').then((m) => {\n    FirebaseService = m.FirebaseService;\n  });\n} catch {\n  // FirebaseService no disponible\n}\n\n/**\n * Servicio principal de autenticación.\n *\n * @example\n * ```typescript\n * import { AuthService } from 'valtech-components';\n *\n * @Component({...})\n * export class LoginComponent {\n *   private auth = inject(AuthService);\n *\n *   async login() {\n *     await firstValueFrom(this.auth.signin({ email, password }));\n *     if (this.auth.mfaPending().required) {\n *       // Mostrar UI de MFA\n *     } else {\n *       this.router.navigate(['/']);\n *     }\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class AuthService implements OnDestroy {\n  private config = inject(VALTECH_AUTH_CONFIG);\n  private http = inject(HttpClient);\n  private router = inject(Router);\n  private stateService = inject(AuthStateService);\n  private tokenService = inject(TokenService);\n  private storageService = inject(AuthStorageService);\n  private syncService = inject(AuthSyncService);\n\n  // Timer para refresh proactivo\n  private refreshTimerId: ReturnType<typeof setTimeout> | null = null;\n  private syncSubscription: Subscription | null = null;\n\n  // =============================================\n  // ESTADO PÚBLICO (Signals readonly)\n  // =============================================\n\n  /** Estado completo de autenticación */\n  readonly state = this.stateService.state;\n\n  /** Usuario está autenticado */\n  readonly isAuthenticated = this.stateService.isAuthenticated;\n\n  /** Estado de carga */\n  readonly isLoading = this.stateService.isLoading;\n\n  /** Información del usuario */\n  readonly user = this.stateService.user;\n\n  /** Token de acceso */\n  readonly accessToken = this.stateService.accessToken;\n\n  /** Roles del usuario */\n  readonly roles = this.stateService.roles;\n\n  /** Permisos del usuario */\n  readonly permissions = this.stateService.permissions;\n\n  /** Usuario es super admin */\n  readonly isSuperAdmin = this.stateService.isSuperAdmin;\n\n  /** Estado de MFA pendiente */\n  readonly mfaPending = this.stateService.mfaPending;\n\n  /** Error actual */\n  readonly error = this.stateService.error;\n\n  // =============================================\n  // INICIALIZACIÓN\n  // =============================================\n\n  /**\n   * Inicializa el servicio de autenticación.\n   * Llamado automáticamente por provideValtechAuth.\n   */\n  async initialize(): Promise<void> {\n    // 1. Cargar estado desde storage\n    const storedState = this.storageService.loadState();\n\n    if (storedState.accessToken) {\n      // 2. Verificar si token es válido\n      if (this.tokenService.isTokenValid(storedState.accessToken)) {\n        this.stateService.restoreFromStorage(storedState);\n\n        // Extraer info del token\n        const claims = this.tokenService.parseToken(storedState.accessToken);\n        if (claims) {\n          this.stateService.updateUserInfo(claims.uid, claims.email);\n        }\n\n        // 3. Iniciar timer de refresco proactivo\n        this.startRefreshTimer();\n      } else if (storedState.refreshToken) {\n        // 4. Token expirado pero hay refresh token - intentar refrescar\n        try {\n          await firstValueFrom(this.refreshAccessToken());\n        } catch {\n          this.clearState();\n        }\n      } else {\n        this.clearState();\n      }\n    }\n\n    // 5. Iniciar sincronización entre pestañas\n    if (this.config.enableTabSync) {\n      this.syncService.start();\n      this.syncSubscription = this.syncService.onEvent$.subscribe((event) =>\n        this.handleSyncEvent(event)\n      );\n    }\n\n    this.stateService.setLoading(false);\n  }\n\n  ngOnDestroy(): void {\n    this.stopRefreshTimer();\n    this.syncSubscription?.unsubscribe();\n  }\n\n  // =============================================\n  // AUTENTICACIÓN\n  // =============================================\n\n  /**\n   * Inicia sesión con email y contraseña.\n   */\n  signin(request: SigninRequest): Observable<SigninResponse> {\n    this.stateService.clearError();\n\n    return this.http\n      .post<SigninResponse>(`${this.baseUrl}/signin`, request)\n      .pipe(\n        tap((response) => {\n          if (response.mfaRequired) {\n            // MFA requerido - guardar estado temporal\n            this.stateService.setMFAPending({\n              required: true,\n              mfaToken: response.mfaToken!,\n              method: response.mfaMethod!,\n            });\n          } else if (response.accessToken) {\n            // Login exitoso sin MFA\n            this.handleSuccessfulAuth(response);\n          }\n        }),\n        catchError((error) => this.handleAuthError(error))\n      );\n  }\n\n  /**\n   * Verifica código MFA.\n   */\n  verifyMFA(code: string): Observable<MFAVerifyResponse> {\n    const mfaState = this.mfaPending();\n    if (!mfaState.mfaToken) {\n      return throwError(() => ({\n        code: 'MFA_NOT_PENDING',\n        message: 'No hay verificación MFA pendiente',\n      }));\n    }\n\n    return this.http\n      .post<MFAVerifyResponse>(`${this.baseUrl}/mfa/verify`, {\n        mfaToken: mfaState.mfaToken,\n        code,\n      })\n      .pipe(\n        tap((response) => {\n          this.stateService.clearMFAPending();\n          this.handleSuccessfulAuth(response);\n        }),\n        catchError((error) => this.handleAuthError(error))\n      );\n  }\n\n  /**\n   * Refresca el token de acceso.\n   */\n  refreshAccessToken(): Observable<RefreshResponse> {\n    const refreshToken = this.state().refreshToken;\n    if (!refreshToken) {\n      return throwError(() => ({\n        code: 'NO_REFRESH_TOKEN',\n        message: 'No hay token de refresco',\n      }));\n    }\n\n    return this.http\n      .post<RefreshResponse>(`${this.baseUrl}/refresh`, { refreshToken })\n      .pipe(\n        tap((response) => {\n          const expiresAt = Date.now() + response.expiresIn * 1000;\n          this.stateService.updateAccessToken(\n            response.accessToken,\n            response.expiresIn\n          );\n          this.storageService.saveAccessToken(response.accessToken, expiresAt);\n          this.startRefreshTimer();\n          this.syncService.broadcast({\n            type: 'TOKEN_REFRESH',\n            payload: { accessToken: response.accessToken, expiresAt },\n          });\n        }),\n        catchError((error) => {\n          this.logout();\n          return throwError(() => error);\n        })\n      );\n  }\n\n  /**\n   * Cierra sesión.\n   */\n  logout(): void {\n    const refreshToken = this.state().refreshToken;\n\n    // Notificar al backend (fire and forget)\n    if (refreshToken) {\n      this.http\n        .post(`${this.baseUrl}/logout`, { refreshToken })\n        .pipe(catchError(() => of(null)))\n        .subscribe();\n    }\n\n    // Cerrar sesión de Firebase si está integrado\n    this.signOutFirebase();\n\n    this.clearState();\n    this.syncService.broadcast({ type: 'LOGOUT' });\n    this.router.navigate([this.config.loginRoute]);\n  }\n\n  // =============================================\n  // MFA SETUP (usuario autenticado)\n  // =============================================\n\n  /**\n   * Configura MFA para el usuario.\n   */\n  setupMFA(method: MFAMethod, phone?: string): Observable<MFASetupResponse> {\n    return this.http\n      .post<MFASetupResponse>(`${this.baseUrl}/mfa/setup`, { method, phone })\n      .pipe(catchError((error) => this.handleAuthError(error)));\n  }\n\n  /**\n   * Confirma la configuración de MFA.\n   */\n  confirmMFA(code: string): Observable<MFAConfirmResponse> {\n    return this.http\n      .post<MFAConfirmResponse>(`${this.baseUrl}/mfa/confirm`, { code })\n      .pipe(catchError((error) => this.handleAuthError(error)));\n  }\n\n  /**\n   * Deshabilita MFA.\n   */\n  disableMFA(password: string): Observable<MFADisableResponse> {\n    return this.http\n      .post<MFADisableResponse>(`${this.baseUrl}/mfa/disable`, { password })\n      .pipe(catchError((error) => this.handleAuthError(error)));\n  }\n\n  // =============================================\n  // PERMISOS\n  // =============================================\n\n  /**\n   * Obtiene los permisos actualizados del backend.\n   */\n  fetchPermissions(): Observable<GetPermissionsResponse> {\n    return this.http\n      .get<GetPermissionsResponse>(`${this.baseUrl}/permissions`)\n      .pipe(\n        tap((response) => {\n          this.stateService.updatePermissions(\n            response.roles,\n            response.permissions,\n            response.isSuperAdmin\n          );\n          this.storageService.savePermissions(response);\n          this.syncService.broadcast({ type: 'PERMISSIONS_UPDATE' });\n        }),\n        catchError((error) => this.handleAuthError(error))\n      );\n  }\n\n  /**\n   * Verifica si el usuario tiene un permiso específico.\n   * Formato: \"resource:action\" (ej: \"templates:edit\")\n   */\n  hasPermission(permission: string): boolean {\n    if (this.isSuperAdmin()) return true;\n\n    const [resource, action] = permission.split(':');\n    return this.permissions().some((p) => {\n      const [pResource, pAction] = p.split(':');\n      return (\n        (pResource === '*' || pResource === resource) &&\n        (pAction === '*' || pAction === action)\n      );\n    });\n  }\n\n  /**\n   * Verifica si el usuario tiene alguno de los permisos dados.\n   */\n  hasAnyPermission(permissions: string[]): boolean {\n    return permissions.some((p) => this.hasPermission(p));\n  }\n\n  /**\n   * Verifica si el usuario tiene todos los permisos dados.\n   */\n  hasAllPermissions(permissions: string[]): boolean {\n    return permissions.every((p) => this.hasPermission(p));\n  }\n\n  /**\n   * Verifica si el usuario tiene un rol específico.\n   */\n  hasRole(role: string): boolean {\n    return this.roles().some(\n      (r) => r.toLowerCase() === role.toLowerCase()\n    );\n  }\n\n  // =============================================\n  // PRIVATE METHODS\n  // =============================================\n\n  private get baseUrl(): string {\n    return `${this.config.apiUrl}${this.config.authPrefix}`;\n  }\n\n  private handleSuccessfulAuth(\n    response: SigninResponse | MFAVerifyResponse\n  ): void {\n    const expiresAt = Date.now() + (response.expiresIn! * 1000);\n    const tokenData = this.tokenService.parseToken(response.accessToken!);\n\n    this.stateService.setAuthenticated({\n      accessToken: response.accessToken!,\n      refreshToken: response.refreshToken!,\n      userId: tokenData?.uid,\n      email: tokenData?.email,\n      roles: response.roles || [],\n      permissions: response.permissions || [],\n      isSuperAdmin: response.permissions?.includes('*:*') || false,\n      expiresAt,\n    });\n\n    this.storageService.saveState({\n      accessToken: response.accessToken!,\n      refreshToken: response.refreshToken!,\n      roles: response.roles || [],\n      permissions: response.permissions || [],\n      isSuperAdmin: response.permissions?.includes('*:*') || false,\n      expiresAt,\n    });\n\n    this.startRefreshTimer();\n    this.syncService.broadcast({ type: 'LOGIN' });\n\n    // Integración con Firebase\n    if (\n      this.config.enableFirebaseIntegration &&\n      'firebaseToken' in response &&\n      response.firebaseToken\n    ) {\n      this.signInWithFirebase(response.firebaseToken);\n    }\n  }\n\n  private clearState(): void {\n    this.stopRefreshTimer();\n    this.stateService.reset();\n    this.storageService.clear();\n  }\n\n  private startRefreshTimer(): void {\n    this.stopRefreshTimer();\n\n    const state = this.stateService.state();\n    if (!state.expiresAt) return;\n\n    const refreshBeforeMs = (this.config.refreshBeforeExpiry || 60) * 1000;\n    const refreshAt = state.expiresAt - refreshBeforeMs;\n    const delay = refreshAt - Date.now();\n\n    if (delay > 0) {\n      this.refreshTimerId = setTimeout(() => {\n        this.refreshAccessToken().subscribe({\n          error: () => this.logout(),\n        });\n      }, delay);\n    } else if (state.refreshToken) {\n      // Token ya debería refrescarse, intentar ahora\n      this.refreshAccessToken().subscribe({\n        error: () => this.logout(),\n      });\n    }\n  }\n\n  private stopRefreshTimer(): void {\n    if (this.refreshTimerId) {\n      clearTimeout(this.refreshTimerId);\n      this.refreshTimerId = null;\n    }\n  }\n\n  private handleSyncEvent(event: AuthSyncEvent): void {\n    switch (event.type) {\n      case 'LOGIN':\n      case 'TOKEN_REFRESH': {\n        // Recargar estado desde storage\n        const state = this.storageService.loadState();\n        if (state.accessToken) {\n          this.stateService.restoreFromStorage(state);\n          const claims = this.tokenService.parseToken(state.accessToken);\n          if (claims) {\n            this.stateService.updateUserInfo(claims.uid, claims.email);\n          }\n          this.startRefreshTimer();\n        }\n        break;\n      }\n      case 'LOGOUT':\n        this.stateService.reset();\n        this.stopRefreshTimer();\n        this.router.navigate([this.config.loginRoute]);\n        break;\n      case 'PERMISSIONS_UPDATE': {\n        const perms = this.storageService.loadPermissions();\n        this.stateService.updatePermissions(\n          perms.roles,\n          perms.permissions,\n          perms.isSuperAdmin\n        );\n        break;\n      }\n    }\n  }\n\n  private handleAuthError(error: HttpErrorResponse): Observable<never> {\n    const authError: AuthError = {\n      code: error.error?.code || 'UNKNOWN_ERROR',\n      message: error.error?.message || 'Error de autenticación desconocido',\n    };\n    this.stateService.setError(authError);\n    return throwError(() => authError);\n  }\n\n  // =============================================\n  // FIREBASE INTEGRATION\n  // =============================================\n\n  private async signInWithFirebase(firebaseToken: string): Promise<void> {\n    try {\n      // Importar FirebaseService dinámicamente\n      const firebase = await import('../firebase');\n      const injector = (await import('@angular/core')).inject;\n\n      // Esto es un workaround - en producción se usaría un patrón más robusto\n      console.log(\n        '[ValtechAuth] Firebase integration: token received, attempting signin...'\n      );\n\n      // Por ahora, solo loguear que se recibió el token\n      // La integración real requiere inyectar FirebaseService\n    } catch {\n      console.warn('[ValtechAuth] FirebaseService no disponible');\n    }\n  }\n\n  private async signOutFirebase(): Promise<void> {\n    if (!this.config.enableFirebaseIntegration) return;\n\n    try {\n      // Similar al signin, la integración real requiere inyección\n      console.log('[ValtechAuth] Firebase signout triggered');\n    } catch {\n      // Ignorar errores de Firebase\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
import { InjectionToken, makeEnvironmentProviders, APP_INITIALIZER, } from '@angular/core';
|
|
2
|
+
import { provideHttpClient, withInterceptors } from '@angular/common/http';
|
|
3
|
+
import { authInterceptor } from './interceptor';
|
|
4
|
+
import { AuthService } from './auth.service';
|
|
5
|
+
/**
|
|
6
|
+
* Token de inyección para la configuración de Auth.
|
|
7
|
+
*/
|
|
8
|
+
export const VALTECH_AUTH_CONFIG = new InjectionToken('ValtechAuthConfig');
|
|
9
|
+
/**
|
|
10
|
+
* Configuración por defecto.
|
|
11
|
+
*/
|
|
12
|
+
export const DEFAULT_AUTH_CONFIG = {
|
|
13
|
+
authPrefix: '/v2/auth',
|
|
14
|
+
storagePrefix: 'valtech_auth_',
|
|
15
|
+
refreshBeforeExpiry: 60,
|
|
16
|
+
enableTabSync: true,
|
|
17
|
+
loginRoute: '/login',
|
|
18
|
+
homeRoute: '/',
|
|
19
|
+
unauthorizedRoute: '/unauthorized',
|
|
20
|
+
enableFirebaseIntegration: false,
|
|
21
|
+
};
|
|
22
|
+
/**
|
|
23
|
+
* Factory para inicializar el AuthService.
|
|
24
|
+
*/
|
|
25
|
+
function initializeAuth(authService) {
|
|
26
|
+
return () => authService.initialize();
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Provee el servicio de autenticación a la aplicación Angular.
|
|
30
|
+
*
|
|
31
|
+
* @param config - Configuración de autenticación
|
|
32
|
+
* @returns EnvironmentProviders para usar en bootstrapApplication
|
|
33
|
+
*
|
|
34
|
+
* @example
|
|
35
|
+
* ```typescript
|
|
36
|
+
* // main.ts
|
|
37
|
+
* import { bootstrapApplication } from '@angular/platform-browser';
|
|
38
|
+
* import { provideValtechAuth } from 'valtech-components';
|
|
39
|
+
* import { environment } from './environments/environment';
|
|
40
|
+
*
|
|
41
|
+
* bootstrapApplication(AppComponent, {
|
|
42
|
+
* providers: [
|
|
43
|
+
* provideValtechAuth({
|
|
44
|
+
* apiUrl: environment.apiUrl,
|
|
45
|
+
* enableFirebaseIntegration: true,
|
|
46
|
+
* }),
|
|
47
|
+
* ],
|
|
48
|
+
* });
|
|
49
|
+
* ```
|
|
50
|
+
*/
|
|
51
|
+
export function provideValtechAuth(config) {
|
|
52
|
+
const mergedConfig = {
|
|
53
|
+
...DEFAULT_AUTH_CONFIG,
|
|
54
|
+
...config,
|
|
55
|
+
};
|
|
56
|
+
return makeEnvironmentProviders([
|
|
57
|
+
{ provide: VALTECH_AUTH_CONFIG, useValue: mergedConfig },
|
|
58
|
+
provideHttpClient(withInterceptors([authInterceptor])),
|
|
59
|
+
// Inicializar AuthService al arrancar la app
|
|
60
|
+
{
|
|
61
|
+
provide: APP_INITIALIZER,
|
|
62
|
+
useFactory: initializeAuth,
|
|
63
|
+
deps: [AuthService],
|
|
64
|
+
multi: true,
|
|
65
|
+
},
|
|
66
|
+
]);
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Provee solo el interceptor (para apps que ya tienen AuthService configurado manualmente).
|
|
70
|
+
*/
|
|
71
|
+
export function provideValtechAuthInterceptor() {
|
|
72
|
+
return makeEnvironmentProviders([
|
|
73
|
+
provideHttpClient(withInterceptors([authInterceptor])),
|
|
74
|
+
]);
|
|
75
|
+
}
|
|
76
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9zZXJ2aWNlcy9hdXRoL2NvbmZpZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBRUwsY0FBYyxFQUNkLHdCQUF3QixFQUN4QixlQUFlLEdBRWhCLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBRTNFLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDaEQsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBRTdDOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sbUJBQW1CLEdBQUcsSUFBSSxjQUFjLENBQ25ELG1CQUFtQixDQUNwQixDQUFDO0FBRUY7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSxtQkFBbUIsR0FBK0I7SUFDN0QsVUFBVSxFQUFFLFVBQVU7SUFDdEIsYUFBYSxFQUFFLGVBQWU7SUFDOUIsbUJBQW1CLEVBQUUsRUFBRTtJQUN2QixhQUFhLEVBQUUsSUFBSTtJQUNuQixVQUFVLEVBQUUsUUFBUTtJQUNwQixTQUFTLEVBQUUsR0FBRztJQUNkLGlCQUFpQixFQUFFLGVBQWU7SUFDbEMseUJBQXlCLEVBQUUsS0FBSztDQUNqQyxDQUFDO0FBRUY7O0dBRUc7QUFDSCxTQUFTLGNBQWMsQ0FBQyxXQUF3QjtJQUM5QyxPQUFPLEdBQUcsRUFBRSxDQUFDLFdBQVcsQ0FBQyxVQUFVLEVBQUUsQ0FBQztBQUN4QyxDQUFDO0FBRUQ7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FzQkc7QUFDSCxNQUFNLFVBQVUsa0JBQWtCLENBQ2hDLE1BQXlCO0lBRXpCLE1BQU0sWUFBWSxHQUFzQjtRQUN0QyxHQUFHLG1CQUFtQjtRQUN0QixHQUFHLE1BQU07S0FDVixDQUFDO0lBRUYsT0FBTyx3QkFBd0IsQ0FBQztRQUM5QixFQUFFLE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxRQUFRLEVBQUUsWUFBWSxFQUFFO1FBQ3hELGlCQUFpQixDQUFDLGdCQUFnQixDQUFDLENBQUMsZUFBZSxDQUFDLENBQUMsQ0FBQztRQUN0RCw2Q0FBNkM7UUFDN0M7WUFDRSxPQUFPLEVBQUUsZUFBZTtZQUN4QixVQUFVLEVBQUUsY0FBYztZQUMxQixJQUFJLEVBQUUsQ0FBQyxXQUFXLENBQUM7WUFDbkIsS0FBSyxFQUFFLElBQUk7U0FDWjtLQUNGLENBQUMsQ0FBQztBQUNMLENBQUM7QUFFRDs7R0FFRztBQUNILE1BQU0sVUFBVSw2QkFBNkI7SUFDM0MsT0FBTyx3QkFBd0IsQ0FBQztRQUM5QixpQkFBaUIsQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDLGVBQWUsQ0FBQyxDQUFDLENBQUM7S0FDdkQsQ0FBQyxDQUFDO0FBQ0wsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEVudmlyb25tZW50UHJvdmlkZXJzLFxuICBJbmplY3Rpb25Ub2tlbixcbiAgbWFrZUVudmlyb25tZW50UHJvdmlkZXJzLFxuICBBUFBfSU5JVElBTElaRVIsXG4gIGluamVjdCxcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBwcm92aWRlSHR0cENsaWVudCwgd2l0aEludGVyY2VwdG9ycyB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcbmltcG9ydCB7IFZhbHRlY2hBdXRoQ29uZmlnIH0gZnJvbSAnLi90eXBlcyc7XG5pbXBvcnQgeyBhdXRoSW50ZXJjZXB0b3IgfSBmcm9tICcuL2ludGVyY2VwdG9yJztcbmltcG9ydCB7IEF1dGhTZXJ2aWNlIH0gZnJvbSAnLi9hdXRoLnNlcnZpY2UnO1xuXG4vKipcbiAqIFRva2VuIGRlIGlueWVjY2nDs24gcGFyYSBsYSBjb25maWd1cmFjacOzbiBkZSBBdXRoLlxuICovXG5leHBvcnQgY29uc3QgVkFMVEVDSF9BVVRIX0NPTkZJRyA9IG5ldyBJbmplY3Rpb25Ub2tlbjxWYWx0ZWNoQXV0aENvbmZpZz4oXG4gICdWYWx0ZWNoQXV0aENvbmZpZydcbik7XG5cbi8qKlxuICogQ29uZmlndXJhY2nDs24gcG9yIGRlZmVjdG8uXG4gKi9cbmV4cG9ydCBjb25zdCBERUZBVUxUX0FVVEhfQ09ORklHOiBQYXJ0aWFsPFZhbHRlY2hBdXRoQ29uZmlnPiA9IHtcbiAgYXV0aFByZWZpeDogJy92Mi9hdXRoJyxcbiAgc3RvcmFnZVByZWZpeDogJ3ZhbHRlY2hfYXV0aF8nLFxuICByZWZyZXNoQmVmb3JlRXhwaXJ5OiA2MCxcbiAgZW5hYmxlVGFiU3luYzogdHJ1ZSxcbiAgbG9naW5Sb3V0ZTogJy9sb2dpbicsXG4gIGhvbWVSb3V0ZTogJy8nLFxuICB1bmF1dGhvcml6ZWRSb3V0ZTogJy91bmF1dGhvcml6ZWQnLFxuICBlbmFibGVGaXJlYmFzZUludGVncmF0aW9uOiBmYWxzZSxcbn07XG5cbi8qKlxuICogRmFjdG9yeSBwYXJhIGluaWNpYWxpemFyIGVsIEF1dGhTZXJ2aWNlLlxuICovXG5mdW5jdGlvbiBpbml0aWFsaXplQXV0aChhdXRoU2VydmljZTogQXV0aFNlcnZpY2UpOiAoKSA9PiBQcm9taXNlPHZvaWQ+IHtcbiAgcmV0dXJuICgpID0+IGF1dGhTZXJ2aWNlLmluaXRpYWxpemUoKTtcbn1cblxuLyoqXG4gKiBQcm92ZWUgZWwgc2VydmljaW8gZGUgYXV0ZW50aWNhY2nDs24gYSBsYSBhcGxpY2FjacOzbiBBbmd1bGFyLlxuICpcbiAqIEBwYXJhbSBjb25maWcgLSBDb25maWd1cmFjacOzbiBkZSBhdXRlbnRpY2FjacOzblxuICogQHJldHVybnMgRW52aXJvbm1lbnRQcm92aWRlcnMgcGFyYSB1c2FyIGVuIGJvb3RzdHJhcEFwcGxpY2F0aW9uXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIC8vIG1haW4udHNcbiAqIGltcG9ydCB7IGJvb3RzdHJhcEFwcGxpY2F0aW9uIH0gZnJvbSAnQGFuZ3VsYXIvcGxhdGZvcm0tYnJvd3Nlcic7XG4gKiBpbXBvcnQgeyBwcm92aWRlVmFsdGVjaEF1dGggfSBmcm9tICd2YWx0ZWNoLWNvbXBvbmVudHMnO1xuICogaW1wb3J0IHsgZW52aXJvbm1lbnQgfSBmcm9tICcuL2Vudmlyb25tZW50cy9lbnZpcm9ubWVudCc7XG4gKlxuICogYm9vdHN0cmFwQXBwbGljYXRpb24oQXBwQ29tcG9uZW50LCB7XG4gKiAgIHByb3ZpZGVyczogW1xuICogICAgIHByb3ZpZGVWYWx0ZWNoQXV0aCh7XG4gKiAgICAgICBhcGlVcmw6IGVudmlyb25tZW50LmFwaVVybCxcbiAqICAgICAgIGVuYWJsZUZpcmViYXNlSW50ZWdyYXRpb246IHRydWUsXG4gKiAgICAgfSksXG4gKiAgIF0sXG4gKiB9KTtcbiAqIGBgYFxuICovXG5leHBvcnQgZnVuY3Rpb24gcHJvdmlkZVZhbHRlY2hBdXRoKFxuICBjb25maWc6IFZhbHRlY2hBdXRoQ29uZmlnXG4pOiBFbnZpcm9ubWVudFByb3ZpZGVycyB7XG4gIGNvbnN0IG1lcmdlZENvbmZpZzogVmFsdGVjaEF1dGhDb25maWcgPSB7XG4gICAgLi4uREVGQVVMVF9BVVRIX0NPTkZJRyxcbiAgICAuLi5jb25maWcsXG4gIH07XG5cbiAgcmV0dXJuIG1ha2VFbnZpcm9ubWVudFByb3ZpZGVycyhbXG4gICAgeyBwcm92aWRlOiBWQUxURUNIX0FVVEhfQ09ORklHLCB1c2VWYWx1ZTogbWVyZ2VkQ29uZmlnIH0sXG4gICAgcHJvdmlkZUh0dHBDbGllbnQod2l0aEludGVyY2VwdG9ycyhbYXV0aEludGVyY2VwdG9yXSkpLFxuICAgIC8vIEluaWNpYWxpemFyIEF1dGhTZXJ2aWNlIGFsIGFycmFuY2FyIGxhIGFwcFxuICAgIHtcbiAgICAgIHByb3ZpZGU6IEFQUF9JTklUSUFMSVpFUixcbiAgICAgIHVzZUZhY3Rvcnk6IGluaXRpYWxpemVBdXRoLFxuICAgICAgZGVwczogW0F1dGhTZXJ2aWNlXSxcbiAgICAgIG11bHRpOiB0cnVlLFxuICAgIH0sXG4gIF0pO1xufVxuXG4vKipcbiAqIFByb3ZlZSBzb2xvIGVsIGludGVyY2VwdG9yIChwYXJhIGFwcHMgcXVlIHlhIHRpZW5lbiBBdXRoU2VydmljZSBjb25maWd1cmFkbyBtYW51YWxtZW50ZSkuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcm92aWRlVmFsdGVjaEF1dGhJbnRlcmNlcHRvcigpOiBFbnZpcm9ubWVudFByb3ZpZGVycyB7XG4gIHJldHVybiBtYWtlRW52aXJvbm1lbnRQcm92aWRlcnMoW1xuICAgIHByb3ZpZGVIdHRwQ2xpZW50KHdpdGhJbnRlcmNlcHRvcnMoW2F1dGhJbnRlcmNlcHRvcl0pKSxcbiAgXSk7XG59XG4iXX0=
|