Преобразование SQL в Rails 3 ActiveRecord - PullRequest
0 голосов
/ 06 апреля 2011

Я ищу наиболее подходящий для Rails способ получения отдельного списка категорий для доступных песен в данном альбоме .

Вот это в SQL для album_id = 1

-- Using subselects
select * from categories where id in (
  select distinct category_id
  from categorizations
  where song_id in (select song_id from album_songs
                    where album_id = 1 and available = 't')
)
order by name asc;

-- Using joins
select distinct c.* from categories c
  inner join categorizations cz on c.id = cz.category_id
  left join album_songs a on cz.song_id = a.song_id
where a.album_id = 1 and a.available = 't'
order by c.name asc;

Моя рабочая (хотя и наивная!) Пытается перенести это на ActiveRecord

## attempting to do it like subselects (although they're not really
## subselects, it executes them individually -- from what i've read
## ActiveRecord won't do subselects?)
Category.where('id IN (?)',
  Categorization.select('DISTINCT category_id').where('song_id IN (?)',
    Album.find(1).songs.available.map(&:song_id)
  ).map(&:category_id)
).order('name ASC')

## joins - although at this point it's pretty much all sql
## as i couldn't find a way to do the left join in pure AR
## i'm also duplicating my AlbumSongs.available scope -- is
## that scope reusable here? (outside the AlbumSongs model?)
Category.select('DISTINCT categories.*')
        .joins(:categorizations,
               'LEFT OUTER JOIN album_songs ON categorizations.song_id = album_songs.song_id')
        .where('album_songs.album_id = ? and available', 1)

Я собираюсь перейти к последнему, но похоже, что я мог бы просто написать это на SQL?

Есть ли способ улучшить это, чтобы быть более подходящим для Rails?

1 Ответ

0 голосов
/ 06 апреля 2011

Ну, это, безусловно, поможет, если вы опубликуете свою модель. Но при условии, что:
* песня has_many: категории,: через => классификации
* в альбоме нет большого количества песен

Почему бы просто не сделать:

Album.includes({:songs => {:categorizations => :categories}}).find(1).songs.collect {|s| s.category}.flatten.uniq
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...