1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
|
require 'thread_safe'
module ActiveRecord
module Type
class TypeMap # :nodoc:
def initialize
@mapping = {}
@cache = ThreadSafe::Cache.new do |h, key|
h.fetch_or_store(key, ThreadSafe::Cache.new)
end
end
def lookup(lookup_key, *args)
fetch(lookup_key, *args) { 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
def default_value
@default_value ||= Value.new
end
end
end
end
|