Sha256: 83ea7234a6a94787ef81a58c63a5ce84e5721e68247dffa535bf660f328cb2f5

Contents?: true

Size: 1.69 KB

Versions: 4

Compression:

Stored size: 1.69 KB

Contents

# frozen_string_literal: true

module MiniSql
  class InlineParamEncoder
    attr_reader :conn

    def initialize(conn)
      @conn = conn
    end

    def encode(sql, *params)
      if Hash === (hash = params[0])
        raise ArgumentError, "Only one hash param is allowed, multiple were sent" if params.length > 1
        encode_hash(sql, hash)
      else
        encode_array(sql, params)
      end
    end

    def encode_hash(sql, hash)
      sql = sql.dup

      # longest key first for gsub to work
      # in an expected way
      hash.sort do |(k, _), (k1, _)|
        k1.to_s.length <=> k.to_s.length
      end.each do |k, v|
        sql.gsub!(":#{k}") do
          # ignore ::int and stuff like that
          # $` is previous to match
          if $` && $`[-1] != ":"
            quote_val(v)
          else
            ":#{k}"
          end
        end
      end
      sql
    end

    def encode_array(sql, array)
      i = -1
      sql.gsub("?") do
        i += 1
        quote_val(array[i])
      end
    end

    def quoted_time(value)
      value.utc.iso8601
    end

    def quote_val(value)
      case value
      when String     then "'#{conn.escape_string(value.to_s)}'"
      when Numeric    then value.to_s
      when BigDecimal then value.to_s("F")
      when Time       then "'#{quoted_time(value)}'"
      when Date       then "'#{value.to_s}'"
      when Symbol     then "'#{conn.escape_string(value.to_s)}'"
      when true       then "true"
      when false      then "false"
      when nil        then "NULL"
      when []         then "NULL"
      when Array      then value.map { |v| quote_val(v) }.join(', ')
      else raise TypeError, "can't quote #{value.class.name}"
      end
    end
  end
end

Version data entries

4 entries across 4 versions & 1 rubygems

Version Path
mini_sql-1.5.0 lib/mini_sql/inline_param_encoder.rb
mini_sql-1.4.0 lib/mini_sql/inline_param_encoder.rb
mini_sql-1.3.0 lib/mini_sql/inline_param_encoder.rb
mini_sql-1.2.0 lib/mini_sql/inline_param_encoder.rb