Armin Ronacher | ab45b84 | 2007-03-18 20:47:50 +0100 | [diff] [blame] | 1 | # -*- coding: utf-8 -*- |
| 2 | """ |
| 3 | unit test for loop functions |
| 4 | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~ |
| 5 | |
Armin Ronacher | 62ccd1b | 2009-01-04 14:26:19 +0100 | [diff] [blame] | 6 | :copyright: (c) 2009 by the Jinja Team. |
Armin Ronacher | ab45b84 | 2007-03-18 20:47:50 +0100 | [diff] [blame] | 7 | :license: BSD, see LICENSE for more details. |
| 8 | """ |
Armin Ronacher | 10f3ba2 | 2008-04-18 11:30:37 +0200 | [diff] [blame] | 9 | from py.test import raises |
Armin Ronacher | 9f258ff | 2008-05-24 22:28:52 +0200 | [diff] [blame] | 10 | from jinja2.exceptions import UndefinedError, TemplateSyntaxError |
Armin Ronacher | 10f3ba2 | 2008-04-18 11:30:37 +0200 | [diff] [blame] | 11 | |
Armin Ronacher | ab45b84 | 2007-03-18 20:47:50 +0100 | [diff] [blame] | 12 | |
| 13 | SIMPLE = '''{% for item in seq %}{{ item }}{% endfor %}''' |
| 14 | ELSE = '''{% for item in seq %}XXX{% else %}...{% endfor %}''' |
| 15 | EMPTYBLOCKS = '''<{% for item in seq %}{% else %}{% endfor %}>''' |
| 16 | CONTEXTVARS = '''{% for item in seq %}\ |
| 17 | {{ loop.index }}|{{ loop.index0 }}|{{ loop.revindex }}|{{ |
| 18 | loop.revindex0 }}|{{ loop.first }}|{{ loop.last }}|{{ |
Armin Ronacher | 10f3ba2 | 2008-04-18 11:30:37 +0200 | [diff] [blame] | 19 | loop.length }}###{% endfor %}''' |
| 20 | CYCLING = '''{% for item in seq %}{{ loop.cycle('<1>', '<2>') }}{% endfor %}\ |
| 21 | {% for item in seq %}{{ loop.cycle(*through) }}{% endfor %}''' |
Armin Ronacher | ab45b84 | 2007-03-18 20:47:50 +0100 | [diff] [blame] | 22 | SCOPE = '''{% for item in seq %}{% endfor %}{{ item }}''' |
Armin Ronacher | 6dba4d6 | 2007-05-21 23:41:36 +0200 | [diff] [blame] | 23 | VARLEN = '''{% for item in iter %}{{ item }}{% endfor %}''' |
Armin Ronacher | 9bcd411 | 2007-05-29 14:17:24 +0200 | [diff] [blame] | 24 | NONITER = '''{% for item in none %}...{% endfor %}''' |
Armin Ronacher | 66a9344 | 2008-05-11 23:42:19 +0200 | [diff] [blame] | 25 | RECURSIVE = '''{% for item in seq recursive -%} |
Armin Ronacher | b455c31 | 2008-05-11 23:43:07 +0200 | [diff] [blame] | 26 | [{{ item.a }}{% if item.b %}<{{ loop(item.b) }}>{% endif %}] |
Armin Ronacher | 66a9344 | 2008-05-11 23:42:19 +0200 | [diff] [blame] | 27 | {%- endfor %}''' |
Armin Ronacher | 24db451 | 2008-05-19 09:28:17 +0200 | [diff] [blame] | 28 | LOOPLOOP = '''{% for row in table %} |
| 29 | {%- set rowloop = loop -%} |
| 30 | {% for cell in row -%} |
| 31 | [{{ rowloop.index }}|{{ loop.index }}] |
| 32 | {%- endfor %} |
| 33 | {%- endfor %}''' |
Armin Ronacher | 105f0dc | 2008-05-23 16:12:47 +0200 | [diff] [blame] | 34 | LOOPERROR1 = '''\ |
| 35 | {% for item in [1] if loop.index == 0 %}...{% endfor %}''' |
| 36 | LOOPERROR2 = '''\ |
| 37 | {% for item in [] %}...{% else %}{{ loop }}{% endfor %}''' |
Armin Ronacher | 6df604e | 2008-05-23 22:18:38 +0200 | [diff] [blame] | 38 | LOOPFILTER = '''\ |
| 39 | {% for item in range(10) if item is even %}[{{ item }}]{% endfor %}''' |
| 40 | EXTENDEDLOOPFILTER = '''\ |
| 41 | {% for item in range(10) if item is even %}[{{ loop.index |
| 42 | }}:{{ item }}]{% endfor %}''' |
Armin Ronacher | 9f258ff | 2008-05-24 22:28:52 +0200 | [diff] [blame] | 43 | LOOPUNASSIGNABLE = '''\ |
| 44 | {% for loop in seq %}...{% endfor %}''' |
Armin Ronacher | ab45b84 | 2007-03-18 20:47:50 +0100 | [diff] [blame] | 45 | |
| 46 | |
| 47 | def test_simple(env): |
| 48 | tmpl = env.from_string(SIMPLE) |
| 49 | assert tmpl.render(seq=range(10)) == '0123456789' |
| 50 | |
| 51 | |
| 52 | def test_else(env): |
| 53 | tmpl = env.from_string(ELSE) |
| 54 | assert tmpl.render() == '...' |
| 55 | |
| 56 | |
| 57 | def test_empty_blocks(env): |
| 58 | tmpl = env.from_string(EMPTYBLOCKS) |
| 59 | assert tmpl.render() == '<>' |
| 60 | |
| 61 | |
| 62 | def test_context_vars(env): |
| 63 | tmpl = env.from_string(CONTEXTVARS) |
| 64 | one, two, _ = tmpl.render(seq=[0, 1]).split('###') |
| 65 | (one_index, one_index0, one_revindex, one_revindex0, one_first, |
Armin Ronacher | 10f3ba2 | 2008-04-18 11:30:37 +0200 | [diff] [blame] | 66 | one_last, one_length) = one.split('|') |
Armin Ronacher | ab45b84 | 2007-03-18 20:47:50 +0100 | [diff] [blame] | 67 | (two_index, two_index0, two_revindex, two_revindex0, two_first, |
Armin Ronacher | 10f3ba2 | 2008-04-18 11:30:37 +0200 | [diff] [blame] | 68 | two_last, two_length) = two.split('|') |
Armin Ronacher | ab45b84 | 2007-03-18 20:47:50 +0100 | [diff] [blame] | 69 | |
| 70 | assert int(one_index) == 1 and int(two_index) == 2 |
| 71 | assert int(one_index0) == 0 and int(two_index0) == 1 |
| 72 | assert int(one_revindex) == 2 and int(two_revindex) == 1 |
| 73 | assert int(one_revindex0) == 1 and int(two_revindex0) == 0 |
| 74 | assert one_first == 'True' and two_first == 'False' |
| 75 | assert one_last == 'False' and two_last == 'True' |
Armin Ronacher | ab45b84 | 2007-03-18 20:47:50 +0100 | [diff] [blame] | 76 | assert one_length == two_length == '2' |
| 77 | |
| 78 | |
| 79 | def test_cycling(env): |
| 80 | tmpl = env.from_string(CYCLING) |
| 81 | output = tmpl.render(seq=range(4), through=('<1>', '<2>')) |
| 82 | assert output == '<1><2>' * 4 |
| 83 | |
| 84 | |
| 85 | def test_scope(env): |
| 86 | tmpl = env.from_string(SCOPE) |
| 87 | output = tmpl.render(seq=range(10)) |
| 88 | assert not output |
Armin Ronacher | 6dba4d6 | 2007-05-21 23:41:36 +0200 | [diff] [blame] | 89 | |
| 90 | |
| 91 | def test_varlen(env): |
| 92 | def inner(): |
| 93 | for item in range(5): |
| 94 | yield item |
| 95 | tmpl = env.from_string(VARLEN) |
| 96 | output = tmpl.render(iter=inner()) |
| 97 | assert output == '01234' |
Armin Ronacher | 9bcd411 | 2007-05-29 14:17:24 +0200 | [diff] [blame] | 98 | |
| 99 | |
| 100 | def test_noniter(env): |
| 101 | tmpl = env.from_string(NONITER) |
Armin Ronacher | 10f3ba2 | 2008-04-18 11:30:37 +0200 | [diff] [blame] | 102 | raises(TypeError, tmpl.render) |
Armin Ronacher | 66a9344 | 2008-05-11 23:42:19 +0200 | [diff] [blame] | 103 | |
| 104 | |
| 105 | def test_recursive(env): |
| 106 | tmpl = env.from_string(RECURSIVE) |
| 107 | assert tmpl.render(seq=[ |
| 108 | dict(a=1, b=[dict(a=1), dict(a=2)]), |
| 109 | dict(a=2, b=[dict(a=1), dict(a=2)]), |
| 110 | dict(a=3, b=[dict(a='a')]) |
Armin Ronacher | b455c31 | 2008-05-11 23:43:07 +0200 | [diff] [blame] | 111 | ]) == '[1<[1][2]>][2<[1][2]>][3<[a]>]' |
Armin Ronacher | 24db451 | 2008-05-19 09:28:17 +0200 | [diff] [blame] | 112 | |
| 113 | |
| 114 | def test_looploop(env): |
| 115 | tmpl = env.from_string(LOOPLOOP) |
| 116 | assert tmpl.render(table=['ab', 'cd']) == '[1|1][1|2][2|1][2|2]' |
Armin Ronacher | 105f0dc | 2008-05-23 16:12:47 +0200 | [diff] [blame] | 117 | |
| 118 | |
Armin Ronacher | 547d0b6 | 2008-07-04 16:35:10 +0200 | [diff] [blame] | 119 | def test_reversed_bug(env): |
| 120 | tmpl = env.from_string('{% for i in items %}{{ i }}{% if not loop.last %}' |
| 121 | ',{% endif %}{% endfor %}') |
| 122 | assert tmpl.render(items=reversed([3, 2, 1])) == '1,2,3' |
| 123 | |
| 124 | |
Armin Ronacher | 105f0dc | 2008-05-23 16:12:47 +0200 | [diff] [blame] | 125 | def test_loop_errors(env): |
| 126 | tmpl = env.from_string(LOOPERROR1) |
| 127 | raises(UndefinedError, tmpl.render) |
| 128 | tmpl = env.from_string(LOOPERROR2) |
| 129 | assert tmpl.render() == '' |
Armin Ronacher | 6df604e | 2008-05-23 22:18:38 +0200 | [diff] [blame] | 130 | |
| 131 | |
| 132 | def test_loop_filter(env): |
| 133 | tmpl = env.from_string(LOOPFILTER) |
| 134 | assert tmpl.render() == '[0][2][4][6][8]' |
| 135 | tmpl = env.from_string(EXTENDEDLOOPFILTER) |
| 136 | assert tmpl.render() == '[1:0][2:2][3:4][4:6][5:8]' |
Armin Ronacher | 9f258ff | 2008-05-24 22:28:52 +0200 | [diff] [blame] | 137 | |
| 138 | |
| 139 | def test_loop_unassignable(env): |
| 140 | raises(TemplateSyntaxError, env.from_string, LOOPUNASSIGNABLE) |
Armin Ronacher | ff53c78 | 2008-08-13 18:55:50 +0200 | [diff] [blame] | 141 | |
| 142 | |
| 143 | def test_scoped_special_var(env): |
| 144 | t = env.from_string('{% for s in seq %}[{{ loop.first }}{% for c in s %}' |
| 145 | '|{{ loop.first }}{% endfor %}]{% endfor %}') |
| 146 | assert t.render(seq=('ab', 'cd')) == '[True|True|False][False|True|False]' |
Armin Ronacher | 833a3b5 | 2008-08-14 12:31:12 +0200 | [diff] [blame] | 147 | |
| 148 | |
| 149 | def test_scoped_loop_var(env): |
| 150 | t = env.from_string('{% for x in seq %}{{ loop.first }}' |
| 151 | '{% for y in seq %}{% endfor %}{% endfor %}') |
| 152 | assert t.render(seq='ab') == 'TrueFalse' |
| 153 | t = env.from_string('{% for x in seq %}{% for y in seq %}' |
| 154 | '{{ loop.first }}{% endfor %}{% endfor %}') |
| 155 | assert t.render(seq='ab') == 'TrueFalseTrueFalse' |
Armin Ronacher | cebd838 | 2008-12-25 18:33:46 +0100 | [diff] [blame] | 156 | |
| 157 | |
| 158 | def test_recursive_empty_loop_iter(env): |
| 159 | t = env.from_string(''' |
| 160 | {%- for item in foo recursive -%}{%- endfor -%} |
| 161 | ''') |
| 162 | assert t.render(dict(foo=[])) == '' |
Armin Ronacher | 330fbc0 | 2009-02-04 19:13:58 +0100 | [diff] [blame^] | 163 | |
| 164 | |
| 165 | def test_call_in_loop(env): |
| 166 | t = env.from_string(''' |
| 167 | {%- macro do_something() -%} |
| 168 | [{{ caller() }}] |
| 169 | {%- endmacro %} |
| 170 | |
| 171 | {%- for i in [1, 2, 3] %} |
| 172 | {%- call do_something() -%} |
| 173 | {{ i }} |
| 174 | {%- endcall %} |
| 175 | {%- endfor -%} |
| 176 | ''') |
| 177 | assert t.render() == '[1][2][3]' |