valtech-components 2.0.457 → 2.0.459

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.
@@ -97,10 +97,12 @@ export class AuthService {
97
97
  await firstValueFrom(this.refreshAccessToken());
98
98
  }
99
99
  catch {
100
+ this.signOutFirebase();
100
101
  this.clearState();
101
102
  }
102
103
  }
103
104
  else {
105
+ this.signOutFirebase();
104
106
  this.clearState();
105
107
  }
106
108
  }
@@ -471,4 +473,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
471
473
  type: Inject,
472
474
  args: [VALTECH_AUTH_CONFIG]
473
475
  }] }, { type: i1.HttpClient }, { type: i2.Router }, { type: i3.AuthStateService }, { type: i4.TokenService }, { type: i5.AuthStorageService }, { type: i6.AuthSyncService }, { type: i7.FirebaseService }] });
474
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/auth.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject, OnDestroy, Injector } from '@angular/core';\nimport { HttpClient, HttpErrorResponse } from '@angular/common/http';\nimport { Router } from '@angular/router';\nimport { Observable, throwError, of, firstValueFrom, Subscription } from 'rxjs';\nimport { tap, catchError } from 'rxjs/operators';\n\nimport { VALTECH_AUTH_CONFIG } from './config';\nimport { AuthStateService } from './auth-state.service';\nimport { TokenService } from './token.service';\nimport { AuthStorageService } from './storage.service';\nimport { AuthSyncService } from './sync.service';\nimport {\n  SigninRequest,\n  SigninResponse,\n  SignupRequest,\n  SignupResponse,\n  VerifyEmailRequest,\n  VerifyEmailResponse,\n  ResendCodeRequest,\n  ResendCodeResponse,\n  MFAVerifyResponse,\n  RefreshResponse,\n  GetPermissionsResponse,\n  MFASetupResponse,\n  MFAConfirmResponse,\n  MFADisableResponse,\n  ForgotPasswordRequest,\n  ForgotPasswordResponse,\n  ResetPasswordRequest,\n  ResetPasswordResponse,\n  MFAMethod,\n  AuthError,\n  AuthSyncEvent,\n  ValtechAuthConfig,\n} from './types';\nimport { FirebaseService } from '../firebase';\n\n\n/**\n * Servicio principal de autenticación.\n *\n * @example\n * ```typescript\n * import { AuthService } from 'valtech-components';\n *\n * @Component({...})\n * export class LoginComponent {\n *   private auth = inject(AuthService);\n *\n *   async login() {\n *     await firstValueFrom(this.auth.signin({ email, password }));\n *     if (this.auth.mfaPending().required) {\n *       // Mostrar UI de MFA\n *     } else {\n *       this.router.navigate(['/']);\n *     }\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class AuthService implements OnDestroy {\n  // Timer para refresh proactivo\n  private refreshTimerId: ReturnType<typeof setTimeout> | null = null;\n  private syncSubscription: Subscription | null = null;\n\n  constructor(\n    @Inject(VALTECH_AUTH_CONFIG) private config: ValtechAuthConfig,\n    private http: HttpClient,\n    private router: Router,\n    private stateService: AuthStateService,\n    private tokenService: TokenService,\n    private storageService: AuthStorageService,\n    private syncService: AuthSyncService,\n    private firebaseService: FirebaseService,\n  ) {}\n\n  // =============================================\n  // ESTADO PÚBLICO (Signals readonly)\n  // =============================================\n\n  /** Estado completo de autenticación */\n  readonly state = this.stateService.state;\n\n  /** Usuario está autenticado */\n  readonly isAuthenticated = this.stateService.isAuthenticated;\n\n  /** Estado de carga */\n  readonly isLoading = this.stateService.isLoading;\n\n  /** Información del usuario */\n  readonly user = this.stateService.user;\n\n  /** Token de acceso */\n  readonly accessToken = this.stateService.accessToken;\n\n  /** Roles del usuario */\n  readonly roles = this.stateService.roles;\n\n  /** Permisos del usuario */\n  readonly permissions = this.stateService.permissions;\n\n  /** Usuario es super admin */\n  readonly isSuperAdmin = this.stateService.isSuperAdmin;\n\n  /** Estado de MFA pendiente */\n  readonly mfaPending = this.stateService.mfaPending;\n\n  /** Error actual */\n  readonly error = this.stateService.error;\n\n  // =============================================\n  // INICIALIZACIÓN\n  // =============================================\n\n  /**\n   * Inicializa el servicio de autenticación.\n   * Llamado automáticamente por provideValtechAuth.\n   */\n  async initialize(): Promise<void> {\n    // 1. Cargar estado desde storage\n    const storedState = this.storageService.loadState();\n\n    if (storedState.accessToken) {\n      // 2. Verificar si token es válido\n      if (this.tokenService.isTokenValid(storedState.accessToken)) {\n        this.stateService.restoreFromStorage(storedState);\n\n        // Extraer info del token\n        const claims = this.tokenService.parseToken(storedState.accessToken);\n        if (claims) {\n          this.stateService.updateUserInfo(claims.uid, claims.email);\n        }\n\n        // 3. Iniciar timer de refresco proactivo\n        this.startRefreshTimer();\n      } else if (storedState.refreshToken) {\n        // 4. Token expirado pero hay refresh token - intentar refrescar\n        try {\n          await firstValueFrom(this.refreshAccessToken());\n        } catch {\n          this.clearState();\n        }\n      } else {\n        this.clearState();\n      }\n    }\n\n    // 5. Iniciar sincronización entre pestañas\n    if (this.config.enableTabSync) {\n      this.syncService.start();\n      this.syncSubscription = this.syncService.onEvent$.subscribe(event =>\n        this.handleSyncEvent(event)\n      );\n    }\n\n    this.stateService.setLoading(false);\n  }\n\n  ngOnDestroy(): void {\n    this.stopRefreshTimer();\n    this.syncSubscription?.unsubscribe();\n  }\n\n  // =============================================\n  // AUTENTICACIÓN\n  // =============================================\n\n  /**\n   * Inicia sesión con email y contraseña.\n   */\n  signin(request: SigninRequest): Observable<SigninResponse> {\n    this.stateService.clearError();\n\n    return this.http.post<SigninResponse>(`${this.baseUrl}/signin`, request).pipe(\n      tap(response => {\n        if (response.mfaRequired) {\n          // MFA requerido - guardar estado temporal\n          this.stateService.setMFAPending({\n            required: true,\n            mfaToken: response.mfaToken!,\n            method: response.mfaMethod!,\n          });\n        } else if (response.accessToken) {\n          // Login exitoso sin MFA\n          this.handleSuccessfulAuth(response);\n        }\n      }),\n      catchError(error => this.handleAuthError(error))\n    );\n  }\n\n  /**\n   * Registra un nuevo usuario.\n   * El usuario queda en estado PENDING hasta verificar su email.\n   */\n  signup(request: SignupRequest): Observable<SignupResponse> {\n    this.stateService.clearError();\n\n    return this.http\n      .post<SignupResponse>(`${this.baseUrl}/signup`, request)\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Verifica email con código de 6 dígitos.\n   * Si es exitoso, hace auto-login y retorna tokens.\n   */\n  verifyEmail(request: VerifyEmailRequest): Observable<VerifyEmailResponse> {\n    this.stateService.clearError();\n\n    return this.http.post<VerifyEmailResponse>(`${this.baseUrl}/verify-email`, request).pipe(\n      tap(response => {\n        if (response.verified && response.accessToken) {\n          // Auto-login: guardar tokens y actualizar estado\n          this.handleSuccessfulAuth(response as unknown as SigninResponse);\n        }\n      }),\n      catchError(error => this.handleAuthError(error))\n    );\n  }\n\n  /**\n   * Reenvía código de verificación al email.\n   */\n  resendCode(request: ResendCodeRequest): Observable<ResendCodeResponse> {\n    return this.http\n      .post<ResendCodeResponse>(`${this.baseUrl}/resend-code`, request)\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Verifica código MFA.\n   */\n  verifyMFA(code: string): Observable<MFAVerifyResponse> {\n    const mfaState = this.mfaPending();\n    if (!mfaState.mfaToken) {\n      return throwError(() => ({\n        code: 'MFA_NOT_PENDING',\n        message: 'No hay verificación MFA pendiente',\n      }));\n    }\n\n    return this.http\n      .post<MFAVerifyResponse>(`${this.baseUrl}/mfa/verify`, {\n        mfaToken: mfaState.mfaToken,\n        code,\n      })\n      .pipe(\n        tap(response => {\n          this.stateService.clearMFAPending();\n          this.handleSuccessfulAuth(response);\n        }),\n        catchError(error => this.handleAuthError(error))\n      );\n  }\n\n  /**\n   * Refresca el token de acceso.\n   */\n  refreshAccessToken(): Observable<RefreshResponse> {\n    const refreshToken = this.state().refreshToken;\n    if (!refreshToken) {\n      return throwError(() => ({\n        code: 'NO_REFRESH_TOKEN',\n        message: 'No hay token de refresco',\n      }));\n    }\n\n    return this.http.post<RefreshResponse>(`${this.baseUrl}/refresh`, { refreshToken }).pipe(\n      tap(response => {\n        const expiresAt = Date.now() + response.expiresIn * 1000;\n        this.stateService.updateAccessToken(response.accessToken, response.expiresIn);\n        this.storageService.saveAccessToken(response.accessToken, expiresAt);\n        this.startRefreshTimer();\n        this.syncService.broadcast({\n          type: 'TOKEN_REFRESH',\n          payload: { accessToken: response.accessToken, expiresAt },\n        });\n      }),\n      catchError(error => {\n        this.logout();\n        return throwError(() => error);\n      })\n    );\n  }\n\n  /**\n   * Cierra sesión.\n   */\n  logout(): void {\n    const refreshToken = this.state().refreshToken;\n\n    // Notificar al backend (fire and forget)\n    if (refreshToken) {\n      this.http\n        .post(`${this.baseUrl}/logout`, { refreshToken })\n        .pipe(catchError(() => of(null)))\n        .subscribe();\n    }\n\n    // Cerrar sesión de Firebase si está integrado\n    this.signOutFirebase();\n\n    this.clearState();\n    this.syncService.broadcast({ type: 'LOGOUT' });\n    this.router.navigate([this.config.loginRoute]);\n  }\n\n  // =============================================\n  // MFA SETUP (usuario autenticado)\n  // =============================================\n\n  /**\n   * Configura MFA para el usuario.\n   */\n  setupMFA(method: MFAMethod, phone?: string): Observable<MFASetupResponse> {\n    return this.http\n      .post<MFASetupResponse>(`${this.baseUrl}/mfa/setup`, { method, phone })\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Confirma la configuración de MFA.\n   */\n  confirmMFA(code: string): Observable<MFAConfirmResponse> {\n    return this.http\n      .post<MFAConfirmResponse>(`${this.baseUrl}/mfa/confirm`, { code })\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Deshabilita MFA.\n   */\n  disableMFA(password: string): Observable<MFADisableResponse> {\n    return this.http\n      .post<MFADisableResponse>(`${this.baseUrl}/mfa/disable`, { password })\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  // =============================================\n  // 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    }\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"]}
476
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/auth.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject, OnDestroy, Injector } from '@angular/core';\nimport { HttpClient, HttpErrorResponse } from '@angular/common/http';\nimport { Router } from '@angular/router';\nimport { Observable, throwError, of, firstValueFrom, Subscription } from 'rxjs';\nimport { tap, catchError } from 'rxjs/operators';\n\nimport { VALTECH_AUTH_CONFIG } from './config';\nimport { AuthStateService } from './auth-state.service';\nimport { TokenService } from './token.service';\nimport { AuthStorageService } from './storage.service';\nimport { AuthSyncService } from './sync.service';\nimport {\n  SigninRequest,\n  SigninResponse,\n  SignupRequest,\n  SignupResponse,\n  VerifyEmailRequest,\n  VerifyEmailResponse,\n  ResendCodeRequest,\n  ResendCodeResponse,\n  MFAVerifyResponse,\n  RefreshResponse,\n  GetPermissionsResponse,\n  MFASetupResponse,\n  MFAConfirmResponse,\n  MFADisableResponse,\n  ForgotPasswordRequest,\n  ForgotPasswordResponse,\n  ResetPasswordRequest,\n  ResetPasswordResponse,\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  // 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    }\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"]}
@@ -4,9 +4,10 @@
4
4
  * Servicio genérico para operaciones CRUD en Firestore.
5
5
  * Soporta lecturas one-time, subscripciones real-time, paginación y queries complejas.
6
6
  */
7
- import { Injectable } from '@angular/core';
7
+ import { inject, Injectable } from '@angular/core';
8
8
  import { addDoc, collection, collectionData, deleteDoc, doc, docData, getDoc, getDocs, limit, orderBy, query, serverTimestamp, setDoc, startAfter, startAt, endBefore, endAt, Timestamp, updateDoc, where, writeBatch, increment, arrayUnion, arrayRemove, } from '@angular/fire/firestore';
9
9
  import { map } from 'rxjs';
10
+ import { VALTECH_FIREBASE_CONFIG } from './config';
10
11
  import { buildPath } from './utils/path-builder';
11
12
  import * as i0 from "@angular/core";
12
13
  import * as i1 from "@angular/fire/firestore";
@@ -46,6 +47,18 @@ import * as i1 from "@angular/fire/firestore";
46
47
  export class FirestoreService {
47
48
  constructor(firestore) {
48
49
  this.firestore = firestore;
50
+ this.config = inject(VALTECH_FIREBASE_CONFIG, { optional: true });
51
+ }
52
+ /**
53
+ * Prefija el path de colección con el appId si está configurado.
54
+ * Si no hay appId, retorna el path sin modificar (backward compatible).
55
+ *
56
+ * @internal
57
+ */
58
+ prefixCollectionPath(collectionPath) {
59
+ if (!this.config?.appId)
60
+ return collectionPath;
61
+ return `apps/${this.config.appId}/${collectionPath}`;
49
62
  }
50
63
  // ===========================================================================
51
64
  // LECTURAS ONE-TIME (Promise)
@@ -66,7 +79,8 @@ export class FirestoreService {
66
79
  * ```
67
80
  */
68
81
  async getDoc(collectionPath, docId) {
69
- const docRef = doc(this.firestore, collectionPath, docId);
82
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
83
+ const docRef = doc(this.firestore, prefixedPath, docId);
70
84
  const snapshot = await getDoc(docRef);
71
85
  if (!snapshot.exists()) {
72
86
  return null;
@@ -91,7 +105,8 @@ export class FirestoreService {
91
105
  * ```
92
106
  */
93
107
  async getDocs(collectionPath, options) {
94
- const collectionRef = collection(this.firestore, collectionPath);
108
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
109
+ const collectionRef = collection(this.firestore, prefixedPath);
95
110
  const constraints = this.buildQueryConstraints(options);
96
111
  const q = query(collectionRef, ...constraints);
97
112
  const snapshot = await getDocs(q);
@@ -123,7 +138,8 @@ export class FirestoreService {
123
138
  * ```
124
139
  */
125
140
  async getPaginated(collectionPath, options) {
126
- const collectionRef = collection(this.firestore, collectionPath);
141
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
142
+ const collectionRef = collection(this.firestore, prefixedPath);
127
143
  const constraints = this.buildQueryConstraints(options);
128
144
  // Pedir uno más para saber si hay más páginas
129
145
  const q = query(collectionRef, ...constraints, limit(options.limit + 1));
@@ -147,7 +163,8 @@ export class FirestoreService {
147
163
  * @returns true si el documento existe
148
164
  */
149
165
  async exists(collectionPath, docId) {
150
- const docRef = doc(this.firestore, collectionPath, docId);
166
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
167
+ const docRef = doc(this.firestore, prefixedPath, docId);
151
168
  const snapshot = await getDoc(docRef);
152
169
  return snapshot.exists();
153
170
  }
@@ -173,7 +190,8 @@ export class FirestoreService {
173
190
  * ```
174
191
  */
175
192
  docChanges(collectionPath, docId) {
176
- const docRef = doc(this.firestore, collectionPath, docId);
193
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
194
+ const docRef = doc(this.firestore, prefixedPath, docId);
177
195
  return docData(docRef, { idField: 'id' }).pipe(map((data) => {
178
196
  if (!data)
179
197
  return null;
@@ -196,7 +214,8 @@ export class FirestoreService {
196
214
  * ```
197
215
  */
198
216
  collectionChanges(collectionPath, options) {
199
- const collectionRef = collection(this.firestore, collectionPath);
217
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
218
+ const collectionRef = collection(this.firestore, prefixedPath);
200
219
  const constraints = this.buildQueryConstraints(options);
201
220
  const q = query(collectionRef, ...constraints);
202
221
  return collectionData(q, { idField: 'id' }).pipe(map((docs) => docs.map((doc) => this.convertTimestamps(doc))));
@@ -222,7 +241,8 @@ export class FirestoreService {
222
241
  * ```
223
242
  */
224
243
  async addDoc(collectionPath, data) {
225
- const collectionRef = collection(this.firestore, collectionPath);
244
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
245
+ const collectionRef = collection(this.firestore, prefixedPath);
226
246
  const timestamp = serverTimestamp();
227
247
  const docData = {
228
248
  ...data,
@@ -252,7 +272,8 @@ export class FirestoreService {
252
272
  * ```
253
273
  */
254
274
  async setDoc(collectionPath, docId, data, options) {
255
- const docRef = doc(this.firestore, collectionPath, docId);
275
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
276
+ const docRef = doc(this.firestore, prefixedPath, docId);
256
277
  const timestamp = serverTimestamp();
257
278
  const docData = {
258
279
  ...data,
@@ -277,7 +298,8 @@ export class FirestoreService {
277
298
  * ```
278
299
  */
279
300
  async updateDoc(collectionPath, docId, data) {
280
- const docRef = doc(this.firestore, collectionPath, docId);
301
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
302
+ const docRef = doc(this.firestore, prefixedPath, docId);
281
303
  await updateDoc(docRef, {
282
304
  ...data,
283
305
  updatedAt: serverTimestamp(),
@@ -295,7 +317,8 @@ export class FirestoreService {
295
317
  * ```
296
318
  */
297
319
  async deleteDoc(collectionPath, docId) {
298
- const docRef = doc(this.firestore, collectionPath, docId);
320
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
321
+ const docRef = doc(this.firestore, prefixedPath, docId);
299
322
  await deleteDoc(docRef);
300
323
  }
301
324
  // ===========================================================================
@@ -320,7 +343,8 @@ export class FirestoreService {
320
343
  const batchApi = {
321
344
  set: (path, data) => {
322
345
  const [collectionPath, docId] = this.splitPath(path);
323
- const docRef = doc(this.firestore, collectionPath, docId);
346
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
347
+ const docRef = doc(this.firestore, prefixedPath, docId);
324
348
  batch.set(docRef, {
325
349
  ...data,
326
350
  createdAt: serverTimestamp(),
@@ -329,7 +353,8 @@ export class FirestoreService {
329
353
  },
330
354
  update: (path, data) => {
331
355
  const [collectionPath, docId] = this.splitPath(path);
332
- const docRef = doc(this.firestore, collectionPath, docId);
356
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
357
+ const docRef = doc(this.firestore, prefixedPath, docId);
333
358
  batch.update(docRef, {
334
359
  ...data,
335
360
  updatedAt: serverTimestamp(),
@@ -337,7 +362,8 @@ export class FirestoreService {
337
362
  },
338
363
  delete: (path) => {
339
364
  const [collectionPath, docId] = this.splitPath(path);
340
- const docRef = doc(this.firestore, collectionPath, docId);
365
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
366
+ const docRef = doc(this.firestore, prefixedPath, docId);
341
367
  batch.delete(docRef);
342
368
  },
343
369
  };
@@ -373,7 +399,8 @@ export class FirestoreService {
373
399
  * @returns ID único generado por Firestore
374
400
  */
375
401
  generateId(collectionPath) {
376
- const collectionRef = collection(this.firestore, collectionPath);
402
+ const prefixedPath = this.prefixCollectionPath(collectionPath);
403
+ const collectionRef = collection(this.firestore, prefixedPath);
377
404
  return doc(collectionRef).id;
378
405
  }
379
406
  /**
@@ -506,4 +533,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
506
533
  type: Injectable,
507
534
  args: [{ providedIn: 'root' }]
508
535
  }], ctorParameters: () => [{ type: i1.Firestore }] });
509
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"firestore.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/firestore.service.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,EACL,MAAM,EACN,UAAU,EACV,cAAc,EACd,SAAS,EACT,GAAG,EACH,OAAO,EAKP,MAAM,EACN,OAAO,EACP,KAAK,EACL,OAAO,EACP,KAAK,EAEL,eAAe,EACf,MAAM,EACN,UAAU,EACV,OAAO,EACP,SAAS,EACT,KAAK,EACL,SAAS,EACT,SAAS,EACT,KAAK,EACL,UAAU,EACV,SAAS,EACT,UAAU,EACV,WAAW,GAEZ,MAAM,yBAAyB,CAAC;AACjC,OAAO,EAAE,GAAG,EAAc,MAAM,MAAM,CAAC;AAGvC,OAAO,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;;;AAEjD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgCG;AAEH,MAAM,OAAO,gBAAgB;IAC3B,YAAoB,SAAoB;QAApB,cAAS,GAAT,SAAS,CAAW;IAAG,CAAC;IAE5C,8EAA8E;IAC9E,8BAA8B;IAC9B,8EAA8E;IAE9E;;;;;;;;;;;;;;OAcG;IACH,KAAK,CAAC,MAAM,CACV,cAAsB,EACtB,KAAa;QAEb,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,EAAE,KAAK,CAAC,CAAC;QAC1D,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,CAAC;QAEtC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,EAAE,CAAC;YACvB,OAAO,IAAI,CAAC;QACd,CAAC;QAED,OAAO,IAAI,CAAC,WAAW,CAAI,QAAQ,CAAC,CAAC;IACvC,CAAC;IAED;;;;;;;;;;;;;;;;OAgBG;IACH,KAAK,CAAC,OAAO,CACX,cAAsB,EACtB,OAAsB;QAEtB,MAAM,aAAa,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,CAAC,CAAC;QACjE,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC;QACxD,MAAM,CAAC,GAAG,KAAK,CAAC,aAAa,EAAE,GAAG,WAAW,CAAC,CAAC;QAC/C,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,CAAC,CAAC,CAAC;QAElC,OAAO,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAI,GAAG,CAAC,CAAC,CAAC;IAC9D,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;OAwBG;IACH,KAAK,CAAC,YAAY,CAChB,cAAsB,EACtB,OAAyC;QAEzC,MAAM,aAAa,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,CAAC,CAAC;QACjE,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC;QAExD,8CAA8C;QAC9C,MAAM,CAAC,GAAG,KAAK,CAAC,aAAa,EAAE,GAAG,WAAW,EAAE,KAAK,CAAC,OAAO,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC;QACzE,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,CAAC,CAAC,CAAC;QAElC,MAAM,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC;QAC3B,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC;QAE5C,yCAAyC;QACzC,MAAM,UAAU,GAAG,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QACtD,MAAM,OAAO,GAAG,UAAU,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAEjF,OAAO;YACL,IAAI,EAAE,UAAU,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAI,GAAG,CAAC,CAAC;YACvD,OAAO;YACP,OAAO;SACR,CAAC;IACJ,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,MAAM,CAAC,cAAsB,EAAE,KAAa;QAChD,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,EAAE,KAAK,CAAC,CAAC;QAC1D,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,CAAC;QACtC,OAAO,QAAQ,CAAC,MAAM,EAAE,CAAC;IAC3B,CAAC;IAED,8EAA8E;IAC9E,wCAAwC;IACxC,8EAA8E;IAE9E;;;;;;;;;;;;;;;;;OAiBG;IACH,UAAU,CACR,cAAsB,EACtB,KAAa;QAEb,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,EAAE,KAAK,CAAC,CAAC;QAC1D,OAAO,OAAO,CAAC,MAAM,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,CAC5C,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE;YACX,IAAI,CAAC,IAAI;gBAAE,OAAO,IAAI,CAAC;YACvB,OAAO,IAAI,CAAC,iBAAiB,CAAC,IAAoB,CAAM,CAAC;QAC3D,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED;;;;;;;;;;;;;;OAcG;IACH,iBAAiB,CACf,cAAsB,EACtB,OAAsB;QAEtB,MAAM,aAAa,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,CAAC,CAAC;QACjE,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC;QACxD,MAAM,CAAC,GAAG,KAAK,CAAC,aAAa,EAAE,GAAG,WAAW,CAAC,CAAC;QAE/C,OAAO,cAAc,CAAC,CAAC,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,CAC9C,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAM,CAAC,CAAC,CACnE,CAAC;IACJ,CAAC;IAED,8EAA8E;IAC9E,YAAY;IACZ,8EAA8E;IAE9E;;;;;;;;;;;;;;;;OAgBG;IACH,KAAK,CAAC,MAAM,CACV,cAAsB,EACtB,IAA+C;QAE/C,MAAM,aAAa,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,CAAC,CAAC;QACjE,MAAM,SAAS,GAAG,eAAe,EAAE,CAAC;QAEpC,MAAM,OAAO,GAAG;YACd,GAAG,IAAI;YACP,SAAS,EAAE,SAAS;YACpB,SAAS,EAAE,SAAS;SACrB,CAAC;QAEF,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,aAAa,EAAE,OAAO,CAAC,CAAC;QAEpD,uEAAuE;QACvE,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,CAAC;QACtC,OAAO,IAAI,CAAC,WAAW,CAAI,QAAQ,CAAC,CAAC;IACvC,CAAC;IAED;;;;;;;;;;;;;;;;OAgBG;IACH,KAAK,CAAC,MAAM,CACV,cAAsB,EACtB,KAAa,EACb,IAAmB,EACnB,OAA6B;QAE7B,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,EAAE,KAAK,CAAC,CAAC;QAC1D,MAAM,SAAS,GAAG,eAAe,EAAE,CAAC;QAEpC,MAAM,OAAO,GAAG;YACd,GAAG,IAAI;YACP,SAAS,EAAE,SAAS;YACpB,GAAG,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,SAAS,EAAE,SAAS,EAAE,CAAC;SACpD,CAAC;QAEF,MAAM,MAAM,CAAC,MAAM,EAAE,OAAO,EAAE,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,IAAI,KAAK,EAAE,CAAC,CAAC;IACpE,CAAC;IAED;;;;;;;;;;;;;;OAcG;IACH,KAAK,CAAC,SAAS,CACb,cAAsB,EACtB,KAAa,EACb,IAA0C;QAE1C,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,EAAE,KAAK,CAAC,CAAC;QAE1D,MAAM,SAAS,CAAC,MAAM,EAAE;YACtB,GAAG,IAAI;YACP,SAAS,EAAE,eAAe,EAAE;SAC7B,CAAC,CAAC;IACL,CAAC;IAED;;;;;;;;;;OAUG;IACH,KAAK,CAAC,SAAS,CAAC,cAAsB,EAAE,KAAa;QACnD,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,EAAE,KAAK,CAAC,CAAC;QAC1D,MAAM,SAAS,CAAC,MAAM,CAAC,CAAC;IAC1B,CAAC;IAED,8EAA8E;IAC9E,sBAAsB;IACtB,8EAA8E;IAE9E;;;;;;;;;;;;;OAaG;IACH,KAAK,CAAC,KAAK,CACT,UAIU;QAEV,MAAM,KAAK,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAEzC,MAAM,QAAQ,GAAG;YACf,GAAG,EAAE,CAAI,IAAY,EAAE,IAAO,EAAE,EAAE;gBAChC,MAAM,CAAC,cAAc,EAAE,KAAK,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;gBACrD,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,EAAE,KAAK,CAAC,CAAC;gBAC1D,KAAK,CAAC,GAAG,CAAC,MAAM,EAAE;oBAChB,GAAG,IAAI;oBACP,SAAS,EAAE,eAAe,EAAE;oBAC5B,SAAS,EAAE,eAAe,EAAE;iBACb,CAAC,CAAC;YACrB,CAAC;YACD,MAAM,EAAE,CAAI,IAAY,EAAE,IAAgB,EAAE,EAAE;gBAC5C,MAAM,CAAC,cAAc,EAAE,KAAK,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;gBACrD,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,EAAE,KAAK,CAAC,CAAC;gBAC1D,KAAK,CAAC,MAAM,CAAC,MAAM,EAAE;oBACnB,GAAG,IAAI;oBACP,SAAS,EAAE,eAAe,EAAE;iBACb,CAAC,CAAC;YACrB,CAAC;YACD,MAAM,EAAE,CAAC,IAAY,EAAE,EAAE;gBACvB,MAAM,CAAC,cAAc,EAAE,KAAK,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;gBACrD,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,EAAE,KAAK,CAAC,CAAC;gBAC1D,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YACvB,CAAC;SACF,CAAC;QAEF,UAAU,CAAC,QAAQ,CAAC,CAAC;QACrB,MAAM,KAAK,CAAC,MAAM,EAAE,CAAC;IACvB,CAAC;IAED,8EAA8E;IAC9E,aAAa;IACb,8EAA8E;IAE9E;;;;;;;;;;;;;;;OAeG;IACH,SAAS,CAAC,QAAgB,EAAE,MAA8B;QACxD,OAAO,SAAS,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;IACrC,CAAC;IAED;;;;;OAKG;IACH,UAAU,CAAC,cAAsB;QAC/B,MAAM,aAAa,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,CAAC,CAAC;QACjE,OAAO,GAAG,CAAC,aAAa,CAAC,CAAC,EAAE,CAAC;IAC/B,CAAC;IAED;;;OAGG;IACH,eAAe;QACb,OAAO,eAAe,EAAE,CAAC;IAC3B,CAAC;IAED;;;;;;;;;OASG;IACH,UAAU,CAAC,GAAG,QAAmB;QAC/B,OAAO,UAAU,CAAC,GAAG,QAAQ,CAAC,CAAC;IACjC,CAAC;IAED;;OAEG;IACH,WAAW,CAAC,GAAG,QAAmB;QAChC,OAAO,WAAW,CAAC,GAAG,QAAQ,CAAC,CAAC;IAClC,CAAC;IAED;;;;;;;;;OASG;IACH,SAAS,CAAC,CAAS;QACjB,OAAO,SAAS,CAAC,CAAC,CAAC,CAAC;IACtB,CAAC;IAED,8EAA8E;IAC9E,mBAAmB;IACnB,8EAA8E;IAE9E;;OAEG;IACK,qBAAqB,CAAC,OAAsB;QAClD,MAAM,WAAW,GAAsB,EAAE,CAAC;QAE1C,IAAI,CAAC,OAAO;YAAE,OAAO,WAAW,CAAC;QAEjC,gBAAgB;QAChB,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;YAClB,KAAK,MAAM,MAAM,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;gBACnC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,QAAQ,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;YACvE,CAAC;QACH,CAAC;QAED,kBAAkB;QAClB,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpB,KAAK,MAAM,MAAM,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;gBACrC,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC;YAC5D,CAAC;QACH,CAAC;QAED,0BAA0B;QAC1B,IAAI,OAAO,CAAC,UAAU,EAAE,CAAC;YACvB,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC;QACnD,CAAC;QACD,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpB,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC;QAC7C,CAAC;QACD,IAAI,OAAO,CAAC,SAAS,EAAE,CAAC;YACtB,WAAW,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,CAAC;QACjD,CAAC;QACD,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;YAClB,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC;QACzC,CAAC;QAED,6BAA6B;QAC7B,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;YAClB,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC;QACzC,CAAC;QAED,OAAO,WAAW,CAAC;IACrB,CAAC;IAED;;OAEG;IACK,WAAW,CACjB,QAAwC;QAExC,MAAM,IAAI,GAAG,QAAQ,CAAC,IAAI,EAAE,CAAC;QAC7B,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,MAAM,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC;QAC9C,CAAC;QAED,OAAO;YACL,EAAE,EAAE,QAAQ,CAAC,EAAE;YACf,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC;SAC3B,CAAC;IACT,CAAC;IAED;;OAEG;IACK,iBAAiB,CAAC,IAAkB;QAC1C,MAAM,MAAM,GAAiB,EAAE,CAAC;QAEhC,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;YAChD,IAAI,KAAK,YAAY,SAAS,EAAE,CAAC;gBAC/B,MAAM,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC;YAC/B,CAAC;iBAAM,IAAI,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;gBACvE,MAAM,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC;YAC9C,CAAC;iBAAM,CAAC;gBACN,MAAM,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;YACtB,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACK,SAAS,CAAC,IAAY;QAC5B,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACjC,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;YACrD,MAAM,IAAI,KAAK,CAAC,+BAA+B,IAAI,EAAE,CAAC,CAAC;QACzD,CAAC;QACD,MAAM,KAAK,GAAG,QAAQ,CAAC,GAAG,EAAG,CAAC;QAC9B,MAAM,cAAc,GAAG,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC1C,OAAO,CAAC,cAAc,EAAE,KAAK,CAAC,CAAC;IACjC,CAAC;+GA5hBU,gBAAgB;mHAAhB,gBAAgB,cADH,MAAM;;4FACnB,gBAAgB;kBAD5B,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE","sourcesContent":["/**\n * Firestore Service\n *\n * Servicio genérico para operaciones CRUD en Firestore.\n * Soporta lecturas one-time, subscripciones real-time, paginación y queries complejas.\n */\n\nimport { Injectable } from '@angular/core';\nimport {\n  addDoc,\n  collection,\n  collectionData,\n  deleteDoc,\n  doc,\n  docData,\n  DocumentData,\n  DocumentReference,\n  DocumentSnapshot,\n  Firestore,\n  getDoc,\n  getDocs,\n  limit,\n  orderBy,\n  query,\n  QueryConstraint,\n  serverTimestamp,\n  setDoc,\n  startAfter,\n  startAt,\n  endBefore,\n  endAt,\n  Timestamp,\n  updateDoc,\n  where,\n  writeBatch,\n  increment,\n  arrayUnion,\n  arrayRemove,\n  FieldValue,\n} from '@angular/fire/firestore';\nimport { map, Observable } from 'rxjs';\n\nimport { FirestoreDocument, PaginatedResult, QueryOptions } from './types';\nimport { buildPath } from './utils/path-builder';\n\n/**\n * Servicio para operaciones CRUD en Firestore.\n *\n * @example\n * ```typescript\n * interface User extends FirestoreDocument {\n *   name: string;\n *   email: string;\n *   role: 'admin' | 'user';\n * }\n *\n * @Component({...})\n * export class UsersComponent {\n *   private firestore = inject(FirestoreService);\n *\n *   // Lectura one-time\n *   async loadUser(id: string) {\n *     const user = await this.firestore.getDoc<User>('users', id);\n *   }\n *\n *   // Subscripción real-time\n *   users$ = this.firestore.collectionChanges<User>('users', {\n *     where: [{ field: 'role', operator: '==', value: 'admin' }],\n *     orderBy: [{ field: 'name', direction: 'asc' }]\n *   });\n *\n *   // Crear documento\n *   async createUser(data: Omit<User, 'id'>) {\n *     const user = await this.firestore.addDoc<User>('users', data);\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class FirestoreService {\n  constructor(private firestore: Firestore) {}\n\n  // ===========================================================================\n  // LECTURAS ONE-TIME (Promise)\n  // ===========================================================================\n\n  /**\n   * Obtiene un documento por ID (lectura única).\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @returns Documento o null si no existe\n   *\n   * @example\n   * ```typescript\n   * const user = await firestoreService.getDoc<User>('users', 'abc123');\n   * if (user) {\n   *   console.log(user.name);\n   * }\n   * ```\n   */\n  async getDoc<T extends FirestoreDocument>(\n    collectionPath: string,\n    docId: string\n  ): Promise<T | null> {\n    const docRef = doc(this.firestore, collectionPath, docId);\n    const snapshot = await getDoc(docRef);\n\n    if (!snapshot.exists()) {\n      return null;\n    }\n\n    return this.mapDocument<T>(snapshot);\n  }\n\n  /**\n   * Obtiene múltiples documentos con opciones de query.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param options - Opciones de query (where, orderBy, limit)\n   * @returns Array de documentos\n   *\n   * @example\n   * ```typescript\n   * // Todos los usuarios activos ordenados por nombre\n   * const users = await firestoreService.getDocs<User>('users', {\n   *   where: [{ field: 'active', operator: '==', value: true }],\n   *   orderBy: [{ field: 'name', direction: 'asc' }],\n   *   limit: 50\n   * });\n   * ```\n   */\n  async getDocs<T extends FirestoreDocument>(\n    collectionPath: string,\n    options?: QueryOptions\n  ): Promise<T[]> {\n    const collectionRef = collection(this.firestore, collectionPath);\n    const constraints = this.buildQueryConstraints(options);\n    const q = query(collectionRef, ...constraints);\n    const snapshot = await getDocs(q);\n\n    return snapshot.docs.map((doc) => this.mapDocument<T>(doc));\n  }\n\n  /**\n   * Obtiene documentos con paginación basada en cursores.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param options - Opciones de query (debe incluir limit)\n   * @returns Resultado paginado con cursor para la siguiente página\n   *\n   * @example\n   * ```typescript\n   * // Primera página\n   * const page1 = await firestoreService.getPaginated<User>('users', {\n   *   orderBy: [{ field: 'createdAt', direction: 'desc' }],\n   *   limit: 10\n   * });\n   *\n   * // Siguiente página\n   * if (page1.hasMore) {\n   *   const page2 = await firestoreService.getPaginated<User>('users', {\n   *     orderBy: [{ field: 'createdAt', direction: 'desc' }],\n   *     limit: 10,\n   *     startAfter: page1.lastDoc\n   *   });\n   * }\n   * ```\n   */\n  async getPaginated<T extends FirestoreDocument>(\n    collectionPath: string,\n    options: QueryOptions & { limit: number }\n  ): Promise<PaginatedResult<T>> {\n    const collectionRef = collection(this.firestore, collectionPath);\n    const constraints = this.buildQueryConstraints(options);\n\n    // Pedir uno más para saber si hay más páginas\n    const q = query(collectionRef, ...constraints, limit(options.limit + 1));\n    const snapshot = await getDocs(q);\n\n    const docs = snapshot.docs;\n    const hasMore = docs.length > options.limit;\n\n    // Si hay más, remover el documento extra\n    const resultDocs = hasMore ? docs.slice(0, -1) : docs;\n    const lastDoc = resultDocs.length > 0 ? resultDocs[resultDocs.length - 1] : null;\n\n    return {\n      data: resultDocs.map((doc) => this.mapDocument<T>(doc)),\n      hasMore,\n      lastDoc,\n    };\n  }\n\n  /**\n   * Verifica si un documento existe.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @returns true si el documento existe\n   */\n  async exists(collectionPath: string, docId: string): Promise<boolean> {\n    const docRef = doc(this.firestore, collectionPath, docId);\n    const snapshot = await getDoc(docRef);\n    return snapshot.exists();\n  }\n\n  // ===========================================================================\n  // SUBSCRIPCIONES REAL-TIME (Observable)\n  // ===========================================================================\n\n  /**\n   * Suscribe a cambios de un documento (real-time).\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @returns Observable que emite cuando el documento cambia\n   *\n   * @example\n   * ```typescript\n   * // En el componente\n   * user$ = this.firestoreService.docChanges<User>('users', this.userId);\n   *\n   * // En el template\n   * @if (user$ | async; as user) {\n   *   <p>{{ user.name }}</p>\n   * }\n   * ```\n   */\n  docChanges<T extends FirestoreDocument>(\n    collectionPath: string,\n    docId: string\n  ): Observable<T | null> {\n    const docRef = doc(this.firestore, collectionPath, docId);\n    return docData(docRef, { idField: 'id' }).pipe(\n      map((data) => {\n        if (!data) return null;\n        return this.convertTimestamps(data as DocumentData) as T;\n      })\n    );\n  }\n\n  /**\n   * Suscribe a cambios de una colección (real-time).\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param options - Opciones de query\n   * @returns Observable que emite cuando la colección cambia\n   *\n   * @example\n   * ```typescript\n   * // Usuarios activos en tiempo real\n   * activeUsers$ = this.firestoreService.collectionChanges<User>('users', {\n   *   where: [{ field: 'status', operator: '==', value: 'online' }]\n   * });\n   * ```\n   */\n  collectionChanges<T extends FirestoreDocument>(\n    collectionPath: string,\n    options?: QueryOptions\n  ): Observable<T[]> {\n    const collectionRef = collection(this.firestore, collectionPath);\n    const constraints = this.buildQueryConstraints(options);\n    const q = query(collectionRef, ...constraints);\n\n    return collectionData(q, { idField: 'id' }).pipe(\n      map((docs) => docs.map((doc) => this.convertTimestamps(doc) as T))\n    );\n  }\n\n  // ===========================================================================\n  // ESCRITURA\n  // ===========================================================================\n\n  /**\n   * Agrega un documento con ID auto-generado.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param data - Datos del documento (sin id, createdAt, updatedAt)\n   * @returns Documento creado con su ID\n   *\n   * @example\n   * ```typescript\n   * const newUser = await firestoreService.addDoc<User>('users', {\n   *   name: 'John Doe',\n   *   email: 'john@example.com',\n   *   role: 'user'\n   * });\n   * console.log('Created user with ID:', newUser.id);\n   * ```\n   */\n  async addDoc<T extends FirestoreDocument>(\n    collectionPath: string,\n    data: Omit<T, 'id' | 'createdAt' | 'updatedAt'>\n  ): Promise<T> {\n    const collectionRef = collection(this.firestore, collectionPath);\n    const timestamp = serverTimestamp();\n\n    const docData = {\n      ...data,\n      createdAt: timestamp,\n      updatedAt: timestamp,\n    };\n\n    const docRef = await addDoc(collectionRef, docData);\n\n    // Obtener el documento creado para retornarlo con timestamps resueltos\n    const snapshot = await getDoc(docRef);\n    return this.mapDocument<T>(snapshot);\n  }\n\n  /**\n   * Crea o sobrescribe un documento con ID específico.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @param data - Datos del documento\n   * @param options - Opciones (merge: true para merge en lugar de sobrescribir)\n   *\n   * @example\n   * ```typescript\n   * // Sobrescribir completamente\n   * await firestoreService.setDoc<User>('users', 'user123', userData);\n   *\n   * // Merge con datos existentes\n   * await firestoreService.setDoc<User>('users', 'user123', { name: 'New Name' }, { merge: true });\n   * ```\n   */\n  async setDoc<T extends FirestoreDocument>(\n    collectionPath: string,\n    docId: string,\n    data: Omit<T, 'id'>,\n    options?: { merge?: boolean }\n  ): Promise<void> {\n    const docRef = doc(this.firestore, collectionPath, docId);\n    const timestamp = serverTimestamp();\n\n    const docData = {\n      ...data,\n      updatedAt: timestamp,\n      ...(options?.merge ? {} : { createdAt: timestamp }),\n    };\n\n    await setDoc(docRef, docData, { merge: options?.merge ?? false });\n  }\n\n  /**\n   * Actualiza campos específicos de un documento.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @param data - Campos a actualizar\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.updateDoc<User>('users', 'user123', {\n   *   name: 'Updated Name',\n   *   lastLogin: new Date()\n   * });\n   * ```\n   */\n  async updateDoc<T extends FirestoreDocument>(\n    collectionPath: string,\n    docId: string,\n    data: Partial<Omit<T, 'id' | 'createdAt'>>\n  ): Promise<void> {\n    const docRef = doc(this.firestore, collectionPath, docId);\n\n    await updateDoc(docRef, {\n      ...data,\n      updatedAt: serverTimestamp(),\n    });\n  }\n\n  /**\n   * Elimina un documento.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.deleteDoc('users', 'user123');\n   * ```\n   */\n  async deleteDoc(collectionPath: string, docId: string): Promise<void> {\n    const docRef = doc(this.firestore, collectionPath, docId);\n    await deleteDoc(docRef);\n  }\n\n  // ===========================================================================\n  // OPERACIONES EN LOTE\n  // ===========================================================================\n\n  /**\n   * Ejecuta múltiples operaciones de escritura de forma atómica.\n   *\n   * @param operations - Función que recibe el batch y agrega operaciones\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.batch((batch) => {\n   *   batch.set('users/user1', { name: 'User 1' });\n   *   batch.update('users/user2', { status: 'inactive' });\n   *   batch.delete('users/user3');\n   * });\n   * ```\n   */\n  async batch(\n    operations: (batch: {\n      set: <T>(path: string, data: T) => void;\n      update: <T>(path: string, data: Partial<T>) => void;\n      delete: (path: string) => void;\n    }) => void\n  ): Promise<void> {\n    const batch = writeBatch(this.firestore);\n\n    const batchApi = {\n      set: <T>(path: string, data: T) => {\n        const [collectionPath, docId] = this.splitPath(path);\n        const docRef = doc(this.firestore, collectionPath, docId);\n        batch.set(docRef, {\n          ...data,\n          createdAt: serverTimestamp(),\n          updatedAt: serverTimestamp(),\n        } as DocumentData);\n      },\n      update: <T>(path: string, data: Partial<T>) => {\n        const [collectionPath, docId] = this.splitPath(path);\n        const docRef = doc(this.firestore, collectionPath, docId);\n        batch.update(docRef, {\n          ...data,\n          updatedAt: serverTimestamp(),\n        } as DocumentData);\n      },\n      delete: (path: string) => {\n        const [collectionPath, docId] = this.splitPath(path);\n        const docRef = doc(this.firestore, collectionPath, docId);\n        batch.delete(docRef);\n      },\n    };\n\n    operations(batchApi);\n    await batch.commit();\n  }\n\n  // ===========================================================================\n  // UTILIDADES\n  // ===========================================================================\n\n  /**\n   * Construye una ruta a partir de un template.\n   *\n   * @param template - Template con placeholders {param}\n   * @param params - Valores para los placeholders\n   * @returns Ruta construida\n   *\n   * @example\n   * ```typescript\n   * const path = firestoreService.buildPath('users/{userId}/documents/{docId}', {\n   *   userId: 'user123',\n   *   docId: 'doc456'\n   * });\n   * // => 'users/user123/documents/doc456'\n   * ```\n   */\n  buildPath(template: string, params: Record<string, string>): string {\n    return buildPath(template, params);\n  }\n\n  /**\n   * Genera un ID único para un documento (sin crearlo).\n   *\n   * @param collectionPath - Ruta de la colección\n   * @returns ID único generado por Firestore\n   */\n  generateId(collectionPath: string): string {\n    const collectionRef = collection(this.firestore, collectionPath);\n    return doc(collectionRef).id;\n  }\n\n  /**\n   * Retorna un valor de timestamp del servidor.\n   * Usar en campos de fecha para que Firestore asigne el timestamp.\n   */\n  serverTimestamp(): FieldValue {\n    return serverTimestamp();\n  }\n\n  /**\n   * Retorna un valor para agregar elementos a un array.\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.updateDoc('users', 'user123', {\n   *   tags: firestoreService.arrayUnion('new-tag')\n   * });\n   * ```\n   */\n  arrayUnion(...elements: unknown[]): FieldValue {\n    return arrayUnion(...elements);\n  }\n\n  /**\n   * Retorna un valor para remover elementos de un array.\n   */\n  arrayRemove(...elements: unknown[]): FieldValue {\n    return arrayRemove(...elements);\n  }\n\n  /**\n   * Retorna un valor para incrementar un campo numérico.\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.updateDoc('users', 'user123', {\n   *   loginCount: firestoreService.increment(1)\n   * });\n   * ```\n   */\n  increment(n: number): FieldValue {\n    return increment(n);\n  }\n\n  // ===========================================================================\n  // MÉTODOS PRIVADOS\n  // ===========================================================================\n\n  /**\n   * Construye los QueryConstraints a partir de QueryOptions\n   */\n  private buildQueryConstraints(options?: QueryOptions): QueryConstraint[] {\n    const constraints: QueryConstraint[] = [];\n\n    if (!options) return constraints;\n\n    // Where clauses\n    if (options.where) {\n      for (const clause of options.where) {\n        constraints.push(where(clause.field, clause.operator, clause.value));\n      }\n    }\n\n    // OrderBy clauses\n    if (options.orderBy) {\n      for (const clause of options.orderBy) {\n        constraints.push(orderBy(clause.field, clause.direction));\n      }\n    }\n\n    // Cursors para paginación\n    if (options.startAfter) {\n      constraints.push(startAfter(options.startAfter));\n    }\n    if (options.startAt) {\n      constraints.push(startAt(options.startAt));\n    }\n    if (options.endBefore) {\n      constraints.push(endBefore(options.endBefore));\n    }\n    if (options.endAt) {\n      constraints.push(endAt(options.endAt));\n    }\n\n    // Limit (se agrega al final)\n    if (options.limit) {\n      constraints.push(limit(options.limit));\n    }\n\n    return constraints;\n  }\n\n  /**\n   * Mapea un DocumentSnapshot a nuestro tipo\n   */\n  private mapDocument<T extends FirestoreDocument>(\n    snapshot: DocumentSnapshot<DocumentData>\n  ): T {\n    const data = snapshot.data();\n    if (!data) {\n      throw new Error('Documento no tiene datos');\n    }\n\n    return {\n      id: snapshot.id,\n      ...this.convertTimestamps(data),\n    } as T;\n  }\n\n  /**\n   * Convierte Timestamps de Firestore a Date de JavaScript\n   */\n  private convertTimestamps(data: DocumentData): DocumentData {\n    const result: DocumentData = {};\n\n    for (const [key, value] of Object.entries(data)) {\n      if (value instanceof Timestamp) {\n        result[key] = value.toDate();\n      } else if (value && typeof value === 'object' && !Array.isArray(value)) {\n        result[key] = this.convertTimestamps(value);\n      } else {\n        result[key] = value;\n      }\n    }\n\n    return result;\n  }\n\n  /**\n   * Divide una ruta de documento en colección e ID\n   */\n  private splitPath(path: string): [string, string] {\n    const segments = path.split('/');\n    if (segments.length < 2 || segments.length % 2 !== 0) {\n      throw new Error(`Ruta de documento inválida: ${path}`);\n    }\n    const docId = segments.pop()!;\n    const collectionPath = segments.join('/');\n    return [collectionPath, docId];\n  }\n}\n"]}
536
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"firestore.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/firestore.service.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AACnD,OAAO,EACL,MAAM,EACN,UAAU,EACV,cAAc,EACd,SAAS,EACT,GAAG,EACH,OAAO,EAKP,MAAM,EACN,OAAO,EACP,KAAK,EACL,OAAO,EACP,KAAK,EAEL,eAAe,EACf,MAAM,EACN,UAAU,EACV,OAAO,EACP,SAAS,EACT,KAAK,EACL,SAAS,EACT,SAAS,EACT,KAAK,EACL,UAAU,EACV,SAAS,EACT,UAAU,EACV,WAAW,GAEZ,MAAM,yBAAyB,CAAC;AACjC,OAAO,EAAE,GAAG,EAAc,MAAM,MAAM,CAAC;AAEvC,OAAO,EAAE,uBAAuB,EAAE,MAAM,UAAU,CAAC;AAEnD,OAAO,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;;;AAEjD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgCG;AAEH,MAAM,OAAO,gBAAgB;IAG3B,YAAoB,SAAoB;QAApB,cAAS,GAAT,SAAS,CAAW;QAFhC,WAAM,GAAG,MAAM,CAAC,uBAAuB,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;IAE1B,CAAC;IAE5C;;;;;OAKG;IACK,oBAAoB,CAAC,cAAsB;QACjD,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,KAAK;YAAE,OAAO,cAAc,CAAC;QAC/C,OAAO,QAAQ,IAAI,CAAC,MAAM,CAAC,KAAK,IAAI,cAAc,EAAE,CAAC;IACvD,CAAC;IAED,8EAA8E;IAC9E,8BAA8B;IAC9B,8EAA8E;IAE9E;;;;;;;;;;;;;;OAcG;IACH,KAAK,CAAC,MAAM,CACV,cAAsB,EACtB,KAAa;QAEb,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;QAC/D,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,EAAE,KAAK,CAAC,CAAC;QACxD,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,CAAC;QAEtC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,EAAE,CAAC;YACvB,OAAO,IAAI,CAAC;QACd,CAAC;QAED,OAAO,IAAI,CAAC,WAAW,CAAI,QAAQ,CAAC,CAAC;IACvC,CAAC;IAED;;;;;;;;;;;;;;;;OAgBG;IACH,KAAK,CAAC,OAAO,CACX,cAAsB,EACtB,OAAsB;QAEtB,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;QAC/D,MAAM,aAAa,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;QAC/D,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC;QACxD,MAAM,CAAC,GAAG,KAAK,CAAC,aAAa,EAAE,GAAG,WAAW,CAAC,CAAC;QAC/C,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,CAAC,CAAC,CAAC;QAElC,OAAO,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAI,GAAG,CAAC,CAAC,CAAC;IAC9D,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;OAwBG;IACH,KAAK,CAAC,YAAY,CAChB,cAAsB,EACtB,OAAyC;QAEzC,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;QAC/D,MAAM,aAAa,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;QAC/D,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC;QAExD,8CAA8C;QAC9C,MAAM,CAAC,GAAG,KAAK,CAAC,aAAa,EAAE,GAAG,WAAW,EAAE,KAAK,CAAC,OAAO,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC;QACzE,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,CAAC,CAAC,CAAC;QAElC,MAAM,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC;QAC3B,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC;QAE5C,yCAAyC;QACzC,MAAM,UAAU,GAAG,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QACtD,MAAM,OAAO,GAAG,UAAU,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAEjF,OAAO;YACL,IAAI,EAAE,UAAU,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAI,GAAG,CAAC,CAAC;YACvD,OAAO;YACP,OAAO;SACR,CAAC;IACJ,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,MAAM,CAAC,cAAsB,EAAE,KAAa;QAChD,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;QAC/D,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,EAAE,KAAK,CAAC,CAAC;QACxD,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,CAAC;QACtC,OAAO,QAAQ,CAAC,MAAM,EAAE,CAAC;IAC3B,CAAC;IAED,8EAA8E;IAC9E,wCAAwC;IACxC,8EAA8E;IAE9E;;;;;;;;;;;;;;;;;OAiBG;IACH,UAAU,CACR,cAAsB,EACtB,KAAa;QAEb,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;QAC/D,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,EAAE,KAAK,CAAC,CAAC;QACxD,OAAO,OAAO,CAAC,MAAM,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,CAC5C,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE;YACX,IAAI,CAAC,IAAI;gBAAE,OAAO,IAAI,CAAC;YACvB,OAAO,IAAI,CAAC,iBAAiB,CAAC,IAAoB,CAAM,CAAC;QAC3D,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED;;;;;;;;;;;;;;OAcG;IACH,iBAAiB,CACf,cAAsB,EACtB,OAAsB;QAEtB,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;QAC/D,MAAM,aAAa,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;QAC/D,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC;QACxD,MAAM,CAAC,GAAG,KAAK,CAAC,aAAa,EAAE,GAAG,WAAW,CAAC,CAAC;QAE/C,OAAO,cAAc,CAAC,CAAC,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,CAC9C,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAM,CAAC,CAAC,CACnE,CAAC;IACJ,CAAC;IAED,8EAA8E;IAC9E,YAAY;IACZ,8EAA8E;IAE9E;;;;;;;;;;;;;;;;OAgBG;IACH,KAAK,CAAC,MAAM,CACV,cAAsB,EACtB,IAA+C;QAE/C,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;QAC/D,MAAM,aAAa,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;QAC/D,MAAM,SAAS,GAAG,eAAe,EAAE,CAAC;QAEpC,MAAM,OAAO,GAAG;YACd,GAAG,IAAI;YACP,SAAS,EAAE,SAAS;YACpB,SAAS,EAAE,SAAS;SACrB,CAAC;QAEF,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,aAAa,EAAE,OAAO,CAAC,CAAC;QAEpD,uEAAuE;QACvE,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,CAAC;QACtC,OAAO,IAAI,CAAC,WAAW,CAAI,QAAQ,CAAC,CAAC;IACvC,CAAC;IAED;;;;;;;;;;;;;;;;OAgBG;IACH,KAAK,CAAC,MAAM,CACV,cAAsB,EACtB,KAAa,EACb,IAAmB,EACnB,OAA6B;QAE7B,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;QAC/D,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,EAAE,KAAK,CAAC,CAAC;QACxD,MAAM,SAAS,GAAG,eAAe,EAAE,CAAC;QAEpC,MAAM,OAAO,GAAG;YACd,GAAG,IAAI;YACP,SAAS,EAAE,SAAS;YACpB,GAAG,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,SAAS,EAAE,SAAS,EAAE,CAAC;SACpD,CAAC;QAEF,MAAM,MAAM,CAAC,MAAM,EAAE,OAAO,EAAE,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,IAAI,KAAK,EAAE,CAAC,CAAC;IACpE,CAAC;IAED;;;;;;;;;;;;;;OAcG;IACH,KAAK,CAAC,SAAS,CACb,cAAsB,EACtB,KAAa,EACb,IAA0C;QAE1C,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;QAC/D,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,EAAE,KAAK,CAAC,CAAC;QAExD,MAAM,SAAS,CAAC,MAAM,EAAE;YACtB,GAAG,IAAI;YACP,SAAS,EAAE,eAAe,EAAE;SAC7B,CAAC,CAAC;IACL,CAAC;IAED;;;;;;;;;;OAUG;IACH,KAAK,CAAC,SAAS,CAAC,cAAsB,EAAE,KAAa;QACnD,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;QAC/D,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,EAAE,KAAK,CAAC,CAAC;QACxD,MAAM,SAAS,CAAC,MAAM,CAAC,CAAC;IAC1B,CAAC;IAED,8EAA8E;IAC9E,sBAAsB;IACtB,8EAA8E;IAE9E;;;;;;;;;;;;;OAaG;IACH,KAAK,CAAC,KAAK,CACT,UAIU;QAEV,MAAM,KAAK,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAEzC,MAAM,QAAQ,GAAG;YACf,GAAG,EAAE,CAAI,IAAY,EAAE,IAAO,EAAE,EAAE;gBAChC,MAAM,CAAC,cAAc,EAAE,KAAK,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;gBACrD,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;gBAC/D,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,EAAE,KAAK,CAAC,CAAC;gBACxD,KAAK,CAAC,GAAG,CAAC,MAAM,EAAE;oBAChB,GAAG,IAAI;oBACP,SAAS,EAAE,eAAe,EAAE;oBAC5B,SAAS,EAAE,eAAe,EAAE;iBACb,CAAC,CAAC;YACrB,CAAC;YACD,MAAM,EAAE,CAAI,IAAY,EAAE,IAAgB,EAAE,EAAE;gBAC5C,MAAM,CAAC,cAAc,EAAE,KAAK,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;gBACrD,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;gBAC/D,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,EAAE,KAAK,CAAC,CAAC;gBACxD,KAAK,CAAC,MAAM,CAAC,MAAM,EAAE;oBACnB,GAAG,IAAI;oBACP,SAAS,EAAE,eAAe,EAAE;iBACb,CAAC,CAAC;YACrB,CAAC;YACD,MAAM,EAAE,CAAC,IAAY,EAAE,EAAE;gBACvB,MAAM,CAAC,cAAc,EAAE,KAAK,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;gBACrD,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;gBAC/D,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,EAAE,KAAK,CAAC,CAAC;gBACxD,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YACvB,CAAC;SACF,CAAC;QAEF,UAAU,CAAC,QAAQ,CAAC,CAAC;QACrB,MAAM,KAAK,CAAC,MAAM,EAAE,CAAC;IACvB,CAAC;IAED,8EAA8E;IAC9E,aAAa;IACb,8EAA8E;IAE9E;;;;;;;;;;;;;;;OAeG;IACH,SAAS,CAAC,QAAgB,EAAE,MAA8B;QACxD,OAAO,SAAS,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;IACrC,CAAC;IAED;;;;;OAKG;IACH,UAAU,CAAC,cAAsB;QAC/B,MAAM,YAAY,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC;QAC/D,MAAM,aAAa,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;QAC/D,OAAO,GAAG,CAAC,aAAa,CAAC,CAAC,EAAE,CAAC;IAC/B,CAAC;IAED;;;OAGG;IACH,eAAe;QACb,OAAO,eAAe,EAAE,CAAC;IAC3B,CAAC;IAED;;;;;;;;;OASG;IACH,UAAU,CAAC,GAAG,QAAmB;QAC/B,OAAO,UAAU,CAAC,GAAG,QAAQ,CAAC,CAAC;IACjC,CAAC;IAED;;OAEG;IACH,WAAW,CAAC,GAAG,QAAmB;QAChC,OAAO,WAAW,CAAC,GAAG,QAAQ,CAAC,CAAC;IAClC,CAAC;IAED;;;;;;;;;OASG;IACH,SAAS,CAAC,CAAS;QACjB,OAAO,SAAS,CAAC,CAAC,CAAC,CAAC;IACtB,CAAC;IAED,8EAA8E;IAC9E,mBAAmB;IACnB,8EAA8E;IAE9E;;OAEG;IACK,qBAAqB,CAAC,OAAsB;QAClD,MAAM,WAAW,GAAsB,EAAE,CAAC;QAE1C,IAAI,CAAC,OAAO;YAAE,OAAO,WAAW,CAAC;QAEjC,gBAAgB;QAChB,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;YAClB,KAAK,MAAM,MAAM,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;gBACnC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,QAAQ,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;YACvE,CAAC;QACH,CAAC;QAED,kBAAkB;QAClB,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpB,KAAK,MAAM,MAAM,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;gBACrC,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC;YAC5D,CAAC;QACH,CAAC;QAED,0BAA0B;QAC1B,IAAI,OAAO,CAAC,UAAU,EAAE,CAAC;YACvB,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC;QACnD,CAAC;QACD,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpB,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC;QAC7C,CAAC;QACD,IAAI,OAAO,CAAC,SAAS,EAAE,CAAC;YACtB,WAAW,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,CAAC;QACjD,CAAC;QACD,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;YAClB,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC;QACzC,CAAC;QAED,6BAA6B;QAC7B,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;YAClB,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC;QACzC,CAAC;QAED,OAAO,WAAW,CAAC;IACrB,CAAC;IAED;;OAEG;IACK,WAAW,CACjB,QAAwC;QAExC,MAAM,IAAI,GAAG,QAAQ,CAAC,IAAI,EAAE,CAAC;QAC7B,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,MAAM,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC;QAC9C,CAAC;QAED,OAAO;YACL,EAAE,EAAE,QAAQ,CAAC,EAAE;YACf,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC;SAC3B,CAAC;IACT,CAAC;IAED;;OAEG;IACK,iBAAiB,CAAC,IAAkB;QAC1C,MAAM,MAAM,GAAiB,EAAE,CAAC;QAEhC,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;YAChD,IAAI,KAAK,YAAY,SAAS,EAAE,CAAC;gBAC/B,MAAM,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC;YAC/B,CAAC;iBAAM,IAAI,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;gBACvE,MAAM,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC;YAC9C,CAAC;iBAAM,CAAC;gBACN,MAAM,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;YACtB,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACK,SAAS,CAAC,IAAY;QAC5B,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACjC,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;YACrD,MAAM,IAAI,KAAK,CAAC,+BAA+B,IAAI,EAAE,CAAC,CAAC;QACzD,CAAC;QACD,MAAM,KAAK,GAAG,QAAQ,CAAC,GAAG,EAAG,CAAC;QAC9B,MAAM,cAAc,GAAG,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC1C,OAAO,CAAC,cAAc,EAAE,KAAK,CAAC,CAAC;IACjC,CAAC;+GAvjBU,gBAAgB;mHAAhB,gBAAgB,cADH,MAAM;;4FACnB,gBAAgB;kBAD5B,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE","sourcesContent":["/**\n * Firestore Service\n *\n * Servicio genérico para operaciones CRUD en Firestore.\n * Soporta lecturas one-time, subscripciones real-time, paginación y queries complejas.\n */\n\nimport { inject, Injectable } from '@angular/core';\nimport {\n  addDoc,\n  collection,\n  collectionData,\n  deleteDoc,\n  doc,\n  docData,\n  DocumentData,\n  DocumentReference,\n  DocumentSnapshot,\n  Firestore,\n  getDoc,\n  getDocs,\n  limit,\n  orderBy,\n  query,\n  QueryConstraint,\n  serverTimestamp,\n  setDoc,\n  startAfter,\n  startAt,\n  endBefore,\n  endAt,\n  Timestamp,\n  updateDoc,\n  where,\n  writeBatch,\n  increment,\n  arrayUnion,\n  arrayRemove,\n  FieldValue,\n} from '@angular/fire/firestore';\nimport { map, Observable } from 'rxjs';\n\nimport { VALTECH_FIREBASE_CONFIG } from './config';\nimport { FirestoreDocument, PaginatedResult, QueryOptions } from './types';\nimport { buildPath } from './utils/path-builder';\n\n/**\n * Servicio para operaciones CRUD en Firestore.\n *\n * @example\n * ```typescript\n * interface User extends FirestoreDocument {\n *   name: string;\n *   email: string;\n *   role: 'admin' | 'user';\n * }\n *\n * @Component({...})\n * export class UsersComponent {\n *   private firestore = inject(FirestoreService);\n *\n *   // Lectura one-time\n *   async loadUser(id: string) {\n *     const user = await this.firestore.getDoc<User>('users', id);\n *   }\n *\n *   // Subscripción real-time\n *   users$ = this.firestore.collectionChanges<User>('users', {\n *     where: [{ field: 'role', operator: '==', value: 'admin' }],\n *     orderBy: [{ field: 'name', direction: 'asc' }]\n *   });\n *\n *   // Crear documento\n *   async createUser(data: Omit<User, 'id'>) {\n *     const user = await this.firestore.addDoc<User>('users', data);\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class FirestoreService {\n  private config = inject(VALTECH_FIREBASE_CONFIG, { optional: true });\n\n  constructor(private firestore: Firestore) {}\n\n  /**\n   * Prefija el path de colección con el appId si está configurado.\n   * Si no hay appId, retorna el path sin modificar (backward compatible).\n   *\n   * @internal\n   */\n  private prefixCollectionPath(collectionPath: string): string {\n    if (!this.config?.appId) return collectionPath;\n    return `apps/${this.config.appId}/${collectionPath}`;\n  }\n\n  // ===========================================================================\n  // LECTURAS ONE-TIME (Promise)\n  // ===========================================================================\n\n  /**\n   * Obtiene un documento por ID (lectura única).\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @returns Documento o null si no existe\n   *\n   * @example\n   * ```typescript\n   * const user = await firestoreService.getDoc<User>('users', 'abc123');\n   * if (user) {\n   *   console.log(user.name);\n   * }\n   * ```\n   */\n  async getDoc<T extends FirestoreDocument>(\n    collectionPath: string,\n    docId: string\n  ): Promise<T | null> {\n    const prefixedPath = this.prefixCollectionPath(collectionPath);\n    const docRef = doc(this.firestore, prefixedPath, docId);\n    const snapshot = await getDoc(docRef);\n\n    if (!snapshot.exists()) {\n      return null;\n    }\n\n    return this.mapDocument<T>(snapshot);\n  }\n\n  /**\n   * Obtiene múltiples documentos con opciones de query.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param options - Opciones de query (where, orderBy, limit)\n   * @returns Array de documentos\n   *\n   * @example\n   * ```typescript\n   * // Todos los usuarios activos ordenados por nombre\n   * const users = await firestoreService.getDocs<User>('users', {\n   *   where: [{ field: 'active', operator: '==', value: true }],\n   *   orderBy: [{ field: 'name', direction: 'asc' }],\n   *   limit: 50\n   * });\n   * ```\n   */\n  async getDocs<T extends FirestoreDocument>(\n    collectionPath: string,\n    options?: QueryOptions\n  ): Promise<T[]> {\n    const prefixedPath = this.prefixCollectionPath(collectionPath);\n    const collectionRef = collection(this.firestore, prefixedPath);\n    const constraints = this.buildQueryConstraints(options);\n    const q = query(collectionRef, ...constraints);\n    const snapshot = await getDocs(q);\n\n    return snapshot.docs.map((doc) => this.mapDocument<T>(doc));\n  }\n\n  /**\n   * Obtiene documentos con paginación basada en cursores.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param options - Opciones de query (debe incluir limit)\n   * @returns Resultado paginado con cursor para la siguiente página\n   *\n   * @example\n   * ```typescript\n   * // Primera página\n   * const page1 = await firestoreService.getPaginated<User>('users', {\n   *   orderBy: [{ field: 'createdAt', direction: 'desc' }],\n   *   limit: 10\n   * });\n   *\n   * // Siguiente página\n   * if (page1.hasMore) {\n   *   const page2 = await firestoreService.getPaginated<User>('users', {\n   *     orderBy: [{ field: 'createdAt', direction: 'desc' }],\n   *     limit: 10,\n   *     startAfter: page1.lastDoc\n   *   });\n   * }\n   * ```\n   */\n  async getPaginated<T extends FirestoreDocument>(\n    collectionPath: string,\n    options: QueryOptions & { limit: number }\n  ): Promise<PaginatedResult<T>> {\n    const prefixedPath = this.prefixCollectionPath(collectionPath);\n    const collectionRef = collection(this.firestore, prefixedPath);\n    const constraints = this.buildQueryConstraints(options);\n\n    // Pedir uno más para saber si hay más páginas\n    const q = query(collectionRef, ...constraints, limit(options.limit + 1));\n    const snapshot = await getDocs(q);\n\n    const docs = snapshot.docs;\n    const hasMore = docs.length > options.limit;\n\n    // Si hay más, remover el documento extra\n    const resultDocs = hasMore ? docs.slice(0, -1) : docs;\n    const lastDoc = resultDocs.length > 0 ? resultDocs[resultDocs.length - 1] : null;\n\n    return {\n      data: resultDocs.map((doc) => this.mapDocument<T>(doc)),\n      hasMore,\n      lastDoc,\n    };\n  }\n\n  /**\n   * Verifica si un documento existe.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @returns true si el documento existe\n   */\n  async exists(collectionPath: string, docId: string): Promise<boolean> {\n    const prefixedPath = this.prefixCollectionPath(collectionPath);\n    const docRef = doc(this.firestore, prefixedPath, docId);\n    const snapshot = await getDoc(docRef);\n    return snapshot.exists();\n  }\n\n  // ===========================================================================\n  // SUBSCRIPCIONES REAL-TIME (Observable)\n  // ===========================================================================\n\n  /**\n   * Suscribe a cambios de un documento (real-time).\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @returns Observable que emite cuando el documento cambia\n   *\n   * @example\n   * ```typescript\n   * // En el componente\n   * user$ = this.firestoreService.docChanges<User>('users', this.userId);\n   *\n   * // En el template\n   * @if (user$ | async; as user) {\n   *   <p>{{ user.name }}</p>\n   * }\n   * ```\n   */\n  docChanges<T extends FirestoreDocument>(\n    collectionPath: string,\n    docId: string\n  ): Observable<T | null> {\n    const prefixedPath = this.prefixCollectionPath(collectionPath);\n    const docRef = doc(this.firestore, prefixedPath, docId);\n    return docData(docRef, { idField: 'id' }).pipe(\n      map((data) => {\n        if (!data) return null;\n        return this.convertTimestamps(data as DocumentData) as T;\n      })\n    );\n  }\n\n  /**\n   * Suscribe a cambios de una colección (real-time).\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param options - Opciones de query\n   * @returns Observable que emite cuando la colección cambia\n   *\n   * @example\n   * ```typescript\n   * // Usuarios activos en tiempo real\n   * activeUsers$ = this.firestoreService.collectionChanges<User>('users', {\n   *   where: [{ field: 'status', operator: '==', value: 'online' }]\n   * });\n   * ```\n   */\n  collectionChanges<T extends FirestoreDocument>(\n    collectionPath: string,\n    options?: QueryOptions\n  ): Observable<T[]> {\n    const prefixedPath = this.prefixCollectionPath(collectionPath);\n    const collectionRef = collection(this.firestore, prefixedPath);\n    const constraints = this.buildQueryConstraints(options);\n    const q = query(collectionRef, ...constraints);\n\n    return collectionData(q, { idField: 'id' }).pipe(\n      map((docs) => docs.map((doc) => this.convertTimestamps(doc) as T))\n    );\n  }\n\n  // ===========================================================================\n  // ESCRITURA\n  // ===========================================================================\n\n  /**\n   * Agrega un documento con ID auto-generado.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param data - Datos del documento (sin id, createdAt, updatedAt)\n   * @returns Documento creado con su ID\n   *\n   * @example\n   * ```typescript\n   * const newUser = await firestoreService.addDoc<User>('users', {\n   *   name: 'John Doe',\n   *   email: 'john@example.com',\n   *   role: 'user'\n   * });\n   * console.log('Created user with ID:', newUser.id);\n   * ```\n   */\n  async addDoc<T extends FirestoreDocument>(\n    collectionPath: string,\n    data: Omit<T, 'id' | 'createdAt' | 'updatedAt'>\n  ): Promise<T> {\n    const prefixedPath = this.prefixCollectionPath(collectionPath);\n    const collectionRef = collection(this.firestore, prefixedPath);\n    const timestamp = serverTimestamp();\n\n    const docData = {\n      ...data,\n      createdAt: timestamp,\n      updatedAt: timestamp,\n    };\n\n    const docRef = await addDoc(collectionRef, docData);\n\n    // Obtener el documento creado para retornarlo con timestamps resueltos\n    const snapshot = await getDoc(docRef);\n    return this.mapDocument<T>(snapshot);\n  }\n\n  /**\n   * Crea o sobrescribe un documento con ID específico.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @param data - Datos del documento\n   * @param options - Opciones (merge: true para merge en lugar de sobrescribir)\n   *\n   * @example\n   * ```typescript\n   * // Sobrescribir completamente\n   * await firestoreService.setDoc<User>('users', 'user123', userData);\n   *\n   * // Merge con datos existentes\n   * await firestoreService.setDoc<User>('users', 'user123', { name: 'New Name' }, { merge: true });\n   * ```\n   */\n  async setDoc<T extends FirestoreDocument>(\n    collectionPath: string,\n    docId: string,\n    data: Omit<T, 'id'>,\n    options?: { merge?: boolean }\n  ): Promise<void> {\n    const prefixedPath = this.prefixCollectionPath(collectionPath);\n    const docRef = doc(this.firestore, prefixedPath, docId);\n    const timestamp = serverTimestamp();\n\n    const docData = {\n      ...data,\n      updatedAt: timestamp,\n      ...(options?.merge ? {} : { createdAt: timestamp }),\n    };\n\n    await setDoc(docRef, docData, { merge: options?.merge ?? false });\n  }\n\n  /**\n   * Actualiza campos específicos de un documento.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @param data - Campos a actualizar\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.updateDoc<User>('users', 'user123', {\n   *   name: 'Updated Name',\n   *   lastLogin: new Date()\n   * });\n   * ```\n   */\n  async updateDoc<T extends FirestoreDocument>(\n    collectionPath: string,\n    docId: string,\n    data: Partial<Omit<T, 'id' | 'createdAt'>>\n  ): Promise<void> {\n    const prefixedPath = this.prefixCollectionPath(collectionPath);\n    const docRef = doc(this.firestore, prefixedPath, docId);\n\n    await updateDoc(docRef, {\n      ...data,\n      updatedAt: serverTimestamp(),\n    });\n  }\n\n  /**\n   * Elimina un documento.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.deleteDoc('users', 'user123');\n   * ```\n   */\n  async deleteDoc(collectionPath: string, docId: string): Promise<void> {\n    const prefixedPath = this.prefixCollectionPath(collectionPath);\n    const docRef = doc(this.firestore, prefixedPath, docId);\n    await deleteDoc(docRef);\n  }\n\n  // ===========================================================================\n  // OPERACIONES EN LOTE\n  // ===========================================================================\n\n  /**\n   * Ejecuta múltiples operaciones de escritura de forma atómica.\n   *\n   * @param operations - Función que recibe el batch y agrega operaciones\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.batch((batch) => {\n   *   batch.set('users/user1', { name: 'User 1' });\n   *   batch.update('users/user2', { status: 'inactive' });\n   *   batch.delete('users/user3');\n   * });\n   * ```\n   */\n  async batch(\n    operations: (batch: {\n      set: <T>(path: string, data: T) => void;\n      update: <T>(path: string, data: Partial<T>) => void;\n      delete: (path: string) => void;\n    }) => void\n  ): Promise<void> {\n    const batch = writeBatch(this.firestore);\n\n    const batchApi = {\n      set: <T>(path: string, data: T) => {\n        const [collectionPath, docId] = this.splitPath(path);\n        const prefixedPath = this.prefixCollectionPath(collectionPath);\n        const docRef = doc(this.firestore, prefixedPath, docId);\n        batch.set(docRef, {\n          ...data,\n          createdAt: serverTimestamp(),\n          updatedAt: serverTimestamp(),\n        } as DocumentData);\n      },\n      update: <T>(path: string, data: Partial<T>) => {\n        const [collectionPath, docId] = this.splitPath(path);\n        const prefixedPath = this.prefixCollectionPath(collectionPath);\n        const docRef = doc(this.firestore, prefixedPath, docId);\n        batch.update(docRef, {\n          ...data,\n          updatedAt: serverTimestamp(),\n        } as DocumentData);\n      },\n      delete: (path: string) => {\n        const [collectionPath, docId] = this.splitPath(path);\n        const prefixedPath = this.prefixCollectionPath(collectionPath);\n        const docRef = doc(this.firestore, prefixedPath, docId);\n        batch.delete(docRef);\n      },\n    };\n\n    operations(batchApi);\n    await batch.commit();\n  }\n\n  // ===========================================================================\n  // UTILIDADES\n  // ===========================================================================\n\n  /**\n   * Construye una ruta a partir de un template.\n   *\n   * @param template - Template con placeholders {param}\n   * @param params - Valores para los placeholders\n   * @returns Ruta construida\n   *\n   * @example\n   * ```typescript\n   * const path = firestoreService.buildPath('users/{userId}/documents/{docId}', {\n   *   userId: 'user123',\n   *   docId: 'doc456'\n   * });\n   * // => 'users/user123/documents/doc456'\n   * ```\n   */\n  buildPath(template: string, params: Record<string, string>): string {\n    return buildPath(template, params);\n  }\n\n  /**\n   * Genera un ID único para un documento (sin crearlo).\n   *\n   * @param collectionPath - Ruta de la colección\n   * @returns ID único generado por Firestore\n   */\n  generateId(collectionPath: string): string {\n    const prefixedPath = this.prefixCollectionPath(collectionPath);\n    const collectionRef = collection(this.firestore, prefixedPath);\n    return doc(collectionRef).id;\n  }\n\n  /**\n   * Retorna un valor de timestamp del servidor.\n   * Usar en campos de fecha para que Firestore asigne el timestamp.\n   */\n  serverTimestamp(): FieldValue {\n    return serverTimestamp();\n  }\n\n  /**\n   * Retorna un valor para agregar elementos a un array.\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.updateDoc('users', 'user123', {\n   *   tags: firestoreService.arrayUnion('new-tag')\n   * });\n   * ```\n   */\n  arrayUnion(...elements: unknown[]): FieldValue {\n    return arrayUnion(...elements);\n  }\n\n  /**\n   * Retorna un valor para remover elementos de un array.\n   */\n  arrayRemove(...elements: unknown[]): FieldValue {\n    return arrayRemove(...elements);\n  }\n\n  /**\n   * Retorna un valor para incrementar un campo numérico.\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.updateDoc('users', 'user123', {\n   *   loginCount: firestoreService.increment(1)\n   * });\n   * ```\n   */\n  increment(n: number): FieldValue {\n    return increment(n);\n  }\n\n  // ===========================================================================\n  // MÉTODOS PRIVADOS\n  // ===========================================================================\n\n  /**\n   * Construye los QueryConstraints a partir de QueryOptions\n   */\n  private buildQueryConstraints(options?: QueryOptions): QueryConstraint[] {\n    const constraints: QueryConstraint[] = [];\n\n    if (!options) return constraints;\n\n    // Where clauses\n    if (options.where) {\n      for (const clause of options.where) {\n        constraints.push(where(clause.field, clause.operator, clause.value));\n      }\n    }\n\n    // OrderBy clauses\n    if (options.orderBy) {\n      for (const clause of options.orderBy) {\n        constraints.push(orderBy(clause.field, clause.direction));\n      }\n    }\n\n    // Cursors para paginación\n    if (options.startAfter) {\n      constraints.push(startAfter(options.startAfter));\n    }\n    if (options.startAt) {\n      constraints.push(startAt(options.startAt));\n    }\n    if (options.endBefore) {\n      constraints.push(endBefore(options.endBefore));\n    }\n    if (options.endAt) {\n      constraints.push(endAt(options.endAt));\n    }\n\n    // Limit (se agrega al final)\n    if (options.limit) {\n      constraints.push(limit(options.limit));\n    }\n\n    return constraints;\n  }\n\n  /**\n   * Mapea un DocumentSnapshot a nuestro tipo\n   */\n  private mapDocument<T extends FirestoreDocument>(\n    snapshot: DocumentSnapshot<DocumentData>\n  ): T {\n    const data = snapshot.data();\n    if (!data) {\n      throw new Error('Documento no tiene datos');\n    }\n\n    return {\n      id: snapshot.id,\n      ...this.convertTimestamps(data),\n    } as T;\n  }\n\n  /**\n   * Convierte Timestamps de Firestore a Date de JavaScript\n   */\n  private convertTimestamps(data: DocumentData): DocumentData {\n    const result: DocumentData = {};\n\n    for (const [key, value] of Object.entries(data)) {\n      if (value instanceof Timestamp) {\n        result[key] = value.toDate();\n      } else if (value && typeof value === 'object' && !Array.isArray(value)) {\n        result[key] = this.convertTimestamps(value);\n      } else {\n        result[key] = value;\n      }\n    }\n\n    return result;\n  }\n\n  /**\n   * Divide una ruta de documento en colección e ID\n   */\n  private splitPath(path: string): [string, string] {\n    const segments = path.split('/');\n    if (segments.length < 2 || segments.length % 2 !== 0) {\n      throw new Error(`Ruta de documento inválida: ${path}`);\n    }\n    const docId = segments.pop()!;\n    const collectionPath = segments.join('/');\n    return [collectionPath, docId];\n  }\n}\n"]}
@@ -31,8 +31,8 @@
31
31
  export * from './types';
32
32
  // Configuración
33
33
  export { VALTECH_FIREBASE_CONFIG, hasEmulators, provideValtechFirebase } from './config';
34
- // Configuración compartida del monorepo
35
- export { APP_IDS, FIREBASE_PROJECTS, SHARED_EMULATOR_CONFIG, collections, createFirebaseConfig, isEmulatorMode, storagePaths, } from './shared-config';
34
+ // Configuración compartida (helpers genéricos)
35
+ export { collections, createFirebaseConfig, isEmulatorMode, storagePaths, } from './shared-config';
36
36
  // Servicios
37
37
  export { FirebaseService } from './firebase.service';
38
38
  // Firestore
@@ -46,4 +46,4 @@ export { buildPath, extractPathParams, getCollectionPath, getDocumentId, isColle
46
46
  export { StorageService } from './storage.service';
47
47
  // Messaging (FCM)
48
48
  export { MessagingService } from './messaging.service';
49
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2ZpcmViYXNlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNEJHO0FBRUgsUUFBUTtBQUNSLGNBQWMsU0FBUyxDQUFDO0FBRXhCLGdCQUFnQjtBQUNoQixPQUFPLEVBQUUsdUJBQXVCLEVBQUUsWUFBWSxFQUFFLHNCQUFzQixFQUFFLE1BQU0sVUFBVSxDQUFDO0FBRXpGLHdDQUF3QztBQUN4QyxPQUFPLEVBQ0wsT0FBTyxFQUNQLGlCQUFpQixFQUNqQixzQkFBc0IsRUFDdEIsV0FBVyxFQUNYLG9CQUFvQixFQUNwQixjQUFjLEVBQ2QsWUFBWSxHQUdiLE1BQU0saUJBQWlCLENBQUM7QUFFekIsWUFBWTtBQUNaLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUVyRCxZQUFZO0FBQ1osT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFFdkQsMENBQTBDO0FBQzFDLE9BQU8sRUFHTCwwQkFBMEIsRUFFMUIsZUFBZSxHQUNoQixNQUFNLHdCQUF3QixDQUFDO0FBRWhDLGFBQWE7QUFDYixPQUFPLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBQzVELE9BQU8sRUFDTCxTQUFTLEVBQ1QsaUJBQWlCLEVBQ2pCLGlCQUFpQixFQUNqQixhQUFhLEVBQ2IsZ0JBQWdCLEVBQ2hCLGNBQWMsRUFDZCxXQUFXLEVBQ1gsUUFBUSxHQUNULE1BQU0sc0JBQXNCLENBQUM7QUFFOUIsVUFBVTtBQUNWLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUVuRCxrQkFBa0I7QUFDbEIsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0scUJBQXFCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEZpcmViYXNlIFNlcnZpY2VzXG4gKlxuICogU2VydmljaW9zIHJldXRpbGl6YWJsZXMgcGFyYSBpbnRlZ3JhY2nDs24gY29uIEZpcmViYXNlLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiAvLyBFbiBtYWluLnRzXG4gKiBpbXBvcnQgeyBwcm92aWRlVmFsdGVjaEZpcmViYXNlIH0gZnJvbSAndmFsdGVjaC1jb21wb25lbnRzJztcbiAqXG4gKiBib290c3RyYXBBcHBsaWNhdGlvbihBcHBDb21wb25lbnQsIHtcbiAqICAgcHJvdmlkZXJzOiBbXG4gKiAgICAgcHJvdmlkZVZhbHRlY2hGaXJlYmFzZSh7XG4gKiAgICAgICBmaXJlYmFzZTogZW52aXJvbm1lbnQuZmlyZWJhc2UsXG4gKiAgICAgICBwZXJzaXN0ZW5jZTogdHJ1ZSxcbiAqICAgICB9KSxcbiAqICAgXSxcbiAqIH0pO1xuICpcbiAqIC8vIEVuIGNvbXBvbmVudGVzXG4gKiBpbXBvcnQgeyBGaXJlYmFzZVNlcnZpY2UsIEZpcmVzdG9yZVNlcnZpY2UgfSBmcm9tICd2YWx0ZWNoLWNvbXBvbmVudHMnO1xuICpcbiAqIEBDb21wb25lbnQoey4uLn0pXG4gKiBleHBvcnQgY2xhc3MgTXlDb21wb25lbnQge1xuICogICBwcml2YXRlIGZpcmViYXNlID0gaW5qZWN0KEZpcmViYXNlU2VydmljZSk7XG4gKiAgIHByaXZhdGUgZmlyZXN0b3JlID0gaW5qZWN0KEZpcmVzdG9yZVNlcnZpY2UpO1xuICogfVxuICogYGBgXG4gKi9cblxuLy8gVGlwb3NcbmV4cG9ydCAqIGZyb20gJy4vdHlwZXMnO1xuXG4vLyBDb25maWd1cmFjacOzblxuZXhwb3J0IHsgVkFMVEVDSF9GSVJFQkFTRV9DT05GSUcsIGhhc0VtdWxhdG9ycywgcHJvdmlkZVZhbHRlY2hGaXJlYmFzZSB9IGZyb20gJy4vY29uZmlnJztcblxuLy8gQ29uZmlndXJhY2nDs24gY29tcGFydGlkYSBkZWwgbW9ub3JlcG9cbmV4cG9ydCB7XG4gIEFQUF9JRFMsXG4gIEZJUkVCQVNFX1BST0pFQ1RTLFxuICBTSEFSRURfRU1VTEFUT1JfQ09ORklHLFxuICBjb2xsZWN0aW9ucyxcbiAgY3JlYXRlRmlyZWJhc2VDb25maWcsXG4gIGlzRW11bGF0b3JNb2RlLFxuICBzdG9yYWdlUGF0aHMsXG4gIHR5cGUgQXBwSWQsXG4gIHR5cGUgQ3JlYXRlRmlyZWJhc2VDb25maWdPcHRpb25zLFxufSBmcm9tICcuL3NoYXJlZC1jb25maWcnO1xuXG4vLyBTZXJ2aWNpb3NcbmV4cG9ydCB7IEZpcmViYXNlU2VydmljZSB9IGZyb20gJy4vZmlyZWJhc2Uuc2VydmljZSc7XG5cbi8vIEZpcmVzdG9yZVxuZXhwb3J0IHsgRmlyZXN0b3JlU2VydmljZSB9IGZyb20gJy4vZmlyZXN0b3JlLnNlcnZpY2UnO1xuXG4vLyBGaXJlc3RvcmUgQ29sbGVjdGlvbnMgKEZhY3RvcnkgUGF0dGVybilcbmV4cG9ydCB7XG4gIENvbGxlY3Rpb25PcHRpb25zLFxuICBGaXJlc3RvcmVDb2xsZWN0aW9uLFxuICBGaXJlc3RvcmVDb2xsZWN0aW9uRmFjdG9yeSxcbiAgU3ViQ29sbGVjdGlvblJlZixcbiAgVHlwZWRDb2xsZWN0aW9uLFxufSBmcm9tICcuL2ZpcmVzdG9yZS1jb2xsZWN0aW9uJztcblxuLy8gVXRpbGlkYWRlc1xuZXhwb3J0IHsgUXVlcnlCdWlsZGVyLCBxdWVyeSB9IGZyb20gJy4vdXRpbHMvcXVlcnktYnVpbGRlcic7XG5leHBvcnQge1xuICBidWlsZFBhdGgsXG4gIGV4dHJhY3RQYXRoUGFyYW1zLFxuICBnZXRDb2xsZWN0aW9uUGF0aCxcbiAgZ2V0RG9jdW1lbnRJZCxcbiAgaXNDb2xsZWN0aW9uUGF0aCxcbiAgaXNEb2N1bWVudFBhdGgsXG4gIGlzVmFsaWRQYXRoLFxuICBqb2luUGF0aCxcbn0gZnJvbSAnLi91dGlscy9wYXRoLWJ1aWxkZXInO1xuXG4vLyBTdG9yYWdlXG5leHBvcnQgeyBTdG9yYWdlU2VydmljZSB9IGZyb20gJy4vc3RvcmFnZS5zZXJ2aWNlJztcblxuLy8gTWVzc2FnaW5nIChGQ00pXG5leHBvcnQgeyBNZXNzYWdpbmdTZXJ2aWNlIH0gZnJvbSAnLi9tZXNzYWdpbmcuc2VydmljZSc7XG4iXX0=
49
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2ZpcmViYXNlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNEJHO0FBRUgsUUFBUTtBQUNSLGNBQWMsU0FBUyxDQUFDO0FBRXhCLGdCQUFnQjtBQUNoQixPQUFPLEVBQUUsdUJBQXVCLEVBQUUsWUFBWSxFQUFFLHNCQUFzQixFQUFFLE1BQU0sVUFBVSxDQUFDO0FBRXpGLCtDQUErQztBQUMvQyxPQUFPLEVBQ0wsV0FBVyxFQUNYLG9CQUFvQixFQUNwQixjQUFjLEVBQ2QsWUFBWSxHQUdiLE1BQU0saUJBQWlCLENBQUM7QUFFekIsWUFBWTtBQUNaLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUVyRCxZQUFZO0FBQ1osT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFFdkQsMENBQTBDO0FBQzFDLE9BQU8sRUFHTCwwQkFBMEIsRUFFMUIsZUFBZSxHQUNoQixNQUFNLHdCQUF3QixDQUFDO0FBRWhDLGFBQWE7QUFDYixPQUFPLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBQzVELE9BQU8sRUFDTCxTQUFTLEVBQ1QsaUJBQWlCLEVBQ2pCLGlCQUFpQixFQUNqQixhQUFhLEVBQ2IsZ0JBQWdCLEVBQ2hCLGNBQWMsRUFDZCxXQUFXLEVBQ1gsUUFBUSxHQUNULE1BQU0sc0JBQXNCLENBQUM7QUFFOUIsVUFBVTtBQUNWLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUVuRCxrQkFBa0I7QUFDbEIsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0scUJBQXFCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEZpcmViYXNlIFNlcnZpY2VzXG4gKlxuICogU2VydmljaW9zIHJldXRpbGl6YWJsZXMgcGFyYSBpbnRlZ3JhY2nDs24gY29uIEZpcmViYXNlLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiAvLyBFbiBtYWluLnRzXG4gKiBpbXBvcnQgeyBwcm92aWRlVmFsdGVjaEZpcmViYXNlIH0gZnJvbSAndmFsdGVjaC1jb21wb25lbnRzJztcbiAqXG4gKiBib290c3RyYXBBcHBsaWNhdGlvbihBcHBDb21wb25lbnQsIHtcbiAqICAgcHJvdmlkZXJzOiBbXG4gKiAgICAgcHJvdmlkZVZhbHRlY2hGaXJlYmFzZSh7XG4gKiAgICAgICBmaXJlYmFzZTogZW52aXJvbm1lbnQuZmlyZWJhc2UsXG4gKiAgICAgICBwZXJzaXN0ZW5jZTogdHJ1ZSxcbiAqICAgICB9KSxcbiAqICAgXSxcbiAqIH0pO1xuICpcbiAqIC8vIEVuIGNvbXBvbmVudGVzXG4gKiBpbXBvcnQgeyBGaXJlYmFzZVNlcnZpY2UsIEZpcmVzdG9yZVNlcnZpY2UgfSBmcm9tICd2YWx0ZWNoLWNvbXBvbmVudHMnO1xuICpcbiAqIEBDb21wb25lbnQoey4uLn0pXG4gKiBleHBvcnQgY2xhc3MgTXlDb21wb25lbnQge1xuICogICBwcml2YXRlIGZpcmViYXNlID0gaW5qZWN0KEZpcmViYXNlU2VydmljZSk7XG4gKiAgIHByaXZhdGUgZmlyZXN0b3JlID0gaW5qZWN0KEZpcmVzdG9yZVNlcnZpY2UpO1xuICogfVxuICogYGBgXG4gKi9cblxuLy8gVGlwb3NcbmV4cG9ydCAqIGZyb20gJy4vdHlwZXMnO1xuXG4vLyBDb25maWd1cmFjacOzblxuZXhwb3J0IHsgVkFMVEVDSF9GSVJFQkFTRV9DT05GSUcsIGhhc0VtdWxhdG9ycywgcHJvdmlkZVZhbHRlY2hGaXJlYmFzZSB9IGZyb20gJy4vY29uZmlnJztcblxuLy8gQ29uZmlndXJhY2nDs24gY29tcGFydGlkYSAoaGVscGVycyBnZW7DqXJpY29zKVxuZXhwb3J0IHtcbiAgY29sbGVjdGlvbnMsXG4gIGNyZWF0ZUZpcmViYXNlQ29uZmlnLFxuICBpc0VtdWxhdG9yTW9kZSxcbiAgc3RvcmFnZVBhdGhzLFxuICB0eXBlIEFwcElkLFxuICB0eXBlIENyZWF0ZUZpcmViYXNlQ29uZmlnT3B0aW9ucyxcbn0gZnJvbSAnLi9zaGFyZWQtY29uZmlnJztcblxuLy8gU2VydmljaW9zXG5leHBvcnQgeyBGaXJlYmFzZVNlcnZpY2UgfSBmcm9tICcuL2ZpcmViYXNlLnNlcnZpY2UnO1xuXG4vLyBGaXJlc3RvcmVcbmV4cG9ydCB7IEZpcmVzdG9yZVNlcnZpY2UgfSBmcm9tICcuL2ZpcmVzdG9yZS5zZXJ2aWNlJztcblxuLy8gRmlyZXN0b3JlIENvbGxlY3Rpb25zIChGYWN0b3J5IFBhdHRlcm4pXG5leHBvcnQge1xuICBDb2xsZWN0aW9uT3B0aW9ucyxcbiAgRmlyZXN0b3JlQ29sbGVjdGlvbixcbiAgRmlyZXN0b3JlQ29sbGVjdGlvbkZhY3RvcnksXG4gIFN1YkNvbGxlY3Rpb25SZWYsXG4gIFR5cGVkQ29sbGVjdGlvbixcbn0gZnJvbSAnLi9maXJlc3RvcmUtY29sbGVjdGlvbic7XG5cbi8vIFV0aWxpZGFkZXNcbmV4cG9ydCB7IFF1ZXJ5QnVpbGRlciwgcXVlcnkgfSBmcm9tICcuL3V0aWxzL3F1ZXJ5LWJ1aWxkZXInO1xuZXhwb3J0IHtcbiAgYnVpbGRQYXRoLFxuICBleHRyYWN0UGF0aFBhcmFtcyxcbiAgZ2V0Q29sbGVjdGlvblBhdGgsXG4gIGdldERvY3VtZW50SWQsXG4gIGlzQ29sbGVjdGlvblBhdGgsXG4gIGlzRG9jdW1lbnRQYXRoLFxuICBpc1ZhbGlkUGF0aCxcbiAgam9pblBhdGgsXG59IGZyb20gJy4vdXRpbHMvcGF0aC1idWlsZGVyJztcblxuLy8gU3RvcmFnZVxuZXhwb3J0IHsgU3RvcmFnZVNlcnZpY2UgfSBmcm9tICcuL3N0b3JhZ2Uuc2VydmljZSc7XG5cbi8vIE1lc3NhZ2luZyAoRkNNKVxuZXhwb3J0IHsgTWVzc2FnaW5nU2VydmljZSB9IGZyb20gJy4vbWVzc2FnaW5nLnNlcnZpY2UnO1xuIl19