Rinser/node_modules/express/lib/response.js

847 lines
20 KiB
JavaScript
Raw Normal View History

2015-07-20 13:42:07 +00:00
/*!
* express
* Copyright(c) 2009-2013 TJ Holowaychuk
* Copyright(c) 2014-2015 Douglas Christopher Wilson
* MIT Licensed
*/
/**
* Module dependencies.
2015-08-14 19:58:05 +00:00
* @api private
2015-07-20 13:42:07 +00:00
*/
var contentDisposition = require('content-disposition');
var deprecate = require('depd')('express');
var escapeHtml = require('escape-html');
var merge = require('utils-merge');
2015-08-14 19:58:05 +00:00
var parseUrl = require('parseurl');
2015-07-20 13:42:07 +00:00
var vary = require('vary');
2015-08-14 19:58:05 +00:00
var http = require('http')
, path = require('path')
, connect = require('connect')
, sign = require('cookie-signature').sign
, normalizeType = require('./utils').normalizeType
, normalizeTypes = require('./utils').normalizeTypes
, setCharset = require('./utils').setCharset
, statusCodes = http.STATUS_CODES
, cookie = require('cookie')
, send = require('send')
, mime = connect.mime
, resolve = require('url').resolve
, basename = path.basename
, extname = path.extname;
2015-07-20 13:42:07 +00:00
/**
* Response prototype.
*/
var res = module.exports = {
__proto__: http.ServerResponse.prototype
};
/**
* Set status `code`.
*
* @param {Number} code
* @return {ServerResponse}
2015-08-14 19:58:05 +00:00
* @api public
2015-07-20 13:42:07 +00:00
*/
2015-08-14 19:58:05 +00:00
res.status = function(code){
2015-07-20 13:42:07 +00:00
this.statusCode = code;
return this;
};
/**
* Set Link header field with the given `links`.
*
* Examples:
*
* res.links({
* next: 'http://api.example.com/users?page=2',
* last: 'http://api.example.com/users?page=5'
* });
*
* @param {Object} links
* @return {ServerResponse}
2015-08-14 19:58:05 +00:00
* @api public
2015-07-20 13:42:07 +00:00
*/
res.links = function(links){
var link = this.get('Link') || '';
if (link) link += ', ';
return this.set('Link', link + Object.keys(links).map(function(rel){
return '<' + links[rel] + '>; rel="' + rel + '"';
}).join(', '));
};
/**
* Send a response.
*
* Examples:
*
* res.send(new Buffer('wahoo'));
* res.send({ some: 'json' });
* res.send('<p>some html</p>');
2015-08-14 19:58:05 +00:00
* res.send(404, 'Sorry, cant find that');
* res.send(404);
2015-07-20 13:42:07 +00:00
*
2015-08-14 19:58:05 +00:00
* @param {Mixed} body or status
* @param {Mixed} body
* @return {ServerResponse}
* @api public
2015-07-20 13:42:07 +00:00
*/
2015-08-14 19:58:05 +00:00
res.send = function(body){
2015-07-20 13:42:07 +00:00
var req = this.req;
var type;
2015-08-14 19:58:05 +00:00
var encoding;
var len;
2015-07-20 13:42:07 +00:00
// settings
var app = this.app;
// allow status / body
2015-08-14 19:58:05 +00:00
if (2 == arguments.length) {
2015-07-20 13:42:07 +00:00
// res.send(body, status) backwards compat
2015-08-14 19:58:05 +00:00
if ('number' != typeof body && 'number' == typeof arguments[1]) {
2015-07-20 13:42:07 +00:00
this.statusCode = arguments[1];
} else {
2015-08-14 19:58:05 +00:00
this.statusCode = body;
body = arguments[1];
2015-07-20 13:42:07 +00:00
}
}
// disambiguate res.send(status) and res.send(status, num)
2015-08-14 19:58:05 +00:00
if (typeof body === 'number' && arguments.length === 1) {
2015-07-20 13:42:07 +00:00
// res.send(status) will set status message as text string
2015-08-14 19:58:05 +00:00
this.get('Content-Type') || this.type('txt');
this.statusCode = body;
body = http.STATUS_CODES[body];
2015-07-20 13:42:07 +00:00
}
2015-08-14 19:58:05 +00:00
switch (typeof body) {
2015-07-20 13:42:07 +00:00
// string defaulting to html
case 'string':
if (!this.get('Content-Type')) {
2015-08-14 19:58:05 +00:00
this.charset = this.charset || 'utf-8';
2015-07-20 13:42:07 +00:00
this.type('html');
}
break;
case 'boolean':
case 'number':
case 'object':
2015-08-14 19:58:05 +00:00
if (null == body) {
body = '';
} else if (Buffer.isBuffer(body)) {
this.get('Content-Type') || this.type('bin');
2015-07-20 13:42:07 +00:00
} else {
2015-08-14 19:58:05 +00:00
return this.json(body);
2015-07-20 13:42:07 +00:00
}
break;
}
// write strings in utf-8
2015-08-14 19:58:05 +00:00
if ('string' === typeof body) {
2015-07-20 13:42:07 +00:00
encoding = 'utf8';
type = this.get('Content-Type');
// reflect this in content-type
2015-08-14 19:58:05 +00:00
if ('string' === typeof type) {
2015-07-20 13:42:07 +00:00
this.set('Content-Type', setCharset(type, 'utf-8'));
}
}
// populate Content-Length
2015-08-14 19:58:05 +00:00
if (undefined !== body && !this.get('Content-Length')) {
len = Buffer.isBuffer(body)
? body.length
: Buffer.byteLength(body, encoding);
2015-07-20 13:42:07 +00:00
this.set('Content-Length', len);
}
// populate ETag
var etag;
var generateETag = len !== undefined && app.get('etag fn');
if (typeof generateETag === 'function' && !this.get('ETag')) {
2015-08-14 19:58:05 +00:00
if ((etag = generateETag(body, encoding))) {
2015-07-20 13:42:07 +00:00
this.set('ETag', etag);
}
}
// freshness
if (req.fresh) this.statusCode = 304;
// strip irrelevant headers
if (204 == this.statusCode || 304 == this.statusCode) {
this.removeHeader('Content-Type');
this.removeHeader('Content-Length');
this.removeHeader('Transfer-Encoding');
2015-08-14 19:58:05 +00:00
body = '';
2015-07-20 13:42:07 +00:00
}
if (req.method === 'HEAD') {
// skip body for HEAD
this.end();
} else {
// respond
2015-08-14 19:58:05 +00:00
this.end(body, encoding);
2015-07-20 13:42:07 +00:00
}
return this;
};
/**
* Send JSON response.
*
* Examples:
*
* res.json(null);
* res.json({ user: 'tj' });
2015-08-14 19:58:05 +00:00
* res.json(500, 'oh noes!');
* res.json(404, 'I dont have that');
2015-07-20 13:42:07 +00:00
*
2015-08-14 19:58:05 +00:00
* @param {Mixed} obj or status
* @param {Mixed} obj
* @return {ServerResponse}
* @api public
2015-07-20 13:42:07 +00:00
*/
2015-08-14 19:58:05 +00:00
res.json = function(obj){
2015-07-20 13:42:07 +00:00
// allow status / body
2015-08-14 19:58:05 +00:00
if (2 == arguments.length) {
2015-07-20 13:42:07 +00:00
// res.json(body, status) backwards compat
2015-08-14 19:58:05 +00:00
if ('number' == typeof arguments[1]) {
2015-07-20 13:42:07 +00:00
this.statusCode = arguments[1];
2015-08-14 19:58:05 +00:00
if (typeof obj === 'number') {
deprecate('res.json(obj, status): Use res.json(status, obj) instead');
} else {
deprecate('res.json(num, status): Use res.status(status).json(num) instead');
}
2015-07-20 13:42:07 +00:00
} else {
2015-08-14 19:58:05 +00:00
this.statusCode = obj;
obj = arguments[1];
2015-07-20 13:42:07 +00:00
}
}
// settings
var app = this.app;
var replacer = app.get('json replacer');
var spaces = app.get('json spaces');
2015-08-14 19:58:05 +00:00
var body = JSON.stringify(obj, replacer, spaces);
2015-07-20 13:42:07 +00:00
// content-type
2015-08-14 19:58:05 +00:00
this.charset = this.charset || 'utf-8';
this.get('Content-Type') || this.set('Content-Type', 'application/json');
2015-07-20 13:42:07 +00:00
return this.send(body);
};
/**
* Send JSON response with JSONP callback support.
*
* Examples:
*
* res.jsonp(null);
* res.jsonp({ user: 'tj' });
2015-08-14 19:58:05 +00:00
* res.jsonp(500, 'oh noes!');
* res.jsonp(404, 'I dont have that');
2015-07-20 13:42:07 +00:00
*
2015-08-14 19:58:05 +00:00
* @param {Mixed} obj or status
* @param {Mixed} obj
* @return {ServerResponse}
* @api public
2015-07-20 13:42:07 +00:00
*/
2015-08-14 19:58:05 +00:00
res.jsonp = function(obj){
2015-07-20 13:42:07 +00:00
// allow status / body
2015-08-14 19:58:05 +00:00
if (2 == arguments.length) {
2015-07-20 13:42:07 +00:00
// res.json(body, status) backwards compat
2015-08-14 19:58:05 +00:00
if ('number' == typeof arguments[1]) {
2015-07-20 13:42:07 +00:00
this.statusCode = arguments[1];
2015-08-14 19:58:05 +00:00
if (typeof obj === 'number') {
deprecate('res.jsonp(obj, status): Use res.jsonp(status, obj) instead');
} else {
deprecate('res.jsonp(num, status): Use res.status(status).jsonp(num) instead');
}
2015-07-20 13:42:07 +00:00
} else {
2015-08-14 19:58:05 +00:00
this.statusCode = obj;
obj = arguments[1];
2015-07-20 13:42:07 +00:00
}
}
// settings
var app = this.app;
var replacer = app.get('json replacer');
var spaces = app.get('json spaces');
2015-08-14 19:58:05 +00:00
var body = JSON.stringify(obj, replacer, spaces);
2015-07-20 13:42:07 +00:00
var callback = this.req.query[app.get('jsonp callback name')];
// content-type
if (!this.get('Content-Type')) {
2015-08-14 19:58:05 +00:00
this.charset = 'utf-8';
2015-07-20 13:42:07 +00:00
this.set('X-Content-Type-Options', 'nosniff');
this.set('Content-Type', 'application/json');
}
// fixup callback
if (Array.isArray(callback)) {
callback = callback[0];
}
// jsonp
if (typeof callback === 'string' && callback.length !== 0) {
this.charset = 'utf-8';
this.set('X-Content-Type-Options', 'nosniff');
this.set('Content-Type', 'text/javascript');
// restrict callback charset
callback = callback.replace(/[^\[\]\w$.]/g, '');
// replace chars not allowed in JavaScript that are in JSON
body = body
.replace(/\u2028/g, '\\u2028')
.replace(/\u2029/g, '\\u2029');
// the /**/ is a specific security mitigation for "Rosetta Flash JSONP abuse"
// the typeof check is just to reduce client error noise
body = '/**/ typeof ' + callback + ' === \'function\' && ' + callback + '(' + body + ');';
}
return this.send(body);
};
/**
* Transfer the file at the given `path`.
*
* Automatically sets the _Content-Type_ response header field.
2015-08-14 19:58:05 +00:00
* The callback `fn(err)` is invoked when the transfer is complete
2015-07-20 13:42:07 +00:00
* or when an error occurs. Be sure to check `res.sentHeader`
* if you wish to attempt responding, as the header and some data
* may have already been transferred.
*
* Options:
*
2015-08-14 19:58:05 +00:00
* - `maxAge` defaulting to 0
2015-07-20 13:42:07 +00:00
* - `root` root directory for relative filenames
* - `dotfiles` serve dotfiles, defaulting to false; can be `"allow"` to send them
*
* Other options are passed along to `send`.
*
* Examples:
*
2015-08-14 19:58:05 +00:00
* The following example illustrates how `res.sendfile()` may
2015-07-20 13:42:07 +00:00
* be used as an alternative for the `static()` middleware for
2015-08-14 19:58:05 +00:00
* dynamic situations. The code backing `res.sendfile()` is actually
2015-07-20 13:42:07 +00:00
* the same code, so HTTP cache support etc is identical.
*
* app.get('/user/:uid/photos/:file', function(req, res){
* var uid = req.params.uid
* , file = req.params.file;
*
* req.user.mayViewFilesFrom(uid, function(yes){
* if (yes) {
2015-08-14 19:58:05 +00:00
* res.sendfile('/uploads/' + uid + '/' + file);
2015-07-20 13:42:07 +00:00
* } else {
* res.send(403, 'Sorry! you cant see that.');
* }
* });
* });
*
2015-08-14 19:58:05 +00:00
* @param {String} path
* @param {Object|Function} options or fn
* @param {Function} fn
* @api public
2015-07-20 13:42:07 +00:00
*/
2015-08-14 19:58:05 +00:00
res.sendfile = function(path, options, fn){
var self = this
, req = self.req
, next = this.req.next
, options = options || {}
, done;
2015-07-20 13:42:07 +00:00
// support function as second arg
2015-08-14 19:58:05 +00:00
if ('function' == typeof options) {
fn = options;
options = {};
2015-07-20 13:42:07 +00:00
}
2015-08-14 19:58:05 +00:00
// socket errors
req.socket.on('error', error);
2015-07-20 13:42:07 +00:00
2015-08-14 19:58:05 +00:00
// errors
function error(err) {
if (done) return;
done = true;
2015-07-20 13:42:07 +00:00
2015-08-14 19:58:05 +00:00
// clean up
cleanup();
if (!self.headersSent) self.removeHeader('Content-Disposition');
2015-07-20 13:42:07 +00:00
2015-08-14 19:58:05 +00:00
// callback available
if (fn) return fn(err);
2015-07-20 13:42:07 +00:00
2015-08-14 19:58:05 +00:00
// list in limbo if there's no callback
if (self.headersSent) return;
2015-07-20 13:42:07 +00:00
2015-08-14 19:58:05 +00:00
// delegate
next(err);
}
2015-07-20 13:42:07 +00:00
2015-08-14 19:58:05 +00:00
// streaming
function stream(stream) {
if (done) return;
cleanup();
if (fn) stream.on('end', fn);
2015-07-20 13:42:07 +00:00
}
2015-08-14 19:58:05 +00:00
// cleanup
function cleanup() {
req.socket.removeListener('error', error);
}
2015-07-20 13:42:07 +00:00
// transfer
2015-08-14 19:58:05 +00:00
var file = send(req, path, options);
file.on('error', error);
file.on('directory', next);
file.on('stream', stream);
file.pipe(this);
this.on('finish', cleanup);
2015-07-20 13:42:07 +00:00
};
/**
* Transfer the file at the given `path` as an attachment.
*
* Optionally providing an alternate attachment `filename`,
2015-08-14 19:58:05 +00:00
* and optional callback `fn(err)`. The callback is invoked
2015-07-20 13:42:07 +00:00
* when the data transfer is complete, or when an error has
* ocurred. Be sure to check `res.headersSent` if you plan to respond.
*
* This method uses `res.sendfile()`.
*
2015-08-14 19:58:05 +00:00
* @param {String} path
* @param {String|Function} filename or fn
* @param {Function} fn
* @api public
2015-07-20 13:42:07 +00:00
*/
2015-08-14 19:58:05 +00:00
res.download = function download(path, filename, fn) {
2015-07-20 13:42:07 +00:00
// support function as second arg
if (typeof filename === 'function') {
2015-08-14 19:58:05 +00:00
fn = filename;
filename = null;
2015-07-20 13:42:07 +00:00
}
2015-08-14 19:58:05 +00:00
filename = filename || path;
2015-07-20 13:42:07 +00:00
2015-08-14 19:58:05 +00:00
this.set('Content-Disposition', contentDisposition(filename));
2015-07-20 13:42:07 +00:00
2015-08-14 19:58:05 +00:00
return this.sendfile(path, fn);
2015-07-20 13:42:07 +00:00
};
/**
* Set _Content-Type_ response header with `type` through `mime.lookup()`
* when it does not contain "/", or set the Content-Type to `type` otherwise.
*
* Examples:
*
* res.type('.html');
* res.type('html');
* res.type('json');
* res.type('application/json');
* res.type('png');
*
* @param {String} type
* @return {ServerResponse} for chaining
2015-08-14 19:58:05 +00:00
* @api public
2015-07-20 13:42:07 +00:00
*/
res.contentType =
2015-08-14 19:58:05 +00:00
res.type = function(type){
return this.set('Content-Type', ~type.indexOf('/')
? type
: mime.lookup(type));
2015-07-20 13:42:07 +00:00
};
/**
* Respond to the Acceptable formats using an `obj`
* of mime-type callbacks.
*
* This method uses `req.accepted`, an array of
* acceptable types ordered by their quality values.
* When "Accept" is not present the _first_ callback
* is invoked, otherwise the first match is used. When
* no match is performed the server responds with
* 406 "Not Acceptable".
*
* Content-Type is set for you, however if you choose
* you may alter this within the callback using `res.type()`
* or `res.set('Content-Type', ...)`.
*
* res.format({
* 'text/plain': function(){
* res.send('hey');
* },
*
* 'text/html': function(){
* res.send('<p>hey</p>');
* },
*
* 'appliation/json': function(){
* res.send({ message: 'hey' });
* }
* });
*
* In addition to canonicalized MIME types you may
* also use extnames mapped to these types:
*
* res.format({
* text: function(){
* res.send('hey');
* },
*
* html: function(){
* res.send('<p>hey</p>');
* },
*
* json: function(){
* res.send({ message: 'hey' });
* }
* });
*
* By default Express passes an `Error`
* with a `.status` of 406 to `next(err)`
* if a match is not made. If you provide
* a `.default` callback it will be invoked
* instead.
*
* @param {Object} obj
* @return {ServerResponse} for chaining
2015-08-14 19:58:05 +00:00
* @api public
2015-07-20 13:42:07 +00:00
*/
res.format = function(obj){
2015-08-14 19:58:05 +00:00
var req = this.req
, next = req.next;
2015-07-20 13:42:07 +00:00
var fn = obj.default;
if (fn) delete obj.default;
var keys = Object.keys(obj);
2015-08-14 19:58:05 +00:00
var key = req.accepts(keys);
2015-07-20 13:42:07 +00:00
this.vary("Accept");
if (key) {
2015-08-14 19:58:05 +00:00
var type = normalizeType(key).value;
var charset = mime.charsets.lookup(type);
if (charset) type += '; charset=' + charset;
this.set('Content-Type', type);
2015-07-20 13:42:07 +00:00
obj[key](req, this, next);
} else if (fn) {
fn();
} else {
var err = new Error('Not Acceptable');
2015-08-14 19:58:05 +00:00
err.status = 406;
2015-07-20 13:42:07 +00:00
err.types = normalizeTypes(keys).map(function(o){ return o.value });
next(err);
}
return this;
};
/**
* Set _Content-Disposition_ header to _attachment_ with optional `filename`.
*
* @param {String} filename
* @return {ServerResponse}
2015-08-14 19:58:05 +00:00
* @api public
2015-07-20 13:42:07 +00:00
*/
res.attachment = function attachment(filename) {
if (filename) {
this.type(extname(filename));
}
this.set('Content-Disposition', contentDisposition(filename));
return this;
};
/**
* Set header `field` to `val`, or pass
* an object of header fields.
*
* Examples:
*
* res.set('Foo', ['bar', 'baz']);
* res.set('Accept', 'application/json');
* res.set({ Accept: 'text/plain', 'X-API-Key': 'tobi' });
*
* Aliased as `res.header()`.
*
2015-08-14 19:58:05 +00:00
* @param {String|Object|Array} field
* @param {String} val
2015-07-20 13:42:07 +00:00
* @return {ServerResponse} for chaining
2015-08-14 19:58:05 +00:00
* @api public
2015-07-20 13:42:07 +00:00
*/
res.set =
2015-08-14 19:58:05 +00:00
res.header = function(field, val){
if (2 == arguments.length) {
if (Array.isArray(val)) val = val.map(String);
else val = String(val);
this.setHeader(field, val);
2015-07-20 13:42:07 +00:00
} else {
for (var key in field) {
this.set(key, field[key]);
}
}
return this;
};
/**
* Get value for header `field`.
*
* @param {String} field
* @return {String}
2015-08-14 19:58:05 +00:00
* @api public
2015-07-20 13:42:07 +00:00
*/
res.get = function(field){
return this.getHeader(field);
};
/**
* Clear cookie `name`.
*
* @param {String} name
* @param {Object} options
2015-08-14 19:58:05 +00:00
* @param {ServerResponse} for chaining
* @api public
2015-07-20 13:42:07 +00:00
*/
2015-08-14 19:58:05 +00:00
res.clearCookie = function(name, options){
var opts = { expires: new Date(1), path: '/' };
return this.cookie(name, '', options
? merge(opts, options)
: opts);
2015-07-20 13:42:07 +00:00
};
/**
2015-08-14 19:58:05 +00:00
* Set cookie `name` to `val`, with the given `options`.
2015-07-20 13:42:07 +00:00
*
* Options:
*
* - `maxAge` max-age in milliseconds, converted to `expires`
* - `signed` sign the cookie
* - `path` defaults to "/"
*
* Examples:
*
* // "Remember Me" for 15 minutes
* res.cookie('rememberme', '1', { expires: new Date(Date.now() + 900000), httpOnly: true });
*
* // save as above
* res.cookie('rememberme', '1', { maxAge: 900000, httpOnly: true })
*
* @param {String} name
2015-08-14 19:58:05 +00:00
* @param {String|Object} val
2015-07-20 13:42:07 +00:00
* @param {Options} options
2015-08-14 19:58:05 +00:00
* @api public
2015-07-20 13:42:07 +00:00
*/
2015-08-14 19:58:05 +00:00
res.cookie = function(name, val, options){
options = merge({}, options);
2015-07-20 13:42:07 +00:00
var secret = this.req.secret;
2015-08-14 19:58:05 +00:00
var signed = options.signed;
if (signed && !secret) throw new Error('connect.cookieParser("secret") required for signed cookies');
if ('number' == typeof val) val = val.toString();
if ('object' == typeof val) val = 'j:' + JSON.stringify(val);
if (signed) val = 's:' + sign(val, secret);
if ('maxAge' in options) {
options.expires = new Date(Date.now() + options.maxAge);
options.maxAge /= 1000;
}
if (null == options.path) options.path = '/';
this.set('Set-Cookie', cookie.serialize(name, String(val), options));
2015-07-20 13:42:07 +00:00
return this;
};
2015-08-14 19:58:05 +00:00
2015-07-20 13:42:07 +00:00
/**
* Set the location header to `url`.
*
* The given `url` can also be "back", which redirects
* to the _Referrer_ or _Referer_ headers or "/".
*
* Examples:
*
* res.location('/foo/bar').;
* res.location('http://example.com');
2015-08-14 19:58:05 +00:00
* res.location('../login'); // /blog/post/1 -> /blog/login
*
* Mounting:
*
* When an application is mounted and `res.location()`
* is given a path that does _not_ lead with "/" it becomes
* relative to the mount-point. For example if the application
* is mounted at "/blog", the following would become "/blog/login".
*
* res.location('login');
*
* While the leading slash would result in a location of "/login":
*
* res.location('/login');
2015-07-20 13:42:07 +00:00
*
* @param {String} url
2015-08-14 19:58:05 +00:00
* @api public
2015-07-20 13:42:07 +00:00
*/
2015-08-14 19:58:05 +00:00
res.location = function(url){
var app = this.app
, req = this.req
, path;
2015-07-20 13:42:07 +00:00
// "back" is an alias for the referrer
2015-08-14 19:58:05 +00:00
if ('back' == url) url = req.get('Referrer') || '/';
// relative
if (!~url.indexOf('://') && 0 != url.indexOf('//')) {
// relative to path
if ('.' == url[0]) {
path = parseUrl.original(req).pathname;
path = path + ('/' == path[path.length - 1] ? '' : '/');
url = resolve(path, url);
// relative to mount-point
} else if ('/' != url[0]) {
path = app.path();
url = path + '/' + url;
}
2015-07-20 13:42:07 +00:00
}
2015-08-14 19:58:05 +00:00
// Respond
this.set('Location', url);
2015-07-20 13:42:07 +00:00
return this;
};
/**
* Redirect to the given `url` with optional response `status`
* defaulting to 302.
*
* The resulting `url` is determined by `res.location()`, so
* it will play nicely with mounted apps, relative paths,
* `"back"` etc.
*
* Examples:
*
* res.redirect('/foo/bar');
* res.redirect('http://example.com');
* res.redirect(301, 'http://example.com');
* res.redirect('../login'); // /blog/post/1 -> /blog/login
*
2015-08-14 19:58:05 +00:00
* @api public
2015-07-20 13:42:07 +00:00
*/
2015-08-14 19:58:05 +00:00
res.redirect = function(url){
var head = 'HEAD' == this.req.method
, status = 302
, body;
2015-07-20 13:42:07 +00:00
// allow status / url
2015-08-14 19:58:05 +00:00
if (2 == arguments.length) {
if ('number' == typeof url) {
status = url;
url = arguments[1];
2015-07-20 13:42:07 +00:00
} else {
deprecate('res.redirect(url, status): Use res.redirect(status, url) instead');
status = arguments[1];
}
}
// Set location header
2015-08-14 19:58:05 +00:00
this.location(url);
url = this.get('Location');
2015-07-20 13:42:07 +00:00
// Support text/{plain,html} by default
this.format({
text: function(){
2015-08-14 19:58:05 +00:00
body = statusCodes[status] + '. Redirecting to ' + encodeURI(url);
2015-07-20 13:42:07 +00:00
},
html: function(){
2015-08-14 19:58:05 +00:00
var u = escapeHtml(url);
2015-07-20 13:42:07 +00:00
body = '<p>' + statusCodes[status] + '. Redirecting to <a href="' + u + '">' + u + '</a></p>';
},
default: function(){
body = '';
}
});
// Respond
this.statusCode = status;
this.set('Content-Length', Buffer.byteLength(body));
2015-08-14 19:58:05 +00:00
this.end(head ? null : body);
2015-07-20 13:42:07 +00:00
};
/**
* Add `field` to Vary. If already present in the Vary set, then
* this call is simply ignored.
*
* @param {Array|String} field
2015-08-14 19:58:05 +00:00
* @param {ServerResponse} for chaining
* @api public
2015-07-20 13:42:07 +00:00
*/
res.vary = function(field){
// checks for back-compat
2015-08-14 19:58:05 +00:00
if (!field) return this;
if (Array.isArray(field) && !field.length) return this;
2015-07-20 13:42:07 +00:00
vary(this, field);
return this;
};
/**
* Render `view` with the given `options` and optional callback `fn`.
* When a callback function is given a response will _not_ be made
* automatically, otherwise a response of _200_ and _text/html_ is given.
*
* Options:
*
* - `cache` boolean hinting to the engine it should cache
* - `filename` filename of the view being rendered
*
2015-08-14 19:58:05 +00:00
* @param {String} view
* @param {Object|Function} options or callback function
* @param {Function} fn
* @api public
2015-07-20 13:42:07 +00:00
*/
2015-08-14 19:58:05 +00:00
res.render = function(view, options, fn){
var self = this
, options = options || {}
, req = this.req
, app = req.app;
2015-07-20 13:42:07 +00:00
// support callback function as second arg
2015-08-14 19:58:05 +00:00
if ('function' == typeof options) {
fn = options, options = {};
2015-07-20 13:42:07 +00:00
}
// merge res.locals
2015-08-14 19:58:05 +00:00
options._locals = self.locals;
2015-07-20 13:42:07 +00:00
// default callback to respond
2015-08-14 19:58:05 +00:00
fn = fn || function(err, str){
2015-07-20 13:42:07 +00:00
if (err) return req.next(err);
self.send(str);
};
// render
2015-08-14 19:58:05 +00:00
app.render(view, options, fn);
2015-07-20 13:42:07 +00:00
};