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

firebase-framework

v1.4.0

Published

A framework designed to increase productivity in serverless applications using firebase

Downloads

108

Readme

Firebase Framework

Getting Started

This documentation describes features of the firebase-framework but generally does not cover firebase options in general. For example, to find out how you can specify a specific version of nodejs see the firebase docs here: https://firebase.google.com/docs/functions/manage-functions#set_nodejs_version

To begin you will need nodejs and firebase-tools

To install firebase-tools run

npm i -G firebase-tools

Go to https://console.firebase.google.com and create a new Firebase project

mkdir firebase-project
cd firebase-project
firebase init

Select Firestore, Cloud Functions and the firebase project that you created, the rest is up to you.

You will need a serviceAccountKey.json field in the root of the project directory to run the functions locally. (This can be obtained from the Firebase console under project users and permissions.)

cd functions
npm install --save firebase-framework

Your functions/index.js file should contain:

const admin = require('firebase-admin');
const { createFunctions } = require('firebase-framework');
const serviceAccount = require('../serviceAccountKey.json');
const services = require('./services');

admin.initializeApp({
  credential: admin.credential.cert(serviceAccount),
  databaseURL: '<your firebase database url>',
});

const config = {};

module.exports = createFunctions(config, services);
Config

The config object can contain the following values:

| key | required | default | type | description | | ----------------- | -------- | -------------------------------------------------------------------------------------------------- | ---------- | --------------------------------------------------------------------------------------------------------------------------------------------------------- | | region | false | us-central1 | string | GCP region in which to deploy functions. Valid values are listed here: https://firebase.google.com/docs/functions/locations | | validatePrivilege | false | (privilege) => (req, res, next) => next() | Function | A function that accepts a privilege and returns an expressJs middleware function to validate whether the client has the correct privilege for the request | | middleware | false | [] | Function[] | An array of expressJs middleware to be added to all routes across all services | | corsEnabled | false | true | boolean | whether the expressJs cors middleware should be enabled https://www.npmjs.com/package/cors | | corsOptions | false | {origin: true,methods: "GET,PUT,POST,DELETE,OPTIONS", allowedHeaders: "token, role, content-type"} | Object | expressJs cors middleware options https://www.npmjs.com/package/cors#configuring-cors |

Creating your first service

  1. create a folder inside of functions called hello
  2. create an index.js file inside of functions/hello
  3. add the following to index.js
const schema = {
  name: {
    type: 'string',
    required: true,
  },
  age: {
    type: 'number',
    required: true,
  },
};

module.exports = {
  basePath: 'hello',
  schema,
  routes: [
    {
      path: '/',
      function: (req) => [200, { message: 'hello-world' }],
    },
  ],
};
  1. create a file inside of functions/ called services.js
  2. add the following into into services.js
const hello = require('./hello');

module.exports = [hello];

Service Configuration

  • Service configuration is exported from the index.js file in the service's directory
  • the config object's structure is as follows:

| key | required | type | description | | -------------- | --------- | ------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | basePath | true | string | defines the services base-path | | resourcePath | false | string | defines which documents in the db should be published when changed ex: 'posts/{id}' (uses the cloud functions firestore triggers syntax) | | schema | false | object | should contain a reference to the required schema file | | postSchema | false | object | optional alternative used for services that require special fields during creation | | publishChanges | false | boolean | whether the service should publish changes to it's data as messages on cloud pub sub | | withModifiers | false | boolean | declares that the schema can contain writeModifier keys that define a function that will modify values before they are processes/saved | | middleware | false | Array | ExpressJs middleware that will apply to all routes in the service | | routes | false | Array | these are the functions triggered within the service by http requests (see routes below) | | events | false | Array | pub sub events that the service will listed to (see events below) | | schedule | false | Array | cloud schedules that will trigger functions within this service (see schedule below) | | keepAlive | false | boolean | whether a scheduled function should be set up that will trigger the http function (routes) every 5 minutes to prevent cold starts* | | runtimeOptions | false | object | Firebase runtime options usually passed to runWith as documented here: https://firebase.google.com/docs/reference/functions/function_configuration.runtimeoptions| | maxAge | undefined | number | The maxAge in ms before the event listener will not attempt to process an event. Can be used to prevent infinite retries of a retry-able event |

* Though billing is required, you can expect the overall cost to be manageable, as each Cloud Scheduler job costs $0.10 (USD) per month, and there is an allowance of three free jobs per Google account (as of the time of writing). * The keepAlive feature adds a route to the service at '/heartbeat'. This will not conflict with wildcard routes in the service but would conflict with a route named the same.

Routes

| key | required | type | description | | ---------- | -------- | ----------------- | ---------------------------------------------------------------------------------------------------------- | | path | true | string | expressJs style paths that can contain parameters | | method | true | string | ['get', 'post', 'put', 'delete'] | | function | false | function | to be executed when a request reaches the defined path(optional if privilege defines functions) | | privilege | false | string | one of the privileges defined in validatePrivilege middleware | | ignoreBody | false | boolean | if set to true the body of POST / PUT requests to the route will not be checked against the service schema | | middleware | false | Array | an array of middleware that will apply to this route |

Events

| key | required | type | description | | ---------------- | -------- | -------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | topic | true | string | the pub sub topic to subscribe to | | type | false | string | the event type to listen to. This is passed to the subscriber but will not affect which message in the topic trigger the subscriber, it can function as a not about which types of events from the topic the function cares about | | function | false | function | to be executed when the described event is triggered | | ensureIdempotent | false | boolean | whether the framework should check messages against a store (requires a firestore database in the project) to ensure that messages are never processed more than once | | runtimeOptions | false | object | Firebase runtime options usually passed to runWith as documented here: https://firebase.google.com/docs/reference/functions/function_configuration.runtimeoptions |

Schedule

| key | required | type | description | | -------------- | -------- | -------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | name | true | string | The name of the schedule (it can be anything) | | time | true | string | Both Unix Crontab and AppEngine syntax are supported by Google Cloud Scheduler. | | function | false | function | to be executed when the cronjob is run | | runtimeOptions | false | object | Firebase runtime options usually passed to runWith as documented here: https://firebase.google.com/docs/reference/functions/function_configuration.runtimeoptions |

Service Schemas

  • The schema for a service's data is defined in a file named schema.js which sits in the root directory of the service (ex: functions/posts/schema.js)
  • A separate schema used when creating records can be defined in a file named postSchema.js which sits in the root directory of the service (ex: functions/posts/postSchema.js) it is used only for post requests

A valid schema consists of an array of objects containing keys defined below

| key | type | required | valid values | | ------------- | ----------------------------------------- | -------- | ------------------------------------------------------------------ | | type | string | true | 'boolean', 'string', 'object', 'number' | | default | boolean, string, object, number, function | false | any, (record: Object, req: Request) => any / Promise | | required | boolean, function | false | true, false, (value: any, record: Object, req: Request) => boolean | | enum | Array | false | any[] | | readOnly | boolean | false | true, false | | immutable | boolean | false | true, false | | nullable | boolean | false | true, false | | validator | function | false | (value: any, record: Object, req: Request) => boolean | | writeModifier | function | false | (value: any, record: Object, req: Request) => any |

Schema Validation

A function that will validate schemas is available. It can be used as follows;

const { validateSchema } = require('firebase-framework');
const services = require('./services');

services.forEach(validateSchema);

Route Handlers

HTTP functions must have the following signature:

(req, req) => [statusCode, body, headers?] | Promise<[statusCode, body, headers?]> | void

The req and res arguments are regular express.js request and response objects. the key distinction being that while you can call res.status(200).send{hello: 'world} to send a response, have found it preferable to return a tuple of [statusCode, responseBody, responseHeaders] from the route handler function.

Event Handlers

Event Handlers have the following signature:

(message, context) => Promise<void> | void

Message:

| key | description | | ------------- | ----------------------------------------------------------------------------------------------- | | data | data transmitted with the message (already parsed JSON) | | dataBefore | only on db change triggers set off by an update operation, describes the data before the change | | changeContext | context of a db change event message (db changes are proxied over Google PubSub ) | | type | type of db change the occurred, ['create', 'update', 'delete'] |

Deployment

To switch to the default project run firebase use default

  • To deploy to firebase execute the following commands:

    • firebase login
    • firebase use default
    • firebase deploy
  • To test http triggers locally run firebase serve

    • some functions require runtime env variables. For local development these can be stored in /functions/.runtimeconfig.json
    • This file is should not be committed to version control as it may contain secrets. To auto-generate it with the current project env:
      • cd into src
      • on macOS or linux run firebase functions:config:get > .runtimeconfig.json
      • on windows, in powershell run firebase functions:config:get | ac .runtimeconfig.json

Data Relationships

Being that the DB used is Firestore and it is a NoSQL (non-relational) database, we compile related data after writes as opposed to during reads. This helps improve read performance due to the ability to return a data structure as it exists at rest.

  • when a resource is updated it should emit an event on cloud pub sub describing the update that occurred, and the resource it affected.
  • services can listen to these events and update their data accordingly
  • fields updated in this way should be marked as readOnly | immutable in their schema to prevent CRUD operations on that resource from causing them to get out of sync
  • an example of this would be the a counter:
    • when an action occurred that should increment that counter the service will dispatch an event describing the update
    • the counter service can listen to that event and, within a transaction, update the counter

Event Idempotency

Google PubSub ensures at-least-once delivery. What this means for us is that while all events are ensured to be delivered, if a function times out or takes too long to come up, the event will be delivered again. The issue with this is that events that are not inherently idempotent (specifically counter operations) can cause corrupted data when run more than once.

To solve this, the ensureIdempotency value for an event should be set to true if duplicate execution of that event could be an issue. This is especially important if Retry on failure is set to true in the google cloud console.

Setting this flag to true requires that you accept a different first argument to your event handler function. instead of


function (message, context) {
}

It should be:

async function (setComplete, message, context){
  // do your thing
  await setComplete();
}

In the case of a counter where a transaction or batch is used to update a value, the idempotent success confirmation callback should be run inside of the transaction. This is done by calling setComplete within your transaction and passing it a reference to your transaction or passing it the batch reference.

async function (setComplete, message, context){
  await db.runTransaction(async t => {
    // do your thing
    await setComplete(t);
  })
}

Inter-Service Communication

Services are highly siloed from each other. They cannot call functions in each other, access object, and they should not access each others db storage.

To request data from a service the iFC inter-function communicator can be used.

It is simply a http client wrapped in a function that will automatically find and make a call to a service given standard http parameters.

Additional documentation can be found in iFC.js

Custom domains

Firebase supports custom domains for functions via firebase hosting. The official firebase docs are available here: https://firebase.google.com/docs/hosting/functions. There is a bug (maybe?) in the firebase runtime that effects functions using a custom domain. This framework uses an express router under the hood for all http functions. When a request comes in through the default firebase url to the / path of a service with a basePath of users the request path is /. This matches the configured route. When the same function is reached via a firebase hosting rule that directs <your-domain>/users to the users function the request path is /users. We cannot check the domain of the request because the request domain is the same whether the requests originates from a custom domain or the default one. In order to provide support for both use-cases we have to mount the service's express.js router to 2 paths: / and /<service-basePath>, in this case / and /users. For many projects this will present no issue but in the case of a route that looks like this /users/users the behavior may be unpredictable. Hopefully in the future this issue will be fixed in the firebase runtime by the firebase team. Until then we must live with the constraints that this provides.