valtech-components 2.0.452 → 2.0.454

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.
Files changed (48) hide show
  1. package/esm2022/lib/components/molecules/pin-input/pin-input.component.mjs +10 -2
  2. package/esm2022/lib/services/auth/auth-state.service.mjs +173 -0
  3. package/esm2022/lib/services/auth/auth.service.mjs +454 -0
  4. package/esm2022/lib/services/auth/config.mjs +76 -0
  5. package/esm2022/lib/services/auth/guards.mjs +194 -0
  6. package/esm2022/lib/services/auth/index.mjs +70 -0
  7. package/esm2022/lib/services/auth/interceptor.mjs +98 -0
  8. package/esm2022/lib/services/auth/storage.service.mjs +141 -0
  9. package/esm2022/lib/services/auth/sync.service.mjs +149 -0
  10. package/esm2022/lib/services/auth/token.service.mjs +113 -0
  11. package/esm2022/lib/services/auth/types.mjs +29 -0
  12. package/esm2022/lib/services/firebase/config.mjs +108 -0
  13. package/esm2022/lib/services/firebase/firebase.service.mjs +288 -0
  14. package/esm2022/lib/services/firebase/firestore-collection.mjs +254 -0
  15. package/esm2022/lib/services/firebase/firestore.service.mjs +509 -0
  16. package/esm2022/lib/services/firebase/index.mjs +49 -0
  17. package/esm2022/lib/services/firebase/messaging.service.mjs +512 -0
  18. package/esm2022/lib/services/firebase/shared-config.mjs +138 -0
  19. package/esm2022/lib/services/firebase/storage.service.mjs +422 -0
  20. package/esm2022/lib/services/firebase/types.mjs +8 -0
  21. package/esm2022/lib/services/firebase/utils/path-builder.mjs +195 -0
  22. package/esm2022/lib/services/firebase/utils/query-builder.mjs +302 -0
  23. package/esm2022/public-api.mjs +3 -5
  24. package/fesm2022/valtech-components.mjs +4204 -5
  25. package/fesm2022/valtech-components.mjs.map +1 -1
  26. package/lib/services/auth/auth-state.service.d.ts +85 -0
  27. package/lib/services/auth/auth.service.d.ts +146 -0
  28. package/lib/services/auth/config.d.ts +38 -0
  29. package/lib/services/auth/guards.d.ts +123 -0
  30. package/lib/services/auth/index.d.ts +63 -0
  31. package/lib/services/auth/interceptor.d.ts +22 -0
  32. package/lib/services/auth/storage.service.d.ts +48 -0
  33. package/lib/services/auth/sync.service.d.ts +49 -0
  34. package/lib/services/auth/token.service.d.ts +51 -0
  35. package/lib/services/auth/types.d.ts +315 -0
  36. package/lib/services/firebase/config.d.ts +49 -0
  37. package/lib/services/firebase/firebase.service.d.ts +140 -0
  38. package/lib/services/firebase/firestore-collection.d.ts +175 -0
  39. package/lib/services/firebase/firestore.service.d.ts +304 -0
  40. package/lib/services/firebase/index.d.ts +39 -0
  41. package/lib/services/firebase/messaging.service.d.ts +263 -0
  42. package/lib/services/firebase/shared-config.d.ts +126 -0
  43. package/lib/services/firebase/storage.service.d.ts +206 -0
  44. package/lib/services/firebase/types.d.ts +281 -0
  45. package/lib/services/firebase/utils/path-builder.d.ts +132 -0
  46. package/lib/services/firebase/utils/query-builder.d.ts +210 -0
  47. package/package.json +1 -1
  48. package/public-api.d.ts +2 -0
@@ -0,0 +1,509 @@
1
+ /**
2
+ * Firestore Service
3
+ *
4
+ * Servicio genérico para operaciones CRUD en Firestore.
5
+ * Soporta lecturas one-time, subscripciones real-time, paginación y queries complejas.
6
+ */
7
+ import { Injectable } from '@angular/core';
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
+ import { map } from 'rxjs';
10
+ import { buildPath } from './utils/path-builder';
11
+ import * as i0 from "@angular/core";
12
+ import * as i1 from "@angular/fire/firestore";
13
+ /**
14
+ * Servicio para operaciones CRUD en Firestore.
15
+ *
16
+ * @example
17
+ * ```typescript
18
+ * interface User extends FirestoreDocument {
19
+ * name: string;
20
+ * email: string;
21
+ * role: 'admin' | 'user';
22
+ * }
23
+ *
24
+ * @Component({...})
25
+ * export class UsersComponent {
26
+ * private firestore = inject(FirestoreService);
27
+ *
28
+ * // Lectura one-time
29
+ * async loadUser(id: string) {
30
+ * const user = await this.firestore.getDoc<User>('users', id);
31
+ * }
32
+ *
33
+ * // Subscripción real-time
34
+ * users$ = this.firestore.collectionChanges<User>('users', {
35
+ * where: [{ field: 'role', operator: '==', value: 'admin' }],
36
+ * orderBy: [{ field: 'name', direction: 'asc' }]
37
+ * });
38
+ *
39
+ * // Crear documento
40
+ * async createUser(data: Omit<User, 'id'>) {
41
+ * const user = await this.firestore.addDoc<User>('users', data);
42
+ * }
43
+ * }
44
+ * ```
45
+ */
46
+ export class FirestoreService {
47
+ constructor(firestore) {
48
+ this.firestore = firestore;
49
+ }
50
+ // ===========================================================================
51
+ // LECTURAS ONE-TIME (Promise)
52
+ // ===========================================================================
53
+ /**
54
+ * Obtiene un documento por ID (lectura única).
55
+ *
56
+ * @param collectionPath - Ruta de la colección
57
+ * @param docId - ID del documento
58
+ * @returns Documento o null si no existe
59
+ *
60
+ * @example
61
+ * ```typescript
62
+ * const user = await firestoreService.getDoc<User>('users', 'abc123');
63
+ * if (user) {
64
+ * console.log(user.name);
65
+ * }
66
+ * ```
67
+ */
68
+ async getDoc(collectionPath, docId) {
69
+ const docRef = doc(this.firestore, collectionPath, docId);
70
+ const snapshot = await getDoc(docRef);
71
+ if (!snapshot.exists()) {
72
+ return null;
73
+ }
74
+ return this.mapDocument(snapshot);
75
+ }
76
+ /**
77
+ * Obtiene múltiples documentos con opciones de query.
78
+ *
79
+ * @param collectionPath - Ruta de la colección
80
+ * @param options - Opciones de query (where, orderBy, limit)
81
+ * @returns Array de documentos
82
+ *
83
+ * @example
84
+ * ```typescript
85
+ * // Todos los usuarios activos ordenados por nombre
86
+ * const users = await firestoreService.getDocs<User>('users', {
87
+ * where: [{ field: 'active', operator: '==', value: true }],
88
+ * orderBy: [{ field: 'name', direction: 'asc' }],
89
+ * limit: 50
90
+ * });
91
+ * ```
92
+ */
93
+ async getDocs(collectionPath, options) {
94
+ const collectionRef = collection(this.firestore, collectionPath);
95
+ const constraints = this.buildQueryConstraints(options);
96
+ const q = query(collectionRef, ...constraints);
97
+ const snapshot = await getDocs(q);
98
+ return snapshot.docs.map((doc) => this.mapDocument(doc));
99
+ }
100
+ /**
101
+ * Obtiene documentos con paginación basada en cursores.
102
+ *
103
+ * @param collectionPath - Ruta de la colección
104
+ * @param options - Opciones de query (debe incluir limit)
105
+ * @returns Resultado paginado con cursor para la siguiente página
106
+ *
107
+ * @example
108
+ * ```typescript
109
+ * // Primera página
110
+ * const page1 = await firestoreService.getPaginated<User>('users', {
111
+ * orderBy: [{ field: 'createdAt', direction: 'desc' }],
112
+ * limit: 10
113
+ * });
114
+ *
115
+ * // Siguiente página
116
+ * if (page1.hasMore) {
117
+ * const page2 = await firestoreService.getPaginated<User>('users', {
118
+ * orderBy: [{ field: 'createdAt', direction: 'desc' }],
119
+ * limit: 10,
120
+ * startAfter: page1.lastDoc
121
+ * });
122
+ * }
123
+ * ```
124
+ */
125
+ async getPaginated(collectionPath, options) {
126
+ const collectionRef = collection(this.firestore, collectionPath);
127
+ const constraints = this.buildQueryConstraints(options);
128
+ // Pedir uno más para saber si hay más páginas
129
+ const q = query(collectionRef, ...constraints, limit(options.limit + 1));
130
+ const snapshot = await getDocs(q);
131
+ const docs = snapshot.docs;
132
+ const hasMore = docs.length > options.limit;
133
+ // Si hay más, remover el documento extra
134
+ const resultDocs = hasMore ? docs.slice(0, -1) : docs;
135
+ const lastDoc = resultDocs.length > 0 ? resultDocs[resultDocs.length - 1] : null;
136
+ return {
137
+ data: resultDocs.map((doc) => this.mapDocument(doc)),
138
+ hasMore,
139
+ lastDoc,
140
+ };
141
+ }
142
+ /**
143
+ * Verifica si un documento existe.
144
+ *
145
+ * @param collectionPath - Ruta de la colección
146
+ * @param docId - ID del documento
147
+ * @returns true si el documento existe
148
+ */
149
+ async exists(collectionPath, docId) {
150
+ const docRef = doc(this.firestore, collectionPath, docId);
151
+ const snapshot = await getDoc(docRef);
152
+ return snapshot.exists();
153
+ }
154
+ // ===========================================================================
155
+ // SUBSCRIPCIONES REAL-TIME (Observable)
156
+ // ===========================================================================
157
+ /**
158
+ * Suscribe a cambios de un documento (real-time).
159
+ *
160
+ * @param collectionPath - Ruta de la colección
161
+ * @param docId - ID del documento
162
+ * @returns Observable que emite cuando el documento cambia
163
+ *
164
+ * @example
165
+ * ```typescript
166
+ * // En el componente
167
+ * user$ = this.firestoreService.docChanges<User>('users', this.userId);
168
+ *
169
+ * // En el template
170
+ * @if (user$ | async; as user) {
171
+ * <p>{{ user.name }}</p>
172
+ * }
173
+ * ```
174
+ */
175
+ docChanges(collectionPath, docId) {
176
+ const docRef = doc(this.firestore, collectionPath, docId);
177
+ return docData(docRef, { idField: 'id' }).pipe(map((data) => {
178
+ if (!data)
179
+ return null;
180
+ return this.convertTimestamps(data);
181
+ }));
182
+ }
183
+ /**
184
+ * Suscribe a cambios de una colección (real-time).
185
+ *
186
+ * @param collectionPath - Ruta de la colección
187
+ * @param options - Opciones de query
188
+ * @returns Observable que emite cuando la colección cambia
189
+ *
190
+ * @example
191
+ * ```typescript
192
+ * // Usuarios activos en tiempo real
193
+ * activeUsers$ = this.firestoreService.collectionChanges<User>('users', {
194
+ * where: [{ field: 'status', operator: '==', value: 'online' }]
195
+ * });
196
+ * ```
197
+ */
198
+ collectionChanges(collectionPath, options) {
199
+ const collectionRef = collection(this.firestore, collectionPath);
200
+ const constraints = this.buildQueryConstraints(options);
201
+ const q = query(collectionRef, ...constraints);
202
+ return collectionData(q, { idField: 'id' }).pipe(map((docs) => docs.map((doc) => this.convertTimestamps(doc))));
203
+ }
204
+ // ===========================================================================
205
+ // ESCRITURA
206
+ // ===========================================================================
207
+ /**
208
+ * Agrega un documento con ID auto-generado.
209
+ *
210
+ * @param collectionPath - Ruta de la colección
211
+ * @param data - Datos del documento (sin id, createdAt, updatedAt)
212
+ * @returns Documento creado con su ID
213
+ *
214
+ * @example
215
+ * ```typescript
216
+ * const newUser = await firestoreService.addDoc<User>('users', {
217
+ * name: 'John Doe',
218
+ * email: 'john@example.com',
219
+ * role: 'user'
220
+ * });
221
+ * console.log('Created user with ID:', newUser.id);
222
+ * ```
223
+ */
224
+ async addDoc(collectionPath, data) {
225
+ const collectionRef = collection(this.firestore, collectionPath);
226
+ const timestamp = serverTimestamp();
227
+ const docData = {
228
+ ...data,
229
+ createdAt: timestamp,
230
+ updatedAt: timestamp,
231
+ };
232
+ const docRef = await addDoc(collectionRef, docData);
233
+ // Obtener el documento creado para retornarlo con timestamps resueltos
234
+ const snapshot = await getDoc(docRef);
235
+ return this.mapDocument(snapshot);
236
+ }
237
+ /**
238
+ * Crea o sobrescribe un documento con ID específico.
239
+ *
240
+ * @param collectionPath - Ruta de la colección
241
+ * @param docId - ID del documento
242
+ * @param data - Datos del documento
243
+ * @param options - Opciones (merge: true para merge en lugar de sobrescribir)
244
+ *
245
+ * @example
246
+ * ```typescript
247
+ * // Sobrescribir completamente
248
+ * await firestoreService.setDoc<User>('users', 'user123', userData);
249
+ *
250
+ * // Merge con datos existentes
251
+ * await firestoreService.setDoc<User>('users', 'user123', { name: 'New Name' }, { merge: true });
252
+ * ```
253
+ */
254
+ async setDoc(collectionPath, docId, data, options) {
255
+ const docRef = doc(this.firestore, collectionPath, docId);
256
+ const timestamp = serverTimestamp();
257
+ const docData = {
258
+ ...data,
259
+ updatedAt: timestamp,
260
+ ...(options?.merge ? {} : { createdAt: timestamp }),
261
+ };
262
+ await setDoc(docRef, docData, { merge: options?.merge ?? false });
263
+ }
264
+ /**
265
+ * Actualiza campos específicos de un documento.
266
+ *
267
+ * @param collectionPath - Ruta de la colección
268
+ * @param docId - ID del documento
269
+ * @param data - Campos a actualizar
270
+ *
271
+ * @example
272
+ * ```typescript
273
+ * await firestoreService.updateDoc<User>('users', 'user123', {
274
+ * name: 'Updated Name',
275
+ * lastLogin: new Date()
276
+ * });
277
+ * ```
278
+ */
279
+ async updateDoc(collectionPath, docId, data) {
280
+ const docRef = doc(this.firestore, collectionPath, docId);
281
+ await updateDoc(docRef, {
282
+ ...data,
283
+ updatedAt: serverTimestamp(),
284
+ });
285
+ }
286
+ /**
287
+ * Elimina un documento.
288
+ *
289
+ * @param collectionPath - Ruta de la colección
290
+ * @param docId - ID del documento
291
+ *
292
+ * @example
293
+ * ```typescript
294
+ * await firestoreService.deleteDoc('users', 'user123');
295
+ * ```
296
+ */
297
+ async deleteDoc(collectionPath, docId) {
298
+ const docRef = doc(this.firestore, collectionPath, docId);
299
+ await deleteDoc(docRef);
300
+ }
301
+ // ===========================================================================
302
+ // OPERACIONES EN LOTE
303
+ // ===========================================================================
304
+ /**
305
+ * Ejecuta múltiples operaciones de escritura de forma atómica.
306
+ *
307
+ * @param operations - Función que recibe el batch y agrega operaciones
308
+ *
309
+ * @example
310
+ * ```typescript
311
+ * await firestoreService.batch((batch) => {
312
+ * batch.set('users/user1', { name: 'User 1' });
313
+ * batch.update('users/user2', { status: 'inactive' });
314
+ * batch.delete('users/user3');
315
+ * });
316
+ * ```
317
+ */
318
+ async batch(operations) {
319
+ const batch = writeBatch(this.firestore);
320
+ const batchApi = {
321
+ set: (path, data) => {
322
+ const [collectionPath, docId] = this.splitPath(path);
323
+ const docRef = doc(this.firestore, collectionPath, docId);
324
+ batch.set(docRef, {
325
+ ...data,
326
+ createdAt: serverTimestamp(),
327
+ updatedAt: serverTimestamp(),
328
+ });
329
+ },
330
+ update: (path, data) => {
331
+ const [collectionPath, docId] = this.splitPath(path);
332
+ const docRef = doc(this.firestore, collectionPath, docId);
333
+ batch.update(docRef, {
334
+ ...data,
335
+ updatedAt: serverTimestamp(),
336
+ });
337
+ },
338
+ delete: (path) => {
339
+ const [collectionPath, docId] = this.splitPath(path);
340
+ const docRef = doc(this.firestore, collectionPath, docId);
341
+ batch.delete(docRef);
342
+ },
343
+ };
344
+ operations(batchApi);
345
+ await batch.commit();
346
+ }
347
+ // ===========================================================================
348
+ // UTILIDADES
349
+ // ===========================================================================
350
+ /**
351
+ * Construye una ruta a partir de un template.
352
+ *
353
+ * @param template - Template con placeholders {param}
354
+ * @param params - Valores para los placeholders
355
+ * @returns Ruta construida
356
+ *
357
+ * @example
358
+ * ```typescript
359
+ * const path = firestoreService.buildPath('users/{userId}/documents/{docId}', {
360
+ * userId: 'user123',
361
+ * docId: 'doc456'
362
+ * });
363
+ * // => 'users/user123/documents/doc456'
364
+ * ```
365
+ */
366
+ buildPath(template, params) {
367
+ return buildPath(template, params);
368
+ }
369
+ /**
370
+ * Genera un ID único para un documento (sin crearlo).
371
+ *
372
+ * @param collectionPath - Ruta de la colección
373
+ * @returns ID único generado por Firestore
374
+ */
375
+ generateId(collectionPath) {
376
+ const collectionRef = collection(this.firestore, collectionPath);
377
+ return doc(collectionRef).id;
378
+ }
379
+ /**
380
+ * Retorna un valor de timestamp del servidor.
381
+ * Usar en campos de fecha para que Firestore asigne el timestamp.
382
+ */
383
+ serverTimestamp() {
384
+ return serverTimestamp();
385
+ }
386
+ /**
387
+ * Retorna un valor para agregar elementos a un array.
388
+ *
389
+ * @example
390
+ * ```typescript
391
+ * await firestoreService.updateDoc('users', 'user123', {
392
+ * tags: firestoreService.arrayUnion('new-tag')
393
+ * });
394
+ * ```
395
+ */
396
+ arrayUnion(...elements) {
397
+ return arrayUnion(...elements);
398
+ }
399
+ /**
400
+ * Retorna un valor para remover elementos de un array.
401
+ */
402
+ arrayRemove(...elements) {
403
+ return arrayRemove(...elements);
404
+ }
405
+ /**
406
+ * Retorna un valor para incrementar un campo numérico.
407
+ *
408
+ * @example
409
+ * ```typescript
410
+ * await firestoreService.updateDoc('users', 'user123', {
411
+ * loginCount: firestoreService.increment(1)
412
+ * });
413
+ * ```
414
+ */
415
+ increment(n) {
416
+ return increment(n);
417
+ }
418
+ // ===========================================================================
419
+ // MÉTODOS PRIVADOS
420
+ // ===========================================================================
421
+ /**
422
+ * Construye los QueryConstraints a partir de QueryOptions
423
+ */
424
+ buildQueryConstraints(options) {
425
+ const constraints = [];
426
+ if (!options)
427
+ return constraints;
428
+ // Where clauses
429
+ if (options.where) {
430
+ for (const clause of options.where) {
431
+ constraints.push(where(clause.field, clause.operator, clause.value));
432
+ }
433
+ }
434
+ // OrderBy clauses
435
+ if (options.orderBy) {
436
+ for (const clause of options.orderBy) {
437
+ constraints.push(orderBy(clause.field, clause.direction));
438
+ }
439
+ }
440
+ // Cursors para paginación
441
+ if (options.startAfter) {
442
+ constraints.push(startAfter(options.startAfter));
443
+ }
444
+ if (options.startAt) {
445
+ constraints.push(startAt(options.startAt));
446
+ }
447
+ if (options.endBefore) {
448
+ constraints.push(endBefore(options.endBefore));
449
+ }
450
+ if (options.endAt) {
451
+ constraints.push(endAt(options.endAt));
452
+ }
453
+ // Limit (se agrega al final)
454
+ if (options.limit) {
455
+ constraints.push(limit(options.limit));
456
+ }
457
+ return constraints;
458
+ }
459
+ /**
460
+ * Mapea un DocumentSnapshot a nuestro tipo
461
+ */
462
+ mapDocument(snapshot) {
463
+ const data = snapshot.data();
464
+ if (!data) {
465
+ throw new Error('Documento no tiene datos');
466
+ }
467
+ return {
468
+ id: snapshot.id,
469
+ ...this.convertTimestamps(data),
470
+ };
471
+ }
472
+ /**
473
+ * Convierte Timestamps de Firestore a Date de JavaScript
474
+ */
475
+ convertTimestamps(data) {
476
+ const result = {};
477
+ for (const [key, value] of Object.entries(data)) {
478
+ if (value instanceof Timestamp) {
479
+ result[key] = value.toDate();
480
+ }
481
+ else if (value && typeof value === 'object' && !Array.isArray(value)) {
482
+ result[key] = this.convertTimestamps(value);
483
+ }
484
+ else {
485
+ result[key] = value;
486
+ }
487
+ }
488
+ return result;
489
+ }
490
+ /**
491
+ * Divide una ruta de documento en colección e ID
492
+ */
493
+ splitPath(path) {
494
+ const segments = path.split('/');
495
+ if (segments.length < 2 || segments.length % 2 !== 0) {
496
+ throw new Error(`Ruta de documento inválida: ${path}`);
497
+ }
498
+ const docId = segments.pop();
499
+ const collectionPath = segments.join('/');
500
+ return [collectionPath, docId];
501
+ }
502
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirestoreService, deps: [{ token: i1.Firestore }], target: i0.ɵɵFactoryTarget.Injectable }); }
503
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirestoreService, providedIn: 'root' }); }
504
+ }
505
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirestoreService, decorators: [{
506
+ type: Injectable,
507
+ args: [{ providedIn: 'root' }]
508
+ }], 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":"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","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"]}
@@ -0,0 +1,49 @@
1
+ /**
2
+ * Firebase Services
3
+ *
4
+ * Servicios reutilizables para integración con Firebase.
5
+ *
6
+ * @example
7
+ * ```typescript
8
+ * // En main.ts
9
+ * import { provideValtechFirebase } from 'valtech-components';
10
+ *
11
+ * bootstrapApplication(AppComponent, {
12
+ * providers: [
13
+ * provideValtechFirebase({
14
+ * firebase: environment.firebase,
15
+ * persistence: true,
16
+ * }),
17
+ * ],
18
+ * });
19
+ *
20
+ * // En componentes
21
+ * import { FirebaseService, FirestoreService } from 'valtech-components';
22
+ *
23
+ * @Component({...})
24
+ * export class MyComponent {
25
+ * private firebase = inject(FirebaseService);
26
+ * private firestore = inject(FirestoreService);
27
+ * }
28
+ * ```
29
+ */
30
+ // Tipos
31
+ export * from './types';
32
+ // Configuración
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';
36
+ // Servicios
37
+ export { FirebaseService } from './firebase.service';
38
+ // Firestore
39
+ export { FirestoreService } from './firestore.service';
40
+ // Firestore Collections (Factory Pattern)
41
+ export { FirestoreCollectionFactory, TypedCollection, } from './firestore-collection';
42
+ // Utilidades
43
+ export { QueryBuilder, query } from './utils/query-builder';
44
+ export { buildPath, extractPathParams, getCollectionPath, getDocumentId, isCollectionPath, isDocumentPath, isValidPath, joinPath, } from './utils/path-builder';
45
+ // Storage
46
+ export { StorageService } from './storage.service';
47
+ // Messaging (FCM)
48
+ export { MessagingService } from './messaging.service';
49
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2ZpcmViYXNlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNEJHO0FBRUgsUUFBUTtBQUNSLGNBQWMsU0FBUyxDQUFDO0FBRXhCLGdCQUFnQjtBQUNoQixPQUFPLEVBQUUsdUJBQXVCLEVBQUUsWUFBWSxFQUFFLHNCQUFzQixFQUFFLE1BQU0sVUFBVSxDQUFDO0FBRXpGLHdDQUF3QztBQUN4QyxPQUFPLEVBQ0wsT0FBTyxFQUNQLGlCQUFpQixFQUNqQixzQkFBc0IsRUFDdEIsV0FBVyxFQUNYLG9CQUFvQixFQUNwQixjQUFjLEVBQ2QsWUFBWSxHQUdiLE1BQU0saUJBQWlCLENBQUM7QUFFekIsWUFBWTtBQUNaLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUVyRCxZQUFZO0FBQ1osT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFFdkQsMENBQTBDO0FBQzFDLE9BQU8sRUFHTCwwQkFBMEIsRUFFMUIsZUFBZSxHQUNoQixNQUFNLHdCQUF3QixDQUFDO0FBRWhDLGFBQWE7QUFDYixPQUFPLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBQzVELE9BQU8sRUFDTCxTQUFTLEVBQ1QsaUJBQWlCLEVBQ2pCLGlCQUFpQixFQUNqQixhQUFhLEVBQ2IsZ0JBQWdCLEVBQ2hCLGNBQWMsRUFDZCxXQUFXLEVBQ1gsUUFBUSxHQUNULE1BQU0sc0JBQXNCLENBQUM7QUFFOUIsVUFBVTtBQUNWLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUVuRCxrQkFBa0I7QUFDbEIsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0scUJBQXFCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEZpcmViYXNlIFNlcnZpY2VzXG4gKlxuICogU2VydmljaW9zIHJldXRpbGl6YWJsZXMgcGFyYSBpbnRlZ3JhY2nDs24gY29uIEZpcmViYXNlLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiAvLyBFbiBtYWluLnRzXG4gKiBpbXBvcnQgeyBwcm92aWRlVmFsdGVjaEZpcmViYXNlIH0gZnJvbSAndmFsdGVjaC1jb21wb25lbnRzJztcbiAqXG4gKiBib290c3RyYXBBcHBsaWNhdGlvbihBcHBDb21wb25lbnQsIHtcbiAqICAgcHJvdmlkZXJzOiBbXG4gKiAgICAgcHJvdmlkZVZhbHRlY2hGaXJlYmFzZSh7XG4gKiAgICAgICBmaXJlYmFzZTogZW52aXJvbm1lbnQuZmlyZWJhc2UsXG4gKiAgICAgICBwZXJzaXN0ZW5jZTogdHJ1ZSxcbiAqICAgICB9KSxcbiAqICAgXSxcbiAqIH0pO1xuICpcbiAqIC8vIEVuIGNvbXBvbmVudGVzXG4gKiBpbXBvcnQgeyBGaXJlYmFzZVNlcnZpY2UsIEZpcmVzdG9yZVNlcnZpY2UgfSBmcm9tICd2YWx0ZWNoLWNvbXBvbmVudHMnO1xuICpcbiAqIEBDb21wb25lbnQoey4uLn0pXG4gKiBleHBvcnQgY2xhc3MgTXlDb21wb25lbnQge1xuICogICBwcml2YXRlIGZpcmViYXNlID0gaW5qZWN0KEZpcmViYXNlU2VydmljZSk7XG4gKiAgIHByaXZhdGUgZmlyZXN0b3JlID0gaW5qZWN0KEZpcmVzdG9yZVNlcnZpY2UpO1xuICogfVxuICogYGBgXG4gKi9cblxuLy8gVGlwb3NcbmV4cG9ydCAqIGZyb20gJy4vdHlwZXMnO1xuXG4vLyBDb25maWd1cmFjacOzblxuZXhwb3J0IHsgVkFMVEVDSF9GSVJFQkFTRV9DT05GSUcsIGhhc0VtdWxhdG9ycywgcHJvdmlkZVZhbHRlY2hGaXJlYmFzZSB9IGZyb20gJy4vY29uZmlnJztcblxuLy8gQ29uZmlndXJhY2nDs24gY29tcGFydGlkYSBkZWwgbW9ub3JlcG9cbmV4cG9ydCB7XG4gIEFQUF9JRFMsXG4gIEZJUkVCQVNFX1BST0pFQ1RTLFxuICBTSEFSRURfRU1VTEFUT1JfQ09ORklHLFxuICBjb2xsZWN0aW9ucyxcbiAgY3JlYXRlRmlyZWJhc2VDb25maWcsXG4gIGlzRW11bGF0b3JNb2RlLFxuICBzdG9yYWdlUGF0aHMsXG4gIHR5cGUgQXBwSWQsXG4gIHR5cGUgQ3JlYXRlRmlyZWJhc2VDb25maWdPcHRpb25zLFxufSBmcm9tICcuL3NoYXJlZC1jb25maWcnO1xuXG4vLyBTZXJ2aWNpb3NcbmV4cG9ydCB7IEZpcmViYXNlU2VydmljZSB9IGZyb20gJy4vZmlyZWJhc2Uuc2VydmljZSc7XG5cbi8vIEZpcmVzdG9yZVxuZXhwb3J0IHsgRmlyZXN0b3JlU2VydmljZSB9IGZyb20gJy4vZmlyZXN0b3JlLnNlcnZpY2UnO1xuXG4vLyBGaXJlc3RvcmUgQ29sbGVjdGlvbnMgKEZhY3RvcnkgUGF0dGVybilcbmV4cG9ydCB7XG4gIENvbGxlY3Rpb25PcHRpb25zLFxuICBGaXJlc3RvcmVDb2xsZWN0aW9uLFxuICBGaXJlc3RvcmVDb2xsZWN0aW9uRmFjdG9yeSxcbiAgU3ViQ29sbGVjdGlvblJlZixcbiAgVHlwZWRDb2xsZWN0aW9uLFxufSBmcm9tICcuL2ZpcmVzdG9yZS1jb2xsZWN0aW9uJztcblxuLy8gVXRpbGlkYWRlc1xuZXhwb3J0IHsgUXVlcnlCdWlsZGVyLCBxdWVyeSB9IGZyb20gJy4vdXRpbHMvcXVlcnktYnVpbGRlcic7XG5leHBvcnQge1xuICBidWlsZFBhdGgsXG4gIGV4dHJhY3RQYXRoUGFyYW1zLFxuICBnZXRDb2xsZWN0aW9uUGF0aCxcbiAgZ2V0RG9jdW1lbnRJZCxcbiAgaXNDb2xsZWN0aW9uUGF0aCxcbiAgaXNEb2N1bWVudFBhdGgsXG4gIGlzVmFsaWRQYXRoLFxuICBqb2luUGF0aCxcbn0gZnJvbSAnLi91dGlscy9wYXRoLWJ1aWxkZXInO1xuXG4vLyBTdG9yYWdlXG5leHBvcnQgeyBTdG9yYWdlU2VydmljZSB9IGZyb20gJy4vc3RvcmFnZS5zZXJ2aWNlJztcblxuLy8gTWVzc2FnaW5nIChGQ00pXG5leHBvcnQgeyBNZXNzYWdpbmdTZXJ2aWNlIH0gZnJvbSAnLi9tZXNzYWdpbmcuc2VydmljZSc7XG4iXX0=