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.
		
		
		
		
		
			
		
			
				
					
					
						
							182 lines
						
					
					
						
							3.2 KiB
						
					
					
				
			
		
		
	
	
							182 lines
						
					
					
						
							3.2 KiB
						
					
					
				/*! | 
						|
 * 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'); | 
						|
 | 
						|
/** | 
						|
 * 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) | 
						|
 | 
						|
    // default engine export | 
						|
    var fn = require(mod).__express | 
						|
 | 
						|
    if (typeof fn !== 'function') { | 
						|
      throw new Error('Module "' + mod + '" does not provide a view engine.') | 
						|
    } | 
						|
 | 
						|
    opts.engines[this.ext] = fn | 
						|
  } | 
						|
 | 
						|
  // 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; | 
						|
  } | 
						|
}
 | 
						|
 |