module Mobility module Backend class ActiveRecord::Table::QueryMethods < ActiveRecord::QueryMethods def initialize(attributes, association_name: nil, model_class: nil, subclass_name: nil, **options) super @association_name = association_name @translation_class = translation_class = model_class.const_get(subclass_name) define_join_method(association_name, translation_class, **options) define_query_methods(association_name, translation_class, **options) attributes.each do |attribute| define_method :"find_by_#{attribute}" do |value| find_by(attribute.to_sym => value) end end end def extended(relation) super association_name = @association_name attributes_extractor = @attributes_extractor translation_class = @translation_class mod = Module.new do define_method :not do |opts, *rest| if i18n_keys = attributes_extractor.call(opts) opts = opts.with_indifferent_access i18n_keys.each { |attr| opts["#{translation_class.table_name}.#{attr}"] = opts.delete(attr) } super(opts, *rest).send("join_#{association_name}") else super(opts, *rest) end end end relation.mobility_where_chain.include(mod) end private def define_join_method(association_name, translation_class, foreign_key: nil, table_name: nil, **_) define_method :"join_#{association_name}" do |**options| return self if (@__mobility_table_joined || []).include?(table_name) (@__mobility_table_joined ||= []) << table_name t = translation_class.arel_table m = arel_table join_type = options[:outer_join] ? Arel::Nodes::OuterJoin : Arel::Nodes::InnerJoin joins(m.join(t, join_type). on(t[foreign_key].eq(m[:id]). and(t[:locale].eq(Mobility.locale))).join_sources) end end def define_query_methods(association_name, translation_class, **_) attributes_extractor = @attributes_extractor # Note that Mobility will try to use inner/outer joins appropriate to the query, # so for example: # # Article.where(title: nil, content: nil) #=> OUTER JOIN (all nils) # Article.where(title: "foo", content: nil) #=> INNER JOIN (one non-nil) # # In the first case, if we are in (say) the "en" locale, then we should match articles # that have *no* article_translations with English locales (since no translation is # equivalent to a nil value). If we used an inner join in the first case, an article # with no English translations would be filtered out, so we use an outer join. # # However, if you call `where` multiple times, you may end up with an outer join # when a (faster) inner join would have worked fine: # # Article.where(title: nil).where(content: "foo") #=> OUTER JOIN # # In this case, we are searching for a match on the article_translations table # which has a NULL title and a content equal to "foo". Since we need a positive # match for content, there must be an English translation on the article, thus # we can use an inner join. However, Mobility will use an outer join since we don't # want to modify the existing relation which has already been joined. # # To avoid this problem, simply make sure to either order your queries to place nil # values last, or include all queried attributes in a single `where`: # # Article.where(title: nil, content: "foo") #=> INNER JOIN # define_method :where! do |opts, *rest| if i18n_keys = attributes_extractor.call(opts) opts = opts.with_indifferent_access options = { outer_join: i18n_keys.all? { |attr| opts[attr].nil? } } i18n_keys.each { |attr| opts["#{translation_class.table_name}.#{attr}"] = opts.delete(attr) } super(opts, *rest).send("join_#{association_name}", options) else super(opts, *rest) end end end end end end