Вы на правильном пути.Предположим, что ваше скользкое отображение определено как:
case class RecordRow(id: Int)
case class TagRow(id: Int)
case class RecordTagRow(recordId: Int, tagId: Int)
case class MarketRow(id: Int)
case class RecordMarketRow(recordId: Int, marketId: Int)
class RecordTable(_tableTag: Tag)
extends Table[RecordRow](_tableTag, "record") {
val id = column[Int]("id", O.PrimaryKey, O.AutoInc)
override def * = id <> ((id: Int) => RecordRow(id), RecordRow.unapply)
}
class TagTable(_tableTag: Tag) extends Table[TagRow](_tableTag, "tag") {
val id = column[Int]("id", O.PrimaryKey, O.AutoInc)
override def * = id <> ((id: Int) => TagRow(id), TagRow.unapply)
}
class RecordTagTable(_tableTag: Tag)
extends Table[RecordTagRow](_tableTag, "record_tag") {
val recordId = column[Int]("record_id")
val tagId = column[Int]("tag_id")
val pk = primaryKey("record_tag_pkey", (recordId, tagId))
foreignKey("record_tag_record_fk", recordId, RecordQuery)(r => r.id)
foreignKey("record_tag_tag_fk", tagId, TagQuery)(r => r.id)
override def * =
(recordId, tagId) <> (RecordTagRow.tupled, RecordTagRow.unapply)
}
class MarketTable(_tableTag: Tag)
extends Table[MarketRow](_tableTag, "market") {
val id = column[Int]("id", O.PrimaryKey, O.AutoInc)
override def * = id <> ((id: Int) => MarketRow(id), MarketRow.unapply)
}
class RecordMarketTable(_tableTag: Tag)
extends Table[RecordMarketRow](_tableTag, "record_market") {
val recordId = column[Int]("record_id")
val marketId = column[Int]("market_id")
val pk = primaryKey("record_tag_pkey", (recordId, marketId))
foreignKey("record_market_record_fk", recordId, RecordQuery)(r => r.id)
foreignKey("record_market_market_fk", marketId, MarketQuery)(r => r.id)
override def * =
(recordId, marketId) <> (RecordMarketRow.tupled, RecordMarketRow.unapply)
}
val RecordQuery = new TableQuery(tag => new RecordTable(tag))
val TagQuery = new TableQuery(tag => new TagTable(tag))
val RecordTagQuery = new TableQuery(tag => new RecordTagTable(tag))
val MarketQuery = new TableQuery(tag => new MarketTable(tag))
val RecordMarketQuery = new TableQuery(tag => new RecordMarketTable(tag))
Чтобы объединить таблицы с отношениями «многие ко многим», вы должны объединить левые объединения с внутренними объединениями следующим образом:
val recordsQuery = RecordQuery
.joinLeft(RecordTagQuery.join(TagQuery).on(_.tagId === _.id)).on(_.id === _._1.recordId)
.joinLeft(RecordMarketQuery.join(MarketQuery).on(_.marketId === _.id)).on(_._1.id === _._1.recordId)
Это переведенощелкнув следующий SQL с профилем PostgreSQL:
select
x2."id",
x3."id",
x4."record_id",
x4."tag_id",
x3."id",
x5."id",
x6."record_id",
x6."market_id",
x5."id"
from
"record" x2
left outer join
"record_tag" x4
inner join
"tag" x3
on x4."tag_id" = x3."id"
on x2."id" = x4."record_id"
left outer join
"record_market" x6
inner join
"market" x5
on x6."market_id" = x5."id"
on x2."id" = x6."record_id"
Последний шаг - правильно сопоставить результат этого запроса с классами scala.Я сделал это следующим образом:
db.run {
recordsQuery.result
.map(result => {
result
.groupBy(_._1._1) // RecordRow as a key
.mapValues(values =>values.map(value => (value._1._2.map(_._2), value._2.map(_._2)))) // Seq[(Option[TagRow], Option[MarketRow])] as value
.map(mapEntry =>(mapEntry._1, mapEntry._2.flatMap(_._1), mapEntry._2.flatMap(_._2))) // map to Seq[(RecordRow, Seq[TagRow], Seq[MarketRow])]
.toSeq
})
}
Это вернет Future[Seq[(RecordRow, Seq[TagRow], Seq[MarketRow])]]