UNPKG

fastify

Version:

Fast and low overhead web framework, for Node.js

278 lines (228 loc) 9.8 kB
<h1 align="center">Fastify</h1> ## Reply The second parameter of the handler function is `Reply`. Reply is a core Fastify object that exposes the following functions and properties: - `.code(statusCode)` - Sets the status code. - `.status(statusCode)` - An alias for `.code(statusCode)`. - `.header(name, value)` - Sets a response header. - `.getHeader(name)` - Retrieve value of already set header. - `.hasHeader(name)` - Determine if a header has been set. - `.type(value)` - Sets the header `Content-Type`. - `.redirect([code,] url)` - Redirect to the specified url, the status code is optional (default to `302`). - `.callNotFound()` - Invokes the custom not found handler. - `.serialize(payload)` - Serializes the specified payload using the default json serializer or using the custom serializer (if one is set) and returns the serialized payload. - `.serializer(function)` - Sets a custom serializer for the payload. - `.send(payload)` - Sends the payload to the user, could be a plain text, a buffer, JSON, stream, or an Error object. - `.sent` - A boolean value that you can use if you need to know if `send` has already been called. - `.res` - The [`http.ServerResponse`](https://nodejs.org/dist/latest/docs/api/http.html#http_class_http_serverresponse) from Node core. ```js fastify.get('/', options, function (request, reply) { // Your code reply .code(200) .header('Content-Type', 'application/json; charset=utf-8') .send({ hello: 'world' }) }) ``` Additionally, `Reply` provides access to the context of the request: ```js fastify.get('/', {config: {foo: 'bar'}}, function (request, reply) { reply.send('handler config.foo = ' + reply.context.config.foo) }) ``` <a name="code"></a> ### .code(statusCode) If not set via `reply.code`, the resulting `statusCode` will be `200`. <a name="header"></a> ### .header(key, value) Sets a response header. If the value is omitted or undefined it is coerced to `''`. For more information, see [`http.ServerResponse#setHeader`](https://nodejs.org/dist/latest/docs/api/http.html#http_response_setheader_name_value). <a name="getHeader"></a> ### .getHeader(key) Retrieves the value of a previously set header. ```js reply.header('x-foo', 'foo') reply.getHeader('x-foo') // 'foo' ``` <a name="getHeader"></a> ### .removeHeader(key) Removed the value of a previously set header. ```js reply.header('x-foo', 'foo') reply.removeHeader('x-foo') reply.getHeader('x-foo') // undefined ``` <a name="hasHeader"></a> ### .hasHeader(key) Returns a boolean indicating if the specified header has been set. <a name="redirect"></a> ### .redirect(dest) Redirects a request to the specified url, the status code is optional, default to `302` (if status code is not already set by calling `code`). ```js reply.redirect('/home') ``` <a name="call-not-found"></a> ### .callNotFound() Invokes the custom not found handler. ```js reply.callNotFound() ``` <a name="type"></a> ### .type(contentType) Sets the content type for the response. This is a shortcut for `reply.header('Content-Type', 'the/type')`. ```js reply.type('text/html') ``` <a name="serializer"></a> ### .serializer(func) `.send()` will by default JSON-serialize any value that is not one of: `Buffer`, `stream`, `string`, `undefined`, `Error`. If you need to replace the default serializer with a custom serializer for a particular request, you can do so with the `.serializer()` utility. Be aware that if you are using a custom serializer, you must set a custom `'Content-Type'` header. ```js reply .header('Content-Type', 'application/x-protobuf') .serializer(protoBuf.serialize) ``` Note that you don't need to use this utility inside a `handler` because Buffers, streams, and strings (unless a serializer is set) are considered to already be serialized. ```js reply .header('Content-Type', 'application/x-protobuf') .send(protoBuf.serialize(data)) ``` See [`.send()`](#send) for more information on sending different types of values. <a name="sent"></a> ### .sent As the name suggests, `.sent` is a property to indicate if a response has been sent via `reply.send()`. In case a route handler is defined as an async function or it returns a promise, it is possible to set `reply.sent = true` to indicate that the automatic invocation of `reply.send()` once the handler promise resolve should be skipped. By setting `reply.sent = true`, an application claims full responsibility of the low-level request and response. Moreover, hooks will not be invoked. As an example: ```js app.get('/', (req, reply) => { reply.sent = true reply.res.end('hello world') return Promise.resolve('this will be skipped') }) ``` If the handler rejects, the error will be logged. <a name="send"></a> ### .send(data) As the name suggests, `.send()` is the function that sends the payload to the end user. <a name="send-object"></a> #### Objects As noted above, if you are sending JSON objects, `send` will serialize the object with [fast-json-stringify](https://www.npmjs.com/package/fast-json-stringify) if you set an output schema, otherwise `JSON.stringify()` will be used. ```js fastify.get('/json', options, function (request, reply) { reply.send({ hello: 'world' }) }) ``` <a name="send-string"></a> #### Strings If you pass a string to `send` without a `Content-Type`, it will be sent as `text/plain; charset=utf-8`. If you set the `Content-Type` header and pass a string to `send`, it will be serialized with the custom serializer if one is set, otherwise it will be sent unmodified (unless the `Content-Type` header is set to `application/json; charset=utf-8`, in which case it will be JSON-serialized like an object — see the section above). ```js fastify.get('/json', options, function (request, reply) { reply.send('plain string') }) ``` <a name="send-streams"></a> #### Streams *send* can also handle streams out of the box, internally uses [pump](https://www.npmjs.com/package/pump) to avoid leaks of file descriptors. If you are sending a stream and you have not set a `'Content-Type'` header, *send* will set it at `'application/octet-stream'`. ```js fastify.get('/streams', function (request, reply) { const fs = require('fs') const stream = fs.createReadStream('some-file', 'utf8') reply.send(stream) }) ``` <a name="send-buffers"></a> #### Buffers If you are sending a buffer and you have not set a `'Content-Type'` header, *send* will set it to `'application/octet-stream'`. ```js const fs = require('fs') fastify.get('/streams', function (request, reply) { fs.readFile('some-file', (err, fileBuffer) => { reply.send(err || fileBuffer) }) }) ``` <a name="errors"></a> #### Errors If you pass to *send* an object that is an instance of *Error*, Fastify will automatically create an error structured as the following: ```js { error: String // the http error message message: String // the user error message statusCode: Number // the http status code } ``` You can add some custom property to the Error object, such as `statusCode` and `headers`, that will be used to enhance the http response.<br> *Note: If you are passing an error to `send` and the statusCode is less than 400, Fastify will automatically set it at 500.* Tip: you can simplify errors by using the [`http-errors`](https://npm.im/http-errors) module or [`fastify-sensible`](https://github.com/fastify/fastify-sensible) plugin to generate errors: ```js fastify.get('/', function (request, reply) { reply.send(httpErrors.Gone()) }) ``` If you want to completely customize the error handling, checkout [`setErrorHandler`](https://github.com/fastify/fastify/blob/master/docs/Server.md#seterrorhandler) API.<br> *Note: you are responsibile for logging when customizing the error handler* API: ```js fastify.setErrorHandler(function (error, request, reply) { request.log.warn(error) var statusCode = error.statusCode >= 400 ? error.statusCode : 500 reply .code(statusCode) .type('text/plain') .send(statusCode >= 500 ? 'Internal server error' : error.message) }) ``` The not found errors generated by the router will use the [`setNotFoundHandler`](https://github.com/fastify/fastify/blob/master/docs/Server.md#setnotfoundhandler) API: ```js fastify.setNotFoundHandler(function (request, reply) { reply .code(404) .type('text/plain') .send('a custom not found') }) ``` <a name="payload-type"></a> #### Type of the final payload The type of the sent payload (after serialization and going through any [`onSend` hooks](https://github.com/fastify/fastify/blob/master/docs/Hooks.md#the-onsend-hook)) must be one of the following types, otherwise an error will be thrown: - `string` - `Buffer` - `stream` - `undefined` - `null` <a name="async-await-promise"></a> #### Async-Await and Promises Fastify natively handles promises and supports async-await.<br> *Note that in the following examples we are not using reply.send.* ```js fastify.get('/promises', options, function (request, reply) { return new Promise(function (resolve) { setTimeout(resolve, 200, { hello: 'world' }) }) }) fastify.get('/async-await', options, async function (request, reply) { var res = await new Promise(function (resolve) { setTimeout(resolve, 200, { hello: 'world' }) }) return res }) ``` Rejected promises default to a `500` HTTP status code. Reject the promise, or `throw` in an `async function`, with an object that has `statusCode` (or `status`) and `message` properties to modify the reply. ```js fastify.get('/teapot', async function (request, reply) => { const err = new Error() err.statusCode = 418 err.message = 'short and stout' throw err }) ``` If you want to know more please review [Routes#async-await](https://github.com/fastify/fastify/blob/master/docs/Routes.md#async-await).