parser.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699
  1. /*!
  2. * Jade - Parser
  3. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  4. * MIT Licensed
  5. */
  6. /**
  7. * Module dependencies.
  8. */
  9. var Lexer = require('./lexer')
  10. , nodes = require('./nodes')
  11. , utils = require('./utils')
  12. , filters = require('./filters')
  13. , path = require('path')
  14. , extname = path.extname;
  15. /**
  16. * Initialize `Parser` with the given input `str` and `filename`.
  17. *
  18. * @param {String} str
  19. * @param {String} filename
  20. * @param {Object} options
  21. * @api public
  22. */
  23. var Parser = exports = module.exports = function Parser(str, filename, options){
  24. this.input = str;
  25. this.lexer = new Lexer(str, options);
  26. this.filename = filename;
  27. this.blocks = {};
  28. this.mixins = {};
  29. this.options = options;
  30. this.contexts = [this];
  31. };
  32. /**
  33. * Tags that may not contain tags.
  34. */
  35. var textOnly = exports.textOnly = ['script', 'style'];
  36. /**
  37. * Parser prototype.
  38. */
  39. Parser.prototype = {
  40. /**
  41. * Push `parser` onto the context stack,
  42. * or pop and return a `Parser`.
  43. */
  44. context: function(parser){
  45. if (parser) {
  46. this.contexts.push(parser);
  47. } else {
  48. return this.contexts.pop();
  49. }
  50. },
  51. /**
  52. * Return the next token object.
  53. *
  54. * @return {Object}
  55. * @api private
  56. */
  57. advance: function(){
  58. return this.lexer.advance();
  59. },
  60. /**
  61. * Skip `n` tokens.
  62. *
  63. * @param {Number} n
  64. * @api private
  65. */
  66. skip: function(n){
  67. while (n--) this.advance();
  68. },
  69. /**
  70. * Single token lookahead.
  71. *
  72. * @return {Object}
  73. * @api private
  74. */
  75. peek: function() {
  76. return this.lookahead(1);
  77. },
  78. /**
  79. * Return lexer lineno.
  80. *
  81. * @return {Number}
  82. * @api private
  83. */
  84. line: function() {
  85. return this.lexer.lineno;
  86. },
  87. /**
  88. * `n` token lookahead.
  89. *
  90. * @param {Number} n
  91. * @return {Object}
  92. * @api private
  93. */
  94. lookahead: function(n){
  95. return this.lexer.lookahead(n);
  96. },
  97. /**
  98. * Parse input returning a string of js for evaluation.
  99. *
  100. * @return {String}
  101. * @api public
  102. */
  103. parse: function(){
  104. var block = new nodes.Block, parser;
  105. block.line = this.line();
  106. while ('eos' != this.peek().type) {
  107. if ('newline' == this.peek().type) {
  108. this.advance();
  109. } else {
  110. block.push(this.parseExpr());
  111. }
  112. }
  113. if (parser = this.extending) {
  114. this.context(parser);
  115. var ast = parser.parse();
  116. this.context();
  117. // hoist mixins
  118. for (var name in this.mixins)
  119. ast.unshift(this.mixins[name]);
  120. return ast;
  121. }
  122. return block;
  123. },
  124. /**
  125. * Expect the given type, or throw an exception.
  126. *
  127. * @param {String} type
  128. * @api private
  129. */
  130. expect: function(type){
  131. if (this.peek().type === type) {
  132. return this.advance();
  133. } else {
  134. throw new Error('expected "' + type + '", but got "' + this.peek().type + '"');
  135. }
  136. },
  137. /**
  138. * Accept the given `type`.
  139. *
  140. * @param {String} type
  141. * @api private
  142. */
  143. accept: function(type){
  144. if (this.peek().type === type) {
  145. return this.advance();
  146. }
  147. },
  148. /**
  149. * tag
  150. * | doctype
  151. * | mixin
  152. * | include
  153. * | filter
  154. * | comment
  155. * | text
  156. * | each
  157. * | code
  158. * | yield
  159. * | id
  160. * | class
  161. * | interpolation
  162. */
  163. parseExpr: function(){
  164. switch (this.peek().type) {
  165. case 'tag':
  166. return this.parseTag();
  167. case 'mixin':
  168. return this.parseMixin();
  169. case 'block':
  170. return this.parseBlock();
  171. case 'case':
  172. return this.parseCase();
  173. case 'when':
  174. return this.parseWhen();
  175. case 'default':
  176. return this.parseDefault();
  177. case 'extends':
  178. return this.parseExtends();
  179. case 'include':
  180. return this.parseInclude();
  181. case 'doctype':
  182. return this.parseDoctype();
  183. case 'filter':
  184. return this.parseFilter();
  185. case 'comment':
  186. return this.parseComment();
  187. case 'text':
  188. return this.parseText();
  189. case 'each':
  190. return this.parseEach();
  191. case 'code':
  192. return this.parseCode();
  193. case 'call':
  194. return this.parseCall();
  195. case 'interpolation':
  196. return this.parseInterpolation();
  197. case 'yield':
  198. this.advance();
  199. var block = new nodes.Block;
  200. block.yield = true;
  201. return block;
  202. case 'id':
  203. case 'class':
  204. var tok = this.advance();
  205. this.lexer.defer(this.lexer.tok('tag', 'div'));
  206. this.lexer.defer(tok);
  207. return this.parseExpr();
  208. default:
  209. throw new Error('unexpected token "' + this.peek().type + '"');
  210. }
  211. },
  212. /**
  213. * Text
  214. */
  215. parseText: function(){
  216. var tok = this.expect('text');
  217. var node = new nodes.Text(tok.val);
  218. node.line = this.line();
  219. return node;
  220. },
  221. /**
  222. * ':' expr
  223. * | block
  224. */
  225. parseBlockExpansion: function(){
  226. if (':' == this.peek().type) {
  227. this.advance();
  228. return new nodes.Block(this.parseExpr());
  229. } else {
  230. return this.block();
  231. }
  232. },
  233. /**
  234. * case
  235. */
  236. parseCase: function(){
  237. var val = this.expect('case').val;
  238. var node = new nodes.Case(val);
  239. node.line = this.line();
  240. node.block = this.block();
  241. return node;
  242. },
  243. /**
  244. * when
  245. */
  246. parseWhen: function(){
  247. var val = this.expect('when').val
  248. return new nodes.Case.When(val, this.parseBlockExpansion());
  249. },
  250. /**
  251. * default
  252. */
  253. parseDefault: function(){
  254. this.expect('default');
  255. return new nodes.Case.When('default', this.parseBlockExpansion());
  256. },
  257. /**
  258. * code
  259. */
  260. parseCode: function(){
  261. var tok = this.expect('code');
  262. var node = new nodes.Code(tok.val, tok.buffer, tok.escape);
  263. var block;
  264. var i = 1;
  265. node.line = this.line();
  266. while (this.lookahead(i) && 'newline' == this.lookahead(i).type) ++i;
  267. block = 'indent' == this.lookahead(i).type;
  268. if (block) {
  269. this.skip(i-1);
  270. node.block = this.block();
  271. }
  272. return node;
  273. },
  274. /**
  275. * comment
  276. */
  277. parseComment: function(){
  278. var tok = this.expect('comment');
  279. var node;
  280. if ('indent' == this.peek().type) {
  281. node = new nodes.BlockComment(tok.val, this.block(), tok.buffer);
  282. } else {
  283. node = new nodes.Comment(tok.val, tok.buffer);
  284. }
  285. node.line = this.line();
  286. return node;
  287. },
  288. /**
  289. * doctype
  290. */
  291. parseDoctype: function(){
  292. var tok = this.expect('doctype');
  293. var node = new nodes.Doctype(tok.val);
  294. node.line = this.line();
  295. return node;
  296. },
  297. /**
  298. * filter attrs? text-block
  299. */
  300. parseFilter: function(){
  301. var tok = this.expect('filter');
  302. var attrs = this.accept('attrs');
  303. var block;
  304. this.lexer.pipeless = true;
  305. block = this.parseTextBlock();
  306. this.lexer.pipeless = false;
  307. var node = new nodes.Filter(tok.val, block, attrs && attrs.attrs);
  308. node.line = this.line();
  309. return node;
  310. },
  311. /**
  312. * each block
  313. */
  314. parseEach: function(){
  315. var tok = this.expect('each');
  316. var node = new nodes.Each(tok.code, tok.val, tok.key);
  317. node.line = this.line();
  318. node.block = this.block();
  319. if (this.peek().type == 'code' && this.peek().val == 'else') {
  320. this.advance();
  321. node.alternative = this.block();
  322. }
  323. return node;
  324. },
  325. /**
  326. * 'extends' name
  327. */
  328. parseExtends: function(){
  329. var path = require('path');
  330. var fs = require('fs');
  331. var dirname = path.dirname;
  332. var basename = path.basename;
  333. var join = path.join;
  334. if (!this.filename)
  335. throw new Error('the "filename" option is required to extend templates');
  336. path = this.expect('extends').val.trim();
  337. var dir = dirname(this.filename);
  338. path = join(dir, path + '.jade');
  339. var str = fs.readFileSync(path, 'utf8');
  340. var parser = new Parser(str, path, this.options);
  341. parser.blocks = this.blocks;
  342. parser.contexts = this.contexts;
  343. this.extending = parser;
  344. // TODO: null node
  345. return new nodes.Literal('');
  346. },
  347. /**
  348. * 'block' name block
  349. */
  350. parseBlock: function(){
  351. var block = this.expect('block');
  352. var mode = block.mode;
  353. var name = block.val.trim();
  354. block = 'indent' == this.peek().type
  355. ? this.block()
  356. : new nodes.Block(new nodes.Literal(''));
  357. var prev = this.blocks[name];
  358. if (prev) {
  359. switch (prev.mode) {
  360. case 'append':
  361. block.nodes = block.nodes.concat(prev.nodes);
  362. prev = block;
  363. break;
  364. case 'prepend':
  365. block.nodes = prev.nodes.concat(block.nodes);
  366. prev = block;
  367. break;
  368. }
  369. }
  370. block.mode = mode;
  371. return this.blocks[name] = prev || block;
  372. },
  373. /**
  374. * include block?
  375. */
  376. parseInclude: function(){
  377. var path = require('path');
  378. var fs = require('fs');
  379. var dirname = path.dirname;
  380. var basename = path.basename;
  381. var join = path.join;
  382. var str;
  383. path = this.expect('include').val.trim();
  384. var dir = dirname(this.filename);
  385. if (!this.filename)
  386. throw new Error('the "filename" option is required to use includes');
  387. // no extension
  388. if (!~basename(path).indexOf('.')) {
  389. path += '.jade';
  390. }
  391. // non-jade
  392. if ('.jade' != path.substr(-5)) {
  393. path = join(dir, path);
  394. str = fs.readFileSync(path, 'utf8').replace(/\r/g, '');
  395. var ext = extname(path).slice(1);
  396. var filter = filters[ext];
  397. if (filter) str = filter(str, { filename: path }).replace(/\\n/g, '\n');
  398. return new nodes.Literal(str);
  399. }
  400. path = join(dir, path);
  401. str = fs.readFileSync(path, 'utf8');
  402. var parser = new Parser(str, path, this.options);
  403. parser.blocks = utils.merge({}, this.blocks);
  404. parser.mixins = this.mixins;
  405. this.context(parser);
  406. var ast = parser.parse();
  407. this.context();
  408. ast.filename = path;
  409. if ('indent' == this.peek().type) {
  410. ast.includeBlock().push(this.block());
  411. }
  412. return ast;
  413. },
  414. /**
  415. * call ident block
  416. */
  417. parseCall: function(){
  418. var tok = this.expect('call');
  419. var name = tok.val;
  420. var args = tok.args;
  421. var mixin = new nodes.Mixin(name, args, new nodes.Block, true);
  422. this.tag(mixin);
  423. if (mixin.block.isEmpty()) mixin.block = null;
  424. return mixin;
  425. },
  426. /**
  427. * mixin block
  428. */
  429. parseMixin: function(){
  430. var tok = this.expect('mixin');
  431. var name = tok.val;
  432. var args = tok.args;
  433. var mixin;
  434. // definition
  435. if ('indent' == this.peek().type) {
  436. mixin = new nodes.Mixin(name, args, this.block(), false);
  437. this.mixins[name] = mixin;
  438. return mixin;
  439. // call
  440. } else {
  441. return new nodes.Mixin(name, args, null, true);
  442. }
  443. },
  444. /**
  445. * indent (text | newline)* outdent
  446. */
  447. parseTextBlock: function(){
  448. var block = new nodes.Block;
  449. block.line = this.line();
  450. var spaces = this.expect('indent').val;
  451. if (null == this._spaces) this._spaces = spaces;
  452. var indent = Array(spaces - this._spaces + 1).join(' ');
  453. while ('outdent' != this.peek().type) {
  454. switch (this.peek().type) {
  455. case 'newline':
  456. this.advance();
  457. break;
  458. case 'indent':
  459. this.parseTextBlock().nodes.forEach(function(node){
  460. block.push(node);
  461. });
  462. break;
  463. default:
  464. var text = new nodes.Text(indent + this.advance().val);
  465. text.line = this.line();
  466. block.push(text);
  467. }
  468. }
  469. if (spaces == this._spaces) this._spaces = null;
  470. this.expect('outdent');
  471. return block;
  472. },
  473. /**
  474. * indent expr* outdent
  475. */
  476. block: function(){
  477. var block = new nodes.Block;
  478. block.line = this.line();
  479. this.expect('indent');
  480. while ('outdent' != this.peek().type) {
  481. if ('newline' == this.peek().type) {
  482. this.advance();
  483. } else {
  484. block.push(this.parseExpr());
  485. }
  486. }
  487. this.expect('outdent');
  488. return block;
  489. },
  490. /**
  491. * interpolation (attrs | class | id)* (text | code | ':')? newline* block?
  492. */
  493. parseInterpolation: function(){
  494. var tok = this.advance();
  495. var tag = new nodes.Tag(tok.val);
  496. tag.buffer = true;
  497. return this.tag(tag);
  498. },
  499. /**
  500. * tag (attrs | class | id)* (text | code | ':')? newline* block?
  501. */
  502. parseTag: function(){
  503. // ast-filter look-ahead
  504. var i = 2;
  505. if ('attrs' == this.lookahead(i).type) ++i;
  506. var tok = this.advance();
  507. var tag = new nodes.Tag(tok.val);
  508. tag.selfClosing = tok.selfClosing;
  509. return this.tag(tag);
  510. },
  511. /**
  512. * Parse tag.
  513. */
  514. tag: function(tag){
  515. var dot;
  516. tag.line = this.line();
  517. // (attrs | class | id)*
  518. out:
  519. while (true) {
  520. switch (this.peek().type) {
  521. case 'id':
  522. case 'class':
  523. var tok = this.advance();
  524. tag.setAttribute(tok.type, "'" + tok.val + "'");
  525. continue;
  526. case 'attrs':
  527. var tok = this.advance()
  528. , obj = tok.attrs
  529. , escaped = tok.escaped
  530. , names = Object.keys(obj);
  531. if (tok.selfClosing) tag.selfClosing = true;
  532. for (var i = 0, len = names.length; i < len; ++i) {
  533. var name = names[i]
  534. , val = obj[name];
  535. tag.setAttribute(name, val, escaped[name]);
  536. }
  537. continue;
  538. default:
  539. break out;
  540. }
  541. }
  542. // check immediate '.'
  543. if ('.' == this.peek().val) {
  544. dot = tag.textOnly = true;
  545. this.advance();
  546. }
  547. // (text | code | ':')?
  548. switch (this.peek().type) {
  549. case 'text':
  550. tag.block.push(this.parseText());
  551. break;
  552. case 'code':
  553. tag.code = this.parseCode();
  554. break;
  555. case ':':
  556. this.advance();
  557. tag.block = new nodes.Block;
  558. tag.block.push(this.parseExpr());
  559. break;
  560. }
  561. // newline*
  562. while ('newline' == this.peek().type) this.advance();
  563. tag.textOnly = tag.textOnly || ~textOnly.indexOf(tag.name);
  564. // script special-case
  565. if ('script' == tag.name) {
  566. var type = tag.getAttribute('type');
  567. if (!dot && type && 'text/javascript' != type.replace(/^['"]|['"]$/g, '')) {
  568. tag.textOnly = false;
  569. }
  570. }
  571. // block?
  572. if ('indent' == this.peek().type) {
  573. if (tag.textOnly) {
  574. this.lexer.pipeless = true;
  575. tag.block = this.parseTextBlock();
  576. this.lexer.pipeless = false;
  577. } else {
  578. var block = this.block();
  579. if (tag.block) {
  580. for (var i = 0, len = block.nodes.length; i < len; ++i) {
  581. tag.block.push(block.nodes[i]);
  582. }
  583. } else {
  584. tag.block = block;
  585. }
  586. }
  587. }
  588. return tag;
  589. }
  590. };