koa-lite/docs/api/index.md

232 lines
7.4 KiB
Markdown
Raw Permalink Normal View History

2014-05-01 23:36:16 +00:00
# Installation
2017-02-25 06:30:11 +00:00
Koa requires __node v7.6.0__ or higher for ES2015 and async function support.
2014-05-01 23:36:16 +00:00
2017-02-26 10:54:54 +00:00
You can quickly install a supported version of node with your favorite version manager:
2015-02-02 00:35:17 +00:00
```bash
2017-02-25 06:30:11 +00:00
$ nvm install 7
2015-02-02 00:35:17 +00:00
$ npm i koa
$ node my-koa-app.js
2014-05-01 23:36:16 +00:00
```
## Async Functions with Babel
2017-10-18 08:22:26 +00:00
To use `async` functions in Koa in versions of node < 7.6, we recommend using [babel's require hook](http://babeljs.io/docs/usage/babel-register/).
```js
2017-10-18 08:22:26 +00:00
require('babel-register');
// require the rest of the app that needs to be transpiled after the hook
const app = require('./app');
```
To parse and transpile async functions,
you should at a minimum have the [transform-async-to-generator](http://babeljs.io/docs/plugins/transform-async-to-generator/)
or [transform-async-to-module-method](http://babeljs.io/docs/plugins/transform-async-to-module-method/) plugins.
For example, in your `.babelrc` file, you should have:
```json
{
"plugins": ["transform-async-to-generator"]
}
```
2017-02-26 11:21:17 +00:00
You can also use the [env preset](http://babeljs.io/docs/plugins/preset-env/) with a target option `"node": "current"` instead.
2013-12-18 07:47:14 +00:00
# Application
A Koa application is an object containing an array of middleware functions
which are composed and executed in a stack-like manner upon request. Koa is similar to many
other middleware systems that you may have encountered such as Ruby's Rack, Connect, and so on -
however a key design decision was made to provide high level "sugar" at the otherwise low-level
middleware layer. This improves interoperability, robustness, and makes writing middleware much
more enjoyable.
This includes methods for common tasks like content-negotiation, cache freshness, proxy support, and redirection
among others. Despite supplying a reasonably large number of helpful methods Koa maintains a small footprint, as
no middleware are bundled.
The obligatory hello world application:
<!-- runkit:endpoint -->
```js
const Koa = require('koa');
2015-10-24 09:21:47 +00:00
const app = new Koa();
app.use(async ctx => {
ctx.body = 'Hello World';
});
app.listen(3000);
```
## Cascading
2013-12-22 15:33:37 +00:00
Koa middleware cascade in a more traditional way as you may be used to with similar tools -
2013-12-19 21:38:51 +00:00
this was previously difficult to make user friendly with node's use of callbacks.
However with async functions we can achieve "true" middleware. Contrasting Connect's implementation which
simply passes control through series of functions until one returns, Koa invoke "downstream", then
control flows back "upstream".
The following example responds with "Hello World", however first the request flows through
the `x-response-time` and `logging` middleware to mark when the request started, then continue
to yield control through the response middleware. When a middleware invokes `next()`
the function suspends and passes control to the next middleware defined. After there are no more
middleware to execute downstream, the stack will unwind and each middleware is resumed to perform
its upstream behaviour.
<!-- runkit:endpoint -->
```js
const Koa = require('koa');
2015-10-24 09:21:47 +00:00
const app = new Koa();
// logger
app.use(async (ctx, next) => {
await next();
const rt = ctx.response.get('X-Response-Time');
console.log(`${ctx.method} ${ctx.url} - ${rt}`);
});
// x-response-time
app.use(async (ctx, next) => {
const start = Date.now();
await next();
const ms = Date.now() - start;
ctx.set('X-Response-Time', `${ms}ms`);
});
// response
app.use(async ctx => {
ctx.body = 'Hello World';
});
app.listen(3000);
```
2013-12-19 04:12:12 +00:00
## Settings
Application settings are properties on the `app` instance, currently
the following are supported:
- `app.env` defaulting to the __NODE_ENV__ or "development"
- `app.proxy` when true proxy header fields will be trusted
- `app.subdomainOffset` offset of `.subdomains` to ignore [2]
## app.listen(...)
2017-09-13 17:58:43 +00:00
A Koa application is not a 1-to-1 representation of an HTTP server.
One or more Koa applications may be mounted together to form larger
applications with a single HTTP server.
2013-12-19 04:12:12 +00:00
Create and return an HTTP server, passing the given arguments to
`Server#listen()`. These arguments are documented on [nodejs.org](http://nodejs.org/api/http.html#http_server_listen_port_hostname_backlog_callback). The following is a useless Koa application bound to port `3000`:
```js
const Koa = require('koa');
2015-10-24 09:21:47 +00:00
const app = new Koa();
app.listen(3000);
```
The `app.listen(...)` method is simply sugar for the following:
```js
2015-10-24 09:21:47 +00:00
const http = require('http');
const Koa = require('koa');
2015-10-24 09:21:47 +00:00
const app = new Koa();
http.createServer(app.callback()).listen(3000);
```
This means you can spin up the same application as both HTTP and HTTPS
or on multiple addresses:
```js
2015-10-24 09:21:47 +00:00
const http = require('http');
const https = require('https');
const Koa = require('koa');
2015-10-24 09:21:47 +00:00
const app = new Koa();
http.createServer(app.callback()).listen(3000);
https.createServer(app.callback()).listen(3001);
```
2013-12-18 07:47:14 +00:00
## app.callback()
Return a callback function suitable for the `http.createServer()`
method to handle a request.
2018-01-31 05:05:00 +00:00
You may also use this callback function to mount your Koa app in a
Connect/Express app.
2013-12-18 07:47:14 +00:00
## app.use(function)
Add the given middleware function to this application. See [Middleware](https://github.com/koajs/koa/wiki#middleware) for
more information.
2013-12-18 07:47:14 +00:00
## app.keys=
2013-11-15 18:03:40 +00:00
Set signed cookie keys.
2013-11-15 18:03:40 +00:00
These are passed to [KeyGrip](https://github.com/jed/keygrip),
however you may also pass your own `KeyGrip` instance. For
example the following are acceptable:
```js
2013-11-15 18:03:40 +00:00
app.keys = ['im a newer secret', 'i like turtle'];
app.keys = new KeyGrip(['im a newer secret', 'i like turtle'], 'sha256');
```
These keys may be rotated and are used when signing cookies
with the `{ signed: true }` option:
```js
2016-03-22 18:03:10 +00:00
ctx.cookies.set('name', 'tobi', { signed: true });
2013-11-15 18:03:40 +00:00
```
2015-08-31 03:20:22 +00:00
## app.context
`app.context` is the prototype from which `ctx` is created.
You may add additional properties to `ctx` by editing `app.context`.
This is useful for adding properties or methods to `ctx` to be used across your entire app,
which may be more performant (no middleware) and/or easier (fewer `require()`s)
at the expense of relying more on `ctx`, which could be considered an anti-pattern.
For example, to add a reference to your database from `ctx`:
2015-08-31 03:20:22 +00:00
```js
app.context.db = db();
app.use(async ctx => {
console.log(ctx.db);
});
2015-08-31 03:20:22 +00:00
```
Note:
- Many properties on `ctx` are defined using getters, setters, and `Object.defineProperty()`. You can only edit these properties (not recommended) by using `Object.defineProperty()` on `app.context`. See https://github.com/koajs/koa/issues/652.
2018-06-30 17:30:22 +00:00
- Mounted apps currently use their parent's `ctx` and settings. Thus, mounted apps are really just groups of middleware.
## Error Handling
By default outputs all errors to stderr unless `app.silent` is `true`.
2018-01-31 05:05:19 +00:00
The default error handler also won't output errors when `err.status` is `404` or `err.expose` is `true`.
To perform custom error-handling logic such as centralized logging you can add an "error" event listener:
```js
app.on('error', err => {
2016-03-15 15:35:52 +00:00
log.error('server error', err)
});
```
2015-10-13 20:09:01 +00:00
If an error is in the req/res cycle and it is _not_ possible to respond to the client, the `Context` instance is also passed:
```js
app.on('error', (err, ctx) => {
2016-03-15 15:35:52 +00:00
log.error('server error', err, ctx)
});
```
When an error occurs _and_ it is still possible to respond to the client, aka no data has been written to the socket, Koa will respond
appropriately with a 500 "Internal Server Error". In either case
an app-level "error" is emitted for logging purposes.