PostgreSQL
 sql >> Baza danych >  >> RDS >> PostgreSQL

Problemy z nieskończonym zakresem czasu w Railsach

Nie możesz przechowywać Infinity jako części zakresu czasu w Railsach. Wierzę, że dzieje się tak dlatego, że Infinity zostanie wstawiona jako wartość ciągu i zinterpretowana jako pływak po wyciągnięciu z natywnego oidu PSQL. Tak więc żaden zakres dat z Date -> Float nie będzie opłacalny. Ale możesz stworzyć własny zakres z pseudodatami (1 milion lat od teraz) lub możesz po prostu użyć dwóch oddzielnych pól dat i odpowiednio je zinterpretować w modelu. Data rozpoczęcia, data zakończenia.

W Rails 4.2+ możesz przechowywać wartość Float::INFINITY wewnątrz twojego typu datetime. Przykład.

User.first.update(begin_date: DateTime.now, end_date: 'infinity')
User.first.end_date # => Infinity

Jednak end_date nie będzie prawidłową datą. Po prostu przechowujesz ciąg w bazie danych i wyciągasz float kiedy to zadzwonisz.

Oto rzeczywisty kod (Rails 4.2), który to obsługuje:

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

Ponownie, nie być w stanie zrobić porównania daty i czasu za pomocą zmiennoprzecinkowego. Ale prawdopodobnie wystarczy mieć specjalny przypadek dla tych dwóch wartości -::Float::INFINITY i ::Float::INFINITY




  1. Database
  2.   
  3. Mysql
  4.   
  5. Oracle
  6.   
  7. Sqlserver
  8.   
  9. PostgreSQL
  10.   
  11. Access
  12.   
  13. SQLite
  14.   
  15. MariaDB
  1. Postgresql escape w pojedynczym cudzysłowie w klauzuli where

  2. Asynchroniczny DBAppender z logbackiem

  3. Usługa bazy danych PostgreSQL

  4. Jak przetwarzać transakcje rozproszone w ramach postgresql?

  5. Nie można utworzyć tabeli bazy danych o nazwie „użytkownik” w PostgreSQL