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