wfgilman

wfgilman

Ecto Association vs Foreign Key Constraints

I’m having trouble understanding the difference between cast_assoc, assoc_constraint and foreign_key_constraint in a changeset. More specifically, I don’t seem to understand when you should build your associations explicitly (by putting foreign keys) versus letting Ecto build them with associations.

I have a schema like this:

schema "institution_codes" do # InstitutionCode
  field :code, :string
  belongs_to :source, Source
  belongs_to :institution, Institution
end

def changeset(struct, params \\ %{}) do
  struct
  |> cast(params, [:code])
  |> cast_assoc, assoc_constraint or foreign_key_constraint?
end

…and in my migration:

create table(:institution_codes) do
  add :code, :string
  add :source_id, references(:sources)
  add :institution_id, references(:institutions)
end
create unique_index(:institution_codes, [:source_id, :institution_id], 
                     name: :institution_codes_index)

If I want to insert a record into institution_codes, should I be looking up the foreign keys of the Source and Institution schema and explicitly putting them in source_id and institution_id in my InstitutionCode schema params then inserting the record, or putting associations?

The unique constraint on InstitutionCode doesn’t seem to work when putting associations. When I try to insert the record I get errors in my changeset that there are uniqueness violations in the Source and Institution schemas because Ecto is trying to insert the associations. I want it to put the keys of the associations as foreign keys in the InstitutionCode record, not insert the associations themselves.

I feel like I have a fundamental misunderstanding of how these should be used.

Marked As Solved

wfgilman

wfgilman

After some digging, I found this stackoverflow post which really helped explain the difference.

cast_assoc/3 is only used in the changesets of schemas that have has_one or has_many. It invokes the cast/3 validation in the changeset of the associated schema.

assoc_constraint/3 is used in the changesets of schemas that have belongs_to. It checks to make sure the parent schema is in the database so an orphaned record isn’t inserted.

foreign_key_constraint/3 is used similarly to assoc_constraint/3, except that it uses the database to validate the foreign key constraint, whereas assoc_constraint/3 uses the Ecto schema definition.

The reason I was having issues inserting records with associations was because of the way I defined the association. Ecto was trying to insert the parent record along with the child record. As noted in that linked article, the correct way to insert a child record with a parent association is to either build the the association using build_assoc/3 from the parent record, or, in more complex casts, put the parent foreign key in the parameters of the child record and pass that to the changeset.

See one, do one, teach one :wink:

32
Post #2

Where Next?

Popular in Questions Top

sergio
In Ruby, I can go: User.find_by(email: "foobar@email.com").update(email: "hello@email.com") How can I do something similar in Elixir? ...
New
marius95
Hello everyone, I try to use an Javascript Event Handler in my root.html.leex file. Therefore I created a function in the app.js file: ...
New
Fl4m3Ph03n1x
About me? ( if you have nothing better to do than reading about some random guy in the internet :stuck_out_tongue: ) Hello all, this is ...
New
ovidiubadita
Hey all, I discovered Elixir and I love it. I always wanted to learn a functional programming and I intended to go for Haskell, but afte...
New
jononomo
I am trying to figure out how Mix knows whether the environment is test, dev, or prod -- where is this set? Thanks.
New
minhajuddin
I have seen a lot of code which picks the first element from a list using Enum.at(0) instead of List.first. Is there a reason why people ...
New
vonH
When I run the Plug and I recompile I wind up having to use Ctrl C to quit iex and start again. Witht the help of rlwrap I can use the cu...
New
Lily
In templates/appointment/index.html.eex: <%= for appointment <- @appointments do %> <tr> <td><%= appoi...
New
script
If I have a string “1000 cfu/ml” . I want to remove the characters and / and space . So the string is like this "1000" What is the ...
New
nobody
Hi! In PHP: $SERVER['SERVERADDR'] - in Elixir? Searched the docs for ip address and the web, no good results. Thanks!
New

Other popular topics Top

mcarvalho
What is the difference between System.get_env and Application.get_env? For example, what are best practices to use one versus another.
New
Patoshizzle
After calling mix ecto.create I get this error: 17:00:32.162 [error] GenServer #PID<0.412.0> terminating ** (Postgrex.Error) FATAL...
New
pmjoe
I have a relationship of love and hate with Elixir. Lots of things are just absolutely right, but there are some things that are kind of ...
New
Emily
I have VueJS GUIs with the project generated using Webpack. I have Elixir modules that will need to be used by the VueJS GUIs. I fore...
New
fireproofsocks
Forgive me if this is obvious, but how does one delete a database record WITHOUT selecting it first? https://hexdocs.pm/ecto/Ecto.Repo.h...
New
aalberti333
As the title describes, I’m trying to run Enum.map() over a list of key/value pairs, where the value is a map. My data looks like this: ...
New
chrismccord
This release brings a number of exciting features, including integration with the new Phoenix LiveDashboard and Phoenix LiveView. There h...
New
Brian
What is the proper way to load a module from a file in to IEX? In the python world, doing something like this pretty standard: from ....
New
openscript
Hello! Sorry for this astonishing simple question, but I’m really stuck. I try to set up the intellij-elixir plugin, but I don’t know ho...
New
hariharasudhan94
Lets say i have map like this fetching from my database %{"_id" => #BSON.ObjectId<58eb1a7a9ad169198c3dXXXX>, "email" => "XX...
New

We're in Beta

About us Mission Statement