Take your HTTP server to
data:image/s3,"s3://crabby-images/89ead/89ead9dc005f50ce4bdb922f290993c90f487816" alt=""
by @matteocollina
Ludicrous speed is from Spaceballs
Node Core
'use strict'
var server = require('http').createServer(handle)
server.listen(3000)
function handle (req, res) {
res.setHeader('Content-Type', 'application/json')
res.end(JSON.stringify({ hello: 'world' }))
}
Express
'use strict'
var express = require('express')
var app = express()
app.disable('etag').disable('x-powered-by')
app.get('/', function (req, res) {
res.json({ hello: 'world' })
})
app.listen(3000)
data:image/s3,"s3://crabby-images/dc00d/dc00d45d640e6e741933b6cb9d2b35df0478e4c5" alt=""
data:image/s3,"s3://crabby-images/4f8c0/4f8c0bf9ce99e1b2d5713a128c39992f23515c2d" alt=""
data:image/s3,"s3://crabby-images/6e2de/6e2dee301e2628e5bff2556a44e69ae55bdaf199" alt=""
data:image/s3,"s3://crabby-images/6c921/6c9217c932cd1ed035135479d742d4bfa4e30e9f" alt=""
data:image/s3,"s3://crabby-images/755ca/755cad7a888c7dbb94874e94838b04cf38f46b3e" alt=""
data:image/s3,"s3://crabby-images/b5cc2/b5cc295708a762ca1019fe29c5aaccee4fa89ab3" alt=""
The path to fast
-
Start from scratch
-
Add a feature
-
Measure
-
Optimize until it has no overhead
-
GOTO 2
Can we speed up Core?
The problem with JSON.stringify
- recursive function, hard to optimize
- generic code, cannot be type specific
Serialization
data:image/s3,"s3://crabby-images/227a9/227a95860d50390b5c2c2bece006b90ac82441fe" alt=""
fast-json-stringify
- schema-based JSON rendering
- generates code based on the schema
- new Function(), but it is safe
require('http').createServer(handle).listen(3000)
var flatstr = require('flatstr')
var stringify = require('fast-json-stringify')({
type: 'object',
properties: { hello: { type: 'string' } }
})
function handle (req, res) {
res.setHeader('Content-Type', 'application/json')
res.end(flatstr(stringify({ hello: 'world' })))
}
data:image/s3,"s3://crabby-images/d154d/d154d74bb4aa078dc07deba45a0c9fef404750b7" alt=""
Adding an HTTP router
- https://github.com/delvedor/find-my-way
- avoid closure allocation
- built on a radix-tree
- safe
routing
data:image/s3,"s3://crabby-images/f1d71/f1d7125b74ed6fc9bf8b9e8a4417e23301c78326" alt=""
const router = require('find-my-way')()
const flatstr = require('flatstr')
require('http').createServer(router.lookup.bind(router)).listen(3000)
var stringify = require('fast-json-stringify')({
type: 'object',
properties: { hello: { type: 'string' } }
})
router.on('GET', '/', function root (req, res, params) {
res.setHeader('Content-Type', 'application/json')
res.end(flatstr(stringify({ hello: 'world' })))
}
data:image/s3,"s3://crabby-images/8615d/8615de6d6132d4be5c8f6ec34124e83cb722a717" alt=""
data:image/s3,"s3://crabby-images/1f88b/1f88b2e2298025a8916bdb3cd7510341715df339" alt=""
https://github.com/fastify/fastify
'use strict'
const fastify = require('fastify')()
fastify.get('/', async function (req, reply) {
return { hello: 'world' }
})
fastify.listen(3000)
fastify.get('/', {
schema: {
response: {
200: {
type: 'object',
properties: { hello: { type: 'string' } }
}
}
}
}, (req, reply) => {
reply.send({ hello: 'world' })
})
data:image/s3,"s3://crabby-images/8e2b0/8e2b0b0db37db2876da84e6a9c98f4bfd6e6ea77" alt=""
data:image/s3,"s3://crabby-images/7bc03/7bc03e8131f1e6b5d8a5012d97348897c1b36376" alt=""
data:image/s3,"s3://crabby-images/755ca/755cad7a888c7dbb94874e94838b04cf38f46b3e" alt=""
Features Comparison
Feature | Express | Hapi | Fastify |
---|---|---|---|
router | ✔ | ✔ | ✔ |
middleware | ✔ | ❌ | ✔ |
plugins | ❌ | ✔ | ✔ |
validation | ❌ | ✔ | ✔ |
hooks | ❌ | ✔ | ✔ |
decorators | ❌ | ✔ | ✔ |
logging | ❌ | ✔ | ✔ |
async/await | ❌ | ✔ | ✔ |
req/sec | 18k | 20k | 36k |
data:image/s3,"s3://crabby-images/5cdb4/5cdb4392b8e3f85b06760bcb6d5b7464087babeb" alt=""
data:image/s3,"s3://crabby-images/f3f27/f3f271c8cdc7deaddea1785a2b2744987b04885a" alt=""
The problem with closures
function process (bigdata, cb) {
remoteCall(bigdata, function (err, something) {
storeSomething(something, function (err, res) {
// this function is short-lived and hard to optimize
// bigdata is still in scope!
cb(null, res * 2)
})
})
}
Avoid nested closures 1/2
function process (bigdata, cb) {
remoteCall(bigdata, function (err, something) {
// bigdata exits scope here
callStoreSomething(something, cb)
})
}
Avoid nested closures 2/2
function callStoreSomething(something, cb) {
/* this function can be optimized! */
storeSomething(something, function (err, res) {
cb(null, res * 2)
})
}
Fastify is extremely careful in maintaining the least amount of state in the most efficient way.
middleware
fastify consumes (req, res, next) middlewares!
helmet comparison
data:image/s3,"s3://crabby-images/7a25b/7a25b0bc38d0fa2fa8ce56928a7e8c2c686271fc" alt=""
data:image/s3,"s3://crabby-images/a30a1/a30a1e05a650c5702eb00d9d66d5ff7153b15abd" alt=""
data:image/s3,"s3://crabby-images/ec13f/ec13fdb69d9680b91b17337fe91ce5b12edaee12" alt=""
validation speed
data:image/s3,"s3://crabby-images/ef207/ef207da87fd068d242db8bb35b3d2e154ca9071f" alt=""
https://github.com/ebdrup/json-schema-benchmark
validation correctness
data:image/s3,"s3://crabby-images/b7d12/b7d12387969bbad0e347e59184922cb7056bb557" alt=""
https://github.com/ebdrup/json-schema-benchmark
Fast is not everything
Fastify plugin system
fastify.register(
require('my-plugin'),
{ options }
)
Everything that could be a plugin, should be a plugin
module.exports = function myPlugin (fastify, opts, next) {
// register other plugins
fastify.register(...)
// add hooks
fastify.addHook(...)
// add decorator
fastify.decorate(...)
// add routes
fastify.route(...)
next()
}
data:image/s3,"s3://crabby-images/b49c3/b49c3d61f68800b53488f77a76dfea2ed9f4b41e" alt=""
data:image/s3,"s3://crabby-images/08983/08983709e61e83c96e9b670c06b4a9b0c55c70fd" alt=""
const fp = require('fastify-plugin')
async function myPlugin (fastify, options) {
fastify.decorate('util', yourAwesomeUtility)
// now you can use it with `fastify.util`
}
module.exports = fp(myPlugin)
data:image/s3,"s3://crabby-images/3fbb9/3fbb90b1b616f8239589ee83aed6e2328016d5c6" alt=""
data:image/s3,"s3://crabby-images/bfff9/bfff9644b3166b275a78f03f59257671e99b225e" alt=""
const fastify = require('fastify')()
fastify.register(require('./api/v1'), {
prefix: '/v1',
logLevel: 'error'
})
fastify.register(require('./api/v2'), {
prefix: '/v2',
logLevel: 'debug'
})
data:image/s3,"s3://crabby-images/1f88b/1f88b2e2298025a8916bdb3cd7510341715df339" alt=""
https://www.fastify.io
Demo
Fastify is not built by one developer
Open Open Source
Individuals making significant and valuable *contributions* are given commit-access to the project to contribute as they see fit.
openopensource.org10 Collaborators
110 Contributors
>= 1800 Commits
>= 98 versions
87 plugins
Most code does not need to go
data:image/s3,"s3://crabby-images/89ead/89ead9dc005f50ce4bdb922f290993c90f487816" alt=""