123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176 |
- /*!
- * express
- * Copyright(c) 2009-2013 TJ Holowaychuk
- * Copyright(c) 2013 Roman Shtylman
- * Copyright(c) 2014-2015 Douglas Christopher Wilson
- * MIT Licensed
- */
- 'use strict';
- /**
- * Module dependencies.
- * @private
- */
- var debug = require('debug')('express:view');
- var path = require('path');
- var fs = require('fs');
- var utils = require('./utils');
- /**
- * Module variables.
- * @private
- */
- var dirname = path.dirname;
- var basename = path.basename;
- var extname = path.extname;
- var join = path.join;
- var resolve = path.resolve;
- /**
- * Module exports.
- * @public
- */
- module.exports = View;
- /**
- * Initialize a new `View` with the given `name`.
- *
- * Options:
- *
- * - `defaultEngine` the default template engine name
- * - `engines` template engine require() cache
- * - `root` root path for view lookup
- *
- * @param {string} name
- * @param {object} options
- * @public
- */
- function View(name, options) {
- var opts = options || {};
- this.defaultEngine = opts.defaultEngine;
- this.ext = extname(name);
- this.name = name;
- this.root = opts.root;
- if (!this.ext && !this.defaultEngine) {
- throw new Error('No default engine was specified and no extension was provided.');
- }
- var fileName = name;
- if (!this.ext) {
- // get extension from default engine name
- this.ext = this.defaultEngine[0] !== '.'
- ? '.' + this.defaultEngine
- : this.defaultEngine;
- fileName += this.ext;
- }
- if (!opts.engines[this.ext]) {
- // load engine
- var mod = this.ext.substr(1)
- debug('require "%s"', mod)
- opts.engines[this.ext] = require(mod).__express
- }
- // store loaded engine
- this.engine = opts.engines[this.ext];
- // lookup path
- this.path = this.lookup(fileName);
- }
- /**
- * Lookup view by the given `name`
- *
- * @param {string} name
- * @private
- */
- View.prototype.lookup = function lookup(name) {
- var path;
- var roots = [].concat(this.root);
- debug('lookup "%s"', name);
- for (var i = 0; i < roots.length && !path; i++) {
- var root = roots[i];
- // resolve the path
- var loc = resolve(root, name);
- var dir = dirname(loc);
- var file = basename(loc);
- // resolve the file
- path = this.resolve(dir, file);
- }
- return path;
- };
- /**
- * Render with the given options.
- *
- * @param {object} options
- * @param {function} callback
- * @private
- */
- View.prototype.render = function render(options, callback) {
- debug('render "%s"', this.path);
- this.engine(this.path, options, callback);
- };
- /**
- * Resolve the file within the given directory.
- *
- * @param {string} dir
- * @param {string} file
- * @private
- */
- View.prototype.resolve = function resolve(dir, file) {
- var ext = this.ext;
- // <path>.<ext>
- var path = join(dir, file);
- var stat = tryStat(path);
- if (stat && stat.isFile()) {
- return path;
- }
- // <path>/index.<ext>
- path = join(dir, basename(file, ext), 'index' + ext);
- stat = tryStat(path);
- if (stat && stat.isFile()) {
- return path;
- }
- };
- /**
- * Return a stat, maybe.
- *
- * @param {string} path
- * @return {fs.Stats}
- * @private
- */
- function tryStat(path) {
- debug('stat "%s"', path);
- try {
- return fs.statSync(path);
- } catch (e) {
- return undefined;
- }
- }
|