erlydtl_compiler.erl 71 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509
  1. %%%-------------------------------------------------------------------
  2. %%% File: erlydtl_compiler.erl
  3. %%% @author Roberto Saccon <rsaccon@gmail.com> [http://rsaccon.com]
  4. %%% @author Evan Miller <emmiller@gmail.com>
  5. %%% @author Andreas Stenius <kaos@astekk.se>
  6. %%% @copyright 2008 Roberto Saccon, Evan Miller
  7. %%% @doc
  8. %%% ErlyDTL template compiler
  9. %%% @end
  10. %%%
  11. %%% The MIT License
  12. %%%
  13. %%% Copyright (c) 2007 Roberto Saccon, Evan Miller
  14. %%%
  15. %%% Permission is hereby granted, free of charge, to any person obtaining a copy
  16. %%% of this software and associated documentation files (the "Software"), to deal
  17. %%% in the Software without restriction, including without limitation the rights
  18. %%% to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  19. %%% copies of the Software, and to permit persons to whom the Software is
  20. %%% furnished to do so, subject to the following conditions:
  21. %%%
  22. %%% The above copyright notice and this permission notice shall be included in
  23. %%% all copies or substantial portions of the Software.
  24. %%%
  25. %%% THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  26. %%% IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  27. %%% FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  28. %%% AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  29. %%% LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  30. %%% OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  31. %%% THE SOFTWARE.
  32. %%%
  33. %%% @since 2007-12-16 by Roberto Saccon, Evan Miller
  34. %%%-------------------------------------------------------------------
  35. -module(erlydtl_compiler).
  36. -author('rsaccon@gmail.com').
  37. -author('emmiller@gmail.com').
  38. -author('Andreas Stenius <kaos@astekk.se>').
  39. %% --------------------------------------------------------------------
  40. %% Definitions
  41. %% --------------------------------------------------------------------
  42. -export([compile/2, compile/3, compile_dir/2, compile_dir/3, parse/1]).
  43. %% exported for use by extension modules
  44. -export([
  45. merge_info/2,
  46. format/3,
  47. value_ast/5,
  48. resolve_scoped_variable_ast/2
  49. ]).
  50. -include("erlydtl_ext.hrl").
  51. compile(Binary, Module) when is_binary(Binary) ->
  52. compile(Binary, Module, []);
  53. compile(File, Module) ->
  54. compile(File, Module, []).
  55. compile(Binary, Module, Options) when is_binary(Binary) ->
  56. File = "",
  57. CheckSum = "",
  58. Context = init_dtl_context(File, Module, Options),
  59. case parse(Binary, Context) of
  60. {ok, DjangoParseTree} ->
  61. case compile_to_binary(File, DjangoParseTree, Context, CheckSum) of
  62. {ok, Module1, _, _} ->
  63. {ok, Module1};
  64. Err ->
  65. Err
  66. end;
  67. Err ->
  68. Err
  69. end;
  70. compile(File, Module, Options) ->
  71. Context = init_dtl_context(File, Module, Options),
  72. case parse(File, Context) of
  73. ok ->
  74. ok;
  75. {ok, DjangoParseTree, CheckSum} ->
  76. case compile_to_binary(File, DjangoParseTree, Context, CheckSum) of
  77. {ok, Module1, Bin, Warnings} ->
  78. write_binary(Module1, Bin, Options, Warnings);
  79. Err ->
  80. Err
  81. end;
  82. Err ->
  83. Err
  84. end.
  85. compile_dir(Dir, Module) ->
  86. compile_dir(Dir, Module, []).
  87. compile_dir(Dir, Module, Options) ->
  88. Context = init_dtl_context_dir(Dir, Module, Options),
  89. %% Find all files in Dir (recursively), matching the regex (no
  90. %% files ending in "~").
  91. Files = filelib:fold_files(Dir, ".+[^~]$", true, fun(F1,Acc1) -> [F1 | Acc1] end, []),
  92. {ParserResults, ParserErrors} = lists:foldl(fun
  93. (File, {ResultAcc, ErrorAcc}) ->
  94. case filename:basename(File) of
  95. "."++_ ->
  96. {ResultAcc, ErrorAcc};
  97. _ ->
  98. FilePath = filename:absname(File),
  99. case filelib:is_dir(FilePath) of
  100. true ->
  101. {ResultAcc, ErrorAcc};
  102. false ->
  103. case parse(FilePath, Context) of
  104. ok -> {ResultAcc, ErrorAcc};
  105. {ok, DjangoParseTree, CheckSum} ->
  106. {[{File, DjangoParseTree, CheckSum}|ResultAcc], ErrorAcc};
  107. Err -> {ResultAcc, [Err|ErrorAcc]}
  108. end
  109. end
  110. end
  111. end, {[], []}, Files),
  112. case ParserErrors of
  113. [] ->
  114. case compile_multiple_to_binary(Dir, ParserResults, Context) of
  115. {ok, Module1, Bin, Warnings} ->
  116. write_binary(Module1, Bin, Options, Warnings);
  117. Err ->
  118. Err
  119. end;
  120. [Error|_] ->
  121. Error
  122. end.
  123. %%====================================================================
  124. %% Internal functions
  125. %%====================================================================
  126. write_binary(Module1, Bin, Options, Warnings) ->
  127. Verbose = proplists:get_value(verbose, Options, false),
  128. case proplists:get_value(out_dir, Options) of
  129. undefined ->
  130. print(Verbose, "Template module: ~w not saved (no out_dir option)\n", [Module1]),
  131. ok;
  132. OutDir ->
  133. BeamFile = filename:join([OutDir, atom_to_list(Module1) ++ ".beam"]),
  134. print(Verbose, "Template module: ~w -> ~s~s\n",
  135. [Module1, BeamFile,
  136. case Warnings of
  137. [] -> "";
  138. _ -> io_lib:format("\n Warnings: ~p", [Warnings])
  139. end]),
  140. case file:write_file(BeamFile, Bin) of
  141. ok ->
  142. ok;
  143. {error, Reason} ->
  144. {error, lists:flatten(
  145. io_lib:format("Beam generation of '~s' failed: ~p",
  146. [BeamFile, file:format_error(Reason)]))}
  147. end
  148. end.
  149. compile_multiple_to_binary(Dir, ParserResults, Context) ->
  150. MatchAst = options_match_ast(Context),
  151. {Functions, {AstInfo, _}} = lists:mapfoldl(fun({File, DjangoParseTree, CheckSum}, {AstInfo, TreeWalker}) ->
  152. FilePath = full_path(File, Context#dtl_context.doc_root),
  153. {{BodyAst, BodyInfo}, TreeWalker1} = with_dependency({FilePath, CheckSum}, body_ast(DjangoParseTree, Context, TreeWalker)),
  154. FunctionName = filename:rootname(filename:basename(File)),
  155. Function1 = erl_syntax:function(erl_syntax:atom(FunctionName),
  156. [erl_syntax:clause([erl_syntax:variable("_Variables")], none,
  157. [erl_syntax:application(none, erl_syntax:atom(FunctionName),
  158. [erl_syntax:variable("_Variables"), erl_syntax:list([])])])]),
  159. Function2 = erl_syntax:function(erl_syntax:atom(FunctionName),
  160. [erl_syntax:clause([erl_syntax:variable("_Variables"), erl_syntax:variable("RenderOptions")], none,
  161. MatchAst ++ [BodyAst])]),
  162. {{FunctionName, Function1, Function2}, {merge_info(AstInfo, BodyInfo), TreeWalker1}}
  163. end, {#ast_info{}, init_treewalker(Context)}, ParserResults),
  164. Forms = custom_forms(Dir, Context#dtl_context.module, Functions, AstInfo),
  165. compile_forms_and_reload(Dir, Forms, Context).
  166. compile_to_binary(File, DjangoParseTree, Context, CheckSum) ->
  167. try body_ast(DjangoParseTree, Context, init_treewalker(Context)) of
  168. {{BodyAst, BodyInfo}, BodyTreeWalker} ->
  169. try custom_tags_ast(BodyInfo#ast_info.custom_tags, Context, BodyTreeWalker) of
  170. {{CustomTagsAst, CustomTagsInfo}, _} ->
  171. Forms = forms(File, Context#dtl_context.module, {BodyAst, BodyInfo},
  172. {CustomTagsAst, CustomTagsInfo}, CheckSum, Context, BodyTreeWalker),
  173. compile_forms_and_reload(File, Forms, Context)
  174. catch
  175. throw:Error -> Error
  176. end
  177. catch
  178. throw:Error -> Error
  179. end.
  180. compile_forms_and_reload(File, Forms, Context) ->
  181. case proplists:get_value(debug_compiler, Context#dtl_context.compiler_options) of
  182. true ->
  183. io:format("template source:~n~s~n",
  184. [try
  185. erl_prettypr:format(erl_syntax:form_list(Forms))
  186. catch
  187. error:Err ->
  188. io_lib:format("Pretty printing failed: ~p~nContext: ~n~p~nForms: ~n~p~n",
  189. [Err, Context, Forms])
  190. end
  191. ]);
  192. _ -> nop
  193. end,
  194. case compile:forms(Forms, Context#dtl_context.compiler_options) of
  195. {ok, Module1, Bin} ->
  196. load_code(Module1, Bin, []);
  197. {ok, Module1, Bin, Warnings} ->
  198. load_code(Module1, Bin, Warnings);
  199. error ->
  200. {error, lists:concat(["compilation failed: ", File])};
  201. OtherError ->
  202. OtherError
  203. end.
  204. load_code(Module, Bin, Warnings) ->
  205. code:purge(Module),
  206. case code:load_binary(Module, atom_to_list(Module) ++ ".erl", Bin) of
  207. {module, _} -> {ok, Module, Bin, Warnings};
  208. _ -> {error, lists:concat(["code reload failed: ", Module])}
  209. end.
  210. init_context(IsCompilingDir, ParseTrail, DefDir, Module, Options) ->
  211. Ctx = #dtl_context{},
  212. Context = #dtl_context{
  213. parse_trail = ParseTrail,
  214. module = Module,
  215. doc_root = proplists:get_value(doc_root, Options, DefDir),
  216. filter_modules = proplists:get_value(custom_filters_modules, Options, Ctx#dtl_context.filter_modules) ++ [erlydtl_filters],
  217. custom_tags_dir = proplists:get_value(custom_tags_dir, Options, filename:join([erlydtl_deps:get_base_dir(), "priv", "custom_tags"])),
  218. custom_tags_modules = proplists:get_value(custom_tags_modules, Options, Ctx#dtl_context.custom_tags_modules),
  219. blocktrans_fun = proplists:get_value(blocktrans_fun, Options, Ctx#dtl_context.blocktrans_fun),
  220. blocktrans_locales = proplists:get_value(blocktrans_locales, Options, Ctx#dtl_context.blocktrans_locales),
  221. vars = proplists:get_value(vars, Options, Ctx#dtl_context.vars),
  222. reader = proplists:get_value(reader, Options, Ctx#dtl_context.reader),
  223. compiler_options = proplists:get_value(compiler_options, Options, Ctx#dtl_context.compiler_options),
  224. binary_strings = proplists:get_value(binary_strings, Options, Ctx#dtl_context.binary_strings),
  225. force_recompile = proplists:get_value(force_recompile, Options, Ctx#dtl_context.force_recompile),
  226. locale = proplists:get_value(locale, Options, Ctx#dtl_context.locale),
  227. verbose = proplists:get_value(verbose, Options, Ctx#dtl_context.verbose),
  228. is_compiling_dir = IsCompilingDir,
  229. extension_module = proplists:get_value(extension_module, Options, Ctx#dtl_context.extension_module)
  230. },
  231. case call_extension(Context, init_context, [Context]) of
  232. {ok, C} when is_record(C, dtl_context) -> C;
  233. undefined -> Context
  234. end.
  235. init_dtl_context(File, Module, Options) when is_list(Module) ->
  236. init_dtl_context(File, list_to_atom(Module), Options);
  237. init_dtl_context(File, Module, Options) ->
  238. init_context(false, [File], filename:dirname(File), Module, Options).
  239. init_dtl_context_dir(Dir, Module, Options) when is_list(Module) ->
  240. init_dtl_context_dir(Dir, list_to_atom(Module), Options);
  241. init_dtl_context_dir(Dir, Module, Options) ->
  242. init_context(true, [], Dir, Module, Options).
  243. init_treewalker(Context) ->
  244. TreeWalker = #treewalker{},
  245. case call_extension(Context, init_treewalker, [TreeWalker]) of
  246. {ok, TW} when is_record(TW, treewalker) -> TW;
  247. undefined -> TreeWalker
  248. end.
  249. is_up_to_date(_, #dtl_context{force_recompile = true}) ->
  250. false;
  251. is_up_to_date(CheckSum, Context) ->
  252. Module = Context#dtl_context.module,
  253. {M, F} = Context#dtl_context.reader,
  254. case catch Module:source() of
  255. {_, CheckSum} ->
  256. case catch Module:dependencies() of
  257. L when is_list(L) ->
  258. RecompileList = lists:foldl(fun
  259. ({XFile, XCheckSum}, Acc) ->
  260. case catch M:F(XFile) of
  261. {ok, Data} ->
  262. case binary_to_list(erlang:md5(Data)) of
  263. XCheckSum ->
  264. Acc;
  265. _ ->
  266. [recompile | Acc]
  267. end;
  268. _ ->
  269. [recompile | Acc]
  270. end
  271. end, [], L),
  272. case RecompileList of
  273. [] -> true;
  274. _ -> false
  275. end;
  276. _ ->
  277. false
  278. end;
  279. _ ->
  280. false
  281. end.
  282. parse(Data) ->
  283. parse(Data, #dtl_context{}).
  284. parse(Data, Context) when is_binary(Data) ->
  285. check_scan(erlydtl_scanner:scan(binary_to_list(Data)), Context);
  286. parse(File, Context) ->
  287. {M, F} = Context#dtl_context.reader,
  288. case catch M:F(File) of
  289. {ok, Data} ->
  290. CheckSum = binary_to_list(erlang:md5(Data)),
  291. case parse(CheckSum, Data, Context) of
  292. {error, Msg} when is_list(Msg) ->
  293. {error, File ++ ": " ++ Msg};
  294. {error, Msg} ->
  295. {error, {File, [Msg]}};
  296. Result ->
  297. Result
  298. end;
  299. _ ->
  300. {error, {File, [{0, Context#dtl_context.module, "Failed to read file"}]}}
  301. end.
  302. parse(CheckSum, Data, Context) ->
  303. case is_up_to_date(CheckSum, Context) of
  304. true ->
  305. ok;
  306. _ ->
  307. case parse(Data, Context) of
  308. {ok, Val} ->
  309. {ok, Val, CheckSum};
  310. Err ->
  311. Err
  312. end
  313. end.
  314. call_extension(#dtl_context{ extension_module=undefined }, _Fun, _Args) ->
  315. undefined;
  316. call_extension(#dtl_context{ extension_module=Mod }, Fun, Args)
  317. when is_atom(Mod), is_atom(Fun), is_list(Args) ->
  318. M = case code:is_loaded(Mod) of
  319. false ->
  320. case code:load_file(Mod) of
  321. {module, Mod} ->
  322. Mod;
  323. _ ->
  324. undefined
  325. end;
  326. _ -> Mod
  327. end,
  328. if M /= undefined ->
  329. case erlang:function_exported(M, Fun, length(Args)) of
  330. true ->
  331. apply(M, Fun, Args);
  332. false ->
  333. undefined
  334. end;
  335. true ->
  336. undefined
  337. end.
  338. check_scan({ok, Tokens}, Context) ->
  339. Tokens1 = case call_extension(Context, post_scan, [Tokens]) of
  340. undefined -> Tokens;
  341. {ok, T} -> T
  342. end,
  343. check_parse(erlydtl_parser:parse(Tokens1), [], Context#dtl_context{ scanned_tokens=Tokens1 });
  344. check_scan({error, Err, State}, Context) ->
  345. case call_extension(Context, scan, [State]) of
  346. undefined ->
  347. {error, Err};
  348. {ok, NewState} ->
  349. check_scan(erlydtl_scanner:resume(NewState), Context);
  350. ExtRes ->
  351. ExtRes
  352. end.
  353. check_parse({ok, _}=Ok, [], _Context) -> Ok;
  354. check_parse({ok, _, _}=Ok, [], _Context) -> Ok;
  355. check_parse({ok, Parsed}, Acc, _Context) -> {ok, Acc ++ Parsed};
  356. check_parse({ok, Parsed, C}, Acc, _Context) -> {ok, Acc ++ Parsed, C};
  357. check_parse({error, _}=Err, _, _Context) -> Err;
  358. check_parse({error, Err, State}, Acc, Context) ->
  359. {State1, Parsed} = reset_parse_state(State, Context),
  360. case call_extension(Context, parse, [State1]) of
  361. undefined ->
  362. {error, Err};
  363. {ok, ExtParsed} ->
  364. {ok, Acc ++ Parsed ++ ExtParsed};
  365. {error, ExtErr, ExtState} ->
  366. case reset_parse_state(ExtState, Context) of
  367. {_, []} ->
  368. %% todo: see if this is indeed a sensible ext error,
  369. %% or if we should rather present the original Err message
  370. {error, ExtErr};
  371. {State2, ExtParsed} ->
  372. check_parse(erlydtl_parser:resume(State2), Acc ++ Parsed ++ ExtParsed, Context)
  373. end;
  374. ExtRes ->
  375. ExtRes
  376. end.
  377. %% backtrack up to the nearest opening tag, and keep the value stack parsed ok so far
  378. reset_parse_state([[{Tag, _, _}|_]=Ts, Tzr, _, _, Stack], Context)
  379. when Tag==open_tag; Tag==open_var ->
  380. %% reached opening tag, so the stack should be sensible here
  381. {[reset_token_stream(Ts, Context#dtl_context.scanned_tokens),
  382. Tzr, 0, [], []], lists:flatten(Stack)};
  383. reset_parse_state([_, _, 0, [], []]=State, _Context) ->
  384. %% top of (empty) stack
  385. {State, []};
  386. reset_parse_state([Ts, Tzr, _, [0 | []], [Parsed | []]], Context)
  387. when is_list(Parsed) ->
  388. %% top of good stack
  389. {[reset_token_stream(Ts, Context#dtl_context.scanned_tokens),
  390. Tzr, 0, [], []], Parsed};
  391. reset_parse_state([Ts, Tzr, _, [S | Ss], [T | Stack]], Context) ->
  392. %% backtrack...
  393. reset_parse_state([[T|Ts], Tzr, S, Ss, Stack], Context).
  394. reset_token_stream([T|_], [T|Ts]) -> [T|Ts];
  395. reset_token_stream(Ts, [_|S]) ->
  396. reset_token_stream(Ts, S).
  397. %% we should find the next token in the list of scanned tokens, or something is real fishy
  398. custom_tags_ast(CustomTags, Context, TreeWalker) ->
  399. {{CustomTagsClauses, CustomTagsInfo}, TreeWalker1} = custom_tags_clauses_ast(CustomTags, Context, TreeWalker),
  400. %% This doesn't work since erl_syntax:revert/1 chokes on the airity_qualifier in the -compile attribute..
  401. %% bug report sent to the erlang-bugs mailing list.
  402. %% {{erl_syntax:form_list(
  403. %% [erl_syntax:attribute(
  404. %% erl_syntax:atom(compile),
  405. %% [erl_syntax:tuple(
  406. %% [erl_syntax:atom(nowarn_unused_function),
  407. %% erl_syntax:arity_qualifier(
  408. %% erl_syntax:atom(render_tag),
  409. %% erl_syntax:integer(3))])]),
  410. %% erl_syntax:function(erl_syntax:atom(render_tag), CustomTagsClauses)]),
  411. {{erl_syntax:function(erl_syntax:atom(render_tag), CustomTagsClauses),
  412. CustomTagsInfo},
  413. TreeWalker1}.
  414. custom_tags_clauses_ast(CustomTags, Context, TreeWalker) ->
  415. custom_tags_clauses_ast1(CustomTags, [], [], #ast_info{}, Context, TreeWalker).
  416. custom_tags_clauses_ast1([], _ExcludeTags, ClauseAcc, InfoAcc, Context, TreeWalker) ->
  417. {{DefaultAst, DefaultInfo}, TreeWalker1} =
  418. case call_extension(Context, custom_tag_ast, [Context, TreeWalker]) of
  419. undefined ->
  420. {{erl_syntax:clause(
  421. [erl_syntax:variable("_TagName"), erl_syntax:underscore(), erl_syntax:underscore()],
  422. none,
  423. [erl_syntax:list([])]),
  424. InfoAcc},
  425. TreeWalker};
  426. {{ExtAst, ExtInfo}, ExtTreeWalker} ->
  427. Clause = erl_syntax:clause(
  428. [erl_syntax:variable("TagName"), erl_syntax:variable("_Variables"), erl_syntax:variable("RenderOptions")],
  429. none, options_match_ast(Context, ExtTreeWalker) ++ [ExtAst]),
  430. {{Clause, merge_info(ExtInfo, InfoAcc)}, ExtTreeWalker}
  431. end,
  432. {{lists:reverse([DefaultAst|ClauseAcc]), DefaultInfo}, TreeWalker1};
  433. custom_tags_clauses_ast1([Tag|CustomTags], ExcludeTags, ClauseAcc, InfoAcc, Context, TreeWalker) ->
  434. case lists:member(Tag, ExcludeTags) of
  435. true ->
  436. custom_tags_clauses_ast1(CustomTags, ExcludeTags, ClauseAcc, InfoAcc, Context, TreeWalker);
  437. false ->
  438. CustomTagFile = full_path(Tag, Context#dtl_context.custom_tags_dir),
  439. case filelib:is_file(CustomTagFile) of
  440. true ->
  441. case parse(CustomTagFile, Context) of
  442. {ok, DjangoParseTree, CheckSum} ->
  443. {{BodyAst, BodyAstInfo}, TreeWalker1} = with_dependency(
  444. {CustomTagFile, CheckSum}, body_ast(DjangoParseTree, Context, TreeWalker)),
  445. MatchAst = options_match_ast(Context, TreeWalker),
  446. Clause = erl_syntax:clause(
  447. [erl_syntax:atom(Tag), erl_syntax:variable("_Variables"), erl_syntax:variable("RenderOptions")],
  448. none, MatchAst ++ [BodyAst]),
  449. custom_tags_clauses_ast1(CustomTags, [Tag|ExcludeTags],
  450. [Clause|ClauseAcc], merge_info(BodyAstInfo, InfoAcc),
  451. Context, TreeWalker1);
  452. Error ->
  453. throw(Error)
  454. end;
  455. false ->
  456. case call_extension(Context, custom_tag_ast, [Tag, Context, TreeWalker]) of
  457. undefined ->
  458. custom_tags_clauses_ast1(
  459. CustomTags, [Tag | ExcludeTags],
  460. ClauseAcc, InfoAcc, Context, TreeWalker);
  461. {{Ast, Info}, TW} ->
  462. Clause = erl_syntax:clause(
  463. [erl_syntax:atom(Tag), erl_syntax:variable("_Variables"), erl_syntax:variable("RenderOptions")],
  464. none, options_match_ast(Context, TW) ++ [Ast]),
  465. custom_tags_clauses_ast1(
  466. CustomTags, [Tag | ExcludeTags],
  467. [Clause|ClauseAcc], merge_info(Info, InfoAcc),
  468. Context, TW)
  469. end
  470. end
  471. end.
  472. dependencies_function(Dependencies) ->
  473. erl_syntax:function(
  474. erl_syntax:atom(dependencies), [erl_syntax:clause([], none,
  475. [erl_syntax:list(lists:map(fun
  476. ({XFile, XCheckSum}) ->
  477. erl_syntax:tuple([erl_syntax:string(XFile), erl_syntax:string(XCheckSum)])
  478. end, Dependencies))])]).
  479. translatable_strings_function(TranslatableStrings) ->
  480. erl_syntax:function(
  481. erl_syntax:atom(translatable_strings), [erl_syntax:clause([], none,
  482. [erl_syntax:list(lists:map(fun(String) ->
  483. erl_syntax:string(String)
  484. end,
  485. TranslatableStrings))])]).
  486. translated_blocks_function(TranslatedBlocks) ->
  487. erl_syntax:function(
  488. erl_syntax:atom(translated_blocks), [erl_syntax:clause([], none,
  489. [erl_syntax:list(lists:map(fun(String) ->
  490. erl_syntax:string(String)
  491. end,
  492. TranslatedBlocks))])]).
  493. variables_function(Variables) ->
  494. erl_syntax:function(
  495. erl_syntax:atom(variables), [erl_syntax:clause([], none,
  496. [erl_syntax:list([erl_syntax:atom(S) || S <- lists:usort(Variables)])])]).
  497. custom_forms(Dir, Module, Functions, AstInfo) ->
  498. ModuleAst = erl_syntax:attribute(erl_syntax:atom(module), [erl_syntax:atom(Module)]),
  499. ExportAst = erl_syntax:attribute(erl_syntax:atom(export),
  500. [erl_syntax:list([
  501. erl_syntax:arity_qualifier(erl_syntax:atom(source_dir), erl_syntax:integer(0)),
  502. erl_syntax:arity_qualifier(erl_syntax:atom(dependencies), erl_syntax:integer(0)),
  503. erl_syntax:arity_qualifier(erl_syntax:atom(translatable_strings), erl_syntax:integer(0))
  504. |
  505. lists:foldl(fun({FunctionName, _, _}, Acc) ->
  506. [erl_syntax:arity_qualifier(erl_syntax:atom(FunctionName), erl_syntax:integer(1)),
  507. erl_syntax:arity_qualifier(erl_syntax:atom(FunctionName), erl_syntax:integer(2))|Acc]
  508. end, [], Functions)]
  509. )]),
  510. SourceFunctionAst = erl_syntax:function(
  511. erl_syntax:atom(source_dir), [erl_syntax:clause([], none, [erl_syntax:string(Dir)])]),
  512. DependenciesFunctionAst = dependencies_function(AstInfo#ast_info.dependencies),
  513. TranslatableStringsFunctionAst = translatable_strings_function(AstInfo#ast_info.translatable_strings),
  514. FunctionAsts = lists:foldl(fun({_, Function1, Function2}, Acc) -> [Function1, Function2 | Acc] end, [], Functions),
  515. [erl_syntax:revert(X) || X <- [ModuleAst, ExportAst, SourceFunctionAst, DependenciesFunctionAst, TranslatableStringsFunctionAst
  516. | FunctionAsts] ++ AstInfo#ast_info.pre_render_asts].
  517. forms(File, Module, {BodyAst, BodyInfo}, {CustomTagsFunctionAst, CustomTagsInfo}, CheckSum, Context, TreeWalker) ->
  518. MergedInfo = merge_info(BodyInfo, CustomTagsInfo),
  519. Render0FunctionAst = erl_syntax:function(erl_syntax:atom(render),
  520. [erl_syntax:clause([], none, [erl_syntax:application(none,
  521. erl_syntax:atom(render), [erl_syntax:list([])])])]),
  522. Render1FunctionAst = erl_syntax:function(erl_syntax:atom(render),
  523. [erl_syntax:clause([erl_syntax:variable("_Variables")], none,
  524. [erl_syntax:application(none,
  525. erl_syntax:atom(render),
  526. [erl_syntax:variable("_Variables"), erl_syntax:list([])])])]),
  527. Function2 = erl_syntax:application(none, erl_syntax:atom(render_internal),
  528. [erl_syntax:variable("_Variables"), erl_syntax:variable("RenderOptions")]),
  529. ClauseOk = erl_syntax:clause([erl_syntax:variable("Val")], none,
  530. [erl_syntax:tuple([erl_syntax:atom(ok), erl_syntax:variable("Val")])]),
  531. ClauseCatch = erl_syntax:clause([erl_syntax:variable("Err")], none,
  532. [erl_syntax:tuple([erl_syntax:atom(error), erl_syntax:variable("Err")])]),
  533. Render2FunctionAst = erl_syntax:function(erl_syntax:atom(render),
  534. [erl_syntax:clause([erl_syntax:variable("_Variables"),
  535. erl_syntax:variable("RenderOptions")], none,
  536. [erl_syntax:try_expr([Function2], [ClauseOk], [ClauseCatch])])]),
  537. SourceFunctionTuple = erl_syntax:tuple(
  538. [erl_syntax:string(File), erl_syntax:string(CheckSum)]),
  539. SourceFunctionAst = erl_syntax:function(
  540. erl_syntax:atom(source),
  541. [erl_syntax:clause([], none, [SourceFunctionTuple])]),
  542. DependenciesFunctionAst = dependencies_function(MergedInfo#ast_info.dependencies),
  543. TranslatableStringsAst = translatable_strings_function(MergedInfo#ast_info.translatable_strings),
  544. TranslatedBlocksAst = translated_blocks_function(MergedInfo#ast_info.translated_blocks),
  545. VariablesAst = variables_function(MergedInfo#ast_info.var_names),
  546. MatchAst = options_match_ast(Context, TreeWalker),
  547. BodyAstTmp = MatchAst ++ [
  548. erl_syntax:application(
  549. erl_syntax:atom(erlydtl_runtime),
  550. erl_syntax:atom(stringify_final),
  551. [BodyAst, erl_syntax:atom(Context#dtl_context.binary_strings)])
  552. ],
  553. RenderInternalFunctionAst = erl_syntax:function(
  554. erl_syntax:atom(render_internal),
  555. [erl_syntax:clause([
  556. erl_syntax:variable("_Variables"),
  557. erl_syntax:variable("RenderOptions")],
  558. none, BodyAstTmp)]
  559. ),
  560. ModuleAst = erl_syntax:attribute(erl_syntax:atom(module), [erl_syntax:atom(Module)]),
  561. ExportAst = erl_syntax:attribute(erl_syntax:atom(export),
  562. [erl_syntax:list([erl_syntax:arity_qualifier(erl_syntax:atom(render), erl_syntax:integer(0)),
  563. erl_syntax:arity_qualifier(erl_syntax:atom(render), erl_syntax:integer(1)),
  564. erl_syntax:arity_qualifier(erl_syntax:atom(render), erl_syntax:integer(2)),
  565. erl_syntax:arity_qualifier(erl_syntax:atom(source), erl_syntax:integer(0)),
  566. erl_syntax:arity_qualifier(erl_syntax:atom(dependencies), erl_syntax:integer(0)),
  567. erl_syntax:arity_qualifier(erl_syntax:atom(translatable_strings), erl_syntax:integer(0)),
  568. erl_syntax:arity_qualifier(erl_syntax:atom(translated_blocks), erl_syntax:integer(0)),
  569. erl_syntax:arity_qualifier(erl_syntax:atom(variables), erl_syntax:integer(0))
  570. ])]),
  571. erl_syntax:revert_forms(
  572. erl_syntax:form_list(
  573. [ModuleAst, ExportAst, Render0FunctionAst, Render1FunctionAst, Render2FunctionAst,
  574. SourceFunctionAst, DependenciesFunctionAst, TranslatableStringsAst,
  575. TranslatedBlocksAst, VariablesAst, RenderInternalFunctionAst,
  576. CustomTagsFunctionAst
  577. |BodyInfo#ast_info.pre_render_asts])).
  578. options_match_ast(Context) -> options_match_ast(Context, undefined).
  579. options_match_ast(Context, TreeWalker) ->
  580. [
  581. erl_syntax:match_expr(
  582. erl_syntax:variable("_TranslationFun"),
  583. erl_syntax:application(
  584. erl_syntax:atom(proplists),
  585. erl_syntax:atom(get_value),
  586. [erl_syntax:atom(translation_fun), erl_syntax:variable("RenderOptions"), erl_syntax:atom(none)])),
  587. erl_syntax:match_expr(
  588. erl_syntax:variable("_CurrentLocale"),
  589. erl_syntax:application(
  590. erl_syntax:atom(proplists),
  591. erl_syntax:atom(get_value),
  592. [erl_syntax:atom(locale), erl_syntax:variable("RenderOptions"), erl_syntax:atom(none)]))
  593. | case call_extension(Context, setup_render_ast, [Context, TreeWalker]) of
  594. undefined -> [];
  595. Ast when is_list(Ast) -> Ast
  596. end
  597. ].
  598. % child templates should only consist of blocks at the top level
  599. body_ast([{'extends', {string_literal, _Pos, String}} | ThisParseTree], Context, TreeWalker) ->
  600. File = full_path(unescape_string_literal(String), Context#dtl_context.doc_root),
  601. case lists:member(File, Context#dtl_context.parse_trail) of
  602. true ->
  603. throw({error, "Circular file inclusion!"});
  604. _ ->
  605. case parse(File, Context) of
  606. {ok, ParentParseTree, CheckSum} ->
  607. BlockDict = lists:foldl(
  608. fun
  609. ({block, {identifier, _, Name}, Contents}, Dict) ->
  610. dict:store(Name, Contents, Dict);
  611. (_, Dict) ->
  612. Dict
  613. end, dict:new(), ThisParseTree),
  614. with_dependency({File, CheckSum}, body_ast(ParentParseTree, Context#dtl_context{
  615. block_dict = dict:merge(fun(_Key, _ParentVal, ChildVal) -> ChildVal end,
  616. BlockDict, Context#dtl_context.block_dict),
  617. parse_trail = [File | Context#dtl_context.parse_trail]}, TreeWalker));
  618. Err ->
  619. throw(Err)
  620. end
  621. end;
  622. body_ast(DjangoParseTree, Context, TreeWalker) ->
  623. {AstInfoList, TreeWalker2} = lists:mapfoldl(
  624. fun
  625. ({'autoescape', {identifier, _, OnOrOff}, Contents}, TreeWalkerAcc) ->
  626. body_ast(Contents, Context#dtl_context{auto_escape = OnOrOff},
  627. TreeWalkerAcc);
  628. ({'block', {identifier, _, Name}, Contents}, TreeWalkerAcc) ->
  629. Block = case dict:find(Name, Context#dtl_context.block_dict) of
  630. {ok, ChildBlock} -> ChildBlock;
  631. _ -> Contents
  632. end,
  633. body_ast(Block, Context, TreeWalkerAcc);
  634. ({'blocktrans', Args, Contents}, TreeWalkerAcc) ->
  635. blocktrans_ast(Args, Contents, Context, TreeWalkerAcc);
  636. ({'call', {identifier, _, Name}}, TreeWalkerAcc) ->
  637. call_ast(Name, TreeWalkerAcc);
  638. ({'call', {identifier, _, Name}, With}, TreeWalkerAcc) ->
  639. call_with_ast(Name, With, Context, TreeWalkerAcc);
  640. ({'comment', _Contents}, TreeWalkerAcc) ->
  641. empty_ast(TreeWalkerAcc);
  642. ({'cycle', Names}, TreeWalkerAcc) ->
  643. cycle_ast(Names, Context, TreeWalkerAcc);
  644. ({'cycle_compat', Names}, TreeWalkerAcc) ->
  645. cycle_compat_ast(Names, Context, TreeWalkerAcc);
  646. ({'date', 'now', {string_literal, _Pos, FormatString}}, TreeWalkerAcc) ->
  647. now_ast(FormatString, Context, TreeWalkerAcc);
  648. ({'filter', FilterList, Contents}, TreeWalkerAcc) ->
  649. filter_tag_ast(FilterList, Contents, Context, TreeWalkerAcc);
  650. ({'firstof', Vars}, TreeWalkerAcc) ->
  651. firstof_ast(Vars, Context, TreeWalkerAcc);
  652. ({'for', {'in', IteratorList, Variable, Reversed}, Contents}, TreeWalkerAcc) ->
  653. {EmptyAstInfo, TreeWalker1} = empty_ast(TreeWalkerAcc),
  654. for_loop_ast(IteratorList, Variable, Reversed, Contents, EmptyAstInfo, Context, TreeWalker1);
  655. ({'for', {'in', IteratorList, Variable, Reversed}, Contents, EmptyPartContents}, TreeWalkerAcc) ->
  656. {EmptyAstInfo, TreeWalker1} = body_ast(EmptyPartContents, Context, TreeWalkerAcc),
  657. for_loop_ast(IteratorList, Variable, Reversed, Contents, EmptyAstInfo, Context, TreeWalker1);
  658. ({'if', Expression, Contents, Elif}, TreeWalkerAcc) ->
  659. {IfAstInfo, TreeWalker1} = body_ast(Contents, Context, TreeWalkerAcc),
  660. {ElifAstInfo, TreeWalker2} = body_ast(Elif, Context, TreeWalker1),
  661. ifelse_ast(Expression, IfAstInfo, ElifAstInfo, Context, TreeWalker2);
  662. ({'if', Expression, Contents}, TreeWalkerAcc) ->
  663. {IfAstInfo, TreeWalker1} = body_ast(Contents, Context, TreeWalkerAcc),
  664. {ElseAstInfo, TreeWalker2} = empty_ast(TreeWalker1),
  665. ifelse_ast(Expression, IfAstInfo, ElseAstInfo, Context, TreeWalker2);
  666. ({'ifchanged', '$undefined', Contents}, TreeWalkerAcc) ->
  667. {IfAstInfo, TreeWalker1} = body_ast(Contents, Context, TreeWalkerAcc),
  668. {ElseAstInfo, TreeWalker2} = empty_ast(TreeWalker1),
  669. ifchanged_contents_ast(Contents, IfAstInfo, ElseAstInfo, Context, TreeWalker2);
  670. ({'ifchanged', Values, Contents}, TreeWalkerAcc) ->
  671. {IfAstInfo, TreeWalker1} = body_ast(Contents, Context, TreeWalkerAcc),
  672. {ElseAstInfo, TreeWalker2} = empty_ast(TreeWalker1),
  673. ifchanged_values_ast(Values, IfAstInfo, ElseAstInfo, Context, TreeWalker2);
  674. ({'ifchangedelse', '$undefined', IfContents, ElseContents}, TreeWalkerAcc) ->
  675. {IfAstInfo, TreeWalker1} = body_ast(IfContents, Context, TreeWalkerAcc),
  676. {ElseAstInfo, TreeWalker2} = body_ast(ElseContents, Context, TreeWalker1),
  677. ifchanged_contents_ast(IfContents, IfAstInfo, ElseAstInfo, Context, TreeWalker2);
  678. ({'ifchangedelse', Values, IfContents, ElseContents}, TreeWalkerAcc) ->
  679. {IfAstInfo, TreeWalker1} = body_ast(IfContents, Context, TreeWalkerAcc),
  680. {ElseAstInfo, TreeWalker2} = body_ast(ElseContents, Context, TreeWalker1),
  681. ifchanged_values_ast(Values, IfAstInfo, ElseAstInfo, Context, TreeWalker2);
  682. ({'ifelse', Expression, IfContents, ElseContents}, TreeWalkerAcc) ->
  683. {IfAstInfo, TreeWalker1} = body_ast(IfContents, Context, TreeWalkerAcc),
  684. {ElseAstInfo, TreeWalker2} = body_ast(ElseContents, Context, TreeWalker1),
  685. ifelse_ast(Expression, IfAstInfo, ElseAstInfo, Context, TreeWalker2);
  686. ({'ifequal', [Arg1, Arg2], Contents}, TreeWalkerAcc) ->
  687. {IfAstInfo, TreeWalker1} = body_ast(Contents, Context, TreeWalkerAcc),
  688. {ElseAstInfo, TreeWalker2} = empty_ast(TreeWalker1),
  689. ifelse_ast({'expr', "eq", Arg1, Arg2}, IfAstInfo, ElseAstInfo, Context, TreeWalker2);
  690. ({'ifequalelse', [Arg1, Arg2], IfContents, ElseContents}, TreeWalkerAcc) ->
  691. {IfAstInfo, TreeWalker1} = body_ast(IfContents, Context, TreeWalkerAcc),
  692. {ElseAstInfo, TreeWalker2} = body_ast(ElseContents, Context,TreeWalker1),
  693. ifelse_ast({'expr', "eq", Arg1, Arg2}, IfAstInfo, ElseAstInfo, Context, TreeWalker2);
  694. ({'ifnotequal', [Arg1, Arg2], Contents}, TreeWalkerAcc) ->
  695. {IfAstInfo, TreeWalker1} = body_ast(Contents, Context, TreeWalkerAcc),
  696. {ElseAstInfo, TreeWalker2} = empty_ast(TreeWalker1),
  697. ifelse_ast({'expr', "ne", Arg1, Arg2}, IfAstInfo, ElseAstInfo, Context, TreeWalker2);
  698. ({'ifnotequalelse', [Arg1, Arg2], IfContents, ElseContents}, TreeWalkerAcc) ->
  699. {IfAstInfo, TreeWalker1} = body_ast(IfContents, Context, TreeWalkerAcc),
  700. {ElseAstInfo, TreeWalker2} = body_ast(ElseContents, Context, TreeWalker1),
  701. ifelse_ast({'expr', "ne", Arg1, Arg2}, IfAstInfo, ElseAstInfo, Context, TreeWalker2);
  702. ({'include', {string_literal, _, File}, Args}, TreeWalkerAcc) ->
  703. include_ast(unescape_string_literal(File), Args, Context#dtl_context.local_scopes, Context, TreeWalkerAcc);
  704. ({'include_only', {string_literal, _, File}, Args}, TreeWalkerAcc) ->
  705. include_ast(unescape_string_literal(File), Args, [], Context, TreeWalkerAcc);
  706. ({'regroup', {ListVariable, Grouper, {identifier, _, NewVariable}}, Contents}, TreeWalkerAcc) ->
  707. regroup_ast(ListVariable, Grouper, NewVariable, Contents, Context, TreeWalkerAcc);
  708. ({'spaceless', Contents}, TreeWalkerAcc) ->
  709. spaceless_ast(Contents, Context, TreeWalkerAcc);
  710. ({'ssi', Arg}, TreeWalkerAcc) ->
  711. ssi_ast(Arg, Context, TreeWalkerAcc);
  712. ({'ssi_parsed', {string_literal, _, FileName}}, TreeWalkerAcc) ->
  713. include_ast(unescape_string_literal(FileName), [], Context#dtl_context.local_scopes, Context, TreeWalkerAcc);
  714. ({'string', _Pos, String}, TreeWalkerAcc) ->
  715. string_ast(String, Context, TreeWalkerAcc);
  716. ({'tag', {identifier, _, Name}, Args}, TreeWalkerAcc) ->
  717. tag_ast(Name, Args, Context, TreeWalkerAcc);
  718. ({'templatetag', {_, _, TagName}}, TreeWalkerAcc) ->
  719. templatetag_ast(TagName, Context, TreeWalkerAcc);
  720. ({'trans', Value}, TreeWalkerAcc) ->
  721. translated_ast(Value, Context, TreeWalkerAcc);
  722. ({'widthratio', Numerator, Denominator, Scale}, TreeWalkerAcc) ->
  723. widthratio_ast(Numerator, Denominator, Scale, Context, TreeWalkerAcc);
  724. ({'with', Args, Contents}, TreeWalkerAcc) ->
  725. with_ast(Args, Contents, Context, TreeWalkerAcc);
  726. ({'extension', Tag}, TreeWalkerAcc) ->
  727. extension_ast(Tag, Context, TreeWalkerAcc);
  728. (ValueToken, TreeWalkerAcc) ->
  729. {{ValueAst,ValueInfo},ValueTreeWalker} = value_ast(ValueToken, true, true, Context, TreeWalkerAcc),
  730. {{format(ValueAst, Context, ValueTreeWalker),ValueInfo},ValueTreeWalker}
  731. end, TreeWalker, DjangoParseTree),
  732. {AstList, {Info, TreeWalker3}} = lists:mapfoldl(
  733. fun({Ast, Info}, {InfoAcc, TreeWalkerAcc}) ->
  734. PresetVars = lists:foldl(fun
  735. (X, Acc) ->
  736. case proplists:lookup(X, Context#dtl_context.vars) of
  737. none ->
  738. Acc;
  739. Val ->
  740. [erl_syntax:abstract(Val) | Acc]
  741. end
  742. end, [], Info#ast_info.var_names),
  743. case PresetVars of
  744. [] ->
  745. {Ast, {merge_info(Info, InfoAcc), TreeWalkerAcc}};
  746. _ ->
  747. Counter = TreeWalkerAcc#treewalker.counter,
  748. Name = lists:concat([pre_render, Counter]),
  749. Ast1 = erl_syntax:application(none, erl_syntax:atom(Name),
  750. [erl_syntax:list(PresetVars)]),
  751. PreRenderAst = erl_syntax:function(erl_syntax:atom(Name),
  752. [erl_syntax:clause([erl_syntax:variable("_Variables")], none, [Ast])]),
  753. PreRenderAsts = Info#ast_info.pre_render_asts,
  754. Info1 = Info#ast_info{pre_render_asts = [PreRenderAst | PreRenderAsts]},
  755. {Ast1, {merge_info(Info1, InfoAcc), TreeWalkerAcc#treewalker{counter = Counter + 1}}}
  756. end
  757. end, {#ast_info{}, TreeWalker2}, AstInfoList),
  758. {{erl_syntax:list(AstList), Info}, TreeWalker3}.
  759. value_ast(ValueToken, AsString, EmptyIfUndefined, Context, TreeWalker) ->
  760. case ValueToken of
  761. {'expr', Operator, Value} ->
  762. {{ValueAst,InfoValue}, TreeWalker1} = value_ast(Value, false, EmptyIfUndefined, Context, TreeWalker),
  763. Ast = erl_syntax:application(erl_syntax:atom(erlydtl_runtime),
  764. erl_syntax:atom(Operator),
  765. [ValueAst]),
  766. {{Ast, InfoValue}, TreeWalker1};
  767. {'expr', Operator, Value1, Value2} ->
  768. {{Value1Ast,InfoValue1}, TreeWalker1} = value_ast(Value1, false, EmptyIfUndefined, Context, TreeWalker),
  769. {{Value2Ast,InfoValue2}, TreeWalker2} = value_ast(Value2, false, EmptyIfUndefined, Context, TreeWalker1),
  770. Ast = erl_syntax:application(erl_syntax:atom(erlydtl_runtime),
  771. erl_syntax:atom(Operator),
  772. [Value1Ast, Value2Ast]),
  773. {{Ast, merge_info(InfoValue1,InfoValue2)}, TreeWalker2};
  774. {'string_literal', _Pos, String} ->
  775. string_ast(unescape_string_literal(String), Context, TreeWalker);
  776. {'number_literal', _Pos, Number} ->
  777. case AsString of
  778. true -> string_ast(Number, Context, TreeWalker);
  779. false -> {{erl_syntax:integer(list_to_integer(Number)), #ast_info{}}, TreeWalker}
  780. end;
  781. {'apply_filter', Variable, Filter} ->
  782. filter_ast(Variable, Filter, Context, TreeWalker);
  783. {'attribute', _} = Variable ->
  784. resolve_variable_ast(Variable, Context, TreeWalker, EmptyIfUndefined);
  785. {'variable', _} = Variable ->
  786. resolve_variable_ast(Variable, Context, TreeWalker, EmptyIfUndefined);
  787. {extension, Tag} ->
  788. extension_ast(Tag, Context, TreeWalker)
  789. end.
  790. extension_ast(Tag, Context, TreeWalker) ->
  791. case call_extension(Context, compile_ast, [Tag, Context, TreeWalker]) of
  792. undefined ->
  793. throw({error, {unknown_extension, Tag}});
  794. Result ->
  795. Result
  796. end.
  797. merge_info(Info1, Info2) ->
  798. #ast_info{
  799. dependencies =
  800. lists:merge(
  801. lists:sort(Info1#ast_info.dependencies),
  802. lists:sort(Info2#ast_info.dependencies)),
  803. var_names =
  804. lists:merge(
  805. lists:sort(Info1#ast_info.var_names),
  806. lists:sort(Info2#ast_info.var_names)),
  807. translatable_strings =
  808. lists:merge(
  809. lists:sort(Info1#ast_info.translatable_strings),
  810. lists:sort(Info2#ast_info.translatable_strings)),
  811. translated_blocks =
  812. lists:merge(
  813. lists:sort(Info1#ast_info.translated_blocks),
  814. lists:sort(Info2#ast_info.translated_blocks)),
  815. custom_tags =
  816. lists:merge(
  817. lists:sort(Info1#ast_info.custom_tags),
  818. lists:sort(Info2#ast_info.custom_tags)),
  819. pre_render_asts =
  820. lists:merge(
  821. Info1#ast_info.pre_render_asts,
  822. Info2#ast_info.pre_render_asts)}.
  823. with_dependencies([], Args) ->
  824. Args;
  825. with_dependencies([Dependency | Rest], Args) ->
  826. with_dependencies(Rest, with_dependency(Dependency, Args)).
  827. with_dependency(FilePath, {{Ast, Info}, TreeWalker}) ->
  828. {{Ast, Info#ast_info{dependencies = [FilePath | Info#ast_info.dependencies]}}, TreeWalker}.
  829. empty_ast(TreeWalker) ->
  830. {{erl_syntax:list([]), #ast_info{}}, TreeWalker}.
  831. blocktrans_ast(ArgList, Contents, Context, TreeWalker) ->
  832. {NewScope, {ArgInfo, TreeWalker1}} = lists:mapfoldl(fun
  833. ({{identifier, _, LocalVarName}, Value}, {AstInfo1, TreeWalker1}) ->
  834. {{Ast, Info}, TreeWalker2} = value_ast(Value, false, false, Context, TreeWalker1),
  835. {{LocalVarName, Ast}, {merge_info(AstInfo1, Info), TreeWalker2}}
  836. end, {#ast_info{}, TreeWalker}, ArgList),
  837. NewContext = Context#dtl_context{ local_scopes = [NewScope|Context#dtl_context.local_scopes] },
  838. SourceText = lists:flatten(erlydtl_unparser:unparse(Contents)),
  839. {{DefaultAst, AstInfo}, TreeWalker2} = body_ast(Contents, NewContext, TreeWalker1),
  840. MergedInfo = merge_info(AstInfo, ArgInfo),
  841. case Context#dtl_context.blocktrans_fun of
  842. none ->
  843. {{DefaultAst, MergedInfo}, TreeWalker2};
  844. BlockTransFun when is_function(BlockTransFun) ->
  845. {FinalAstInfo, FinalTreeWalker, Clauses} = lists:foldr(fun(Locale, {AstInfoAcc, ThisTreeWalker, ClauseAcc}) ->
  846. case BlockTransFun(SourceText, Locale) of
  847. default ->
  848. {AstInfoAcc, ThisTreeWalker, ClauseAcc};
  849. Body ->
  850. {ok, DjangoParseTree} = parse(Body, Context),
  851. {{ThisAst, ThisAstInfo}, TreeWalker3} = body_ast(DjangoParseTree, NewContext, ThisTreeWalker),
  852. {merge_info(ThisAstInfo, AstInfoAcc), TreeWalker3,
  853. [erl_syntax:clause([erl_syntax:string(Locale)], none, [ThisAst])|ClauseAcc]}
  854. end
  855. end, {MergedInfo, TreeWalker2, []}, Context#dtl_context.blocktrans_locales),
  856. Ast = erl_syntax:case_expr(erl_syntax:variable("_CurrentLocale"),
  857. Clauses ++ [erl_syntax:clause([erl_syntax:underscore()], none, [DefaultAst])]),
  858. {{Ast, FinalAstInfo#ast_info{ translated_blocks = [SourceText] }}, FinalTreeWalker}
  859. end.
  860. translated_ast({string_literal, _, String}, Context, TreeWalker) ->
  861. NewStr = unescape_string_literal(String),
  862. DefaultString = case Context#dtl_context.locale of
  863. none -> NewStr;
  864. Locale -> erlydtl_i18n:translate(NewStr,Locale)
  865. end,
  866. translated_ast2(erl_syntax:string(NewStr), erl_syntax:string(DefaultString),
  867. #ast_info{translatable_strings = [NewStr]}, TreeWalker);
  868. translated_ast(ValueToken, Context, TreeWalker) ->
  869. {{Ast, Info}, TreeWalker1} = value_ast(ValueToken, true, false, Context, TreeWalker),
  870. translated_ast2(Ast, Ast, Info, TreeWalker1).
  871. translated_ast2(NewStrAst, DefaultStringAst, AstInfo, TreeWalker) ->
  872. StringLookupAst = erl_syntax:application(
  873. erl_syntax:atom(erlydtl_runtime),
  874. erl_syntax:atom(translate),
  875. [NewStrAst, erl_syntax:variable("_TranslationFun"), DefaultStringAst]),
  876. {{StringLookupAst, AstInfo}, TreeWalker}.
  877. % Completely unnecessary in ErlyDTL (use {{ "{%" }} etc), but implemented for compatibility.
  878. templatetag_ast("openblock", Context, TreeWalker) ->
  879. string_ast("{%", Context, TreeWalker);
  880. templatetag_ast("closeblock", Context, TreeWalker) ->
  881. string_ast("%}", Context, TreeWalker);
  882. templatetag_ast("openvariable", Context, TreeWalker) ->
  883. string_ast("{{", Context, TreeWalker);
  884. templatetag_ast("closevariable", Context, TreeWalker) ->
  885. string_ast("}}", Context, TreeWalker);
  886. templatetag_ast("openbrace", Context, TreeWalker) ->
  887. string_ast("{", Context, TreeWalker);
  888. templatetag_ast("closebrace", Context, TreeWalker) ->
  889. string_ast("}", Context, TreeWalker);
  890. templatetag_ast("opencomment", Context, TreeWalker) ->
  891. string_ast("{#", Context, TreeWalker);
  892. templatetag_ast("closecomment", Context, TreeWalker) ->
  893. string_ast("#}", Context, TreeWalker).
  894. widthratio_ast(Numerator, Denominator, Scale, Context, TreeWalker) ->
  895. {{NumAst, NumInfo}, TreeWalker1} = value_ast(Numerator, false, true, Context, TreeWalker),
  896. {{DenAst, DenInfo}, TreeWalker2} = value_ast(Denominator, false, true, Context, TreeWalker1),
  897. {{ScaleAst, ScaleInfo}, TreeWalker3} = value_ast(Scale, false, true, Context, TreeWalker2),
  898. {{format_number_ast(erl_syntax:application(
  899. erl_syntax:atom(erlydtl_runtime),
  900. erl_syntax:atom(widthratio),
  901. [NumAst, DenAst, ScaleAst])), merge_info(ScaleInfo, merge_info(NumInfo, DenInfo))},
  902. TreeWalker3}.
  903. binary_string(String) ->
  904. erl_syntax:binary([erl_syntax:binary_field(erl_syntax:integer(X)) || X <- String]).
  905. string_ast(String, #dtl_context{ binary_strings = true }, TreeWalker) when is_list(String) ->
  906. {{binary_string(String), #ast_info{}}, TreeWalker};
  907. string_ast(String, #dtl_context{ binary_strings = false }, TreeWalker) when is_list(String) ->
  908. {{erl_syntax:string(String), #ast_info{}}, TreeWalker}; %% less verbose AST, better for development and debugging
  909. string_ast(S, Context, TreeWalker) when is_atom(S) ->
  910. string_ast(atom_to_list(S), Context, TreeWalker).
  911. include_ast(File, ArgList, Scopes, Context, TreeWalker) ->
  912. FilePath = full_path(File, Context#dtl_context.doc_root),
  913. case parse(FilePath, Context) of
  914. {ok, InclusionParseTree, CheckSum} ->
  915. {NewScope, {ArgInfo, TreeWalker1}} = lists:mapfoldl(fun
  916. ({{identifier, _, LocalVarName}, Value}, {AstInfo1, TreeWalker1}) ->
  917. {{Ast, Info}, TreeWalker2} = value_ast(Value, false, false, Context, TreeWalker1),
  918. {{LocalVarName, Ast}, {merge_info(AstInfo1, Info), TreeWalker2}}
  919. end, {#ast_info{}, TreeWalker}, ArgList),
  920. {{BodyAst, BodyInfo}, TreeWalker2} = with_dependency({FilePath, CheckSum},
  921. body_ast(InclusionParseTree, Context#dtl_context{
  922. parse_trail = [FilePath | Context#dtl_context.parse_trail],
  923. local_scopes = [NewScope|Scopes]
  924. }, TreeWalker1)),
  925. {{BodyAst, merge_info(BodyInfo, ArgInfo)}, TreeWalker2};
  926. Err ->
  927. throw(Err)
  928. end.
  929. % include at run-time
  930. ssi_ast(FileName, Context, TreeWalker) ->
  931. {{Ast, Info}, TreeWalker1} = value_ast(FileName, true, true, Context, TreeWalker),
  932. {Mod, Fun} = Context#dtl_context.reader,
  933. {{erl_syntax:application(
  934. erl_syntax:atom(erlydtl_runtime),
  935. erl_syntax:atom(read_file),
  936. [erl_syntax:atom(Mod), erl_syntax:atom(Fun), erl_syntax:string(Context#dtl_context.doc_root), Ast]), Info}, TreeWalker1}.
  937. filter_tag_ast(FilterList, Contents, Context, TreeWalker) ->
  938. {{InnerAst, Info}, TreeWalker1} = body_ast(Contents, Context#dtl_context{auto_escape = did}, TreeWalker),
  939. {{FilteredAst, FilteredInfo}, TreeWalker2} = lists:foldl(fun
  940. ([{identifier, _, 'escape'}], {{AstAcc, InfoAcc}, TreeWalkerAcc}) ->
  941. {{AstAcc, InfoAcc}, TreeWalkerAcc#treewalker{safe = true}};
  942. ([{identifier, _, 'safe'}], {{AstAcc, InfoAcc}, TreeWalkerAcc}) ->
  943. {{AstAcc, InfoAcc}, TreeWalkerAcc#treewalker{safe = true}};
  944. ([{identifier, _, 'safeseq'}], {{AstAcc, InfoAcc}, TreeWalkerAcc}) ->
  945. {{AstAcc, InfoAcc}, TreeWalkerAcc#treewalker{safe = true}};
  946. (Filter, {{AstAcc, InfoAcc}, TreeWalkerAcc}) ->
  947. {{Ast, AstInfo}, TW} = filter_ast1(Filter, AstAcc, Context, TreeWalkerAcc),
  948. {{Ast, merge_info(InfoAcc, AstInfo)}, TW}
  949. end, {{erl_syntax:application(
  950. erl_syntax:atom(erlang),
  951. erl_syntax:atom(iolist_to_binary),
  952. [InnerAst]), Info}, TreeWalker1}, FilterList),
  953. EscapedAst = case search_for_escape_filter(lists:reverse(FilterList), Context) of
  954. on ->
  955. erl_syntax:application(
  956. erl_syntax:atom(erlydtl_filters),
  957. erl_syntax:atom(force_escape),
  958. [FilteredAst]);
  959. _ ->
  960. FilteredAst
  961. end,
  962. {{EscapedAst, FilteredInfo}, TreeWalker2}.
  963. search_for_escape_filter(FilterList, #dtl_context{auto_escape = on}) ->
  964. search_for_safe_filter(FilterList);
  965. search_for_escape_filter(_, #dtl_context{auto_escape = did}) ->
  966. off;
  967. search_for_escape_filter([[{identifier, _, 'escape'}]|Rest], _Context) ->
  968. search_for_safe_filter(Rest);
  969. search_for_escape_filter([_|Rest], Context) ->
  970. search_for_escape_filter(Rest, Context);
  971. search_for_escape_filter([], _Context) ->
  972. off.
  973. search_for_safe_filter([[{identifier, _, 'safe'}]|_]) ->
  974. off;
  975. search_for_safe_filter([[{identifier, _, 'safeseq'}]|_]) ->
  976. off;
  977. search_for_safe_filter([_|Rest]) ->
  978. search_for_safe_filter(Rest);
  979. search_for_safe_filter([]) ->
  980. on.
  981. filter_ast(Variable, Filter, Context, TreeWalker) ->
  982. % the escape filter is special; it is always applied last, so we have to go digging for it
  983. % AutoEscape = 'did' means we (will have) decided whether to escape the current variable,
  984. % so don't do any more escaping
  985. {{UnescapedAst, Info}, TreeWalker2} = filter_ast_noescape(Variable, Filter,
  986. Context#dtl_context{auto_escape = did}, TreeWalker),
  987. EscapedAst = case search_for_escape_filter(Variable, Filter, Context) of
  988. on ->
  989. erl_syntax:application(
  990. erl_syntax:atom(erlydtl_filters),
  991. erl_syntax:atom(force_escape),
  992. [UnescapedAst]);
  993. _ ->
  994. UnescapedAst
  995. end,
  996. {{EscapedAst, Info}, TreeWalker2}.
  997. filter_ast_noescape(Variable, [{identifier, _, 'escape'}], Context, TreeWalker) ->
  998. value_ast(Variable, true, false, Context, TreeWalker#treewalker{safe = true});
  999. filter_ast_noescape(Variable, [{identifier, _, 'safe'}], Context, TreeWalker) ->
  1000. value_ast(Variable, true, false, Context, TreeWalker#treewalker{safe = true});
  1001. filter_ast_noescape(Variable, [{identifier, _, 'safeseq'}], Context, TreeWalker) ->
  1002. value_ast(Variable, true, false, Context, TreeWalker#treewalker{safe = true});
  1003. filter_ast_noescape(Variable, Filter, Context, TreeWalker) ->
  1004. {{VariableAst, Info1}, TreeWalker2} = value_ast(Variable, true, false, Context, TreeWalker),
  1005. {{VarValue, Info2}, TreeWalker3} = filter_ast1(Filter, VariableAst, Context, TreeWalker2),
  1006. {{VarValue, merge_info(Info1, Info2)}, TreeWalker3}.
  1007. filter_ast1([{identifier, _, Name}, {string_literal, _, ArgName}], VariableAst,
  1008. #dtl_context{ binary_strings = true } = Context, TreeWalker) ->
  1009. filter_ast2(Name, VariableAst, [binary_string(unescape_string_literal(ArgName))], #ast_info{}, Context, TreeWalker);
  1010. filter_ast1([{identifier, _, Name}, {string_literal, _, ArgName}], VariableAst,
  1011. #dtl_context{ binary_strings = false } = Context, TreeWalker) ->
  1012. filter_ast2(Name, VariableAst, [erl_syntax:string(unescape_string_literal(ArgName))], #ast_info{}, Context, TreeWalker);
  1013. filter_ast1([{identifier, _, Name}, {number_literal, _, ArgName}], VariableAst, Context, TreeWalker) ->
  1014. filter_ast2(Name, VariableAst, [erl_syntax:integer(list_to_integer(ArgName))], #ast_info{}, Context, TreeWalker);
  1015. filter_ast1([{identifier, _, Name}, ArgVariable], VariableAst, Context, TreeWalker) ->
  1016. {{ArgAst, ArgInfo}, TreeWalker2} = resolve_variable_ast(ArgVariable, Context, TreeWalker, false),
  1017. filter_ast2(Name, VariableAst, [ArgAst], ArgInfo, Context, TreeWalker2);
  1018. filter_ast1([{identifier, _, Name}], VariableAst, Context, TreeWalker) ->
  1019. filter_ast2(Name, VariableAst, [], #ast_info{}, Context, TreeWalker).
  1020. filter_ast2(Name, VariableAst, [], VarInfo, #dtl_context{ filter_modules = [Module|Rest] } = Context, TreeWalker) ->
  1021. case lists:member({Name, 1}, Module:module_info(exports)) of
  1022. true ->
  1023. {{erl_syntax:application(
  1024. erl_syntax:atom(Module), erl_syntax:atom(Name),
  1025. [VariableAst]),
  1026. VarInfo},
  1027. TreeWalker};
  1028. false ->
  1029. filter_ast2(Name, VariableAst, [], VarInfo, Context#dtl_context{ filter_modules = Rest }, TreeWalker)
  1030. end;
  1031. filter_ast2(Name, VariableAst, [Arg], VarInfo, #dtl_context{ filter_modules = [Module|Rest] } = Context, TreeWalker) ->
  1032. case lists:member({Name, 2}, Module:module_info(exports)) of
  1033. true ->
  1034. {{erl_syntax:application(
  1035. erl_syntax:atom(Module), erl_syntax:atom(Name),
  1036. [VariableAst, Arg]),
  1037. VarInfo},
  1038. TreeWalker};
  1039. false ->
  1040. filter_ast2(Name, VariableAst, [Arg], VarInfo, Context#dtl_context{ filter_modules = Rest }, TreeWalker)
  1041. end.
  1042. search_for_escape_filter(Variable, Filter, #dtl_context{auto_escape = on}) ->
  1043. search_for_safe_filter(Variable, Filter);
  1044. search_for_escape_filter(_, _, #dtl_context{auto_escape = did}) ->
  1045. off;
  1046. search_for_escape_filter(Variable, [{identifier, _, 'escape'}] = Filter, _Context) ->
  1047. search_for_safe_filter(Variable, Filter);
  1048. search_for_escape_filter({apply_filter, Variable, Filter}, _, Context) ->
  1049. search_for_escape_filter(Variable, Filter, Context);
  1050. search_for_escape_filter(_Variable, _Filter, _Context) ->
  1051. off.
  1052. search_for_safe_filter(_, [{identifier, _, 'safe'}]) ->
  1053. off;
  1054. search_for_safe_filter(_, [{identifier, _, 'safeseq'}]) ->
  1055. off;
  1056. search_for_safe_filter({apply_filter, Variable, Filter}, _) ->
  1057. search_for_safe_filter(Variable, Filter);
  1058. search_for_safe_filter(_Variable, _Filter) ->
  1059. on.
  1060. finder_function(true) -> {erlydtl_runtime, fetch_value};
  1061. finder_function(false) -> {erlydtl_runtime, find_value}.
  1062. finder_function(EmptyIfUndefined, Context) ->
  1063. case call_extension(Context, finder_function, [EmptyIfUndefined]) of
  1064. undefined -> finder_function(EmptyIfUndefined);
  1065. Result -> Result
  1066. end.
  1067. resolve_variable_ast({extension, Tag}, Context, TreeWalker, _) ->
  1068. extension_ast(Tag, Context, TreeWalker);
  1069. resolve_variable_ast(VarTuple, Context, TreeWalker, EmptyIfUndefined)
  1070. when is_boolean(EmptyIfUndefined) ->
  1071. resolve_variable_ast(VarTuple, Context, TreeWalker, finder_function(EmptyIfUndefined, Context));
  1072. resolve_variable_ast(VarTuple, Context, TreeWalker, FinderFunction) ->
  1073. resolve_variable_ast1(VarTuple, Context, TreeWalker, FinderFunction).
  1074. resolve_variable_ast1({attribute, {{identifier, {Row, Col}, AttrName}, Variable}}, Context, TreeWalker, FinderFunction) ->
  1075. {{VarAst, VarInfo}, TreeWalker1} = resolve_variable_ast(Variable, Context, TreeWalker, FinderFunction),
  1076. FileNameAst = case Context#dtl_context.parse_trail of
  1077. [] -> erl_syntax:atom(undefined);
  1078. [H|_] -> erl_syntax:string(H)
  1079. end,
  1080. {Runtime, Finder} = FinderFunction,
  1081. {{erl_syntax:application(
  1082. erl_syntax:atom(Runtime),
  1083. erl_syntax:atom(Finder),
  1084. [erl_syntax:atom(AttrName), VarAst, FileNameAst,
  1085. erl_syntax:tuple([erl_syntax:integer(Row), erl_syntax:integer(Col)])
  1086. ]),
  1087. VarInfo},
  1088. TreeWalker1};
  1089. resolve_variable_ast1({variable, {identifier, {Row, Col}, VarName}}, Context, TreeWalker, FinderFunction) ->
  1090. VarValue = case resolve_scoped_variable_ast(VarName, Context) of
  1091. undefined ->
  1092. FileNameAst = case Context#dtl_context.parse_trail of
  1093. [] -> erl_syntax:atom(undefined);
  1094. [H|_] -> erl_syntax:string(H)
  1095. end,
  1096. {Runtime, Finder} = FinderFunction,
  1097. erl_syntax:application(
  1098. erl_syntax:atom(Runtime), erl_syntax:atom(Finder),
  1099. [erl_syntax:atom(VarName), erl_syntax:variable("_Variables"), FileNameAst,
  1100. erl_syntax:tuple([erl_syntax:integer(Row), erl_syntax:integer(Col)])
  1101. ]);
  1102. Val ->
  1103. Val
  1104. end,
  1105. {{VarValue, #ast_info{ var_names=[VarName] }}, TreeWalker}.
  1106. resolve_scoped_variable_ast(VarName, Context) ->
  1107. lists:foldl(fun(Scope, Value) ->
  1108. case Value of
  1109. undefined -> proplists:get_value(VarName, Scope);
  1110. _ -> Value
  1111. end
  1112. end, undefined, Context#dtl_context.local_scopes).
  1113. format(Ast, Context, TreeWalker) ->
  1114. auto_escape(format_number_ast(Ast), Context, TreeWalker).
  1115. format_number_ast(Ast) ->
  1116. erl_syntax:application(erl_syntax:atom(erlydtl_filters), erl_syntax:atom(format_number),
  1117. [Ast]).
  1118. auto_escape(Value, _, #treewalker{safe = true}) ->
  1119. Value;
  1120. auto_escape(Value, #dtl_context{auto_escape = on}, _) ->
  1121. erl_syntax:application(erl_syntax:atom(erlydtl_filters), erl_syntax:atom(force_escape), [Value]);
  1122. auto_escape(Value, _, _) ->
  1123. Value.
  1124. firstof_ast(Vars, Context, TreeWalker) ->
  1125. body_ast([lists:foldr(fun
  1126. ({L, _, _}=Var, []) when L=:=string_literal;L=:=number_literal ->
  1127. Var;
  1128. ({L, _, _}, _) when L=:=string_literal;L=:=number_literal ->
  1129. erlang:error(errbadliteral);
  1130. (Var, []) ->
  1131. {'ifelse', Var, [Var], []};
  1132. (Var, Acc) ->
  1133. {'ifelse', Var, [Var], [Acc]} end,
  1134. [], Vars)], Context, TreeWalker).
  1135. ifelse_ast(Expression, {IfContentsAst, IfContentsInfo}, {ElseContentsAst, ElseContentsInfo}, Context, TreeWalker) ->
  1136. Info = merge_info(IfContentsInfo, ElseContentsInfo),
  1137. {{Ast, ExpressionInfo}, TreeWalker1} = value_ast(Expression, false, false, Context, TreeWalker),
  1138. {{erl_syntax:case_expr(erl_syntax:application(erl_syntax:atom(erlydtl_runtime), erl_syntax:atom(is_true), [Ast]),
  1139. [erl_syntax:clause([erl_syntax:atom(true)], none,
  1140. [IfContentsAst]),
  1141. erl_syntax:clause([erl_syntax:underscore()], none,
  1142. [ElseContentsAst])
  1143. ]), merge_info(ExpressionInfo, Info)}, TreeWalker1}.
  1144. with_ast(ArgList, Contents, Context, TreeWalker) ->
  1145. {ArgAstList, {ArgInfo, TreeWalker1}} = lists:mapfoldl(fun
  1146. ({{identifier, _, _LocalVarName}, Value}, {AstInfo1, TreeWalker1}) ->
  1147. {{Ast, Info}, TreeWalker2} = value_ast(Value, false, false, Context, TreeWalker1),
  1148. {Ast, {merge_info(AstInfo1, Info), TreeWalker2}}
  1149. end, {#ast_info{}, TreeWalker}, ArgList),
  1150. NewScope = lists:map(fun({{identifier, _, LocalVarName}, _Value}) ->
  1151. {LocalVarName, erl_syntax:variable(lists:concat(["Var_", LocalVarName]))}
  1152. end, ArgList),
  1153. {{InnerAst, InnerInfo}, TreeWalker2} = body_ast(Contents,
  1154. Context#dtl_context{local_scopes = [NewScope|Context#dtl_context.local_scopes]}, TreeWalker1),
  1155. {{erl_syntax:application(
  1156. erl_syntax:fun_expr([
  1157. erl_syntax:clause(lists:map(fun({_, Var}) -> Var end, NewScope), none,
  1158. [InnerAst])]), ArgAstList), merge_info(ArgInfo, InnerInfo)}, TreeWalker2}.
  1159. regroup_ast(ListVariable, GrouperVariable, LocalVarName, Contents, Context, TreeWalker) ->
  1160. {{ListAst, ListInfo}, TreeWalker1} = value_ast(ListVariable, false, true, Context, TreeWalker),
  1161. NewScope = [{LocalVarName, erl_syntax:variable(lists:concat(["Var_", LocalVarName]))}],
  1162. {{InnerAst, InnerInfo}, TreeWalker2} = body_ast(Contents,
  1163. Context#dtl_context{ local_scopes = [NewScope|Context#dtl_context.local_scopes] }, TreeWalker1),
  1164. Ast = {erl_syntax:application(
  1165. erl_syntax:fun_expr([
  1166. erl_syntax:clause([erl_syntax:variable(lists:concat(["Var_", LocalVarName]))], none,
  1167. [InnerAst])]),
  1168. [erl_syntax:application(erl_syntax:atom(erlydtl_runtime), erl_syntax:atom(regroup),
  1169. [ListAst, regroup_filter(GrouperVariable,[])])]), merge_info(ListInfo, InnerInfo)},
  1170. {Ast,TreeWalker2}.
  1171. regroup_filter({attribute,{{identifier,_,Ident},Next}},Acc) ->
  1172. regroup_filter(Next,[erl_syntax:atom(Ident)|Acc]);
  1173. regroup_filter({variable,{identifier,_,Var}},Acc) ->
  1174. erl_syntax:list([erl_syntax:atom(Var)|Acc]).
  1175. for_loop_ast(IteratorList, LoopValue, IsReversed, Contents, {EmptyContentsAst, EmptyContentsInfo}, Context, TreeWalker) ->
  1176. Vars = lists:map(fun({identifier, _, Iterator}) ->
  1177. erl_syntax:variable(lists:concat(["Var_", Iterator]))
  1178. end, IteratorList),
  1179. {{InnerAst, Info}, TreeWalker1} = body_ast(Contents,
  1180. Context#dtl_context{local_scopes = [
  1181. [{'forloop', erl_syntax:variable("Counters")} | lists:map(
  1182. fun({identifier, _, Iterator}) ->
  1183. {Iterator, erl_syntax:variable(lists:concat(["Var_", Iterator]))}
  1184. end, IteratorList)] | Context#dtl_context.local_scopes]}, TreeWalker),
  1185. CounterAst = erl_syntax:application(erl_syntax:atom(erlydtl_runtime),
  1186. erl_syntax:atom(increment_counter_stats), [erl_syntax:variable("Counters")]),
  1187. {{LoopValueAst, LoopValueInfo}, TreeWalker2} = value_ast(LoopValue, false, true, Context, TreeWalker1),
  1188. LoopValueAst0 = case IsReversed of
  1189. true -> erl_syntax:application(erl_syntax:atom(lists), erl_syntax:atom(reverse), [LoopValueAst]);
  1190. false -> LoopValueAst
  1191. end,
  1192. CounterVars0 = case resolve_scoped_variable_ast('forloop', Context) of
  1193. undefined ->
  1194. erl_syntax:application(erl_syntax:atom(erlydtl_runtime), erl_syntax:atom(init_counter_stats), [LoopValueAst0]);
  1195. Value ->
  1196. erl_syntax:application(erl_syntax:atom(erlydtl_runtime), erl_syntax:atom(init_counter_stats), [LoopValueAst0, Value])
  1197. end,
  1198. {{erl_syntax:case_expr(
  1199. erl_syntax:application(
  1200. erl_syntax:atom('erlydtl_runtime'), erl_syntax:atom('forloop'),
  1201. [erl_syntax:fun_expr([
  1202. erl_syntax:clause([erl_syntax:tuple(Vars), erl_syntax:variable("Counters")], none,
  1203. [erl_syntax:tuple([InnerAst, CounterAst])]),
  1204. erl_syntax:clause(case Vars of [H] -> [H, erl_syntax:variable("Counters")];
  1205. _ -> [erl_syntax:list(Vars), erl_syntax:variable("Counters")] end, none,
  1206. [erl_syntax:tuple([InnerAst, CounterAst])])
  1207. ]),
  1208. CounterVars0, LoopValueAst0]),
  1209. [erl_syntax:clause(
  1210. [erl_syntax:tuple([erl_syntax:underscore(),
  1211. erl_syntax:list([erl_syntax:tuple([erl_syntax:atom(counter), erl_syntax:integer(1)])],
  1212. erl_syntax:underscore())])],
  1213. none, [EmptyContentsAst]),
  1214. erl_syntax:clause(
  1215. [erl_syntax:tuple([erl_syntax:variable("L"), erl_syntax:underscore()])],
  1216. none, [erl_syntax:variable("L")])]
  1217. ),
  1218. merge_info(merge_info(Info, EmptyContentsInfo), LoopValueInfo)
  1219. }, TreeWalker2}.
  1220. ifchanged_values_ast(Values, {IfContentsAst, IfContentsInfo}, {ElseContentsAst, ElseContentsInfo}, Context, TreeWalker) ->
  1221. Info = merge_info(IfContentsInfo, ElseContentsInfo),
  1222. ValueAstFun = fun(Expr, {LTreeWalker, LInfo, Acc}) ->
  1223. {{EAst, EInfo}, ETw} = value_ast(Expr, false, true, Context, LTreeWalker),
  1224. {ETw, merge_info(LInfo, EInfo), [erl_syntax:tuple([erl_syntax:integer(erlang:phash2(Expr)), EAst])|Acc]} end,
  1225. {TreeWalker1, MergedInfo, Changed} = lists:foldl(ValueAstFun, {TreeWalker, Info, []}, Values),
  1226. {{erl_syntax:case_expr(erl_syntax:application(erl_syntax:atom(erlydtl_runtime), erl_syntax:atom(ifchanged), [erl_syntax:list(Changed)]),
  1227. [erl_syntax:clause([erl_syntax:atom(true)], none,
  1228. [IfContentsAst]),
  1229. erl_syntax:clause([erl_syntax:underscore()], none,
  1230. [ElseContentsAst])
  1231. ]), MergedInfo}, TreeWalker1}.
  1232. ifchanged_contents_ast(Contents, {IfContentsAst, IfContentsInfo}, {ElseContentsAst, ElseContentsInfo}, _Context, TreeWalker) ->
  1233. Info = merge_info(IfContentsInfo, ElseContentsInfo),
  1234. Key = erl_syntax:integer(erlang:phash2(Contents)),
  1235. {{erl_syntax:case_expr(erl_syntax:application(erl_syntax:atom(erlydtl_runtime), erl_syntax:atom(ifchanged), [erl_syntax:list([erl_syntax:tuple([Key, IfContentsAst])])]),
  1236. [erl_syntax:clause([erl_syntax:atom(true)], none,
  1237. [IfContentsAst]),
  1238. erl_syntax:clause([erl_syntax:underscore()], none,
  1239. [ElseContentsAst])
  1240. ]), Info}, TreeWalker}.
  1241. cycle_ast(Names, Context, TreeWalker) ->
  1242. {NamesTuple, VarNames} = lists:mapfoldl(fun
  1243. ({string_literal, _, Str}, VarNamesAcc) ->
  1244. {{S, _}, _} = string_ast(unescape_string_literal(Str), Context, TreeWalker),
  1245. {S, VarNamesAcc};
  1246. ({variable, _}=Var, VarNamesAcc) ->
  1247. {{V, #ast_info{ var_names=[VarName] }}, _} = resolve_variable_ast(Var, Context, TreeWalker, true),
  1248. {V, [VarName|VarNamesAcc]};
  1249. ({number_literal, _, Num}, VarNamesAcc) ->
  1250. {format(erl_syntax:integer(Num), Context, TreeWalker), VarNamesAcc};
  1251. (_, VarNamesAcc) ->
  1252. {[], VarNamesAcc}
  1253. end, [], Names),
  1254. {{erl_syntax:application(
  1255. erl_syntax:atom('erlydtl_runtime'), erl_syntax:atom('cycle'),
  1256. [erl_syntax:tuple(NamesTuple), erl_syntax:variable("Counters")]), #ast_info{ var_names = VarNames }}, TreeWalker}.
  1257. %% Older Django templates treat cycle with comma-delimited elements as strings
  1258. cycle_compat_ast(Names, Context, TreeWalker) ->
  1259. NamesTuple = lists:map(fun
  1260. ({identifier, _, X}) ->
  1261. {{S, _}, _} = string_ast(X, Context, TreeWalker),
  1262. S
  1263. end, Names),
  1264. {{erl_syntax:application(
  1265. erl_syntax:atom('erlydtl_runtime'), erl_syntax:atom('cycle'),
  1266. [erl_syntax:tuple(NamesTuple), erl_syntax:variable("Counters")]), #ast_info{}}, TreeWalker}.
  1267. now_ast(FormatString, Context, TreeWalker) ->
  1268. % Note: we can't use unescape_string_literal here
  1269. % because we want to allow escaping in the format string.
  1270. % We only want to remove the surrounding escapes,
  1271. % i.e. \"foo\" becomes "foo"
  1272. UnescapeOuter = string:strip(FormatString, both, 34),
  1273. {{StringAst, Info}, TreeWalker1} = string_ast(UnescapeOuter, Context, TreeWalker),
  1274. {{erl_syntax:application(
  1275. erl_syntax:atom(erlydtl_dateformat),
  1276. erl_syntax:atom(format),
  1277. [StringAst]), Info}, TreeWalker1}.
  1278. spaceless_ast(Contents, Context, TreeWalker) ->
  1279. {{Ast, Info}, TreeWalker1} = body_ast(Contents, Context, TreeWalker),
  1280. {{erl_syntax:application(
  1281. erl_syntax:atom(erlydtl_runtime),
  1282. erl_syntax:atom(spaceless),
  1283. [Ast]), Info}, TreeWalker1}.
  1284. unescape_string_literal(String) ->
  1285. unescape_string_literal(string:strip(String, both, 34), [], noslash).
  1286. unescape_string_literal([], Acc, noslash) ->
  1287. lists:reverse(Acc);
  1288. unescape_string_literal([$\\ | Rest], Acc, noslash) ->
  1289. unescape_string_literal(Rest, Acc, slash);
  1290. unescape_string_literal([C | Rest], Acc, noslash) ->
  1291. unescape_string_literal(Rest, [C | Acc], noslash);
  1292. unescape_string_literal("n" ++ Rest, Acc, slash) ->
  1293. unescape_string_literal(Rest, [$\n | Acc], noslash);
  1294. unescape_string_literal("r" ++ Rest, Acc, slash) ->
  1295. unescape_string_literal(Rest, [$\r | Acc], noslash);
  1296. unescape_string_literal("t" ++ Rest, Acc, slash) ->
  1297. unescape_string_literal(Rest, [$\t | Acc], noslash);
  1298. unescape_string_literal([C | Rest], Acc, slash) ->
  1299. unescape_string_literal(Rest, [C | Acc], noslash).
  1300. full_path(File, DocRoot) ->
  1301. case filename:absname(File) of
  1302. File -> File;
  1303. _ -> filename:join([DocRoot, File])
  1304. end.
  1305. %%-------------------------------------------------------------------
  1306. %% Custom tags
  1307. %%-------------------------------------------------------------------
  1308. tag_ast(Name, Args, Context, TreeWalker) ->
  1309. {{InterpretedArgs, AstInfo1}, TreeWalker1} =
  1310. lists:foldr(
  1311. fun ({{identifier, _, Key}, {trans, StringLiteral}}, {{ArgsAcc, AstInfoAcc}, TreeWalkerAcc}) ->
  1312. {{TransAst, TransAstInfo}, TreeWalker0} = translated_ast(StringLiteral, Context, TreeWalkerAcc),
  1313. {{[erl_syntax:tuple([erl_syntax:atom(Key), TransAst])|ArgsAcc], merge_info(TransAstInfo, AstInfoAcc)}, TreeWalker0};
  1314. ({{identifier, _, Key}, Value}, {{ArgsAcc, AstInfoAcc}, TreeWalkerAcc}) ->
  1315. {{Ast0, AstInfo0}, TreeWalker0} = value_ast(Value, false, false, Context, TreeWalkerAcc),
  1316. {{[erl_syntax:tuple([erl_syntax:atom(Key), Ast0])|ArgsAcc], merge_info(AstInfo0, AstInfoAcc)}, TreeWalker0};
  1317. ({extension, Tag}, {{ArgsAcc, AstInfoAcc}, TreeWalkerAcc}) ->
  1318. {{ExtAst, ExtInfo}, ExtTreeWalker} = extension_ast(Tag, Context, TreeWalkerAcc),
  1319. {{[ExtAst|ArgsAcc], merge_info(ExtInfo, AstInfoAcc)}, ExtTreeWalker}
  1320. end, {{[], #ast_info{}}, TreeWalker}, Args),
  1321. {RenderAst, RenderInfo} = custom_tags_modules_ast(Name, InterpretedArgs, Context),
  1322. {{RenderAst, merge_info(AstInfo1, RenderInfo)}, TreeWalker1}.
  1323. custom_tags_modules_ast(Name, InterpretedArgs, #dtl_context{ custom_tags_modules = [], is_compiling_dir = false }) ->
  1324. {erl_syntax:application(none, erl_syntax:atom(render_tag),
  1325. [erl_syntax:atom(Name), erl_syntax:list(InterpretedArgs),
  1326. erl_syntax:variable("RenderOptions")]),
  1327. #ast_info{custom_tags = [Name]}};
  1328. custom_tags_modules_ast(Name, InterpretedArgs, #dtl_context{ custom_tags_modules = [], is_compiling_dir = true, module = Module }) ->
  1329. {erl_syntax:application(erl_syntax:atom(Module), erl_syntax:atom(Name),
  1330. [erl_syntax:list(InterpretedArgs), erl_syntax:variable("RenderOptions")]),
  1331. #ast_info{ custom_tags = [Name] }};
  1332. custom_tags_modules_ast(Name, InterpretedArgs, #dtl_context{ custom_tags_modules = [Module|Rest] } = Context) ->
  1333. try lists:max([I || {N,I} <- Module:module_info(exports), N =:= Name]) of
  1334. 2 ->
  1335. {erl_syntax:application(erl_syntax:atom(Module), erl_syntax:atom(Name),
  1336. [erl_syntax:list(InterpretedArgs),
  1337. erl_syntax:variable("RenderOptions")]), #ast_info{}};
  1338. 1 ->
  1339. {erl_syntax:application(erl_syntax:atom(Module), erl_syntax:atom(Name),
  1340. [erl_syntax:list(InterpretedArgs)]), #ast_info{}};
  1341. I ->
  1342. throw({unsupported_custom_tag_fun, {Module, Name, I}})
  1343. catch _:function_clause ->
  1344. custom_tags_modules_ast(Name, InterpretedArgs,
  1345. Context#dtl_context{ custom_tags_modules = Rest })
  1346. end.
  1347. print(true, Fmt, Args) ->
  1348. io:format(Fmt, Args);
  1349. print(_, _Fmt, _Args) ->
  1350. ok.
  1351. call_ast(Module, TreeWalkerAcc) ->
  1352. call_ast(Module, erl_syntax:variable("_Variables"), #ast_info{}, TreeWalkerAcc).
  1353. call_with_ast(Module, Variable, Context, TreeWalker) ->
  1354. {{VarAst, VarInfo}, TreeWalker2} = resolve_variable_ast(Variable, Context, TreeWalker, false),
  1355. call_ast(Module, VarAst, VarInfo, TreeWalker2).
  1356. call_ast(Module, Variable, AstInfo, TreeWalker) ->
  1357. AppAst = erl_syntax:application(
  1358. erl_syntax:atom(Module),
  1359. erl_syntax:atom(render),
  1360. [Variable, erl_syntax:variable("RenderOptions")]),
  1361. RenderedAst = erl_syntax:variable("Rendered"),
  1362. OkAst = erl_syntax:clause(
  1363. [erl_syntax:tuple([erl_syntax:atom(ok), RenderedAst])],
  1364. none,
  1365. [RenderedAst]),
  1366. ReasonAst = erl_syntax:variable("Reason"),
  1367. ErrStrAst = erl_syntax:application(
  1368. erl_syntax:atom(io_lib),
  1369. erl_syntax:atom(format),
  1370. [erl_syntax:string("error: ~p"), erl_syntax:list([ReasonAst])]),
  1371. ErrorAst = erl_syntax:clause(
  1372. [erl_syntax:tuple([erl_syntax:atom(error), ReasonAst])],
  1373. none,
  1374. [ErrStrAst]),
  1375. CallAst = erl_syntax:case_expr(AppAst, [OkAst, ErrorAst]),
  1376. with_dependencies(Module:dependencies(), {{CallAst, AstInfo}, TreeWalker}).