require 'thinking_sphinx/source/internal_properties' require 'thinking_sphinx/source/sql' module ThinkingSphinx class Source include ThinkingSphinx::Source::InternalProperties include ThinkingSphinx::Source::SQL attr_accessor :model, :fields, :attributes, :joins, :conditions, :groupings, :options attr_reader :base, :index, :database_configuration def initialize(index, options = {}) @index = index @model = index.model @fields = [] @attributes = [] @joins = [] @conditions = [] @groupings = [] @options = options @associations = {} @database_configuration = @model.connection. instance_variable_get(:@config).clone @base = join_dependency_class.new( @model, [], initial_joins ) unless @model.descends_from_active_record? stored_class = @model.store_full_sti_class ? @model.name : @model.name.demodulize @conditions << "#{@model.quoted_table_name}.#{quote_column(@model.inheritance_column)} = '#{stored_class}'" end add_internal_attributes_and_facets end def name index.name end def to_riddle_for_core(offset, position) source = Riddle::Configuration::SQLSource.new( "#{index.core_name}_#{position}", adapter.sphinx_identifier ) set_source_database_settings source set_source_fields source set_source_attributes source, offset set_source_settings source set_source_sql source, offset source end def to_riddle_for_delta(offset, position) source = Riddle::Configuration::SQLSource.new( "#{index.delta_name}_#{position}", adapter.sphinx_identifier ) source.parent = "#{index.core_name}_#{position}" set_source_database_settings source set_source_fields source set_source_attributes source, offset, true set_source_settings source set_source_sql source, offset, true source end def delta? !@index.delta_object.nil? end # Gets the association stack for a specific key. # def association(key) @associations[key] ||= Association.children(@model, key) end private def adapter @adapter ||= @model.sphinx_database_adapter end def available_attributes attributes.select { |attrib| attrib.available? } end def set_source_database_settings(source) config = @database_configuration source.sql_host = config[:host] || "localhost" source.sql_user = config[:username] || config[:user] || 'root' source.sql_pass = (config[:password].to_s || "").gsub('#', '\#') source.sql_db = config[:database] source.sql_port = config[:port] source.sql_sock = config[:socket] end def set_source_fields(source) fields.each do |field| source.sql_file_field << field.unique_name if field.file? source.sql_field_string << field.unique_name if field.with_attribute? source.sql_field_str2wordcount << field.unique_name if field.with_wordcount? end end def set_source_attributes(source, offset, delta = false) available_attributes.each do |attrib| source.send(attrib.type_to_config) << attrib.config_value(offset, delta) end end def set_source_sql(source, offset, delta = false) source.sql_query = to_sql(:offset => offset, :delta => delta).gsub(/\n/, ' ') source.sql_query_range = to_sql_query_range(:delta => delta) source.sql_query_info = to_sql_query_info(offset) source.sql_query_pre += send(!delta ? :sql_query_pre_for_core : :sql_query_pre_for_delta) if @index.local_options[:group_concat_max_len] source.sql_query_pre << "SET SESSION group_concat_max_len = #{@index.local_options[:group_concat_max_len]}" end source.sql_query_pre += [adapter.utf8_query_pre].compact if utf8? source.sql_query_pre << adapter.utc_query_pre end def set_source_settings(source) config = ThinkingSphinx::Configuration.instance config.source_options.each do |key, value| source.send("#{key}=".to_sym, value) end source_options = ThinkingSphinx::Configuration::SourceOptions @options.each do |key, value| if source_options.include?(key.to_s) && !value.nil? source.send("#{key}=".to_sym, value) end end end # Returns all associations used amongst all the fields and attributes. # This includes all associations between the model and what the actual # columns are from. # def all_associations @all_associations ||= ( # field associations @fields.collect { |field| field.associations.values }.flatten + # attribute associations @attributes.collect { |attrib| attrib.associations.values if attrib.include_as_association? }.compact.flatten + # explicit joins @joins.collect { |join| join.associations }.flatten ).uniq.collect { |assoc| # get ancestors as well as column-level associations assoc.ancestors }.flatten.uniq end def utf8? @index.options[:charset_type] =~ /utf-8|zh_cn.utf-8/ end def join_dependency_class if rails_3_1? ::ActiveRecord::Associations::JoinDependency else ::ActiveRecord::Associations::ClassMethods::JoinDependency end end def initial_joins if rails_3_1? [] else nil end end def rails_3_1? ::ActiveRecord::Associations.constants.include?(:JoinDependency) || ::ActiveRecord::Associations.constants.include?('JoinDependency') end end end