我正在与Poison一起使用的伞形应用程序对Json进行编码和解码。为了对ecto模型进行编码,我编写了以下实现:
def defimpl Poison.Encoder, for: Any do
def encode(%{__struct__: App.Models.Task} = struct, options) do
struct =
if struct.geom do
{lon, lat} = struct.geom.coordinates
%{struct | lon: lon, lat: lat}
else
struct
end
enconde_model(struct, options)
end
def encode(%{__struct__: _} = struct, options) do
enconde_model(struct, options)
end
def encode({lon, lat}, options) when is_float(lon) and is_float(lat) do
%{lon: lon, lat: lat}
|> sanitize_map()
|> Poison.Encoder.Map.encode(options)
end
defp enconde_model(model, options) do
model
|> Map.from_struct()
|> sanitize_map()
|> Poison.Encoder.Map.encode(options)
end
defp sanitize_map(map) do
map
|> Enum.filter(
fn({_, %Ecto.Association.NotLoaded{}}) -> false
({:__meta__, _}) -> false
({:__struct__, _}) -> false
({_, _}) -> true
end)
|> Map.new()
end
结束
认为,除非我注释掉上面刚刚显示的代码,否则一旦运行mix release
,程序就会引发错误。
那么,还有另一种无需使用实现即可实现相同功能的方法吗?
答案 0 :(得分:1)
由于覆盖了现有的编码器实现,因此出现了该错误。正确的方法是为要编码的每个结构实现Poison.Encoder
。使用@derive
,这非常简单:
@derive {Poison.Encoder, only: [:the, :field, :names, :you, :want, :in, :json]}
schema "tasks" do
...
end
对于元组,您需要将其手动转换为包含这些元组的结构的编码形式的映射:
defimpl Poison.Encoder, for: Something do
def encode(%{a_tuple: {lat, lon}, something: else_}, options) do
Poison.encode!(%{location: %{lat: lat, lon: lon}, something: else_}, options)
end
end