aboutsummaryrefslogblamecommitdiffstats
path: root/activerecord/lib/active_record/type/type_map.rb
blob: 58f25ba075396ee088990f75e9ceb9ae92c0f22e (plain) (tree)
1
2
3
4
5
6
7
8
9

                             
                        
 
                   



                           

                                                    
           


                                   
                                                       

         
                                          


                                                  



                                                   
                    



















                                                   



                                                                  
 




                                                      
           


       
# frozen_string_literal: true

require "concurrent/map"

module ActiveRecord
  module Type
    class TypeMap # :nodoc:
      def initialize
        @mapping = {}
        @cache = Concurrent::Map.new do |h, key|
          h.fetch_or_store(key, Concurrent::Map.new)
        end
      end

      def lookup(lookup_key, *args)
        fetch(lookup_key, *args) { Type.default_value }
      end

      def fetch(lookup_key, *args, &block)
        @cache[lookup_key].fetch_or_store(args) do
          perform_fetch(lookup_key, *args, &block)
        end
      end

      def register_type(key, value = nil, &block)
        raise ::ArgumentError unless value || block
        @cache.clear

        if block
          @mapping[key] = block
        else
          @mapping[key] = proc { value }
        end
      end

      def alias_type(key, target_key)
        register_type(key) do |sql_type, *args|
          metadata = sql_type[/\(.*\)/, 0]
          lookup("#{target_key}#{metadata}", *args)
        end
      end

      def clear
        @mapping.clear
      end

      private
        def perform_fetch(lookup_key, *args)
          matching_pair = @mapping.reverse_each.detect do |key, _|
            key === lookup_key
          end

          if matching_pair
            matching_pair.last.call(lookup_key, *args)
          else
            yield lookup_key, *args
          end
        end
    end
  end
end