npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

@razaman2/collection-proxy

v0.0.22

Published

an opinionated approach to reading and writing documents in firebase/firestore.

Downloads

814

Readme

@razaman2/collection-proxy

A custom firestore client that provides a simple and consistent interface for interacting with the database.

Install the dependencies

npm install --save @razaman2/collection-proxy
pnpm add @razaman2/collection-proxy

THINGS YOU SHOULD KNOW

All documents are created with the auto generated properties outlined by the following json object.

{
   "belongsTo": [
      "document-path"
   ],
   "createdAt": "serverTimestamp",
   "updatedAt": "serverTimestamp",
   "id": "document-id"
}
  1. The belongsTo is an array that stores all the document paths that has a relationship with the current document. The relationship will always include the current document path.

A settings document with id 111, that belongs to a user with id 222, and a company with id 333, would look like this {belongsTo: ["settings/111", "users/222", "companies/333"]}. This relationship is intended to be queried using the array-contains or the array-contains-any operators. This approach provides the ability to run queries like get all settings documents that belong to user with id 222 or get all settings documents that belong to company with id 333 or even get all settings documents that belong to user with id 222 and company 333. The latter would require that when adding relationships, you concatenate the permutations you want to use for querying. {belongsTo: ["111 settings", "222 users", "333 companies", "222 users|333 companies]} now you can query a settings document that belongs to a specific user and company.

  1. The createdAt key stores a firestore serverTimestamp that represents exactly when the document was created.
  2. The updatedAt key stores a firestore serverTimestamp that represents the last time the document was updated. It will be automatically updated when the document is updated using the update method from the firestore-proxy.
  3. The id key stores the assigned document id on the document data.

Initializing the Collection

The firestore-proxy is compatible with both the namespaced and the modular firestore api's. The collection has a static configuration which allows you to configure the api once in the entry point of your app.

be sure to initialize your firestore app!

Namespaced API, including the Admin SDK

All you need to do to configure the namespaced api is to provide it with your firestore instance.

import firebase from "firebase/compat/app";
import "firebase/compat/firestore";
import {Collection} from "@razaman2/collection-proxy";

firebase.initializeApp({projectId: "demo-app"});

Collection.setConfig({
    getFirestore: firestore,
    // you can enable operation logging to console
    logging: true,
});

// And thats it!

Modular API

Import the functions from the firestore library and provide them to the collection config. You can provide only the functions that are needed for the operation you are trying to perform.

import {initializeApp} from "firebase/app";
import {getFirestore, collection, doc, getDoc, getDocs, collectionGroup, onSnapshot, setDoc, updateDoc, deleteDoc, deleteField, arrayUnion, serverTimestamp, writeBatch} from "firebase/firestore";
import {Collection} from "@razaman2/collection-proxy";

initializeApp({projectId: "demo-app"});

Collection.setConfig({
    // you can enable operation logging to console
    logging: true,
   
    // base setup to start working with the collection
    modular: true,
    getFirestore: getFirestore(),
    collection,
    doc,

    // when reading single document
    getDoc,

    // when reading multiple documents
    getDocs,

    // when reading from a collection group
    collectionGroup,

    // when reading documents using a snapshot
    onSnapshot,

    // when creating, updating or deleting documents
    setDoc,

    // when updating documents
    updateDoc,

    // when deleting documents
    deleteDoc,

    // when you want to delete undefined document fields
    deleteField,

    // when creating documents
    arrayUnion,

    // when creating or updating documents
    serverTimestamp,

    // when updating documents
    writeBatch,
});

Reading Documents

// init with document id
const collection = await Collection.setCollectionName("users").init(1);

const data = collection.getData();
// get document for provided id
const collection = await Collection.setCollectionName("users").getDocument(1, {realtime: false});

const data = collection.getData();
// set
const collection = await Collection.setCollectionName("users").setDoc(1).init();

Creating Documents

// create document with auto-generated id
await Collection.setCollectionName("tests").create({
    data: {name: "john doe"},
});
// create document with id from setDoc
await Collection.setCollectionName("tests").setDoc("123").create({
    data: {name: "john doe"},
});
// create document with id from payload data
await Collection.setCollectionName("tests", {
    payload: {
        data: {id: "123", name: "john doe"},
    },
}).create();
// create document with id from document path
await Collection.setPath("tests/123").create({
    data: {name: "john doe"},
});
// create document with setPath and setDoc
await Collection.setPath("tests").setDoc("123").create({
    data: {name: "john doe"},
});

Updating Documents

// update document with auto-generated id
await Collection.setCollectionName("tests").update({
    data: {name: "jane doe"},
});
// update document with id from setDoc
await Collection.setCollectionName("tests").setDoc("123").update({
    data: {name: "jane doe"},
});
// update document with id from payload data
await Collection.setCollectionName("tests", {
    payload: {
        data: {id: "123", name: "jane doe"},
    },
}).update();
// update document with id from document path
await Collection.setPath("tests/123").update({
    data: {name: "jane doe"},
});
// update document with setPath and setDoc
await Collection.setPath("tests").setDoc("123").update({
    data: {name: "jane doe"},
});

Deleting Documents

// delete document with id from setDoc
await Collection.setCollectionName("tests").setDoc("123").delete();
// delete document with id from payload data
await Collection.setCollectionName("tests").setData({id: "123"}).delete();
// delete document with id from document path
await Collection.setPath("tests/123").delete();
// delete document with setPath and setDoc
await Collection.setPath("tests").setDoc("123").delete();

Audit trail

you can attach built in audit trail to create, update and delete operations

import {Update} from "@razaman2/collection-proxy";

// audit trail any create, update or delete operations on the proxy
Collection.setCollectionName("tests").onWrite({
   handler: (collection) => new Update(collection),
   triggers: ["create", "update", "delete"], // when you perform an operation listed in the triggers array, an associated update will be created to reflect the before and after state of the document.
});

// attach audit trail when creating a document
Collection.setCollectionName("tests").create({
   data: {id: "123", name: "john doe"},
   update: (collection) => new Update(collection),
});

// attach audit trail when updating a document
Collection.setCollectionName("tests").update({
   data: {id: "123", name: "jane doe"},
   update: (collection) => new Update(collection),
});

// attach audit trail when deleting a document
Collection.setCollectionName("tests").delete({
   data: {id: "123", name: "jane doe"},
   update: (collection) => new Update(collection),
});