Node-Red configuration
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

application.js 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661
  1. /*!
  2. * express
  3. * Copyright(c) 2009-2013 TJ Holowaychuk
  4. * Copyright(c) 2013 Roman Shtylman
  5. * Copyright(c) 2014-2015 Douglas Christopher Wilson
  6. * MIT Licensed
  7. */
  8. 'use strict';
  9. /**
  10. * Module dependencies.
  11. * @private
  12. */
  13. var finalhandler = require('finalhandler');
  14. var Router = require('./router');
  15. var methods = require('methods');
  16. var middleware = require('./middleware/init');
  17. var query = require('./middleware/query');
  18. var debug = require('debug')('express:application');
  19. var View = require('./view');
  20. var http = require('http');
  21. var compileETag = require('./utils').compileETag;
  22. var compileQueryParser = require('./utils').compileQueryParser;
  23. var compileTrust = require('./utils').compileTrust;
  24. var deprecate = require('depd')('express');
  25. var flatten = require('array-flatten');
  26. var merge = require('utils-merge');
  27. var resolve = require('path').resolve;
  28. var setPrototypeOf = require('setprototypeof')
  29. /**
  30. * Module variables.
  31. * @private
  32. */
  33. var hasOwnProperty = Object.prototype.hasOwnProperty
  34. var slice = Array.prototype.slice;
  35. /**
  36. * Application prototype.
  37. */
  38. var app = exports = module.exports = {};
  39. /**
  40. * Variable for trust proxy inheritance back-compat
  41. * @private
  42. */
  43. var trustProxyDefaultSymbol = '@@symbol:trust_proxy_default';
  44. /**
  45. * Initialize the server.
  46. *
  47. * - setup default configuration
  48. * - setup default middleware
  49. * - setup route reflection methods
  50. *
  51. * @private
  52. */
  53. app.init = function init() {
  54. this.cache = {};
  55. this.engines = {};
  56. this.settings = {};
  57. this.defaultConfiguration();
  58. };
  59. /**
  60. * Initialize application configuration.
  61. * @private
  62. */
  63. app.defaultConfiguration = function defaultConfiguration() {
  64. var env = process.env.NODE_ENV || 'development';
  65. // default settings
  66. this.enable('x-powered-by');
  67. this.set('etag', 'weak');
  68. this.set('env', env);
  69. this.set('query parser', 'extended');
  70. this.set('subdomain offset', 2);
  71. this.set('trust proxy', false);
  72. // trust proxy inherit back-compat
  73. Object.defineProperty(this.settings, trustProxyDefaultSymbol, {
  74. configurable: true,
  75. value: true
  76. });
  77. debug('booting in %s mode', env);
  78. this.on('mount', function onmount(parent) {
  79. // inherit trust proxy
  80. if (this.settings[trustProxyDefaultSymbol] === true
  81. && typeof parent.settings['trust proxy fn'] === 'function') {
  82. delete this.settings['trust proxy'];
  83. delete this.settings['trust proxy fn'];
  84. }
  85. // inherit protos
  86. setPrototypeOf(this.request, parent.request)
  87. setPrototypeOf(this.response, parent.response)
  88. setPrototypeOf(this.engines, parent.engines)
  89. setPrototypeOf(this.settings, parent.settings)
  90. });
  91. // setup locals
  92. this.locals = Object.create(null);
  93. // top-most app is mounted at /
  94. this.mountpath = '/';
  95. // default locals
  96. this.locals.settings = this.settings;
  97. // default configuration
  98. this.set('view', View);
  99. this.set('views', resolve('views'));
  100. this.set('jsonp callback name', 'callback');
  101. if (env === 'production') {
  102. this.enable('view cache');
  103. }
  104. Object.defineProperty(this, 'router', {
  105. get: function() {
  106. throw new Error('\'app.router\' is deprecated!\nPlease see the 3.x to 4.x migration guide for details on how to update your app.');
  107. }
  108. });
  109. };
  110. /**
  111. * lazily adds the base router if it has not yet been added.
  112. *
  113. * We cannot add the base router in the defaultConfiguration because
  114. * it reads app settings which might be set after that has run.
  115. *
  116. * @private
  117. */
  118. app.lazyrouter = function lazyrouter() {
  119. if (!this._router) {
  120. this._router = new Router({
  121. caseSensitive: this.enabled('case sensitive routing'),
  122. strict: this.enabled('strict routing')
  123. });
  124. this._router.use(query(this.get('query parser fn')));
  125. this._router.use(middleware.init(this));
  126. }
  127. };
  128. /**
  129. * Dispatch a req, res pair into the application. Starts pipeline processing.
  130. *
  131. * If no callback is provided, then default error handlers will respond
  132. * in the event of an error bubbling through the stack.
  133. *
  134. * @private
  135. */
  136. app.handle = function handle(req, res, callback) {
  137. var router = this._router;
  138. // final handler
  139. var done = callback || finalhandler(req, res, {
  140. env: this.get('env'),
  141. onerror: logerror.bind(this)
  142. });
  143. // no routes
  144. if (!router) {
  145. debug('no routes defined on app');
  146. done();
  147. return;
  148. }
  149. router.handle(req, res, done);
  150. };
  151. /**
  152. * Proxy `Router#use()` to add middleware to the app router.
  153. * See Router#use() documentation for details.
  154. *
  155. * If the _fn_ parameter is an express app, then it will be
  156. * mounted at the _route_ specified.
  157. *
  158. * @public
  159. */
  160. app.use = function use(fn) {
  161. var offset = 0;
  162. var path = '/';
  163. // default path to '/'
  164. // disambiguate app.use([fn])
  165. if (typeof fn !== 'function') {
  166. var arg = fn;
  167. while (Array.isArray(arg) && arg.length !== 0) {
  168. arg = arg[0];
  169. }
  170. // first arg is the path
  171. if (typeof arg !== 'function') {
  172. offset = 1;
  173. path = fn;
  174. }
  175. }
  176. var fns = flatten(slice.call(arguments, offset));
  177. if (fns.length === 0) {
  178. throw new TypeError('app.use() requires a middleware function')
  179. }
  180. // setup router
  181. this.lazyrouter();
  182. var router = this._router;
  183. fns.forEach(function (fn) {
  184. // non-express app
  185. if (!fn || !fn.handle || !fn.set) {
  186. return router.use(path, fn);
  187. }
  188. debug('.use app under %s', path);
  189. fn.mountpath = path;
  190. fn.parent = this;
  191. // restore .app property on req and res
  192. router.use(path, function mounted_app(req, res, next) {
  193. var orig = req.app;
  194. fn.handle(req, res, function (err) {
  195. setPrototypeOf(req, orig.request)
  196. setPrototypeOf(res, orig.response)
  197. next(err);
  198. });
  199. });
  200. // mounted an app
  201. fn.emit('mount', this);
  202. }, this);
  203. return this;
  204. };
  205. /**
  206. * Proxy to the app `Router#route()`
  207. * Returns a new `Route` instance for the _path_.
  208. *
  209. * Routes are isolated middleware stacks for specific paths.
  210. * See the Route api docs for details.
  211. *
  212. * @public
  213. */
  214. app.route = function route(path) {
  215. this.lazyrouter();
  216. return this._router.route(path);
  217. };
  218. /**
  219. * Register the given template engine callback `fn`
  220. * as `ext`.
  221. *
  222. * By default will `require()` the engine based on the
  223. * file extension. For example if you try to render
  224. * a "foo.ejs" file Express will invoke the following internally:
  225. *
  226. * app.engine('ejs', require('ejs').__express);
  227. *
  228. * For engines that do not provide `.__express` out of the box,
  229. * or if you wish to "map" a different extension to the template engine
  230. * you may use this method. For example mapping the EJS template engine to
  231. * ".html" files:
  232. *
  233. * app.engine('html', require('ejs').renderFile);
  234. *
  235. * In this case EJS provides a `.renderFile()` method with
  236. * the same signature that Express expects: `(path, options, callback)`,
  237. * though note that it aliases this method as `ejs.__express` internally
  238. * so if you're using ".ejs" extensions you don't need to do anything.
  239. *
  240. * Some template engines do not follow this convention, the
  241. * [Consolidate.js](https://github.com/tj/consolidate.js)
  242. * library was created to map all of node's popular template
  243. * engines to follow this convention, thus allowing them to
  244. * work seamlessly within Express.
  245. *
  246. * @param {String} ext
  247. * @param {Function} fn
  248. * @return {app} for chaining
  249. * @public
  250. */
  251. app.engine = function engine(ext, fn) {
  252. if (typeof fn !== 'function') {
  253. throw new Error('callback function required');
  254. }
  255. // get file extension
  256. var extension = ext[0] !== '.'
  257. ? '.' + ext
  258. : ext;
  259. // store engine
  260. this.engines[extension] = fn;
  261. return this;
  262. };
  263. /**
  264. * Proxy to `Router#param()` with one added api feature. The _name_ parameter
  265. * can be an array of names.
  266. *
  267. * See the Router#param() docs for more details.
  268. *
  269. * @param {String|Array} name
  270. * @param {Function} fn
  271. * @return {app} for chaining
  272. * @public
  273. */
  274. app.param = function param(name, fn) {
  275. this.lazyrouter();
  276. if (Array.isArray(name)) {
  277. for (var i = 0; i < name.length; i++) {
  278. this.param(name[i], fn);
  279. }
  280. return this;
  281. }
  282. this._router.param(name, fn);
  283. return this;
  284. };
  285. /**
  286. * Assign `setting` to `val`, or return `setting`'s value.
  287. *
  288. * app.set('foo', 'bar');
  289. * app.set('foo');
  290. * // => "bar"
  291. *
  292. * Mounted servers inherit their parent server's settings.
  293. *
  294. * @param {String} setting
  295. * @param {*} [val]
  296. * @return {Server} for chaining
  297. * @public
  298. */
  299. app.set = function set(setting, val) {
  300. if (arguments.length === 1) {
  301. // app.get(setting)
  302. var settings = this.settings
  303. while (settings && settings !== Object.prototype) {
  304. if (hasOwnProperty.call(settings, setting)) {
  305. return settings[setting]
  306. }
  307. settings = Object.getPrototypeOf(settings)
  308. }
  309. return undefined
  310. }
  311. debug('set "%s" to %o', setting, val);
  312. // set value
  313. this.settings[setting] = val;
  314. // trigger matched settings
  315. switch (setting) {
  316. case 'etag':
  317. this.set('etag fn', compileETag(val));
  318. break;
  319. case 'query parser':
  320. this.set('query parser fn', compileQueryParser(val));
  321. break;
  322. case 'trust proxy':
  323. this.set('trust proxy fn', compileTrust(val));
  324. // trust proxy inherit back-compat
  325. Object.defineProperty(this.settings, trustProxyDefaultSymbol, {
  326. configurable: true,
  327. value: false
  328. });
  329. break;
  330. }
  331. return this;
  332. };
  333. /**
  334. * Return the app's absolute pathname
  335. * based on the parent(s) that have
  336. * mounted it.
  337. *
  338. * For example if the application was
  339. * mounted as "/admin", which itself
  340. * was mounted as "/blog" then the
  341. * return value would be "/blog/admin".
  342. *
  343. * @return {String}
  344. * @private
  345. */
  346. app.path = function path() {
  347. return this.parent
  348. ? this.parent.path() + this.mountpath
  349. : '';
  350. };
  351. /**
  352. * Check if `setting` is enabled (truthy).
  353. *
  354. * app.enabled('foo')
  355. * // => false
  356. *
  357. * app.enable('foo')
  358. * app.enabled('foo')
  359. * // => true
  360. *
  361. * @param {String} setting
  362. * @return {Boolean}
  363. * @public
  364. */
  365. app.enabled = function enabled(setting) {
  366. return Boolean(this.set(setting));
  367. };
  368. /**
  369. * Check if `setting` is disabled.
  370. *
  371. * app.disabled('foo')
  372. * // => true
  373. *
  374. * app.enable('foo')
  375. * app.disabled('foo')
  376. * // => false
  377. *
  378. * @param {String} setting
  379. * @return {Boolean}
  380. * @public
  381. */
  382. app.disabled = function disabled(setting) {
  383. return !this.set(setting);
  384. };
  385. /**
  386. * Enable `setting`.
  387. *
  388. * @param {String} setting
  389. * @return {app} for chaining
  390. * @public
  391. */
  392. app.enable = function enable(setting) {
  393. return this.set(setting, true);
  394. };
  395. /**
  396. * Disable `setting`.
  397. *
  398. * @param {String} setting
  399. * @return {app} for chaining
  400. * @public
  401. */
  402. app.disable = function disable(setting) {
  403. return this.set(setting, false);
  404. };
  405. /**
  406. * Delegate `.VERB(...)` calls to `router.VERB(...)`.
  407. */
  408. methods.forEach(function(method){
  409. app[method] = function(path){
  410. if (method === 'get' && arguments.length === 1) {
  411. // app.get(setting)
  412. return this.set(path);
  413. }
  414. this.lazyrouter();
  415. var route = this._router.route(path);
  416. route[method].apply(route, slice.call(arguments, 1));
  417. return this;
  418. };
  419. });
  420. /**
  421. * Special-cased "all" method, applying the given route `path`,
  422. * middleware, and callback to _every_ HTTP method.
  423. *
  424. * @param {String} path
  425. * @param {Function} ...
  426. * @return {app} for chaining
  427. * @public
  428. */
  429. app.all = function all(path) {
  430. this.lazyrouter();
  431. var route = this._router.route(path);
  432. var args = slice.call(arguments, 1);
  433. for (var i = 0; i < methods.length; i++) {
  434. route[methods[i]].apply(route, args);
  435. }
  436. return this;
  437. };
  438. // del -> delete alias
  439. app.del = deprecate.function(app.delete, 'app.del: Use app.delete instead');
  440. /**
  441. * Render the given view `name` name with `options`
  442. * and a callback accepting an error and the
  443. * rendered template string.
  444. *
  445. * Example:
  446. *
  447. * app.render('email', { name: 'Tobi' }, function(err, html){
  448. * // ...
  449. * })
  450. *
  451. * @param {String} name
  452. * @param {Object|Function} options or fn
  453. * @param {Function} callback
  454. * @public
  455. */
  456. app.render = function render(name, options, callback) {
  457. var cache = this.cache;
  458. var done = callback;
  459. var engines = this.engines;
  460. var opts = options;
  461. var renderOptions = {};
  462. var view;
  463. // support callback function as second arg
  464. if (typeof options === 'function') {
  465. done = options;
  466. opts = {};
  467. }
  468. // merge app.locals
  469. merge(renderOptions, this.locals);
  470. // merge options._locals
  471. if (opts._locals) {
  472. merge(renderOptions, opts._locals);
  473. }
  474. // merge options
  475. merge(renderOptions, opts);
  476. // set .cache unless explicitly provided
  477. if (renderOptions.cache == null) {
  478. renderOptions.cache = this.enabled('view cache');
  479. }
  480. // primed cache
  481. if (renderOptions.cache) {
  482. view = cache[name];
  483. }
  484. // view
  485. if (!view) {
  486. var View = this.get('view');
  487. view = new View(name, {
  488. defaultEngine: this.get('view engine'),
  489. root: this.get('views'),
  490. engines: engines
  491. });
  492. if (!view.path) {
  493. var dirs = Array.isArray(view.root) && view.root.length > 1
  494. ? 'directories "' + view.root.slice(0, -1).join('", "') + '" or "' + view.root[view.root.length - 1] + '"'
  495. : 'directory "' + view.root + '"'
  496. var err = new Error('Failed to lookup view "' + name + '" in views ' + dirs);
  497. err.view = view;
  498. return done(err);
  499. }
  500. // prime the cache
  501. if (renderOptions.cache) {
  502. cache[name] = view;
  503. }
  504. }
  505. // render
  506. tryRender(view, renderOptions, done);
  507. };
  508. /**
  509. * Listen for connections.
  510. *
  511. * A node `http.Server` is returned, with this
  512. * application (which is a `Function`) as its
  513. * callback. If you wish to create both an HTTP
  514. * and HTTPS server you may do so with the "http"
  515. * and "https" modules as shown here:
  516. *
  517. * var http = require('http')
  518. * , https = require('https')
  519. * , express = require('express')
  520. * , app = express();
  521. *
  522. * http.createServer(app).listen(80);
  523. * https.createServer({ ... }, app).listen(443);
  524. *
  525. * @return {http.Server}
  526. * @public
  527. */
  528. app.listen = function listen() {
  529. var server = http.createServer(this);
  530. return server.listen.apply(server, arguments);
  531. };
  532. /**
  533. * Log error using console.error.
  534. *
  535. * @param {Error} err
  536. * @private
  537. */
  538. function logerror(err) {
  539. /* istanbul ignore next */
  540. if (this.get('env') !== 'test') console.error(err.stack || err.toString());
  541. }
  542. /**
  543. * Try rendering a view.
  544. * @private
  545. */
  546. function tryRender(view, options, callback) {
  547. try {
  548. view.render(options, callback);
  549. } catch (err) {
  550. callback(err);
  551. }
  552. }