defprotocol Phoenix.HTML.Safe do
@moduledoc """
Defines the HTML safe protocol.
In order to promote HTML safety, Phoenix templates
do not use `Kernel.to_string/1` to convert data types to
strings in templates. Instead, Phoenix uses this
protocol which must be implemented by data structures
and guarantee that a HTML safe representation is returned.
Furthermore, this protocol relies on iodata, which provides
better performance when sending or streaming data to the client.
"""
def to_iodata(data)
end
defimpl Phoenix.HTML.Safe, for: Atom do
def to_iodata(nil), do: ""
def to_iodata(atom), do: Phoenix.HTML.Engine.html_escape(Atom.to_string(atom))
end
defimpl Phoenix.HTML.Safe, for: BitString do
defdelegate to_iodata(data), to: Phoenix.HTML.Engine, as: :html_escape
end
defimpl Phoenix.HTML.Safe, for: Time do
defdelegate to_iodata(data), to: Time, as: :to_iso8601
end
defimpl Phoenix.HTML.Safe, for: Date do
defdelegate to_iodata(data), to: Date, as: :to_iso8601
end
defimpl Phoenix.HTML.Safe, for: NaiveDateTime do
defdelegate to_iodata(data), to: NaiveDateTime, as: :to_iso8601
end
defimpl Phoenix.HTML.Safe, for: DateTime do
def to_iodata(data) do
# Call escape in case someone can inject reserved
# characters in the timezone or its abbreviation
Phoenix.HTML.Engine.html_escape(DateTime.to_iso8601(data))
end
end
defimpl Phoenix.HTML.Safe, for: List do
def to_iodata([h | t]) do
[to_iodata(h) | to_iodata(t)]
end
def to_iodata([]) do
[]
end
def to_iodata(?<), do: "<"
def to_iodata(?>), do: ">"
def to_iodata(?&), do: "&"
def to_iodata(?"), do: """
def to_iodata(?'), do: "'"
def to_iodata(h) when is_integer(h) and h <= 255 do
h
end
def to_iodata(h) when is_integer(h) do
raise ArgumentError,
"lists in Phoenix.HTML templates only support iodata, and not chardata. Integers may only represent bytes. " <>
"It's likely you meant to pass a string with double quotes instead of a char list with single quotes."
end
def to_iodata(h) when is_binary(h) do
Phoenix.HTML.Engine.html_escape(h)
end
def to_iodata({:safe, data}) do
data
end
def to_iodata(other) do
raise ArgumentError,
"lists in Phoenix.HTML and templates may only contain integers representing bytes, binaries or other lists, " <>
"got invalid entry: #{inspect(other)}"
end
end
defimpl Phoenix.HTML.Safe, for: Integer do
defdelegate to_iodata(data), to: Integer, as: :to_string
end
defimpl Phoenix.HTML.Safe, for: Float do
defdelegate to_iodata(data), to: Float, as: :to_string
end
defimpl Phoenix.HTML.Safe, for: Tuple do
def to_iodata({:safe, data}), do: data
def to_iodata(value), do: raise(Protocol.UndefinedError, protocol: @protocol, value: value)
end
defimpl Phoenix.HTML.Safe, for: URI do
def to_iodata(data), do: Phoenix.HTML.Engine.html_escape(URI.to_string(data))
end