This repository was archived by the owner on Jan 1, 2025. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathlupa.lua
1810 lines (1695 loc) · 72.1 KB
/
lupa.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
-- Copyright 2015-2020 Mitchell. See LICENSE.
-- Sponsored by the Library of the University of Antwerp.
-- Contributions from Ana Balan.
-- Lupa templating engine.
--[[ This comment is for LuaDoc.
---
-- Lupa is a Jinja2 template engine implementation written in Lua and supports
-- Lua syntax within tags and variables.
module('lupa')]]
local M = {}
local lpeg = require('lpeg')
lpeg.locale(lpeg)
local space, newline = lpeg.space, lpeg.P('\r')^-1 * '\n'
local P, S, V = lpeg.P, lpeg.S, lpeg.V
local C, Cc, Cg, Cp, Ct = lpeg.C, lpeg.Cc, lpeg.Cg, lpeg.Cp, lpeg.Ct
---
-- Lupa's expression filters.
-- @class table
-- @name filters
M.filters = {}
---
-- Lupa's value tests.
-- @class table
-- @name tests
M.tests = {}
---
-- Lupa's template loaders.
-- @class table
-- @name loaders
M.loaders = {}
-- Lua version compatibility.
if _VERSION == 'Lua 5.1' then
function load(ld, source, mode, env)
local f, err = loadstring(ld)
if f and env then return setfenv(f, env) end
return f, err
end
table.unpack = unpack
end
local newline_sequence, keep_trailing_newline, autoescape = '\n', false, false
local loader
-- Creates and returns a token pattern with token name *name* and pattern
-- *patt*.
-- The returned pattern captures three values: the token's position and name,
-- and either a string value or table of capture values.
-- Tokens are used to construct an Abstract Syntax Tree (AST) for a template.
-- @param name The name of the token.
-- @param patt The pattern to match. It must contain only one capture: either a
-- string or table of captures.
-- @see evaluate
local function token(name, patt) return Cp() * Cc(name) * patt end
-- Returns an LPeg pattern that immediately raises an error with message
-- *errmsg* for invalid syntax when parsing a template.
-- @param errmsg The error message to raise an error with.
local function lpeg_error(errmsg)
return P(function(input, index)
input = input:sub(1, index)
local _, line_num = input:gsub('\n', '')
local col_num = #input:match('[^\n]*$')
error(string.format('Parse Error in file "%s" on line %d, column %d: %s',
M._FILENAME, line_num + 1, col_num, errmsg), 0)
end)
end
---
-- Configures the basic delimiters and options for templates.
-- This function then regenerates the grammar for parsing templates.
-- Note: this function cannot be used iteratively to configure Lupa options.
-- Any options not provided are reset to their default values.
-- @param ts The tag start delimiter. The default value is '{%'.
-- @param te The tag end delimiter. The default value is '%}'.
-- @param vs The variable start delimiter. The default value is '{{'.
-- @param ve The variable end delimiter. The default value is '}}'.
-- @param cs The comment start delimiter. The default value is '{#'.
-- @param ce The comment end delimiter. The default value is '#}'.
-- @param options Optional set of options for templates:
--
-- * `trim_blocks`: Trim the first newline after blocks.
-- * `lstrip_blocks`: Strip line-leading whitespace in front of tags.
-- * `newline_sequence`: The end-of-line character to use.
-- * `keep_trailing_newline`: Whether or not to keep a newline at the end of
-- a template.
-- * `autoescape`: Whether or not to autoescape HTML entities. May be a
-- function that accepts the template's filename as an argument and returns
-- a boolean.
-- * `loader`: Function that receives a template name to load and returns the
-- path to that template.
-- @name configure
function M.configure(ts, te, vs, ve, cs, ce, options)
if type(ts) == 'table' then options, ts = ts, nil end
if not ts then ts = '{%' end
if not te then te = '%}' end
if not vs then vs = '{{' end
if not ve then ve = '}}' end
if not cs then cs = '{#' end
if not ce then ce = '#}' end
-- Tokens for whitespace control.
local lstrip = token('lstrip', C('-')) + '+' -- '+' is handled by grammar
local rstrip = token('rstrip', -(P(te) + ve + ce) * C('-'))
-- Configure delimiters, including whitespace control.
local tag_start = P(ts) * lstrip^-1 * space^0
local tag_end = space^0 * rstrip^-1 * P(te)
local variable_start = P(vs) * lstrip^-1 * space^0
local variable_end = space^0 * rstrip^-1 * P(ve)
local comment_start = P(cs) * lstrip^-1 * space^0
local comment_end = space^0 * rstrip^-1 * P(ce)
if options and options.trim_blocks then
-- Consider whitespace, including a newline, immediately following a tag as
-- part of that tag so it is not captured as plain text. Basically, strip
-- the trailing newline from tags.
tag_end = tag_end * S(' \t')^0 * newline^-1
comment_end = comment_end * S(' \t')^0 * newline^-1
end
-- Error messages.
local variable_end_error = lpeg_error('"'..ve..'" expected')
local comment_end_error = lpeg_error('"'..ce..'" expected')
local tag_end_error = lpeg_error('"'..te..'" expected')
local endraw_error = lpeg_error('additional tag or "'..ts..' endraw '..te..
'" expected')
local expr_error = lpeg_error('expression expected')
local endblock_error = lpeg_error('additional tag or "'..ts..' endblock '..
te..'" expected')
local endfor_error = lpeg_error('additional tag or "'..ts..' endfor '..te..
'" expected')
local endif_error = lpeg_error('additional tag or "'..ts..' endif '..te..
'" expected')
local endmacro_error = lpeg_error('additional tag or "'..ts..' endmacro '..
te..'" expected')
local endcall_error = lpeg_error('additional tag or "'..ts..' endcall '..te..
'" expected')
local endfilter_error = lpeg_error('additional tag or "'..ts..' endfilter '..
te..'" expected')
local tag_error = lpeg_error('unknown or unexpected tag')
local main_error = lpeg_error('unexpected character; text or tag expected')
-- Grammar.
M.grammar = Ct(P{
-- Utility patterns used by tokens.
entity_start = tag_start + variable_start + comment_start,
any_text = (1 - V('entity_start'))^1,
-- Allow '{{' by default in expression text since it is valid in Lua.
expr_text = (1 - tag_end - tag_start - comment_start)^1,
-- When `options.lstrip_blocks` is enabled, ignore leading whitespace
-- immediately followed by a tag (as long as '+' is not present) so that
-- whitespace not captured as plain text. Basically, strip leading spaces
-- from tags.
line_text = (1 - newline - V('entity_start'))^1,
lstrip_entity_start = -P(vs) * (P(ts) + cs) * -P('+'),
lstrip_space = S(' \t')^1 * #V('lstrip_entity_start'),
text_lines = V('line_text') * (newline * -(S(' \t')^0 * V('lstrip_entity_start')) * V('line_text'))^0 * newline^-1 + newline,
-- Plain text.
text = (not options or not options.lstrip_blocks) and
token('text', C(V('any_text'))) or
V('lstrip_space') + token('text', C(V('text_lines'))),
-- Variables: {{ expr }}.
lua_table = '{' * ((1 - S('{}')) + V('lua_table'))^0 * '}',
variable = variable_start *
token('variable', C((V('lua_table') + (1 - variable_end))^0)) *
(variable_end + variable_end_error),
-- Filters: handled in variable evaluation.
-- Tests: handled in control structure expression evaluation.
-- Comments: {# comment #}.
comment = comment_start * (1 - comment_end)^0 * (comment_end + comment_end_error),
-- Whitespace control: handled in tag/variable/comment start/end.
-- Escaping: {% raw %} body {% endraw %}.
raw_block = tag_start * 'raw' * (tag_end + tag_end_error) *
token('text', C((1 - (tag_start * 'endraw' * tag_end))^0)) *
(tag_start * 'endraw' * tag_end + endraw_error),
-- Note: line statements are not supported since this grammer cannot parse
-- Lua itself.
-- Template inheritence.
-- {% block ... %} body {% endblock %}
block_block = tag_start * 'block' * space^1 * token('block', Ct((Cg(V('expr_text'), 'expression') + expr_error) * (tag_end + tag_end_error) *
V('body')^-1)) *
(tag_start * 'endblock' * tag_end + endblock_error),
-- {% extends ... %}
extends_tag = tag_start * 'extends' * space^1 * token('extends', C(V('expr_text')) + expr_error) * (tag_end + tag_end_error),
-- Super blocks are handled in variables.
-- Note: named block end tags are not supported since keeping track of that
-- state information is difficult.
-- Note: block nesting and scope is not applicable since blocks always have
-- access to scoped variables in this implementation.
-- Control Structures.
-- {% for expr %} body {% else %} body {% endfor %}
for_block = tag_start * 'for' * space^1 * token('for', Ct((Cg(V('expr_text'), 'expression') + expr_error) * (tag_end + tag_end_error) *
V('body')^-1 *
Cg(Ct(tag_start * 'else' * tag_end *
V('body')^-1), 'else')^-1)) *
(tag_start * 'endfor' * tag_end + endfor_error),
-- {% if expr %} body {% elseif expr %} body {% else %} body {% endif %}
if_block = tag_start * 'if' * space^1 * token('if', Ct((Cg(V('expr_text'), 'expression') + expr_error) * (tag_end + tag_end_error) *
V('body')^-1 *
Cg(Ct(Ct(tag_start * 'elseif' * space^1 * (Cg(V('expr_text'), 'expression') + expr_error) * (tag_end + tag_end_error) *
V('body')^-1)^1), 'elseif')^-1 *
Cg(Ct(tag_start * 'else' * tag_end *
V('body')^-1), 'else')^-1)) *
(tag_start * 'endif' * tag_end + endif_error),
-- {% macro expr %} body {% endmacro %}
macro_block = tag_start * 'macro' * space^1 * token('macro', Ct((Cg(V('expr_text'), 'expression') + expr_error) * (tag_end + tag_end_error) *
V('body')^-1)) *
(tag_start * 'endmacro' * tag_end + endmacro_error),
-- {% call expr %} body {% endcall %}
call_block = tag_start * 'call' * (space^1 + #P('(')) * token('call', Ct((Cg(V('expr_text'), 'expression') + expr_error) * (tag_end + tag_end_error) *
V('body')^-1)) *
(tag_start * 'endcall' * tag_end + endcall_error),
-- {% filter expr %} body {% endfilter %}
filter_block = tag_start * 'filter' * space^1 * token('filter', Ct((Cg(V('expr_text'), 'expression') + expr_error) * (tag_end + tag_end_error) *
V('body')^-1)) *
(tag_start * 'endfilter' * tag_end + endfilter_error),
-- {% set ... %}
set_tag = tag_start * 'set' * space^1 * token('set', C(V('expr_text')) + expr_error) * (tag_end + tag_end_error),
-- {% include ... %}
include_tag = tag_start * 'include' * space^1 * token('include', C(V('expr_text')) + expr_error) * (tag_end + tag_end_error),
-- {% import ... %}
import_tag = tag_start * 'import' * space^1 * token('import', C(V('expr_text')) + expr_error) * (tag_end + tag_end_error),
-- Note: i18n is not supported since it is out of scope for this
-- implementation.
-- Expression statement: {% do ... %}.
do_tag = tag_start * 'do' * space^1 * token('do', C(V('expr_text')) + expr_error) * (tag_end + tag_end_error),
-- Note: loop controls are not supported since that would require jumping
-- between "scopes" (e.g. from within an "if" block to outside that "if"
-- block's parent "for" block when coming across a {% break %} tag).
-- Note: with statement is not supported since it is out of scope for this
-- implementation.
-- Note: autoescape is not supported since it is out of scope for this
-- implementation.
-- Any valid blocks of text or tags.
body = (V('text') + V('variable') + V('comment') + V('raw_block') +
V('block_block') + V('extends_tag') + V('for_block') +
V('if_block') + V('macro_block') + V('call_block') +
V('filter_block') + V('set_tag') + V('include_tag') +
V('import_tag') + V('do_tag'))^0,
-- Main pattern.
V('body') * (-1 + tag_start * tag_error + main_error),
})
-- Other options.
if options and options.newline_sequence then
assert(options.newline_sequence:find('^\r?\n$'),
'options.newline_sequence must be "\r\n" or "\n"')
newline_sequence = options.newline_sequence
else
newline_sequence = '\n'
end
if options and options.keep_trailing_newline then
keep_trailing_newline = options.keep_trailing_newline
else
keep_trailing_newline = false
end
if options and options.autoescape then
autoescape = options.autoescape
else
autoescape = false
end
if options and options.loader then
assert(type(options.loader) == 'function',
'options.loader must be a function that returns a filename')
loader = options.loader
else
loader = M.loaders.filesystem()
end
end
-- Wraps Lua's `assert()` in template environment *env* such that, when called
-- in conjunction with another Lua function that produces an error message (e.g.
-- `load()` and `pcall()`), that error message's context (source and line
-- number) is replaced by the template's context.
-- This results in Lua's error messages pointing to a template position rather
-- than this library's source code.
-- @param env The environment for the currently running template. It must have
-- a `_SOURCE` field with the template's source text and a `_POSITION` field
-- with the current position of expansion.
-- @param ... Arguments to Lua's `assert()`.
local function env_assert(env, ...)
if not select(1, ...) then
local input = env._LUPASOURCE:sub(1, env._LUPAPOSITION)
local _, line_num = input:gsub('\n', '')
local col_num = #input:match('[^\n]*$')
local errmsg = select(2, ...)
errmsg = errmsg:match(':%d+: (.*)$') or errmsg -- reformat if necessary
error(string.format('Runtime Error in file "%s" on line %d, column %d: %s',
env._LUPAFILENAME, line_num + 1, col_num, errmsg), 0)
end
return ...
end
-- Returns a generator that returns the position and filter in a list of
-- filters, taking into account '|'s that may be within filter arguments.
-- @usage for pos, filter in each_filter('foo|join("|")|bar') do ... end
local function each_filter(s)
local init = 1
return function(s)
local pos, filter, e = s:match('^%s*()([^|(]+%b()[^|]*)|?()', init)
if not pos then pos, filter, e = s:match('()([^|]+)|?()', init) end
init = e
return pos, filter
end, s
end
-- Evaluates template variable *expression* subject to template environment
-- *env*, applying any filters given in *expression*.
-- @param expression The string expression to evaluate.
-- @param env The environment to evaluate the expression in.
local function eval(expression, env)
local expr, pos, filters = expression:match('^([^|]*)|?()(.-)$')
-- Evaluate base expression.
local f = env_assert(env, load('return '..expr, nil, nil, env))
local result = select(2, env_assert(env, pcall(f)))
-- Apply any filters.
local results, multiple_results = nil, false
local p = env._LUPAPOSITION + pos - 1 -- mark position at first filter
for pos, filter in each_filter(filters) do
env._LUPAPOSITION = p + pos - 1 -- update position for error messages
local name, params = filter:match('^%s*([%w_]+)%(?(.-)%)?%s*$')
f = M.filters[name]
env_assert(env, f, 'unknown filter "'..name..'"')
local args = env_assert(env, load('return {'..params..'}', nil, nil, env),
'invalid filter parameter(s) for "'..name..'"')()
if not multiple_results then
results = {select(2,
env_assert(env, pcall(f, result, table.unpack(args))))}
else
for i = 1, #results do table.insert(args, i, results[i]) end
results = {select(2, env_assert(env, pcall(f, table.unpack(args))))}
end
result, multiple_results = results[1], #results > 1
end
if multiple_results then return table.unpack(results) end
return result
end
local iterate
-- Iterates over *ast*, a collection of tokens from a portion of a template's
-- Abstract Syntax Tree (AST), evaluating any expressions in template
-- environment *env*, and returns a concatenation of the results.
-- @param ast A template's AST or portion of its AST (e.g. portion inside a
-- 'for' control structure).
-- @param env Environment to evaluate any expressions in.
local function evaluate(ast, env)
local chunks = {}
local extends -- text of a parent template
local rstrip -- flag for stripping leading whitespace of next token
for i = 1, #ast, 3 do
local pos, token, block = ast[i], ast[i + 1], ast[i + 2]
env._LUPAPOSITION = pos
if token == 'text' then
chunks[#chunks + 1] = block
elseif token == 'variable' then
local value = eval(block, env)
if autoescape then
local escape = autoescape
if type(autoescape) == 'function' then
escape = autoescape(env._LUPAFILENAME) -- TODO: test
end
if escape and type(value) == 'string' then
value = M.filters.escape(value)
end
end
chunks[#chunks + 1] = value ~= nil and tostring(value) or ''
elseif token == 'extends' then
env_assert(env, not extends,
'cannot have multiple "extends" in the same scope')
local file = eval(block, env) -- covers strings and variables
extends = file
env._LUPAEXTENDED = true -- used by parent templates
elseif token == 'block' then
local name = block.expression:match('^[%w_]+$')
env_assert(env, name, 'invalid block name')
-- Store the block for potential use by the parent template if this
-- template is a child template, or for use by `self`.
if not env._LUPABLOCKS then env._LUPABLOCKS = {} end
if not env._LUPABLOCKS[name] then env._LUPABLOCKS[name] = {} end
table.insert(env._LUPABLOCKS[name], 1, block)
-- Handle the block properly.
if not extends then
if not env._LUPAEXTENDED then
-- Evaluate the block normally.
chunks[#chunks + 1] = evaluate(block, env)
else
-- A child template is overriding this parent's named block. Evaluate
-- the child's block and use it instead of the parent's.
local blocks = env._LUPABLOCKS[name]
local super_env = setmetatable({super = function()
-- Loop through the chain of defined blocks, evaluating from top to
-- bottom, and return the bottom block. In each sub-block, the
-- 'super' variable needs to point to the next-highest block's
-- evaluated result.
local super = evaluate(block, env) -- start with parent block
local sub_env = setmetatable({super = function() return super end},
{__index = env})
for i = 1, #blocks - 1 do super = evaluate(blocks[i], sub_env) end
return super
end}, {__index = env})
chunks[#chunks + 1] = evaluate(blocks[#blocks], super_env)
end
end
elseif token == 'for' then
local expr = block.expression
local p = env._LUPAPOSITION -- mark position at beginning of expression
-- Extract variable list and generator.
local patt = '^([%w_,%s]+)%s+in%s+()(.+)%s+if%s+(.+)$'
local var_list, pos, generator, if_expr = expr:match(patt)
if not var_list then
var_list, pos, generator = expr:match('^([%w_,%s]+)%s+in%s+()(.+)$')
end
env_assert(env, var_list and generator, 'invalid for expression')
-- Store variable names in a list for loop assignment.
local variables = {}
for variable, pos in var_list:gmatch('([^,%s]+)()') do
env._LUPAPOSITION = p + pos - 1 -- update position for error messages
env_assert(env, variable:find('^[%a_]') and variable ~= 'loop',
'invalid variable name')
variables[#variables + 1] = variable
end
-- Evaluate the generator and perform the iteration.
env._LUPAPOSITION = p + pos - 1 -- update position to generator
if not generator:find('|') then
generator = env_assert(env, load('return '..generator, nil, nil, env))
else
local generator_expr = generator
generator = function() return eval(generator_expr, env) end
end
local new_env = setmetatable({}, {__index = env})
chunks[#chunks + 1] = iterate(generator, variables, if_expr, block,
new_env, 1, ast[i + 4] == 'lstrip')
elseif token == 'if' then
if eval(block.expression, env) then
chunks[#chunks + 1] = evaluate(block, env)
else
local evaluate_else = true
local elseifs = block['elseif']
if elseifs then
for j = 1, #elseifs do
if eval(elseifs[j].expression, env) then
chunks[#chunks + 1] = evaluate(elseifs[j], env)
evaluate_else = false
break
end
end
end
if evaluate_else and block['else'] then
chunks[#chunks + 1] = evaluate(block['else'], env)
end
end
elseif token == 'macro' then
-- Parse the macro's name and parameter list.
local signature = block.expression
local name, param_list = signature:match('^([%w_]+)(%b())')
env_assert(env, name and param_list, 'invalid macro expression')
param_list = param_list:sub(2, -2)
local p = env._LUPAPOSITION + #name + 1 -- mark pos at beginning of args
local params, defaults = {}, {}
for param, pos, default in param_list:gmatch('([%w_]+)=?()([^,]*)') do
params[#params + 1] = param
if default ~= '' then
env._LUPAPOSITION = p + pos - 1 -- update position for error messages
local f = env_assert(env, load('return '..default))
defaults[param] = select(2, env_assert(env, pcall(f)))
end
end
-- Create the function associated with the macro such that when the
-- function is called (from within {{ ... }}), the macro's body is
-- evaluated subject to an environment where parameter names are variables
-- whose values are the ones passed to the macro itself.
env[name] = function(...)
local new_env = setmetatable({}, {__index = function(_, k)
if k == 'caller' and type(env[k]) ~= 'function' then return nil end
return env[k]
end})
local args = {...}
-- Assign the given parameter values.
for i = 1, #args do
if i > #params then break end
new_env[params[i]] = args[i]
end
-- Clear all other unspecified parameter values or set them to their
-- defined defaults.
for i = #args + 1, #params do
new_env[params[i]] = defaults[params[i]]
end
-- Store extra parameters in "varargs" variable.
new_env.varargs = {}
for i = #params + 1, #args do
new_env.varargs[#new_env.varargs + 1] = args[i]
end
local chunk = evaluate(block, new_env)
if ast[i + 4] == 'lstrip' then chunk = chunk:gsub('%s*$', '') end
return chunk
end
elseif token == 'call' then
-- Parse the call block's parameter list (if any) and determine the macro
-- to call.
local param_list = block.expression:match('^(%b())')
local params = {}
if param_list then
for param in param_list:gmatch('[%w_]+') do
params[#params + 1] = param
end
end
local macro = block.expression:match('^%b()(.+)$') or block.expression
-- Evaluate the given macro, subject to a "caller" function that returns
-- the contents of this call block. Any arguments passed to the caller
-- function are used as values of this parameters parsed earlier.
local old_caller = M.env.caller -- save
M.env.caller = function(...)
local new_env = setmetatable({}, {__index = env})
local args = {...}
-- Assign the given parameter values (if any).
for i = 1, #args do new_env[params[i]] = args[i] end
local chunk = evaluate(block, new_env)
if ast[i + 4] == 'lstrip' then chunk = chunk:gsub('%s*$', '') end
return chunk
end
chunks[#chunks + 1] = eval(macro, env)
M.env.caller = old_caller -- restore
elseif token == 'filter' then
local text = evaluate(block, env)
local p = env._LUPAPOSITION -- mark position at beginning of expression
for pos, filter in each_filter(block.expression) do
env._LUPAPOSITION = p + pos - 1 -- update position for error messages
local name, params = filter:match('^%s*([%w_]+)%(?(.-)%)?%s*$')
local f = M.filters[name]
env_assert(env, f, 'unknown filter "'..name..'"')
local args = env_assert(env, load('return {'..params..'}'),
'invalid filter parameter(s) for "'..name..
'"')()
text = select(2, env_assert(env, pcall(f, text, table.unpack(args))))
end
chunks[#chunks + 1] = text
elseif token == 'set' then
local var, expr = block:match('^([%a_][%w_]*)%s*=%s*(.+)$')
env_assert(env, var and expr, 'invalid variable name or expression')
env[var] = eval(expr, env)
elseif token == 'do' then
env_assert(env, pcall(env_assert(env, load(block, nil, nil, env))))
elseif token == 'include' then
-- Parse the include block for flags.
local without_context = block:find('without%s+context%s*')
local ignore_missing = block:find('ignore%s+missing%s*')
block = block:gsub('witho?u?t?%s+context%s*', '')
:gsub('ignore%s+missing%s*', '')
-- Evaluate the include expression in order to determine the file to
-- include. If the result is a table, use the first file that exists.
local file = eval(block, env) -- covers strings and variables
if type(file) == 'table' then
local files = file
for i = 1, #files do
file = loader(files[i], env)
if file then break end
end
if type(file) == 'table' then file = nil end
elseif type(file) == 'string' then
file = loader(file, env)
else
error('"include" requires a string or table of files')
end
-- If the file exists, include it. Otherwise throw an error unless the
-- "ignore missing" flag was given.
env_assert(env, file or ignore_missing, 'no file(s) found to include')
if file then
chunks[#chunks + 1] = M.expand_file(file, not without_context and env or
M.env)
end
elseif token == 'import' then
local file, global = block:match('^%s*(.+)%s+as%s+([%a][%w_]*)%s*')
local new_env = setmetatable({}, {
__index = block:find('with%s+context%s*$') and env or M.env
})
M.expand_file(eval(file or block, env), new_env)
-- Copy any defined macros and variables over into the proper namespace.
if global then env[global] = {} end
local namespace = global and env[global] or env
for k, v in pairs(new_env) do if not env[k] then namespace[k] = v end end
elseif token == 'lstrip' and chunks[#chunks] then
chunks[#chunks] = chunks[#chunks]:gsub('%s*$', '')
elseif token == 'rstrip' then
rstrip = true -- can only strip after determining the next chunk
end
if rstrip and token ~= 'rstrip' then
chunks[#chunks] = chunks[#chunks]:gsub('^%s*', '')
rstrip = false
end
end
return not extends and table.concat(chunks) or M.expand_file(extends, env)
end
local pairs_gen, ipairs_gen = pairs({}), ipairs({})
-- Iterates over the generator *generator* subject to string "if" expression
-- *if_expr*, assigns that generator's returned values to the variable names
-- listed in *variables* within template environment *env*, evaluates any
-- expressions in *block* (a portion of a template's AST), and returns a
-- concatenation of the results.
-- @param generator Either a function that returns a generator function, or a
-- table to iterate over. In the latter case, `ipairs()` is used as the
-- generator function.
-- @param variables List of variable names to assign values returned by
-- *generator* to.
-- @param if_expr A conditional expression that when `false`, skips the current
-- loop item.
-- @param block The portion inside the 'for' structure of a template's AST to
-- iterate with.
-- @param env The environment iteration variables are defined in and where
-- expressions are evaluated in.
-- @param depth The current recursion depth. Recursion is performed by calling
-- `loop(t)` with a table to iterate over.
-- @param lstrip Whether or not the "endfor" block strips whitespace on the
-- left. When `true`, all blocks produced by iteration are left-stripped.
iterate = function(generator, variables, if_expr, block, env, depth, lstrip)
local chunks = {}
local orig_variables = {} -- used to store original loop variables' values
for i = 1, #variables do orig_variables[variables[i]] = env[variables[i]] end
local i, n = 1 -- used for loop variables
local _, s, v -- state variables
if type(generator) == 'function' then
_, generator, s, v = env_assert(env, pcall(generator))
-- In practice, a generator's state variable is normally unused and hidden.
-- This is not the case for 'pairs()' and 'ipairs', though.
if variables[1] ~= '_index' and generator ~= pairs_gen and
generator ~= ipairs_gen then
table.insert(variables, 1, '_index')
end
end
if type(generator) == 'table' then
n = #generator
generator, s, v = ipairs(generator)
-- "for x in y" translates to "for _, x in ipairs(y)"; hide _ state variable
if variables[1] ~= '_index' then table.insert(variables, 1, '_index') end
end
if generator then
local first_results -- for preventing infinite loop from invalid generator
while true do
local results = {generator(s, v)}
if results[1] == nil then break end
-- If the results from the generator look like results returned by a
-- generator itself (function, state, initial variable), verify last two
-- results are different. If they are the same, then the original
-- generator is invalid and will loop infinitely.
if first_results == nil then
first_results = #results == 3 and type(results[1]) == 'function' and
results
elseif first_results then
env_assert(env, results[3] ~= first_results[3] or
results[2] ~= first_results[2],
'invalid generator (infinite loop)')
end
-- Assign context variables and evaluate the body of the loop.
-- As long as the result (ignoring the _index variable) is not a single
-- table and there is only one loop variable defined (again, ignoring
-- _index variable), assignment occurs as normal in Lua. Otherwise,
-- unpacking on the table is done (like assignment to ...).
if not (type(results[2]) == 'table' and #results == 2 and
#variables > 2) then
for j = 1, #variables do env[variables[j]] = results[j] end
else
for j = 2, #variables do env[variables[j]] = results[2][j - 1] end
end
if not if_expr or eval(if_expr, env) then
env.loop = setmetatable({
index = i, index0 = i - 1,
revindex = n and n - (i - 1), revindex0 = n and n - i,
first = i == 1, last = i == n, length = n,
cycle = function(...)
return select((i - 1) % select('#', ...) + 1, ...)
end,
depth = depth, depth0 = depth - 1
}, {__call = function(_, t)
return iterate(t, variables, if_expr, block, env, depth + 1, lstrip)
end})
chunks[#chunks + 1] = evaluate(block, env)
if lstrip then chunks[#chunks] = chunks[#chunks]:gsub('%s*$', '') end
i = i + 1
end
-- Prepare for next iteration.
v = results[1]
end
end
if i == 1 and block['else'] then
chunks[#chunks + 1] = evaluate(block['else'], env)
end
for i = 1, #variables do env[variables[i]] = orig_variables[variables[i]] end
return table.concat(chunks)
end
-- Expands string template *template* from source *source*, subject to template
-- environment *env*, and returns the result.
-- @param template String template to expand.
-- @param env Environment for the given template.
-- @param source Filename or identifier the template comes from for error
-- messages and debugging.
local function expand(template, env, source)
template = template:gsub('\r?\n', newline_sequence) -- normalize
if not keep_trailing_newline then template = template:gsub('\r?\n$', '') end
-- Set up environment.
if not env then env = {} end
if not getmetatable(env) then env = setmetatable(env, {__index = M.env}) end
env.self = setmetatable({}, {__index = function(_, k)
env_assert(env, env._LUPABLOCKS and env._LUPABLOCKS[k],
'undefined block "'..k..'"')
return function() return evaluate(env._LUPABLOCKS[k][1], env) end
end})
-- Set context variables and expand the template.
env._LUPASOURCE, env._LUPAFILENAME = template, source
M._FILENAME = source -- for lpeg errors only
local ast = assert(lpeg.match(M.grammar, template), "internal error")
local result = evaluate(ast, env)
return result
end
---
-- Expands the string template *template*, subject to template environment
-- *env*, and returns the result.
-- @param template String template to expand.
-- @param env Optional environment for the given template.
-- @name expand
function M.expand(template, env) return expand(template, env, '<string>') end
---
-- Expands the template within file *filename*, subject to template environment
-- *env*, and returns the result.
-- @param filename Filename containing the template to expand.
-- @param env Optional environment for the template to expand.
-- @name expand_file
function M.expand_file(filename, env)
filename = loader(filename, env) or filename
local f = (not env or not env._LUPASOURCE) and assert(io.open(filename)) or
env_assert(env, io.open(filename))
local template = f:read('*a')
f:close()
return expand(template, env, filename)
end
---
-- Returns a loader for templates that uses the filesystem starting at directory
-- *directory*.
-- When looking up the template for a given filename, the loader considers the
-- following: if no template is being expanded, the loader assumes the given
-- filename is relative to *directory* and returns the full path; otherwise the
-- loader assumes the given filename is relative to the current template's
-- directory and returns the full path.
-- The returned path may be passed to `io.open()`.
-- @param directory Optional the template root directory. The default value is
-- ".", which is the current working directory.
-- @name loaders.filesystem
-- @see configure
function M.loaders.filesystem(directory)
return function(filename, env)
if not filename then return nil end
local current_dir = env and env._LUPAFILENAME and
env._LUPAFILENAME:match('^(.+)[/\\]')
if not filename:find('^/') and not filename:find('^%a:[/\\]') then
filename = (current_dir or directory or '.')..'/'..filename
end
local f = io.open(filename)
if not f then return nil end
f:close()
return filename
end
end
-- Globally defined functions.
---
-- Returns a sequence of integers from *start* to *stop*, inclusive, in
-- increments of *step*.
-- The complete sequence is generated at once -- no generator is returned.
-- @param start Optional number to start at. The default value is `1`.
-- @param stop Number to stop at.
-- @param step Optional increment between sequence elements. The default value
-- is `1`.
-- @name _G.range
function range(start, stop, step)
if not stop and not step then stop, start = start, 1 end
if not step then step = 1 end
local t = {}
for i = start, stop, step do t[#t + 1] = i end
return t
end
---
-- Returns an object that cycles through the given values by calls to its
-- `next()` function.
-- A `current` field contains the cycler's current value and a `reset()`
-- function resets the cycler to its beginning.
-- @param ... Values to cycle through.
-- @usage c = cycler(1, 2, 3)
-- @usage c:next(), c:next() --> 1, 2
-- @usage c:reset() --> c.current == 1
-- @name _G.cycler
function cycler(...)
local c = {...}
c.n, c.i, c.current = #c, 1, c[1]
function c:next()
local current = self.current
self.i = self.i + 1
if self.i > self.n then self.i = 1 end
self.current = self[self.i]
return current
end
function c:reset() self.i, self.current = 1, self[1] end
return c
end
-- Create the default sandbox environment for templates.
local safe = {
-- Lua globals.
'_VERSION', 'ipairs', 'math', 'pairs', 'select', 'tonumber', 'tostring',
'type', 'bit32', 'os.date', 'os.time', 'string', 'table', 'utf8',
-- Lupa globals.
'range', 'cycler'
}
local sandbox_env = setmetatable({}, {__index = M.tests})
for i = 1, #safe do
local v = safe[i]
if not v:find('%.') then
sandbox_env[v] = _G[v]
else
local mod, func = v:match('^([^.]+)%.(.+)$')
if not sandbox_env[mod] then sandbox_env[mod] = {} end
sandbox_env[mod][func] = _G[mod][func]
end
end
sandbox_env._G = sandbox_env
---
-- Resets Lupa's default delimiters, options, and environments to their
-- original default values.
-- @name reset
function M.reset()
M.configure('{%', '%}', '{{', '}}', '{#', '#}')
M.env = setmetatable({}, {__index = sandbox_env})
end
M.reset()
---
-- The default template environment.
-- @class table
-- @name env
local env
-- Lupa filters.
---
-- Returns the absolute value of number *n*.
-- @param n The number to compute the absolute value of.
-- @name filters.abs
M.filters.abs = math.abs
-- Returns a table that, when indexed with an integer, indexes table *t* with
-- that integer along with string *attribute*.
-- This is used by filters that operate on particular attributes of table
-- elements.
-- @param t The table to index.
-- @param attribute The additional attribute to index with.
local function attr_accessor(t, attribute)
return setmetatable({}, {__index = function(_, i)
local value = t[i]
attribute = tonumber(attribute) or attribute
if type(attribute) == 'number' then return value[attribute] end
for k in attribute:gmatch('[^.]+') do value = value[k] end
return value
end})
end
---
-- Returns a generator that produces all of the items in table *t* in batches
-- of size *size*, filling any empty spaces with value *fill*.
-- Combine this with the "list" filter to produce a list.
-- @param t The table to split into batches.
-- @param size The batch size.
-- @param fill The value to use when filling in any empty space in the last
-- batch.
-- @usage expand('{% for i in {1, 2, 3}|batch(2, 0) %}{{ i|string }}
-- {% endfor %}') --> {1, 2} {3, 0}
-- @see filters.list
-- @name filters.batch
function M.filters.batch(t, size, fill)
assert(t, 'input to filter "batch" was nil instead of a table')
local n = #t
return function(t, i)
if i > n then return nil end
local batch = {}
for j = i, i + size - 1 do batch[j - i + 1] = t[j] end
if i + size > n and fill then
for j = n + 1, i + size - 1 do batch[#batch + 1] = fill end
end
return i + size, batch
end, t, 1
end
---
-- Capitalizes string *s*.
-- The first character will be uppercased, the others lowercased.
-- @param s The string to capitalize.
-- @usage expand('{{ "foo bar"|capitalize }}') --> Foo bar
-- @name filters.capitalize
function M.filters.capitalize(s)
assert(s, 'input to filter "capitalize" was nil instead of a string')
local first, rest = s:match('^(.)(.*)$')
return first and first:upper()..rest:lower() or s
end
---
-- Centers string *s* within a string of length *width*.
-- @param s The string to center.
-- @param width The length of the centered string.
-- @usage expand('{{ "foo"|center(9) }}') --> " foo "
-- @name filters.center
function M.filters.center(s, width)
assert(s, 'input to filter "center" was nil instead of a string')
local padding = (width or 80) - #s
local left, right = math.ceil(padding / 2), math.floor(padding / 2)
return ("%s%s%s"):format((' '):rep(left), s, (' '):rep(right))
end
---
-- Returns value *value* or value *default*, depending on whether or not *value*
-- is "true" and whether or not boolean *false_defaults* is `true`.
-- @param value The value return if "true" or if `false` and *false_defaults*
-- is `true`.
-- @param default The value to return if *value* is `nil` or `false` (the latter
-- applies only if *false_defaults* is `true`).
-- @param false_defaults Optional flag indicating whether or not to return
-- *default* if *value* is `false`. The default value is `false`.
-- @usage expand('{{ false|default("no") }}') --> false
-- @usage expand('{{ false|default("no", true) }') --> no
-- @name filters.default
function M.filters.default(value, default, false_defaults)
if value == nil or false_defaults and not value then return default end
return value
end
---
-- Returns a table constructed from table *t* such that each element is a list
-- that contains a single key-value pair and all elements are sorted according
-- to string *by* (which is either "key" or "value") and boolean
-- *case_sensitive*.
-- @param value The table to sort.
-- @param case_sensitive Optional flag indicating whether or not to consider
-- case when sorting string values. The default value is `false`.
-- @param by Optional string that specifies which of the key-value to sort by,
-- either "key" or "value". The default value is `"key"`.
-- @usage expand('{{ {b = 1, a = 2}|dictsort|string }}') --> {{"a", 2},
-- {"b", 1}}
-- @name filters.dictsort
function M.filters.dictsort(t, case_sensitive, by)
assert(t, 'input to filter "dictsort" was nil instead of a table')
assert(not by or by == 'key' or by == 'value',
'filter "dictsort" can only sort tables by "key" or "value"')
local i = (not by or by == 'key') and 1 or 2
local items = {}
for k, v in pairs(t) do items[#items + 1] = {k, v} end
table.sort(items, function(a, b)
a, b = a[i], b[i]
if not case_sensitive then
if type(a) == 'string' then a = a:lower() end
if type(b) == 'string' then b = b:lower() end
end
return a < b
end)
return items
end
---
-- Returns an HTML-safe copy of string *s*.
-- @param s String to ensure is HTML-safe.
-- @usage expand([[{{ '<">&'|e}}]]) --> <">&
-- @name filters.escape
function M.filters.escape(s)
assert(s, 'input to filter "escape" was nil instead of a string')