pixsy/node_modules/pug-cli/index.js

311 lines
8.6 KiB
JavaScript

#!/usr/bin/env node
'use strict';
var fs = require('fs');
var path = require('path');
var program = require('commander');
var mkdirp = require('mkdirp');
var chalk = require('chalk');
var pug = require('pug');
var basename = path.basename;
var dirname = path.dirname;
var resolve = path.resolve;
var normalize = path.normalize;
var join = path.join;
var relative = path.relative;
// Pug options
var options = {};
// options
program
.version(
'pug version: ' + require('pug/package.json').version + '\n' +
'pug-cli version: ' + require( './package.json').version
)
.usage('[options] [dir|file ...]')
.option('-O, --obj <str|path>', 'JSON/JavaScript options object or file')
.option('-o, --out <dir>', 'output the rendered HTML or compiled JavaScript to <dir>')
.option('-p, --path <path>', 'filename used to resolve includes')
.option('-b, --basedir <path>', 'path used as root directory to resolve absolute includes')
.option('-P, --pretty', 'compile pretty HTML output')
.option('-c, --client', 'compile function for client-side')
.option('-n, --name <str>', 'the name of the compiled template (requires --client)')
.option('-D, --no-debug', 'compile without debugging (smaller functions)')
.option('-w, --watch', 'watch files for changes and automatically re-render')
.option('-E, --extension <ext>', 'specify the output file extension')
.option('-s, --silent', 'do not output logs')
.option('--name-after-file', 'name the template after the last section of the file path (requires --client and overriden by --name)')
.option('--doctype <str>', 'specify the doctype on the command line (useful if it is not specified by the template)')
program.on('--help', function(){
console.log(' Examples:');
console.log('');
console.log(' # Render all files in the `templates` directory:');
console.log(' $ pug templates');
console.log('');
console.log(' # Create {foo,bar}.html:');
console.log(' $ pug {foo,bar}.pug');
console.log('');
console.log(' # Using `pug` over standard input and output streams');
console.log(' $ pug < my.pug > my.html');
console.log(' $ echo \'h1 Pug!\' | pug');
console.log('');
console.log(' # Render all files in `foo` and `bar` directories to `/tmp`:');
console.log(' $ pug foo bar --out /tmp');
console.log('');
console.log(' # Specify options through a string:');
console.log(' $ pug -O \'{"doctype": "html"}\' foo.pug');
console.log(' # or, using JavaScript instead of JSON');
console.log(' $ pug -O "{doctype: \'html\'}" foo.pug');
console.log('');
console.log(' # Specify options through a file:');
console.log(' $ echo "exports.doctype = \'html\';" > options.js');
console.log(' $ pug -O options.js foo.pug');
console.log(' # or, JSON works too');
console.log(' $ echo \'{"doctype": "html"}\' > options.json');
console.log(' $ pug -O options.json foo.pug');
console.log('');
});
program.parse(process.argv);
// options given, parse them
if (program.obj) {
options = parseObj(program.obj);
}
/**
* Parse object either in `input` or in the file called `input`. The latter is
* searched first.
*/
function parseObj (input) {
try {
return require(path.resolve(input));
} catch (e) {
var str;
try {
str = fs.readFileSync(program.obj, 'utf8');
} catch (e) {
str = program.obj;
}
try {
return JSON.parse(str);
} catch (e) {
return eval('(' + str + ')');
}
}
}
[
['path', 'filename'], // --path
['debug', 'compileDebug'], // --no-debug
['client', 'client'], // --client
['pretty', 'pretty'], // --pretty
['basedir', 'basedir'], // --basedir
['doctype', 'doctype'], // --doctype
].forEach(function (o) {
options[o[1]] = program[o[0]] !== undefined ? program[o[0]] : options[o[1]];
});
// --name
if (typeof program.name === 'string') {
options.name = program.name;
}
// --silent
var consoleLog = program.silent ? function() {} : console.log;
// left-over args are file paths
var files = program.args;
// object of reverse dependencies of a watched file, including itself if
// applicable
var watchList = {};
// function for rendering
var render = program.watch ? tryRender : renderFile;
// compile files
if (files.length) {
consoleLog();
if (program.watch) {
process.on('SIGINT', function() {
process.exit(1);
});
}
files.forEach(function (file) {
render(file);
});
// stdio
} else {
stdin();
}
/**
* Watch for changes on path
*
* Renders `base` if specified, otherwise renders `path`.
*/
function watchFile(path, base, rootPath) {
path = normalize(path);
var log = ' ' + chalk.gray('watching') + ' ' + chalk.cyan(path);
if (!base) {
base = path;
} else {
base = normalize(base);
log += '\n ' + chalk.gray('as a dependency of') + ' ';
log += chalk.cyan(base);
}
if (watchList[path]) {
if (watchList[path].indexOf(base) !== -1) return;
consoleLog(log);
watchList[path].push(base);
return;
}
consoleLog(log);
watchList[path] = [base];
fs.watchFile(path, {persistent: true, interval: 200},
function (curr, prev) {
// File doesn't exist anymore. Keep watching.
if (curr.mtime.getTime() === 0) return;
// istanbul ignore if
if (curr.mtime.getTime() === prev.mtime.getTime()) return;
watchList[path].forEach(function(file) {
tryRender(file, rootPath);
});
});
}
/**
* Convert error to string
*/
function errorToString(e) {
return e.stack || /* istanbul ignore next */ (e.message || e);
}
/**
* Try to render `path`; if an exception is thrown it is printed to stderr and
* otherwise ignored.
*
* This is used in watch mode.
*/
function tryRender(path, rootPath) {
try {
renderFile(path, rootPath);
} catch (e) {
// keep watching when error occured.
console.error(errorToString(e));
}
}
/**
* Compile from stdin.
*/
function stdin() {
var buf = '';
process.stdin.setEncoding('utf8');
process.stdin.on('data', function(chunk){ buf += chunk; });
process.stdin.on('end', function(){
var output;
if (options.client) {
output = pug.compileClient(buf, options);
} else {
var fn = pug.compile(buf, options);
var output = fn(options);
}
process.stdout.write(output);
}).resume();
}
/**
* Process the given path, compiling the pug files found.
* Always walk the subdirectories.
*
* @param path path of the file, might be relative
* @param rootPath path relative to the directory specified in the command
*/
function renderFile(path, rootPath) {
var re = /\.(?:pug|jade)$/;
var stat = fs.lstatSync(path);
// Found pug file
if (stat.isFile() && re.test(path)) {
// Try to watch the file if needed. watchFile takes care of duplicates.
if (program.watch) watchFile(path, null, rootPath);
if (program.nameAfterFile) {
options.name = getNameFromFileName(path);
}
var fn = options.client
? pug.compileFileClient(path, options)
: pug.compileFile(path, options);
if (program.watch && fn.dependencies) {
// watch dependencies, and recompile the base
fn.dependencies.forEach(function (dep) {
watchFile(dep, path, rootPath);
});
}
// --extension
var extname;
if (program.extension) extname = '.' + program.extension;
else if (options.client) extname = '.js';
else extname = '.html';
// path: foo.pug -> foo.<ext>
path = path.replace(re, extname);
if (program.out) {
// prepend output directory
if (rootPath) {
// replace the rootPath of the resolved path with output directory
path = relative(rootPath, path);
} else {
// if no rootPath handling is needed
path = basename(path);
}
path = resolve(program.out, path);
}
var dir = resolve(dirname(path));
mkdirp.sync(dir);
var output = options.client ? fn : fn(options);
fs.writeFileSync(path, output);
consoleLog(' ' + chalk.gray('rendered') + ' ' + chalk.cyan('%s'), normalize(path));
// Found directory
} else if (stat.isDirectory()) {
var files = fs.readdirSync(path);
files.map(function(filename) {
return path + '/' + filename;
}).forEach(function (file) {
render(file, rootPath || path);
});
}
}
/**
* Get a sensible name for a template function from a file path
*
* @param {String} filename
* @returns {String}
*/
function getNameFromFileName(filename) {
var file = basename(filename).replace(/\.(?:pug|jade)$/, '');
return file.toLowerCase().replace(/[^a-z0-9]+([a-z])/g, function (_, character) {
return character.toUpperCase();
}) + 'Template';
}