valtech-components 2.0.417 → 2.0.418
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/public-api.mjs +4 -2
- package/fesm2022/valtech-components.mjs +4 -2607
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/package.json +1 -1
- package/public-api.d.ts +0 -1
- package/esm2022/lib/services/firebase/config.mjs +0 -108
- package/esm2022/lib/services/firebase/firebase.service.mjs +0 -285
- package/esm2022/lib/services/firebase/firestore-collection.mjs +0 -266
- package/esm2022/lib/services/firebase/firestore.service.mjs +0 -508
- package/esm2022/lib/services/firebase/index.mjs +0 -46
- package/esm2022/lib/services/firebase/messaging.service.mjs +0 -503
- package/esm2022/lib/services/firebase/storage.service.mjs +0 -421
- package/esm2022/lib/services/firebase/types.mjs +0 -8
- package/esm2022/lib/services/firebase/utils/path-builder.mjs +0 -195
- package/esm2022/lib/services/firebase/utils/query-builder.mjs +0 -302
- package/lib/services/firebase/config.d.ts +0 -49
- package/lib/services/firebase/firebase.service.d.ts +0 -140
- package/lib/services/firebase/firestore-collection.d.ts +0 -195
- package/lib/services/firebase/firestore.service.d.ts +0 -303
- package/lib/services/firebase/index.d.ts +0 -38
- package/lib/services/firebase/messaging.service.d.ts +0 -254
- package/lib/services/firebase/storage.service.d.ts +0 -204
- package/lib/services/firebase/types.d.ts +0 -281
- package/lib/services/firebase/utils/path-builder.d.ts +0 -132
- package/lib/services/firebase/utils/query-builder.d.ts +0 -210
package/package.json
CHANGED
package/public-api.d.ts
CHANGED
|
@@ -212,7 +212,6 @@ export * from './lib/services/qr-generator/qr-generator.service';
|
|
|
212
212
|
export * from './lib/services/qr-generator/types';
|
|
213
213
|
export * from './lib/services/modal/modal.service';
|
|
214
214
|
export * from './lib/services/modal/types';
|
|
215
|
-
export * from './lib/services/firebase';
|
|
216
215
|
export * from './lib/components/types';
|
|
217
216
|
export * from './lib/shared/pipes/process-links.pipe';
|
|
218
217
|
export * from './lib/shared/utils/content';
|
|
@@ -1,108 +0,0 @@
|
|
|
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"]}
|
|
@@ -1,285 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Firebase Service
|
|
3
|
-
*
|
|
4
|
-
* Servicio principal para la autenticación con Firebase usando Custom Tokens.
|
|
5
|
-
* Permite que usuarios autenticados con tu backend (Cognito, etc.) accedan
|
|
6
|
-
* a servicios de Firebase (Firestore, Storage, FCM) de manera segura.
|
|
7
|
-
*/
|
|
8
|
-
import { inject, Injectable } from '@angular/core';
|
|
9
|
-
import { Auth, authState, signInWithCustomToken, signOut } from '@angular/fire/auth';
|
|
10
|
-
import { BehaviorSubject, distinctUntilChanged, map } from 'rxjs';
|
|
11
|
-
import { VALTECH_FIREBASE_CONFIG } from './config';
|
|
12
|
-
import * as i0 from "@angular/core";
|
|
13
|
-
/**
|
|
14
|
-
* Servicio de autenticación de Firebase.
|
|
15
|
-
*
|
|
16
|
-
* Este servicio NO maneja el login de usuarios directamente.
|
|
17
|
-
* En su lugar, trabaja con Custom Tokens generados por tu backend.
|
|
18
|
-
*
|
|
19
|
-
* @example
|
|
20
|
-
* ```typescript
|
|
21
|
-
* // Después de autenticarte con tu backend (ej: Cognito)
|
|
22
|
-
* @Component({...})
|
|
23
|
-
* export class LoginComponent {
|
|
24
|
-
* private authService = inject(AuthService); // Tu servicio de auth
|
|
25
|
-
* private firebase = inject(FirebaseService); // Este servicio
|
|
26
|
-
*
|
|
27
|
-
* async login(email: string, password: string) {
|
|
28
|
-
* // 1. Autenticar con tu backend
|
|
29
|
-
* const response = await this.authService.login(email, password);
|
|
30
|
-
*
|
|
31
|
-
* // 2. El backend devuelve un Firebase Custom Token
|
|
32
|
-
* if (response.firebaseToken) {
|
|
33
|
-
* await this.firebase.signInWithCustomToken(response.firebaseToken);
|
|
34
|
-
* }
|
|
35
|
-
*
|
|
36
|
-
* // Ahora el usuario puede acceder a Firestore, Storage, etc.
|
|
37
|
-
* }
|
|
38
|
-
*
|
|
39
|
-
* async logout() {
|
|
40
|
-
* await this.authService.logout();
|
|
41
|
-
* await this.firebase.signOut();
|
|
42
|
-
* }
|
|
43
|
-
* }
|
|
44
|
-
* ```
|
|
45
|
-
*/
|
|
46
|
-
export class FirebaseService {
|
|
47
|
-
constructor() {
|
|
48
|
-
this.auth = inject(Auth);
|
|
49
|
-
this.config = inject(VALTECH_FIREBASE_CONFIG);
|
|
50
|
-
/** Estado interno de la sesión */
|
|
51
|
-
this.sessionState = new BehaviorSubject({
|
|
52
|
-
user: null,
|
|
53
|
-
isAuthenticated: false,
|
|
54
|
-
isLoading: true,
|
|
55
|
-
error: null,
|
|
56
|
-
});
|
|
57
|
-
/** Estado actual de la sesión como Observable */
|
|
58
|
-
this.state$ = this.sessionState.asObservable();
|
|
59
|
-
/** Usuario actual de Firebase como Observable */
|
|
60
|
-
this.user$ = authState(this.auth).pipe(map((user) => (user ? this.mapUser(user) : null)), distinctUntilChanged((a, b) => a?.uid === b?.uid));
|
|
61
|
-
/** Indica si el usuario está autenticado en Firebase */
|
|
62
|
-
this.isAuthenticated$ = this.user$.pipe(map((user) => !!user), distinctUntilChanged());
|
|
63
|
-
// Escuchar cambios en el estado de autenticación
|
|
64
|
-
authState(this.auth).subscribe({
|
|
65
|
-
next: (user) => {
|
|
66
|
-
this.sessionState.next({
|
|
67
|
-
user: user ? this.mapUser(user) : null,
|
|
68
|
-
isAuthenticated: !!user,
|
|
69
|
-
isLoading: false,
|
|
70
|
-
error: null,
|
|
71
|
-
});
|
|
72
|
-
},
|
|
73
|
-
error: (error) => {
|
|
74
|
-
this.sessionState.next({
|
|
75
|
-
user: null,
|
|
76
|
-
isAuthenticated: false,
|
|
77
|
-
isLoading: false,
|
|
78
|
-
error,
|
|
79
|
-
});
|
|
80
|
-
},
|
|
81
|
-
});
|
|
82
|
-
}
|
|
83
|
-
// ===========================================================================
|
|
84
|
-
// AUTENTICACIÓN
|
|
85
|
-
// ===========================================================================
|
|
86
|
-
/**
|
|
87
|
-
* Autentica al usuario con un Custom Token generado por el backend.
|
|
88
|
-
*
|
|
89
|
-
* @param token - Firebase Custom Token generado por tu backend
|
|
90
|
-
* @returns UserCredential con la información del usuario
|
|
91
|
-
* @throws Error si el token es inválido o expiró
|
|
92
|
-
*
|
|
93
|
-
* @example
|
|
94
|
-
* ```typescript
|
|
95
|
-
* // Después de login exitoso con tu backend
|
|
96
|
-
* const { firebaseToken } = await backendAuth.login(email, password);
|
|
97
|
-
* await firebaseService.signInWithCustomToken(firebaseToken);
|
|
98
|
-
* ```
|
|
99
|
-
*/
|
|
100
|
-
async signInWithCustomToken(token) {
|
|
101
|
-
try {
|
|
102
|
-
const credential = await signInWithCustomToken(this.auth, token);
|
|
103
|
-
return credential;
|
|
104
|
-
}
|
|
105
|
-
catch (error) {
|
|
106
|
-
const message = this.getErrorMessage(error);
|
|
107
|
-
throw new Error(message);
|
|
108
|
-
}
|
|
109
|
-
}
|
|
110
|
-
/**
|
|
111
|
-
* Cierra la sesión de Firebase.
|
|
112
|
-
* Llamar junto con el logout de tu sistema de autenticación principal.
|
|
113
|
-
*
|
|
114
|
-
* @example
|
|
115
|
-
* ```typescript
|
|
116
|
-
* async logout() {
|
|
117
|
-
* await this.backendAuth.logout(); // Tu auth
|
|
118
|
-
* await this.firebase.signOut(); // Firebase
|
|
119
|
-
* }
|
|
120
|
-
* ```
|
|
121
|
-
*/
|
|
122
|
-
async signOut() {
|
|
123
|
-
try {
|
|
124
|
-
await signOut(this.auth);
|
|
125
|
-
}
|
|
126
|
-
catch (error) {
|
|
127
|
-
const message = this.getErrorMessage(error);
|
|
128
|
-
throw new Error(message);
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
// ===========================================================================
|
|
132
|
-
// GETTERS SÍNCRONOS
|
|
133
|
-
// ===========================================================================
|
|
134
|
-
/**
|
|
135
|
-
* Obtiene el usuario actual de Firebase (síncrono).
|
|
136
|
-
* Retorna null si no hay usuario autenticado.
|
|
137
|
-
*/
|
|
138
|
-
get currentUser() {
|
|
139
|
-
const user = this.auth.currentUser;
|
|
140
|
-
return user ? this.mapUser(user) : null;
|
|
141
|
-
}
|
|
142
|
-
/**
|
|
143
|
-
* Obtiene el UID del usuario actual.
|
|
144
|
-
* Retorna null si no hay usuario autenticado.
|
|
145
|
-
*/
|
|
146
|
-
get uid() {
|
|
147
|
-
return this.auth.currentUser?.uid ?? null;
|
|
148
|
-
}
|
|
149
|
-
/**
|
|
150
|
-
* Indica si hay un usuario autenticado actualmente.
|
|
151
|
-
*/
|
|
152
|
-
get isAuthenticated() {
|
|
153
|
-
return !!this.auth.currentUser;
|
|
154
|
-
}
|
|
155
|
-
// ===========================================================================
|
|
156
|
-
// TOKENS
|
|
157
|
-
// ===========================================================================
|
|
158
|
-
/**
|
|
159
|
-
* Obtiene el ID Token de Firebase para el usuario actual.
|
|
160
|
-
* Útil para validar el usuario en tu backend.
|
|
161
|
-
*
|
|
162
|
-
* @param forceRefresh - Si true, fuerza la renovación del token
|
|
163
|
-
* @returns ID Token o null si no hay usuario
|
|
164
|
-
*/
|
|
165
|
-
async getIdToken(forceRefresh = false) {
|
|
166
|
-
const user = this.auth.currentUser;
|
|
167
|
-
if (!user)
|
|
168
|
-
return null;
|
|
169
|
-
try {
|
|
170
|
-
return await user.getIdToken(forceRefresh);
|
|
171
|
-
}
|
|
172
|
-
catch {
|
|
173
|
-
return null;
|
|
174
|
-
}
|
|
175
|
-
}
|
|
176
|
-
/**
|
|
177
|
-
* Obtiene los claims personalizados del token del usuario.
|
|
178
|
-
* Los claims son establecidos por tu backend al crear el Custom Token.
|
|
179
|
-
*
|
|
180
|
-
* @returns Objeto con los claims o vacío si no hay usuario
|
|
181
|
-
*/
|
|
182
|
-
async getClaims() {
|
|
183
|
-
const user = this.auth.currentUser;
|
|
184
|
-
if (!user)
|
|
185
|
-
return {};
|
|
186
|
-
try {
|
|
187
|
-
const result = await user.getIdTokenResult();
|
|
188
|
-
return result.claims;
|
|
189
|
-
}
|
|
190
|
-
catch {
|
|
191
|
-
return {};
|
|
192
|
-
}
|
|
193
|
-
}
|
|
194
|
-
/**
|
|
195
|
-
* Verifica si el usuario tiene un rol específico.
|
|
196
|
-
* El rol debe estar definido en los claims del Custom Token.
|
|
197
|
-
*
|
|
198
|
-
* @param role - Nombre del rol a verificar
|
|
199
|
-
* @returns true si el usuario tiene el rol
|
|
200
|
-
*/
|
|
201
|
-
async hasRole(role) {
|
|
202
|
-
const claims = await this.getClaims();
|
|
203
|
-
return claims['role'] === role || (Array.isArray(claims['roles']) && claims['roles'].includes(role));
|
|
204
|
-
}
|
|
205
|
-
// ===========================================================================
|
|
206
|
-
// UTILIDADES
|
|
207
|
-
// ===========================================================================
|
|
208
|
-
/**
|
|
209
|
-
* Espera a que el estado de autenticación esté determinado.
|
|
210
|
-
* Útil en guards o al inicializar la app.
|
|
211
|
-
*
|
|
212
|
-
* @returns Usuario actual o null
|
|
213
|
-
*/
|
|
214
|
-
waitForAuth() {
|
|
215
|
-
return new Promise((resolve) => {
|
|
216
|
-
const subscription = this.state$.subscribe((state) => {
|
|
217
|
-
if (!state.isLoading) {
|
|
218
|
-
subscription.unsubscribe();
|
|
219
|
-
resolve(state.user);
|
|
220
|
-
}
|
|
221
|
-
});
|
|
222
|
-
});
|
|
223
|
-
}
|
|
224
|
-
/**
|
|
225
|
-
* Obtiene la configuración actual de Firebase.
|
|
226
|
-
*/
|
|
227
|
-
getConfig() {
|
|
228
|
-
return this.config;
|
|
229
|
-
}
|
|
230
|
-
/**
|
|
231
|
-
* Indica si los emuladores están habilitados.
|
|
232
|
-
*/
|
|
233
|
-
isUsingEmulators() {
|
|
234
|
-
return !!(this.config.emulator?.firestore || this.config.emulator?.auth || this.config.emulator?.storage);
|
|
235
|
-
}
|
|
236
|
-
// ===========================================================================
|
|
237
|
-
// MÉTODOS PRIVADOS
|
|
238
|
-
// ===========================================================================
|
|
239
|
-
/**
|
|
240
|
-
* Mapea un User de Firebase a nuestra interface FirebaseUser
|
|
241
|
-
*/
|
|
242
|
-
mapUser(user) {
|
|
243
|
-
return {
|
|
244
|
-
uid: user.uid,
|
|
245
|
-
email: user.email,
|
|
246
|
-
displayName: user.displayName,
|
|
247
|
-
photoURL: user.photoURL,
|
|
248
|
-
emailVerified: user.emailVerified,
|
|
249
|
-
isAnonymous: user.isAnonymous,
|
|
250
|
-
providerId: user.providerId,
|
|
251
|
-
};
|
|
252
|
-
}
|
|
253
|
-
/**
|
|
254
|
-
* Convierte errores de Firebase a mensajes en español
|
|
255
|
-
*/
|
|
256
|
-
getErrorMessage(error) {
|
|
257
|
-
if (error instanceof Error) {
|
|
258
|
-
const code = error.code;
|
|
259
|
-
switch (code) {
|
|
260
|
-
case 'auth/invalid-custom-token':
|
|
261
|
-
return 'Token de autenticación inválido';
|
|
262
|
-
case 'auth/custom-token-mismatch':
|
|
263
|
-
return 'El token no corresponde a este proyecto';
|
|
264
|
-
case 'auth/network-request-failed':
|
|
265
|
-
return 'Error de conexión. Verifica tu conexión a internet';
|
|
266
|
-
case 'auth/too-many-requests':
|
|
267
|
-
return 'Demasiados intentos. Intenta de nuevo más tarde';
|
|
268
|
-
case 'auth/user-disabled':
|
|
269
|
-
return 'Esta cuenta ha sido deshabilitada';
|
|
270
|
-
case 'auth/user-not-found':
|
|
271
|
-
return 'Usuario no encontrado';
|
|
272
|
-
default:
|
|
273
|
-
return error.message || 'Error de autenticación desconocido';
|
|
274
|
-
}
|
|
275
|
-
}
|
|
276
|
-
return 'Error de autenticación desconocido';
|
|
277
|
-
}
|
|
278
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirebaseService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
279
|
-
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirebaseService, providedIn: 'root' }); }
|
|
280
|
-
}
|
|
281
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirebaseService, decorators: [{
|
|
282
|
-
type: Injectable,
|
|
283
|
-
args: [{ providedIn: 'root' }]
|
|
284
|
-
}], ctorParameters: () => [] });
|
|
285
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"firebase.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/firebase.service.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AACnD,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,qBAAqB,EAAE,OAAO,EAAwB,MAAM,oBAAoB,CAAC;AAC3G,OAAO,EAAE,eAAe,EAAE,oBAAoB,EAAE,GAAG,EAAc,MAAM,MAAM,CAAC;AAE9E,OAAO,EAAE,uBAAuB,EAAE,MAAM,UAAU,CAAC;;AAGnD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgCG;AAEH,MAAM,OAAO,eAAe;IA2B1B;QA1BQ,SAAI,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC;QACpB,WAAM,GAAG,MAAM,CAAC,uBAAuB,CAAC,CAAC;QAEjD,kCAAkC;QAC1B,iBAAY,GAAG,IAAI,eAAe,CAAe;YACvD,IAAI,EAAE,IAAI;YACV,eAAe,EAAE,KAAK;YACtB,SAAS,EAAE,IAAI;YACf,KAAK,EAAE,IAAI;SACZ,CAAC,CAAC;QAEH,iDAAiD;QACxC,WAAM,GAA6B,IAAI,CAAC,YAAY,CAAC,YAAY,EAAE,CAAC;QAE7E,iDAAiD;QACxC,UAAK,GAAoC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,CACzE,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,EACjD,oBAAoB,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,KAAK,CAAC,EAAE,GAAG,CAAC,CAClD,CAAC;QAEF,wDAAwD;QAC/C,qBAAgB,GAAwB,IAAI,CAAC,KAAK,CAAC,IAAI,CAC9D,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,EACrB,oBAAoB,EAAE,CACvB,CAAC;QAGA,iDAAiD;QACjD,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC;YAC7B,IAAI,EAAE,CAAC,IAAI,EAAE,EAAE;gBACb,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;oBACrB,IAAI,EAAE,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI;oBACtC,eAAe,EAAE,CAAC,CAAC,IAAI;oBACvB,SAAS,EAAE,KAAK;oBAChB,KAAK,EAAE,IAAI;iBACZ,CAAC,CAAC;YACL,CAAC;YACD,KAAK,EAAE,CAAC,KAAK,EAAE,EAAE;gBACf,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;oBACrB,IAAI,EAAE,IAAI;oBACV,eAAe,EAAE,KAAK;oBACtB,SAAS,EAAE,KAAK;oBAChB,KAAK;iBACN,CAAC,CAAC;YACL,CAAC;SACF,CAAC,CAAC;IACL,CAAC;IAED,8EAA8E;IAC9E,gBAAgB;IAChB,8EAA8E;IAE9E;;;;;;;;;;;;;OAaG;IACH,KAAK,CAAC,qBAAqB,CAAC,KAAa;QACvC,IAAI,CAAC;YACH,MAAM,UAAU,GAAG,MAAM,qBAAqB,CAAC,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;YACjE,OAAO,UAAU,CAAC;QACpB,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,OAAO,GAAG,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;YAC5C,MAAM,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC;QAC3B,CAAC;IACH,CAAC;IAED;;;;;;;;;;;OAWG;IACH,KAAK,CAAC,OAAO;QACX,IAAI,CAAC;YACH,MAAM,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC3B,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,OAAO,GAAG,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;YAC5C,MAAM,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC;QAC3B,CAAC;IACH,CAAC;IAED,8EAA8E;IAC9E,oBAAoB;IACpB,8EAA8E;IAE9E;;;OAGG;IACH,IAAI,WAAW;QACb,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC;QACnC,OAAO,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IAC1C,CAAC;IAED;;;OAGG;IACH,IAAI,GAAG;QACL,OAAO,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,GAAG,IAAI,IAAI,CAAC;IAC5C,CAAC;IAED;;OAEG;IACH,IAAI,eAAe;QACjB,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC;IACjC,CAAC;IAED,8EAA8E;IAC9E,SAAS;IACT,8EAA8E;IAE9E;;;;;;OAMG;IACH,KAAK,CAAC,UAAU,CAAC,YAAY,GAAG,KAAK;QACnC,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC;QACnC,IAAI,CAAC,IAAI;YAAE,OAAO,IAAI,CAAC;QAEvB,IAAI,CAAC;YACH,OAAO,MAAM,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;QAC7C,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,SAAS;QACb,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC;QACnC,IAAI,CAAC,IAAI;YAAE,OAAO,EAAE,CAAC;QAErB,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC7C,OAAO,MAAM,CAAC,MAAM,CAAC;QACvB,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,EAAE,CAAC;QACZ,CAAC;IACH,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,OAAO,CAAC,IAAY;QACxB,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,EAAE,CAAC;QACtC,OAAO,MAAM,CAAC,MAAM,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC;IACvG,CAAC;IAED,8EAA8E;IAC9E,aAAa;IACb,8EAA8E;IAE9E;;;;;OAKG;IACH,WAAW;QACT,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7B,MAAM,YAAY,GAAG,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE;gBACnD,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;oBACrB,YAAY,CAAC,WAAW,EAAE,CAAC;oBAC3B,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;gBACtB,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,SAAS;QACP,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED;;OAEG;IACH,gBAAgB;QACd,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,SAAS,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,IAAI,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;IAC5G,CAAC;IAED,8EAA8E;IAC9E,mBAAmB;IACnB,8EAA8E;IAE9E;;OAEG;IACK,OAAO,CAAC,IAAU;QACxB,OAAO;YACL,GAAG,EAAE,IAAI,CAAC,GAAG;YACb,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,UAAU,EAAE,IAAI,CAAC,UAAU;SAC5B,CAAC;IACJ,CAAC;IAED;;OAEG;IACK,eAAe,CAAC,KAAc;QACpC,IAAI,KAAK,YAAY,KAAK,EAAE,CAAC;YAC3B,MAAM,IAAI,GAAI,KAA2B,CAAC,IAAI,CAAC;YAE/C,QAAQ,IAAI,EAAE,CAAC;gBACb,KAAK,2BAA2B;oBAC9B,OAAO,iCAAiC,CAAC;gBAC3C,KAAK,4BAA4B;oBAC/B,OAAO,yCAAyC,CAAC;gBACnD,KAAK,6BAA6B;oBAChC,OAAO,oDAAoD,CAAC;gBAC9D,KAAK,wBAAwB;oBAC3B,OAAO,iDAAiD,CAAC;gBAC3D,KAAK,oBAAoB;oBACvB,OAAO,mCAAmC,CAAC;gBAC7C,KAAK,qBAAqB;oBACxB,OAAO,uBAAuB,CAAC;gBACjC;oBACE,OAAO,KAAK,CAAC,OAAO,IAAI,oCAAoC,CAAC;YACjE,CAAC;QACH,CAAC;QAED,OAAO,oCAAoC,CAAC;IAC9C,CAAC;+GAlQU,eAAe;mHAAf,eAAe,cADF,MAAM;;4FACnB,eAAe;kBAD3B,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE","sourcesContent":["/**\n * Firebase Service\n *\n * Servicio principal para la autenticación con Firebase usando Custom Tokens.\n * Permite que usuarios autenticados con tu backend (Cognito, etc.) accedan\n * a servicios de Firebase (Firestore, Storage, FCM) de manera segura.\n */\n\nimport { inject, Injectable } from '@angular/core';\nimport { Auth, authState, signInWithCustomToken, signOut, User, UserCredential } from '@angular/fire/auth';\nimport { BehaviorSubject, distinctUntilChanged, map, Observable } from 'rxjs';\n\nimport { VALTECH_FIREBASE_CONFIG } from './config';\nimport { FirebaseUser, SessionState, ValtechFirebaseConfig } from './types';\n\n/**\n * Servicio de autenticación de Firebase.\n *\n * Este servicio NO maneja el login de usuarios directamente.\n * En su lugar, trabaja con Custom Tokens generados por tu backend.\n *\n * @example\n * ```typescript\n * // Después de autenticarte con tu backend (ej: Cognito)\n * @Component({...})\n * export class LoginComponent {\n *   private authService = inject(AuthService);     // Tu servicio de auth\n *   private firebase = inject(FirebaseService);    // Este servicio\n *\n *   async login(email: string, password: string) {\n *     // 1. Autenticar con tu backend\n *     const response = await this.authService.login(email, password);\n *\n *     // 2. El backend devuelve un Firebase Custom Token\n *     if (response.firebaseToken) {\n *       await this.firebase.signInWithCustomToken(response.firebaseToken);\n *     }\n *\n *     // Ahora el usuario puede acceder a Firestore, Storage, etc.\n *   }\n *\n *   async logout() {\n *     await this.authService.logout();\n *     await this.firebase.signOut();\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class FirebaseService {\n  private auth = inject(Auth);\n  private config = inject(VALTECH_FIREBASE_CONFIG);\n\n  /** Estado interno de la sesión */\n  private sessionState = new BehaviorSubject<SessionState>({\n    user: null,\n    isAuthenticated: false,\n    isLoading: true,\n    error: null,\n  });\n\n  /** Estado actual de la sesión como Observable */\n  readonly state$: Observable<SessionState> = this.sessionState.asObservable();\n\n  /** Usuario actual de Firebase como Observable */\n  readonly user$: Observable<FirebaseUser | null> = authState(this.auth).pipe(\n    map((user) => (user ? this.mapUser(user) : null)),\n    distinctUntilChanged((a, b) => a?.uid === b?.uid)\n  );\n\n  /** Indica si el usuario está autenticado en Firebase */\n  readonly isAuthenticated$: Observable<boolean> = this.user$.pipe(\n    map((user) => !!user),\n    distinctUntilChanged()\n  );\n\n  constructor() {\n    // Escuchar cambios en el estado de autenticación\n    authState(this.auth).subscribe({\n      next: (user) => {\n        this.sessionState.next({\n          user: user ? this.mapUser(user) : null,\n          isAuthenticated: !!user,\n          isLoading: false,\n          error: null,\n        });\n      },\n      error: (error) => {\n        this.sessionState.next({\n          user: null,\n          isAuthenticated: false,\n          isLoading: false,\n          error,\n        });\n      },\n    });\n  }\n\n  // ===========================================================================\n  // AUTENTICACIÓN\n  // ===========================================================================\n\n  /**\n   * Autentica al usuario con un Custom Token generado por el backend.\n   *\n   * @param token - Firebase Custom Token generado por tu backend\n   * @returns UserCredential con la información del usuario\n   * @throws Error si el token es inválido o expiró\n   *\n   * @example\n   * ```typescript\n   * // Después de login exitoso con tu backend\n   * const { firebaseToken } = await backendAuth.login(email, password);\n   * await firebaseService.signInWithCustomToken(firebaseToken);\n   * ```\n   */\n  async signInWithCustomToken(token: string): Promise<UserCredential> {\n    try {\n      const credential = await signInWithCustomToken(this.auth, token);\n      return credential;\n    } catch (error) {\n      const message = this.getErrorMessage(error);\n      throw new Error(message);\n    }\n  }\n\n  /**\n   * Cierra la sesión de Firebase.\n   * Llamar junto con el logout de tu sistema de autenticación principal.\n   *\n   * @example\n   * ```typescript\n   * async logout() {\n   *   await this.backendAuth.logout();    // Tu auth\n   *   await this.firebase.signOut();       // Firebase\n   * }\n   * ```\n   */\n  async signOut(): Promise<void> {\n    try {\n      await signOut(this.auth);\n    } catch (error) {\n      const message = this.getErrorMessage(error);\n      throw new Error(message);\n    }\n  }\n\n  // ===========================================================================\n  // GETTERS SÍNCRONOS\n  // ===========================================================================\n\n  /**\n   * Obtiene el usuario actual de Firebase (síncrono).\n   * Retorna null si no hay usuario autenticado.\n   */\n  get currentUser(): FirebaseUser | null {\n    const user = this.auth.currentUser;\n    return user ? this.mapUser(user) : null;\n  }\n\n  /**\n   * Obtiene el UID del usuario actual.\n   * Retorna null si no hay usuario autenticado.\n   */\n  get uid(): string | null {\n    return this.auth.currentUser?.uid ?? null;\n  }\n\n  /**\n   * Indica si hay un usuario autenticado actualmente.\n   */\n  get isAuthenticated(): boolean {\n    return !!this.auth.currentUser;\n  }\n\n  // ===========================================================================\n  // TOKENS\n  // ===========================================================================\n\n  /**\n   * Obtiene el ID Token de Firebase para el usuario actual.\n   * Útil para validar el usuario en tu backend.\n   *\n   * @param forceRefresh - Si true, fuerza la renovación del token\n   * @returns ID Token o null si no hay usuario\n   */\n  async getIdToken(forceRefresh = false): Promise<string | null> {\n    const user = this.auth.currentUser;\n    if (!user) return null;\n\n    try {\n      return await user.getIdToken(forceRefresh);\n    } catch {\n      return null;\n    }\n  }\n\n  /**\n   * Obtiene los claims personalizados del token del usuario.\n   * Los claims son establecidos por tu backend al crear el Custom Token.\n   *\n   * @returns Objeto con los claims o vacío si no hay usuario\n   */\n  async getClaims(): Promise<Record<string, unknown>> {\n    const user = this.auth.currentUser;\n    if (!user) return {};\n\n    try {\n      const result = await user.getIdTokenResult();\n      return result.claims;\n    } catch {\n      return {};\n    }\n  }\n\n  /**\n   * Verifica si el usuario tiene un rol específico.\n   * El rol debe estar definido en los claims del Custom Token.\n   *\n   * @param role - Nombre del rol a verificar\n   * @returns true si el usuario tiene el rol\n   */\n  async hasRole(role: string): Promise<boolean> {\n    const claims = await this.getClaims();\n    return claims['role'] === role || (Array.isArray(claims['roles']) && claims['roles'].includes(role));\n  }\n\n  // ===========================================================================\n  // UTILIDADES\n  // ===========================================================================\n\n  /**\n   * Espera a que el estado de autenticación esté determinado.\n   * Útil en guards o al inicializar la app.\n   *\n   * @returns Usuario actual o null\n   */\n  waitForAuth(): Promise<FirebaseUser | null> {\n    return new Promise((resolve) => {\n      const subscription = this.state$.subscribe((state) => {\n        if (!state.isLoading) {\n          subscription.unsubscribe();\n          resolve(state.user);\n        }\n      });\n    });\n  }\n\n  /**\n   * Obtiene la configuración actual de Firebase.\n   */\n  getConfig(): ValtechFirebaseConfig {\n    return this.config;\n  }\n\n  /**\n   * Indica si los emuladores están habilitados.\n   */\n  isUsingEmulators(): boolean {\n    return !!(this.config.emulator?.firestore || this.config.emulator?.auth || this.config.emulator?.storage);\n  }\n\n  // ===========================================================================\n  // MÉTODOS PRIVADOS\n  // ===========================================================================\n\n  /**\n   * Mapea un User de Firebase a nuestra interface FirebaseUser\n   */\n  private mapUser(user: User): FirebaseUser {\n    return {\n      uid: user.uid,\n      email: user.email,\n      displayName: user.displayName,\n      photoURL: user.photoURL,\n      emailVerified: user.emailVerified,\n      isAnonymous: user.isAnonymous,\n      providerId: user.providerId,\n    };\n  }\n\n  /**\n   * Convierte errores de Firebase a mensajes en español\n   */\n  private getErrorMessage(error: unknown): string {\n    if (error instanceof Error) {\n      const code = (error as { code?: string }).code;\n\n      switch (code) {\n        case 'auth/invalid-custom-token':\n          return 'Token de autenticación inválido';\n        case 'auth/custom-token-mismatch':\n          return 'El token no corresponde a este proyecto';\n        case 'auth/network-request-failed':\n          return 'Error de conexión. Verifica tu conexión a internet';\n        case 'auth/too-many-requests':\n          return 'Demasiados intentos. Intenta de nuevo más tarde';\n        case 'auth/user-disabled':\n          return 'Esta cuenta ha sido deshabilitada';\n        case 'auth/user-not-found':\n          return 'Usuario no encontrado';\n        default:\n          return error.message || 'Error de autenticación desconocido';\n      }\n    }\n\n    return 'Error de autenticación desconocido';\n  }\n}\n"]}
|