firebase-manifest-compiler v1.0.3
Firebase manifest compiler WIP
Aims to take care of redundant firebase tasks that could be skipped by providing data schemes
Only for firestore atm, realtime database & security rules will follow, might implement links to other gcp services later, it all depends on what I need on my own project but PRs are welcomed
Experimental
Features
- Write a schema once and never hardcode string paths to your document references ever again, - firestore.doc('chatrooms/{roomId}/messages/{messageId}')simply becomes- db.room(id).message(msg_id)for documents and- db.room(id).messagesfor the collection itself.
- Save on bandwidth & storage without compromising on code readability. The compiler automatically adds in functions to map collection, document and property names to single characters of your choosing such that you get the full - { name, username, picture_url }object client and server side ( with VSCode autocompletion WIP ) while the database only stores- { n, u, p }. The library comes with wrappers that allow this kind of thing to happen even with deeply nested document data objects.
- Hardcoded documents within nested collections are collapsed in the compiled manifest, such that - users/{uid}/private/profilesimply becomes- db.user(uid).profilewhen there aren't conflicts with other subcollections or wildcard documents within- /private
- Adding the - @preprocessproperty on a wildcard document allows you to change arguments before they are converted to paths, so that if you need to enforce username uniqueness for example, you'd set- '@preprocess' : 'username.toLowerCase()'once under the username document in the manifest and that'd get it lowercased anytime you access documents at- usernames/{username}, be it server or client side.
- Document properties can have defaults, and any property can be converted using an inline function 
Install
npm install --savedev firebase-manifest-compiler
Usage
import { compile_firestore_manifest } from 'firebase-manifest-compiler'
 
compile_firestore_manifest( myManifest, { path : './client.manifest.js' }, { path : './server.manifest.js' } )Example
Simple one level deep manifest :

Compiles down to :

calling myManifest.user(uid) will get you a wrapped DocumentReference, it's just like the one you'd get from firestore but this one converts the data from and to the database without you having to think about it ever again. 
If user had subcollections, they'd get passed as another argument to the function and assigned to the returned DocumentReference.
Manifest syntax
| Type | syntax | prerequisites | example | result | 
|---|---|---|---|---|
| collection [] | [min:name] : { [ hardset \| wildcard documents ] } | Must be property of root or property of a document | [u:users] | manifest.userswill give you a reference to theuserscollection | 
| wildcard document () | (name:argumentName?) : { [properties &\| collections ] } | Must be property of collection | (user:uid) | manifest.userwill be a function callable with an argument nameduidthat returns a wrapped instance of a firestoreDocumentReference | 
| hardset document {} | {min:name} : { [properties &\| collections ] } | Must be property of collection | {p:profile} | manifest.userwill be a function callable with an argument nameduidthat returns a wrapped instance of a firestoreDocumentReference | 
| document property '' | 'min:name' : 'type [= {default}]' | Must be property of a document or property of a property of defined type 'object' | 'n:name' : 'string = "unknown"' | The nameproperty will be renamed to and fromnwhen needed and set to"unknown"anytime it'sundefined( except when updating the document ) | 
| modifier @ | '@modifier:arg1' : 'arg2' | Must be property of a document or property of a property | (user:uid): { '@preprocess' : 'uid.toLowerCase()', 'n:name': { type : 'string', '@from': my_function, '@to' : my_other_function } } | uidwill be lower cased beforefirestore.doc(path)is called.name = my_function(name)will be called whennameis retrieved from the server andname = my_other_function(name)will be called when it goes the other way around | 
Exports
| function | arguments | 
|---|---|
| compile_firestore_manifest | manifest, clientOpts, serverOpts, commonOpts? | 
Options
| Property | Description | default | 
|---|---|---|
| path: string | output file path ( ex ./manifest.js) | undefined | 
| name: string | name of the compiled manifest variable | 'toFirestore' | 
| imports: string | optional imports in the output file ( e.g. firebase itself ) | '' | 
| firestore: string | defines firestore in the output file | client : typeof window !== 'undefined' ? firebase.firestore : noopserver :admin.firestore | 
| prettier: object | string | prettier config object for the output file, can also be a path to .prettierrc | { parser : 'babel' } | 
Todo
- Get non generic VSCode autocompletion working on .data()& alike
- Implement the @linkmodifier ( i.e. assign references to documents in other collections whose ID are one of the current document data properties to each of itsDocumentReferences &DocumentSnapshots )
- Implement the @bindmodifier ( ) //todo
- Implement wraps for collection refs & queries
- Include types in the output
- Security rules compiler
- Realtime database manifest compiler
- Write tests
.md written with StackEdit.