95 lines
3 KiB
Ruby
95 lines
3 KiB
Ruby
|
# frozen_string_literal: true
|
||
|
|
||
|
# from https://gist.github.com/Envek/7077bfc36b17233f60ad
|
||
|
|
||
|
# PostgreSQL interval data type support from https://github.com/rails/rails/pull/16919
|
||
|
# Works with both Rails 5.2 and 6.0
|
||
|
# Place this file to config/initializers/
|
||
|
|
||
|
require "active_support/duration"
|
||
|
|
||
|
# activerecord/lib/active_record/connection_adapters/postgresql/oid/interval.rb
|
||
|
module ActiveRecord
|
||
|
module ConnectionAdapters
|
||
|
module PostgreSQL
|
||
|
module OID # :nodoc:
|
||
|
class Interval < Type::Value # :nodoc:
|
||
|
def type
|
||
|
:interval
|
||
|
end
|
||
|
|
||
|
def cast_value(value)
|
||
|
case value
|
||
|
when ::ActiveSupport::Duration
|
||
|
value
|
||
|
when ::String
|
||
|
begin
|
||
|
::ActiveSupport::Duration.parse(value)
|
||
|
rescue ::ActiveSupport::Duration::ISO8601Parser::ParsingError
|
||
|
nil
|
||
|
end
|
||
|
else
|
||
|
super
|
||
|
end
|
||
|
end
|
||
|
|
||
|
def serialize(value)
|
||
|
case value
|
||
|
when ::ActiveSupport::Duration
|
||
|
value.iso8601(precision: self.precision)
|
||
|
when ::Numeric
|
||
|
# Sometimes operations on Times returns just float number of seconds so we need to handle that.
|
||
|
# Example: Time.current - (Time.current + 1.hour) # => -3600.000001776 (Float)
|
||
|
value.seconds.iso8601(precision: self.precision)
|
||
|
else
|
||
|
super
|
||
|
end
|
||
|
end
|
||
|
|
||
|
def type_cast_for_schema(value)
|
||
|
serialize(value).inspect
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
# activerecord/lib/active_record/connection_adapters/postgresql_adapter.rb
|
||
|
require 'active_record/connection_adapters/postgresql_adapter'
|
||
|
PostgreSQLAdapterWithInterval = Module.new do
|
||
|
def initialize_type_map(m = type_map)
|
||
|
super
|
||
|
m.register_type "interval" do |*_args, sql_type|
|
||
|
precision = extract_precision(sql_type)
|
||
|
::ActiveRecord::ConnectionAdapters::PostgreSQLAdapter::OID::Interval.new(precision: precision)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
def configure_connection
|
||
|
super
|
||
|
execute('SET intervalstyle = iso_8601', 'SCHEMA')
|
||
|
end
|
||
|
|
||
|
ActiveRecord::Type.register(:interval, ::ActiveRecord::ConnectionAdapters::PostgreSQLAdapter::OID::Interval, adapter: :postgresql)
|
||
|
end
|
||
|
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.prepend(PostgreSQLAdapterWithInterval)
|
||
|
|
||
|
# activerecord/lib/active_record/connection_adapters/postgresql/schema_statements.rb
|
||
|
require 'active_record/connection_adapters/postgresql/schema_statements'
|
||
|
module SchemaStatementsWithInterval
|
||
|
def type_to_sql(type, limit: nil, precision: nil, scale: nil, array: nil, **)
|
||
|
case type.to_s
|
||
|
when 'interval'
|
||
|
case precision
|
||
|
when nil; "interval"
|
||
|
when 0..6; "interval(#{precision})"
|
||
|
else raise(ActiveRecordError, "No interval type has precision of #{precision}. The allowed range of precision is from 0 to 6")
|
||
|
end
|
||
|
else
|
||
|
super
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
ActiveRecord::ConnectionAdapters::PostgreSQL::SchemaStatements.prepend(SchemaStatementsWithInterval)
|