-
-
Save JamieCurnow/cba3968a7f1e335d473632f9fc9f6e8b to your computer and use it in GitHub Desktop.
/** | |
* This Gist is part of a medium article - read here: | |
* https://jamiecurnow.medium.com/using-firestore-with-typescript-65bd2a602945 | |
*/ | |
// import firstore (obviously) | |
import { firestore } from "firebase-admin" | |
// Import or define your types | |
// import { YourType } from '~/@types' | |
interface YourType { | |
firstName: string | |
lastName: string | |
isGreat: boolean | |
blackLivesMatter: true | |
} | |
interface YourOtherType { | |
something: boolean | |
somethingElse: boolean | |
} | |
// This helper function pipes your types through a firestore converter | |
const converter = <T>() => ({ | |
toFirestore: (data: Partial<T>) => data, | |
fromFirestore: (snap: FirebaseFirestore.QueryDocumentSnapshot) => snap.data() as T | |
}) | |
// This helper function exposes a 'typed' version of firestore().collection(collectionPath) | |
// Pass it a collectionPath string as the path to the collection in firestore | |
// Pass it a type argument representing the 'type' (schema) of the docs in the collection | |
const dataPoint = <T>(collectionPath: string) => firestore().collection(collectionPath).withConverter(converter<T>()) | |
// Construct a database helper object | |
const db = { | |
// list your collections here | |
users: dataPoint<YourType>('users'), | |
userPosts: (userId: string) => dataPoint<YourOtherType>(`users/${userId}/posts`) | |
} | |
// export your helper | |
export { db } | |
export default db | |
/** | |
* Some examples of how to use: | |
*/ | |
const example = async (id: string) => { | |
// firestore just as you know it, but with types | |
const userDoc = await db.users.doc(id).get() | |
const { blackLivesMatter } = userDoc.data() | |
return blackLivesMatter === true // obviously | |
} | |
const createExample = async (userId: string) => { | |
await db.userPosts(userId).doc().create({ | |
something: false, | |
somethingElse: true | |
}) | |
} | |
// Always use set for updates as firestore doesn't type update function correctly yet! | |
const updateExample = async (id: string) => { | |
await db.users.doc(id).set({ | |
firstName: 'Jamie', | |
blackLivesMatter: true | |
}, { merge: true }) | |
} |
Is anyone using the generic converter with VueFire? Would love to see how you adapted it to work with this: https://vuefire.vuejs.org/guide/realtime-data.html#Firestore-withConverter-
Coming from google.
Seems like there are two FirestoreDataConverter
s, one from @firebase/firestore
and the other from firebase-admin/firestore
.
And fromFirestore
implementation for each is totally different.
I'm using a shared codebase. My solution was this:
// rename the imports
import { FirestoreDataConverter as FrontendFirestoreDataConverter } from "@firebase/firestore";
import { FirestoreDataConverter as BackendFirestoreDataConverter, DocumentData, QueryDocumentSnapshot } from "firebase-admin/firestore";
// create different converters for frontend and backend
export const FrontendResourceConverter: FrontendFirestoreDataConverter<Resource> = {
fromFirestore(snapshot, options) {
return {
id: snapshot.id,
...snapshot.data(options) as Omit<Resource, "id">,
}
},
toFirestore(resource) {
// eslint-disable-next-line @typescript-eslint/no-unused-vars
const { id, ...rest } = resource;
return rest;
}
}
export const BackendResourceConverter: BackendFirestoreDataConverter<Resource> = {
fromFirestore(snapshot: QueryDocumentSnapshot<DocumentData>) { // notice how there's no options here
const data = snapshot.data();
return {
id: snapshot.id,
...data as Omit<Resource, "id">,
}
},
toFirestore(resource) {
// eslint-disable-next-line @typescript-eslint/no-unused-vars
const { id, ...rest } = resource;
return rest;
}
}
I can't destruct the typed object returning from doc.data().
export interface User {
id: number;
name: string;
email: string;
}
// omitting converter & dataPoint functions
const db = {
users: dataPoint<User>("users"),
};
Then in a Cloud Function I have:
const userDoc = await db.users.doc("12345").get();
const { email } = userDoc.data(); // error message: "Property 'email' does not exist on type 'Partial<User> | undefined'"
This should pretty much do the trick of having typed results from Firestore, however, the error message above keeps preventing typescript from compiling the code.
Any thoughts on this?
Thanks!
@erayerdin As of latest versions (firebase 11.0.1
, firebase-admin 13.0.0
) types for both FirestoreDataConverter
s are identical. Check out js & node.
This uses Firebase v10.4.0, based on the
withConverter
example in the docs: https://firebase.google.com/docs/firestore/query-data/get-data#custom_objectsInstead of getting the document, this just returns the reference. The document is converted to/from a model class instance.