在Phoenix中处理嵌套表单/外部变更集的正确方法是什么?

时间:2015-12-23 20:50:29

标签: elixir phoenix-framework ecto

我正在凤凰城编写一个简单的CRUD应用程序,管理员可以在创建新组织时为其配置初始员工帐户。

有效地,组织和用户之间的关系是多对多的。

我想出了以下内容:

  1. 用户架构:

    defmodule MyApp.User do
    use MyApp.Web, :model
    
    schema "users" do
      field :name, :string
      field :email, :string
      field :password, :string, virtual: true
      field :password_hash, :string
    end
    
    def changeset(...) # validate email, password confirmation etc.
    
  2. 组织架构:

    defmodule MyApp.Org do
      use MyApp.Web, :model
    
      schema "orgs" do
        field :official_name, :string
        field :common_name, :string
    
        has_many :org_staff_users, MyApp.OrgStaffUser
        has_many :users, through: [:org_staff_users, :user]
     end
    
     def changeset(model, params \\ :empty) do
      model
      |> cast(params, ~w(official_name common_name), [])
     end
    
     def provisioning_changeset(model, params \\ :empty) do
       model
       |> changeset(params)
       |> cast_assoc(:org_staff_users, required: true)
     end
    
  3. 连接表org_staff_users和相应的Ecto Schema user_idorg_id

  4. 具有以下new操作的控制器:

     def new(conn, _params) do
       data = %Org{org_staff_users: [%User{}]}
       changeset = Org.provisioning_changeset(data)
       render(conn, "new.html", changeset: changeset)
     end
    
  5. 带有以下摘录的模板:

     <%= form_for @changeset, @action, fn f -> %>
          <%= if @changeset.action do %>
            <div class="alert alert-danger">
              <p>Oops, something went wrong! Please check the errors below:</p>
              <ul>
                <%= for {attr, message} <- f.errors do %>
                  <li><%= humanize(attr) %> <%= message %></li>
                <% end %>
              </ul>
            </div>
          <% end %>
    
        <%= text_input f, :official_name, class: "form-control" %>
        <%= text_input f, :common_name, class: "form-control" %>
    
        <%= inputs_for f, :org_staff_users, fn i -> %>
            <%= text_input f, :email, class: "form-control" %>
            <%= text_input f, :password, class: "form-control" %>
            <%= text_input f, :password_confirmation, class: "form-control" %>
        <% end %>
    
        <%= submit "Submit", class: "btn btn-primary" %>
    <% end %>
    
  6. 到目前为止一切顺利,表格显示得很好。

    问题是,我真的不明白构建变更集的规范方法应该是什么才能插入create,同时能够 在验证错误时将其再次传递给视图。

    目前还不清楚我是应该使用一个变更集(以及如何?)还是明确地使用变更集 每个实体(UserOrg和联结表)的三个变更集。

    如何确认这些组合表单的更改,并给出每个更改 model / schema有自己定义的特定验证吗?

    我提交表格时收到的参数都在%{"org" => ...}范围内 地图,包括实际上与user相关的地图。我该怎么办? 正确创建表单?

    我已阅读最近更新的http://blog.plataformatec.com.br/2015/08/working-with-ecto-associations-and-embeds/ 但无论如何,我仍然感到困惑。

    FWIW,我在Phoenix 1.0.4,Phoenix Ecto 2.0和Phoenix HTML 2.3.0上。

    任何提示都将不胜感激。

2 个答案:

答案 0 :(得分:13)

现在,除了在交易中执行所有操作之外,您没有任何其他选项。您将在事务中创建一个组织,该组织具有自己的变更集,如果可行,则创建每个员工。像这样:

if organization_changeset.valid? and Enum.all?(staff_changesets, & &1.valid?) do
  Repo.transaction fn ->
    Repo.insert!(organization_changeset)
    Enum.each staff_changesets, &Repo.insert!/1)
  end
end

注意我正在对变更集进行valid?检查,这是非理想的,因为它不考虑约束。如果变更集中存在约束,则需要使用Repo.insert(无需!)。

请记住,在Ecto 2.0上这会更容易。在Ecto master上,我们已经通过changesets支持belongs_to,这意味着你可以通过生成中间和结束关联来明确地做到这一点:

<%= inputs_for f, :org_staff_users, fn org_staff -> %>
  <%= inputs_for org_staff, :user, fn user -> %>
    # Your user form here
  <% end %>    
<% end %>

但是,我们也会支持many_to_many,这将使它完全直截了当。

答案 1 :(得分:0)

使用here

所述的嵌入式架构