Context, an instance provided by Koa, it is used across the whole lifecycle of user request. It is used within middleware, controller and logic, named ctx
for short.
// use ctx in middleware
module.exports = options => {
// ctx use be passed in as the first params
return (ctx, next) => {
...
}
}
// use ctx in controller
module.exports = class extends think.Controller {
indexAction() {
const ip = this.ctx.ip;
}
}
Framework inhert ctx and use Extend to add many useful properties and methods.
Node's request object.
Node's response object.
No Supported to use response bypass Koa, avoid to use the following properties:
res.statusCode
res.writeHead()
res.write()
res.end()
Koa‘s Request class.
Koa's Response class.
The suggest namespace for sharing data between middleware or sending message to template. We should avoid to put fields into ctx directly which may override existing properties and cause weird issues.
ctx.state.user = await User.find(id);
We can get value in controller through this.ctx.state.user
.
module.exports = class extends think.Controller {
indexAction() {
const user = this.ctx.state.user;
}
}
Application instance, the same as think.app
.
Get cookie, deprecated, use ctx.cookie(name) instead.
Set cookie, deprecated, use ctx.cookie(name, value, options) instead.
Helper method, throw error including .status
, default is 500
. This method allows Koa to response accordingly which support the following combination:
ctx.throw(403)
ctx.throw('name required', 400)
ctx.throw(400, 'name required')
ctx.throw('something exploded')
this.throw('name required', 400)
is equivalent as bellow:
let err = new Error('name required');
err.status = 400;
throw err;
Note, this is user scope error, err.expose
is marked, so these message can be used to response client request. Apparently you won't use it to expose error message if you don't want to leak error detail.
You can pass properties
object, which will be merged to error, to pass message to other middlewares with a nice defined error message.
ctx.throw(401, 'access_denied', { user: user });
ctx.throw('access_denied', { user: user });
Koa use http-errors to create error object.
Helper function to throw error When !value
equals true
, similar to .throw()
. Also similar to node's assert() method.
this.assert(this.user, 401, 'User not found. Please login!');
Koa use http-assert to assert.
If you don't want to use Koa's buildin response, just set ctx.respond = false
. And then you can use the origin res
object to response.
Note Koa doesn't suuport this, because it may break Koa's middleware and Koa itself. It is a hack way, for those want to use traditional fn(req, res)
method with middleware.
Get all header message, equivalent to ctx.request.header
.
const headers = ctx.headers;
Get all header information, equivalent to ctx.header
.
Get request type, uppercase. Like: GET
, POST
, DELETE
.
const method = ctx.method;
Set the request type (will not actually change HTTP request's type), may be useful for some middlewares, example: methodOverride()
.
ctx.method = 'COMMAND';
Get request url.
Set request url, for URL rewrite.
Get original request URL.
Get origin of URL, include protocol and host.
ctx.origin
// => http://example.com
Get full request URL, include protocol, host and url.
ctx.href
// => http://example.com/foo/bar?q=1
Get request pathname.
Set request pathname and retain query-string when present.
Get parsed query-string, returning an empty object when no query-string is present. Note that this getter does not support nested parsing.
For example "color=blue&size=small":
{
color: 'blue',
size: 'small'
}
Set query-string to the given object. Note that this setter does not support nested objects.
ctx.query = { next: '/login' }
Get raw query string void of ?.
Set raw query string.
Get raw query string with the ?.
Set raw query string.
Get host (hostname:port) when present. Supports X-Forwarded-Host when app.proxy is true, otherwise Host is used.
Get hostname when present. Supports X-Forwarded-Host when app.proxy is true, otherwise Host is used.
Get request Content-Type void of parameters such as "charset".
const ct = ctx.type
// => "image/png"
Get request charset when present, or undefined:
ctx.charset
// => "utf-8"
Check if a request cache is "fresh", aka the contents have not changed. This method is for cache negotiation between If-None-Match / ETag, and If-Modified-Since and Last-Modified. It should be referenced after setting one or more of these response headers.
// freshness check requires status 20x or 304
ctx.status = 200;
ctx.set('ETag', '123');
// cache is ok
if (ctx.fresh) {
ctx.status = 304;
return;
}
// cache is stale
// fetch new data
ctx.body = await db.find('something');
Inverse of ctx.fresh.
Return the request socket.
Get request protocal, value is https
or http
, when app.proxy
is ture then protocal value is retreive from X-Forwarded-Proto
header.
Specific details, if req.socket.encrypted
is true, then return https
, otherwise if app.proxy
is true, return X-Forwarded-Proto
header value, default value is http
.
Usually we don't want Node.js to serve client directly, but using an extra web server layout (like nginx). Web server provide HTTP(S) service, and communicate with Node.js with HTTP.
In this situation, Node.js always use https
as protocal, the actual protocal only known to web server. So we need to defined a special header for it, recommand X-Forwarded-Proto
. For safty, only if app.proxy
if true then protocol will read from this header (production.js
default value is true).
ssl on;
# SSL certificate
ssl_certificate /usr/local/nginx/ssl/domain.crt;
ssl_certificate_key /usr/local/nginx/ssl/domain.key;
location = /index.js {
proxy_http_version 1.1;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header Host $http_host;
proxy_set_header X-Forwarded-Proto "https"; # current Node.js protocol https
proxy_set_header X-NginX-Proxy true;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "upgrade";
proxy_pass http://127.0.0.1:$node_port$request_uri;
proxy_redirect off;
}
Shorthand for ctx.protocol == "https" to check if a request was issued via TLS.
Request remote address. Supports X-Forwarded-For when app.proxy is true.
When X-Forwarded-For is present and app.proxy is enabled an array of these ips is returned, ordered from upstream -> downstream. When disabled an empty array is returned.
Return subdomains as an array.
Subdomains are the dot-separated parts of the host before the main domain of the app. By default, the domain of the app is assumed to be the last two parts of the host. This can be changed by setting app.subdomainOffset.
For example, if the domain is "tobi.ferrets.example.com": If app.subdomainOffset is not set, ctx.subdomains is ["ferrets", "tobi"]. If app.subdomainOffset is 3, ctx.subdomains is ["tobi"].
Check if the incoming request contains the "Content-Type" header field, and it contains any of the give mime types. If there is no request body, null is returned. If there is no content type, or the match fails false is returned. Otherwise, it returns the matching content-type.
// With Content-Type: text/html; charset=utf-8
ctx.is('html'); // => 'html'
ctx.is('text/html'); // => 'text/html'
ctx.is('text/*', 'text/html'); // => 'text/html'
// When Content-Type is application/json
ctx.is('json', 'urlencoded'); // => 'json'
ctx.is('application/json'); // => 'application/json'
ctx.is('html', 'application/*'); // => 'application/json'
ctx.is('html'); // => false
For example if you want to ensure that only images are sent to a given route:
if (ctx.is('image/*')) {
// process
} else {
ctx.throw(415, 'images only!');
}
Check if the given type(s) is acceptable, returning the best match when true, otherwise false. The type value may be one or more mime type string such as "application/json", the extension name such as "json", or an array ["json", "html", "text/plain"].
// Accept: text/html
ctx.accepts('html');
// => "html"
// Accept: text/*, application/json
ctx.accepts('html');
// => "html"
ctx.accepts('text/html');
// => "text/html"
ctx.accepts('json', 'text');
// => "json"
ctx.accepts('application/json');
// => "application/json"
// Accept: text/*, application/json
ctx.accepts('image/png');
ctx.accepts('png');
// => false
// Accept: text/*;q=.5, application/json
ctx.accepts(['html', 'json']);
ctx.accepts('html', 'json');
// => "json"
// No Accept header
ctx.accepts('html', 'json');
// => "html"
ctx.accepts('json', 'html');
// => "json"
You may call ctx.accepts() as many times as you like, or use a switch:
switch (ctx.accepts('json', 'html', 'text')) {
case 'json': break;
case 'html': break;
case 'text': break;
default: ctx.throw(406, 'json, html, or text only');
}
Check if encodings are acceptable, returning the best match when true, otherwise false. Note that you should include identity as one of the encodings!
// Accept-Encoding: gzip
ctx.acceptsEncodings('gzip', 'deflate', 'identity');
// => "gzip"
ctx.acceptsEncodings(['gzip', 'deflate', 'identity']);
// => "gzip"
When no arguments are given all accepted encodings are returned as an array:
// Accept-Encoding: gzip, deflate
ctx.acceptsEncodings();
// => ["gzip", "deflate", "identity"]
Note that the identity encoding (which means no encoding) could be unacceptable if the client explicitly sends identity;q=0. Although this is an edge case, you should still handle the case where this method returns false.
Check if charsets are acceptable, returning the best match when true, otherwise false.
// Accept-Charset: utf-8, iso-8859-1;q=0.2, utf-7;q=0.5
ctx.acceptsCharsets('utf-8', 'utf-7');
// => "utf-8"
ctx.acceptsCharsets(['utf-7', 'utf-8']);
// => "utf-8"
When no arguments are given all accepted charsets are returned as an array:
// Accept-Charset: utf-8, iso-8859-1;q=0.2, utf-7;q=0.5
ctx.acceptsCharsets();
// => ["utf-8", "utf-7", "iso-8859-1"]
Check if langs are acceptable, returning the best match when true, otherwise false.
// Accept-Language: en;q=0.8, es, pt
ctx.acceptsLanguages('es', 'en');
// => "es"
ctx.acceptsLanguages(['en', 'es']);
// => "es"
When no arguments are given all accepted languages are returned as an array:
// Accept-Language: en;q=0.8, es, pt
ctx.acceptsLanguages();
// => ["es", "pt", "en"]
Return request header.
const host = ctx.get('host');
Get response body.
Set response body to one of the following:
string written
The Content-Type is defaulted to text/html or text/plain, both with a default charset of utf-8. The Content-Length field is also set.
Buffer written
The Content-Type is defaulted to application/octet-stream, and Content-Length is also set.
Stream piped
The Content-Type is defaulted to application/octet-stream.
Whenever a stream is set as the response body, .onerror is automatically added as a listener to the error event to catch any errors. In addition, whenever the request is closed (even prematurely), the stream is destroyed. If you do not want these two features, do not set the stream as the body directly. For example, you may not want this when setting the body as an HTTP stream in a proxy as it would destroy the underlying connection.
See: https://github.com/koajs/koa/pull/612 for more information.
Here's an example of stream error handling without automatically destroying the stream:
const PassThrough = require('stream').PassThrough;
app.use(function * (next) {
ctx.body = someHTTPStream.on('error', ctx.onerror).pipe(PassThrough());
});
Object || Array json-stringified
The Content-Type is defaulted to application/json. This includes plain objects { foo: 'bar' } and arrays ['foo', 'bar'].
null no content response
If ctx.status has not been set, Koa will automatically set the status to 200 or 204.
Get response status. By default, response.status is set to 404 unlike node's res.statusCode which defaults to 200.
Set response status via numeric code:
NOTE: don't worry too much about memorizing these strings, if you have a typo an error will be thrown, displaying this list so you can make a correction.
Get response status message. By default, response.message is associated with response.status.
Set response status message to the given value.
Set response Content-Length to the given value.
Return response Content-Length as a number when present, or deduce from ctx.body when possible, or undefined.
Get response Content-Type void of parameters such as "charset".
const ct = ctx.type;
// => "image/png"
Set response Content-Type via mime string or file extension.
ctx.type = 'text/plain; charset=utf-8';
ctx.type = 'image/png';
ctx.type = '.png';
ctx.type = 'png';
Note: when appropriate a charset is selected for you, for example response.type = 'html' will default to "utf-8", however when explicitly defined in full as response.type = 'text/html' no charset is assigned.
Check if a response header has already been sent. Useful for seeing if the client may be notified on error.
Perform a [302] redirect to url.
The string "back" is special-cased to provide Referrer support, when Referrer is not present alt or "/" is used.
ctx.redirect('back');
ctx.redirect('back', '/index.html');
ctx.redirect('/login');
ctx.redirect('http://google.com');
To alter the default status of 302, simply assign the status before or after this call. To alter the body, assign it after this call:
ctx.status = 301;
ctx.redirect('/cart');
ctx.body = 'Redirecting to shopping cart';
Set Content-Disposition to "attachment" to signal the client to prompt for download. Optionally specify the filename of the download.
Set several response header fields with an object:
ctx.set({
'Etag': '1234',
'Last-Modified': date
});
Append additional header field with value val.
ctx.append('Link', '<http://127.0.0.1/>');
Remove header field.
Set the Last-Modified header as an appropriate UTC string. You can either set it as a Date or date string.
ctx.lastModified = new Date();
Set the ETag of a response including the wrapped "s. Note that there is no corresponding response.etag getter.
ctx.etag = crypto.createHash('md5').update(ctx.body).digest('hex');
Get module name base on route parsing, this value is always empty on single module project. Default parsing logic is use think-router module.
module.exports = class extends think.Controller {
__before() {
// get module
// Variable name module is used by node, use m instead
const m = this.ctx.module;
}
}
Get controller name base on route parsing, parse by think-router.
module.exports = class extends think.Controller {
__before() {
// get controller
const controller = this.ctx.controller;
}
}
Get action name base on route parsing, parse by think-router .
module.exports = class extends think.Controller {
__before() {
// get action
const action = this.ctx.action;
}
}
Get user agent.
const userAgent = ctx.userAgent;
if(userAgent.indexOf('spider')){
...
}
Judge whether current request method is GET
.
const isGet = ctx.isGet;
if(isGet){
...
}
Judge whether current request method is POST
.
const isPost = ctx.isPost;
if(isPost){
...
}
Judge whether current request type is CLI
(call from command line).
const isCli = ctx.isCli;
if(isCli){
...
}
Return request body raw data, this prop only available when think-payload module enabled.
onlyHost
{Boolean} only return hostreturn
{String}get request referrer.
const referer1 = ctx.referer(); // http://www.thinkjs.org/doc.html
const referer2 = ctx.referer(true); // www.thinkjs.org
equal to referer
.
method
{String} typereturn
{Boolean}Judge whether current request method equals to method value.
const isPut = ctx.isMethod('PUT');
method
{String} request typereturn
{Boolean}Judge whether it is ajax request (by x-requested-with
header value equals to XMLHttpRequest
), if method is passed, this method will also do compare the request type equals to method value.
const isAjax = ctx.isAjax();
const isPostAjax = ctx.isAjax('POST');
callbackField
{String} callback field name,default value is this.config('jsonpCallbackField')
return
{Boolean}Judge whether it is jsonp request.
const isJsonp = ctx.isJson('callback');
if(isJsonp){
ctx.jsonp(data);
}
data
{Mixed} output datacallbackField
{String} callback field name,default value is this.config('jsonpCallbackField')
return
{Boolean} falseOutput jsonp format data, the return value is false. The Content-Type
returned can be specified by configuringjsonContentType
.
ctx.jsonp({name: 'test'});
//output
jsonp111({
name: 'test'
})
data
{Mixed} output datareturn
{Boolean} falseOutput json format data, the return value is false. The Content-Type
returned can be specified by configuringjsonContentType
.
ctx.json({name: 'test'});
//output
{
name: 'test'
}
data
{Mixed} output datamessage
{String} errmsg field datareturn
{Boolean} falseOutput data with errno
anderrmsg
formats. Where errno
is 0 anderrmsg
is message.
{
errno: 0,
errmsg: '',
data: ...
}
The field names errno
anderrmsg
can be modified by configuring errnoField
anderrmsgField
.
errno
{Number} error numbererrmsg
{String} error messagedata
{Mixed} extra error datareturn
{Boolean} false{
errno: 1000,
errmsg: 'no permission',
data: ''
}
The field names errno
anderrmsg
can be modified by configuring errnoField
anderrmsgField
.
time
{Number} cache time,unit is milliseconds. Support time format like 1s
or 1m
.return
{undefined}set Cache-Control
and Expires
cache header.
ctx.expires('1h'); //cache 1 hour
name
{Mixed} config namevalue
{Mixed} confg valuem
{String} module name, for multi-module projectreturn
{Mixed}Get, set configuration items, internal call think.config
method.
ctx.config('name'); //get config
ctx.config('name', value); // get config
ctx.config('name', undefined, 'admin'); //get admin module config, for multi-module project
name
{String} param namevalue
{Mixed} param valuereturn
{Mixed}Get, set the parameter value on the URL. Since the names get, query, etc. have been used by Koa, param can only be used here.
ctx.param('name'); //will undefined if 'name' is not exist
ctx.param(); // Get all the parameter values, including dynamically added parameters
ctx.param('name1,name2'); // Get the specified number of parameter values, separated by commas
ctx.param('name', value); // Reset the parameter value
ctx.param({name: 'value', name2: 'value2'}); // Reset multiple parameter values
name
{String} value
{Mixed} return
{Mixed}Get, Set post value.
ctx.post('name'); //Get the POST value, or undefined if it does not exist
ctx.post(); //Get all the POST values, including dynamically added data
ctx.post('name1,name2'); // Get the specified number of POST values, separated by commas
ctx.post('name', value); // Reset the POST value
ctx.post({name: 'value', name2: 'value2'}); //Reset multiple POST values
Sometimes submitted data is a composite data, this time to get the data format is the following format:
{ action: 'create',
'data[0][username]': '',
'data[0][nickname]': '',
'data[0][password]': ''
}
In fact, we want the data field data to be an array, which we can support using [think-qs] (https://github.com/thinkjs/think-qs) middleware.
name
{String} value
{Mixed} return
{Mixed}Get, set the file data, the file will be saved in a temporary directory, for security, the request will be deleted after the end. If you need to use the corresponding file, you can use the fs.rename
method to move to other places.
ctx.file('name'); // Get the FILE value, or undefined if it does not exist
ctx.file(); // Get all the FILE values, including dynamically added data
ctx.file('name', value); // Reset the FILE value
ctx.file({name: 'value', name2: 'value2'}); // Reset multiple FILE values
File Data Format:
{
"size": 287313, // file size
"path": "/var/folders/4j/g57qvmmd1lb_9h605w_d38_r0000gn/T/upload_fa6bf8c44179851f1cfec99544b4ef22", //temp location
"name": "An Introduction to libuv.pdf", // file name
"type": "application/pdf", // type
"mtime": "2017-07-02T07:55:23.763Z" // last modify time
}
File uploads are resolved by the [think-payload] (https://github.com/thinkjs/think-payload) module, which allows you to configure parameters such as file size restrictions.
const fs = require('fs');
const path = require('path');
const rename = think.promisify(fs.rename, fs); // The promisify method renames the method to a Promise interface
module.exports = class extends think.Controller {
async indexAction(){
const file = this.file('image');
// If you upload png format image file, move to another directory
if(file && file.type === 'image/png') {
const filepath = path.join(think.ROOT_PATH, 'runtime/upload/a.png');
think.mkdir(path.dirname(filepath));
await rename(file.path, filepath)
}
}
}
name
{String} Cookie namevalue
{mixed} Cookie valueoptions
{Object} Cookie configreturn
{Mixed}Get and set the cookie value.
ctx.cookie('name'); //get Cookie
ctx.cookie('name', value); //set Cookie
ctx.cookie(name, null); //delete Cookie
ctx.cookie(name, null, {
path: '/'
})
When setting a cookie, if the length of value is greater than 4094, a cookieLimit
event is fired, which can be captured via think.app.on ("cookieLimit")
.
When Deleting cookie, you must set parameters such as domain
and path
and other parameters and set the same time, otherwise the browser's homologous strategy will reject the delete action.
name
{String} service namem
{String} module name, only for multi-module projectreturn
{Mixed}Get service, if the class is instantiated, or directly return. Equivalent to think.service.
// get src/service/github.js module
const github = ctx.service('github');
filepath
{String} download file pathfilename
{String} download file name, if not exist will get from filepath
.Download file will set Content-Disposition
header base on content-disposition module.
const filepath = path.join(think.ROOT_PATH, 'a.txt');
ctx.download(filepath);
If the file name contains Chinese cause garbled, then you can manually specify the Content-Disposition header information, such as:
const userAgent = this.userAgent().toLowerCase();
let hfilename = '';
if (userAgent.indexOf('msie') >= 0 || userAgent.indexOf('chrome') >= 0) {
hfilename = `=${encodeURIComponent(filename)}`;
} else if(userAgent.indexOf('firefox') >= 0) {
hfilename = `*="utf8''${encodeURIComponent(filename)}"`;
} else {
hfilename = `=${new Buffer(filename).toString('binary')}`;
}
ctx.set('Content-Disposition', `attachment; filename${hfilename}`)
ctx.download(filepath)