PostgreSQL
 sql >> Base de Dados >  >> RDS >> PostgreSQL

Problemas com intervalo de tempo infinito no Rails


Você não pode armazenar Infinity como parte de um intervalo de tempo no Rails. Eu acredito que isso é porque Infinity será inserido como um valor de string e interpretado como um float quando retirado do oid PSQL nativo. Portanto, qualquer intervalo de datas de Date -> Float não será viável. Mas você pode criar seu próprio intervalo com datas pseudo (1 milhão de anos a partir de agora) ou pode simplesmente usar dois campos de data separados e interpretá-los adequadamente no modelo. Data de início, data de término.

No Rails 4.2+, você pode armazenar um valor Float::INFINITY dentro do seu tipo datetime. Exemplo.
User.first.update(begin_date: DateTime.now, end_date: 'infinity')
User.first.end_date # => Infinity

No entanto, end_date não será uma data válida. Você está apenas armazenando a string no banco de dados e você está retirando um float quando você o chama.

Aqui está o código real (Rails 4.2) que lida com isso:
module ActiveRecord
  module ConnectionAdapters
    module PostgreSQL
      module OID # :nodoc:
        class DateTime < Type::DateTime # :nodoc:
          include Infinity

          def type_cast_for_database(value)
            if has_precision? && value.acts_like?(:time) && value.year <= 0
              bce_year = format("%04d", -value.year + 1)
              super.sub(/^-?\d+/, bce_year) + " BC"
            else
              super
            end
          end

          def cast_value(value)
            if value.is_a?(::String)
              case value
              when 'infinity' then ::Float::INFINITY
              when '-infinity' then -::Float::INFINITY
              when / BC$/
                astronomical_year = format("%04d", -value[/^\d+/].to_i + 1)
                super(value.sub(/ BC$/, "").sub(/^\d+/, astronomical_year))
              else
                super
              end
            else
              value
            end
          end
        end
      end
    end
  end
end

Novamente, você não ser capaz de fazer comparações de data e hora com um float. Mas, provavelmente é simples o suficiente para ter um caso especial para esses dois valores -::Float::INFINITY e ::Float::INFINITY