blob: 15f72fc78b856f71c8a528cff1bf1554fac85b9b (
plain)
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
65
66
67
68
69
70
71
72
73
74
|
#
#
# We modify Hash to add a few features we need:
#
# * sorted output of keys to in yaml.
# * reference values either with hsh[key] or hsh.key
# * deep merge
# * select fields
#
# Because the json parsing code we use doesn't support setting a custom class, it is easier for us to just modify Hash.
#
require 'yaml'
class Hash
##
## YAML
##
#
# make the type appear to be a normal Hash in yaml, even for subclasses.
#
def to_yaml_type
"!map"
end
#
# just like Hash#to_yaml, but sorted
#
def to_yaml(opts = {})
YAML::quick_emit(self, opts) do |out|
out.map(taguri, to_yaml_style) do |map|
keys.sort.each do |k|
v = self[k]
map.add(k, v)
end
end
end
end
##
## CONVERTING
##
#
# convert self into a hash, but only include the specified keys
#
def pick(*keys)
keys.map(&:to_s).inject({}) do |hsh, key|
if has_key?(key)
hsh[key] = self[key]
end
hsh
end
end
#
# recursive merging (aka deep merge)
# taken from ActiveSupport::CoreExtensions::Hash::DeepMerge
#
def deep_merge(other_hash)
self.merge(other_hash) do |key, oldval, newval|
oldval = oldval.to_hash if oldval.respond_to?(:to_hash)
newval = newval.to_hash if newval.respond_to?(:to_hash)
oldval.class.to_s == 'Hash' && newval.class.to_s == 'Hash' ? oldval.deep_merge(newval) : newval
end
end
def deep_merge!(other_hash)
replace(deep_merge(other_hash))
end
end
|