123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295 |
- #!/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';
- }
|