Support for Fastify routes returning functional structures, such as fp-ts
Either, Task, TaskEither, or plain JavaScript parameterless functions.
Let's go funky, let's go functional!
First install the package:
npm i @fastify/funky
Next, set up the plugin:
const { fastifyFunky } = require('@fastify/funky')
const fastify = require('fastify');
fastify.register(fastifyFunky);
@fastify/funky
plugin is executed during preSerialization
response lifecycle phase.
While the most convenient way to use this plugin is with fp-ts
library, it is not required.
@fastify/funky
supports the following data structures:
app.get('/', (req, reply) => {
// This will result in a response 200: { id: 1}
return () => { return { id: 1} }
// Same as above
return () => { return { right:
{ id: 1 }
}}
// Same as above
return () => { return Promise.resolve({ id: 1 })}
// Same as above
return () => { return Promise.resolve({ right:
{ id: 1 }
})}
// Plugin will pass-through this value without doing anything
return (id) => { return Promise.resolve({ id })}
});
If the function returns an Either
object, it will be handled in the same way as if you returned that Either
object directly.
If the function returns a Promise, it will be resolved. If Promise resolves to an Either
object, it will be handled in the same way as if you returned that Either
object directly.
If the function directly returns anything else, or if its Promise resolves to anything else, that result is passed further along the chain as the plugin execution result.
Note that functions with parameters will be ignored by the plugin and passed-through as-is.
Right value is passed further along the chain as the plugin execution result:
app.get('/', (req, reply) => {
// This will result in a response 200: { id: 1}
return { right:
{ id: 1 }
}
});
Left value is passed to the error handler:
app.get('/', (req, reply) => {
// This will propagate to fastify error handler, which by default will result in a response 500: Internal server error
return { left: new Error('Invalid state') }
});
With the plugin registered, you can start returning entities of type Either, Task, or plain parameterless functions as router method results:
const { either, task, taskEither } = require('fp-ts')
app.get('/', (req, reply) => {
// This will result in a response 200: { id: 1}
return either.right({ id: 1})
// Same as above
return task.of(Promise.resolve({ id: 1}))
// Same as above
return taskEither.fromEither(either.right({ id: 1}))
// Same as above
return taskEither.fromTask(task.of(Promise.resolve({ id: 1})))
// Same as above
return () => { return { id: 1} }
// This will propagate to fastify error handler, which by default will result in a response 500: Internal server error
return either.left(new Error('Invalid state'))
// Same as above
return taskEither.fromEither(either.left(new Error('Invalid state')))
});