-
Notifications
You must be signed in to change notification settings - Fork 65
/
use-swr-collection.ts
562 lines (515 loc) Β· 17.5 KB
/
use-swr-collection.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
import useSWR, { mutate as mutateStatic, ConfigInterface } from 'swr'
import { fuego } from '../context'
import { useRef, useEffect, useMemo, useCallback } from 'react'
// import { useMemoOne as useMemo } from 'use-memo-one'
import { empty } from '../helpers/empty'
import { collectionCache } from '../classes/Cache'
// type Document<T = {}> = T & { id: string }
import {
FieldPath,
OrderByDirection,
WhereFilterOp,
Query,
} from '@firebase/firestore-types'
import { isDev } from '../helpers/is-dev'
import { withDocumentDatesParsed } from '../helpers/doc-date-parser'
import { Document } from '../types'
type KeyHack = string & {} // hack to also allow strings
// here we get the "key" from our data, to add intellisense for any "orderBy" in the queries and such.
type OrderByArray<Doc extends object = {}, Key = keyof Doc> = [
Key | FieldPath | KeyHack,
OrderByDirection
]
type OrderByItem<Doc extends object = {}, Key = keyof Doc> =
| OrderByArray<Doc>
| Key
| KeyHack
type OrderByType<Doc extends object = {}> =
| OrderByItem<Doc>
| OrderByArray<Doc>[]
type WhereItem<Doc extends object = {}, Key = keyof Doc> = [
Key | FieldPath | KeyHack,
WhereFilterOp,
unknown
]
type WhereArray<Doc extends object = {}> = WhereItem<Doc>[]
type WhereType<Doc extends object = {}> = WhereItem<Doc> | WhereArray<Doc>
export type CollectionQueryType<Doc extends object = {}> = {
limit?: number
orderBy?: OrderByType<Doc>
where?: WhereType<Doc>
isCollectionGroup?: boolean
/**
* For now, this can only be a number, since it has to be JSON serializable.
*
* **TODO** allow DocumentSnapshot here too. This will probably be used with a useStaticCollection hook in the future.
*/
startAt?: number
/**
* For now, this can only be a number, since it has to be JSON serializable.
*
* **TODO** allow DocumentSnapshot here too. This will probably be used with a useStaticCollection hook in the future.
*/
endAt?: number
/**
* For now, this can only be a number, since it has to be JSON serializable.
*
* **TODO** allow DocumentSnapshot here too. This will probably be used with a useStaticCollection hook in the future.
*/
startAfter?: number
/**
* For now, this can only be a number, since it has to be JSON serializable.
*
* **TODO** allow DocumentSnapshot here too. This will probably be used with a useStaticCollection hook in the future.
*/
endBefore?: number
// THESE ARE NOT JSON SERIALIZABLE
// startAt?: number | DocumentSnapshot
// endAt?: number | DocumentSnapshot
// startAfter?: number | DocumentSnapshot
// endBefore?: number | DocumentSnapshot
}
export const getCollection = async <Doc extends Document = Document>(
path: string,
// queryString: string = '{}',
query: CollectionQueryType<Doc> = {},
{
parseDates,
ignoreFirestoreDocumentSnapshotField,
}: {
parseDates?: (string | keyof Doc)[]
/**
* If `true`, docs returned in `data` will not include the firestore `__snapshot` field. If `false`, it will include a `__snapshot` field. This lets you access the document snapshot, but makes the document not JSON serializable.
*
* Default: `false`
*/
ignoreFirestoreDocumentSnapshotField?: boolean
} = empty.object
) => {
const ref = createFirestoreRef(path, query)
const data: Doc[] = await ref.get().then(querySnapshot => {
const array: typeof data = []
querySnapshot.forEach(doc => {
const docData =
doc.data({
serverTimestamps: 'estimate',
}) ?? empty.object
const docToAdd = withDocumentDatesParsed(
{
...docData,
id: doc.id,
exists: doc.exists,
hasPendingWrites: doc.metadata.hasPendingWrites,
__snapshot: ignoreFirestoreDocumentSnapshotField ? undefined : doc,
} as any,
parseDates
)
// update individual docs in the cache
mutateStatic(doc.ref.path, docToAdd, false)
if (
isDev &&
// @ts-ignore
(docData.exists || docData.id || docData.hasPendingWrites)
) {
console.warn(
'[get-collection] warning: Your document, ',
doc.id,
' is using one of the following reserved fields: [exists, id, hasPendingWrites]. These fields are reserved. Please remove them from your documents.'
)
}
array.push(docToAdd)
})
return array
})
return data
}
const createFirestoreRef = <Doc extends object = {}>(
path: string,
{
where,
orderBy,
limit,
startAt,
endAt,
startAfter,
endBefore,
isCollectionGroup,
}: CollectionQueryType<Doc>
) =>
// { isCollectionGroup = false }: { isCollectionGroup?: boolean } = empty.object
{
let ref: Query = fuego.db.collection(path)
if (isCollectionGroup) {
ref = fuego.db.collectionGroup(path)
}
if (where) {
function multipleConditions(w: WhereType<Doc>): w is WhereArray<Doc> {
return !!(w as WhereArray) && Array.isArray(w[0])
}
if (multipleConditions(where)) {
where.forEach(w => {
ref = ref.where(w[0] as string | FieldPath, w[1], w[2])
})
} else if (typeof where[0] === 'string' && typeof where[1] === 'string') {
ref = ref.where(where[0], where[1], where[2])
}
}
if (orderBy) {
if (typeof orderBy === 'string') {
ref = ref.orderBy(orderBy)
} else if (Array.isArray(orderBy)) {
function multipleOrderBy(
o: OrderByType<Doc>
): o is OrderByArray<Doc>[] {
return Array.isArray((o as OrderByArray<Doc>[])[0])
}
if (multipleOrderBy(orderBy)) {
orderBy.forEach(([order, direction]) => {
ref = ref.orderBy(order as string | FieldPath, direction)
})
} else {
const [order, direction] = orderBy
ref = ref.orderBy(order as string | FieldPath, direction)
}
}
}
if (startAt) {
ref = ref.startAt(startAt)
}
if (endAt) {
ref = ref.endAt(endAt)
}
if (startAfter) {
ref = ref.startAfter(startAfter)
}
if (endBefore) {
ref = ref.endBefore(endBefore)
}
if (limit) {
ref = ref.limit(limit)
}
return ref
}
type ListenerReturnType<Doc extends Document = Document> = {
initialData: Doc[] | null
unsubscribe: ReturnType<ReturnType<typeof fuego['db']['doc']>['onSnapshot']>
}
const createListenerAsync = async <Doc extends Document = Document>(
path: string,
queryString: string,
{
parseDates,
ignoreFirestoreDocumentSnapshotField = true,
}: // isCollectionGroup = false,
{
parseDates?: (string | keyof Doc)[]
/**
* If `true`, docs returned in `data` will not include the firestore `__snapshot` field. If `false`, it will include a `__snapshot` field. This lets you access the document snapshot, but makes the document not JSON serializable.
*
* Default: `true`
*/
ignoreFirestoreDocumentSnapshotField?: boolean
}
): Promise<ListenerReturnType<Doc>> => {
return new Promise(resolve => {
const query: CollectionQueryType = JSON.parse(queryString) ?? {}
const ref = createFirestoreRef(path, query)
const unsubscribe = ref.onSnapshot(
{ includeMetadataChanges: true },
querySnapshot => {
const data: Doc[] = []
querySnapshot.forEach(doc => {
const docData =
doc.data({
serverTimestamps: 'estimate',
}) ?? empty.object
const docToAdd = withDocumentDatesParsed(
{
...docData,
id: doc.id,
exists: doc.exists,
hasPendingWrites: doc.metadata.hasPendingWrites,
__snapshot: ignoreFirestoreDocumentSnapshotField
? undefined
: doc,
} as any,
parseDates
)
if (
isDev &&
// @ts-ignore
(docData.exists || docData.id || docData.hasPendingWrites)
) {
console.warn(
'[use-collection] warning: Your document, ',
doc.id,
' is using one of the following reserved fields: [exists, id, hasPendingWrites]. These fields are reserved. Please remove them from your documents.'
)
}
// update individual docs in the cache
mutateStatic(doc.ref.path, docToAdd, false)
data.push(docToAdd)
})
// resolve initial data
resolve({
initialData: data,
unsubscribe,
})
// update on listener fire
mutateStatic([path, queryString], data, false)
}
)
})
}
export type CollectionSWROptions<
Doc extends Document = Document
> = ConfigInterface<Doc[] | null>
/**
* Call a Firestore Collection
* @template Doc
* @param path String if the document is ready. If it's not ready yet, pass `null`, and the request won't start yet.
* @param [query] - Dictionary with options to query the collection *AND* optionally accepts `listen`, `parseDates`, and `ignoreFirestoreDocumentSnapshotField` as well.
* @param [options] - Dictionary of options to pass to the underlying useSWR library.
*/
export const useCollection = <
Data extends object = {},
Doc extends Document = Document<Data>
>(
path: string | null,
query: CollectionQueryType<Data> & {
/**
* If `true`, sets up a real-time subscription to the Firestore backend.
*
* Default: `false`
*/
listen?: boolean
/**
* An array of key strings that indicate where there will be dates in the document.
*
* Example: if your dates are in the `lastUpdated` and `user.createdAt` fields, then pass `{parseDates: ["lastUpdated", "user.createdAt"]}`.
*
* This will automatically turn all Firestore dates into JS Date objects, removing the need to do `.toDate()` on your dates.
*/
parseDates?: (string | keyof Doc)[]
/**
* If `true`, docs returned in `data` will not include the firestore `__snapshot` field. If `false`, it will include a `__snapshot` field. This lets you access the document snapshot, but makes the document not JSON serializable.
*
* Default: `true`
*/
ignoreFirestoreDocumentSnapshotField?: boolean
} = empty.object,
options: CollectionSWROptions<Doc> = empty.object
) => {
const unsubscribeRef = useRef<ListenerReturnType['unsubscribe'] | null>(null)
const {
where,
endAt,
endBefore,
startAfter,
startAt,
orderBy,
limit,
listen = false,
parseDates,
// __unstableCollectionGroup: isCollectionGroup = false,
isCollectionGroup,
ignoreFirestoreDocumentSnapshotField = true,
} = query
// if we're listening, the firestore listener handles all revalidation
const {
refreshInterval = listen ? 0 : undefined,
refreshWhenHidden = listen ? false : undefined,
refreshWhenOffline = listen ? false : undefined,
revalidateOnFocus = listen ? false : undefined,
revalidateOnReconnect = listen ? false : undefined,
dedupingInterval = listen ? 0 : undefined,
} = options
const swrOptions = {
...options,
refreshInterval,
refreshWhenHidden,
refreshWhenOffline,
revalidateOnFocus,
revalidateOnReconnect,
dedupingInterval,
}
// why not just put this into the ref directly?
// so that we can use the useEffect down below that triggers revalidate()
const memoQueryString = useMemo(
() =>
JSON.stringify({
where,
endAt,
endBefore,
startAfter,
startAt,
orderBy,
limit,
isCollectionGroup,
}),
[
endAt,
endBefore,
isCollectionGroup,
limit,
orderBy,
startAfter,
startAt,
where,
]
)
// we move this to a Ref
// why? because we shouldn't have to include it in the key
// if we do, then calling mutate() won't be consistent for all
// collections with the same path & query
// TODO figure out if this is the right behavior...probably not because of the paths. hm.
// TODO it's not, move this to the
// const isCollectionGroupQuery = useRef(isCollectionGroup)
// useEffect(() => {
// isCollectionGroupQuery.current = isCollectionGroup
// }, [isCollectionGroup])
const dateParser = useRef(parseDates)
useEffect(() => {
dateParser.current = parseDates
}, [parseDates])
// we move listen to a Ref
// why? because we shouldn't have to include "listen" in the key
// if we do, then calling mutate() won't be consistent for all
// collections with the same path & query
const shouldListen = useRef(listen)
useEffect(() => {
shouldListen.current = listen
})
const shouldIgnoreSnapshot = useRef(ignoreFirestoreDocumentSnapshotField)
useEffect(() => {
shouldIgnoreSnapshot.current = ignoreFirestoreDocumentSnapshotField
}, [ignoreFirestoreDocumentSnapshotField])
const swr = useSWR<Doc[] | null>(
// if the path is null, this means we don't want to fetch yet.
path === null ? null : [path, memoQueryString],
async (path: string, queryString: string) => {
if (shouldListen.current) {
if (unsubscribeRef.current) {
unsubscribeRef.current()
unsubscribeRef.current = null
}
const { unsubscribe, initialData } = await createListenerAsync<Doc>(
path,
queryString,
{
parseDates: dateParser.current,
ignoreFirestoreDocumentSnapshotField: shouldIgnoreSnapshot.current,
}
)
unsubscribeRef.current = unsubscribe
return initialData
}
const data = await getCollection<Doc>(
path,
JSON.parse(queryString) as CollectionQueryType<Doc>,
{
parseDates: dateParser.current,
ignoreFirestoreDocumentSnapshotField: shouldIgnoreSnapshot.current,
}
)
return data
},
swrOptions
)
// if listen or changes,
// we run revalidate.
// This triggers SWR to fetch again
// Why? because we don't want to put listen
// in the useSWR key. If we did, then we couldn't mutate
// based on query alone. If we had useSWR(['users', true]),
// but then a `users` fetch with `listen` set to `false` updated, it wouldn't mutate both.
// thus, we move the `listen` and option to a ref user in `useSWR`,
// and we call `revalidate` if it changes.
const mounted = useRef(false)
useEffect(() => {
// TODO should this only happen if listen is false? No, BC swr should revalidate on a change.
if (mounted.current) revalidateRef.current()
else mounted.current = true
}, [listen])
// this MUST be after the previous effect to avoid duplicate initial validations.
// only happens on updates, not initial mounting
const revalidateRef = useRef(swr.revalidate)
useEffect(() => {
revalidateRef.current = swr.revalidate
})
useEffect(() => {
// TODO should this only be for listen, since SWR updates with the others?
// also should it go before the useSWR?
return () => {
// clean up listener on unmount if it exists
if (unsubscribeRef.current) {
unsubscribeRef.current()
unsubscribeRef.current = null
}
}
// should depend on the path, queyr, and listen being the same...
}, [path, listen, memoQueryString])
// add the collection to the cache,
// so that we can mutate it from document calls later
useEffect(() => {
if (path) collectionCache.addCollectionToCache(path, memoQueryString)
}, [path, memoQueryString])
const { data, isValidating, revalidate, mutate, error } = swr
/**
* `add(data)`: Extends the Firestore document [`add` function](https://firebase.google.com/docs/firestore/manage-data/add-data).
* - It also updates the local cache using SWR's `mutate`. This will prove highly convenient over the regular `add` function provided by Firestore.
*/
const add = useCallback(
<T extends Data | Data[]>(
data: T
): Promise<T extends Data ? string : string[]> | null => {
if (!path) return null
const multiple = Array.isArray(data)
const dataArray = multiple ? (data as T[]) : [data]
const ref = fuego.db.collection(path)
const docsToAdd: Doc[] = (dataArray.map(doc => ({
...doc,
// generate IDs we can use that in the local cache that match the server
id: ref.doc().id,
})) as unknown) as Doc[] // solve this annoying TS bug π
// add to cache
if (!listen) {
// we only update the local cache if we don't have a listener set up
// why? because Firestore automatically handles this part for subscriptions
mutate(prevState => {
const state = prevState ?? empty.array
return [...state, ...docsToAdd]
}, false)
}
// add to network
const batch = fuego.db.batch()
docsToAdd.forEach(({ id, ...doc }) => {
// take the ID out of the document
batch.set(ref.doc(id), doc)
})
return batch.commit().then(() => {
const ids = docsToAdd.map(({ id }) => id)
const returnValue = multiple ? ids : ids[0]
return returnValue as T extends Data ? string : string[]
})
},
[listen, mutate, path]
)
return {
data,
isValidating,
revalidate,
mutate,
error,
add,
loading: !data && !error,
/**
* A function that, when called, unsubscribes the Firestore listener.
*
* The function can be null, so make sure to check that it exists before calling it.
*
* Note: This is not necessary to use. `useCollection` already unmounts the listener for you. This is only intended if you want to unsubscribe on your own.
*/
unsubscribe: unsubscribeRef.current,
}
}