我正在凤凰城编写一个简单的CRUD应用程序,管理员可以在创建新组织时为其配置初始员工帐户。
有效地,组织和用户之间的关系是多对多的。
我想出了以下内容:
用户架构:
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.
组织架构:
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
连接表org_staff_users
和相应的Ecto Schema
user_id
和org_id
具有以下new
操作的控制器:
def new(conn, _params) do
data = %Org{org_staff_users: [%User{}]}
changeset = Org.provisioning_changeset(data)
render(conn, "new.html", changeset: changeset)
end
带有以下摘录的模板:
<%= 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 %>
到目前为止一切顺利,表格显示得很好。
问题是,我真的不明白构建变更集的规范方法应该是什么才能插入create
,同时能够
在验证错误时将其再次传递给视图。
目前还不清楚我是应该使用一个变更集(以及如何?)还是明确地使用变更集
每个实体(User
,Org
和联结表)的三个变更集。
如何确认这些组合表单的更改,并给出每个更改 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上。
任何提示都将不胜感激。
答案 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
所述的嵌入式架构