Sha256: 88bc3699ecfff088cef8a484c7f3d595ca6f2a0351b258b01bcc9842cb08e667

Contents?: true

Size: 1.93 KB

Versions: 1

Compression:

Stored size: 1.93 KB

Contents

# rubocop:disable Style/ClassVars
# rubocop:disable Metrics/AbcSize

module Simple
  module SQL
    def insert(table, records)
      if records.is_a?(Hash)
        insert_many(table, [records]).first
      else
        insert_many table, records
      end
    end

    private

    def insert_many(table, records)
      return [] if records.empty?

      inserter = Inserter.create(table_name: table.to_s, columns: records.first.keys)
      inserter.insert(records: records)
    end

    class Inserter
      SQL = ::Simple::SQL

      @@inserters = {}

      def self.create(table_name:, columns:)
        @@inserters[[table_name, columns]] ||= new(table_name: table_name, columns: columns)
      end

      #
      # - table_name - the name of the table
      # - columns - name of columns, as Array[String] or Array[Symbol]
      #
      def initialize(table_name:, columns:)
        @columns = columns

        cols = []
        vals = []

        cols += columns
        vals += columns.each_with_index.map { |_, idx| "$#{idx + 1}" }

        timestamp_columns = timestamp_columns_in_table(table_name) - columns.map(&:to_s)

        cols += timestamp_columns
        vals += timestamp_columns.map { "now()" }

        @sql = "INSERT INTO #{table_name} (#{cols.join(',')}) VALUES(#{vals.join(',')}) RETURNING id"
      end

      # timestamp_columns are columns that will be set to the current time when
      # inserting a record. This includes:
      #
      # - inserted_at (for Ecto)
      # - created_at (for ActiveRecord)
      # - updated_at (for Ecto and ActiveRecord)
      def timestamp_columns_in_table(table_name)
        columns_for_table = SQL::Reflection.columns(table_name).keys
        columns_for_table & %w(inserted_at created_at updated_at)
      end

      def insert(records:)
        SQL.transaction do
          records.map do |record|
            SQL.ask @sql, *record.values_at(*@columns)
          end
        end
      end
    end
  end
end

Version data entries

1 entries across 1 versions & 1 rubygems

Version Path
simple-sql-0.2.7 lib/simple/sql/insert.rb