🔥Write callable functions systematically like a Firelord. No more chaotic error handling, no more unsafe endpoint data type, no more messy validation. Be the Master of Fire you always wanted to be.
API interface not quite good to be honest, will redesign the API in V3
🔥 Write callable functions systematically like a Firelord. No more chaotic error handling, no more unsafe endpoint data type, no more messy validation. Be the Master of Fire you always wanted to be.
Guarantee:
Optional: For maximum benefit, please use FireCaller in front end.
support firebase-functions-test:
When coding a callable function (or any endpoint in general), we need to deal with 5 basic errors, which is basically 99% of your errors, the rest are system errors.
Error handling is chaotic, error handling is hard, error handling make you go nut.
Some developer return error as 200 response and attach his own error code and message as data, now imagine every developer return his unique format of error, this is not fun.
With FireCall, no more “you return your error, I return my error, he return his error”, everybody simply return a god damn standard HTTPS error.
FireCall standardize the way of handling these errors, there is only ONE way.
There is also one common issue where developer often calling the wrong function name which lead to CORS error, basically front end and backend are not tally with each other.
So to solve this is we prepare a schema and share it to both front end and back end, by doing this not only we make sure that the function name is correct, but also we make sure that the data type is correct.
It is very similar to how Graphql schema sharing works, but way much simpler and we all know how convoluted Graphql is.
Long thing short, FireCall make sure that there is only one way to do stuff and providing you absolute type safe at both compile and run time with single source of truth(zod).
npm i firecall zod firebase-functions regenerator-runtime
and of course you need typescript
.
Add this to your very first line of code
import 'regenerator-runtime/runtime'
You only need to add this line once
First, you need to create schema with zod
, you can share this file to front end and use FireCaller with it.
FireCall can works without FireCaller on front end but it is recommended to use FireCaller with it or else there is no point sharing schema to front end.
import { z } from 'zod'
export const updateUserSchema = {
//request data schema
req: z.object({
name: z.string(),
age: z.number(),
address: z.string(),
}),
// response data schema
res: z.undefined(),
// function name
name: 'updateUser',
}
export const getUserSchema = {
//request data schema
req: z.string(), // userId
// response data schema
res: z.object({
name: z.string(),
age: z.number(),
}),
name: 'getUser',
}
req
: request data schema
res
: response data schema
name
: onCall function name
import { updateUserSchema, getUserSchema } from './someFiles'
import { onCall } from 'firecall'
// use any variable name you want
const updateUser = onCall(
updateUserSchema,
{ route: 'private' }, // 'private' for protected route, user must sign in first, else automatically throw unauthenticated error (with customize-able message)
// handler
async (data, context) => {
const { name, age, address } = data // request data is what you define in schema.req
const {
auth: { uid }, // if route is protected, auth object is not undefined
} = context
try {
await updateWithSomeDatabase({ uid, name, age, address })
return { code: 'ok', data: undefined } // response data is what you define in schema.res
} catch (err) {
// this is the error we catch, however if we did not catch the error and in case of error in runtime, FireCall will automatically throw unknown error for us
// if we handle the error and return it, like this piece of code, FireCall will infer the type for us in <Error Logging> (check this section for details)
return {
code: 'unknown',
message: 'update user failed',
err, // this is the details of the error, could be object, could be string, could be anything, it is up to us
}
}
}
)
const getUser = onCall(
getUserSchema,
{ route: 'public' }, // 'public' for unprotected route
// handler
async data => {
const uid = data // request data is what you define in schema.req
try {
const { name, age, secret } = await getUserFromDatabase({
uid,
})
return { code: 'ok', data: { name, age } } // response data is what you define in schema.res
} catch (err) {
// this is the error we catch, however if we did not catch the error and in case of error in runtime, FireCall will automatically throw unknown error for us
// if we handle the error and return it, like this piece of code, FireCall will infer the type for us in <Error Logging> (check this section for details)
return {
code: 'unknown',
message: 'get user failed',
err, // this is the details of the error, could be object, could be string, could be anything, it is up to us
}
}
}
)
If the response is ok, handler must return object with code
and data
property, where
code
: ok
data
: value that has same type as type you define in schema.res
if the response is not ok
, handler must return object with code
and message
properties, and an optional err
property, where
code
: Firebase Functions Error Code except ‘ok’
message
: string
err
?: user defined error, put anything you want here, normally the error object or just skip it
This is helper function to export functions. Since function name is now an object property, we need a runtime check(deploy phase runtime) to make sure each function name is unique and throw error if duplicate found.
import { updateUser, getUser } from './someOtherFile'
import { exp } from 'firecall'
exp({ updateUser, getUser }).forEach(func => {
const { name, onCall } = func
exports[name] = onCall
})
If everything in someOtherFile
is FireCall function, you can write something like this
import * as allFunc from './someOtherFile'
import { exp } from 'firecall'
exp(allFunc).forEach(func => {
const { name, onCall } = func
exports[name] = onCall
})
You can use FireCall with firebase-functions-test:
ok test example:
const wrapped = test.wrap(
onCall(
schema,
{
route: 'private',
},
async () => {
return { code: 'ok', data: 'okie' }
}
).onCall
)
await expect(wrapped('someData', { auth: { uid: '123' } })).resolves.toEqual(
'okie'
)
error test examples:
const wrapped = test.wrap(
onCall(
schema,
{
route: 'private',
},
() => {
return { code: 'cancelled', message: 'cancelled' }
}
).onCall
)
await expect(wrapped('someData', { auth: { uid: '123' } })).rejects.toEqual(
new functions.https.HttpsError('cancelled', 'cancelled')
)
const wrapped = test.wrap(
onCall(
schema,
{
route: 'private',
},
async () => {
return { code: 'ok', data: 'okRes' }
}
).onCall
)
await expect(wrapped('someData')).rejects.toEqual(
new functions.https.HttpsError('unauthenticated', 'Please Login First')
)
You can use const assertion if the handler is returning response from another callback, example from the transaction.
import { onCall } from 'firecall'
export const someFun = onCall(someSchema, { route: 'private' }, async () => {
// return the transaction
return await db.runTransaction(async transaction => {
return { code: 'ok', data: null } as const // do const assertion here
})
})
If you need custom setting for you function like changing ram or region, you can pass function builder to onCall
config.
import * as functions from 'firebase-functions'
import { onCall } from 'firecall'
const someFunc = onCall(
someSchema,
{
route: 'public', // route is not optional, you can use either 'public' or 'private' value
func: functions
.runWith({
timeoutSeconds: 300,
memory: '1GB',
})
.region('europe-west1'),
},
handler
)
func
accept functions
or functions.FunctionBuilder
By default, FireCall does not log anything.
Pass a function to config.onErrorLogging if you want to log:
const someFunc = onCall(
someSchema,
{
route: 'public', // route is not optional, you can use either 'public' or 'private' value
config: {
onErrorLogging: ({ context, reqData, reqZodError, resZodError, err }) => {
// you can do something else here, eg save error to file
// example of what you can return
return undefined // no logging, default behavior
return { abc: reqData } // will log { abc: reqData }
return { logType: 'info', abc: reqData } // will log { abc: reqData }, the log type is info
},
},
},
handler
)
onErrorLogging
?: ({ reqData, context, reqZodError?, resZodError?, err? }) => (Record<string,unknown> & { logType?: 'log' | 'info' | 'warn' | 'error' }) | undefined
reqData
: the request data
context
: Firebase function context callable
reqZodError
: may exist, the error that occurs when trying to parse the request data
resZodError
: may exist, the error that occurs when trying to parse the response data
err
: may exist, it is the user defined error you return to the handler(the response). Its type is unknown until there is user defined error in the onCall callback, which mean you don’t need to type cast, FireCall will infer all the type for you.
Whatever object literal the function return and(empty object = nothing to log) get logged on the console, except the logType
props.
logType
props is an optional prop that set the type of your log, by default it is error
.
Here is how you customize error messages:
const someFunc = onCall(
someSchema,
{
route: 'public', // route is not optional, you can use either 'public' or 'private' value
config: {
changeBuiltInErrorCodeAndMessage: {
unauthenticated: {
code: 'someCode' // default: unauthenticated
message: 'someMessage' // default: Please Login First
},
unknown: {
code: 'someCode' // default: unknown
message: 'someMessage' // default: unknown
},
resZodError: {
code: 'someCode' // default: invalid-argument
message: 'someMessage' // default: invalid-argument
},
reqZodError: {
code: 'someCode' // default: internal
message: 'someMessage' // default: invalid response
}
},
},
},
handler
)
Every prop of changeBuiltInErrorCodeAndMessage
is optional.
If no values are supplied, it uses default codes and messages.
The code
value is limited to Firebase Functions Error Code except ‘ok’.