95 lines
2.6 KiB
Elixir
95 lines
2.6 KiB
Elixir
|
defmodule Jason.OrderedObject do
|
||
|
@doc """
|
||
|
Struct implementing a JSON object retaining order of properties.
|
||
|
|
||
|
A wrapper around a keyword (that supports non-atom keys) allowing for
|
||
|
proper protocol implementations.
|
||
|
|
||
|
Implements the `Access` behaviour and `Enumerable` protocol with
|
||
|
complexity similar to keywords/lists.
|
||
|
"""
|
||
|
|
||
|
@behaviour Access
|
||
|
|
||
|
@type t :: %__MODULE__{values: [{String.Chars.t(), term()}]}
|
||
|
|
||
|
defstruct values: []
|
||
|
|
||
|
def new(values) when is_list(values) do
|
||
|
%__MODULE__{values: values}
|
||
|
end
|
||
|
|
||
|
@impl Access
|
||
|
def fetch(%__MODULE__{values: values}, key) do
|
||
|
case :lists.keyfind(key, 1, values) do
|
||
|
{_, value} -> {:ok, value}
|
||
|
false -> :error
|
||
|
end
|
||
|
end
|
||
|
|
||
|
@impl Access
|
||
|
def get_and_update(%__MODULE__{values: values} = obj, key, function) do
|
||
|
{result, new_values} = get_and_update(values, [], key, function)
|
||
|
{result, %{obj | values: new_values}}
|
||
|
end
|
||
|
|
||
|
@impl Access
|
||
|
def pop(%__MODULE__{values: values} = obj, key, default \\ nil) do
|
||
|
case :lists.keyfind(key, 1, values) do
|
||
|
{_, value} -> {value, %{obj | values: delete_key(values, key)}}
|
||
|
false -> {default, obj}
|
||
|
end
|
||
|
end
|
||
|
|
||
|
defp get_and_update([{key, current} | t], acc, key, fun) do
|
||
|
case fun.(current) do
|
||
|
{get, value} ->
|
||
|
{get, :lists.reverse(acc, [{key, value} | t])}
|
||
|
|
||
|
:pop ->
|
||
|
{current, :lists.reverse(acc, t)}
|
||
|
|
||
|
other ->
|
||
|
raise "the given function must return a two-element tuple or :pop, got: #{inspect(other)}"
|
||
|
end
|
||
|
end
|
||
|
|
||
|
defp get_and_update([{_, _} = h | t], acc, key, fun), do: get_and_update(t, [h | acc], key, fun)
|
||
|
|
||
|
defp get_and_update([], acc, key, fun) do
|
||
|
case fun.(nil) do
|
||
|
{get, update} ->
|
||
|
{get, [{key, update} | :lists.reverse(acc)]}
|
||
|
|
||
|
:pop ->
|
||
|
{nil, :lists.reverse(acc)}
|
||
|
|
||
|
other ->
|
||
|
raise "the given function must return a two-element tuple or :pop, got: #{inspect(other)}"
|
||
|
end
|
||
|
end
|
||
|
|
||
|
defp delete_key([{key, _} | tail], key), do: delete_key(tail, key)
|
||
|
defp delete_key([{_, _} = pair | tail], key), do: [pair | delete_key(tail, key)]
|
||
|
defp delete_key([], _key), do: []
|
||
|
end
|
||
|
|
||
|
defimpl Enumerable, for: Jason.OrderedObject do
|
||
|
def count(%{values: []}), do: {:ok, 0}
|
||
|
def count(_obj), do: {:error, __MODULE__}
|
||
|
|
||
|
def member?(%{values: []}, _value), do: {:ok, false}
|
||
|
def member?(_obj, _value), do: {:error, __MODULE__}
|
||
|
|
||
|
def slice(%{values: []}), do: {:ok, 0, fn _, _ -> [] end}
|
||
|
def slice(_obj), do: {:error, __MODULE__}
|
||
|
|
||
|
def reduce(%{values: values}, acc, fun), do: Enumerable.List.reduce(values, acc, fun)
|
||
|
end
|
||
|
|
||
|
defimpl Jason.Encoder, for: Jason.OrderedObject do
|
||
|
def encode(%{values: values}, opts) do
|
||
|
Jason.Encode.keyword(values, opts)
|
||
|
end
|
||
|
end
|