class ActiveRecord::InsertAll::Builder
:nodoc:
def columns_list
def columns_list format_columns(insert_all.keys_including_timestamps) end
def conflict_target
def conflict_target if index = insert_all.unique_by sql = +"(#{format_columns(index.columns)})" sql << " WHERE #{index.where}" if index.where sql elsif update_duplicates? "(#{format_columns(insert_all.primary_keys)})" end end
def extract_types_from_columns_on(table_name, keys:)
def extract_types_from_columns_on(table_name, keys:) columns = @model.schema_cache.columns_hash(table_name) unknown_column = (keys - columns.keys).first raise UnknownAttributeError.new(model.new, unknown_column) if unknown_column keys.index_with { |key| model.type_for_attribute(key) } end
def format_columns(columns)
def format_columns(columns) columns.respond_to?(:map) ? quote_columns(columns).join(",") : columns end
def initialize(insert_all)
def initialize(insert_all) @insert_all, @model, @connection = insert_all, insert_all.model, insert_all.connection end
def into
def into "INTO #{model.quoted_table_name} (#{columns_list})" end
def quote_column(column)
def quote_column(column) connection.quote_column_name(column) end
def quote_columns(columns)
def quote_columns(columns) columns.map { |column| quote_column(column) } end
def raw_update_sql
def raw_update_sql insert_all.update_sql end
def returning
def returning return unless insert_all.returning if insert_all.returning.is_a?(String) insert_all.returning else Array(insert_all.returning).map do |attribute| if model.attribute_alias?(attribute) "#{quote_column(model.attribute_alias(attribute))} AS #{quote_column(attribute)}" else quote_column(attribute) end end.join(",") end end
def touch_model_timestamps_unless(&block)
def touch_model_timestamps_unless(&block) return "" unless update_duplicates? && record_timestamps? model.timestamp_attributes_for_update_in_model.filter_map do |column_name| if touch_timestamp_attribute?(column_name) "#{column_name}=(CASE WHEN (#{updatable_columns.map(&block).join(" AND ")}) THEN #{model.quoted_table_name}.#{column_name} ELSE #{connection.high_precision_current_timestamp} END)," end end.join end
def touch_timestamp_attribute?(column_name)
def touch_timestamp_attribute?(column_name) insert_all.updatable_columns.exclude?(column_name) end
def updatable_columns
def updatable_columns quote_columns(insert_all.updatable_columns) end
def values_list
def values_list types = extract_types_from_columns_on(model.table_name, keys: keys_including_timestamps) values_list = insert_all.map_key_with_value do |key, value| next value if Arel::Nodes::SqlLiteral === value ActiveModel::Type::SerializeCastValue.serialize(type = types[key], type.cast(value)) end connection.visitor.compile(Arel::Nodes::ValuesList.new(values_list)) end