Package Exports
- firelord
This package does not declare an exports field, so the exports above have been automatically detected and optimized by JSPM instead. If any package subpath is missing, it is recommended to post an issue to the original package (firelord) to support the "exports" field. If that is not possible, create a JSPM override to customize the exports field for this package.
Readme
firelord(BETA, Nodejs ONLY)
🐤 admin firestore typing wrapper.
🚀 All read and write operation are typed, field path, field value, collection path, document path, all typed!
🔥 Automatically convert base type to corresponding read and write time(good at handling timestamp and field value).
✨ Api closely resemble firestore api, low learning curve.
⛲️ Out of box typescript support.
🦙 Purpose
Unfortunately withConverter is not enough to solve the type problems, there is still no feasible solutions to deal with type like date, firestore.Timestamp, number and array where different types in read, write and compare(query) are needed. This library is a wrapper that introduce deeper typing solution to handle each case.
Not only this library deal with data type, it also provide type safe for collection path, document path, firestore limitations(whenever is possible).
require typescript 4.1 and above
Overview:
- generate read(get operation), write type(set/update operation) and compare type(for query) for field value, example:- server timestamp: {write: Firestore.FieldValue, read: Firestore.Timestamp, compare: Date | Firestore.Timestamp}
- number: {write: FieldValue | number, read: number, compare:number}
- xArray: {write: x[] | FieldValue, read: x[], compare: x[]}
- see conversion table for more
 
- server timestamp: 
- Typed collection path and document path.- auto generate sub collection path type.
 
- auto generate updatedAtandcreatedAttimestamp.- auto update updatedAtserver timestamp to update operation.
- auto add createdAtandupdatedAtserver timestamp to create and set operation.
 
- auto update 
- much better where and orderBy clause- field value are typed accordingly to field path
- comparator are depend on field value, eg:you cannot apply array-containsoperator on non-array field path
- depend on comparators, you can or cannot chain orderBy due to firestore limitation, see image below. Go to Order And Limit for more information.
 

basically all read operation return read type data, all write operation require write type data and all query require compare type data, you only need to define base type and the wrapper generate the other 3 types for you.
You don't need to do any kind of manipulation onto read, write and compare types nor you need to utilize them.
the documentation explains how the types work, the library itself is intuitive in practice, thoroughly refer to the documentation only if you want to have better understanding on how the typing work.
You SHOULD NOT try to memorize how the typing work, keep in mind the purpose is not for you to fit into the type but is to let the type GUIDE you.
🦜 Getting Started
npm i firelordimport { firelord, FireLord } from 'firelord'
// use base type to generate read and write type
type User = FireLord.ReadWriteCreator<
    {
        name: string
        age: number
        birthday: Date
        joinDate: 'ServerTimestamp'
        beenTo: ('USA' | 'CANADA' | 'RUSSIA' | 'CHINA')[]
    }, // base type
    'Users', // collection path type
    string // document path type
>
// read type
type UserRead = User['read'] // {name: string, age:number, birthday:firestore.Timestamp, joinDate: firestore.Timestamp, beenTo:('USA' | 'CANADA' | 'RUSSIA' | 'CHINA')[], createdAt: Date | firestore.Timestamp, updatedAt: Date | firestore.Timestamp}
// write type
type UserWrite = User['write'] // {name: string, age:number|FirebaseFirestore.FieldValue, birthday:firestore.Timestamp | Date, joinDate:FirebaseFirestore.FieldValue, beenTo:('USA' | 'CANADA' | 'RUSSIA' | 'CHINA')[] | FirebaseFirestore.FieldValue, createdAt:'ServerTimestamp', updatedAt:'ServerTimestamp'}
// compare type
type UserCompare = User['compare'] // {name: string, age:number, birthday:Date | firestore.Timestamp, joinDate: Date | firestore.Timestamp, beenTo:('USA' | 'CANADA' | 'RUSSIA' | 'CHINA')[], createdAt: Date | firestore.Timestamp, updatedAt: Date | firestore.Timestamp}
// implement wrapper
const userCreator = firelord<User>()
// collection reference
const users = userCreator.col('Users') // collection path type is "Users"
// collection group reference
const userGroup = userCreator.colGroup('Users') // collection path type is "Users"
// user reference
const user = users.doc('1234567890') // document path is string
// subCollection of User
type Transaction = FireLord.ReadWriteCreator<
    {
        amount: number
        date: 'ServerTimestamp'
        status: 'Fail' | 'Success'
    }, // base type
    'Transactions', // collection path type
    string, // document path type
    User // insert parent collection, it will auto construct the collection path for you
>
// implement the wrapper
const transactions = firelord<Transaction>().col('Users/283277782/Transactions') // the type for col is `User/${string}/Transactions`
const transaction = users.doc('1234567890') // document path is string🦔 Conversion Table
| Base | Read | Write | Compare | 
|---|---|---|---|
| number | number | number | FirebaseFirestore.FieldValue (firestore.FieldValue.increment*) | number | 
| string | string | string | string | 
| null | null | null | null | 
| undefined | undefined | undefined | undefined | 
| Date*** | firestore.Timestamp | firestore.Timestamp |Date | firestore.Timestamp |Date | 
| firestore.Timestamp*** | firestore.Timestamp | firestore.Timestamp |Date | firestore.Timestamp |Date | 
| 'ServerTimestamp' | firestore.Timestamp | FirebaseFirestore.FieldValue (firestore.FieldValue.serverTimestamp*) |FirebaseFirestore.FieldValue(firestore.FieldValue.arrayRemove/arrayUnion*) | firestore.Timestamp |Date | 
| object | not supported** | not supported** | not supported** | 
| number[] | number[] | number[] |FirebaseFirestore.FieldValue(firestore.FieldValue.arrayRemove/arrayUnion*) | number[] | 
| string[] | string[] | string[] |FirebaseFirestore.FieldValue(firestore.FieldValue.arrayRemove/arrayUnion*) | string[] | 
| null[] | null[] | null[] |FirebaseFirestore.FieldValue(firestore.FieldValue.arrayRemove/arrayUnion*) | null[] | 
| undefined[] | undefined[] | undefined[] |FirebaseFirestore.FieldValue(firestore.FieldValue.arrayRemove/arrayUnion*) | undefined[] | 
| Date[]*** | firestore.Timestamp[] | (firestore.Timestamp |Date )[] |FirebaseFirestore.FieldValue(firestore.FieldValue.arrayRemove/arrayUnion*) | (Date | firestore.Timestamp)[] | 
| firestore.Timestamp[]*** | firestore.Timestamp[] | (firestore.Timestamp |Date )[] |FirebaseFirestore.FieldValue(firestore.FieldValue.arrayRemove/arrayUnion*) | (Date | firestore.Timestamp)[] | 
| 'ServerTimestamp'[] | firestore.Timestamp[] | FirebaseFirestore.FieldValue (firestore.FieldValue.serverTimestamp*)[] |FirebaseFirestore.FieldValue(firestore.FieldValue.arrayRemove/arrayUnion*) | (Date | firestore.Timestamp)[] | 
| object[] | not supported** | not supported** | not supported** | 
| n-dimension array | n-dimension array | n-dimension array | FirebaseFirestore.FieldValue(firestore.FieldValue.arrayRemove/arrayUnion*) only supported for 1st dimension array | compare only elements in 1st dimension array | 
you can union any types, it will generates the types distributively, for example type string | number | number[] | (string | number)[] | (string | number)[][] | (string | number)[][][] generates:
read type: string | number | number[] | (string | number)[] | (string | number)[][] | (string | number)[][][]
write type: string | number | FirebaseFirestore.FieldValue | number[] | (string | number)[] | (string | number)[][] | (string | number)[][][]
compare type: string | number | number[] | (string | number)[] | (string | number)[][] | (string | number)[][][]
In practice, any union is not recommended, data should has only one type, except union with undefined or null that bear certain meaning(value missing or never initialized).
*I am not able to narrow down FirebaseFirestore.FieldValue, you might end up using increment on array or assign server time stamp on number or array union number onto string array field, solution is welcomed.
**It is not totally impossible to type object, however it is too painful to deal with right now, any elegant solution is welcomed. In practice, object type is not recommended, primitive data type can do thing just fine and easier to deal with in both usage and typing.
***Date | firestore.Timestamp, (Date | firestore.Timestamp)[], and Date[] | firestore.Timestamp[] unions are redundant, because Date and firestore.Timestamp generate same read, write and compare types.
🐘 Document operations: Write, Read and Listen
all the document operations api is similar to firestore write, read and listen.
// import user
import { firestore } from 'firebase-admin'
// get data(type is `read type`)
user.get().then(snapshot => {
    const data = snapshot.data()
})
// listen to data(type is `read type`)
user.onSnapshot(snapshot => {
    const data = snapshot.data()
})
const serverTimestamp = firestore.FieldValue.serverTimestamp()
// create if only exist, else fail
// require all members in `write type` except `updatedAt` and `createdAt`
// auto add `createdAt` and `updatedAt`
user.create({
    name: 'John',
    age: 24,
    birthday: new Date(1995, 11, 17),
    joinDate: serverTimestamp,
    beenTo: ['RUSSIA'],
})
// create if not exist, else overwrite
// although it can overwrite, this is intended to use as create
// require all members in `write type` except `updatedAt` and `createdAt`
// auto add `createdAt` and `updatedAt`
user.set({
    name: 'John',
    age: 24,
    birthday: new Date(1995, 11, 17),
    joinDate: serverTimestamp,
    beenTo: ['RUSSIA'],
})
// create if not exist, else update
// although it can create if not exist, this is intended to use as update
// all member are partial members, you can leave any of the member out, however typescript will stop you from explicitly assign `undefined` value to any of the member unless you union the type with `undefined` or mark it as optional in `base type`
// auto update `updatedAt`
// the only value for `merge` is `true`
// NOTE: there will be typescript missing property error if all member is not present, to fix this just fill in `{ merge:true }` in option as shown below.
user.set({ name: 'Michael' }, { merge: true })
// create if not exist, else update
// although it can create if not exist, this is intended to use as update
// all member are partial members, you can leave any of the member out, however typescript will stop you from explicitly assign `undefined` value to any of the member unless you union the type with `undefined` or mark it as optional in `base type`
// auto update `updatedAt`
// the merge keys are keys of `base type`
// NOTE: there will be typescript missing property error if all member is not present, to fix this just fill in `mergeField: [<keys>]` in option as shown below.
user.set(
    { name: 'Michael', age: 32, birthday: new Date(1987, 8, 9) },
    { mergeField: ['name', 'age'] } // update only `name` and `age` fields
)
// update if exist, else fail
// all member are partial members, you can leave any of the member out, however typescript will stop you from explicitly assign `undefined` value to any of the member unless you union the type with `undefined` or mark it as optional in `base type`
// auto update `updatedAt`
user.update({ name: 'Michael' })
// delete document
user.delete()🦩 Document operations: Batch
all api are similar to firestore batch, the only difference is, the batch is member of doc, hence you don't need to define document reference.
// import user
import { firestore } from 'firebase-admin'
// implement the wrapper
const user = firelord<User>().col('Users').doc('1234567890')
// create batch
const batch = firestore().batch()
const userBatch = user.batch(batch)
// delete document
userBatch.delete()
// create if exist, else fail
// require all members in `write type` except `updatedAt` and `createdAt`
// auto add `updatedAt` and `createdAt`
userBatch.create({ name: 'Michael', age: 32, birthday: new Date(1987, 8, 9) })
// update if exist, else fail
// all member are partial members, you can leave any of the member out, however typescript will stop you from explicitly assign `undefined` value to any of the member unless you union the type with `undefined` or mark it as optional in `base type`
// auto update `updatedAt`
userBatch.update({ name: 'Ozai' })
//commit
batch.commit()🐠 Document Operations: Transaction
all api are similar to firestore transaction, the only difference is, the batch is member of doc, hence you don't need to define document reference.
// import user
user.runTransaction(async transaction => {
    // get `read type` data
    await transaction.get().then(snapshot => {
        const data = snapshot.data()
    })
    // create if only exist, else fail
    // require all members in `write type` except `updatedAt` and `createdAt`
    // auto add `createdAt` and `updatedAt`
    await transaction.create({
        name: 'John',
        age: 24,
        birthday: new Date(1995, 11, 17),
        joinDate: serverTimestamp,
        beenTo: ['RUSSIA'],
    })
    // create if not exist, else overwrite
    // although it can overwrite, this is intended to use as create
    // require all members in `write type` except `updatedAt` and `createdAt`
    // auto add `createdAt` and `updatedAt`
    user.set({
        name: 'John',
        age: 24,
        birthday: new Date(1995, 11, 17),
        joinDate: serverTimestamp,
        beenTo: ['RUSSIA'],
    })
    // create if not exist, else update
    // although it can create if not exist, this is intended to use as update
    // all member are partial members, you can leave any of the member out, however typescript will stop you from explicitly assign `undefined` value to any of the member unless you union the type with `undefined` or mark it as optional in `base type`
    // auto update `updatedAt`
    // the only value for `merge` is `true`
    // NOTE: there will be typescript missing property error if all member is not present, to fix this just fill in `{ merge:true }` in option as shown below.
    await transaction.set({ name: 'Michael' }, { merge: true })
    // create if not exist, else update
    // although it can create if not exist, this is intended to use as update
    // all member are partial members, you can leave any of the member out, however typescript will stop you from explicitly assign `undefined` value to any of the member unless you union the type with `undefined` or mark it as optional in `base type`
    // auto update `updatedAt`
    // the merge keys are keys of `base type`
    // NOTE: there will be typescript missing property error if all member is not present, to fix this just fill in `mergeField: [<keys>]` in option as shown below.
    await transaction.set(
        { name: 'Michael', age: 32, birthday: new Date(1987, 8, 9) },
        { mergeField: ['name', 'age'] } // update only `name` and `age` fields
    )
    // update if exist, else fail
    // all member are partial members, you can leave any of the member out, however typescript will stop you from explicitly assign `undefined` value to any of the member unless you union the type with `undefined` or mark it as optional in `base type`
    // auto update `updatedAt`
    await transaction.update({ name: 'Michael' })
    // delete document
    await transaction.delete()
    // keep in mind you need to return promise in transaction
    // example code here is just example to show api, this is not the correct way to do it
    // refer back firestore guide https://firebase.google.com/docs/firestore/manage-data/transactions
    return Promise.resolve('')
})🌞 Collection Operations: Query
all the api are similar to firestore query, clauses are chain-able.
// import users
// non array data type
// the field path is the keys of the `base type`
// type of opStr is '<' | '<=' | '==' | '!=' | '>=' | '>' | 'not-in' | 'in'
// if type of opStr is '<' | '<=' | '==' | '!=' | '>=' | '>', the value type is same as the member's type in `compare type`
users.where('name', '==', 'John').get()
// if type of opStr is 'not-in' | 'in', the value type is array of member's type in `compare type`
users.where('name', 'in', ['John', 'Michael']).get()
// array data type
// the field path is the keys of the `base type`
// type of `opStr` is  'in' | 'array-contains-any'
// if type of opStr is 'array-contains', the value type is the non-array version of member's type in `compare type`
users.where('beenTo', 'array-contains', 'USA').get()
// if type of opStr is 'array-contains-any', the value type is same as the member's type in `compare type`
users.where('beenTo', 'array-contains-any', ['USA']).get()
// if type of opStr is 'in', the value type is the array of member's type in `compare type`
users.where('beenTo', 'in', [['CANADA', 'RUSSIA']]).get()🐳 Collection Operations: Order And Limit
all the api are similar to firestore order and limit with slight different, but work the same, clauses are chain-able.
// import users
// orderBy field path only include members that is NOT array type in `compare type`
users.orderBy('name', 'desc').limit(3).get()
// you can chain `orderBy` claus with SAME field path as `where` clause if the comparator is `!=`
users.where('age', '!=', 20).orderBy('age', 'desc').get()
// you can chain `orderBy` claus with DIFFERENT field path as `where` clause if the comparator is `array-contains` or `array-contains-any`
users.where('beenTo', 'array-contains', 'USA').orderBy('age', 'desc').get()
users
    .where('beenTo', 'array-contains-any', ['USA', 'CHINA'])
    .orderBy('age', 'desc')
    .get()
// You cannot order your query by any field included in an equality `==` or `in` clause
// https://firebase.google.com/docs/firestore/query-data/order-limit-data#limitations
users.where('age', '==', 20).orderBy('age', 'desc').get() // ERROR: Property 'orderBy' does not exist
users.where('age', 'in', [20, 30]).orderBy('age', 'desc').get() // ERROR: Property 'orderBy' does not exist
// the first orderBy must have the same field path as `where` clause with <, <=, >, >= comparators
// https://firebase.google.com/docs/firestore/query-data/order-limit-data#limitations
// whenever <, <=, >, >= comparators is used, they cannot chain the first orderBy, you need to use type safe shorthand shown in example below
users.where('age', '>', 20).orderBy('name', 'desc').get() // ERROR: Property 'orderBy' does not exist
// this is also invalid according to firestore nor it make any sense
// we prepare type safe shorthand to handle <, <=, >, >= comparators
// for <, <=, >, >= comparators, the optional 4th parameter(orderBy config object) is available, else the 4th parameter's type is `never`(should not exist)
// field value type is the corresponding field path value type in `compare type`
users.where('age', '>', 20, {}).get() // equivalent to where('age', '>', 20).orderBy('age') or where('age', '>', 20).orderBy('age','asc')
users.where('age', '<=', 20, { directionStr: 'desc' }).get() // equivalent to where('age', '>', 20).orderBy('age','desc')
users
    .where('age', '>=', 20, {
        directionStr: 'desc',
        cursor: { clause: 'endAt', fieldValue: 50 },
    })
    .get() // equivalent to where('age', '>', 20).orderBy('age','desc').endAt(50)
// you of course not able to use `==` and `in` comparator in shorthand as stated in limitation
users.where('age', '==', 20, { directionStr: 'desc' }).get() // ERROR: Argument of type '{}' is not assignable to parameter of type 'undefined'
users.where('age', 'in', [20], { directionStr: 'desc' }).get() // ERROR: Argument of type '{}' is not assignable to parameter of type 'undefined'
// for `not-in` comparator, optional `fieldPath` config member is available in the 4th parameter, (else the config member's type is `never`)
// if `fieldPath` is undefined, it will use the same `fieldPath` as where clause
// field value type is the corresponding field path value type in `compare type`
users
    .where('name', 'not-in', ['John', 'Ozai'], {
        fieldPath: 'age',
        directionStr: 'desc',
        cursor: { clause: 'endAt', fieldValue: 50 },
    })
    .get() // equivalent to where('name', 'not-in', ['John', 'Ozai']).orderBy('age','desc').endAt(50)🌺 Collection Operations: Paginate And Cursor
api are slightly different than firestore paginate and cursor, the cursors became orderBy parameter, it still work the same as firestore original api, clauses are chain-able.
// import users
// field path only include members that is NOT array type in `base type`
// field value type is the corresponding field path value type in `compare type`
// value of cursor clause is 'startAt' | 'startAfter' | 'endAt' | 'endBefore'
users.orderBy('age', 'asc', { clause: 'startAt', fieldValue: 20 }).offset(5) // equivalent to orderBy("age").startAt(20).offset(5)🌵 Collection Group
Api is exactly same as Collection Operations: Query, Order And Limit, Paginate And Cursor
just use collection group reference instead of collection reference, refer back Getting Started on how to create collection group reference
🐇 Limitation
While the wrapper try to solve as much as possible, some problem cannot be solved due to typing difficulty, or require huge effort to implement, or straight up not can be solved.
- object data type is not supported. 
- FirebaseFirestore.FieldValue is not narrowed down. 
- no type safe measurement for Query Limitation. 
🐕 Opinion
One element of the wrapper is opinionated, that is createdAt and updatedAt timestamp that add or update automatically.
when a document is created via add, create or set without option, two things will happen:
- createdAt field path is created and the value is firestore server timestamp(current server timestamp).
- updatedAt field path is created and the value is new Date(0), it starts at beginning of the time.
when a document is updated via update or set with option, updatedAt field path is updated and the value is firestore server timestamp.