-module(erlydtl_unittests).
-export([run_tests/0]).
tests() ->
[
{"comment", [
{"Comment block is excised",
<<"Bob {% comment %}(moron){% endcomment %} Loblaw">>,
[], <<"Bob Loblaw">>},
{"Inline comment is excised",
<<"You're {# not #} a very nice person">>,
[], <<"You're a very nice person">>}
]},
{"autoescape", [
{"Autoescape works",
<<"{% autoescape on %}{{ var1 }}{% endautoescape %}">>,
[{var1, "bold"}], <<"<b>bold</b>">>},
{"Nested autoescape",
<<"{% autoescape on %}{{ var1 }}{% autoescape off %}{{ var1 }}{% endautoescape %}{% endautoescape %}">>,
[{var1, ""}], <<"<b>">>}
]},
{"string literal", [
{"Render literal",
<<"{{ \"foo\" }} is my name">>, [], <<"foo is my name">>},
{"Newlines are escaped",
<<"{{ \"foo\\n\" }}">>, [], <<"foo\n">>}
]},
{"cycle", [
{"Cycling through quoted strings",
<<"{% for i in test %}{% cycle 'a' 'b' %}{{ i }},{% endfor %}">>,
[{test, ["0", "1", "2", "3", "4"]}], <<"a0,b1,a2,b3,a4,">>},
{"Cycling through normal variables",
<<"{% for i in test %}{% cycle aye bee %}{{ i }},{% endfor %}">>,
[{test, ["0", "1", "2", "3", "4"]}, {aye, "a"}, {bee, "b"}],
<<"a0,b1,a2,b3,a4,">>}
]},
{"number literal", [
{"Render integer",
<<"{{ 5 }}">>, [], <<"5">>}
]},
{"variable", [
{"Render variable",
<<"{{ var1 }} is my game">>, [{var1, "bar"}], <<"bar is my game">>},
{"Render variable with attribute",
<<"I enjoy {{ var1.game }}">>, [{var1, [{game, "Othello"}]}], <<"I enjoy Othello">>},
{"Render variable in dict",
<<"{{ var1 }}">>, dict:store(var1, "bar", dict:new()), <<"bar">>},
{"Render variable in gb_tree",
<<"{{ var1 }}">>, gb_trees:insert(var1, "bar", gb_trees:empty()), <<"bar">>},
{"Render variable with attribute in dict",
<<"{{ var1.attr }}">>, [{var1, dict:store(attr, "Othello", dict:new())}], <<"Othello">>},
{"Render variable with attribute in gb_tree",
<<"{{ var1.attr }}">>, [{var1, gb_trees:insert(attr, "Othello", gb_trees:empty())}], <<"Othello">>},
{"Render variable in parameterized module",
<<"{{ var1.some_var }}">>, [{var1, erlydtl_example_variable_storage:new("foo")}], <<"foo">>},
{"Nested attributes",
<<"{{ person.city.state.country }}">>, [{person, [{city, [{state, [{country, "Italy"}]}]}]}],
<<"Italy">>}
]},
{"now", [
{"now functional",
<<"It is the {% now \"jS o\\f F Y\" %}.">>, [{var1, ""}], generate_test_date()}
]},
{"if", [
{"If/else",
<<"{% if var1 %}boo{% else %}yay{% endif %}">>, [{var1, ""}], <<"yay">>},
{"If",
<<"{% if var1 %}boo{% endif %}">>, [{var1, ""}], <<>>},
{"If not",
<<"{% if not var1 %}yay{% endif %}">>, [{var1, ""}], <<"yay">>},
{"If \"0\"",
<<"{% if var1 %}boo{% endif %}">>, [{var1, "0"}], <<>>},
{"If false",
<<"{% if var1 %}boo{% endif %}">>, [{var1, false}], <<>>},
{"If undefined",
<<"{% if var1 %}boo{% endif %}">>, [{var1, undefined}], <<>>},
{"If other atom",
<<"{% if var1 %}yay{% endif %}">>, [{var1, foobar}], <<"yay">>},
{"If non-empty string",
<<"{% if var1 %}yay{% endif %}">>, [{var1, "hello"}], <<"yay">>},
{"If proplist",
<<"{% if var1 %}yay{% endif %}">>, [{var1, [{foo, "bar"}]}], <<"yay">>}
]},
{"for", [
{"Simple loop",
<<"{% for x in list %}{{ x }}{% endfor %}">>, [{'list', ["1", "2", "3"]}],
<<"123">>},
{"Expand list",
<<"{% for x, y in list %}{{ x }},{{ y }}\n{% endfor %}">>, [{'list', [["X", "1"], ["X", "2"]]}],
<<"X,1\nX,2\n">>},
{"Expand tuple",
<<"{% for x, y in list %}{{ x }},{{ y }}\n{% endfor %}">>, [{'list', [{"X", "1"}, {"X", "2"}]}],
<<"X,1\nX,2\n">>},
{"Resolve variable attribute",
<<"{% for number in person.numbers %}{{ number }}\n{% endfor %}">>, [{person, [{numbers, ["411", "911"]}]}],
<<"411\n911\n">>},
{"Resolve nested variable attribute",
<<"{% for number in person.home.numbers %}{{ number }}\n{% endfor %}">>, [{person, [{home, [{numbers, ["411", "911"]}]}]}],
<<"411\n911\n">>},
{"Counter0",
<<"{% for number in numbers %}{{ forloop.counter0 }}. {{ number }}\n{% endfor %}">>,
[{numbers, ["Zero", "One", "Two"]}], <<"0. Zero\n1. One\n2. Two\n">>},
{"Counter",
<<"{% for number in numbers %}{{ forloop.counter }}. {{ number }}\n{% endfor %}">>,
[{numbers, ["One", "Two", "Three"]}], <<"1. One\n2. Two\n3. Three\n">>},
{"Reverse Counter0",
<<"{% for number in numbers %}{{ forloop.revcounter0 }}. {{ number }}\n{% endfor %}">>,
[{numbers, ["Two", "One", "Zero"]}], <<"2. Two\n1. One\n0. Zero\n">>},
{"Reverse Counter",
<<"{% for number in numbers %}{{ forloop.revcounter }}. {{ number }}\n{% endfor %}">>,
[{numbers, ["Three", "Two", "One"]}], <<"3. Three\n2. Two\n1. One\n">>},
{"Counter \"first\"",
<<"{% for number in numbers %}{% if forloop.first %}{{ number }}{% endif %}{% endfor %}">>,
[{numbers, ["One", "Two", "Three"]}], <<"One">>},
{"Counter \"last\"",
<<"{% for number in numbers %}{% if forloop.last %}{{ number }}{% endif %}{% endfor %}">>,
[{numbers, ["One", "Two", "Three"]}], <<"Three">>},
{"Nested for loop",
<<"{% for outer in list %}{% for inner in outer %}{{ inner }}\n{% endfor %}{% endfor %}">>,
[{'list', [["Al", "Albert"], ["Jo", "Joseph"]]}],
<<"Al\nAlbert\nJo\nJoseph\n">>},
{"Access parent loop counters",
<<"{% for outer in list %}{% for inner in outer %}({{ forloop.parentloop.counter0 }}, {{ forloop.counter0 }})\n{% endfor %}{% endfor %}">>,
[{'list', [["One", "two"], ["One", "two"]]}], <<"(0, 0)\n(0, 1)\n(1, 0)\n(1, 1)\n">>}
]},
{"ifequal", [
{"Compare variable to literal",
<<"{% ifequal var1 \"foo\" %}yay{% endifequal %}">>,
[{var1, "foo"}], <<"yay">>},
{"Compare variable to unequal literal",
<<"{% ifequal var1 \"foo\" %}boo{% endifequal %}">>,
[{var1, "bar"}], <<>>},
{"Compare literal to variable",
<<"{% ifequal \"foo\" var1 %}yay{% endifequal %}">>,
[{var1, "foo"}], <<"yay">>},
{"Compare literal to unequal variable",
<<"{% ifequal \"foo\" var1 %}boo{% endifequal %}">>,
[{var1, "bar"}], <<>>}
]},
{"ifequal/else", [
{"Compare variable to literal",
<<"{% ifequal var1 \"foo\" %}yay{% else %}boo{% endifequal %}">>,
[{var1, "foo"}], <<"yay">>},
{"Compare variable to unequal literal",
<<"{% ifequal var1 \"foo\" %}boo{% else %}yay{% endifequal %}">>,
[{var1, "bar"}], <<"yay">>},
{"Compare literal to variable",
<<"{% ifequal \"foo\" var1 %}yay{% else %}boo{% endifequal %}">>,
[{var1, "foo"}], <<"yay">>},
{"Compare literal to unequal variable",
<<"{% ifequal \"foo\" var1 %}boo{% else %}yay{% endifequal %}">>,
[{var1, "bar"}], <<"yay">>}
]},
{"ifnotequal", [
{"Compare variable to literal",
<<"{% ifnotequal var1 \"foo\" %}boo{% endifnotequal %}">>,
[{var1, "foo"}], <<>>},
{"Compare variable to unequal literal",
<<"{% ifnotequal var1 \"foo\" %}yay{% endifnotequal %}">>,
[{var1, "bar"}], <<"yay">>},
{"Compare literal to variable",
<<"{% ifnotequal \"foo\" var1 %}boo{% endifnotequal %}">>,
[{var1, "foo"}], <<>>},
{"Compare literal to unequal variable",
<<"{% ifnotequal \"foo\" var1 %}yay{% endifnotequal %}">>,
[{var1, "bar"}], <<"yay">>}
]},
{"ifnotequal/else", [
{"Compare variable to literal",
<<"{% ifnotequal var1 \"foo\" %}boo{% else %}yay{% endifnotequal %}">>,
[{var1, "foo"}], <<"yay">>},
{"Compare variable to unequal literal",
<<"{% ifnotequal var1 \"foo\" %}yay{% else %}boo{% endifnotequal %}">>,
[{var1, "bar"}], <<"yay">>},
{"Compare literal to variable",
<<"{% ifnotequal \"foo\" var1 %}boo{% else %}yay{% endifnotequal %}">>,
[{var1, "foo"}], <<"yay">>},
{"Compare literal to unequal variable",
<<"{% ifnotequal \"foo\" var1 %}yay{% else %}boo{% endifnotequal %}">>,
[{var1, "bar"}], <<"yay">>}
]},
{"filters", [
{"Filter a literal",
<<"{{ \"pop\"|capfirst }}">>, [],
<<"Pop">>},
{"Filters applied in order",
<<"{{ var1|force_escape|length }}">>, [{var1, <<"&">>}],
<<"5">>},
{"Escape is applied last",
<<"{{ var1|escape|linebreaksbr }}">>, [{var1, <<"\n">>}],
<<"<br />">>},
{"|add:4",
<<"{{ one|add:4 }}">>, [{one, "1"}],
<<"5">>},
{"|capfirst",
<<"{{ var1|capfirst }}">>, [{var1, "dana boyd"}],
<<"Dana boyd">>},
{"|center:10",
<<"{{ var1|center:10 }}">>, [{var1, "MB"}],
<<" MB ">>},
{"|center:1",
<<"{{ var1|center:1 }}">>, [{var1, "KBR"}],
<<"B">>},
{"|date 1",
<<"{{ var1|date:\"r\" }}">>,
[{var1, {1975,7,24}}],
<<"Thu, 24 Jul 1975 00:00:00 +0000">>},
{"|date 2",
<<"{{ var1|date:\"r\" }}">>,
[{var1, {{1975,7,24}, {7,13,1}}}],
<<"Thu, 24 Jul 1975 07:13:01 +0000">>},
{"|escapejs",
<<"{{ var1|escapejs }}">>, [{var1, "Skip's \"Old-Timey\" Diner"}],
<<"Skip\\'s \\\"Old-Timey\\\" Diner">>},
{"|first",
<<"{{ var1|first }}">>, [{var1, "James"}],
<<"J">>},
{"|fix_ampersands",
<<"{{ var1|fix_ampersands }}">>, [{var1, "Ben & Jerry's"}],
<<"Ben & Jerry's">>},
{"|force_escape",
<<"{{ var1|force_escape }}">>, [{var1, "Ben & Jerry's <=> \"The World's Best Ice Cream\""}],
<<"Ben & Jerry's <=> "The World's Best Ice Cream"">>},
{"|format_integer",
<<"{{ var1|format_integer }}">>, [{var1, 28}], <<"28">>},
{"|join:\", \"",
<<"{{ var1|join:\", \" }}">>, [{var1, ["Liberte", "Egalite", "Fraternite"]}],
<<"Liberte, Egalite, Fraternite">>},
{"|last",
<<"{{ var1|last }}">>, [{var1, "XYZ"}],
<<"Z">>},
{"|length",
<<"{{ var1|length }}">>, [{var1, "antidisestablishmentarianism"}],
<<"28">>},
{"|linebreaksbr",
<<"{{ var1|linebreaksbr }}">>, [{var1, "One\nTwo\n\nThree\n\n\n"}],
<<"One
Two
Three
">>},
{"|linebreaksbr",
<<"{{ \"One\\nTwo\\n\\nThree\\n\\n\\n\"|linebreaksbr }}">>, [],
<<"One
Two
Three
">>},
{"|ljust:10",
<<"{{ var1|ljust:10 }}">>, [{var1, "Gore"}],
<<"Gore ">>},
{"|lower",
<<"{{ var1|lower }}">>, [{var1, "E. E. Cummings"}],
<<"e. e. cummings">>},
{"|rjust:10",
<<"{{ var1|rjust:10 }}">>, [{var1, "Bush"}],
<<" Bush">>},
{"|upper",
<<"{{ message|upper }}">>, [{message, "That man has a gun."}],
<<"THAT MAN HAS A GUN.">>},
{"|urlencode",
<<"{{ url|urlencode }}">>, [{url, "You #$*@!!"}],
<<"You+%23%24%2A%40%21%21">>}
]}
].
run_tests() ->
Failures = lists:foldl(
fun({Group, Assertions}, GroupAcc) ->
io:format("Running test group ~p...~n", [Group]),
lists:foldl(fun({Name, DTL, Vars, Output}, Acc) ->
case erlydtl_compiler:compile(DTL, erlydtl_running_test, []) of
{ok, _} ->
{ok, IOList} = erlydtl_running_test:render(Vars),
{ok, IOListBin} = erlydtl_running_test:render(vars_to_binary(Vars)),
case {iolist_to_binary(IOList), iolist_to_binary(IOListBin)} of
{Output, Output} ->
Acc;
{Output, Unexpected} ->
[{Group, Name, 'binary', Unexpected, Output} | Acc];
{Unexpected, Output} ->
[{Group, Name, 'list', Unexpected, Output} | Acc];
{Unexpected1, Unexpected2} ->
[{Group, Name, 'list', Unexpected1, Output},
{Group, Name, 'binary', Unexpected2, Output} | Acc]
end;
Err ->
[{Group, Name, Err} | Acc]
end
end, GroupAcc, Assertions)
end, [], tests()),
io:format("Failures: ~p~n", [Failures]),
erlang:halt().
vars_to_binary(Vars) when is_list(Vars) ->
lists:map(fun
({Key, [H|_] = Value}) when is_tuple(H) ->
{Key, vars_to_binary(Value)};
({Key, [H|_] = Value}) when is_integer(H) ->
{Key, list_to_binary(Value)};
({Key, Value}) ->
{Key, Value}
end, Vars);
vars_to_binary(Vars) ->
Vars.
generate_test_date() ->
{{Y,M,D}, _} = erlang:localtime(),
MonthName = [
"January", "February", "March", "April",
"May", "June", "July", "August", "September",
"October", "November", "December"
],
OrdinalSuffix = [
"st","nd","rd","th","th","th","th","th","th","th", % 1-10
"th","th","th","th","th","th","th","th","th","th", % 10-20
"st","nd","rd","th","th","th","th","th","th","th", % 20-30
"st"
],
list_to_binary([
"It is the ",
integer_to_list(D),
lists:nth(D, OrdinalSuffix),
" of ", lists:nth(M, MonthName),
" ", integer_to_list(Y), "."
]).