После долгих потрясений коллега придумал следующий оператор SQL:
SELECT id, max(updated_at) FROM (
SELECT (activities.id) as id, (notes.updated_at) as updated_at FROM `activities`
INNER JOIN `notes` ON `notes`.`activity_id` = `activities`.`id`
WHERE (context_id = 8) AND (notes.updated_at > '2011-12-01 18:56:16')
UNION ALL (
SELECT (activities.id) as id, (transitions.updated_at) as updated_at
FROM `activities`
INNER JOIN `transitions` ON (
`transitions`.`transitionable_id` = `activities`.`id`
AND `transitions`.`transitionable_type` = 'Activity'
)
WHERE (context_id = 8) AND (transitions.updated_at > '2011-12-01 18:56:16')
)
) transitions
GROUP BY id ORDER BY updated_at DESC
Теперь давайте не будем вдаваться в детали или правильность этого утверждения - он спросил меня, могу ли я создать область ARдля этого.
Я пришел к следующему решению:
scope :most_recent_since, lambda { |way_back|
activity_t = Activity.arel_table
transition_t = Transition.arel_table
note_t = Note.arel_table
note_activities = activity_t.join(note_t).on(
note_t[:activity_id].eq(activity_t[:id])
).where(
activity_t[:context_id].eq(8).and(note_t[:updated_at].gt(way_back))
)
transition_activities = activity_t.join(transition_t).on(
transition_t[:transitionable_id].eq(activity_t[:id]).and(
transition_t[:transitionable_type].eq('Activity')
)
).where(
activity_t[:context_id].eq(8).and(
transition_t[:updated_at].gt(way_back)
)
)
union = note_activities.project(
activity_t[:id], note_t[:updated_at]
).union(
transition_activities.project(activity_t[:id], transition_t[:updated_at])
)
sql = transition_t.project(
transition_t[:id]
).group(
transition_t[:id]
).order(
'updated_at'
).to_sql
# now fiddle in the union sql - cant seem to find how to do it with Arel
sql.gsub!("FROM `trans", "FROM #{union.to_sql} `trans")
where( :id => Transition.find_by_sql(sql) )
}
Нет, мой вопрос;как, черт возьми, я могу заставить этого gsub уйти и выбрать ARel из нескольких источников (включая SQL / union)