valtech-components 2.0.452 → 2.0.453
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/auth/auth-state.service.mjs +173 -0
- package/esm2022/lib/services/auth/auth.service.mjs +454 -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 +141 -0
- package/esm2022/lib/services/auth/sync.service.mjs +149 -0
- package/esm2022/lib/services/auth/token.service.mjs +113 -0
- package/esm2022/lib/services/auth/types.mjs +29 -0
- package/esm2022/lib/services/firebase/config.mjs +108 -0
- package/esm2022/lib/services/firebase/firebase.service.mjs +288 -0
- package/esm2022/lib/services/firebase/firestore-collection.mjs +254 -0
- package/esm2022/lib/services/firebase/firestore.service.mjs +509 -0
- package/esm2022/lib/services/firebase/index.mjs +49 -0
- package/esm2022/lib/services/firebase/messaging.service.mjs +512 -0
- package/esm2022/lib/services/firebase/shared-config.mjs +138 -0
- package/esm2022/lib/services/firebase/storage.service.mjs +422 -0
- package/esm2022/lib/services/firebase/types.mjs +8 -0
- package/esm2022/lib/services/firebase/utils/path-builder.mjs +195 -0
- package/esm2022/lib/services/firebase/utils/query-builder.mjs +302 -0
- package/esm2022/public-api.mjs +3 -5
- package/fesm2022/valtech-components.mjs +4195 -4
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/services/auth/auth-state.service.d.ts +85 -0
- package/lib/services/auth/auth.service.d.ts +146 -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 +315 -0
- package/lib/services/firebase/config.d.ts +49 -0
- package/lib/services/firebase/firebase.service.d.ts +140 -0
- package/lib/services/firebase/firestore-collection.d.ts +175 -0
- package/lib/services/firebase/firestore.service.d.ts +304 -0
- package/lib/services/firebase/index.d.ts +39 -0
- package/lib/services/firebase/messaging.service.d.ts +263 -0
- package/lib/services/firebase/shared-config.d.ts +126 -0
- package/lib/services/firebase/storage.service.d.ts +206 -0
- package/lib/services/firebase/types.d.ts +281 -0
- package/lib/services/firebase/utils/path-builder.d.ts +132 -0
- package/lib/services/firebase/utils/query-builder.d.ts +210 -0
- package/package.json +1 -1
- package/public-api.d.ts +2 -0
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import { Inject, Injectable } from '@angular/core';
|
|
2
|
+
import { Subject } from 'rxjs';
|
|
3
|
+
import { VALTECH_AUTH_CONFIG } from './config';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
/**
|
|
6
|
+
* Servicio para sincronización de estado de autenticación entre pestañas.
|
|
7
|
+
* Usa BroadcastChannel API con fallback a storage events.
|
|
8
|
+
*/
|
|
9
|
+
export class AuthSyncService {
|
|
10
|
+
constructor(config) {
|
|
11
|
+
this.config = config;
|
|
12
|
+
this.channel = null;
|
|
13
|
+
this.eventSubject = new Subject();
|
|
14
|
+
this.storageListener = null;
|
|
15
|
+
/** Observable de eventos de sincronización */
|
|
16
|
+
this.onEvent$ = this.eventSubject.asObservable();
|
|
17
|
+
const prefix = this.config.storagePrefix || 'valtech_auth_';
|
|
18
|
+
this.channelName = `${prefix}sync_channel`;
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Inicia la sincronización entre pestañas.
|
|
22
|
+
*/
|
|
23
|
+
start() {
|
|
24
|
+
if (!this.config.enableTabSync) {
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
// Intentar usar BroadcastChannel API (mejor rendimiento)
|
|
28
|
+
if (typeof BroadcastChannel !== 'undefined') {
|
|
29
|
+
this.initBroadcastChannel();
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
// Fallback a storage events
|
|
33
|
+
this.initStorageEvents();
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Detiene la sincronización.
|
|
38
|
+
*/
|
|
39
|
+
stop() {
|
|
40
|
+
if (this.channel) {
|
|
41
|
+
this.channel.close();
|
|
42
|
+
this.channel = null;
|
|
43
|
+
}
|
|
44
|
+
if (this.storageListener) {
|
|
45
|
+
window.removeEventListener('storage', this.storageListener);
|
|
46
|
+
this.storageListener = null;
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Envía un evento a otras pestañas.
|
|
51
|
+
*/
|
|
52
|
+
broadcast(event) {
|
|
53
|
+
if (!this.config.enableTabSync) {
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
const fullEvent = {
|
|
57
|
+
...event,
|
|
58
|
+
timestamp: Date.now(),
|
|
59
|
+
};
|
|
60
|
+
if (this.channel) {
|
|
61
|
+
this.channel.postMessage(fullEvent);
|
|
62
|
+
}
|
|
63
|
+
else {
|
|
64
|
+
// Fallback: usar localStorage para notificar
|
|
65
|
+
this.broadcastViaStorage(fullEvent);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
ngOnDestroy() {
|
|
69
|
+
this.stop();
|
|
70
|
+
this.eventSubject.complete();
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* Inicializa BroadcastChannel API.
|
|
74
|
+
*/
|
|
75
|
+
initBroadcastChannel() {
|
|
76
|
+
try {
|
|
77
|
+
this.channel = new BroadcastChannel(this.channelName);
|
|
78
|
+
this.channel.onmessage = (event) => {
|
|
79
|
+
this.handleEvent(event.data);
|
|
80
|
+
};
|
|
81
|
+
this.channel.onmessageerror = () => {
|
|
82
|
+
console.warn('[ValtechAuth] Error en BroadcastChannel, usando fallback');
|
|
83
|
+
this.channel?.close();
|
|
84
|
+
this.channel = null;
|
|
85
|
+
this.initStorageEvents();
|
|
86
|
+
};
|
|
87
|
+
}
|
|
88
|
+
catch {
|
|
89
|
+
// BroadcastChannel no disponible, usar fallback
|
|
90
|
+
this.initStorageEvents();
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* Inicializa fallback con storage events.
|
|
95
|
+
*/
|
|
96
|
+
initStorageEvents() {
|
|
97
|
+
const storageKey = `${this.config.storagePrefix}sync_event`;
|
|
98
|
+
this.storageListener = (event) => {
|
|
99
|
+
if (event.key === storageKey && event.newValue) {
|
|
100
|
+
try {
|
|
101
|
+
const syncEvent = JSON.parse(event.newValue);
|
|
102
|
+
this.handleEvent(syncEvent);
|
|
103
|
+
}
|
|
104
|
+
catch {
|
|
105
|
+
// Ignorar eventos mal formados
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
};
|
|
109
|
+
window.addEventListener('storage', this.storageListener);
|
|
110
|
+
}
|
|
111
|
+
/**
|
|
112
|
+
* Envía evento via localStorage (fallback).
|
|
113
|
+
*/
|
|
114
|
+
broadcastViaStorage(event) {
|
|
115
|
+
const storageKey = `${this.config.storagePrefix}sync_event`;
|
|
116
|
+
try {
|
|
117
|
+
// Escribir y luego limpiar para permitir múltiples eventos del mismo tipo
|
|
118
|
+
localStorage.setItem(storageKey, JSON.stringify(event));
|
|
119
|
+
// Usar setTimeout para permitir que otras pestañas lean el valor
|
|
120
|
+
setTimeout(() => {
|
|
121
|
+
localStorage.removeItem(storageKey);
|
|
122
|
+
}, 100);
|
|
123
|
+
}
|
|
124
|
+
catch {
|
|
125
|
+
console.warn('[ValtechAuth] Error enviando evento via storage');
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
/**
|
|
129
|
+
* Maneja un evento recibido.
|
|
130
|
+
*/
|
|
131
|
+
handleEvent(event) {
|
|
132
|
+
// Verificar que el evento no sea muy antiguo (más de 5 segundos)
|
|
133
|
+
const age = Date.now() - event.timestamp;
|
|
134
|
+
if (age > 5000) {
|
|
135
|
+
return;
|
|
136
|
+
}
|
|
137
|
+
this.eventSubject.next(event);
|
|
138
|
+
}
|
|
139
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthSyncService, deps: [{ token: VALTECH_AUTH_CONFIG }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
140
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthSyncService, providedIn: 'root' }); }
|
|
141
|
+
}
|
|
142
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthSyncService, decorators: [{
|
|
143
|
+
type: Injectable,
|
|
144
|
+
args: [{ providedIn: 'root' }]
|
|
145
|
+
}], ctorParameters: () => [{ type: undefined, decorators: [{
|
|
146
|
+
type: Inject,
|
|
147
|
+
args: [VALTECH_AUTH_CONFIG]
|
|
148
|
+
}] }] });
|
|
149
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sync.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/sync.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, OnDestroy } from '@angular/core';\nimport { Subject, Observable } from 'rxjs';\nimport { VALTECH_AUTH_CONFIG } from './config';\nimport { AuthSyncEvent, AuthSyncEventType, ValtechAuthConfig } from './types';\n\n/**\n * Servicio para sincronización de estado de autenticación entre pestañas.\n * Usa BroadcastChannel API con fallback a storage events.\n */\n@Injectable({ providedIn: 'root' })\nexport class AuthSyncService implements OnDestroy {\n  private channel: BroadcastChannel | null = null;\n  private channelName: string;\n  private eventSubject = new Subject<AuthSyncEvent>();\n  private storageListener: ((event: StorageEvent) => void) | null = null;\n\n  /** Observable de eventos de sincronización */\n  readonly onEvent$: Observable<AuthSyncEvent> = this.eventSubject.asObservable();\n\n  constructor(\n    @Inject(VALTECH_AUTH_CONFIG) private config: ValtechAuthConfig\n  ) {\n    const prefix = this.config.storagePrefix || 'valtech_auth_';\n    this.channelName = `${prefix}sync_channel`;\n  }\n\n  /**\n   * Inicia la sincronización entre pestañas.\n   */\n  start(): void {\n    if (!this.config.enableTabSync) {\n      return;\n    }\n\n    // Intentar usar BroadcastChannel API (mejor rendimiento)\n    if (typeof BroadcastChannel !== 'undefined') {\n      this.initBroadcastChannel();\n    } else {\n      // Fallback a storage events\n      this.initStorageEvents();\n    }\n  }\n\n  /**\n   * Detiene la sincronización.\n   */\n  stop(): void {\n    if (this.channel) {\n      this.channel.close();\n      this.channel = null;\n    }\n\n    if (this.storageListener) {\n      window.removeEventListener('storage', this.storageListener);\n      this.storageListener = null;\n    }\n  }\n\n  /**\n   * Envía un evento a otras pestañas.\n   */\n  broadcast(event: Omit<AuthSyncEvent, 'timestamp'>): void {\n    if (!this.config.enableTabSync) {\n      return;\n    }\n\n    const fullEvent: AuthSyncEvent = {\n      ...event,\n      timestamp: Date.now(),\n    };\n\n    if (this.channel) {\n      this.channel.postMessage(fullEvent);\n    } else {\n      // Fallback: usar localStorage para notificar\n      this.broadcastViaStorage(fullEvent);\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.stop();\n    this.eventSubject.complete();\n  }\n\n  /**\n   * Inicializa BroadcastChannel API.\n   */\n  private initBroadcastChannel(): void {\n    try {\n      this.channel = new BroadcastChannel(this.channelName);\n\n      this.channel.onmessage = (event: MessageEvent<AuthSyncEvent>) => {\n        this.handleEvent(event.data);\n      };\n\n      this.channel.onmessageerror = () => {\n        console.warn('[ValtechAuth] Error en BroadcastChannel, usando fallback');\n        this.channel?.close();\n        this.channel = null;\n        this.initStorageEvents();\n      };\n    } catch {\n      // BroadcastChannel no disponible, usar fallback\n      this.initStorageEvents();\n    }\n  }\n\n  /**\n   * Inicializa fallback con storage events.\n   */\n  private initStorageEvents(): void {\n    const storageKey = `${this.config.storagePrefix}sync_event`;\n\n    this.storageListener = (event: StorageEvent) => {\n      if (event.key === storageKey && event.newValue) {\n        try {\n          const syncEvent: AuthSyncEvent = JSON.parse(event.newValue);\n          this.handleEvent(syncEvent);\n        } catch {\n          // Ignorar eventos mal formados\n        }\n      }\n    };\n\n    window.addEventListener('storage', this.storageListener);\n  }\n\n  /**\n   * Envía evento via localStorage (fallback).\n   */\n  private broadcastViaStorage(event: AuthSyncEvent): void {\n    const storageKey = `${this.config.storagePrefix}sync_event`;\n\n    try {\n      // Escribir y luego limpiar para permitir múltiples eventos del mismo tipo\n      localStorage.setItem(storageKey, JSON.stringify(event));\n      // Usar setTimeout para permitir que otras pestañas lean el valor\n      setTimeout(() => {\n        localStorage.removeItem(storageKey);\n      }, 100);\n    } catch {\n      console.warn('[ValtechAuth] Error enviando evento via storage');\n    }\n  }\n\n  /**\n   * Maneja un evento recibido.\n   */\n  private handleEvent(event: AuthSyncEvent): void {\n    // Verificar que el evento no sea muy antiguo (más de 5 segundos)\n    const age = Date.now() - event.timestamp;\n    if (age > 5000) {\n      return;\n    }\n\n    this.eventSubject.next(event);\n  }\n}\n"]}
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* Servicio para manejo de tokens JWT.
|
|
5
|
+
* Parseo y validación de tokens sin dependencias externas.
|
|
6
|
+
*/
|
|
7
|
+
export class TokenService {
|
|
8
|
+
/**
|
|
9
|
+
* Parsea un token JWT y extrae los claims.
|
|
10
|
+
* @param token - Token JWT
|
|
11
|
+
* @returns Claims del token o null si es inválido
|
|
12
|
+
*/
|
|
13
|
+
parseToken(token) {
|
|
14
|
+
try {
|
|
15
|
+
const parts = token.split('.');
|
|
16
|
+
if (parts.length !== 3) {
|
|
17
|
+
return null;
|
|
18
|
+
}
|
|
19
|
+
const payload = parts[1];
|
|
20
|
+
const decoded = this.base64UrlDecode(payload);
|
|
21
|
+
return JSON.parse(decoded);
|
|
22
|
+
}
|
|
23
|
+
catch {
|
|
24
|
+
console.warn('[ValtechAuth] Error al parsear token JWT');
|
|
25
|
+
return null;
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Verifica si un token es válido (no expirado).
|
|
30
|
+
* @param token - Token JWT
|
|
31
|
+
* @returns true si el token es válido
|
|
32
|
+
*/
|
|
33
|
+
isTokenValid(token) {
|
|
34
|
+
const claims = this.parseToken(token);
|
|
35
|
+
if (!claims) {
|
|
36
|
+
return false;
|
|
37
|
+
}
|
|
38
|
+
// exp está en segundos, Date.now() en milisegundos
|
|
39
|
+
const expirationMs = claims.exp * 1000;
|
|
40
|
+
return Date.now() < expirationMs;
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* Obtiene el tiempo restante del token en segundos.
|
|
44
|
+
* @param token - Token JWT
|
|
45
|
+
* @returns Segundos restantes o 0 si expirado
|
|
46
|
+
*/
|
|
47
|
+
getTimeToExpiry(token) {
|
|
48
|
+
const claims = this.parseToken(token);
|
|
49
|
+
if (!claims) {
|
|
50
|
+
return 0;
|
|
51
|
+
}
|
|
52
|
+
const expirationMs = claims.exp * 1000;
|
|
53
|
+
const remaining = expirationMs - Date.now();
|
|
54
|
+
return remaining > 0 ? Math.floor(remaining / 1000) : 0;
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Obtiene el timestamp de expiración del token.
|
|
58
|
+
* @param token - Token JWT
|
|
59
|
+
* @returns Timestamp en milisegundos o null
|
|
60
|
+
*/
|
|
61
|
+
getExpirationTime(token) {
|
|
62
|
+
const claims = this.parseToken(token);
|
|
63
|
+
if (!claims) {
|
|
64
|
+
return null;
|
|
65
|
+
}
|
|
66
|
+
return claims.exp * 1000;
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Extrae el user ID del token.
|
|
70
|
+
* @param token - Token JWT
|
|
71
|
+
* @returns User ID o null
|
|
72
|
+
*/
|
|
73
|
+
getUserId(token) {
|
|
74
|
+
const claims = this.parseToken(token);
|
|
75
|
+
return claims?.uid || null;
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Extrae el email del token.
|
|
79
|
+
* @param token - Token JWT
|
|
80
|
+
* @returns Email o null
|
|
81
|
+
*/
|
|
82
|
+
getEmail(token) {
|
|
83
|
+
const claims = this.parseToken(token);
|
|
84
|
+
return claims?.email || null;
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Decodifica base64url a string.
|
|
88
|
+
* Base64url usa - y _ en lugar de + y /
|
|
89
|
+
*/
|
|
90
|
+
base64UrlDecode(str) {
|
|
91
|
+
// Reemplazar caracteres base64url por base64 estándar
|
|
92
|
+
let base64 = str.replace(/-/g, '+').replace(/_/g, '/');
|
|
93
|
+
// Agregar padding si es necesario
|
|
94
|
+
const padding = base64.length % 4;
|
|
95
|
+
if (padding) {
|
|
96
|
+
base64 += '='.repeat(4 - padding);
|
|
97
|
+
}
|
|
98
|
+
// Decodificar
|
|
99
|
+
const decoded = atob(base64);
|
|
100
|
+
// Manejar caracteres UTF-8
|
|
101
|
+
return decodeURIComponent(decoded
|
|
102
|
+
.split('')
|
|
103
|
+
.map((c) => '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2))
|
|
104
|
+
.join(''));
|
|
105
|
+
}
|
|
106
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TokenService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
107
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TokenService, providedIn: 'root' }); }
|
|
108
|
+
}
|
|
109
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TokenService, decorators: [{
|
|
110
|
+
type: Injectable,
|
|
111
|
+
args: [{ providedIn: 'root' }]
|
|
112
|
+
}] });
|
|
113
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG9rZW4uc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3NyYy9saWIvc2VydmljZXMvYXV0aC90b2tlbi5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBRzNDOzs7R0FHRztBQUVILE1BQU0sT0FBTyxZQUFZO0lBQ3ZCOzs7O09BSUc7SUFDSCxVQUFVLENBQUMsS0FBYTtRQUN0QixJQUFJLENBQUM7WUFDSCxNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1lBQy9CLElBQUksS0FBSyxDQUFDLE1BQU0sS0FBSyxDQUFDLEVBQUUsQ0FBQztnQkFDdkIsT0FBTyxJQUFJLENBQUM7WUFDZCxDQUFDO1lBRUQsTUFBTSxPQUFPLEdBQUcsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQ3pCLE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxlQUFlLENBQUMsT0FBTyxDQUFDLENBQUM7WUFDOUMsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQzdCLENBQUM7UUFBQyxNQUFNLENBQUM7WUFDUCxPQUFPLENBQUMsSUFBSSxDQUFDLDBDQUEwQyxDQUFDLENBQUM7WUFDekQsT0FBTyxJQUFJLENBQUM7UUFDZCxDQUFDO0lBQ0gsQ0FBQztJQUVEOzs7O09BSUc7SUFDSCxZQUFZLENBQUMsS0FBYTtRQUN4QixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3RDLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQztZQUNaLE9BQU8sS0FBSyxDQUFDO1FBQ2YsQ0FBQztRQUVELG1EQUFtRDtRQUNuRCxNQUFNLFlBQVksR0FBRyxNQUFNLENBQUMsR0FBRyxHQUFHLElBQUksQ0FBQztRQUN2QyxPQUFPLElBQUksQ0FBQyxHQUFHLEVBQUUsR0FBRyxZQUFZLENBQUM7SUFDbkMsQ0FBQztJQUVEOzs7O09BSUc7SUFDSCxlQUFlLENBQUMsS0FBYTtRQUMzQixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3RDLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQztZQUNaLE9BQU8sQ0FBQyxDQUFDO1FBQ1gsQ0FBQztRQUVELE1BQU0sWUFBWSxHQUFHLE1BQU0sQ0FBQyxHQUFHLEdBQUcsSUFBSSxDQUFDO1FBQ3ZDLE1BQU0sU0FBUyxHQUFHLFlBQVksR0FBRyxJQUFJLENBQUMsR0FBRyxFQUFFLENBQUM7UUFDNUMsT0FBTyxTQUFTLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLFNBQVMsR0FBRyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQzFELENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsaUJBQWlCLENBQUMsS0FBYTtRQUM3QixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3RDLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQztZQUNaLE9BQU8sSUFBSSxDQUFDO1FBQ2QsQ0FBQztRQUVELE9BQU8sTUFBTSxDQUFDLEdBQUcsR0FBRyxJQUFJLENBQUM7SUFDM0IsQ0FBQztJQUVEOzs7O09BSUc7SUFDSCxTQUFTLENBQUMsS0FBYTtRQUNyQixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3RDLE9BQU8sTUFBTSxFQUFFLEdBQUcsSUFBSSxJQUFJLENBQUM7SUFDN0IsQ0FBQztJQUVEOzs7O09BSUc7SUFDSCxRQUFRLENBQUMsS0FBYTtRQUNwQixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3RDLE9BQU8sTUFBTSxFQUFFLEtBQUssSUFBSSxJQUFJLENBQUM7SUFDL0IsQ0FBQztJQUVEOzs7T0FHRztJQUNLLGVBQWUsQ0FBQyxHQUFXO1FBQ2pDLHNEQUFzRDtRQUN0RCxJQUFJLE1BQU0sR0FBRyxHQUFHLENBQUMsT0FBTyxDQUFDLElBQUksRUFBRSxHQUFHLENBQUMsQ0FBQyxPQUFPLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxDQUFDO1FBRXZELGtDQUFrQztRQUNsQyxNQUFNLE9BQU8sR0FBRyxNQUFNLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQztRQUNsQyxJQUFJLE9BQU8sRUFBRSxDQUFDO1lBQ1osTUFBTSxJQUFJLEdBQUcsQ0FBQyxNQUFNLENBQUMsQ0FBQyxHQUFHLE9BQU8sQ0FBQyxDQUFDO1FBQ3BDLENBQUM7UUFFRCxjQUFjO1FBQ2QsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBRTdCLDJCQUEyQjtRQUMzQixPQUFPLGtCQUFrQixDQUN2QixPQUFPO2FBQ0osS0FBSyxDQUFDLEVBQUUsQ0FBQzthQUNULEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsR0FBRyxHQUFHLENBQUMsSUFBSSxHQUFHLENBQUMsQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDLENBQUMsUUFBUSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7YUFDakUsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUNaLENBQUM7SUFDSixDQUFDOytHQWhIVSxZQUFZO21IQUFaLFlBQVksY0FEQyxNQUFNOzs0RkFDbkIsWUFBWTtrQkFEeEIsVUFBVTttQkFBQyxFQUFFLFVBQVUsRUFBRSxNQUFNLEVBQUUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBKV1RDbGFpbXMgfSBmcm9tICcuL3R5cGVzJztcblxuLyoqXG4gKiBTZXJ2aWNpbyBwYXJhIG1hbmVqbyBkZSB0b2tlbnMgSldULlxuICogUGFyc2VvIHkgdmFsaWRhY2nDs24gZGUgdG9rZW5zIHNpbiBkZXBlbmRlbmNpYXMgZXh0ZXJuYXMuXG4gKi9cbkBJbmplY3RhYmxlKHsgcHJvdmlkZWRJbjogJ3Jvb3QnIH0pXG5leHBvcnQgY2xhc3MgVG9rZW5TZXJ2aWNlIHtcbiAgLyoqXG4gICAqIFBhcnNlYSB1biB0b2tlbiBKV1QgeSBleHRyYWUgbG9zIGNsYWltcy5cbiAgICogQHBhcmFtIHRva2VuIC0gVG9rZW4gSldUXG4gICAqIEByZXR1cm5zIENsYWltcyBkZWwgdG9rZW4gbyBudWxsIHNpIGVzIGludsOhbGlkb1xuICAgKi9cbiAgcGFyc2VUb2tlbih0b2tlbjogc3RyaW5nKTogSldUQ2xhaW1zIHwgbnVsbCB7XG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHBhcnRzID0gdG9rZW4uc3BsaXQoJy4nKTtcbiAgICAgIGlmIChwYXJ0cy5sZW5ndGggIT09IDMpIHtcbiAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHBheWxvYWQgPSBwYXJ0c1sxXTtcbiAgICAgIGNvbnN0IGRlY29kZWQgPSB0aGlzLmJhc2U2NFVybERlY29kZShwYXlsb2FkKTtcbiAgICAgIHJldHVybiBKU09OLnBhcnNlKGRlY29kZWQpO1xuICAgIH0gY2F0Y2gge1xuICAgICAgY29uc29sZS53YXJuKCdbVmFsdGVjaEF1dGhdIEVycm9yIGFsIHBhcnNlYXIgdG9rZW4gSldUJyk7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogVmVyaWZpY2Egc2kgdW4gdG9rZW4gZXMgdsOhbGlkbyAobm8gZXhwaXJhZG8pLlxuICAgKiBAcGFyYW0gdG9rZW4gLSBUb2tlbiBKV1RcbiAgICogQHJldHVybnMgdHJ1ZSBzaSBlbCB0b2tlbiBlcyB2w6FsaWRvXG4gICAqL1xuICBpc1Rva2VuVmFsaWQodG9rZW46IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IGNsYWltcyA9IHRoaXMucGFyc2VUb2tlbih0b2tlbik7XG4gICAgaWYgKCFjbGFpbXMpIHtcbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG5cbiAgICAvLyBleHAgZXN0w6EgZW4gc2VndW5kb3MsIERhdGUubm93KCkgZW4gbWlsaXNlZ3VuZG9zXG4gICAgY29uc3QgZXhwaXJhdGlvbk1zID0gY2xhaW1zLmV4cCAqIDEwMDA7XG4gICAgcmV0dXJuIERhdGUubm93KCkgPCBleHBpcmF0aW9uTXM7XG4gIH1cblxuICAvKipcbiAgICogT2J0aWVuZSBlbCB0aWVtcG8gcmVzdGFudGUgZGVsIHRva2VuIGVuIHNlZ3VuZG9zLlxuICAgKiBAcGFyYW0gdG9rZW4gLSBUb2tlbiBKV1RcbiAgICogQHJldHVybnMgU2VndW5kb3MgcmVzdGFudGVzIG8gMCBzaSBleHBpcmFkb1xuICAgKi9cbiAgZ2V0VGltZVRvRXhwaXJ5KHRva2VuOiBzdHJpbmcpOiBudW1iZXIge1xuICAgIGNvbnN0IGNsYWltcyA9IHRoaXMucGFyc2VUb2tlbih0b2tlbik7XG4gICAgaWYgKCFjbGFpbXMpIHtcbiAgICAgIHJldHVybiAwO1xuICAgIH1cblxuICAgIGNvbnN0IGV4cGlyYXRpb25NcyA9IGNsYWltcy5leHAgKiAxMDAwO1xuICAgIGNvbnN0IHJlbWFpbmluZyA9IGV4cGlyYXRpb25NcyAtIERhdGUubm93KCk7XG4gICAgcmV0dXJuIHJlbWFpbmluZyA+IDAgPyBNYXRoLmZsb29yKHJlbWFpbmluZyAvIDEwMDApIDogMDtcbiAgfVxuXG4gIC8qKlxuICAgKiBPYnRpZW5lIGVsIHRpbWVzdGFtcCBkZSBleHBpcmFjacOzbiBkZWwgdG9rZW4uXG4gICAqIEBwYXJhbSB0b2tlbiAtIFRva2VuIEpXVFxuICAgKiBAcmV0dXJucyBUaW1lc3RhbXAgZW4gbWlsaXNlZ3VuZG9zIG8gbnVsbFxuICAgKi9cbiAgZ2V0RXhwaXJhdGlvblRpbWUodG9rZW46IHN0cmluZyk6IG51bWJlciB8IG51bGwge1xuICAgIGNvbnN0IGNsYWltcyA9IHRoaXMucGFyc2VUb2tlbih0b2tlbik7XG4gICAgaWYgKCFjbGFpbXMpIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH1cblxuICAgIHJldHVybiBjbGFpbXMuZXhwICogMTAwMDtcbiAgfVxuXG4gIC8qKlxuICAgKiBFeHRyYWUgZWwgdXNlciBJRCBkZWwgdG9rZW4uXG4gICAqIEBwYXJhbSB0b2tlbiAtIFRva2VuIEpXVFxuICAgKiBAcmV0dXJucyBVc2VyIElEIG8gbnVsbFxuICAgKi9cbiAgZ2V0VXNlcklkKHRva2VuOiBzdHJpbmcpOiBzdHJpbmcgfCBudWxsIHtcbiAgICBjb25zdCBjbGFpbXMgPSB0aGlzLnBhcnNlVG9rZW4odG9rZW4pO1xuICAgIHJldHVybiBjbGFpbXM/LnVpZCB8fCBudWxsO1xuICB9XG5cbiAgLyoqXG4gICAqIEV4dHJhZSBlbCBlbWFpbCBkZWwgdG9rZW4uXG4gICAqIEBwYXJhbSB0b2tlbiAtIFRva2VuIEpXVFxuICAgKiBAcmV0dXJucyBFbWFpbCBvIG51bGxcbiAgICovXG4gIGdldEVtYWlsKHRva2VuOiBzdHJpbmcpOiBzdHJpbmcgfCBudWxsIHtcbiAgICBjb25zdCBjbGFpbXMgPSB0aGlzLnBhcnNlVG9rZW4odG9rZW4pO1xuICAgIHJldHVybiBjbGFpbXM/LmVtYWlsIHx8IG51bGw7XG4gIH1cblxuICAvKipcbiAgICogRGVjb2RpZmljYSBiYXNlNjR1cmwgYSBzdHJpbmcuXG4gICAqIEJhc2U2NHVybCB1c2EgLSB5IF8gZW4gbHVnYXIgZGUgKyB5IC9cbiAgICovXG4gIHByaXZhdGUgYmFzZTY0VXJsRGVjb2RlKHN0cjogc3RyaW5nKTogc3RyaW5nIHtcbiAgICAvLyBSZWVtcGxhemFyIGNhcmFjdGVyZXMgYmFzZTY0dXJsIHBvciBiYXNlNjQgZXN0w6FuZGFyXG4gICAgbGV0IGJhc2U2NCA9IHN0ci5yZXBsYWNlKC8tL2csICcrJykucmVwbGFjZSgvXy9nLCAnLycpO1xuXG4gICAgLy8gQWdyZWdhciBwYWRkaW5nIHNpIGVzIG5lY2VzYXJpb1xuICAgIGNvbnN0IHBhZGRpbmcgPSBiYXNlNjQubGVuZ3RoICUgNDtcbiAgICBpZiAocGFkZGluZykge1xuICAgICAgYmFzZTY0ICs9ICc9Jy5yZXBlYXQoNCAtIHBhZGRpbmcpO1xuICAgIH1cblxuICAgIC8vIERlY29kaWZpY2FyXG4gICAgY29uc3QgZGVjb2RlZCA9IGF0b2IoYmFzZTY0KTtcblxuICAgIC8vIE1hbmVqYXIgY2FyYWN0ZXJlcyBVVEYtOFxuICAgIHJldHVybiBkZWNvZGVVUklDb21wb25lbnQoXG4gICAgICBkZWNvZGVkXG4gICAgICAgIC5zcGxpdCgnJylcbiAgICAgICAgLm1hcCgoYykgPT4gJyUnICsgKCcwMCcgKyBjLmNoYXJDb2RlQXQoMCkudG9TdHJpbmcoMTYpKS5zbGljZSgtMikpXG4gICAgICAgIC5qb2luKCcnKVxuICAgICk7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tipos e interfaces para el servicio de autenticación de Valtech.
|
|
3
|
+
* Alineados con el backend AuthV2.
|
|
4
|
+
*/
|
|
5
|
+
/**
|
|
6
|
+
* Estado inicial de autenticación.
|
|
7
|
+
*/
|
|
8
|
+
export const INITIAL_AUTH_STATE = {
|
|
9
|
+
isAuthenticated: false,
|
|
10
|
+
isLoading: true,
|
|
11
|
+
accessToken: null,
|
|
12
|
+
refreshToken: null,
|
|
13
|
+
userId: null,
|
|
14
|
+
email: null,
|
|
15
|
+
roles: [],
|
|
16
|
+
permissions: [],
|
|
17
|
+
isSuperAdmin: false,
|
|
18
|
+
expiresAt: null,
|
|
19
|
+
error: null,
|
|
20
|
+
};
|
|
21
|
+
/**
|
|
22
|
+
* Estado inicial de MFA.
|
|
23
|
+
*/
|
|
24
|
+
export const INITIAL_MFA_STATE = {
|
|
25
|
+
required: false,
|
|
26
|
+
mfaToken: null,
|
|
27
|
+
method: null,
|
|
28
|
+
};
|
|
29
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/types.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAkFH;;GAEG;AACH,MAAM,CAAC,MAAM,kBAAkB,GAAc;IAC3C,eAAe,EAAE,KAAK;IACtB,SAAS,EAAE,IAAI;IACf,WAAW,EAAE,IAAI;IACjB,YAAY,EAAE,IAAI;IAClB,MAAM,EAAE,IAAI;IACZ,KAAK,EAAE,IAAI;IACX,KAAK,EAAE,EAAE;IACT,WAAW,EAAE,EAAE;IACf,YAAY,EAAE,KAAK;IACnB,SAAS,EAAE,IAAI;IACf,KAAK,EAAE,IAAI;CACZ,CAAC;AAkBF;;GAEG;AACH,MAAM,CAAC,MAAM,iBAAiB,GAAoB;IAChD,QAAQ,EAAE,KAAK;IACf,QAAQ,EAAE,IAAI;IACd,MAAM,EAAE,IAAI;CACb,CAAC","sourcesContent":["/**\n * Tipos e interfaces para el servicio de autenticación de Valtech.\n * Alineados con el backend AuthV2.\n */\n\n// =============================================================================\n// CONFIGURACIÓN\n// =============================================================================\n\n/**\n * Configuración para el servicio de autenticación.\n */\nexport interface ValtechAuthConfig {\n  /** URL base de la API (ej: 'https://api.myvaltech.com') */\n  apiUrl: string;\n  /** Prefijo para endpoints de auth (default: '/v2/auth') */\n  authPrefix?: string;\n  /** Prefijo para las claves de localStorage (default: 'valtech_auth_') */\n  storagePrefix?: string;\n  /** Tiempo antes de expiración para refrescar token en segundos (default: 60) */\n  refreshBeforeExpiry?: number;\n  /** Habilitar sincronización entre pestañas (default: true) */\n  enableTabSync?: boolean;\n  /** Ruta de redirección cuando no autenticado (default: '/login') */\n  loginRoute?: string;\n  /** Ruta de redirección cuando ya autenticado (default: '/') */\n  homeRoute?: string;\n  /** Ruta para acceso denegado (default: '/unauthorized') */\n  unauthorizedRoute?: string;\n  /** Habilitar integración con FirebaseService (default: false) */\n  enableFirebaseIntegration?: boolean;\n}\n\n// =============================================================================\n// ESTADO DE AUTENTICACIÓN\n// =============================================================================\n\n/**\n * Estado completo de autenticación.\n */\nexport interface AuthState {\n  /** Usuario está autenticado */\n  isAuthenticated: boolean;\n  /** Estado de carga inicial */\n  isLoading: boolean;\n  /** Token de acceso actual */\n  accessToken: string | null;\n  /** Token de refresco actual */\n  refreshToken: string | null;\n  /** ID del usuario */\n  userId: string | null;\n  /** Email del usuario */\n  email: string | null;\n  /** Roles del usuario */\n  roles: string[];\n  /** Permisos del usuario (formato 'resource:action') */\n  permissions: string[];\n  /** Usuario es super admin */\n  isSuperAdmin: boolean;\n  /** Timestamp de expiración del accessToken (ms) */\n  expiresAt: number | null;\n  /** Error de autenticación (si existe) */\n  error: AuthError | null;\n}\n\n/**\n * Información del usuario autenticado.\n */\nexport interface AuthUser {\n  userId: string;\n  email: string;\n  name?: string;\n  roles: string[];\n  permissions: string[];\n  isSuperAdmin: boolean;\n}\n\n/**\n * Error de autenticación.\n */\nexport interface AuthError {\n  code: string;\n  message: string;\n}\n\n/**\n * Estado inicial de autenticación.\n */\nexport const INITIAL_AUTH_STATE: AuthState = {\n  isAuthenticated: false,\n  isLoading: true,\n  accessToken: null,\n  refreshToken: null,\n  userId: null,\n  email: null,\n  roles: [],\n  permissions: [],\n  isSuperAdmin: false,\n  expiresAt: null,\n  error: null,\n};\n\n// =============================================================================\n// MFA\n// =============================================================================\n\n/** Métodos de MFA soportados */\nexport type MFAMethod = 'EMAIL' | 'SMS';\n\n/**\n * Estado de MFA pendiente.\n */\nexport interface MFAPendingState {\n  required: boolean;\n  mfaToken: string | null;\n  method: MFAMethod | null;\n}\n\n/**\n * Estado inicial de MFA.\n */\nexport const INITIAL_MFA_STATE: MFAPendingState = {\n  required: false,\n  mfaToken: null,\n  method: null,\n};\n\n/**\n * Resultado de setup de MFA.\n */\nexport interface MFASetupResult {\n  codeSent: boolean;\n  message: string;\n}\n\n/**\n * Estado de MFA del usuario.\n */\nexport interface MFAStatus {\n  enabled: boolean;\n  method: MFAMethod | null;\n}\n\n// =============================================================================\n// REQUESTS/RESPONSES (alineados con backend AuthV2)\n// =============================================================================\n\n/**\n * Request para signup (registro).\n */\nexport interface SignupRequest {\n  email: string;\n  password: string;\n  name: string;\n  phone?: string;\n}\n\n/**\n * Response de signup (registro).\n */\nexport interface SignupResponse {\n  operationId: string;\n  userId: string;\n  message: string;\n}\n\n/**\n * Request para verificar email.\n */\nexport interface VerifyEmailRequest {\n  email: string;\n  code: string;\n}\n\n/**\n * Response de verificación de email.\n * Si es exitoso, incluye tokens para auto-login.\n */\nexport interface VerifyEmailResponse {\n  operationId: string;\n  verified: boolean;\n  accessToken?: string;\n  refreshToken?: string;\n  firebaseToken?: string;\n  expiresIn?: number;\n  tokenType?: string;\n}\n\n/**\n * Request para reenviar código de verificación.\n */\nexport interface ResendCodeRequest {\n  email: string;\n  type: 'EMAIL_VERIFY' | 'PASSWORD_RESET';\n}\n\n/**\n * Response de reenvío de código.\n */\nexport interface ResendCodeResponse {\n  operationId: string;\n  sent: boolean;\n}\n\n/**\n * Request para signin.\n */\nexport interface SigninRequest {\n  email: string;\n  password: string;\n}\n\n/**\n * Response de signin.\n */\nexport interface SigninResponse {\n  operationId: string;\n  accessToken?: string;\n  refreshToken?: string;\n  firebaseToken?: string;\n  expiresIn?: number;\n  tokenType?: string;\n  mfaRequired?: boolean;\n  mfaToken?: string;\n  mfaMethod?: MFAMethod;\n  roles?: string[];\n  permissions?: string[];\n}\n\n/**\n * Request para verificar MFA.\n */\nexport interface MFAVerifyRequest {\n  mfaToken: string;\n  code: string;\n}\n\n/**\n * Response de verificar MFA.\n */\nexport interface MFAVerifyResponse {\n  operationId: string;\n  accessToken: string;\n  refreshToken: string;\n  firebaseToken?: string;\n  expiresIn: number;\n  tokenType: string;\n  roles?: string[];\n  permissions?: string[];\n}\n\n/**\n * Request para refrescar token.\n */\nexport interface RefreshRequest {\n  refreshToken: string;\n}\n\n/**\n * Response de refrescar token.\n */\nexport interface RefreshResponse {\n  operationId: string;\n  accessToken: string;\n  expiresIn: number;\n}\n\n/**\n * Request para logout.\n */\nexport interface LogoutRequest {\n  refreshToken: string;\n}\n\n/**\n * Response de logout.\n */\nexport interface LogoutResponse {\n  operationId: string;\n  success: boolean;\n}\n\n/**\n * Response de obtener permisos.\n */\nexport interface GetPermissionsResponse {\n  operationId: string;\n  roles: string[];\n  permissions: string[];\n  isSuperAdmin: boolean;\n}\n\n/**\n * Request para setup de MFA.\n */\nexport interface MFASetupRequest {\n  method: MFAMethod;\n  phone?: string;\n}\n\n/**\n * Response de setup de MFA.\n */\nexport interface MFASetupResponse {\n  operationId: string;\n  codeSent: boolean;\n  message: string;\n}\n\n/**\n * Request para confirmar MFA.\n */\nexport interface MFAConfirmRequest {\n  code: string;\n}\n\n/**\n * Response de confirmar MFA.\n */\nexport interface MFAConfirmResponse {\n  operationId: string;\n  mfaEnabled: boolean;\n  method: MFAMethod;\n}\n\n/**\n * Request para deshabilitar MFA.\n */\nexport interface MFADisableRequest {\n  password: string;\n}\n\n/**\n * Response de deshabilitar MFA.\n */\nexport interface MFADisableResponse {\n  operationId: string;\n  mfaDisabled: boolean;\n}\n\n// =============================================================================\n// EVENTOS DE SINCRONIZACIÓN\n// =============================================================================\n\n/** Tipos de eventos de sincronización entre pestañas */\nexport type AuthSyncEventType =\n  | 'LOGIN'\n  | 'LOGOUT'\n  | 'TOKEN_REFRESH'\n  | 'PERMISSIONS_UPDATE';\n\n/**\n * Evento de sincronización entre pestañas.\n */\nexport interface AuthSyncEvent {\n  type: AuthSyncEventType;\n  timestamp: number;\n  payload?: {\n    accessToken?: string;\n    expiresAt?: number;\n  };\n}\n\n// =============================================================================\n// JWT CLAIMS\n// =============================================================================\n\n/**\n * Claims del JWT de acceso.\n */\nexport interface JWTClaims {\n  /** User ID */\n  uid: string;\n  /** Email */\n  email: string;\n  /** Session ID */\n  sid?: string;\n  /** Issued at */\n  iat: number;\n  /** Expiration */\n  exp: number;\n}\n\n// =============================================================================\n// STORAGE\n// =============================================================================\n\n/**\n * Datos persistidos en storage.\n */\nexport interface StoredAuthState {\n  accessToken: string;\n  refreshToken: string;\n  roles: string[];\n  permissions: string[];\n  isSuperAdmin: boolean;\n  expiresAt?: number;\n}\n"]}
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Firebase Configuration
|
|
3
|
+
*
|
|
4
|
+
* Configuración e inicialización de Firebase para aplicaciones Angular.
|
|
5
|
+
* Usa provideValtechFirebase() en el bootstrap de tu aplicación.
|
|
6
|
+
*/
|
|
7
|
+
import { InjectionToken, makeEnvironmentProviders } from '@angular/core';
|
|
8
|
+
import { initializeApp, provideFirebaseApp } from '@angular/fire/app';
|
|
9
|
+
import { connectAuthEmulator, getAuth, provideAuth } from '@angular/fire/auth';
|
|
10
|
+
import { connectFirestoreEmulator, enableIndexedDbPersistence, getFirestore, provideFirestore, } from '@angular/fire/firestore';
|
|
11
|
+
import { getMessaging, provideMessaging } from '@angular/fire/messaging';
|
|
12
|
+
import { connectStorageEmulator, getStorage, provideStorage } from '@angular/fire/storage';
|
|
13
|
+
/**
|
|
14
|
+
* Token de inyección para la configuración de Firebase.
|
|
15
|
+
* Usado internamente por los servicios de Firebase.
|
|
16
|
+
*/
|
|
17
|
+
export const VALTECH_FIREBASE_CONFIG = new InjectionToken('ValtechFirebaseConfig');
|
|
18
|
+
/**
|
|
19
|
+
* Provee Firebase a la aplicación Angular.
|
|
20
|
+
*
|
|
21
|
+
* @param config - Configuración de Firebase
|
|
22
|
+
* @returns EnvironmentProviders para usar en bootstrapApplication
|
|
23
|
+
*
|
|
24
|
+
* @example
|
|
25
|
+
* ```typescript
|
|
26
|
+
* // main.ts
|
|
27
|
+
* import { bootstrapApplication } from '@angular/platform-browser';
|
|
28
|
+
* import { provideValtechFirebase } from 'valtech-components';
|
|
29
|
+
* import { environment } from './environments/environment';
|
|
30
|
+
*
|
|
31
|
+
* bootstrapApplication(AppComponent, {
|
|
32
|
+
* providers: [
|
|
33
|
+
* provideValtechFirebase({
|
|
34
|
+
* firebase: environment.firebase,
|
|
35
|
+
* persistence: true,
|
|
36
|
+
* emulator: environment.useEmulators ? {
|
|
37
|
+
* firestore: { host: 'localhost', port: 8080 },
|
|
38
|
+
* auth: { host: 'localhost', port: 9099 },
|
|
39
|
+
* storage: { host: 'localhost', port: 9199 },
|
|
40
|
+
* } : undefined,
|
|
41
|
+
* }),
|
|
42
|
+
* ],
|
|
43
|
+
* });
|
|
44
|
+
* ```
|
|
45
|
+
*/
|
|
46
|
+
export function provideValtechFirebase(config) {
|
|
47
|
+
// Construir array de providers base
|
|
48
|
+
const providers = [
|
|
49
|
+
// Guardar configuración para uso en servicios
|
|
50
|
+
{ provide: VALTECH_FIREBASE_CONFIG, useValue: config },
|
|
51
|
+
// Inicializar Firebase App
|
|
52
|
+
provideFirebaseApp(() => initializeApp(config.firebase)),
|
|
53
|
+
// Firestore con soporte para emuladores y persistencia
|
|
54
|
+
provideFirestore(() => {
|
|
55
|
+
const firestore = getFirestore();
|
|
56
|
+
// Conectar a emulador si está configurado
|
|
57
|
+
if (config.emulator?.firestore) {
|
|
58
|
+
connectFirestoreEmulator(firestore, config.emulator.firestore.host, config.emulator.firestore.port);
|
|
59
|
+
}
|
|
60
|
+
// Habilitar persistencia offline si está configurada
|
|
61
|
+
if (config.persistence) {
|
|
62
|
+
enableIndexedDbPersistence(firestore).catch((err) => {
|
|
63
|
+
if (err.code === 'failed-precondition') {
|
|
64
|
+
console.warn('[ValtechFirebase] Persistencia no disponible: múltiples pestañas abiertas');
|
|
65
|
+
}
|
|
66
|
+
else if (err.code === 'unimplemented') {
|
|
67
|
+
console.warn('[ValtechFirebase] Persistencia no soportada en este navegador');
|
|
68
|
+
}
|
|
69
|
+
});
|
|
70
|
+
}
|
|
71
|
+
return firestore;
|
|
72
|
+
}),
|
|
73
|
+
// Auth con soporte para emulador
|
|
74
|
+
provideAuth(() => {
|
|
75
|
+
const auth = getAuth();
|
|
76
|
+
// Conectar a emulador si está configurado
|
|
77
|
+
if (config.emulator?.auth) {
|
|
78
|
+
connectAuthEmulator(auth, `http://${config.emulator.auth.host}:${config.emulator.auth.port}`, { disableWarnings: true });
|
|
79
|
+
}
|
|
80
|
+
return auth;
|
|
81
|
+
}),
|
|
82
|
+
// Storage con soporte para emulador
|
|
83
|
+
provideStorage(() => {
|
|
84
|
+
const storage = getStorage();
|
|
85
|
+
// Conectar a emulador si está configurado
|
|
86
|
+
if (config.emulator?.storage) {
|
|
87
|
+
connectStorageEmulator(storage, config.emulator.storage.host, config.emulator.storage.port);
|
|
88
|
+
}
|
|
89
|
+
return storage;
|
|
90
|
+
}),
|
|
91
|
+
];
|
|
92
|
+
// Messaging (FCM) - solo si está explícitamente habilitado
|
|
93
|
+
// Requiere Service Worker configurado, puede congelar la app si no está disponible
|
|
94
|
+
if (config.enableMessaging) {
|
|
95
|
+
providers.push(provideMessaging(() => getMessaging()));
|
|
96
|
+
}
|
|
97
|
+
return makeEnvironmentProviders(providers);
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Verifica si los emuladores están configurados.
|
|
101
|
+
*
|
|
102
|
+
* @param config - Configuración de Firebase
|
|
103
|
+
* @returns true si hay al menos un emulador configurado
|
|
104
|
+
*/
|
|
105
|
+
export function hasEmulators(config) {
|
|
106
|
+
return !!(config.emulator?.firestore || config.emulator?.auth || config.emulator?.storage);
|
|
107
|
+
}
|
|
108
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/config.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Firebase Configuration\n *\n * Configuración e inicialización de Firebase para aplicaciones Angular.\n * Usa provideValtechFirebase() en el bootstrap de tu aplicación.\n */\n\nimport { EnvironmentProviders, InjectionToken, makeEnvironmentProviders } from '@angular/core';\nimport { initializeApp, provideFirebaseApp } from '@angular/fire/app';\nimport { connectAuthEmulator, getAuth, provideAuth } from '@angular/fire/auth';\nimport {\n  connectFirestoreEmulator,\n  enableIndexedDbPersistence,\n  getFirestore,\n  provideFirestore,\n} from '@angular/fire/firestore';\nimport { getMessaging, provideMessaging } from '@angular/fire/messaging';\nimport { connectStorageEmulator, getStorage, provideStorage } from '@angular/fire/storage';\n\nimport { ValtechFirebaseConfig } from './types';\n\n/**\n * Token de inyección para la configuración de Firebase.\n * Usado internamente por los servicios de Firebase.\n */\nexport const VALTECH_FIREBASE_CONFIG = new InjectionToken<ValtechFirebaseConfig>(\n  'ValtechFirebaseConfig'\n);\n\n/**\n * Provee Firebase a la aplicación Angular.\n *\n * @param config - Configuración de Firebase\n * @returns EnvironmentProviders para usar en bootstrapApplication\n *\n * @example\n * ```typescript\n * // main.ts\n * import { bootstrapApplication } from '@angular/platform-browser';\n * import { provideValtechFirebase } from 'valtech-components';\n * import { environment } from './environments/environment';\n *\n * bootstrapApplication(AppComponent, {\n *   providers: [\n *     provideValtechFirebase({\n *       firebase: environment.firebase,\n *       persistence: true,\n *       emulator: environment.useEmulators ? {\n *         firestore: { host: 'localhost', port: 8080 },\n *         auth: { host: 'localhost', port: 9099 },\n *         storage: { host: 'localhost', port: 9199 },\n *       } : undefined,\n *     }),\n *   ],\n * });\n * ```\n */\nexport function provideValtechFirebase(config: ValtechFirebaseConfig): EnvironmentProviders {\n  // Construir array de providers base\n  const providers: any[] = [\n    // Guardar configuración para uso en servicios\n    { provide: VALTECH_FIREBASE_CONFIG, useValue: config },\n\n    // Inicializar Firebase App\n    provideFirebaseApp(() => initializeApp(config.firebase)),\n\n    // Firestore con soporte para emuladores y persistencia\n    provideFirestore(() => {\n      const firestore = getFirestore();\n\n      // Conectar a emulador si está configurado\n      if (config.emulator?.firestore) {\n        connectFirestoreEmulator(\n          firestore,\n          config.emulator.firestore.host,\n          config.emulator.firestore.port\n        );\n      }\n\n      // Habilitar persistencia offline si está configurada\n      if (config.persistence) {\n        enableIndexedDbPersistence(firestore).catch((err) => {\n          if (err.code === 'failed-precondition') {\n            console.warn(\n              '[ValtechFirebase] Persistencia no disponible: múltiples pestañas abiertas'\n            );\n          } else if (err.code === 'unimplemented') {\n            console.warn('[ValtechFirebase] Persistencia no soportada en este navegador');\n          }\n        });\n      }\n\n      return firestore;\n    }),\n\n    // Auth con soporte para emulador\n    provideAuth(() => {\n      const auth = getAuth();\n\n      // Conectar a emulador si está configurado\n      if (config.emulator?.auth) {\n        connectAuthEmulator(\n          auth,\n          `http://${config.emulator.auth.host}:${config.emulator.auth.port}`,\n          { disableWarnings: true }\n        );\n      }\n\n      return auth;\n    }),\n\n    // Storage con soporte para emulador\n    provideStorage(() => {\n      const storage = getStorage();\n\n      // Conectar a emulador si está configurado\n      if (config.emulator?.storage) {\n        connectStorageEmulator(\n          storage,\n          config.emulator.storage.host,\n          config.emulator.storage.port\n        );\n      }\n\n      return storage;\n    }),\n  ];\n\n  // Messaging (FCM) - solo si está explícitamente habilitado\n  // Requiere Service Worker configurado, puede congelar la app si no está disponible\n  if (config.enableMessaging) {\n    providers.push(provideMessaging(() => getMessaging()));\n  }\n\n  return makeEnvironmentProviders(providers);\n}\n\n/**\n * Verifica si los emuladores están configurados.\n *\n * @param config - Configuración de Firebase\n * @returns true si hay al menos un emulador configurado\n */\nexport function hasEmulators(config: ValtechFirebaseConfig): boolean {\n  return !!(config.emulator?.firestore || config.emulator?.auth || config.emulator?.storage);\n}\n"]}
|