:map
フィールドはエクトによってPostgreSQLのJSONBフィールドとして格納されています。 Ectoは、そのようなフィールドでのマップ固有の操作を行う機能は提供しませんが、fragment
とカスタムSQLを使用して行うことができます。
SQLクエリfoo.bar ? 'baz'
は、"baz"
の値を含むbar
のfoo
の列が含まれているかどうかをチェックします。
iex(1)> Repo.insert! %MyApp.Map{map: %{}}
iex(2)> Repo.insert! %MyApp.Map{map: %{foo: 1}}
iex(3)> Repo.insert! %MyApp.Map{map: %{foo: 2}}
iex(4)> Repo.insert! %MyApp.Map{map: %{bar: 1}}
iex(5)> Repo.all MyApp.Map |> where([m], fragment("? \\? ?", m.map, "foo"))
[debug] QUERY OK source="maps" db=1.8ms decode=5.3ms
SELECT m0."id", m0."map", m0."inserted_at", m0."updated_at" FROM "maps" AS m0 WHERE (m0."map" ? 'foo') []
[%MyApp.Map{__meta__: #Ecto.Schema.Metadata<:loaded, "maps">, id: 2,
inserted_at: #Ecto.DateTime<2016-12-07 10:19:53>, map: %{"foo" => 1},
updated_at: #Ecto.DateTime<2016-12-07 10:19:53>},
%MyApp.Map{__meta__: #Ecto.Schema.Metadata<:loaded, "maps">, id: 3,
inserted_at: #Ecto.DateTime<2016-12-07 10:19:55>, map: %{"foo" => 2},
updated_at: #Ecto.DateTime<2016-12-07 10:19:55>}]
iex(6)> Repo.all MyApp.Map |> where([m], fragment("? \\? ?", m.map, "bar"))
[debug] QUERY OK source="maps" db=2.9ms queue=0.2ms
SELECT m0."id", m0."map", m0."inserted_at", m0."updated_at" FROM "maps" AS m0 WHERE (m0."map" ? 'bar') []
[%MyApp.Map{__meta__: #Ecto.Schema.Metadata<:loaded, "maps">, id: 4,
inserted_at: #Ecto.DateTime<2016-12-07 10:19:59>, map: %{"bar" => 1},
updated_at: #Ecto.DateTime<2016-12-07 10:19:59>}]
iex(7)> Repo.all MyApp.Map |> where([m], fragment("? \\? ?", m.map, "baz"))
[debug] QUERY OK source="maps" db=2.2ms queue=0.1ms
SELECT m0."id", m0."map", m0."inserted_at", m0."updated_at" FROM "maps" AS m0 WHERE (m0."map" ? 'baz') []
[]
:タイプ
:map
のキーmap
を持つブランドの新しいMap
モデルで:だからあなたのコードがに変更する必要があり
:これは、このような
fragment
で表すことができますうわー...あなたはもう一度私を救った – Edmund