2017-03-16 77 views
1

我正在使用phoenix作为JSON API。我的一个模型如下所示,除非unique_constraint检查失败,否则运行良好。当发生这种情况,我得到以下错误:毒药无法编码错误信息

(Poison.EncodeError) unable to encode value: {"Email address is already registered", []} 

型号

defmodule MyApp.Registration do 
    use MyApp.Web, :model 

    @derive {Poison.Encoder, only: [:name, :email, :category]} 
    schema "registrations" do 
    field :name, :string 
    field :category, :string 
    field :email, :string 

    timestamps 
    end 

    def changeset(model, params \\ :empty) do 
    model 
    |> cast(params, ~w(name email category), []) 
    |> validate_length(:name, min: 1, max: 240) 
    |> unique_constraint(:email, message: "Email address is already registered") 
    end 

end 

控制器

def create(conn, registration_params) do 
    changeset = Registration.changeset(%Registration{}, registration_params) 
    case Repo.insert(changeset) do 
    {:ok, _registration} -> 
     # Success 
    {:error, error} -> 
     conn 
     |> put_status(:unprocessable_entity) 
     |> render(MyApp.ErrorView, "generic.json", error: error) 
    end 
end 

查看

def render("generic.json", error) do 
    error 
end 

我想我可能需要以某种方式将错误消息添加到Poison.Encoder, only: []列表,但我不确定如何做到这一点。

编辑

我要澄清的是,如果我不指定自定义错误我仍然与一般错误消息得到同样的错误。

+0

的错误是Ecto.Changeset,而不是模型的一部分,这样看起来很奇怪。你能否提供控制器代码和渲染json的可能的视图代码? – Svilen

+0

@Svilen添加了控制器和视图代码。我尝试了几种将变更集错误传递给json视图的方法,但目前为止没有喜悦。 – Razzildinho

回答

3

我认为这是自动完成的,如果您使用mix phoenix.gen.json但变更集错误需要转换为json。

查看

def translate_errors(changeset) do 
    Ecto.Changeset.traverse_errors(changeset, &translate_error/1) 
end 

def render("error.json", %{changeset: changeset}) do 
    # When encoded, the changeset returns its errors 
    # as a JSON object. So we just pass it forward. 
    %{errors: translate_errors(changeset)} 
end 

控制器

def create(conn, registration_params) do 
    changeset = Registration.changeset(%Registration{}, registration_params) 
    case Repo.insert(changeset) do 
    {:ok, _registration} -> 
     # Success 
    {:error, changeset} -> 
     conn 
     |> put_status(:unprocessable_entity) 
     |> render(MyApp.ErrorView, "error.json", changeset: changeset) 
    end 
end 

编辑

添加translate_error\1函数定义。这包含在一个名为MyApp.ErrorHelpers的模块中,该模块根据您的凤凰版本导入到web/my_app.exlib/my_app_web.ex中的view函数定义中。

lib/my_app_web/views/error_helpers.ex

defmodule MyAppWeb.ErrorHelpers do 
    def translate_error({msg, opts}) do 
    if count = opts[:count] do 
     Gettext.dngettext(MyAppWeb.Gettext, "errors", msg, msg, count, opts) 
    else 
     Gettext.dgettext(MyAppWeb.Gettext, "errors", msg, opts) 
    end 
    end 
end 
+0

你知道哪里定义了'translate_error/1'函数吗? –

+0

@ Jean-PierreBécotte我已经用函数定义更新了答案。 – Razzildinho

+0

干得好,@Razzildinho! –