UNPKG

elation

Version:

Elation Javascript Component Framework

1,829 lines (1,650 loc) 107 kB
// // Dust - Asynchronous Templating v1.0.0 // http://akdubya.github.com/dustjs // // Copyright (c) 2010, Aleksander Williams // Released under the MIT License. // var dust = {}; function getGlobal(){ return (function(){ return this.dust; }).call(null); } (function(dust) { dust.cache = {}; dust.register = function(name, tmpl) { if (!name) return; dust.cache[name] = tmpl; }; dust.render = function(name, context, callback) { var chunk = new Stub(callback).head; dust.load(name, chunk, Context.wrap(context)).end(); }; dust.stream = function(name, context) { var stream = new Stream(); dust.nextTick(function() { dust.load(name, stream.head, Context.wrap(context)).end(); }); return stream; }; dust.renderSource = function(source, context, callback) { return dust.compileFn(source)(context, callback); }; dust.compileFn = function(source, name) { var tmpl = dust.loadSource(dust.compile(source, name)); return function(context, callback) { var master = callback ? new Stub(callback) : new Stream(); dust.nextTick(function() { tmpl(master.head, Context.wrap(context)).end(); }); return master; } }; dust.load = function(name, chunk, context) { var tmpl = dust.cache[name]; if (tmpl) { return tmpl(chunk, context); } else { if (dust.onLoad) { return chunk.map(function(chunk) { dust.onLoad(name, function(err, src) { if (err) return chunk.setError(err); if (!dust.cache[name]) dust.loadSource(dust.compile(src, name)); dust.cache[name](chunk, context).end(); }); }); } return chunk.setError(new Error("Template Not Found: " + name)); } }; dust.loadSource = function(source, path) { return eval(source); }; if (Array.isArray) { dust.isArray = Array.isArray; } else { dust.isArray = function(arr) { return Object.prototype.toString.call(arr) == "[object Array]"; }; } dust.nextTick = (function() { if (typeof process !== "undefined") { return process.nextTick; } else { return function(callback) { setTimeout(callback,0); } } } )(); dust.isEmpty = function(value) { if (dust.isArray(value) && !value.length) return true; if (value === 0) return false; return (!value); }; dust.filter = function(string, auto, filters) { if (filters) { for (var i=0, len=filters.length; i<len; i++) { var name = filters[i]; if (name === "s") { auto = null; } else { string = dust.filters[name](string); } } } if (auto) { string = dust.filters[auto](string); } return string; }; dust.filters = { h: function(value) { return dust.escapeHtml(value); }, j: function(value) { return dust.escapeJs(value); }, u: encodeURI, uc: encodeURIComponent, js: function(value) { if (!JSON) { return value; } return JSON.stringify(value); }, jp: function(value) { if (!JSON) { return value; } return JSON.parse(value); }, round: function(value) { return (+value).toFixed(4); } }; function Context(stack, global, blocks) { this.stack = stack; this.global = global; this.blocks = blocks; } dust.makeBase = function(global) { return new Context(new Stack(), global); }; Context.wrap = function(context) { if (context instanceof Context) { return context; } return new Context(new Stack(context)); }; Context.prototype.get = function(key) { var ctx = this.stack, value; while(ctx) { if (ctx.isObject) { value = ctx.head[key]; if (!(value === undefined)) { return value; } } ctx = ctx.tail; } return this.global ? this.global[key] : undefined; }; Context.prototype.getPath = function(cur, down) { var ctx = this.stack, len = down.length; if (cur && len === 0) return ctx.head; ctx = ctx.head; var i = 0; while(ctx && i < len) { ctx = ctx[down[i]]; i++; } return ctx; }; Context.prototype.push = function(head, idx, len) { return new Context(new Stack(head, this.stack, idx, len), this.global, this.blocks); }; Context.prototype.rebase = function(head) { return new Context(new Stack(head), this.global, this.blocks); }; Context.prototype.current = function() { return this.stack.head; }; Context.prototype.getBlock = function(key, chk, ctx) { if (typeof key === "function") { key = key(chk, ctx).data; chk.data = ""; } var blocks = this.blocks; if (!blocks) return; var len = blocks.length, fn; while (len--) { fn = blocks[len][key]; if (fn) return fn; } }; Context.prototype.shiftBlocks = function(locals) { var blocks = this.blocks; if (locals) { if (!blocks) { newBlocks = [locals]; } else { newBlocks = blocks.concat([locals]); } return new Context(this.stack, this.global, newBlocks); } return this; }; function Stack(head, tail, idx, len) { this.tail = tail; this.isObject = !dust.isArray(head) && head && typeof head === "object"; this.head = head; this.index = idx; this.of = len; } function Stub(callback) { this.head = new Chunk(this); this.callback = callback; this.out = ''; } Stub.prototype.flush = function() { var chunk = this.head; while (chunk) { if (chunk.flushable) { this.out += chunk.data; } else if (chunk.error) { this.callback(chunk.error); this.flush = function() {}; return; } else { return; } chunk = chunk.next; this.head = chunk; } this.callback(null, this.out); }; function Stream() { this.head = new Chunk(this); } Stream.prototype.flush = function() { var chunk = this.head; while(chunk) { if (chunk.flushable) { this.emit('data', chunk.data); } else if (chunk.error) { this.emit('error', chunk.error); this.flush = function() {}; return; } else { return; } chunk = chunk.next; this.head = chunk; } this.emit('end'); }; Stream.prototype.emit = function(type, data) { if (!this.events) return false; var handler = this.events[type]; if (!handler) return false; if (typeof handler == 'function') { handler(data); } else { var listeners = handler.slice(0); for (var i = 0, l = listeners.length; i < l; i++) { listeners[i](data); } } }; Stream.prototype.on = function(type, callback) { if (!this.events) { this.events = {}; } if (!this.events[type]) { this.events[type] = callback; } else if(typeof this.events[type] === 'function') { this.events[type] = [this.events[type], callback]; } else { this.events[type].push(callback); } return this; }; Stream.prototype.pipe = function(stream) { this.on("data", function(data) { stream.write(data, "utf8"); }).on("end", function() { stream.end(); }).on("error", function(err) { stream.error(err); }); return this; }; function Chunk(root, next, taps) { this.root = root; this.next = next; this.data = ''; this.flushable = false; this.taps = taps; } Chunk.prototype.write = function(data) { var taps = this.taps; if (taps) { data = taps.go(data); } this.data += data; return this; }; Chunk.prototype.end = function(data) { if (data) { this.write(data); } this.flushable = true; this.root.flush(); return this; }; Chunk.prototype.map = function(callback) { var cursor = new Chunk(this.root, this.next, this.taps), branch = new Chunk(this.root, cursor, this.taps); this.next = branch; this.flushable = true; callback(branch); return cursor; }; Chunk.prototype.tap = function(tap) { var taps = this.taps; if (taps) { this.taps = taps.push(tap); } else { this.taps = new Tap(tap); } return this; }; Chunk.prototype.untap = function() { this.taps = this.taps.tail; return this; }; Chunk.prototype.render = function(body, context) { return body(this, context); }; Chunk.prototype.reference = function(elem, context, auto, filters) { if (typeof elem === "function") { elem.isReference = true; // Changed the function calling to use apply with the current context to make sure that "this" is wat we expect it to be inside the function elem = elem.apply(context.current(), [this, context, null, {auto: auto, filters: filters}]); if (elem instanceof Chunk) { return elem; } } if (!dust.isEmpty(elem)) { return this.write(dust.filter(elem, auto, filters)); } else { return this; } }; Chunk.prototype.section = function(elem, context, bodies, params) { if (typeof elem === "function") { elem = elem.apply(context.current(), [this, context, bodies, params]); if (elem instanceof Chunk) { return elem; } } var body = bodies.block, skip = bodies['else']; if (params) { context = context.push(params); } if (dust.isArray(elem)) { if (body) { var len = elem.length, chunk = this; context.stack.head['$len'] = len; for (var i=0; i<len; i++) { context.stack.head['$idx'] = i; chunk = body(chunk, context.push(elem[i], i, len)); } context.stack.head['$idx'] = undefined; context.stack.head['$len'] = undefined; return chunk; } } else if (elem === true) { if (body) return body(this, context); } else if (elem || elem === 0) { if (body) { context.stack.head['$idx'] = 0; context.stack.head['$len'] = 1; chunk = body(this, context.push(elem)); context.stack.head['$idx'] = undefined; context.stack.head['$len'] = undefined; return chunk; } } else if (skip) { return skip(this, context); } return this; }; Chunk.prototype.exists = function(elem, context, bodies) { var body = bodies.block, skip = bodies['else']; if (!dust.isEmpty(elem)) { if (body) return body(this, context); } else if (skip) { return skip(this, context); } return this; }; Chunk.prototype.notexists = function(elem, context, bodies) { var body = bodies.block, skip = bodies['else']; if (dust.isEmpty(elem)) { if (body) return body(this, context); } else if (skip) { return skip(this, context); } return this; }; Chunk.prototype.block = function(elem, context, bodies) { var body = bodies.block; if (elem) { body = elem; } if (body) { return body(this, context); } return this; }; Chunk.prototype.partial = function(elem, context, params) { var ctx = context.stack, tempHead = ctx.head; if (params){ //put the params context second to match what section does. {.} matches the current context without parameters //remove head context = context.rebase(ctx.tail); //put params on context = context.push(params); //reattach the head context = context.push(tempHead); } if (typeof elem === "function") { return this.capture(elem, context, function(name, chunk) { dust.load(name, chunk, context).end(); }); } return dust.load(elem, this, context); }; Chunk.prototype.helper = function(name, context, bodies, params) { return dust.helpers[name](this, context, bodies, params); }; Chunk.prototype.capture = function(body, context, callback) { return this.map(function(chunk) { var stub = new Stub(function(err, out) { if (err) { chunk.setError(err); } else { callback(out, chunk); } }); body(stub.head, context).end(); }); }; Chunk.prototype.setError = function(err) { this.error = err; this.root.flush(); return this; }; function Tap(head, tail) { this.head = head; this.tail = tail; } Tap.prototype.push = function(tap) { return new Tap(tap, this); }; Tap.prototype.go = function(value) { var tap = this; while(tap) { value = tap.head(value); tap = tap.tail; } return value; }; var HCHARS = new RegExp(/[&<>\"\']/), AMP = /&/g, LT = /</g, GT = />/g, QUOT = /\"/g, SQUOT = /\'/g; dust.escapeHtml = function(s) { if (typeof s === "string") { if (!HCHARS.test(s)) { return s; } return s.replace(AMP,'&amp;').replace(LT,'&lt;').replace(GT,'&gt;').replace(QUOT,'&quot;').replace(SQUOT, '&#39;'); } return s; }; var BS = /\\/g, CR = /\r/g, LS = /\u2028/g, PS = /\u2029/g, NL = /\n/g, LF = /\f/g, SQ = /'/g, DQ = /"/g, TB = /\t/g; dust.escapeJs = function(s) { if (typeof s === "string") { return s .replace(BS, '\\\\') .replace(DQ, '\\"') .replace(SQ, "\\'") .replace(CR, '\\r') .replace(LS, '\\u2028') .replace(PS, '\\u2029') .replace(NL, '\\n') .replace(LF, '\\f') .replace(TB, "\\t"); } return s; }; })(dust); // if (typeof exports !== "undefined") { // //TODO: Remove the helpers from dust core in the next release. // dust.helpers = require("./dust-helpers").helpers; // if (typeof process !== "undefined") { // require('./server')(dust); // } // module.exports = dust; // } (function(dust){ /* make a safe version of console if it is not available * currently supporting: * _console.log * */ var _console = (typeof console !== 'undefined')? console: { log: function(){ /* a noop*/ } }; function isSelect(context) { var value = context.current(); return typeof value === "object" && value.isSelect === true; } function filter(chunk, context, bodies, params, filter) { var params = params || {}, actual, expected; if (params.key) { actual = helpers.tap(params.key, chunk, context); } else if (isSelect(context)) { actual = context.current().selectKey; if (context.current().isResolved) { filter = function() { return false; }; } } else { throw "No key specified for filter and no key found in context from select statement"; } expected = helpers.tap(params.value, chunk, context); if (filter(expected, coerce(actual, params.type, context))) { if (isSelect(context)) { context.current().isResolved = true; } return chunk.render(bodies.block, context); } else if (bodies['else']) { return chunk.render(bodies['else'], context); } return chunk.write(''); } function coerce (value, type, context) { if (value) { switch (type || typeof(value)) { case 'number': return +value; case 'string': return String(value); case 'boolean': return Boolean(value); case 'date': return new Date(value); case 'context': return context.get(value); } } return value; } var helpers = { sep: function(chunk, context, bodies) { if (context.stack.index === context.stack.of - 1) { return chunk; } return bodies.block(chunk, context); }, idx: function(chunk, context, bodies) { return bodies.block(chunk, context.push(context.stack.index)); }, contextDump: function(chunk, context, bodies) { _console.log(JSON.stringify(context.stack)); return chunk; }, // Utility helping to resolve dust references in the given chunk tap: function( input, chunk, context ){ // return given input if there is no dust reference to resolve var output = input; // dust compiles a string to function, if there are references if( typeof input === "function"){ if( ( typeof input.isReference !== "undefined" ) && ( input.isReference === true ) ){ // just a plain function, not a dust `body` function output = input(); } else { output = ''; chunk.tap(function(data){ output += data; return ''; }).render(input, context).untap(); if( output === '' ){ output = false; } } } return output; }, /** if helper @param cond, either a string literal value or a dust reference a string literal value, is enclosed in double quotes, e.g. cond="2>3" a dust reference is also enclosed in double quotes, e.g. cond="'{val}'' > 3" cond argument should evaluate to a valid javascript expression **/ "if": function( chunk, context, bodies, params ){ if( params && params.cond ){ var cond = params.cond; cond = this.tap(cond, chunk, context); // eval expressions with given dust references if( eval( cond ) ){ return chunk.render( bodies.block, context ); } if( bodies['else'] ){ return chunk.render( bodies['else'], context ); } } // no condition else { _console.log( "No condition given in the if helper!" ); } return chunk; }, /** select/eq/lt/lte/gt/gte/default helper @param key, either a string literal value or a dust reference a string literal value, is enclosed in double quotes, e.g. key="foo" a dust reference may or may not be enclosed in double quotes, e.g. key="{val}" and key=val are both valid @param type (optiona), supported types are number, boolean, string, date, context, defaults to string **/ select: function(chunk, context, bodies, params) { if( params && params.key){ // returns given input as output, if the input is not a dust reference, else does a context lookup var key = this.tap(params.key, chunk, context); return chunk.render(bodies.block, context.push({ isSelect: true, isResolved: false, selectKey: key })); } // no key else { _console.log( "No key given in the select helper!" ); } return chunk; }, eq: function(chunk, context, bodies, params) { return filter(chunk, context, bodies, params, function(expected, actual) { return actual === expected; }); }, lt: function(chunk, context, bodies, params) { return filter(chunk, context, bodies, params, function(expected, actual) { return actual < expected; }); }, lte: function(chunk, context, bodies, params) { return filter(chunk, context, bodies, params, function(expected, actual) { return actual <= expected; }); }, gt: function(chunk, context, bodies, params) { return filter(chunk, context, bodies, params, function(expected, actual) { return actual > expected; }); }, gte: function(chunk, context, bodies, params) { return filter(chunk, context, bodies, params, function(expected, actual) { return actual >= expected; }); }, "default": function(chunk, context, bodies, params) { return filter(chunk, context, bodies, params, function(expected, actual) { return true; }); }, size: function( chunk, context, bodies, params ) { var subject = params.subject; var value = 0; if (!subject) { //undefined, "", 0 value = 0; } else if(dust.isArray(subject)) { //array value = subject.length; } else if (!isNaN(subject)) { //numeric values value = subject; } else if (Object(subject) === subject) { //object test var nr = 0; for(var k in subject) if(Object.hasOwnProperty.call(subject,k)) nr++; value = nr; } else { value = (subject + '').length; //any other value (strings etc.) } return chunk.write(value); } }; dust.helpers = helpers; })(typeof exports !== 'undefined' ? exports : getGlobal()); (function(dust) { dust.compile = function(source, name) { try { var ast = filterAST(dust.parse(source)); return compile(ast, name); } catch(err) { if(!err.line || !err.column) throw err; throw new SyntaxError(err.message + " At line : " + err.line + ", column : " + err.column); } }; function filterAST(ast) { var context = {}; return dust.filterNode(context, ast); } dust.filterNode = function(context, node) { return dust.optimizers[node[0]](context, node); } dust.optimizers = { body: compactBuffers, buffer: noop, special: convertSpecial, format: nullify, // TODO: convert format reference: visit, "#": visit, "?": visit, "^": visit, "<": visit, "+": visit, "@": visit, "%": visit, partial: visit, context: visit, params: visit, bodies: visit, param: visit, filters: noop, key: noop, path: noop, literal: noop, comment: nullify } dust.pragmas = { esc: function(compiler, context, bodies, params) { var old = compiler.auto; if (!context) context = 'h'; compiler.auto = (context === 's') ? '' : context; var out = compileParts(compiler, bodies.block); compiler.auto = old; return out; } } function visit(context, node) { var out = [node[0]]; for (var i=1, len=node.length; i<len; i++) { var res = dust.filterNode(context, node[i]); if (res) out.push(res); } return out; } // Compacts consecutive buffer nodes into a single node function compactBuffers(context, node) { var out = [node[0]], memo; for (var i=1, len=node.length; i<len; i++) { var res = dust.filterNode(context, node[i]); if (res) { if (res[0] === 'buffer') { if (memo) { memo[1] += res[1]; } else { memo = res; out.push(res); } } else { memo = null; out.push(res); } } } return out; } var specialChars = { "s": " ", "n": "\n", "r": "\r", "lb": "{", "rb": "}" }; function convertSpecial(context, node) { return ['buffer', specialChars[node[1]]] } function noop(context, node) { return node } function nullify(){} function compile(ast, name) { var context = { name: name, bodies: [], blocks: {}, index: 0, auto: "h" } return "(function(){dust.register(" + (name ? "\"" + name + "\"" : "null") + "," + dust.compileNode(context, ast) + ");" + compileBlocks(context) + compileBodies(context) + "return body_0;" + "})();"; } function compileBlocks(context) { var out = [], blocks = context.blocks; for (var name in blocks) { out.push("'" + name + "':" + blocks[name]); } if (out.length) { context.blocks = "ctx=ctx.shiftBlocks(blocks);"; return "var blocks={" + out.join(',') + "};"; } return context.blocks = ""; } function compileBodies(context) { var out = [], bodies = context.bodies, blx = context.blocks; for (var i=0, len=bodies.length; i<len; i++) { out[i] = "function body_" + i + "(chk,ctx){" + blx + "return chk" + bodies[i] + ";}"; } return out.join(''); } function compileParts(context, body) { var parts = ''; for (var i=1, len=body.length; i<len; i++) { parts += dust.compileNode(context, body[i]); } return parts; } dust.compileNode = function(context, node) { return dust.nodes[node[0]](context, node); } dust.nodes = { body: function(context, node) { var id = context.index++, name = "body_" + id; context.bodies[id] = compileParts(context, node); return name; }, buffer: function(context, node) { return ".write(" + escape(node[1]) + ")"; }, format: function(context, node) { return ".write(" + escape(node[1] + node[2]) + ")"; }, reference: function(context, node) { return ".reference(" + dust.compileNode(context, node[1]) + ",ctx," + dust.compileNode(context, node[2]) + ")"; }, "#": function(context, node) { return compileSection(context, node, "section"); }, "?": function(context, node) { return compileSection(context, node, "exists"); }, "^": function(context, node) { return compileSection(context, node, "notexists"); }, "<": function(context, node) { var bodies = node[4]; for (var i=1, len=bodies.length; i<len; i++) { var param = bodies[i], type = param[1][1]; if (type === "block") { context.blocks[node[1].text] = dust.compileNode(context, param[2]); return ''; } } return ''; }, "+": function(context, node) { if(typeof(node[1].text) === "undefined" && typeof(node[4]) === "undefined"){ return ".block(ctx.getBlock(" + dust.compileNode(context, node[1]) + ",chk, ctx)," + dust.compileNode(context, node[2]) + ", {}," + dust.compileNode(context, node[3]) + ")"; }else { return ".block(ctx.getBlock(" + escape(node[1].text) + ")," + dust.compileNode(context, node[2]) + "," + dust.compileNode(context, node[4]) + "," + dust.compileNode(context, node[3]) + ")"; } }, "@": function(context, node) { return ".helper(" + escape(node[1].text) + "," + dust.compileNode(context, node[2]) + "," + dust.compileNode(context, node[4]) + "," + dust.compileNode(context, node[3]) + ")"; }, "%": function(context, node) { // TODO: Move these hacks into pragma precompiler var name = node[1][1]; if (!dust.pragmas[name]) return ''; var rawBodies = node[4]; var bodies = {}; for (var i=1, len=rawBodies.length; i<len; i++) { var b = rawBodies[i]; bodies[b[1][1]] = b[2]; } var rawParams = node[3]; var params = {}; for (var i=1, len=rawParams.length; i<len; i++) { var p = rawParams[i]; params[p[1][1]] = p[2][1]; } var ctx = node[2][1] ? node[2][1].text : null; return dust.pragmas[name](context, ctx, bodies, params); }, partial: function(context, node) { return ".partial(" + dust.compileNode(context, node[1]) + "," + dust.compileNode(context, node[2]) + "," + dust.compileNode(context, node[3]) + ")"; }, context: function(context, node) { if (node[1]) { return "ctx.rebase(" + dust.compileNode(context, node[1]) + ")"; } return "ctx"; }, params: function(context, node) { var out = []; for (var i=1, len=node.length; i<len; i++) { out.push(dust.compileNode(context, node[i])); } if (out.length) { return "{" + out.join(',') + "}"; } return "null"; }, bodies: function(context, node) { var out = []; for (var i=1, len=node.length; i<len; i++) { out.push(dust.compileNode(context, node[i])); } return "{" + out.join(',') + "}"; }, param: function(context, node) { return dust.compileNode(context, node[1]) + ":" + dust.compileNode(context, node[2]); }, filters: function(context, node) { var list = []; for (var i=1, len=node.length; i<len; i++) { var filter = node[i]; list.push("\"" + filter + "\""); } return "\"" + context.auto + "\"" + (list.length ? ",[" + list.join(',') + "]" : ''); }, key: function(context, node) { return "ctx.get(\"" + node[1] + "\")"; }, path: function(context, node) { var current = node[1], keys = node[2], list = []; for (var i=0,len=keys.length; i<len; i++) { list.push("\"" + keys[i] + "\""); } return "ctx.getPath(" + current + ",[" + list.join(',') + "])"; }, literal: function(context, node) { return escape(node[1]); } } function compileSection(context, node, cmd) { return "." + cmd + "(" + dust.compileNode(context, node[1]) + "," + dust.compileNode(context, node[2]) + "," + dust.compileNode(context, node[4]) + "," + dust.compileNode(context, node[3]) + ")"; } var escape = (typeof JSON === "undefined") ? function(str) { return "\"" + dust.escapeJs(str) + "\"" } : JSON.stringify; })(typeof exports !== 'undefined' ? exports : getGlobal()); (function(dust){ var parser = (function(){ /* * Generated by PEG.js 0.7.0. * * http://pegjs.majda.cz/ */ function quote(s) { /* * ECMA-262, 5th ed., 7.8.4: All characters may appear literally in a * string literal except for the closing quote character, backslash, * carriage return, line separator, paragraph separator, and line feed. * Any character may appear in the form of an escape sequence. * * For portability, we also escape escape all control and non-ASCII * characters. Note that "\0" and "\v" escape sequences are not used * because JSHint does not like the first and IE the second. */ return '"' + s .replace(/\\/g, '\\\\') // backslash .replace(/"/g, '\\"') // closing quote character .replace(/\x08/g, '\\b') // backspace .replace(/\t/g, '\\t') // horizontal tab .replace(/\n/g, '\\n') // line feed .replace(/\f/g, '\\f') // form feed .replace(/\r/g, '\\r') // carriage return .replace(/[\x00-\x07\x0B\x0E-\x1F\x80-\uFFFF]/g, escape) + '"'; } var result = { /* * Parses the input with a generated parser. If the parsing is successfull, * returns a value explicitly or implicitly specified by the grammar from * which the parser was generated (see |PEG.buildParser|). If the parsing is * unsuccessful, throws |PEG.parser.SyntaxError| describing the error. */ parse: function(input, startRule) { var parseFunctions = { "body": parse_body, "part": parse_part, "section": parse_section, "sec_tag_start": parse_sec_tag_start, "end_tag": parse_end_tag, "context": parse_context, "params": parse_params, "bodies": parse_bodies, "reference": parse_reference, "partial": parse_partial, "filters": parse_filters, "special": parse_special, "identifier": parse_identifier, "number": parse_number, "frac": parse_frac, "integer": parse_integer, "path": parse_path, "key": parse_key, "array": parse_array, "array_part": parse_array_part, "inline": parse_inline, "inline_part": parse_inline_part, "buffer": parse_buffer, "literal": parse_literal, "esc": parse_esc, "comment": parse_comment, "tag": parse_tag, "ld": parse_ld, "rd": parse_rd, "eol": parse_eol, "ws": parse_ws }; if (startRule !== undefined) { if (parseFunctions[startRule] === undefined) { throw new Error("Invalid rule name: " + quote(startRule) + "."); } } else { startRule = "body"; } var pos = { offset: 0, line: 1, column: 1, seenCR: false }; var reportFailures = 0; var rightmostFailuresPos = { offset: 0, line: 1, column: 1, seenCR: false }; var rightmostFailuresExpected = []; function padLeft(input, padding, length) { var result = input; var padLength = length - input.length; for (var i = 0; i < padLength; i++) { result = padding + result; } return result; } function escape(ch) { var charCode = ch.charCodeAt(0); var escapeChar; var length; if (charCode <= 0xFF) { escapeChar = 'x'; length = 2; } else { escapeChar = 'u'; length = 4; } return '\\' + escapeChar + padLeft(charCode.toString(16).toUpperCase(), '0', length); } function clone(object) { var result = {}; for (var key in object) { result[key] = object[key]; } return result; } function advance(pos, n) { var endOffset = pos.offset + n; for (var offset = pos.offset; offset < endOffset; offset++) { var ch = input.charAt(offset); if (ch === "\n") { if (!pos.seenCR) { pos.line++; } pos.column = 1; pos.seenCR = false; } else if (ch === "\r" || ch === "\u2028" || ch === "\u2029") { pos.line++; pos.column = 1; pos.seenCR = true; } else { pos.column++; pos.seenCR = false; } } pos.offset += n; } function matchFailed(failure) { if (pos.offset < rightmostFailuresPos.offset) { return; } if (pos.offset > rightmostFailuresPos.offset) { rightmostFailuresPos = clone(pos); rightmostFailuresExpected = []; } rightmostFailuresExpected.push(failure); } function parse_body() { var result0, result1; var pos0; pos0 = clone(pos); result0 = []; result1 = parse_part(); while (result1 !== null) { result0.push(result1); result1 = parse_part(); } if (result0 !== null) { result0 = (function(offset, line, column, p) { return ["body"].concat(p) })(pos0.offset, pos0.line, pos0.column, result0); } if (result0 === null) { pos = clone(pos0); } return result0; } function parse_part() { var result0; result0 = parse_comment(); if (result0 === null) { result0 = parse_section(); if (result0 === null) { result0 = parse_partial(); if (result0 === null) { result0 = parse_special(); if (result0 === null) { result0 = parse_reference(); if (result0 === null) { result0 = parse_buffer(); } } } } } return result0; } function parse_section() { var result0, result1, result2, result3, result4, result5, result6; var pos0, pos1; reportFailures++; pos0 = clone(pos); pos1 = clone(pos); result0 = parse_sec_tag_start(); if (result0 !== null) { result1 = []; result2 = parse_ws(); while (result2 !== null) { result1.push(result2); result2 = parse_ws(); } if (result1 !== null) { result2 = parse_rd(); if (result2 !== null) { result3 = parse_body(); if (result3 !== null) { result4 = parse_bodies(); if (result4 !== null) { result5 = parse_end_tag(); if (result5 !== null) { result6 = (function(offset, line, column, t, b, e, n) { return t[1].text === n.text;})(pos.offset, pos.line, pos.column, result0, result3, result4, result5) ? "" : null; if (result6 !== null) { result0 = [result0, result1, result2, result3, result4, result5, result6]; } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } if (result0 !== null) { result0 = (function(offset, line, column, t, b, e, n) { e.push(["param", ["literal", "block"], b]); t.push(e); return t })(pos0.offset, pos0.line, pos0.column, result0[0], result0[3], result0[4], result0[5]); } if (result0 === null) { pos = clone(pos0); } if (result0 === null) { pos0 = clone(pos); pos1 = clone(pos); result0 = parse_sec_tag_start(); if (result0 !== null) { result1 = []; result2 = parse_ws(); while (result2 !== null) { result1.push(result2); result2 = parse_ws(); } if (result1 !== null) { if (input.charCodeAt(pos.offset) === 47) { result2 = "/"; advance(pos, 1); } else { result2 = null; if (reportFailures === 0) { matchFailed("\"/\""); } } if (result2 !== null) { result3 = parse_rd(); if (result3 !== null) { result0 = [result0, result1, result2, result3]; } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } if (result0 !== null) { result0 = (function(offset, line, column, t) { t.push(["bodies"]); return t })(pos0.offset, pos0.line, pos0.column, result0[0]); } if (result0 === null) { pos = clone(pos0); } } reportFailures--; if (reportFailures === 0 && result0 === null) { matchFailed("section"); } return result0; } function parse_sec_tag_start() { var result0, result1, result2, result3, result4, result5; var pos0, pos1; pos0 = clone(pos); pos1 = clone(pos); result0 = parse_ld(); if (result0 !== null) { if (/^[#?^<+@%]/.test(input.charAt(pos.offset))) { result1 = input.charAt(pos.offset); advance(pos, 1); } else { result1 = null; if (reportFailures === 0) { matchFailed("[#?^<+@%]"); } } if (result1 !== null) { result2 = []; result3 = parse_ws(); while (result3 !== null) { result2.push(result3); result3 = parse_ws(); } if (result2 !== null) { result3 = parse_identifier(); if (result3 !== null) { result4 = parse_context(); if (result4 !== null) { result5 = parse_params(); if (result5 !== null) { result0 = [result0, result1, result2, result3, result4, result5]; } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } if (result0 !== null) { result0 = (function(offset, line, column, t, n, c, p) { return [t, n, c, p] })(pos0.offset, pos0.line, pos0.column, result0[1], result0[3], result0[4], result0[5]); } if (result0 === null) { pos = clone(pos0); } return result0; } function parse_end_tag() { var result0, result1, result2, result3, result4, result5; var pos0, pos1; reportFailures++; pos0 = clone(pos); pos1 = clone(pos); result0 = parse_ld(); if (result0 !== null) { if (input.charCodeAt(pos.offset) === 47) { result1 = "/"; advance(pos, 1); } else { result1 = null; if (reportFailures === 0) { matchFailed("\"/\""); } } if (result1 !== null) { result2 = []; result3 = parse_ws(); while (result3 !== null) { result2.push(result3); result3 = parse_ws(); } if (result2 !== null) { result3 = parse_identifier(); if (result3 !== null) { result4 = []; result5 = parse_ws(); while (result5 !== null) { result4.push(result5); result5 = parse_ws(); } if (result4 !== null) { result5 = parse_rd(); if (result5 !== null) { result0 = [result0, result1, result2, result3, result4, result5]; } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } } else { result0 = null; pos = clone(pos1); } if (result0 !== null) { result0 = (function(offset, line, column, n) { return n })(pos0.offset, pos0.line, pos0.column, result0[3]); } if (result0 === null) { pos = clone(pos0); } reportFailures--; if (reportFailures === 0 && result0 === null) { matchFailed("end tag"); } return result0; } function parse_context() { var result0, result1; var pos0, pos1, pos2; pos0 = clone(pos); pos1 = clone(pos); pos2 = clone(pos); if (input.charCodeAt(pos.offset) === 58) { result0 = ":"; advance(pos, 1); } else { result0 = null; if (reportFailures === 0) { matchFailed("\":\""); } } if (result0 !== null) { result1 = parse_identifier(); if (result1 !== null) { result0 = [result0, result1]; } else { result0 = null; pos = clone(pos2); } } else { result0 = null; pos = clone(pos2); } if (result0 !== null) { result0 = (function(offset, line, column, n) {return n})(pos1.offset, pos1.line, pos1.column, result0[1]); } if (result0 === null) { pos = clone(pos1); } result0 = result0 !== null ? result0 : ""; if (result0 !== null) { result0 = (function(offset, line, column, n) { return n ? ["context", n] : ["context"] })(pos0.offset, pos0.line, pos0.column, result0); } if (result0 === null) { pos = clone(pos0); } return result0; } function parse_params() { var result0, result1, result2, result3, result4; var pos0, pos1, pos2; reportFailures++; pos0 = clone(pos); result0 = []; pos1 = clone(pos); pos2 = clone(pos); result2 = parse_ws(); if (result2 !== null) { result1 = []; while (result2 !== null) { result1.push(result2); result2 = parse_ws(); } } else { result1 = null; } if (result1 !== null) { result2 = parse_key(); if (result2 !== null) { if (input.charCodeAt(pos.offset) === 61) { result3 = "="; advance(pos, 1); } else { result3 = null; if (reportFailures === 0) { matchFailed("\"=\""); } } if (result3 !== null) { result4 = parse_number(); if (result4 === null) { result4 = parse_identifier(); if (result4 === null) { result4 = parse_inline(); } } if (result4 !== null) { result1 = [result1, result2, result3, result4]; } else { result1 = null; pos = clone(pos2); } } else { result1 = null; pos = clone(pos2); } } else { result1 = null; pos = clone(pos2); } } else { result1 = null; pos = clone(pos2); } if (result1 !== null) { result1 = (function(offset, line, column, k, v) {return ["param", ["literal", k], v]})(pos1.offset, pos1.line, pos1.column, result1[1], result1[3]); } if (result1 === null) { pos = clone(pos1); } while (result1 !== null) { result0.push(result1); pos1 = clone(pos); pos2 = clone(pos); result2 = parse_ws(); if (result2 !== null) { result1 = []; while (result2 !== null) { result1.push(result2); result2 = parse_ws(); } } else { result1 = null; } if (result1 !== null) { result2 = parse_key(); if (result2 !== null) { if (input.charCodeAt(pos.offset) === 61) { result3 = "="; advance(pos, 1); } else { result3 = null; if (reportFailures === 0) { matchFailed("\"=\""); } } if (result3 !== null) { result4 = parse_number(); if (result4 === null) { result4 = parse_identifier(); if (result4 === null) { result4 = parse_inline(); } } if (result4 !== null) { result1 = [result1, result2, result3, result4]; } else { result1 = null; pos = clone(pos2); } } else { result1 = null; pos = clone(pos2); } } else { result1 = null; pos = clone(pos2); } } else { result1 = null; pos = clone(pos2); } if (result1 !== null) { result1 = (function(offset, line, column, k, v) {return ["param", ["literal", k], v]})(pos1.offset, pos1.line, pos1.column, result1[1], result1[3]); } if (result1 === null) { pos = clone(pos1); } } if (result0 !== null) { result0 = (function(offset, line, column, p) { return ["params"].concat(p) })(pos0.offset, pos0.line, pos0.column, result0); } if (result0 === null) { pos = clone(pos0); } reportFailures--; if (reportFailures === 0 && result0 === null) { matchFailed("params"); } return result0; } function parse_bodies() { var result0, result1, result2, result3, result4, result5; var pos0, pos1, pos2; reportFailures++; pos0 = clone(pos); result0 = []; pos1 = clone(pos); pos2 = clone(pos); result1 = parse_ld(); if (result1 !== null) { if (input.charCodeAt(pos.offset) === 58) { result2 = ":"; advance(pos, 1); } else { result2 = null; if (reportFailures === 0) { matchFailed("\":\""); } } if (result2 !== null) { result3 = parse_key(); if (result3 !== null) { result4 = parse_rd(); if (result4 !== null) { result5 = parse_body(); if (result5 !== null) { result1 = [result1, result2, result3, result4, result5]; } else { result1 = null; pos = clone(pos2); } } else { result1 = null; pos = clone(pos2); } } else { result1 = null; pos = clone(pos2); } } else { result1 = null; pos = clone(pos2); } } else { result1 = null; pos = clone(pos2); } if (result1 !== null) { result1 = (function(offset, line, column, k, v) {return ["param", ["literal", k], v]})(pos1.offset, pos1.line, pos1.column, result1[2], result1[4]); } if (result1 === null) { pos = clone(pos1); } while (result1 !== null) { result0.push(result1); pos1 = clone(pos); pos2 = clone(pos); result1 = parse_ld(); if (result1 !== null) { if (input.charCodeAt(pos.offset) === 58) { result2 = ":"; advance(pos, 1); } else { result2 = null; if (reportFailures === 0) { matchFailed("\":\""); } } if (result2 !== null) { result3 = parse_key(); if (result3 !== null) { result4 = parse_rd(); if (result4 !== null) { result5 = parse_body(); if (result5 !== null) { result1 = [result1, result2, result3, result4, result5]; } else { result1 = null; pos = clone(pos2); } } else { result1 = null; pos = clone(pos2); } } else { result1 = null; pos = clone(pos2); } } else { result1 = null;