You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
296 lines
7.7 KiB
JavaScript
296 lines
7.7 KiB
JavaScript
5 years ago
|
#!/usr/bin/env node
|
||
|
|
||
|
/**
|
||
|
* Module dependencies.
|
||
|
*/
|
||
|
|
||
|
var fs = require('fs')
|
||
|
, program = require('commander')
|
||
|
, path = require('path')
|
||
|
, basename = path.basename
|
||
|
, dirname = path.dirname
|
||
|
, resolve = path.resolve
|
||
|
, normalize = path.normalize
|
||
|
, join = path.join
|
||
|
, mkdirp = require('mkdirp')
|
||
|
, jade = require('../');
|
||
|
|
||
|
// jade options
|
||
|
|
||
|
var options = {};
|
||
|
|
||
|
// options
|
||
|
|
||
|
program
|
||
|
.version(require('../package.json').version)
|
||
|
.usage('[options] [dir|file ...]')
|
||
|
.option('-O, --obj <str|path>', 'JavaScript options object or JSON file containing it')
|
||
|
.option('-o, --out <dir>', 'output the compiled html to <dir>')
|
||
|
.option('-p, --path <path>', 'filename used to resolve includes')
|
||
|
.option('-P, --pretty', 'compile pretty html output')
|
||
|
.option('-c, --client', 'compile function for client-side runtime.js')
|
||
|
.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('-H, --hierarchy', 'keep directory hierarchy when a directory is specified')
|
||
|
.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(' # translate jade the templates dir');
|
||
|
console.log(' $ jade templates');
|
||
|
console.log('');
|
||
|
console.log(' # create {foo,bar}.html');
|
||
|
console.log(' $ jade {foo,bar}.jade');
|
||
|
console.log('');
|
||
|
console.log(' # jade over stdio');
|
||
|
console.log(' $ jade < my.jade > my.html');
|
||
|
console.log('');
|
||
|
console.log(' # jade over stdio');
|
||
|
console.log(' $ echo \'h1 Jade!\' | jade');
|
||
|
console.log('');
|
||
|
console.log(' # foo, bar dirs rendering to /tmp');
|
||
|
console.log(' $ jade foo bar --out /tmp ');
|
||
|
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) {
|
||
|
var str, out;
|
||
|
try {
|
||
|
str = fs.readFileSync(program.obj);
|
||
|
} catch (e) {
|
||
|
return eval('(' + program.obj + ')');
|
||
|
}
|
||
|
// We don't want to catch exceptions thrown in JSON.parse() so have to
|
||
|
// use this two-step approach.
|
||
|
return JSON.parse(str);
|
||
|
}
|
||
|
|
||
|
// --path
|
||
|
|
||
|
if (program.path) options.filename = program.path;
|
||
|
|
||
|
// --no-debug
|
||
|
|
||
|
options.compileDebug = program.debug;
|
||
|
|
||
|
// --client
|
||
|
|
||
|
options.client = program.client;
|
||
|
|
||
|
// --pretty
|
||
|
|
||
|
options.pretty = program.pretty;
|
||
|
|
||
|
// --watch
|
||
|
|
||
|
options.watch = program.watch;
|
||
|
|
||
|
// --name
|
||
|
|
||
|
if (typeof program.name === 'string') {
|
||
|
options.name = program.name;
|
||
|
}
|
||
|
|
||
|
// --doctype
|
||
|
|
||
|
options.doctype = program.doctype;
|
||
|
|
||
|
// left-over args are file paths
|
||
|
|
||
|
var files = program.args;
|
||
|
|
||
|
// array of paths that are being watched
|
||
|
|
||
|
var watchList = [];
|
||
|
|
||
|
// function for rendering
|
||
|
var render = program.watch ? tryRender : renderFile;
|
||
|
|
||
|
// compile files
|
||
|
|
||
|
if (files.length) {
|
||
|
console.log();
|
||
|
if (options.watch) {
|
||
|
process.on('SIGINT', function() {
|
||
|
process.exit(1);
|
||
|
});
|
||
|
}
|
||
|
files.forEach(function (file) {
|
||
|
render(file);
|
||
|
});
|
||
|
process.on('exit', function () {
|
||
|
console.log();
|
||
|
});
|
||
|
// stdio
|
||
|
} else {
|
||
|
stdin();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Watch for changes on path
|
||
|
*
|
||
|
* Renders `base` if specified, otherwise renders `path`.
|
||
|
*/
|
||
|
function watchFile(path, base, rootPath) {
|
||
|
path = normalize(path);
|
||
|
if (watchList.indexOf(path) !== -1) return;
|
||
|
console.log(" \033[90mwatching \033[36m%s\033[0m", path);
|
||
|
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;
|
||
|
tryRender(base || path, rootPath);
|
||
|
});
|
||
|
watchList.push(path);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 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 = jade.compileClient(buf, options);
|
||
|
} else {
|
||
|
var fn = jade.compile(buf, options);
|
||
|
var output = fn(options);
|
||
|
}
|
||
|
process.stdout.write(output);
|
||
|
}).resume();
|
||
|
|
||
|
process.on('SIGINT', function() {
|
||
|
process.stdout.write('\n');
|
||
|
process.stdin.emit('end');
|
||
|
process.stdout.write('\n');
|
||
|
process.exit();
|
||
|
})
|
||
|
}
|
||
|
|
||
|
var hierarchyWarned = false;
|
||
|
|
||
|
/**
|
||
|
* Process the given path, compiling the jade 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 = /\.jade$/;
|
||
|
var stat = fs.lstatSync(path);
|
||
|
// Found jade file/\.jade$/
|
||
|
if (stat.isFile() && re.test(path)) {
|
||
|
// Try to watch the file if needed. watchFile takes care of duplicates.
|
||
|
if (options.watch) watchFile(path, null, rootPath);
|
||
|
if (program.nameAfterFile) {
|
||
|
options.name = getNameFromFileName(path);
|
||
|
}
|
||
|
var fn = options.client
|
||
|
? jade.compileFileClient(path, options)
|
||
|
: jade.compileFile(path, options);
|
||
|
if (options.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.jade -> foo.<ext>
|
||
|
path = path.replace(re, extname);
|
||
|
if (program.out) {
|
||
|
// prepend output directory
|
||
|
if (rootPath && program.hierarchy) {
|
||
|
// replace the rootPath of the resolved path with output directory
|
||
|
path = resolve(path).replace(new RegExp('^' + resolve(rootPath)), '');
|
||
|
path = join(program.out, path);
|
||
|
} else {
|
||
|
if (rootPath && !hierarchyWarned) {
|
||
|
console.warn('In Jade 2.0.0 --hierarchy will become the default.');
|
||
|
hierarchyWarned = true;
|
||
|
}
|
||
|
// old behavior or if no rootPath handling is needed
|
||
|
path = join(program.out, basename(path));
|
||
|
}
|
||
|
}
|
||
|
var dir = resolve(dirname(path));
|
||
|
mkdirp.sync(dir, 0755);
|
||
|
var output = options.client ? fn : fn(options);
|
||
|
fs.writeFileSync(path, output);
|
||
|
console.log(' \033[90mrendered \033[36m%s\033[0m', 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, '.jade');
|
||
|
return file.toLowerCase().replace(/[^a-z0-9]+([a-z])/g, function (_, character) {
|
||
|
return character.toUpperCase();
|
||
|
}) + 'Template';
|
||
|
}
|