time_zone_conversion.rb 2.6 KB
Newer Older
1

J
Joshua Peek 已提交
2
module ActiveRecord
3
  ActiveSupport.on_load(:active_record_model) do
J
Jon Leighton 已提交
4 5 6 7 8 9 10
    mattr_accessor :time_zone_aware_attributes, instance_accessor: false
    self.time_zone_aware_attributes = false

    mattr_accessor :skip_time_zone_conversion_for_attributes, instance_accessor: false
    self.skip_time_zone_conversion_for_attributes = []
  end

J
Joshua Peek 已提交
11 12
  module AttributeMethods
    module TimeZoneConversion
13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
      class Type # :nodoc:
        def initialize(column)
          @column = column
        end

        def type_cast(value)
          value = @column.type_cast(value)
          value.acts_like?(:time) ? value.in_time_zone : value
        end

        def type
          @column.type
        end
      end

J
Joshua Peek 已提交
28 29
      extend ActiveSupport::Concern

30
      included do
J
Jon Leighton 已提交
31
        config_attribute :time_zone_aware_attributes, global: true
32
        config_attribute :skip_time_zone_conversion_for_attributes
J
Joshua Peek 已提交
33 34 35 36
      end

      module ClassMethods
        protected
37 38 39 40 41 42 43 44 45
        # Defined for all +datetime+ and +timestamp+ attributes when +time_zone_aware_attributes+ are enabled.
        # This enhanced write method will automatically convert the time passed to it to the zone stored in Time.zone.
        def define_method_attribute=(attr_name)
          if create_time_zone_conversion_attribute?(attr_name, columns_hash[attr_name])
            method_body, line = <<-EOV, __LINE__ + 1
              def #{attr_name}=(original_time)
                time = original_time
                unless time.acts_like?(:time)
                  time = time.is_a?(String) ? Time.zone.parse(time) : time.to_time rescue time
46
                end
47 48 49 50 51 52
                zoned_time   = time && time.in_time_zone rescue nil
                rounded_time = round_usec(zoned_time)
                rounded_value = round_usec(read_attribute("#{attr_name}"))
                if (rounded_value != rounded_time) || (!rounded_value && original_time)
                  write_attribute("#{attr_name}", original_time)
                  #{attr_name}_will_change!
53
                  @attributes_cache["#{attr_name}"] = zoned_time
54
                end
55 56 57 58 59
              end
            EOV
            generated_attribute_methods.module_eval(method_body, __FILE__, line)
          else
            super
J
Joshua Peek 已提交
60
          end
61
        end
J
Joshua Peek 已提交
62

63
        private
64
        def create_time_zone_conversion_attribute?(name, column)
65 66 67
          time_zone_aware_attributes &&
            !self.skip_time_zone_conversion_for_attributes.include?(name.to_sym) &&
            [:datetime, :timestamp].include?(column.type)
68
        end
J
Joshua Peek 已提交
69
      end
70 71 72 73 74 75

      private
      def round_usec(value)
        return unless value
        value.change(:usec => 0)
      end
J
Joshua Peek 已提交
76 77 78
    end
  end
end