510 lines
15 KiB
Elixir
510 lines
15 KiB
Elixir
|
defmodule Phoenix.Template do
|
||
|
@moduledoc """
|
||
|
Templates are markup languages that are compiled to Elixir code.
|
||
|
|
||
|
This module provides functions for loading and compiling templates
|
||
|
from disk. A markup language is compiled to Elixir code via an engine.
|
||
|
See `Phoenix.Template.Engine`.
|
||
|
|
||
|
In practice, developers rarely use `Phoenix.Template` directly. Instead,
|
||
|
libraries such as `Phoenix.View` and `Phoenix.LiveView` use it as a
|
||
|
building block.
|
||
|
|
||
|
## Custom Template Engines
|
||
|
|
||
|
Phoenix supports custom template engines. Engines tell
|
||
|
Phoenix how to convert a template path into quoted expressions.
|
||
|
See `Phoenix.Template.Engine` for more information on
|
||
|
the API required to be implemented by custom engines.
|
||
|
|
||
|
Once a template engine is defined, you can tell Phoenix
|
||
|
about it via the template engines option:
|
||
|
|
||
|
config :phoenix, :template_engines,
|
||
|
eex: Phoenix.Template.EExEngine,
|
||
|
exs: Phoenix.Template.ExsEngine
|
||
|
|
||
|
## Format encoders
|
||
|
|
||
|
Besides template engines, Phoenix has the concept of format encoders.
|
||
|
Format encoders work per format and are responsible for encoding a
|
||
|
given format to a string. For example, when rendering JSON, your
|
||
|
templates may return a regular Elixir map. Then the JSON format
|
||
|
encoder is invoked to convert it to JSON.
|
||
|
|
||
|
A format encoder must export a function called `encode_to_iodata!/1`
|
||
|
which receives the rendering artifact and returns iodata.
|
||
|
|
||
|
New encoders can be added via the format encoder option:
|
||
|
|
||
|
config :phoenix_template, :format_encoders,
|
||
|
html: Phoenix.HTML.Engine
|
||
|
|
||
|
"""
|
||
|
|
||
|
@type path :: binary
|
||
|
@type root :: binary
|
||
|
|
||
|
@default_pattern "*"
|
||
|
|
||
|
@doc """
|
||
|
Ensure `__mix_recompile__?/0` will be defined.
|
||
|
"""
|
||
|
defmacro __using__(_opts) do
|
||
|
quote do
|
||
|
Phoenix.Template.__idempotent_setup__(__MODULE__, %{})
|
||
|
end
|
||
|
end
|
||
|
|
||
|
@doc """
|
||
|
A convenience macro for embeding templates as functions.
|
||
|
|
||
|
This macro is built on top of the more general `compile_all/3`
|
||
|
functionality.
|
||
|
|
||
|
## Options
|
||
|
|
||
|
* `:root` - The root directory to embed files. Defaults to the current
|
||
|
module's directory (`__DIR__`)
|
||
|
* `:suffix` - The string value to append to embedded function names. By
|
||
|
default, function names will be the name of the template file excluding
|
||
|
the format and engine.
|
||
|
|
||
|
A wildcard pattern may be used to select all files within a directory tree.
|
||
|
For example, imagine a directory listing:
|
||
|
|
||
|
├── pages
|
||
|
│ ├── about.html.heex
|
||
|
│ └── sitemap.xml.eex
|
||
|
|
||
|
Then to embed the templates in your module:
|
||
|
|
||
|
defmodule MyAppWeb.Renderer do
|
||
|
import Phoenix.Template, only: [embed_templates: 1]
|
||
|
embed_templates "pages/*"
|
||
|
end
|
||
|
|
||
|
Now, your module will have a `about/1` and `sitemap/1` functions.
|
||
|
Note that functions across different formats were embedded. In case
|
||
|
you want to distinguish between them, you can give a more specific
|
||
|
pattern:
|
||
|
|
||
|
defmodule MyAppWeb.Emails do
|
||
|
import Phoenix.Template, only: [embed_templates: 2]
|
||
|
|
||
|
embed_templates "pages/*.html", suffix: "_html"
|
||
|
embed_templates "pages/*.xml", suffix: "_xml"
|
||
|
end
|
||
|
|
||
|
Now the functions will be `about_html` and `sitemap_xml`.
|
||
|
"""
|
||
|
@doc type: :macro
|
||
|
defmacro embed_templates(pattern, opts \\ []) do
|
||
|
quote bind_quoted: [pattern: pattern, opts: opts] do
|
||
|
Phoenix.Template.compile_all(
|
||
|
&Phoenix.Template.__embed__(&1, opts[:suffix]),
|
||
|
Path.expand(opts[:root] || __DIR__, __DIR__),
|
||
|
pattern
|
||
|
)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
@doc false
|
||
|
def __embed__(path, suffix),
|
||
|
do:
|
||
|
path
|
||
|
|> Path.basename()
|
||
|
|> Path.rootname()
|
||
|
|> Path.rootname()
|
||
|
|> Kernel.<>(suffix || "")
|
||
|
|
||
|
@doc """
|
||
|
Renders the template and returns iodata.
|
||
|
"""
|
||
|
def render_to_iodata(module, template, format, assign) do
|
||
|
module
|
||
|
|> render(template, format, assign)
|
||
|
|> encode(format)
|
||
|
end
|
||
|
|
||
|
@doc """
|
||
|
Renders the template to string.
|
||
|
"""
|
||
|
def render_to_string(module, template, format, assign) do
|
||
|
module
|
||
|
|> render_to_iodata(template, format, assign)
|
||
|
|> IO.iodata_to_binary()
|
||
|
end
|
||
|
|
||
|
@doc """
|
||
|
Renders template from module.
|
||
|
|
||
|
For a module called `MyApp.FooHTML` and template "index.html.heex",
|
||
|
it will:
|
||
|
|
||
|
* First attempt to call `MyApp.FooHTML.index(assigns)`
|
||
|
|
||
|
* Then fallback to `MyApp.FooHTML.render("index.html", assigns)`
|
||
|
|
||
|
* Raise otherwise
|
||
|
|
||
|
It expects the HTML module, the template as a string, the format, and a
|
||
|
set of assigns.
|
||
|
|
||
|
Notice that this function returns the inner representation of a
|
||
|
template. If you want the encoded template as a result, use
|
||
|
`render_to_iodata/4` instead.
|
||
|
|
||
|
## Examples
|
||
|
|
||
|
Phoenix.Template.render(YourApp.UserView, "index", "html", name: "John Doe")
|
||
|
#=> {:safe, "Hello John Doe"}
|
||
|
|
||
|
## Assigns
|
||
|
|
||
|
Assigns are meant to be user data that will be available in templates.
|
||
|
However, there are keys under assigns that are specially handled by
|
||
|
Phoenix, they are:
|
||
|
|
||
|
* `:layout` - tells Phoenix to wrap the rendered result in the
|
||
|
given layout. See next section
|
||
|
|
||
|
## Layouts
|
||
|
|
||
|
Templates can be rendered within other templates using the `:layout`
|
||
|
option. `:layout` accepts a tuple of the form
|
||
|
`{LayoutModule, "template.extension"}`.
|
||
|
|
||
|
To template that goes inside the layout will be placed in the `@inner_content`
|
||
|
assign:
|
||
|
|
||
|
<%= @inner_content %>
|
||
|
|
||
|
"""
|
||
|
def render(module, template, format, assigns) do
|
||
|
assigns
|
||
|
|> Map.new()
|
||
|
|> Map.pop(:layout, false)
|
||
|
|> render_within_layout(module, template, format)
|
||
|
end
|
||
|
|
||
|
defp render_within_layout({false, assigns}, module, template, format) do
|
||
|
render_with_fallback(module, template, format, assigns)
|
||
|
end
|
||
|
|
||
|
defp render_within_layout({{layout_mod, layout_tpl}, assigns}, module, template, format)
|
||
|
when is_atom(layout_mod) and is_binary(layout_tpl) do
|
||
|
content = render_with_fallback(module, template, format, assigns)
|
||
|
assigns = Map.put(assigns, :inner_content, content)
|
||
|
render_with_fallback(layout_mod, layout_tpl, format, assigns)
|
||
|
end
|
||
|
|
||
|
defp render_within_layout({layout, _assigns}, _module, _template, _format) do
|
||
|
raise ArgumentError, """
|
||
|
invalid value for reserved key :layout in Phoenix.Template.render/4 assigns.
|
||
|
:layout accepts a tuple of the form {LayoutModule, "template.extension"},
|
||
|
got: #{inspect(layout)}
|
||
|
"""
|
||
|
end
|
||
|
|
||
|
defp encode(content, format) do
|
||
|
if encoder = format_encoder(format) do
|
||
|
encoder.encode_to_iodata!(content)
|
||
|
else
|
||
|
content
|
||
|
end
|
||
|
end
|
||
|
|
||
|
defp render_with_fallback(module, template, format, assigns)
|
||
|
when is_atom(module) and is_binary(template) and is_binary(format) and is_map(assigns) do
|
||
|
:erlang.module_loaded(module) or :code.ensure_loaded(module)
|
||
|
|
||
|
try do
|
||
|
String.to_existing_atom(template)
|
||
|
catch
|
||
|
_, _ -> fallback_render(module, template, format, assigns)
|
||
|
else
|
||
|
atom ->
|
||
|
if function_exported?(module, atom, 1) do
|
||
|
apply(module, atom, [assigns])
|
||
|
else
|
||
|
fallback_render(module, template, format, assigns)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
@compile {:inline, fallback_render: 4}
|
||
|
defp fallback_render(module, template, format, assigns) do
|
||
|
if function_exported?(module, :render, 2) do
|
||
|
module.render(template <> "." <> format, assigns)
|
||
|
else
|
||
|
reason =
|
||
|
if Code.ensure_loaded?(module) do
|
||
|
" (the module exists but does not define #{template}/1 nor render/2)"
|
||
|
else
|
||
|
" (the module does not exist)"
|
||
|
end
|
||
|
|
||
|
raise ArgumentError,
|
||
|
"no \"#{template}\" #{format} template defined for #{inspect(module)} #{reason}"
|
||
|
end
|
||
|
end
|
||
|
|
||
|
## Configuration API
|
||
|
|
||
|
@doc """
|
||
|
Returns the format encoder for the given template.
|
||
|
"""
|
||
|
@spec format_encoder(format :: String.t()) :: module | nil
|
||
|
def format_encoder(format) when is_binary(format) do
|
||
|
Map.get(compiled_format_encoders(), format)
|
||
|
end
|
||
|
|
||
|
defp compiled_format_encoders do
|
||
|
case Application.fetch_env(:phoenix_template, :compiled_format_encoders) do
|
||
|
{:ok, encoders} ->
|
||
|
encoders
|
||
|
|
||
|
:error ->
|
||
|
encoders =
|
||
|
default_encoders()
|
||
|
|> Keyword.merge(raw_config(:format_encoders, []))
|
||
|
|> Enum.filter(fn {_, v} -> v end)
|
||
|
|> Enum.into(%{}, fn {k, v} -> {to_string(k), v} end)
|
||
|
|
||
|
Application.put_env(:phoenix_template, :compiled_format_encoders, encoders)
|
||
|
encoders
|
||
|
end
|
||
|
end
|
||
|
|
||
|
defp default_encoders do
|
||
|
[html: Phoenix.HTML.Engine, json: json_library(), js: Phoenix.HTML.Engine]
|
||
|
end
|
||
|
|
||
|
defp json_library() do
|
||
|
Application.get_env(:phoenix_template, :json_library) ||
|
||
|
deprecated_config(:phoenix_view, :json_library) ||
|
||
|
Application.get_env(:phoenix, :json_library, Jason)
|
||
|
end
|
||
|
|
||
|
@doc """
|
||
|
Returns a keyword list with all template engines
|
||
|
extensions followed by their modules.
|
||
|
"""
|
||
|
@spec engines() :: %{atom => module}
|
||
|
def engines do
|
||
|
compiled_engines()
|
||
|
end
|
||
|
|
||
|
defp compiled_engines do
|
||
|
case Application.fetch_env(:phoenix_template, :compiled_template_engines) do
|
||
|
{:ok, engines} ->
|
||
|
engines
|
||
|
|
||
|
:error ->
|
||
|
engines =
|
||
|
default_engines()
|
||
|
|> Keyword.merge(raw_config(:template_engines, []))
|
||
|
|> Enum.filter(fn {_, v} -> v end)
|
||
|
|> Enum.into(%{})
|
||
|
|
||
|
Application.put_env(:phoenix_template, :compiled_template_engines, engines)
|
||
|
engines
|
||
|
end
|
||
|
end
|
||
|
|
||
|
defp default_engines do
|
||
|
[
|
||
|
eex: Phoenix.Template.EExEngine,
|
||
|
exs: Phoenix.Template.ExsEngine,
|
||
|
leex: Phoenix.LiveView.Engine,
|
||
|
heex: Phoenix.LiveView.HTMLEngine
|
||
|
]
|
||
|
end
|
||
|
|
||
|
defp raw_config(name, fallback) do
|
||
|
Application.get_env(:phoenix_template, name) ||
|
||
|
deprecated_config(:phoenix_view, name) ||
|
||
|
Application.get_env(:phoenix, name, fallback)
|
||
|
end
|
||
|
|
||
|
defp deprecated_config(app, name) do
|
||
|
if value = Application.get_env(app, name) do
|
||
|
IO.warn(
|
||
|
"config :#{app}, :#{name} is deprecated, please use config :phoenix_template, :#{name} instead"
|
||
|
)
|
||
|
|
||
|
value
|
||
|
end
|
||
|
end
|
||
|
|
||
|
## Lookup API
|
||
|
|
||
|
@doc """
|
||
|
Returns all template paths in a given template root.
|
||
|
"""
|
||
|
@spec find_all(root, pattern :: String.t(), %{atom => module}) :: [path]
|
||
|
def find_all(root, pattern \\ @default_pattern, engines \\ engines()) do
|
||
|
extensions = engines |> Map.keys() |> Enum.join(",")
|
||
|
|
||
|
root
|
||
|
|> Path.join(pattern <> ".{#{extensions}}")
|
||
|
|> Path.wildcard()
|
||
|
end
|
||
|
|
||
|
@doc """
|
||
|
Returns the hash of all template paths in the given root.
|
||
|
|
||
|
Used by Phoenix to check if a given root path requires recompilation.
|
||
|
"""
|
||
|
@spec hash(root, pattern :: String.t(), %{atom => module}) :: binary
|
||
|
def hash(root, pattern \\ @default_pattern, engines \\ engines()) do
|
||
|
find_all(root, pattern, engines)
|
||
|
|> Enum.sort()
|
||
|
|> :erlang.md5()
|
||
|
end
|
||
|
|
||
|
@doc """
|
||
|
Compiles a function for each template in the given `root`.
|
||
|
|
||
|
`converter` is an anonymous function that receives the template path
|
||
|
and returns the function name (as a string).
|
||
|
|
||
|
For example, to compile all `.eex` templates in a given directory,
|
||
|
you might do:
|
||
|
|
||
|
Phoenix.Template.compile_all(
|
||
|
&(&1 |> Path.basename() |> Path.rootname(".eex")),
|
||
|
__DIR__,
|
||
|
"*.eex"
|
||
|
)
|
||
|
|
||
|
If the directory has templates named `foo.eex` and `bar.eex`,
|
||
|
they will be compiled into the functions `foo/1` and `bar/1`
|
||
|
that receive the template `assigns` as argument.
|
||
|
|
||
|
You may optionally pass a keyword list of engines. If a list
|
||
|
is given, we will lookup and compile only this subset of engines.
|
||
|
If none is passed (`nil`), the default list returned by `engines/0`
|
||
|
is used.
|
||
|
"""
|
||
|
defmacro compile_all(converter, root, pattern \\ @default_pattern, engines \\ nil) do
|
||
|
quote bind_quoted: binding() do
|
||
|
for {path, name, body} <-
|
||
|
Phoenix.Template.__compile_all__(__MODULE__, converter, root, pattern, engines) do
|
||
|
@external_resource path
|
||
|
@file path
|
||
|
def unquote(String.to_atom(name))(var!(assigns)) do
|
||
|
_ = var!(assigns)
|
||
|
unquote(body)
|
||
|
end
|
||
|
|
||
|
{name, path}
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
@doc false
|
||
|
def __compile_all__(module, converter, root, pattern, given_engines) do
|
||
|
engines = given_engines || engines()
|
||
|
paths = find_all(root, pattern, engines)
|
||
|
|
||
|
{triplets, {paths, engines}} =
|
||
|
Enum.map_reduce(paths, {[], %{}}, fn path, {acc_paths, acc_engines} ->
|
||
|
ext = Path.extname(path) |> String.trim_leading(".") |> String.to_atom()
|
||
|
engine = Map.fetch!(engines, ext)
|
||
|
name = converter.(path)
|
||
|
body = engine.compile(path, name)
|
||
|
map = {path, name, body}
|
||
|
reduce = {[path | acc_paths], Map.put(acc_engines, engine, true)}
|
||
|
{map, reduce}
|
||
|
end)
|
||
|
|
||
|
# Store the engines so we define compile-time deps
|
||
|
__idempotent_setup__(module, engines)
|
||
|
|
||
|
# Store the hashes so we define __mix_recompile__?
|
||
|
hash = paths |> Enum.sort() |> :erlang.md5()
|
||
|
|
||
|
args =
|
||
|
if given_engines, do: [root, pattern, Macro.escape(given_engines)], else: [root, pattern]
|
||
|
|
||
|
Module.put_attribute(module, :phoenix_templates_hashes, {hash, args})
|
||
|
triplets
|
||
|
end
|
||
|
|
||
|
@doc false
|
||
|
def __idempotent_setup__(module, engines) do
|
||
|
# Store the used engines so they become requires on before_compile
|
||
|
if used_engines = Module.get_attribute(module, :phoenix_templates_engines) do
|
||
|
Module.put_attribute(module, :phoenix_templates_engines, Map.merge(used_engines, engines))
|
||
|
else
|
||
|
Module.register_attribute(module, :phoenix_templates_hashes, accumulate: true)
|
||
|
Module.put_attribute(module, :phoenix_templates_engines, engines)
|
||
|
Module.put_attribute(module, :before_compile, Phoenix.Template)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
@doc false
|
||
|
defmacro __before_compile__(env) do
|
||
|
hashes = Module.get_attribute(env.module, :phoenix_templates_hashes)
|
||
|
engines = Module.get_attribute(env.module, :phoenix_templates_engines)
|
||
|
|
||
|
body =
|
||
|
Enum.reduce(hashes, false, fn {hash, args}, acc ->
|
||
|
quote do
|
||
|
unquote(acc) or unquote(hash) != Phoenix.Template.hash(unquote_splicing(args))
|
||
|
end
|
||
|
end)
|
||
|
|
||
|
compile_time_deps =
|
||
|
for {engine, _} <- engines do
|
||
|
quote do
|
||
|
unquote(engine).__info__(:module)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
quote do
|
||
|
unquote(compile_time_deps)
|
||
|
|
||
|
@doc false
|
||
|
def __mix_recompile__? do
|
||
|
unquote(body)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
## Deprecated API
|
||
|
|
||
|
@deprecated "Use Phoenix.View.template_path_to_name/3"
|
||
|
def template_path_to_name(path, root) do
|
||
|
path
|
||
|
|> Path.rootname()
|
||
|
|> Path.relative_to(root)
|
||
|
end
|
||
|
|
||
|
@deprecated "Use Phoenix.View.module_to_template_root/3"
|
||
|
def module_to_template_root(module, base, suffix) do
|
||
|
module
|
||
|
|> unsuffix(suffix)
|
||
|
|> Module.split()
|
||
|
|> Enum.drop(length(Module.split(base)))
|
||
|
|> Enum.map(&Macro.underscore/1)
|
||
|
|> join_paths()
|
||
|
end
|
||
|
|
||
|
defp join_paths([]), do: ""
|
||
|
defp join_paths(paths), do: Path.join(paths)
|
||
|
|
||
|
defp unsuffix(value, suffix) do
|
||
|
string = to_string(value)
|
||
|
suffix_size = byte_size(suffix)
|
||
|
prefix_size = byte_size(string) - suffix_size
|
||
|
|
||
|
case string do
|
||
|
<<prefix::binary-size(prefix_size), ^suffix::binary>> -> prefix
|
||
|
_ -> string
|
||
|
end
|
||
|
end
|
||
|
end
|