mirror of
https://github.com/flynx/pWiki.git
synced 2025-10-29 10:00:08 +00:00
1567 lines
45 KiB
JavaScript
Executable File
1567 lines
45 KiB
JavaScript
Executable File
/**********************************************************************
|
|
*
|
|
*
|
|
*
|
|
**********************************************************************/
|
|
((typeof define)[0]=='u'?function(f){module.exports=f(require)}:define)
|
|
(function(require){ var module={} // make module AMD/node compatible...
|
|
/*********************************************************************/
|
|
|
|
var object = require('ig-object')
|
|
var types = require('ig-types')
|
|
|
|
var pwpath = require('./path')
|
|
|
|
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------
|
|
// Parser...
|
|
|
|
// XXX ASAP move the macros here...
|
|
// XXX should we warn about stuff like <macro src=/moo/> -- currently
|
|
// this will simply be ignored, i.e. passed trough the parser
|
|
// without change...
|
|
// XXX might be a good idea to both think of a good async parse and
|
|
// create tools for sync parsing (get links etc.)...
|
|
// XXX need to correctly handle nested and escaped quotes...
|
|
// i.e.
|
|
// "aaa \"bbb \\"ccc\\" bbb\" aaa"
|
|
|
|
var BaseParser =
|
|
module.BaseParser = {
|
|
// patterns...
|
|
//
|
|
// The way the patterns are organized might seem a bit overcomplicated
|
|
// and it has to be to be able to reuse the same pattern in different
|
|
// contexts, e.g. the arguments pattern...
|
|
|
|
//
|
|
// needs:
|
|
// STOP -- '\\>' or ')'
|
|
// PREFIX -- 'inline' or 'elem'
|
|
//
|
|
MACRO_ARGS: ['(\\s*(',[
|
|
// arg='val' | arg="val" | arg=val
|
|
'(?<PREFIXArgName>[a-z:-_]+)\\s*=\\s*(?<PREFIXArgValue>'+([
|
|
// XXX CHROME/NODE BUG: this does not work yet...
|
|
//'\\s+(?<quote>[\'"])[^\\k<quote>]*\\k<quote>',
|
|
'"(?<PREFIXDoubleQuotedValue>(\\"|[^"])*?)"',
|
|
"'(?<PREFIXSingleQuotedValue>(\\'|[^'])*?)'",
|
|
'(?<PREFIXValue>[^\\sSTOP\'"]+)',
|
|
].join('|'))+')',
|
|
// "arg" | 'arg'
|
|
// XXX CHROME/NODE BUG: this does not work yet...
|
|
//'\\s+(?<quote>[\'"])[^\\k<quote>]*\\k<quote>',
|
|
'"(?<PREFIXDoubleQuotedArg>(\\"|[^"])*?)"',
|
|
"'(?<PREFIXSingleQuotedArg>(\\'|[^'])*?)'",
|
|
// arg
|
|
// NOTE: this is last because it could eat up parts of
|
|
// the above alternatives...
|
|
//'|\\s+[^\\s\\/>\'"]+',
|
|
'(?<PREFIXArg>[^\\sSTOP\'"]+)',
|
|
].join('|'),
|
|
'))'].join(''),
|
|
MACRO_ARGS_PATTERN: undefined,
|
|
//
|
|
// .buildArgsPattern(<prefix>[, <stop>[, <flags>]])
|
|
// -> <pattern>
|
|
//
|
|
// .buildArgsPattern(<prefix>[, <stop>[, false]])
|
|
// -> <string>
|
|
//
|
|
buildArgsPattern: function(prefix='elem', stop='', regexp='smig'){
|
|
var pattern = this.MACRO_ARGS
|
|
.replace(/PREFIX/g, prefix)
|
|
.replace(/STOP/g, stop)
|
|
return regexp ?
|
|
new RegExp(pattern, regexp)
|
|
: pattern },
|
|
|
|
//
|
|
// needs:
|
|
// MACROS
|
|
// INLINE_ARGS
|
|
// UNNAMED_ARGS
|
|
// ARGS
|
|
//
|
|
MACRO: '('+([
|
|
// @macro(arg ..)
|
|
'\\\\?@(?<nameInline>MACROS)\\((?<argsInline>INLINE_ARGS)\\)',
|
|
// @(arg ..)
|
|
'\\\\?@\\((?<argsUnnamed>UNNAMED_ARGS)\\)',
|
|
// <macro ..> | <macro ../>
|
|
'<\\s*(?<nameOpen>MACROS)(?<argsOpen>\\sARGS)?\\s*/?>',
|
|
// </macro>
|
|
'</\\s*(?<nameClose>MACROS)\\s*>',
|
|
].join('|'))+')',
|
|
MACRO_PATTERN: undefined,
|
|
MACRO_PATTERN_GROUPS: undefined,
|
|
//
|
|
// .buildMacroPattern(<macros>[, <flags>])
|
|
// -> <pattern>
|
|
//
|
|
// .buildMacroPattern(<macros>[, false])
|
|
// -> <string>
|
|
//
|
|
buildMacroPattern: function(macros=['MACROS'], regexp='smig'){
|
|
var pattern = this.MACRO
|
|
.replace(/MACROS/g,
|
|
macros
|
|
.filter(function(m){
|
|
return m.length > 0 })
|
|
.join('|'))
|
|
.replace(/INLINE_ARGS/g,
|
|
this.buildArgsPattern('inline', ')', false) +'*')
|
|
.replace(/UNNAMED_ARGS/g,
|
|
this.buildArgsPattern('unnamed', ')', false) +'*')
|
|
.replace(/ARGS/g,
|
|
this.buildArgsPattern('elem', '\\/>', false) +'*')
|
|
return regexp ?
|
|
new RegExp(pattern, regexp)
|
|
: pattern },
|
|
countMacroPatternGroups: function(){
|
|
// NOTE: the -2 here is to compensate for the leading and trailing ""'s...
|
|
return '<MACROS>'.split(this.buildMacroPattern()).length - 2 },
|
|
|
|
// XXX should this be closer to .stripComments(..)
|
|
// XXX do we need basic inline and block commets a-la lisp???
|
|
COMMENT_PATTERN: RegExp('('+[
|
|
// <!--[pwiki[ .. ]]-->
|
|
'<!--\\[pwiki\\[(?<uncomment>.*?)\\]\\]-->',
|
|
|
|
// <pwiki-comment> .. </pwiki-comment>
|
|
'<\\s*pwiki-comment[^>]*>.*?<\\/\\s*pwiki-comment\\s*>',
|
|
// <pwiki-comment .. />
|
|
'<\\s*pwiki-comment[^\\/>]*\\/>',
|
|
|
|
// html comments...
|
|
'<!--.*?-->',
|
|
].join('|') +')', 'smig'),
|
|
|
|
|
|
// helpers...
|
|
//
|
|
normalizeFilters: function(filters){
|
|
var skip = new Set()
|
|
return filters
|
|
.flat()
|
|
.tailUnique()
|
|
.filter(function(filter){
|
|
filter[0] == '-'
|
|
&& skip.add(filter.slice(1))
|
|
return filter[0] != '-' })
|
|
.filter(function(filter){
|
|
return !skip.has(filter) })},
|
|
//
|
|
// Spec format:
|
|
// [<orderd>, ... [<keyword>, ...]]
|
|
//
|
|
// Keyword arguments if given without a value are true by default,
|
|
// explicitly setting a keyword argument to 'true' or 'yes' will set
|
|
// it to true, explicitly setting to 'false' or 'no' will set it to
|
|
// false, any other value will be set as-is...
|
|
//
|
|
// NOTE: the input to this is formatted by .lex(..)
|
|
// NOTE: arg pre-parsing is dome by .lex(..) but at that stage we do not
|
|
// yet touch the actual macros (we need them to get the .arg_spec)
|
|
// so the actual parsing is done in .expand(..)
|
|
parseArgs: function(spec, args){
|
|
// spec...
|
|
var order = spec.slice()
|
|
var bools = new Set(
|
|
order[order.length-1] instanceof Array ?
|
|
order.pop()
|
|
: [])
|
|
order = order
|
|
.filter(function(k){
|
|
return !(k in args) })
|
|
|
|
var res = {}
|
|
var pos = Object.entries(args)
|
|
// stage 1: populate res with explicit data and place the rest in pos...
|
|
.reduce(function(pos, [key, value]){
|
|
;/^[0-9]+$/.test(key) ?
|
|
(bools.has(value) ?
|
|
// bool...
|
|
(res[value] = true)
|
|
// positional...
|
|
: (pos[key*1] = value))
|
|
// keyword/bool default values...
|
|
: bools.has(key) ?
|
|
(res[key] =
|
|
// value escaping...
|
|
value[0] == '\\' ?
|
|
value.slice(1)
|
|
: (value == 'true' || value == 'yes') ?
|
|
true
|
|
: (value == 'false' || value == 'no') ?
|
|
false
|
|
: value)
|
|
// keyword...
|
|
: (res[key] = value)
|
|
return pos }, [])
|
|
// stage 2: populate implicit values from pos...
|
|
.forEach(function(e, i){
|
|
order.length == 0 ?
|
|
(res[e] = true)
|
|
: (res[order.shift()] = e) })
|
|
return res },
|
|
// XXX should this be here or on page???
|
|
callMacro: function(page, name, args, body, state, ...rest){
|
|
var macro = this.macros[name]
|
|
return macro.call(page,
|
|
this.parseArgs(
|
|
macro.arg_spec
|
|
?? [],
|
|
args),
|
|
body,
|
|
state,
|
|
...rest) },
|
|
|
|
|
|
// Strip comments...
|
|
//
|
|
stripComments: function(str){
|
|
return str
|
|
.replace(this.COMMENT_PATTERN,
|
|
function(...a){
|
|
return a.pop().uncomment
|
|
|| '' }) },
|
|
|
|
// Lexically split the string (generator)...
|
|
//
|
|
// <item> ::=
|
|
// <string>
|
|
// | {
|
|
// name: <string>,
|
|
// type: 'inline'
|
|
// | 'element'
|
|
// | 'opening'
|
|
// | 'closing',
|
|
// args: {
|
|
// <index>: <value>,
|
|
// <key>: <value>,
|
|
// ...
|
|
// }
|
|
// match: <string>,
|
|
// }
|
|
//
|
|
//
|
|
// NOTE: this internally uses .macros' keys to generate the
|
|
// lexing pattern.
|
|
lex: function*(page, str){
|
|
str = typeof(str) != 'string' ?
|
|
str+''
|
|
: str
|
|
// XXX we can't get .raw from the page without going async...
|
|
//str = str
|
|
// ?? page.raw
|
|
// NOTE: we are doing a separate pass for comments to completely
|
|
// decouple them from the base macro syntax, making them fully
|
|
// transparent...
|
|
str = this.stripComments(str)
|
|
|
|
// XXX should this be cached???
|
|
var macro_pattern = this.MACRO_PATTERN
|
|
?? this.buildMacroPattern(Object.deepKeys(this.macros))
|
|
var macro_pattern_groups = this.MACRO_PATTERN_GROUPS
|
|
?? this.countMacroPatternGroups()
|
|
var macro_args_pattern = this.MACRO_ARGS_PATTERN
|
|
?? this.buildArgsPattern()
|
|
|
|
var lst = str.split(macro_pattern)
|
|
|
|
var macro = false
|
|
while(lst.length > 0){
|
|
if(macro){
|
|
var match = lst.splice(0, macro_pattern_groups)[0]
|
|
// NOTE: we essentially are parsing the detected macro a
|
|
// second time here, this gives us access to named groups
|
|
// avoiding maintaining match indexes with the .split(..)
|
|
// output...
|
|
// XXX for some reason .match(..) here returns a list with a string...
|
|
var cur = [...match.matchAll(macro_pattern)][0].groups
|
|
// special case: escaped inline macro -> keep as text...
|
|
if(match.startsWith('\\@')){
|
|
yield match
|
|
macro = false
|
|
continue }
|
|
// args...
|
|
var args = {}
|
|
var i = -1
|
|
for(var {groups}
|
|
of (cur.argsInline
|
|
?? cur.argsUnnamed
|
|
?? cur.argsOpen
|
|
?? '')
|
|
.matchAll(macro_args_pattern)){
|
|
i++
|
|
args[groups.elemArgName
|
|
?? groups.inlineArgName
|
|
?? groups.unnamedArgName
|
|
?? i] =
|
|
(groups.elemSingleQuotedValue
|
|
?? groups.inlineSingleQuotedValue
|
|
?? groups.unnamedSingleQuotedValue
|
|
?? groups.elemDoubleQuotedValue
|
|
?? groups.inlineDoubleQuotedValue
|
|
?? groups.unnamedDoubleQuotedValue
|
|
?? groups.elemValue
|
|
?? groups.inlineValue
|
|
?? groups.unnamedValue
|
|
?? groups.elemSingleQuotedArg
|
|
?? groups.inlineSingleQuotedArg
|
|
?? groups.unnamedSingleQuotedArg
|
|
?? groups.elemDoubleQuotedArg
|
|
?? groups.inlineDoubleQuotedArg
|
|
?? groups.unnamedDoubleQuotedArg
|
|
?? groups.elemArg
|
|
?? groups.inlineArg
|
|
?? groups.unnamedArg)
|
|
.replace(/\\(["'])/g, '$1') }
|
|
|
|
// macro-spec...
|
|
yield {
|
|
name: (cur.nameInline
|
|
?? cur.nameOpen
|
|
?? cur.nameClose
|
|
?? '')
|
|
.toLowerCase(),
|
|
type: match[0] == '@' ?
|
|
'inline'
|
|
: match[1] == '/' ?
|
|
'closing'
|
|
: match[match.length-2] == '/' ?
|
|
'element'
|
|
: 'opening',
|
|
args,
|
|
match,
|
|
}
|
|
macro = false
|
|
// normal text...
|
|
} else {
|
|
var str = lst.shift()
|
|
// skip empty strings from output...
|
|
if(str != ''){
|
|
yield str }
|
|
macro = true } } },
|
|
|
|
// NOTE: so as to avod cluterring the main parser flow the macros are
|
|
// defined separtly below...
|
|
macros: undefined,
|
|
|
|
// Group block elements (generator)...
|
|
//
|
|
// <item> ::=
|
|
// <string>
|
|
// | {
|
|
// type: 'inline'
|
|
// | 'element'
|
|
// | 'block',
|
|
// body: [
|
|
// <item>,
|
|
// ...
|
|
// ],
|
|
//
|
|
// // rest of items are the same as for lex(..)
|
|
// ...
|
|
// }
|
|
//
|
|
// NOTE: this internaly uses .macros to check for propper nesting
|
|
//group: function*(page, lex, to=false){
|
|
group: function*(page, lex, to=false, parent){
|
|
// XXX we can't get .raw from the page without going async...
|
|
//lex = lex
|
|
// ?? this.lex(page)
|
|
lex = typeof(lex) != 'object' ?
|
|
this.lex(page, lex)
|
|
: lex
|
|
|
|
var quoting = to
|
|
&& (page.QUOTING_MACROS ?? []).includes(to)
|
|
&& []
|
|
|
|
// NOTE: we are not using for .. of .. here as it depletes the
|
|
// generator even if the end is not reached...
|
|
while(true){
|
|
var {value, done} = lex.next()
|
|
// check if unclosed blocks remaining...
|
|
if(done){
|
|
if(to){
|
|
throw new Error(
|
|
'Premature end of input: Expected </'+ to +'>') }
|
|
return }
|
|
|
|
// special case: quoting -> collect text...
|
|
// NOTE: we do not care about nesting here...
|
|
if(quoting !== false){
|
|
if(value.name == to
|
|
&& value.type == 'closing'){
|
|
yield quoting.join('')
|
|
return
|
|
} else {
|
|
quoting.push(
|
|
typeof(value) == 'string' ?
|
|
value
|
|
: value.match ) }
|
|
continue }
|
|
|
|
// assert nesting rules...
|
|
// NOTE: we only check for direct nesting...
|
|
// XXX might be a good idea to link nested block to the parent...
|
|
if(this.macros[value.name] instanceof Array
|
|
&& !this.macros[value.name].includes(to)
|
|
// do not complain about closing nestable tags...
|
|
&& !(value.name == to
|
|
&& value.type == 'closing')){
|
|
throw new Error(
|
|
'Unexpected <'+ value.name +'> macro'
|
|
+(to ?
|
|
' in <'+to+'>'
|
|
: '')) }
|
|
// open block...
|
|
if(value.type == 'opening'){
|
|
//value.body = [...this.group(page, lex, value.name)]
|
|
value.body = [...this.group(page, lex, value.name, value)]
|
|
value.type = 'block'
|
|
// close block...
|
|
} else if(value.type == 'closing'){
|
|
if(value.name != to){
|
|
throw new Error('Unexpected </'+ value.name +'>') }
|
|
// NOTE: we are intentionally not yielding the value here...
|
|
return }
|
|
// normal value...
|
|
yield value } },
|
|
|
|
// Expand macros...
|
|
//
|
|
// <ast> ::= [ <item>, .. ]
|
|
// <item> ::=
|
|
// <func>
|
|
// | <promise>
|
|
// | <string>
|
|
// | { skip: true, ... }
|
|
// | { data: <ast> }
|
|
// | <ast>
|
|
//
|
|
// XXX macros: we are mixing up ast state and parse state...
|
|
// one should only be used for parsing and be forgotten after
|
|
// the ast is constructed the other should be part of the ast...
|
|
// XXX DOC inconsistent await behavior...
|
|
// in var macro, as an example, there are two paths: the assign and
|
|
// the get, the assign calls .parse(..) and awaits for the result
|
|
// while the get path is "sync", this results in the first exec path
|
|
// getting pushed to the next execution frame while the second is run
|
|
// in sync with the caller, here is a simplified demo:
|
|
// console.log(1)
|
|
// // note that we are NOTE await'ing for the function here...
|
|
// (async function f(){
|
|
// console.log(2)})()
|
|
// console.log(3)
|
|
// -> prints 1, 2, 3
|
|
// and:
|
|
// console.log(1)
|
|
// (async function f(){
|
|
// // note the await -- this is the only difference...
|
|
// console.log(await 2)})()
|
|
// console.log(3)
|
|
// -> prints 1, 3, 2
|
|
// this could both be a bug or a feature depending on how you look
|
|
// at it, but it makes promise sequencing very unpredictable...
|
|
// ...another problem here is that in the var macro, simply adding
|
|
// an await of something does not fix the issue, we need to await
|
|
// for something significant -- await this.parse('') works, while
|
|
// await vars[name] does not -- to the contrary of the above example...
|
|
expand: function(page, ast, state={}){
|
|
var that = this
|
|
ast = ast == null ?
|
|
Promise.awaitOrRun(
|
|
page.raw,
|
|
function(raw){
|
|
return that.group(page, raw ?? '') })
|
|
: typeof(ast) != 'object' ?
|
|
this.group(page, ast)
|
|
: ast instanceof types.Generator ?
|
|
ast
|
|
: ast.iter()
|
|
|
|
// NOTE this must execute sequentially for things that depend on
|
|
// lexical scope not to get lost in the mess...
|
|
return Promise.seqiter(ast,
|
|
function(value){
|
|
// text block...
|
|
if(typeof(value) == 'string'){
|
|
return value }
|
|
// macro...
|
|
var {name, args, body} = value
|
|
// nested macro -- skip...
|
|
if(typeof(that.macros[name]) != 'function'){
|
|
return {...value, skip: true} }
|
|
// macro call...
|
|
return Promise.awaitOrRun(
|
|
that.callMacro(page, name, args, body, state),
|
|
function(res){
|
|
res = res ?? ''
|
|
// result...
|
|
if(res instanceof Array
|
|
|| that.macros[name] instanceof types.Generator){
|
|
return res
|
|
} else {
|
|
return [res] } }) },
|
|
function(err){
|
|
console.error(err)
|
|
return [page.parse(
|
|
// XXX add line number and page path...
|
|
'@include("./ParseError'
|
|
+':path='
|
|
// XXX use pwpath.encodeElem(..) ???
|
|
+ page.path
|
|
+':msg='
|
|
+ err.message
|
|
// quote html stuff...
|
|
.replace(/&/g, '&')
|
|
.replace(/</g, '<')
|
|
.replace(/>/g, '>')
|
|
// quote argument syntax...
|
|
.replace(/["']/g, function(c){
|
|
return '%'+ c.charCodeAt().toString(16) })
|
|
.replace(/:/g, ':')
|
|
.replace(/=/g, '=')
|
|
+'")')] })
|
|
.sync() },
|
|
|
|
// recursively resolve and enumerate the ast...
|
|
//
|
|
// <ast> ::= [ <item>, .. ]
|
|
// <item> ::=
|
|
// <string>
|
|
// | { data: <ast> }
|
|
// | <func>
|
|
//
|
|
// <func>(state)
|
|
// -> <ast>
|
|
//
|
|
//
|
|
// NOTE: <func>(..) is called in the context of page...
|
|
//
|
|
// XXX should this also resolve e.data???
|
|
resolve: function(page, ast, state={}){
|
|
var that = this
|
|
ast = ast
|
|
?? this.expand(page, null, state)
|
|
ast = typeof(ast) != 'object' ?
|
|
this.expand(page, ast, state)
|
|
: ast
|
|
// XXX .awaitOrRun(..) will check inside the input array for promises, do
|
|
// we need to do this???
|
|
ast = Promise.awaitOrRun(
|
|
ast,
|
|
function(ast){
|
|
var async_content = false
|
|
return ast
|
|
.map(function(e){
|
|
// expand delayed sections...
|
|
e = typeof(e) == 'function' ?
|
|
e.call(page, state)
|
|
: e
|
|
// promise...
|
|
if(e instanceof Promise){
|
|
async_content = true
|
|
return e
|
|
// expand arrays...
|
|
} else if(e instanceof Array
|
|
|| e instanceof types.Generator){
|
|
return that.resolve(page, e, state)
|
|
// data -- unwrap content...
|
|
} else if(e instanceof Object && 'data' in e){
|
|
var res = Promise.awaitOrRun(
|
|
that.resolve(page, e.data, state),
|
|
function(e){
|
|
return { data: e } })
|
|
res instanceof Promise
|
|
&& (async_content = true)
|
|
return res
|
|
// skipped items...
|
|
} else if(e instanceof Object && e.skip){
|
|
return []
|
|
} else {
|
|
return [e] } })
|
|
// NOTE: if we still have promises in the ast, wrap the
|
|
// whole thing in a promise...
|
|
.run(function(){
|
|
return async_content ?
|
|
Promise.iter(this)
|
|
: this })
|
|
.flat() })
|
|
return ast instanceof Promise ?
|
|
// keep the API consistently array-like...
|
|
ast.iter()
|
|
: ast },
|
|
|
|
// Fully parse a page...
|
|
//
|
|
// This runs in two stages:
|
|
// - resolve the page
|
|
// - lex the page -- .lex(..)
|
|
// - group block elements -- .group(..)
|
|
// - expand macros -- .expand(..)
|
|
// - resolve ast -- .resolve(..)
|
|
// - apply filters
|
|
//
|
|
// NOTE: this has to synchronize everything between stage 1 (up to
|
|
// and including expand) and stage 2 (post-handlers, filters, ...)
|
|
// because the former need a fully loaded and expanded page if
|
|
// we want to do this in 2 stages and not 3...
|
|
// XXX might be fun to try a load-and-tweak approach the first
|
|
// version used -- i.e. setting placeholders and replacing
|
|
// them on demand rather than on encounter (as is now), e.g.
|
|
// a slot when loaded will replace the prior occurrences...
|
|
//
|
|
// XXX add a special filter to clear pending filters... (???)
|
|
parse: function(page, ast, state={}){
|
|
var that = this
|
|
return this.resolve(page, ast, state)
|
|
// filters...
|
|
.map(function(section){
|
|
// normalize types...
|
|
section =
|
|
typeof(section) == 'number' ?
|
|
section + ''
|
|
: section == null ?
|
|
''
|
|
: section
|
|
return (
|
|
// expand section...
|
|
typeof(section) != 'string' ?
|
|
section.data
|
|
// global filters...
|
|
: state.filters ?
|
|
that.normalizeFilters(state.filters)
|
|
.reduce(function(res, filter){
|
|
// unknown filter...
|
|
// NOTE: we try not to break on user errors
|
|
// if we can help it...
|
|
if(page.filters[filter] == null){
|
|
console.warn(
|
|
'.parse(..): unsupported filter: '+ filter)
|
|
return res }
|
|
// NOTE: if a filter returns falsy then it
|
|
// will have no effect on the result...
|
|
return page.filters[filter].call(page, res)
|
|
?? res }, section)
|
|
// no global filters...
|
|
: section ) })
|
|
.flat()
|
|
.join('') },
|
|
}
|
|
|
|
|
|
// XXX do we need anything else like .doc, attrs???
|
|
// XXX might be a good idea to offload arg value parsing to here...
|
|
var Macro =
|
|
module.Macro =
|
|
function(spec, func){
|
|
var args = [...arguments]
|
|
// function...
|
|
func = args.pop()
|
|
// arg sepc...
|
|
;(args.length > 0
|
|
&& args[args.length-1] instanceof Array)
|
|
&& (func.arg_spec = args.pop())
|
|
return func }
|
|
|
|
|
|
var parser =
|
|
module.parser = {
|
|
__proto__: BaseParser,
|
|
|
|
// list of macros that will get raw text of their content...
|
|
QUOTING_MACROS: ['quote'],
|
|
|
|
//
|
|
// <macro>(<args>, <body>, <state>){ .. }
|
|
// -> undefined
|
|
// -> <text>
|
|
// -> <array>
|
|
// -> <iterator>
|
|
// -> <func>(<state>)
|
|
// -> ...
|
|
//
|
|
// XXX do we need to make .macro.__proto__ module level object???
|
|
// XXX ASYNC make these support async page getters...
|
|
macros: {
|
|
//
|
|
// @(<name>[ <else>][ local])
|
|
// @(name=<name>[ else=<value>][ local])
|
|
//
|
|
// @arg(<name>[ <else>][ local])
|
|
// @arg(name=<name>[ else=<value>][ local])
|
|
//
|
|
// <arg <name>[ <else>][ local]/>
|
|
// <arg name=<name>[ else=<value>][ local]/>
|
|
//
|
|
// Resolution order:
|
|
// - local
|
|
// - .renderer
|
|
// - .root
|
|
//
|
|
// NOTE: else (default) value is parsed when accessed...
|
|
arg: Macro(
|
|
['name', 'else', ['local']],
|
|
function(args){
|
|
var v = this.args[args.name]
|
|
|| (!args.local
|
|
&& (this.renderer
|
|
&& this.renderer.args[args.name])
|
|
|| (this.root
|
|
&& this.root.args[args.name]))
|
|
v = v === true ?
|
|
args.name
|
|
: v
|
|
return v
|
|
|| (args['else']
|
|
&& this.parse(args['else'])) }),
|
|
'': Macro(
|
|
['name', 'else', ['local']],
|
|
function(args){
|
|
return this.__parser__.macros.arg.call(this, args) }),
|
|
args: function(){
|
|
return pwpath.obj2args(this.args) },
|
|
//
|
|
// @filter(<filter-spec>)
|
|
// <filter <filter-spec>/>
|
|
//
|
|
// <filter <filter-spec>>
|
|
// ...
|
|
// </filter>
|
|
//
|
|
// <filter-spec> ::=
|
|
// <filter> <filter-spec>
|
|
// | -<filter> <filter-spec>
|
|
//
|
|
// XXX BUG: this does not show any results:
|
|
// pwiki.parse('<filter test>moo test</filter>')
|
|
// -> ''
|
|
// while these do:
|
|
// pwiki.parse('<filter test/>moo test')
|
|
// -> 'moo TEST'
|
|
// await pwiki.parse('<filter test>moo test</filter>@var()')
|
|
// -> 'moo TEST'
|
|
// for more info see:
|
|
// file:///L:/work/pWiki/pwiki2.html#/Editors/Results
|
|
// XXX do we fix this or revise how/when filters work???
|
|
// ...including accounting for variables/expansions and the like...
|
|
// XXX REVISE...
|
|
filter: function(args, body, state, expand=true){
|
|
var that = this
|
|
|
|
var outer = state.filters =
|
|
state.filters ?? []
|
|
var local = Object.keys(args)
|
|
|
|
// trigger quote-filter...
|
|
var quote = local
|
|
.map(function(filter){
|
|
return (that.filters[filter] ?? {})['quote'] ?? [] })
|
|
.flat()
|
|
quote.length > 0
|
|
&& this.__parser__.macros['quote-filter']
|
|
.call(this, Object.fromEntries(Object.entries(quote)), null, state)
|
|
|
|
// local filters...
|
|
if(body != null){
|
|
// expand the body...
|
|
var ast = expand ?
|
|
this.__parser__.expand(this, body, state)
|
|
: body instanceof Array ?
|
|
body
|
|
// NOTE: wrapping the body in an array effectively
|
|
// escapes it from parsing...
|
|
: [body]
|
|
|
|
return function(state){
|
|
// XXX can we loose stuff from state this way???
|
|
// ...at this stage it should more or less be static -- check!
|
|
return Promise.awaitOrRun(
|
|
this.__parser__.parse(this, ast, {
|
|
...state,
|
|
filters: local.includes(this.ISOLATED_FILTERS) ?
|
|
local
|
|
: [...outer, ...local],
|
|
}),
|
|
function(res){
|
|
return {data: res} }) }
|
|
/*/ // XXX ASYNC...
|
|
return async function(state){
|
|
// XXX can we loose stuff from state this way???
|
|
// ...at this stage it should more or less be static -- check!
|
|
var res =
|
|
await this.__parser__.parse(this, ast, {
|
|
...state,
|
|
filters: local.includes(this.ISOLATED_FILTERS) ?
|
|
local
|
|
: [...outer, ...local],
|
|
})
|
|
return {data: res} }
|
|
//*/
|
|
|
|
// global filters...
|
|
} else {
|
|
state.filters = [...outer, ...local] } },
|
|
//
|
|
// @include(<path>)
|
|
//
|
|
// @include(<path> isolated recursive=<text>)
|
|
// @include(src=<path> isolated recursive=<text>)
|
|
//
|
|
// <include src=<path> .. >
|
|
// <text>
|
|
// </include>
|
|
//
|
|
// NOTE: there can be two ways of recursion in pWiki:
|
|
// - flat recursion
|
|
// /A -> /A -> /A -> ..
|
|
// - nested recursion
|
|
// /A -> /A/A -> /A/A/A -> ..
|
|
// Both can be either direct (type I) or indirect (type II).
|
|
// The former is trivial to check for while the later is
|
|
// not quite so, as we can have different contexts at
|
|
// different paths that would lead to different resulting
|
|
// renders.
|
|
// At the moment nested recursion is checked in a fast but
|
|
// not 100% correct manner focusing on path depth and ignoring
|
|
// the context, this potentially can lead to false positives.
|
|
// XXX need a way to make encode option transparent...
|
|
// XXX store a page cache in state...
|
|
include: Macro(
|
|
['src', 'recursive', 'join',
|
|
['s', 'strict', 'isolated']],
|
|
async function*(args, body, state, key='included', handler){
|
|
var macro = 'include'
|
|
if(typeof(args) == 'string'){
|
|
var [macro, args, body, state, key, handler] = arguments
|
|
key = key ?? 'included' }
|
|
var base = this.get(this.path.split(/\*/).shift())
|
|
var src = args.src
|
|
&& this.resolvePathVars(
|
|
await base.parse(args.src, state))
|
|
if(!src){
|
|
return }
|
|
// XXX INHERIT_ARGS special-case: inherit args by default...
|
|
// XXX should this be done when isolated???
|
|
if(this.actions_inherit_args
|
|
&& this.actions_inherit_args.has(pwpath.basename(src))
|
|
&& this.get(pwpath.dirname(src)).path == this.path){
|
|
src += ':$ARGS' }
|
|
var recursive = args.recursive ?? body
|
|
var isolated = args.isolated
|
|
var strict = args.strict
|
|
var strquotes = args.s
|
|
var join = args.join
|
|
&& await base.parse(args.join, state)
|
|
|
|
var depends = state.depends =
|
|
state.depends
|
|
?? new Set()
|
|
// XXX DEPENDS_PATTERN
|
|
depends.add(src)
|
|
|
|
handler = handler
|
|
?? async function(src, state){
|
|
return isolated ?
|
|
//{data: await this.get(src)
|
|
{data: await this
|
|
.parse({
|
|
seen: state.seen,
|
|
depends,
|
|
renderer: state.renderer,
|
|
})}
|
|
//: this.get(src)
|
|
: this
|
|
.parse(state) }
|
|
|
|
var first = true
|
|
for await (var page of this.get(src).asPages(strict)){
|
|
if(join && !first){
|
|
yield join }
|
|
first = false
|
|
|
|
//var full = page.path
|
|
var full = page.location
|
|
|
|
// handle recursion...
|
|
var parent_seen = 'seen' in state
|
|
var seen = state.seen =
|
|
new Set(state.seen ?? [])
|
|
if(seen.has(full)
|
|
// nesting path recursion...
|
|
|| (full.length % (this.NESTING_RECURSION_TEST_THRESHOLD || 50) == 0
|
|
&& (pwpath.split(full).length > 3
|
|
&& new Set([
|
|
await page.find(),
|
|
await page.get('..').find(),
|
|
await page.get('../..').find(),
|
|
]).size == 1
|
|
// XXX HACK???
|
|
|| pwpath.split(full).length > (this.NESTING_DEPTH_LIMIT || 20)))){
|
|
if(recursive == null){
|
|
console.warn(
|
|
`@${key}(..): ${
|
|
seen.has(full) ?
|
|
'direct'
|
|
: 'depth-limit'
|
|
} recursion detected:`, full, seen)
|
|
yield page.get(page.RECURSION_ERROR).parse()
|
|
continue }
|
|
// have the 'recursive' arg...
|
|
yield base.parse(recursive, state)
|
|
continue }
|
|
seen.add(full)
|
|
|
|
// load the included page...
|
|
var res = await handler.call(page, full, state)
|
|
depends.add(full)
|
|
res = strquotes ?
|
|
res
|
|
.replace(/["']/g, function(c){
|
|
return '%'+ c.charCodeAt().toString(16) })
|
|
: res
|
|
|
|
// NOTE: we only track recursion down and not sideways...
|
|
seen.delete(full)
|
|
if(!parent_seen){
|
|
delete state.seen }
|
|
|
|
yield res } }),
|
|
// NOTE: the main difference between this and @include is that
|
|
// this renders the src in the context of current page while
|
|
// include is rendered in the context of its page but with
|
|
// the same state...
|
|
// i.e. for @include(PATH) the paths within the included page
|
|
// are resolved relative to PATH while for @source(PATH)
|
|
// relative to the page containing the @source(..) statement...
|
|
source: Macro(
|
|
// XXX should this have the same args as include???
|
|
['src', 'recursive', 'join',
|
|
['s', 'strict']],
|
|
//['src'],
|
|
async function*(args, body, state){
|
|
var that = this
|
|
yield* this.__parser__.macros.include.call(this,
|
|
'source',
|
|
args, body, state, 'sources',
|
|
async function(src, state){
|
|
//return that.parse(that.get(src).raw, state) }) }),
|
|
return that.parse(this.raw, state) }) }),
|
|
|
|
// Load macro and slot definitions but ignore the page text...
|
|
//
|
|
// NOTE: this is essentially the same as @source(..) but returns ''.
|
|
// XXX revise name...
|
|
load: Macro(
|
|
['src', ['strict']],
|
|
async function*(args, body, state){
|
|
var that = this
|
|
yield* this.__parser__.macros.include.call(this,
|
|
'load',
|
|
args, body, state, 'sources',
|
|
async function(src, state){
|
|
await that.parse(this.raw, state)
|
|
return '' }) }),
|
|
//
|
|
// @quote(<src>)
|
|
//
|
|
// <quote src=<src>[ filter="<filter> ..."]/>
|
|
//
|
|
// <quote text=" .. "[ filter="<filter> ..."]/>
|
|
//
|
|
// <quote[ filter="<filter> ..."]>
|
|
// ..
|
|
// </quote>
|
|
//
|
|
//
|
|
// NOTE: src ant text arguments are mutually exclusive, src takes
|
|
// priority.
|
|
// NOTE: the filter argument has the same semantics as the filter
|
|
// macro with one exception, when used in quote, the body is
|
|
// not expanded...
|
|
// NOTE: the filter argument uses the same filters as @filter(..)
|
|
// NOTE: else argument implies strict mode...
|
|
// XXX need a way to escape macros -- i.e. include </quote> in a quoted text...
|
|
// XXX should join/else be sub-tags???
|
|
quote: Macro(
|
|
['src', 'filter', 'text', 'join', 'else',
|
|
['s', 'expandactions', 'strict']],
|
|
async function*(args, body, state){
|
|
var src = args.src //|| args[0]
|
|
var base = this.get(this.path.split(/\*/).shift())
|
|
var text = args.text
|
|
?? body
|
|
?? []
|
|
var strict = !!(args.strict
|
|
?? args['else']
|
|
?? false)
|
|
// parse arg values...
|
|
src = src ?
|
|
await base.parse(src, state)
|
|
: src
|
|
// XXX INHERIT_ARGS special-case: inherit args by default...
|
|
if(this.actions_inherit_args
|
|
&& this.actions_inherit_args.has(pwpath.basename(src))
|
|
&& this.get(pwpath.dirname(src)).path == this.path){
|
|
src += ':$ARGS' }
|
|
var expandactions =
|
|
args.expandactions
|
|
?? true
|
|
// XXX EXPERIMENTAL
|
|
var strquotes = args.s
|
|
|
|
var depends = state.depends =
|
|
state.depends
|
|
?? new Set()
|
|
// XXX DEPENDS_PATTERN
|
|
depends.add(src)
|
|
|
|
var pages = src ?
|
|
(!expandactions
|
|
&& await this.get(src).type == 'action' ?
|
|
base.get(this.QUOTE_ACTION_PAGE)
|
|
: await this.get(src).asPages(strict))
|
|
: text instanceof Array ?
|
|
[text.join('')]
|
|
: typeof(text) == 'string' ?
|
|
[text]
|
|
: text
|
|
// else...
|
|
pages = ((!pages
|
|
|| pages.length == 0)
|
|
&& args['else']) ?
|
|
[await base.parse(args['else'], state)]
|
|
: pages
|
|
// empty...
|
|
if(!pages || pages.length == 0){
|
|
return }
|
|
|
|
var join = args.join
|
|
&& await base.parse(args.join, state)
|
|
var first = true
|
|
for await (var page of pages){
|
|
if(join && !first){
|
|
yield join }
|
|
first = false
|
|
|
|
text = typeof(page) == 'string' ?
|
|
page
|
|
: (!expandactions
|
|
&& await page.type == 'action') ?
|
|
base.get(this.QUOTE_ACTION_PAGE).raw
|
|
: await page.raw
|
|
text = strquotes ?
|
|
text
|
|
.replace(/["']/g, function(c){
|
|
return '%'+ c.charCodeAt().toString(16) })
|
|
: text
|
|
|
|
page.path
|
|
&& depends.add(page.path)
|
|
|
|
var filters =
|
|
args.filter
|
|
&& args.filter
|
|
.trim()
|
|
.split(/\s+/g)
|
|
|
|
// NOTE: we are delaying .quote_filters handling here to
|
|
// make their semantics the same as general filters...
|
|
// ...and since we are internally calling .filter(..)
|
|
// macro we need to dance around it's architecture too...
|
|
// NOTE: since the body of quote(..) only has filters applied
|
|
// to it doing the first stage of .filter(..) as late
|
|
// as the second stage here will have no ill effect...
|
|
// NOTE: this uses the same filters as @filter(..)
|
|
// NOTE: the function wrapper here isolates text in
|
|
// a closure per function...
|
|
yield (function(text){
|
|
return async function(state){
|
|
// add global quote-filters...
|
|
filters =
|
|
(state.quote_filters
|
|
&& !(filters ?? []).includes(this.ISOLATED_FILTERS)) ?
|
|
[...state.quote_filters, ...(filters ?? [])]
|
|
: filters
|
|
return filters ?
|
|
await this.__parser__.callMacro(
|
|
this, 'filter', filters, text, state, false)
|
|
.call(this, state)
|
|
: text } })(text) } }),
|
|
// very similar to @filter(..) but will affect @quote(..) filters...
|
|
'quote-filter': function(args, body, state){
|
|
var filters = state.quote_filters =
|
|
state.quote_filters ?? []
|
|
filters.splice(filters.length, 0, ...Object.keys(args)) },
|
|
//
|
|
// <slot name=<name>/>
|
|
//
|
|
// <slot name=<name> text=<text>/>
|
|
//
|
|
// <slot name=<name>>
|
|
// ...
|
|
// </slot>
|
|
//
|
|
// Force show a slot...
|
|
// <slot shown ... />
|
|
//
|
|
// Force hide a slot...
|
|
// <slot hidden ... />
|
|
//
|
|
// Insert previous slot content...
|
|
// <content/>
|
|
//
|
|
//
|
|
// NOTE: by default only the first slot with <name> is visible,
|
|
// all other slots with <name> will replace its content, unless
|
|
// explicit shown/hidden arguments are given.
|
|
// NOTE: hidden has precedence over shown if both are given.
|
|
// NOTE: slots are handled in order of occurrence of opening tags
|
|
// in text and not by hierarchy, i.e. the later slot overrides
|
|
// the former and the most nested overrides the parent.
|
|
// This also works for cases where slots override slots they
|
|
// are contained in, this will not lead to recursion.
|
|
//
|
|
// XXX revise the use of hidden/shown use mechanic and if it's
|
|
// needed...
|
|
slot: Macro(
|
|
['name', 'text', ['shown', 'hidden']],
|
|
async function(args, body, state){
|
|
var name = args.name
|
|
var text = args.text
|
|
?? body
|
|
// NOTE: this can't be undefined for .expand(..) to work
|
|
// correctly...
|
|
?? []
|
|
|
|
var slots = state.slots =
|
|
state.slots
|
|
?? {}
|
|
|
|
// parse arg values...
|
|
name = name ?
|
|
await this.parse(name, state)
|
|
: name
|
|
|
|
//var hidden = name in slots
|
|
// XXX EXPERIMENTAL
|
|
var hidden =
|
|
// 'hidden' has priority...
|
|
args.hidden
|
|
// explicitly show... ()
|
|
|| (args.shown ?
|
|
false
|
|
// show first instance...
|
|
: name in slots)
|
|
|
|
// set slot value...
|
|
var stack = []
|
|
slots[name]
|
|
&& stack.push(slots[name])
|
|
delete slots[name]
|
|
var slot = await this.__parser__.expand(this, text, state)
|
|
var original = slot
|
|
slots[name]
|
|
&& stack.unshift(slot)
|
|
slot = slots[name] =
|
|
slots[name]
|
|
?? slot
|
|
// handle <content/>...
|
|
for(prev of stack){
|
|
// get the first <content/>
|
|
for(var i in slot){
|
|
if(typeof(slot[i]) != 'string'
|
|
&& slot[i].name == 'content'){
|
|
break }
|
|
i = null }
|
|
i != null
|
|
&& slot.splice(i, 1,
|
|
...prev
|
|
// remove nested slot handlers...
|
|
.filter(function(e){
|
|
return typeof(e) != 'function'
|
|
|| e.slot != name }) ) }
|
|
return hidden ?
|
|
''
|
|
: Object.assign(
|
|
function(state){
|
|
return (state.slots || {})[name] ?? original },
|
|
{slot: name}) }),
|
|
'content': ['slot'],
|
|
|
|
// XXX EXPERIMENTAL...
|
|
//
|
|
// NOTE: var value is parsed only on assignment and not on dereferencing...
|
|
//
|
|
// XXX should alpha/Alpha be 0 (current) or 1 based???
|
|
// XXX do we need a default attr???
|
|
// ...i.e. if not defined set to ..
|
|
// XXX INC_DEC do we need inc/dec and parent???
|
|
'var': Macro(
|
|
['name', 'text',
|
|
// XXX INC_DEC
|
|
['shown', 'hidden',
|
|
'parent',
|
|
'inc', 'dec',
|
|
'alpha', 'Alpha', 'roman', 'Roman']],
|
|
/*/
|
|
['shown', 'hidden']],
|
|
//*/
|
|
async function(args, body, state){
|
|
var name = args.name
|
|
if(!name){
|
|
return '' }
|
|
name = await this.parse(name, state)
|
|
// XXX INC_DEC
|
|
var inc = args.inc
|
|
var dec = args.dec
|
|
//*/
|
|
var text = args.text
|
|
?? body
|
|
// NOTE: .hidden has priority...
|
|
var show =
|
|
('hidden' in args ?
|
|
!args.hidden
|
|
: undefined)
|
|
?? args.shown
|
|
|
|
var vars = state.vars =
|
|
state.vars
|
|
?? {}
|
|
// XXX INC_DEC
|
|
if(args.parent && name in vars){
|
|
while(!vars.hasOwnProperty(name)
|
|
&& vars.__proto__ !== Object.prototype){
|
|
vars = vars.__proto__ } }
|
|
|
|
var handleFormat = function(value){
|
|
// roman number...
|
|
if(args.roman || args.Roman){
|
|
var n = parseInt(value)
|
|
return isNaN(n) ?
|
|
''
|
|
: args.Roman ?
|
|
n.toRoman()
|
|
: n.toRoman().toLowerCase() }
|
|
// alpha number...
|
|
if(args.alpha || args.Alpha){
|
|
var n = parseInt(value)
|
|
return isNaN(n) ?
|
|
''
|
|
: args.Alpha ?
|
|
n.toAlpha().toUpperCase()
|
|
: n.toAlpha() }
|
|
return value }
|
|
|
|
// inc/dec...
|
|
if(inc || dec){
|
|
if(!(name in vars)
|
|
|| isNaN(parseInt(vars[name]))){
|
|
return '' }
|
|
var cur = parseInt(vars[name])
|
|
cur +=
|
|
inc === true ?
|
|
1
|
|
: !inc ?
|
|
0
|
|
: parseInt(inc)
|
|
cur -=
|
|
dec === true ?
|
|
1
|
|
: !dec ?
|
|
0
|
|
: parseInt(dec)
|
|
vars[name] = cur + ''
|
|
|
|
// as-is...
|
|
return show ?? true ?
|
|
handleFormat(vars[name])
|
|
: '' }
|
|
//*/
|
|
|
|
// set...
|
|
if(text){
|
|
text = vars[name] =
|
|
await this.parse(text, state)
|
|
return show ?? false ?
|
|
text
|
|
: ''
|
|
// get...
|
|
} else {
|
|
return handleFormat(vars[name] ?? '') } }),
|
|
vars: async function(args, body, state){
|
|
var vars = state.vars =
|
|
state.vars
|
|
?? {}
|
|
for(var [name, value] of Object.entries(args)){
|
|
vars[await this.parse(name, state)] =
|
|
await this.parse(value, state) }
|
|
return '' },
|
|
|
|
//
|
|
// <macro src=<url>> .. </macro>
|
|
//
|
|
// <macro name=<name> src=<url> sort=<sort-spec>> .. </macro>
|
|
//
|
|
// <macro ...> ... </macro>
|
|
// <macro ... text=<text>/>
|
|
//
|
|
// <macro ... else=<text>> ... </macro>
|
|
// <macro ...>
|
|
// ...
|
|
//
|
|
//
|
|
// <join>
|
|
// ...
|
|
// </join>
|
|
//
|
|
// <else>
|
|
// ...
|
|
// </else>
|
|
// </macro>
|
|
//
|
|
// Macro variables:
|
|
// macro:count
|
|
// macro:index
|
|
//
|
|
// NOTE: this handles src count argument internally partially
|
|
// overriding <store>.match(..)'s implementation, this is done
|
|
// because @macro(..) needs to account for arbitrary nesting
|
|
// that <store>.match(..) can not know about...
|
|
// XXX should we do the same for offset???
|
|
//
|
|
// XXX BUG: strict does not seem to work:
|
|
// @macro(src="./resolved-page" else="no" text="yes" strict)
|
|
// -> yes
|
|
// should be "no"
|
|
// ...this seems to effect non-pattern pages...
|
|
// XXX should macro:index be 0 or 1 (current) based???
|
|
// XXX SORT sorting not implemented yet...
|
|
macro: Macro(
|
|
['name', 'src', 'sort', 'text', 'join', 'else',
|
|
['strict', 'isolated', 'inheritmacros', 'inheritvars']],
|
|
async function*(args, body, state){
|
|
var that = this
|
|
|
|
// helpers...
|
|
var _getBlock = function(name){
|
|
var block = args[name] ?
|
|
[{
|
|
args: {},
|
|
body: args[name],
|
|
}]
|
|
: (text ?? [])
|
|
.filter(function(e){
|
|
return typeof(e) != 'string'
|
|
&& e.name == name })
|
|
if(block.length == 0){
|
|
return }
|
|
// NOTE: when multiple blocks are present the
|
|
// last one is used...
|
|
block = block.pop()
|
|
block =
|
|
block.args.text
|
|
?? block.body
|
|
return block }
|
|
|
|
var base = this.get(this.path.split(/\*/).shift())
|
|
var macros = state.macros =
|
|
state.macros
|
|
?? {}
|
|
var vars = state.vars =
|
|
state.vars
|
|
?? {}
|
|
var depends = state.depends =
|
|
state.depends
|
|
?? new Set()
|
|
|
|
// uninheritable args...
|
|
// NOTE: arg handling is split in two, to make things simpler
|
|
// to process for retrieved named macros...
|
|
var src = args.src
|
|
var text = args.text
|
|
?? body
|
|
?? []
|
|
text = typeof(text) == 'string' ?
|
|
[...this.__parser__.group(this, text+'</macro>', 'macro')]
|
|
: text
|
|
var join, itext
|
|
var iargs = {}
|
|
|
|
// stored macros...
|
|
if(args.name){
|
|
var name = await base.parse(args.name, state)
|
|
// define new named macro...
|
|
if(text.length != 0){
|
|
// NOTE: we do not need to worry about saving
|
|
// stateful text here because it is only
|
|
// grouped and not expanded...
|
|
macros[name] =
|
|
[ text,
|
|
_getBlock('join'),
|
|
JSON.parse(JSON.stringify(args)), ]
|
|
// use existing macro...
|
|
} else if(macros
|
|
&& name in macros){
|
|
;[itext, join, iargs] = macros[name] } }
|
|
|
|
// inheritable args...
|
|
// XXX is there a point in overloading text???
|
|
text = text.length > 0 ?
|
|
text
|
|
: itext ?? text
|
|
var sort = (args.sort
|
|
?? iargs.sort
|
|
?? '')
|
|
.split(/\s+/g)
|
|
.filter(function(e){
|
|
return e != '' })
|
|
var strict =
|
|
('strict' in args ?
|
|
args.strict
|
|
: iargs.strict)
|
|
//?? true
|
|
?? false
|
|
var isolated =
|
|
('isolated' in args ?
|
|
args.isolated
|
|
: iargs.isolated)
|
|
?? true
|
|
var inheritmacros =
|
|
('inheritmacros' in args ?
|
|
args.inheritmacros
|
|
: iargs.inheritmacros)
|
|
?? true
|
|
var inheritvars =
|
|
('inheritvars' in args ?
|
|
args.inheritvars
|
|
: iargs.inheritvars)
|
|
?? true
|
|
|
|
if(src){
|
|
src = await base.parse(src, state)
|
|
// XXX INHERIT_ARGS special-case: inherit args by default...
|
|
if(this.actions_inherit_args
|
|
&& this.actions_inherit_args.has(pwpath.basename(src))
|
|
&& this.get(pwpath.dirname(src)).path == this.path){
|
|
src += ':$ARGS' }
|
|
// XXX DEPENDS_PATTERN
|
|
depends.add(src)
|
|
|
|
join = _getBlock('join')
|
|
?? join
|
|
join = join
|
|
&& await base.parse(join, state)
|
|
|
|
//var match = this.get(await base.parse(src, state))
|
|
//var match = this.get(src, strict)
|
|
var match = this.get(src)
|
|
|
|
// NOTE: thie does not introduce a dependency on each
|
|
// of the iterated pages, that is handled by the
|
|
// respective include/source/.. macros, this however
|
|
// only depends on page count...
|
|
depends.add(match.path)
|
|
|
|
// populate macrovars...
|
|
var macrovars = {}
|
|
for(var [key, value]
|
|
of Object.entries(
|
|
Object.assign(
|
|
args,
|
|
iargs,
|
|
{
|
|
strict,
|
|
isolated,
|
|
inheritmacros,
|
|
inheritvars,
|
|
}))){
|
|
macrovars['macro:'+ key] =
|
|
value === true ?
|
|
'yes'
|
|
: value === false ?
|
|
'no'
|
|
: value }
|
|
|
|
// handle count...
|
|
// NOTE: this duplicates <store>.match(..)'s functionality
|
|
// because we need to account for arbitrary macro
|
|
// nesting that .match(..) does not know about...
|
|
// XXX revise var naming...
|
|
// XXX these can be overriden in nested macros...
|
|
var count = match.args.count
|
|
if(count){
|
|
var c =
|
|
count == 'inherit' ?
|
|
(!('macro:count' in vars) ?
|
|
this.args.count
|
|
: undefined)
|
|
: count
|
|
if(c !== undefined){
|
|
vars['macro:count'] =
|
|
isNaN(parseInt(c)) ?
|
|
c
|
|
: parseInt(c)
|
|
vars['macro:index'] = 0 } }
|
|
|
|
// expand matches...
|
|
var first = true
|
|
for await(var page of match.asPages(strict)){
|
|
// handle count...
|
|
if('macro:count' in vars){
|
|
if(vars['macro:count'] <= vars['macro:index']){
|
|
break }
|
|
object.sources(vars, 'macro:index')
|
|
.shift()['macro:index']++ }
|
|
// output join between elements....
|
|
if(join && !first){
|
|
yield join }
|
|
first = false
|
|
if(isolated){
|
|
var _state = {
|
|
seen: state.seen,
|
|
depends,
|
|
renderer: state.renderer,
|
|
macros: inheritmacros ?
|
|
{__proto__: macros}
|
|
: {},
|
|
vars: inheritvars ?
|
|
{__proto__: vars,
|
|
...macrovars}
|
|
: {...macrovars},
|
|
}
|
|
yield this.__parser__.parse(page,
|
|
this.__parser__.expand(page,
|
|
text, _state), _state)
|
|
} else {
|
|
yield this.__parser__.expand(page, text, state) } }
|
|
// cleanup...
|
|
delete vars['macro:count']
|
|
delete vars['macro:index']
|
|
// else...
|
|
if(first
|
|
&& (text || args['else'])){
|
|
var else_block = _getBlock('else')
|
|
if(else_block){
|
|
yield this.__parser__.expand(this, else_block, state) } } } }),
|
|
|
|
// nesting rules...
|
|
'else': ['macro'],
|
|
'join': ['macro'],
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
/**********************************************************************
|
|
* vim:set ts=4 sw=4 : */ return module })
|