valtech-components 2.0.463 → 2.0.465
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.service.mjs +17 -12
- package/esm2022/lib/services/auth/interceptor.mjs +24 -11
- package/esm2022/lib/services/firebase/firebase.service.mjs +18 -9
- package/fesm2022/valtech-components.mjs +56 -29
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/services/firebase/firebase.service.d.ts +8 -4
- package/package.json +1 -1
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { Injectable, Inject } from '@angular/core';
|
|
2
|
-
import { throwError, of, firstValueFrom } from 'rxjs';
|
|
3
|
-
import { tap, catchError } from 'rxjs/operators';
|
|
2
|
+
import { throwError, of, from, firstValueFrom } from 'rxjs';
|
|
3
|
+
import { tap, catchError, switchMap, map } from 'rxjs/operators';
|
|
4
4
|
import { VALTECH_AUTH_CONFIG } from './config';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
6
6
|
import * as i1 from "@angular/common/http";
|
|
@@ -282,16 +282,21 @@ export class AuthService {
|
|
|
282
282
|
switchOrg(organizationId) {
|
|
283
283
|
return this.http
|
|
284
284
|
.post(`${this.baseUrl}/switch-org`, { organizationId })
|
|
285
|
-
.pipe(
|
|
285
|
+
.pipe(switchMap(response => {
|
|
286
286
|
// Re-autenticar en Firebase con el nuevo token
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
287
|
+
// Usar switchMap + from para esperar a que Firebase complete
|
|
288
|
+
const firebaseAuth$ = response.firebaseToken
|
|
289
|
+
? from(this.signInWithFirebase(response.firebaseToken))
|
|
290
|
+
: of(undefined);
|
|
291
|
+
return firebaseAuth$.pipe(tap(() => {
|
|
292
|
+
// Notificar cambio a otras pestañas
|
|
293
|
+
this.syncService.broadcast({
|
|
294
|
+
type: 'ORG_SWITCH',
|
|
295
|
+
payload: { activeOrg: response.activeOrg },
|
|
296
|
+
});
|
|
297
|
+
}),
|
|
298
|
+
// Retornar response original después de que Firebase complete
|
|
299
|
+
map(() => response));
|
|
295
300
|
}), catchError(error => this.handleAuthError(error)));
|
|
296
301
|
}
|
|
297
302
|
// =============================================
|
|
@@ -529,4 +534,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
529
534
|
type: Inject,
|
|
530
535
|
args: [VALTECH_AUTH_CONFIG]
|
|
531
536
|
}] }, { type: i1.HttpClient }, { type: i2.Router }, { type: i3.AuthStateService }, { type: i4.TokenService }, { type: i5.AuthStorageService }, { type: i6.AuthSyncService }, { type: i7.FirebaseService }] });
|
|
532
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/auth.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAuB,MAAM,eAAe,CAAC;AAGxE,OAAO,EAAc,UAAU,EAAE,EAAE,EAAE,cAAc,EAAgB,MAAM,MAAM,CAAC;AAChF,OAAO,EAAE,GAAG,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAEjD,OAAO,EAAE,mBAAmB,EAAE,MAAM,UAAU,CAAC;;;;;;;;;AAoC/C;;;;;;;;;;;;;;;;;;;;;GAqBG;AAEH,MAAM,OAAO,WAAW;IAKtB,YACuC,MAAyB,EACtD,IAAgB,EAChB,MAAc,EACd,YAA8B,EAC9B,YAA0B,EAC1B,cAAkC,EAClC,WAA4B,EAC5B,eAAgC;QAPH,WAAM,GAAN,MAAM,CAAmB;QACtD,SAAI,GAAJ,IAAI,CAAY;QAChB,WAAM,GAAN,MAAM,CAAQ;QACd,iBAAY,GAAZ,YAAY,CAAkB;QAC9B,iBAAY,GAAZ,YAAY,CAAc;QAC1B,mBAAc,GAAd,cAAc,CAAoB;QAClC,gBAAW,GAAX,WAAW,CAAiB;QAC5B,oBAAe,GAAf,eAAe,CAAiB;QAZ1C,+BAA+B;QACvB,mBAAc,GAAyC,IAAI,CAAC;QAC5D,qBAAgB,GAAwB,IAAI,CAAC;QAarD,gDAAgD;QAChD,oCAAoC;QACpC,gDAAgD;QAEhD,uCAAuC;QAC9B,UAAK,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC;QAEzC,+BAA+B;QACtB,oBAAe,GAAG,IAAI,CAAC,YAAY,CAAC,eAAe,CAAC;QAE7D,sBAAsB;QACb,cAAS,GAAG,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC;QAEjD,8BAA8B;QACrB,SAAI,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;QAEvC,sBAAsB;QACb,gBAAW,GAAG,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC;QAErD,wBAAwB;QACf,UAAK,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC;QAEzC,2BAA2B;QAClB,gBAAW,GAAG,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC;QAErD,6BAA6B;QACpB,iBAAY,GAAG,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC;QAEvD,8BAA8B;QACrB,eAAU,GAAG,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC;QAEnD,mBAAmB;QACV,UAAK,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC;IAlCtC,CAAC;IAoCJ,gDAAgD;IAChD,iBAAiB;IACjB,gDAAgD;IAEhD;;;OAGG;IACH,KAAK,CAAC,UAAU;QACd,iCAAiC;QACjC,MAAM,WAAW,GAAG,IAAI,CAAC,cAAc,CAAC,SAAS,EAAE,CAAC;QAEpD,IAAI,WAAW,CAAC,WAAW,EAAE,CAAC;YAC5B,kCAAkC;YAClC,IAAI,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,WAAW,CAAC,WAAW,CAAC,EAAE,CAAC;gBAC5D,IAAI,CAAC,YAAY,CAAC,kBAAkB,CAAC,WAAW,CAAC,CAAC;gBAElD,yBAAyB;gBACzB,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,WAAW,CAAC,WAAW,CAAC,CAAC;gBACrE,IAAI,MAAM,EAAE,CAAC;oBACX,IAAI,CAAC,YAAY,CAAC,cAAc,CAAC,MAAM,CAAC,GAAG,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC;gBAC7D,CAAC;gBAED,yCAAyC;gBACzC,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,CAAC;iBAAM,IAAI,WAAW,CAAC,YAAY,EAAE,CAAC;gBACpC,gEAAgE;gBAChE,IAAI,CAAC;oBACH,MAAM,cAAc,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,CAAC;gBAClD,CAAC;gBAAC,MAAM,CAAC;oBACP,IAAI,CAAC,eAAe,EAAE,CAAC;oBACvB,IAAI,CAAC,UAAU,EAAE,CAAC;gBACpB,CAAC;YACH,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,eAAe,EAAE,CAAC;gBACvB,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,CAAC;QACH,CAAC;QAED,2CAA2C;QAC3C,IAAI,IAAI,CAAC,MAAM,CAAC,aAAa,EAAE,CAAC;YAC9B,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE,CAAC;YACzB,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE,CAClE,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAC5B,CAAC;QACJ,CAAC;QAED,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;IACtC,CAAC;IAED,WAAW;QACT,IAAI,CAAC,gBAAgB,EAAE,CAAC;QACxB,IAAI,CAAC,gBAAgB,EAAE,WAAW,EAAE,CAAC;IACvC,CAAC;IAED,gDAAgD;IAChD,gBAAgB;IAChB,gDAAgD;IAEhD;;OAEG;IACH,MAAM,CAAC,OAAsB;QAC3B,IAAI,CAAC,YAAY,CAAC,UAAU,EAAE,CAAC;QAE/B,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAiB,GAAG,IAAI,CAAC,OAAO,SAAS,EAAE,OAAO,CAAC,CAAC,IAAI,CAC3E,GAAG,CAAC,QAAQ,CAAC,EAAE;YACb,IAAI,QAAQ,CAAC,WAAW,EAAE,CAAC;gBACzB,0CAA0C;gBAC1C,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC;oBAC9B,QAAQ,EAAE,IAAI;oBACd,QAAQ,EAAE,QAAQ,CAAC,QAAS;oBAC5B,MAAM,EAAE,QAAQ,CAAC,SAAU;iBAC5B,CAAC,CAAC;YACL,CAAC;iBAAM,IAAI,QAAQ,CAAC,WAAW,EAAE,CAAC;gBAChC,wBAAwB;gBACxB,IAAI,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC;YACtC,CAAC;QACH,CAAC,CAAC,EACF,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CACjD,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,MAAM,CAAC,OAAsB;QAC3B,IAAI,CAAC,YAAY,CAAC,UAAU,EAAE,CAAC;QAE/B,OAAO,IAAI,CAAC,IAAI;aACb,IAAI,CAAiB,GAAG,IAAI,CAAC,OAAO,SAAS,EAAE,OAAO,CAAC;aACvD,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC5D,CAAC;IAED;;;OAGG;IACH,WAAW,CAAC,OAA2B;QACrC,IAAI,CAAC,YAAY,CAAC,UAAU,EAAE,CAAC;QAE/B,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAsB,GAAG,IAAI,CAAC,OAAO,eAAe,EAAE,OAAO,CAAC,CAAC,IAAI,CACtF,GAAG,CAAC,QAAQ,CAAC,EAAE;YACb,IAAI,QAAQ,CAAC,QAAQ,IAAI,QAAQ,CAAC,WAAW,EAAE,CAAC;gBAC9C,iDAAiD;gBACjD,IAAI,CAAC,oBAAoB,CAAC,QAAqC,CAAC,CAAC;YACnE,CAAC;QACH,CAAC,CAAC,EACF,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CACjD,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,UAAU,CAAC,OAA0B;QACnC,OAAO,IAAI,CAAC,IAAI;aACb,IAAI,CAAqB,GAAG,IAAI,CAAC,OAAO,cAAc,EAAE,OAAO,CAAC;aAChE,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC5D,CAAC;IAED;;OAEG;IACH,SAAS,CAAC,IAAY;QACpB,MAAM,QAAQ,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QACnC,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;YACvB,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,CAAC;gBACvB,IAAI,EAAE,iBAAiB;gBACvB,OAAO,EAAE,mCAAmC;aAC7C,CAAC,CAAC,CAAC;QACN,CAAC;QAED,OAAO,IAAI,CAAC,IAAI;aACb,IAAI,CAAoB,GAAG,IAAI,CAAC,OAAO,aAAa,EAAE;YACrD,QAAQ,EAAE,QAAQ,CAAC,QAAQ;YAC3B,IAAI;SACL,CAAC;aACD,IAAI,CACH,GAAG,CAAC,QAAQ,CAAC,EAAE;YACb,IAAI,CAAC,YAAY,CAAC,eAAe,EAAE,CAAC;YACpC,IAAI,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC;QACtC,CAAC,CAAC,EACF,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CACjD,CAAC;IACN,CAAC;IAED;;OAEG;IACH,kBAAkB;QAChB,MAAM,YAAY,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,YAAY,CAAC;QAC/C,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,CAAC;gBACvB,IAAI,EAAE,kBAAkB;gBACxB,OAAO,EAAE,0BAA0B;aACpC,CAAC,CAAC,CAAC;QACN,CAAC;QAED,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAkB,GAAG,IAAI,CAAC,OAAO,UAAU,EAAE,EAAE,YAAY,EAAE,CAAC,CAAC,IAAI,CACtF,GAAG,CAAC,QAAQ,CAAC,EAAE;YACb,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,QAAQ,CAAC,SAAS,GAAG,IAAI,CAAC;YACzD,IAAI,CAAC,YAAY,CAAC,iBAAiB,CAAC,QAAQ,CAAC,WAAW,EAAE,QAAQ,CAAC,SAAS,CAAC,CAAC;YAC9E,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,QAAQ,CAAC,WAAW,EAAE,SAAS,CAAC,CAAC;YACrE,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC;gBACzB,IAAI,EAAE,eAAe;gBACrB,OAAO,EAAE,EAAE,WAAW,EAAE,QAAQ,CAAC,WAAW,EAAE,SAAS,EAAE;aAC1D,CAAC,CAAC;QACL,CAAC,CAAC,EACF,UAAU,CAAC,KAAK,CAAC,EAAE;YACjB,IAAI,CAAC,MAAM,EAAE,CAAC;YACd,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC;QACjC,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,MAAM;QACJ,MAAM,YAAY,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,YAAY,CAAC;QAE/C,yCAAyC;QACzC,IAAI,YAAY,EAAE,CAAC;YACjB,IAAI,CAAC,IAAI;iBACN,IAAI,CAAC,GAAG,IAAI,CAAC,OAAO,SAAS,EAAE,EAAE,YAAY,EAAE,CAAC;iBAChD,IAAI,CAAC,UAAU,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC;iBAChC,SAAS,EAAE,CAAC;QACjB,CAAC;QAED,8CAA8C;QAC9C,IAAI,CAAC,eAAe,EAAE,CAAC;QAEvB,IAAI,CAAC,UAAU,EAAE,CAAC;QAClB,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,CAAC,CAAC;QAC/C,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC;IACjD,CAAC;IAED,gDAAgD;IAChD,kCAAkC;IAClC,gDAAgD;IAEhD;;OAEG;IACH,QAAQ,CAAC,MAAiB,EAAE,KAAc;QACxC,OAAO,IAAI,CAAC,IAAI;aACb,IAAI,CAAmB,GAAG,IAAI,CAAC,OAAO,YAAY,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC;aACtE,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC5D,CAAC;IAED;;OAEG;IACH,UAAU,CAAC,IAAY;QACrB,OAAO,IAAI,CAAC,IAAI;aACb,IAAI,CAAqB,GAAG,IAAI,CAAC,OAAO,cAAc,EAAE,EAAE,IAAI,EAAE,CAAC;aACjE,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC5D,CAAC;IAED;;OAEG;IACH,UAAU,CAAC,QAAgB;QACzB,OAAO,IAAI,CAAC,IAAI;aACb,IAAI,CAAqB,GAAG,IAAI,CAAC,OAAO,cAAc,EAAE,EAAE,QAAQ,EAAE,CAAC;aACrE,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC5D,CAAC;IAED,gDAAgD;IAChD,iBAAiB;IACjB,gDAAgD;IAEhD;;;;;;;;;;;;;OAaG;IACH,SAAS,CAAC,cAAsB;QAC9B,OAAO,IAAI,CAAC,IAAI;aACb,IAAI,CAAoB,GAAG,IAAI,CAAC,OAAO,aAAa,EAAE,EAAE,cAAc,EAAE,CAAC;aACzE,IAAI,CACH,GAAG,CAAC,KAAK,EAAC,QAAQ,EAAC,EAAE;YACnB,+CAA+C;YAC/C,IAAI,QAAQ,CAAC,aAAa,EAAE,CAAC;gBAC3B,MAAM,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC;YACxD,CAAC;YACD,oCAAoC;YACpC,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC;gBACzB,IAAI,EAAE,YAAY;gBAClB,OAAO,EAAE,EAAE,SAAS,EAAE,QAAQ,CAAC,SAAS,EAAE;aAC3C,CAAC,CAAC;QACL,CAAC,CAAC,EACF,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CACjD,CAAC;IACN,CAAC;IAED,gDAAgD;IAChD,oBAAoB;IACpB,gDAAgD;IAEhD;;;OAGG;IACH,UAAU;QACR,OAAO,IAAI,CAAC,IAAI;aACb,GAAG,CAAqB,GAAG,IAAI,CAAC,OAAO,UAAU,CAAC;aAClD,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC5D,CAAC;IAED;;OAEG;IACH,aAAa,CAAC,OAA6B;QACzC,OAAO,IAAI,CAAC,IAAI;aACb,GAAG,CAAwB,GAAG,IAAI,CAAC,OAAO,UAAU,EAAE,OAAO,CAAC;aAC9D,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC5D,CAAC;IAED,gDAAgD;IAChD,6BAA6B;IAC7B,gDAAgD;IAEhD;;;OAGG;IACH,cAAc,CAAC,OAA8B;QAC3C,OAAO,IAAI,CAAC,IAAI;aACb,IAAI,CAAyB,GAAG,IAAI,CAAC,OAAO,kBAAkB,EAAE,OAAO,CAAC;aACxE,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC5D,CAAC;IAED;;OAEG;IACH,aAAa,CAAC,OAA6B;QACzC,OAAO,IAAI,CAAC,IAAI;aACb,IAAI,CAAwB,GAAG,IAAI,CAAC,OAAO,iBAAiB,EAAE,OAAO,CAAC;aACtE,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC5D,CAAC;IAED,gDAAgD;IAChD,WAAW;IACX,gDAAgD;IAEhD;;OAEG;IACH,gBAAgB;QACd,OAAO,IAAI,CAAC,IAAI,CAAC,GAAG,CAAyB,GAAG,IAAI,CAAC,OAAO,cAAc,CAAC,CAAC,IAAI,CAC9E,GAAG,CAAC,QAAQ,CAAC,EAAE;YACb,IAAI,CAAC,YAAY,CAAC,iBAAiB,CACjC,QAAQ,CAAC,KAAK,EACd,QAAQ,CAAC,WAAW,EACpB,QAAQ,CAAC,YAAY,CACtB,CAAC;YACF,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,QAAQ,CAAC,CAAC;YAC9C,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,oBAAoB,EAAE,CAAC,CAAC;QAC7D,CAAC,CAAC,EACF,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CACjD,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,aAAa,CAAC,UAAkB;QAC9B,IAAI,IAAI,CAAC,YAAY,EAAE;YAAE,OAAO,IAAI,CAAC;QAErC,MAAM,CAAC,QAAQ,EAAE,MAAM,CAAC,GAAG,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACjD,OAAO,IAAI,CAAC,WAAW,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE;YACjC,MAAM,CAAC,SAAS,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAC1C,OAAO,CACL,CAAC,SAAS,KAAK,GAAG,IAAI,SAAS,KAAK,QAAQ,CAAC,IAAI,CAAC,OAAO,KAAK,GAAG,IAAI,OAAO,KAAK,MAAM,CAAC,CACzF,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,gBAAgB,CAAC,WAAqB;QACpC,OAAO,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;IACtD,CAAC;IAED;;OAEG;IACH,iBAAiB,CAAC,WAAqB;QACrC,OAAO,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;IACvD,CAAC;IAED;;OAEG;IACH,OAAO,CAAC,IAAY;QAClB,OAAO,IAAI,CAAC,KAAK,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,KAAK,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC;IACxE,CAAC;IAED,gDAAgD;IAChD,kBAAkB;IAClB,gDAAgD;IAEhD,IAAY,OAAO;QACjB,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC;IAC1D,CAAC;IAEO,oBAAoB,CAAC,QAA4C;QACvE,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,QAAQ,CAAC,SAAU,GAAG,IAAI,CAAC;QAC1D,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,QAAQ,CAAC,WAAY,CAAC,CAAC;QAEtE,IAAI,CAAC,YAAY,CAAC,gBAAgB,CAAC;YACjC,WAAW,EAAE,QAAQ,CAAC,WAAY;YAClC,YAAY,EAAE,QAAQ,CAAC,YAAa;YACpC,MAAM,EAAE,SAAS,EAAE,GAAG;YACtB,KAAK,EAAE,SAAS,EAAE,KAAK;YACvB,KAAK,EAAE,QAAQ,CAAC,KAAK,IAAI,EAAE;YAC3B,WAAW,EAAE,QAAQ,CAAC,WAAW,IAAI,EAAE;YACvC,YAAY,EAAE,QAAQ,CAAC,WAAW,EAAE,QAAQ,CAAC,KAAK,CAAC,IAAI,KAAK;YAC5D,SAAS;SACV,CAAC,CAAC;QAEH,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC;YAC5B,WAAW,EAAE,QAAQ,CAAC,WAAY;YAClC,YAAY,EAAE,QAAQ,CAAC,YAAa;YACpC,KAAK,EAAE,QAAQ,CAAC,KAAK,IAAI,EAAE;YAC3B,WAAW,EAAE,QAAQ,CAAC,WAAW,IAAI,EAAE;YACvC,YAAY,EAAE,QAAQ,CAAC,WAAW,EAAE,QAAQ,CAAC,KAAK,CAAC,IAAI,KAAK;YAC5D,SAAS;SACV,CAAC,CAAC;QAEH,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC,CAAC;QAE9C,2BAA2B;QAC3B,IACE,IAAI,CAAC,MAAM,CAAC,yBAAyB;YACrC,eAAe,IAAI,QAAQ;YAC3B,QAAQ,CAAC,aAAa,EACtB,CAAC;YACD,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC;QAClD,CAAC;IACH,CAAC;IAEO,UAAU;QAChB,IAAI,CAAC,gBAAgB,EAAE,CAAC;QACxB,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;QAC1B,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,CAAC;IAC9B,CAAC;IAEO,iBAAiB;QACvB,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAExB,MAAM,KAAK,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;QACxC,IAAI,CAAC,KAAK,CAAC,SAAS;YAAE,OAAO;QAE7B,MAAM,eAAe,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,mBAAmB,IAAI,EAAE,CAAC,GAAG,IAAI,CAAC;QACvE,MAAM,SAAS,GAAG,KAAK,CAAC,SAAS,GAAG,eAAe,CAAC;QACpD,MAAM,KAAK,GAAG,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAErC,IAAI,KAAK,GAAG,CAAC,EAAE,CAAC;YACd,IAAI,CAAC,cAAc,GAAG,UAAU,CAAC,GAAG,EAAE;gBACpC,IAAI,CAAC,kBAAkB,EAAE,CAAC,SAAS,CAAC;oBAClC,KAAK,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE;iBAC3B,CAAC,CAAC;YACL,CAAC,EAAE,KAAK,CAAC,CAAC;QACZ,CAAC;aAAM,IAAI,KAAK,CAAC,YAAY,EAAE,CAAC;YAC9B,+CAA+C;YAC/C,IAAI,CAAC,kBAAkB,EAAE,CAAC,SAAS,CAAC;gBAClC,KAAK,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE;aAC3B,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAEO,gBAAgB;QACtB,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;YACxB,YAAY,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;YAClC,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;QAC7B,CAAC;IACH,CAAC;IAEO,eAAe,CAAC,KAAoB;QAC1C,QAAQ,KAAK,CAAC,IAAI,EAAE,CAAC;YACnB,KAAK,OAAO,CAAC;YACb,KAAK,eAAe,CAAC,CAAC,CAAC;gBACrB,gCAAgC;gBAChC,MAAM,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,SAAS,EAAE,CAAC;gBAC9C,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC;oBACtB,IAAI,CAAC,YAAY,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;oBAC5C,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;oBAC/D,IAAI,MAAM,EAAE,CAAC;wBACX,IAAI,CAAC,YAAY,CAAC,cAAc,CAAC,MAAM,CAAC,GAAG,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC;oBAC7D,CAAC;oBACD,IAAI,CAAC,iBAAiB,EAAE,CAAC;gBAC3B,CAAC;gBACD,MAAM;YACR,CAAC;YACD,KAAK,QAAQ;gBACX,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;gBAC1B,IAAI,CAAC,gBAAgB,EAAE,CAAC;gBACxB,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC;gBAC/C,MAAM;YACR,KAAK,oBAAoB,CAAC,CAAC,CAAC;gBAC1B,MAAM,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,eAAe,EAAE,CAAC;gBACpD,IAAI,CAAC,YAAY,CAAC,iBAAiB,CAAC,KAAK,CAAC,KAAK,EAAE,KAAK,CAAC,WAAW,EAAE,KAAK,CAAC,YAAY,CAAC,CAAC;gBACxF,MAAM;YACR,CAAC;YACD,KAAK,YAAY;gBACf,kFAAkF;gBAClF,MAAM,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;gBACzB,MAAM;QACV,CAAC;IACH,CAAC;IAEO,eAAe,CAAC,KAAwB;QAC9C,MAAM,SAAS,GAAc;YAC3B,IAAI,EAAE,KAAK,CAAC,KAAK,EAAE,IAAI,IAAI,eAAe;YAC1C,OAAO,EAAE,KAAK,CAAC,KAAK,EAAE,OAAO,IAAI,oCAAoC;SACtE,CAAC;QACF,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;QACtC,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,SAAS,CAAC,CAAC;IACrC,CAAC;IAED,gDAAgD;IAChD,uBAAuB;IACvB,gDAAgD;IAExC,KAAK,CAAC,kBAAkB,CAAC,aAAqB;QACpD,IAAI,CAAC;YACH,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;gBACzB,MAAM,IAAI,CAAC,eAAe,CAAC,qBAAqB,CAAC,aAAa,CAAC,CAAC;gBAChE,OAAO,CAAC,GAAG,CAAC,0CAA0C,CAAC,CAAC;YAC1D,CAAC;iBAAM,CAAC;gBACN,OAAO,CAAC,IAAI,CACV,6FAA6F,CAC9F,CAAC;YACJ,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,mDAAmD;YACnD,OAAO,CAAC,KAAK,CAAC,uCAAuC,EAAE,KAAK,CAAC,CAAC;QAChE,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,eAAe;QAC3B,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,yBAAyB;YAAE,OAAO;QAEnD,IAAI,CAAC;YACH,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;gBACzB,MAAM,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,CAAC;gBACrC,OAAO,CAAC,GAAG,CAAC,2CAA2C,CAAC,CAAC;YAC3D,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,sCAAsC;YACtC,OAAO,CAAC,IAAI,CAAC,wCAAwC,EAAE,KAAK,CAAC,CAAC;QAChE,CAAC;IACH,CAAC;+GAnkBU,WAAW,kBAMZ,mBAAmB;mHANlB,WAAW,cADE,MAAM;;4FACnB,WAAW;kBADvB,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE;;0BAO7B,MAAM;2BAAC,mBAAmB","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  GetProfileResponse,\n  UpdateProfileRequest,\n  UpdateProfileResponse,\n  MFASetupResponse,\n  MFAConfirmResponse,\n  MFADisableResponse,\n  ForgotPasswordRequest,\n  ForgotPasswordResponse,\n  ResetPasswordRequest,\n  ResetPasswordResponse,\n  SwitchOrgResponse,\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.signOutFirebase();\n          this.clearState();\n        }\n      } else {\n        this.signOutFirebase();\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  // ORGANIZACIONES\n  // =============================================\n\n  /**\n   * Cambia la organización activa del usuario.\n   * Genera un nuevo Firebase token con el activeOrg actualizado.\n   *\n   * @param organizationId - ID de la organización a la que cambiar\n   * @returns Observable con el nuevo Firebase token y activeOrg\n   *\n   * @example\n   * ```typescript\n   * await firstValueFrom(this.auth.switchOrg('org_xyz789'));\n   * // Firebase ya está re-autenticado con la nueva org\n   * const activeOrg = await this.firebase.getActiveOrg();\n   * ```\n   */\n  switchOrg(organizationId: string): Observable<SwitchOrgResponse> {\n    return this.http\n      .post<SwitchOrgResponse>(`${this.baseUrl}/switch-org`, { organizationId })\n      .pipe(\n        tap(async response => {\n          // Re-autenticar en Firebase con el nuevo token\n          if (response.firebaseToken) {\n            await this.signInWithFirebase(response.firebaseToken);\n          }\n          // Notificar cambio a otras pestañas\n          this.syncService.broadcast({\n            type: 'ORG_SWITCH',\n            payload: { activeOrg: response.activeOrg },\n          });\n        }),\n        catchError(error => this.handleAuthError(error))\n      );\n  }\n\n  // =============================================\n  // PERFIL DE USUARIO\n  // =============================================\n\n  /**\n   * Obtiene el perfil del usuario autenticado.\n   * Incluye información de MFA y teléfono.\n   */\n  getProfile(): Observable<GetProfileResponse> {\n    return this.http\n      .get<GetProfileResponse>(`${this.baseUrl}/profile`)\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Actualiza el perfil del usuario.\n   */\n  updateProfile(request: UpdateProfileRequest): Observable<UpdateProfileResponse> {\n    return this.http\n      .put<UpdateProfileResponse>(`${this.baseUrl}/profile`, request)\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  // =============================================\n  // RECUPERACIÓN DE CONTRASEÑA\n  // =============================================\n\n  /**\n   * Inicia el proceso de recuperación de contraseña.\n   * Envía un código al email del usuario.\n   */\n  forgotPassword(request: ForgotPasswordRequest): Observable<ForgotPasswordResponse> {\n    return this.http\n      .post<ForgotPasswordResponse>(`${this.baseUrl}/forgot-password`, request)\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Resetea la contraseña usando el código enviado por email.\n   */\n  resetPassword(request: ResetPasswordRequest): Observable<ResetPasswordResponse> {\n    return this.http\n      .post<ResetPasswordResponse>(`${this.baseUrl}/reset-password`, request)\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      case 'ORG_SWITCH':\n        // Otra pestaña cambió de organización - recargar página para obtener nuevo estado\n        window.location.reload();\n        break;\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"]}
|
|
537
|
+
//# 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, from, firstValueFrom, Subscription } from 'rxjs';\nimport { tap, catchError, switchMap, map } 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  GetProfileResponse,\n  UpdateProfileRequest,\n  UpdateProfileResponse,\n  MFASetupResponse,\n  MFAConfirmResponse,\n  MFADisableResponse,\n  ForgotPasswordRequest,\n  ForgotPasswordResponse,\n  ResetPasswordRequest,\n  ResetPasswordResponse,\n  SwitchOrgResponse,\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.signOutFirebase();\n          this.clearState();\n        }\n      } else {\n        this.signOutFirebase();\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  // ORGANIZACIONES\n  // =============================================\n\n  /**\n   * Cambia la organización activa del usuario.\n   * Genera un nuevo Firebase token con el activeOrg actualizado.\n   *\n   * @param organizationId - ID de la organización a la que cambiar\n   * @returns Observable con el nuevo Firebase token y activeOrg\n   *\n   * @example\n   * ```typescript\n   * await firstValueFrom(this.auth.switchOrg('org_xyz789'));\n   * // Firebase ya está re-autenticado con la nueva org\n   * const activeOrg = await this.firebase.getActiveOrg();\n   * ```\n   */\n  switchOrg(organizationId: string): Observable<SwitchOrgResponse> {\n    return this.http\n      .post<SwitchOrgResponse>(`${this.baseUrl}/switch-org`, { organizationId })\n      .pipe(\n        switchMap(response => {\n          // Re-autenticar en Firebase con el nuevo token\n          // Usar switchMap + from para esperar a que Firebase complete\n          const firebaseAuth$ = response.firebaseToken\n            ? from(this.signInWithFirebase(response.firebaseToken))\n            : of(undefined);\n\n          return firebaseAuth$.pipe(\n            tap(() => {\n              // Notificar cambio a otras pestañas\n              this.syncService.broadcast({\n                type: 'ORG_SWITCH',\n                payload: { activeOrg: response.activeOrg },\n              });\n            }),\n            // Retornar response original después de que Firebase complete\n            map(() => response)\n          );\n        }),\n        catchError(error => this.handleAuthError(error))\n      );\n  }\n\n  // =============================================\n  // PERFIL DE USUARIO\n  // =============================================\n\n  /**\n   * Obtiene el perfil del usuario autenticado.\n   * Incluye información de MFA y teléfono.\n   */\n  getProfile(): Observable<GetProfileResponse> {\n    return this.http\n      .get<GetProfileResponse>(`${this.baseUrl}/profile`)\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Actualiza el perfil del usuario.\n   */\n  updateProfile(request: UpdateProfileRequest): Observable<UpdateProfileResponse> {\n    return this.http\n      .put<UpdateProfileResponse>(`${this.baseUrl}/profile`, request)\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  // =============================================\n  // RECUPERACIÓN DE CONTRASEÑA\n  // =============================================\n\n  /**\n   * Inicia el proceso de recuperación de contraseña.\n   * Envía un código al email del usuario.\n   */\n  forgotPassword(request: ForgotPasswordRequest): Observable<ForgotPasswordResponse> {\n    return this.http\n      .post<ForgotPasswordResponse>(`${this.baseUrl}/forgot-password`, request)\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Resetea la contraseña usando el código enviado por email.\n   */\n  resetPassword(request: ResetPasswordRequest): Observable<ResetPasswordResponse> {\n    return this.http\n      .post<ResetPasswordResponse>(`${this.baseUrl}/reset-password`, request)\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      case 'ORG_SWITCH':\n        // Otra pestaña cambió de organización - recargar página para obtener nuevo estado\n        window.location.reload();\n        break;\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"]}
|
|
@@ -33,8 +33,8 @@ export const authInterceptor = (request, next) => {
|
|
|
33
33
|
if (!isApiRequest(request, config.apiUrl)) {
|
|
34
34
|
return next(request);
|
|
35
35
|
}
|
|
36
|
-
// Omitir endpoints
|
|
37
|
-
if (
|
|
36
|
+
// Omitir endpoints públicos que no necesitan token
|
|
37
|
+
if (isPublicEndpoint(request, config.authPrefix)) {
|
|
38
38
|
return next(request);
|
|
39
39
|
}
|
|
40
40
|
const accessToken = authService.accessToken();
|
|
@@ -43,7 +43,7 @@ export const authInterceptor = (request, next) => {
|
|
|
43
43
|
request = addAuthHeader(request, accessToken);
|
|
44
44
|
}
|
|
45
45
|
return next(request).pipe(catchError((error) => {
|
|
46
|
-
if (error.status === 401 && !
|
|
46
|
+
if (error.status === 401 && !isNoRefreshEndpoint(request, config.authPrefix)) {
|
|
47
47
|
return handle401Error(request, next, authService);
|
|
48
48
|
}
|
|
49
49
|
if (error.status === 403) {
|
|
@@ -69,22 +69,35 @@ function isApiRequest(request, apiUrl) {
|
|
|
69
69
|
return request.url.startsWith(apiUrl) || request.url.includes('/v2/auth');
|
|
70
70
|
}
|
|
71
71
|
/**
|
|
72
|
-
* Verifica si la request es a un endpoint
|
|
72
|
+
* Verifica si la request es a un endpoint público que NO requiere token.
|
|
73
73
|
*/
|
|
74
|
-
function
|
|
75
|
-
|
|
74
|
+
function isPublicEndpoint(request, authPrefix) {
|
|
75
|
+
const publicEndpoints = [
|
|
76
|
+
'/signin',
|
|
77
|
+
'/signup',
|
|
78
|
+
'/refresh',
|
|
79
|
+
'/logout',
|
|
80
|
+
'/mfa/verify', // Solo durante login
|
|
81
|
+
];
|
|
82
|
+
return publicEndpoints.some((endpoint) => request.url.includes(`${authPrefix}${endpoint}`));
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Verifica si la request es a un endpoint que NO debe reintentar refresh en 401.
|
|
86
|
+
*/
|
|
87
|
+
function isNoRefreshEndpoint(request, authPrefix) {
|
|
88
|
+
// Endpoints que no deben intentar refresh en 401:
|
|
76
89
|
// - Públicos: signin, signup, refresh, logout
|
|
77
|
-
// - MFA: verify (durante login),
|
|
78
|
-
const
|
|
90
|
+
// - MFA: verify (durante login), confirm (401 = código incorrecto), disable (401 = contraseña incorrecta)
|
|
91
|
+
const noRefreshEndpoints = [
|
|
79
92
|
'/signin',
|
|
80
93
|
'/signup',
|
|
81
94
|
'/refresh',
|
|
82
95
|
'/logout',
|
|
83
96
|
'/mfa/verify',
|
|
84
|
-
'/mfa/disable',
|
|
85
97
|
'/mfa/confirm',
|
|
98
|
+
'/mfa/disable',
|
|
86
99
|
];
|
|
87
|
-
return
|
|
100
|
+
return noRefreshEndpoints.some((endpoint) => request.url.includes(`${authPrefix}${endpoint}`));
|
|
88
101
|
}
|
|
89
102
|
/**
|
|
90
103
|
* Maneja errores 401 refrescando el token.
|
|
@@ -106,4 +119,4 @@ function handle401Error(request, next, authService) {
|
|
|
106
119
|
// Esperar a que termine el refresco en curso
|
|
107
120
|
return refreshTokenSubject.pipe(filter((token) => token !== null), take(1), switchMap((token) => next(addAuthHeader(request, token))));
|
|
108
121
|
}
|
|
109
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"interceptor.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/interceptor.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject } from '@angular/core';\nimport {\n  HttpInterceptorFn,\n  HttpRequest,\n  HttpHandlerFn,\n  HttpEvent,\n  HttpErrorResponse,\n} from '@angular/common/http';\nimport { Observable, throwError, BehaviorSubject } from 'rxjs';\nimport { catchError, filter, take, switchMap, finalize } from 'rxjs/operators';\nimport { AuthService } from './auth.service';\nimport { VALTECH_AUTH_CONFIG } from './config';\n\n// Control de estado de refresco (singleton a nivel de módulo)\nlet isRefreshing = false;\nconst refreshTokenSubject = new BehaviorSubject<string | null>(null);\n\n/**\n * Interceptor HTTP que:\n * 1. Agrega header Authorization con Bearer token a requests API\n * 2. Maneja errores 401 refrescando el token automáticamente\n * 3. Encola requests durante el refresco para evitar múltiples refresh\n *\n * @example\n * ```typescript\n * // Incluido automáticamente por provideValtechAuth()\n * // Para uso manual:\n * import { provideHttpClient, withInterceptors } from '@angular/common/http';\n * import { authInterceptor } from 'valtech-components';\n *\n * bootstrapApplication(AppComponent, {\n *   providers: [\n *     provideHttpClient(withInterceptors([authInterceptor])),\n *   ],\n * });\n * ```\n */\nexport const authInterceptor: HttpInterceptorFn = (\n  request: HttpRequest<unknown>,\n  next: HttpHandlerFn\n): Observable<HttpEvent<unknown>> => {\n  const authService = inject(AuthService);\n  const config = inject(VALTECH_AUTH_CONFIG);\n\n  // Omitir requests que no son a nuestra API\n  if (!isApiRequest(request, config.apiUrl)) {\n    return next(request);\n  }\n\n  // Omitir endpoints de auth que no necesitan token\n  if (isAuthEndpoint(request, config.authPrefix!)) {\n    return next(request);\n  }\n\n  const accessToken = authService.accessToken();\n\n  // Agregar header de autorización si hay token\n  if (accessToken) {\n    request = addAuthHeader(request, accessToken);\n  }\n\n  return next(request).pipe(\n    catchError((error: HttpErrorResponse) => {\n      if (error.status === 401 && !isAuthEndpoint(request, config.authPrefix!)) {\n        return handle401Error(request, next, authService);\n      }\n\n      if (error.status === 403) {\n        console.error(\n          '[ValtechAuth] Permiso denegado:',\n          error.error?.message || 'Acceso prohibido'\n        );\n      }\n\n      return throwError(() => error);\n    })\n  );\n};\n\n/**\n * Agrega header de autorización a la request.\n */\nfunction addAuthHeader(\n  request: HttpRequest<unknown>,\n  token: string\n): HttpRequest<unknown> {\n  return request.clone({\n    setHeaders: {\n      Authorization: `Bearer ${token}`,\n    },\n  });\n}\n\n/**\n * Verifica si la request es a nuestra API.\n */\nfunction isApiRequest(request: HttpRequest<unknown>, apiUrl: string): boolean {\n  return request.url.startsWith(apiUrl) || request.url.includes('/v2/auth');\n}\n\n/**\n * Verifica si la request es a un endpoint de auth que no debe reintentar.\n */\nfunction isAuthEndpoint(\n  request: HttpRequest<unknown>,\n  authPrefix: string\n): boolean {\n  // Endpoints que no deben intentar refresh en 401\n  // - Públicos: signin, signup, refresh, logout\n  // - MFA: verify (durante login), disable (401 = contraseña incorrecta), confirm (401 = código incorrecto)\n  const authEndpoints = [\n    '/signin',\n    '/signup',\n    '/refresh',\n    '/logout',\n    '/mfa/verify',\n    '/mfa/disable',\n    '/mfa/confirm',\n  ];\n  return authEndpoints.some((endpoint) =>\n    request.url.includes(`${authPrefix}${endpoint}`)\n  );\n}\n\n/**\n * Maneja errores 401 refrescando el token.\n */\nfunction handle401Error(\n  request: HttpRequest<unknown>,\n  next: HttpHandlerFn,\n  authService: AuthService\n): Observable<HttpEvent<unknown>> {\n  if (!isRefreshing) {\n    isRefreshing = true;\n    refreshTokenSubject.next(null);\n\n    return authService.refreshAccessToken().pipe(\n      switchMap((response) => {\n        refreshTokenSubject.next(response.accessToken);\n        return next(addAuthHeader(request, response.accessToken));\n      }),\n      catchError((error) => {\n        authService.logout();\n        return throwError(() => error);\n      }),\n      finalize(() => {\n        isRefreshing = false;\n      })\n    );\n  }\n\n  // Esperar a que termine el refresco en curso\n  return refreshTokenSubject.pipe(\n    filter((token): token is string => token !== null),\n    take(1),\n    switchMap((token) => next(addAuthHeader(request, token)))\n  );\n}\n"]}
|
|
122
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"interceptor.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/interceptor.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject } from '@angular/core';\nimport {\n  HttpInterceptorFn,\n  HttpRequest,\n  HttpHandlerFn,\n  HttpEvent,\n  HttpErrorResponse,\n} from '@angular/common/http';\nimport { Observable, throwError, BehaviorSubject } from 'rxjs';\nimport { catchError, filter, take, switchMap, finalize } from 'rxjs/operators';\nimport { AuthService } from './auth.service';\nimport { VALTECH_AUTH_CONFIG } from './config';\n\n// Control de estado de refresco (singleton a nivel de módulo)\nlet isRefreshing = false;\nconst refreshTokenSubject = new BehaviorSubject<string | null>(null);\n\n/**\n * Interceptor HTTP que:\n * 1. Agrega header Authorization con Bearer token a requests API\n * 2. Maneja errores 401 refrescando el token automáticamente\n * 3. Encola requests durante el refresco para evitar múltiples refresh\n *\n * @example\n * ```typescript\n * // Incluido automáticamente por provideValtechAuth()\n * // Para uso manual:\n * import { provideHttpClient, withInterceptors } from '@angular/common/http';\n * import { authInterceptor } from 'valtech-components';\n *\n * bootstrapApplication(AppComponent, {\n *   providers: [\n *     provideHttpClient(withInterceptors([authInterceptor])),\n *   ],\n * });\n * ```\n */\nexport const authInterceptor: HttpInterceptorFn = (\n  request: HttpRequest<unknown>,\n  next: HttpHandlerFn\n): Observable<HttpEvent<unknown>> => {\n  const authService = inject(AuthService);\n  const config = inject(VALTECH_AUTH_CONFIG);\n\n  // Omitir requests que no son a nuestra API\n  if (!isApiRequest(request, config.apiUrl)) {\n    return next(request);\n  }\n\n  // Omitir endpoints públicos que no necesitan token\n  if (isPublicEndpoint(request, config.authPrefix!)) {\n    return next(request);\n  }\n\n  const accessToken = authService.accessToken();\n\n  // Agregar header de autorización si hay token\n  if (accessToken) {\n    request = addAuthHeader(request, accessToken);\n  }\n\n  return next(request).pipe(\n    catchError((error: HttpErrorResponse) => {\n      if (error.status === 401 && !isNoRefreshEndpoint(request, config.authPrefix!)) {\n        return handle401Error(request, next, authService);\n      }\n\n      if (error.status === 403) {\n        console.error(\n          '[ValtechAuth] Permiso denegado:',\n          error.error?.message || 'Acceso prohibido'\n        );\n      }\n\n      return throwError(() => error);\n    })\n  );\n};\n\n/**\n * Agrega header de autorización a la request.\n */\nfunction addAuthHeader(\n  request: HttpRequest<unknown>,\n  token: string\n): HttpRequest<unknown> {\n  return request.clone({\n    setHeaders: {\n      Authorization: `Bearer ${token}`,\n    },\n  });\n}\n\n/**\n * Verifica si la request es a nuestra API.\n */\nfunction isApiRequest(request: HttpRequest<unknown>, apiUrl: string): boolean {\n  return request.url.startsWith(apiUrl) || request.url.includes('/v2/auth');\n}\n\n/**\n * Verifica si la request es a un endpoint público que NO requiere token.\n */\nfunction isPublicEndpoint(\n  request: HttpRequest<unknown>,\n  authPrefix: string\n): boolean {\n  const publicEndpoints = [\n    '/signin',\n    '/signup',\n    '/refresh',\n    '/logout',\n    '/mfa/verify', // Solo durante login\n  ];\n  return publicEndpoints.some((endpoint) =>\n    request.url.includes(`${authPrefix}${endpoint}`)\n  );\n}\n\n/**\n * Verifica si la request es a un endpoint que NO debe reintentar refresh en 401.\n */\nfunction isNoRefreshEndpoint(\n  request: HttpRequest<unknown>,\n  authPrefix: string\n): boolean {\n  // Endpoints que no deben intentar refresh en 401:\n  // - Públicos: signin, signup, refresh, logout\n  // - MFA: verify (durante login), confirm (401 = código incorrecto), disable (401 = contraseña incorrecta)\n  const noRefreshEndpoints = [\n    '/signin',\n    '/signup',\n    '/refresh',\n    '/logout',\n    '/mfa/verify',\n    '/mfa/confirm',\n    '/mfa/disable',\n  ];\n  return noRefreshEndpoints.some((endpoint) =>\n    request.url.includes(`${authPrefix}${endpoint}`)\n  );\n}\n\n/**\n * Maneja errores 401 refrescando el token.\n */\nfunction handle401Error(\n  request: HttpRequest<unknown>,\n  next: HttpHandlerFn,\n  authService: AuthService\n): Observable<HttpEvent<unknown>> {\n  if (!isRefreshing) {\n    isRefreshing = true;\n    refreshTokenSubject.next(null);\n\n    return authService.refreshAccessToken().pipe(\n      switchMap((response) => {\n        refreshTokenSubject.next(response.accessToken);\n        return next(addAuthHeader(request, response.accessToken));\n      }),\n      catchError((error) => {\n        authService.logout();\n        return throwError(() => error);\n      }),\n      finalize(() => {\n        isRefreshing = false;\n      })\n    );\n  }\n\n  // Esperar a que termine el refresco en curso\n  return refreshTokenSubject.pipe(\n    filter((token): token is string => token !== null),\n    take(1),\n    switchMap((token) => next(addAuthHeader(request, token)))\n  );\n}\n"]}
|
|
@@ -100,6 +100,11 @@ export class FirebaseService {
|
|
|
100
100
|
async signInWithCustomToken(token) {
|
|
101
101
|
try {
|
|
102
102
|
const credential = await signInWithCustomToken(this.auth, token);
|
|
103
|
+
// Forzar refresh del token para asegurar que los claims estén actualizados
|
|
104
|
+
// Esto es necesario porque getIdTokenResult() cachea el resultado
|
|
105
|
+
if (credential.user) {
|
|
106
|
+
await credential.user.getIdToken(true);
|
|
107
|
+
}
|
|
103
108
|
return credential;
|
|
104
109
|
}
|
|
105
110
|
catch (error) {
|
|
@@ -177,14 +182,15 @@ export class FirebaseService {
|
|
|
177
182
|
* Obtiene los claims personalizados del token del usuario.
|
|
178
183
|
* Los claims son establecidos por tu backend al crear el Custom Token.
|
|
179
184
|
*
|
|
185
|
+
* @param forceRefresh - Si true, fuerza la renovación del token para obtener claims actualizados
|
|
180
186
|
* @returns Objeto con los claims o vacío si no hay usuario
|
|
181
187
|
*/
|
|
182
|
-
async getClaims() {
|
|
188
|
+
async getClaims(forceRefresh = false) {
|
|
183
189
|
const user = this.auth.currentUser;
|
|
184
190
|
if (!user)
|
|
185
191
|
return {};
|
|
186
192
|
try {
|
|
187
|
-
const result = await user.getIdTokenResult();
|
|
193
|
+
const result = await user.getIdTokenResult(forceRefresh);
|
|
188
194
|
return result.claims;
|
|
189
195
|
}
|
|
190
196
|
catch {
|
|
@@ -209,6 +215,7 @@ export class FirebaseService {
|
|
|
209
215
|
* Obtiene las memberships (organizaciones) del usuario.
|
|
210
216
|
* Cada membership contiene el rol y permisos en esa organización.
|
|
211
217
|
*
|
|
218
|
+
* @param forceRefresh - Si true, fuerza la renovación del token
|
|
212
219
|
* @returns Mapa de orgId → MembershipInfo
|
|
213
220
|
*
|
|
214
221
|
* @example
|
|
@@ -217,27 +224,29 @@ export class FirebaseService {
|
|
|
217
224
|
* // { 'org_abc': { roleId: 'admin', roleName: 'admin', permissions: ['users:*', ...] } }
|
|
218
225
|
* ```
|
|
219
226
|
*/
|
|
220
|
-
async getMemberships() {
|
|
221
|
-
const claims = await this.getClaims();
|
|
227
|
+
async getMemberships(forceRefresh = false) {
|
|
228
|
+
const claims = await this.getClaims(forceRefresh);
|
|
222
229
|
return claims['memberships'] || {};
|
|
223
230
|
}
|
|
224
231
|
/**
|
|
225
232
|
* Obtiene la organización activa del usuario.
|
|
226
233
|
* La organización activa se establece al hacer login o al cambiar de org.
|
|
227
234
|
*
|
|
235
|
+
* @param forceRefresh - Si true, fuerza la renovación del token
|
|
228
236
|
* @returns ID de la organización activa o null si no hay ninguna
|
|
229
237
|
*/
|
|
230
|
-
async getActiveOrg() {
|
|
231
|
-
const claims = await this.getClaims();
|
|
238
|
+
async getActiveOrg(forceRefresh = false) {
|
|
239
|
+
const claims = await this.getClaims(forceRefresh);
|
|
232
240
|
return claims['activeOrg'] || null;
|
|
233
241
|
}
|
|
234
242
|
/**
|
|
235
243
|
* Obtiene información de todas las organizaciones del usuario.
|
|
236
244
|
*
|
|
245
|
+
* @param forceRefresh - Si true, fuerza la renovación del token
|
|
237
246
|
* @returns Array con información de cada organización
|
|
238
247
|
*/
|
|
239
|
-
async getOrganizations() {
|
|
240
|
-
const memberships = await this.getMemberships();
|
|
248
|
+
async getOrganizations(forceRefresh = false) {
|
|
249
|
+
const memberships = await this.getMemberships(forceRefresh);
|
|
241
250
|
return Object.entries(memberships).map(([id, info]) => ({
|
|
242
251
|
id,
|
|
243
252
|
roleId: info.roleId,
|
|
@@ -444,4 +453,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
444
453
|
type: Inject,
|
|
445
454
|
args: [VALTECH_FIREBASE_CONFIG]
|
|
446
455
|
}] }] });
|
|
447
|
-
//# 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,EAAQ,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;IAkB1B,YACU,IAAU,EACuB,MAA6B;QAD9D,SAAI,GAAJ,IAAI,CAAM;QACuB,WAAM,GAAN,MAAM,CAAuB;QAnBxE,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;QAY3E,+CAA+C;QAC/C,IAAI,CAAC,KAAK,GAAG,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,CACpC,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,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CACrC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,EACrB,oBAAoB,EAAE,CACvB,CAAC;QACF,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,mCAAmC;IACnC,8EAA8E;IAE9E;;;;;;;;;;;OAWG;IACH,KAAK,CAAC,cAAc;QAClB,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,EAAE,CAAC;QACtC,OAAQ,MAAM,CAAC,aAAa,CAAoC,IAAI,EAAE,CAAC;IACzE,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,YAAY;QAChB,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,EAAE,CAAC;QACtC,OAAQ,MAAM,CAAC,WAAW,CAAY,IAAI,IAAI,CAAC;IACjD,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,gBAAgB;QACpB,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,cAAc,EAAE,CAAC;QAChD,OAAO,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,IAAI,CAAC,EAAE,EAAE,CAAC,CAAC;YACtD,EAAE;YACF,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,WAAW,EAAE,IAAI,CAAC,WAAW;SAC9B,CAAC,CAAC,CAAC;IACN,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,kBAAkB;QACtB,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,cAAc,EAAE,CAAC;QAChD,OAAO,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IAClC,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,UAAU,CAAC,KAAa;QAC5B,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,cAAc,EAAE,CAAC;QAChD,OAAO,KAAK,IAAI,WAAW,CAAC;IAC9B,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,YAAY,CAAC,KAAa;QAC9B,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,cAAc,EAAE,CAAC;QAChD,OAAO,WAAW,CAAC,KAAK,CAAC,EAAE,MAAM,IAAI,IAAI,CAAC;IAC5C,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,mBAAmB,CAAC,KAAa;QACrC,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,cAAc,EAAE,CAAC;QAChD,OAAO,WAAW,CAAC,KAAK,CAAC,EAAE,WAAW,IAAI,EAAE,CAAC;IAC/C,CAAC;IAED;;;;;;;;;;;;;;;;;;OAkBG;IACH,KAAK,CAAC,aAAa,CAAC,KAAa,EAAE,QAAgB,EAAE,MAAc;QACjE,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,CAAC;QAC1D,MAAM,UAAU,GAAG,GAAG,QAAQ,IAAI,MAAM,EAAE,CAAC;QAE3C,OAAO,CACL,WAAW,CAAC,QAAQ,CAAC,UAAU,CAAC;YAChC,WAAW,CAAC,QAAQ,CAAC,GAAG,QAAQ,IAAI,CAAC;YACrC,WAAW,CAAC,QAAQ,CAAC,KAAK,CAAC,CAC5B,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,OAAO,CAAC,KAAa,EAAE,QAAgB;QAC3C,OAAO,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;IACrD,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,QAAQ,CAAC,KAAa,EAAE,QAAgB;QAC5C,OAAO,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE,QAAQ,EAAE,OAAO,CAAC,CAAC;IACtD,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,SAAS,CAAC,KAAa,EAAE,QAAgB;QAC7C,OAAO,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE,QAAQ,EAAE,QAAQ,CAAC,CAAC;IACvD,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,SAAS,CAAC,KAAa,EAAE,QAAgB;QAC7C,OAAO,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE,QAAQ,EAAE,QAAQ,CAAC,CAAC;IACvD,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,SAAS,CAAC,KAAa,EAAE,QAAgB;QAC7C,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,CAAC;QAC1D,OAAO,WAAW,CAAC,QAAQ,CAAC,GAAG,QAAQ,IAAI,CAAC,IAAI,WAAW,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IAC9E,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,YAAY,CAAC,KAAa;QAC9B,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,CAAC;QAC1D,OAAO,WAAW,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IACrC,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,YAAY,CAAC,KAAa;QAC9B,MAAM,IAAI,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QAC5C,OAAO,IAAI,KAAK,OAAO,IAAI,IAAI,KAAK,aAAa,CAAC;IACpD,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;+GAxbU,eAAe,sCAoBhB,uBAAuB;mHApBtB,eAAe,cADF,MAAM;;4FACnB,eAAe;kBAD3B,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE;;0BAqB7B,MAAM;2BAAC,uBAAuB","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, MembershipInfo, OrganizationInfo, 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  /** 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>;\n\n  /** Indica si el usuario está autenticado en Firebase */\n  readonly isAuthenticated$: Observable<boolean>;\n\n  constructor(\n    private auth: Auth,\n    @Inject(VALTECH_FIREBASE_CONFIG) private config: ValtechFirebaseConfig\n  ) {\n    // Inicializar observables que dependen de auth\n    this.user$ = authState(this.auth).pipe(\n      map((user) => (user ? this.mapUser(user) : null)),\n      distinctUntilChanged((a, b) => a?.uid === b?.uid)\n    );\n\n    this.isAuthenticated$ = this.user$.pipe(\n      map((user) => !!user),\n      distinctUntilChanged()\n    );\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  // RBAC - PERMISOS Y ORGANIZACIONES\n  // ===========================================================================\n\n  /**\n   * Obtiene las memberships (organizaciones) del usuario.\n   * Cada membership contiene el rol y permisos en esa organización.\n   *\n   * @returns Mapa de orgId → MembershipInfo\n   *\n   * @example\n   * ```typescript\n   * const memberships = await firebaseService.getMemberships();\n   * // { 'org_abc': { roleId: 'admin', roleName: 'admin', permissions: ['users:*', ...] } }\n   * ```\n   */\n  async getMemberships(): Promise<Record<string, MembershipInfo>> {\n    const claims = await this.getClaims();\n    return (claims['memberships'] as Record<string, MembershipInfo>) || {};\n  }\n\n  /**\n   * Obtiene la organización activa del usuario.\n   * La organización activa se establece al hacer login o al cambiar de org.\n   *\n   * @returns ID de la organización activa o null si no hay ninguna\n   */\n  async getActiveOrg(): Promise<string | null> {\n    const claims = await this.getClaims();\n    return (claims['activeOrg'] as string) || null;\n  }\n\n  /**\n   * Obtiene información de todas las organizaciones del usuario.\n   *\n   * @returns Array con información de cada organización\n   */\n  async getOrganizations(): Promise<OrganizationInfo[]> {\n    const memberships = await this.getMemberships();\n    return Object.entries(memberships).map(([id, info]) => ({\n      id,\n      roleId: info.roleId,\n      roleName: info.roleName,\n      permissions: info.permissions,\n    }));\n  }\n\n  /**\n   * Obtiene los IDs de todas las organizaciones del usuario.\n   *\n   * @returns Array de IDs de organizaciones\n   */\n  async getOrganizationIds(): Promise<string[]> {\n    const memberships = await this.getMemberships();\n    return Object.keys(memberships);\n  }\n\n  /**\n   * Verifica si el usuario pertenece a una organización.\n   *\n   * @param orgId - ID de la organización\n   * @returns true si el usuario es miembro\n   */\n  async isMemberOf(orgId: string): Promise<boolean> {\n    const memberships = await this.getMemberships();\n    return orgId in memberships;\n  }\n\n  /**\n   * Obtiene el rol del usuario en una organización.\n   *\n   * @param orgId - ID de la organización\n   * @returns ID del rol o null si no es miembro\n   */\n  async getRoleInOrg(orgId: string): Promise<string | null> {\n    const memberships = await this.getMemberships();\n    return memberships[orgId]?.roleId || null;\n  }\n\n  /**\n   * Obtiene los permisos del usuario en una organización.\n   *\n   * @param orgId - ID de la organización\n   * @returns Array de permisos en formato 'resource:action'\n   */\n  async getPermissionsInOrg(orgId: string): Promise<string[]> {\n    const memberships = await this.getMemberships();\n    return memberships[orgId]?.permissions || [];\n  }\n\n  /**\n   * Verifica si el usuario tiene un permiso específico en una organización.\n   * Soporta wildcards: 'resource:*' y '*:*' (super admin).\n   *\n   * @param orgId - ID de la organización\n   * @param resource - Recurso a verificar (ej: 'users', 'documents')\n   * @param action - Acción a verificar (ej: 'read', 'write', 'create', 'delete')\n   * @returns true si tiene el permiso\n   *\n   * @example\n   * ```typescript\n   * // Verificar permiso específico\n   * const canReadUsers = await firebaseService.hasPermission('org_abc', 'users', 'read');\n   *\n   * // Verificar en la organización activa\n   * const orgId = await firebaseService.getActiveOrg();\n   * const canEdit = await firebaseService.hasPermission(orgId!, 'documents', 'write');\n   * ```\n   */\n  async hasPermission(orgId: string, resource: string, action: string): Promise<boolean> {\n    const permissions = await this.getPermissionsInOrg(orgId);\n    const permission = `${resource}:${action}`;\n\n    return (\n      permissions.includes(permission) ||\n      permissions.includes(`${resource}:*`) ||\n      permissions.includes('*:*')\n    );\n  }\n\n  /**\n   * Verifica si el usuario puede leer un recurso en una organización.\n   * Atajo para hasPermission(orgId, resource, 'read').\n   */\n  async canRead(orgId: string, resource: string): Promise<boolean> {\n    return this.hasPermission(orgId, resource, 'read');\n  }\n\n  /**\n   * Verifica si el usuario puede escribir un recurso en una organización.\n   * Atajo para hasPermission(orgId, resource, 'write').\n   */\n  async canWrite(orgId: string, resource: string): Promise<boolean> {\n    return this.hasPermission(orgId, resource, 'write');\n  }\n\n  /**\n   * Verifica si el usuario puede crear un recurso en una organización.\n   * Atajo para hasPermission(orgId, resource, 'create').\n   */\n  async canCreate(orgId: string, resource: string): Promise<boolean> {\n    return this.hasPermission(orgId, resource, 'create');\n  }\n\n  /**\n   * Verifica si el usuario puede eliminar un recurso en una organización.\n   * Atajo para hasPermission(orgId, resource, 'delete').\n   */\n  async canDelete(orgId: string, resource: string): Promise<boolean> {\n    return this.hasPermission(orgId, resource, 'delete');\n  }\n\n  /**\n   * Verifica si el usuario puede administrar un recurso en una organización.\n   * Equivale a tener 'resource:*' o '*:*'.\n   */\n  async canManage(orgId: string, resource: string): Promise<boolean> {\n    const permissions = await this.getPermissionsInOrg(orgId);\n    return permissions.includes(`${resource}:*`) || permissions.includes('*:*');\n  }\n\n  /**\n   * Verifica si el usuario es super admin en una organización.\n   * Super admin tiene el permiso '*:*'.\n   */\n  async isSuperAdmin(orgId: string): Promise<boolean> {\n    const permissions = await this.getPermissionsInOrg(orgId);\n    return permissions.includes('*:*');\n  }\n\n  /**\n   * Verifica si el usuario es admin en una organización.\n   */\n  async isAdminInOrg(orgId: string): Promise<boolean> {\n    const role = await this.getRoleInOrg(orgId);\n    return role === 'admin' || role === 'super_admin';\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"]}
|
|
456
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"firebase.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/firebase.service.ts"],"names":[],"mappings":"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","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, MembershipInfo, OrganizationInfo, 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  /** 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>;\n\n  /** Indica si el usuario está autenticado en Firebase */\n  readonly isAuthenticated$: Observable<boolean>;\n\n  constructor(\n    private auth: Auth,\n    @Inject(VALTECH_FIREBASE_CONFIG) private config: ValtechFirebaseConfig\n  ) {\n    // Inicializar observables que dependen de auth\n    this.user$ = authState(this.auth).pipe(\n      map((user) => (user ? this.mapUser(user) : null)),\n      distinctUntilChanged((a, b) => a?.uid === b?.uid)\n    );\n\n    this.isAuthenticated$ = this.user$.pipe(\n      map((user) => !!user),\n      distinctUntilChanged()\n    );\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\n      // Forzar refresh del token para asegurar que los claims estén actualizados\n      // Esto es necesario porque getIdTokenResult() cachea el resultado\n      if (credential.user) {\n        await credential.user.getIdToken(true);\n      }\n\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   * @param forceRefresh - Si true, fuerza la renovación del token para obtener claims actualizados\n   * @returns Objeto con los claims o vacío si no hay usuario\n   */\n  async getClaims(forceRefresh = false): Promise<Record<string, unknown>> {\n    const user = this.auth.currentUser;\n    if (!user) return {};\n\n    try {\n      const result = await user.getIdTokenResult(forceRefresh);\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  // RBAC - PERMISOS Y ORGANIZACIONES\n  // ===========================================================================\n\n  /**\n   * Obtiene las memberships (organizaciones) del usuario.\n   * Cada membership contiene el rol y permisos en esa organización.\n   *\n   * @param forceRefresh - Si true, fuerza la renovación del token\n   * @returns Mapa de orgId → MembershipInfo\n   *\n   * @example\n   * ```typescript\n   * const memberships = await firebaseService.getMemberships();\n   * // { 'org_abc': { roleId: 'admin', roleName: 'admin', permissions: ['users:*', ...] } }\n   * ```\n   */\n  async getMemberships(forceRefresh = false): Promise<Record<string, MembershipInfo>> {\n    const claims = await this.getClaims(forceRefresh);\n    return (claims['memberships'] as Record<string, MembershipInfo>) || {};\n  }\n\n  /**\n   * Obtiene la organización activa del usuario.\n   * La organización activa se establece al hacer login o al cambiar de org.\n   *\n   * @param forceRefresh - Si true, fuerza la renovación del token\n   * @returns ID de la organización activa o null si no hay ninguna\n   */\n  async getActiveOrg(forceRefresh = false): Promise<string | null> {\n    const claims = await this.getClaims(forceRefresh);\n    return (claims['activeOrg'] as string) || null;\n  }\n\n  /**\n   * Obtiene información de todas las organizaciones del usuario.\n   *\n   * @param forceRefresh - Si true, fuerza la renovación del token\n   * @returns Array con información de cada organización\n   */\n  async getOrganizations(forceRefresh = false): Promise<OrganizationInfo[]> {\n    const memberships = await this.getMemberships(forceRefresh);\n    return Object.entries(memberships).map(([id, info]) => ({\n      id,\n      roleId: info.roleId,\n      roleName: info.roleName,\n      permissions: info.permissions,\n    }));\n  }\n\n  /**\n   * Obtiene los IDs de todas las organizaciones del usuario.\n   *\n   * @returns Array de IDs de organizaciones\n   */\n  async getOrganizationIds(): Promise<string[]> {\n    const memberships = await this.getMemberships();\n    return Object.keys(memberships);\n  }\n\n  /**\n   * Verifica si el usuario pertenece a una organización.\n   *\n   * @param orgId - ID de la organización\n   * @returns true si el usuario es miembro\n   */\n  async isMemberOf(orgId: string): Promise<boolean> {\n    const memberships = await this.getMemberships();\n    return orgId in memberships;\n  }\n\n  /**\n   * Obtiene el rol del usuario en una organización.\n   *\n   * @param orgId - ID de la organización\n   * @returns ID del rol o null si no es miembro\n   */\n  async getRoleInOrg(orgId: string): Promise<string | null> {\n    const memberships = await this.getMemberships();\n    return memberships[orgId]?.roleId || null;\n  }\n\n  /**\n   * Obtiene los permisos del usuario en una organización.\n   *\n   * @param orgId - ID de la organización\n   * @returns Array de permisos en formato 'resource:action'\n   */\n  async getPermissionsInOrg(orgId: string): Promise<string[]> {\n    const memberships = await this.getMemberships();\n    return memberships[orgId]?.permissions || [];\n  }\n\n  /**\n   * Verifica si el usuario tiene un permiso específico en una organización.\n   * Soporta wildcards: 'resource:*' y '*:*' (super admin).\n   *\n   * @param orgId - ID de la organización\n   * @param resource - Recurso a verificar (ej: 'users', 'documents')\n   * @param action - Acción a verificar (ej: 'read', 'write', 'create', 'delete')\n   * @returns true si tiene el permiso\n   *\n   * @example\n   * ```typescript\n   * // Verificar permiso específico\n   * const canReadUsers = await firebaseService.hasPermission('org_abc', 'users', 'read');\n   *\n   * // Verificar en la organización activa\n   * const orgId = await firebaseService.getActiveOrg();\n   * const canEdit = await firebaseService.hasPermission(orgId!, 'documents', 'write');\n   * ```\n   */\n  async hasPermission(orgId: string, resource: string, action: string): Promise<boolean> {\n    const permissions = await this.getPermissionsInOrg(orgId);\n    const permission = `${resource}:${action}`;\n\n    return (\n      permissions.includes(permission) ||\n      permissions.includes(`${resource}:*`) ||\n      permissions.includes('*:*')\n    );\n  }\n\n  /**\n   * Verifica si el usuario puede leer un recurso en una organización.\n   * Atajo para hasPermission(orgId, resource, 'read').\n   */\n  async canRead(orgId: string, resource: string): Promise<boolean> {\n    return this.hasPermission(orgId, resource, 'read');\n  }\n\n  /**\n   * Verifica si el usuario puede escribir un recurso en una organización.\n   * Atajo para hasPermission(orgId, resource, 'write').\n   */\n  async canWrite(orgId: string, resource: string): Promise<boolean> {\n    return this.hasPermission(orgId, resource, 'write');\n  }\n\n  /**\n   * Verifica si el usuario puede crear un recurso en una organización.\n   * Atajo para hasPermission(orgId, resource, 'create').\n   */\n  async canCreate(orgId: string, resource: string): Promise<boolean> {\n    return this.hasPermission(orgId, resource, 'create');\n  }\n\n  /**\n   * Verifica si el usuario puede eliminar un recurso en una organización.\n   * Atajo para hasPermission(orgId, resource, 'delete').\n   */\n  async canDelete(orgId: string, resource: string): Promise<boolean> {\n    return this.hasPermission(orgId, resource, 'delete');\n  }\n\n  /**\n   * Verifica si el usuario puede administrar un recurso en una organización.\n   * Equivale a tener 'resource:*' o '*:*'.\n   */\n  async canManage(orgId: string, resource: string): Promise<boolean> {\n    const permissions = await this.getPermissionsInOrg(orgId);\n    return permissions.includes(`${resource}:*`) || permissions.includes('*:*');\n  }\n\n  /**\n   * Verifica si el usuario es super admin en una organización.\n   * Super admin tiene el permiso '*:*'.\n   */\n  async isSuperAdmin(orgId: string): Promise<boolean> {\n    const permissions = await this.getPermissionsInOrg(orgId);\n    return permissions.includes('*:*');\n  }\n\n  /**\n   * Verifica si el usuario es admin en una organización.\n   */\n  async isAdminInOrg(orgId: string): Promise<boolean> {\n    const role = await this.getRoleInOrg(orgId);\n    return role === 'admin' || role === 'super_admin';\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"]}
|