line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Project::Easy::Config; |
2
|
|
|
|
|
|
|
|
3
|
3
|
|
|
3
|
|
1843
|
use Class::Easy; |
|
3
|
|
|
|
|
18175
|
|
|
3
|
|
|
|
|
29
|
|
4
|
|
|
|
|
|
|
|
5
|
|
|
|
|
|
|
our %nonexistent_keys_in_config = (); |
6
|
|
|
|
|
|
|
our @curr_patch_config_path = (); |
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
#sub patch ($$); |
9
|
|
|
|
|
|
|
|
10
|
|
|
|
|
|
|
sub parse { |
11
|
0
|
|
|
0
|
0
|
0
|
my $class = shift; |
12
|
0
|
|
|
|
|
0
|
my $core = shift; |
13
|
0
|
|
|
|
|
0
|
my $instance = shift; |
14
|
|
|
|
|
|
|
|
15
|
0
|
|
|
|
|
0
|
my $path = $core->conf_path; |
16
|
0
|
|
|
|
|
0
|
my $fixup = $core->fixup_path_instance ($instance); |
17
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
# TODO: replace to real splitpath and join '/' for windows users |
19
|
0
|
|
|
|
|
0
|
my $root_path = $core->root->path; |
20
|
0
|
|
|
|
|
0
|
$root_path =~ s/\\/\//g; |
21
|
|
|
|
|
|
|
|
22
|
|
|
|
|
|
|
# here we want to expand some generic params |
23
|
0
|
|
|
|
|
0
|
my $expansion = { |
24
|
|
|
|
|
|
|
root => $root_path, |
25
|
|
|
|
|
|
|
id => $core->id, |
26
|
|
|
|
|
|
|
instance => $core->instance, |
27
|
|
|
|
|
|
|
}; |
28
|
|
|
|
|
|
|
|
29
|
0
|
|
|
|
|
0
|
my $conf = $path->deserialize ($expansion); |
30
|
0
|
|
|
|
|
0
|
my $alt = $fixup->deserialize ($expansion); |
31
|
|
|
|
|
|
|
|
32
|
0
|
|
|
|
|
0
|
patch ($conf, $alt); |
33
|
|
|
|
|
|
|
|
34
|
0
|
|
|
|
|
0
|
return $conf; |
35
|
|
|
|
|
|
|
} |
36
|
|
|
|
|
|
|
|
37
|
|
|
|
|
|
|
my $ext_syn = { |
38
|
|
|
|
|
|
|
'pl' => 'perl', |
39
|
|
|
|
|
|
|
'js' => 'json', |
40
|
|
|
|
|
|
|
}; |
41
|
|
|
|
|
|
|
|
42
|
|
|
|
|
|
|
sub serializer { |
43
|
10
|
|
|
10
|
0
|
1347
|
shift; |
44
|
10
|
|
|
|
|
12
|
my $type = shift; |
45
|
|
|
|
|
|
|
|
46
|
10
|
100
|
|
|
|
33
|
$type = $ext_syn->{$type} |
47
|
|
|
|
|
|
|
if exists $ext_syn->{$type}; |
48
|
|
|
|
|
|
|
|
49
|
10
|
|
|
|
|
20
|
my $pack = "Project::Easy::Config::Format::$type"; |
50
|
|
|
|
|
|
|
|
51
|
10
|
50
|
|
|
|
27
|
die ('no such serializer: ', $type) |
52
|
|
|
|
|
|
|
unless try_to_use ($pack); |
53
|
|
|
|
|
|
|
|
54
|
10
|
|
|
|
|
1158
|
return $pack->new; |
55
|
|
|
|
|
|
|
} |
56
|
|
|
|
|
|
|
|
57
|
|
|
|
|
|
|
sub string_from_template { |
58
|
|
|
|
|
|
|
|
59
|
2
|
|
|
2
|
0
|
5
|
my $template = shift; |
60
|
2
|
|
|
|
|
4
|
my $expansion = shift; |
61
|
|
|
|
|
|
|
|
62
|
2
|
50
|
|
|
|
7
|
return unless $template; |
63
|
|
|
|
|
|
|
|
64
|
2
|
|
|
|
|
9
|
foreach (keys %$expansion) { |
65
|
2
|
50
|
|
|
|
6
|
next unless defined $expansion->{$_}; |
66
|
|
|
|
|
|
|
|
67
|
2
|
|
|
|
|
25
|
$template =~ s/\{\$$_\}/$expansion->{$_}/sg; |
68
|
|
|
|
|
|
|
} |
69
|
|
|
|
|
|
|
|
70
|
2
|
|
|
|
|
8
|
return $template; |
71
|
|
|
|
|
|
|
} |
72
|
|
|
|
|
|
|
|
73
|
|
|
|
|
|
|
sub patch { |
74
|
14
|
|
|
14
|
0
|
991
|
my $struct = shift; |
75
|
14
|
|
|
|
|
18
|
my $patch = shift; |
76
|
14
|
|
100
|
|
|
45
|
my $algorithm = shift || 'ordinary_patch'; |
77
|
|
|
|
|
|
|
|
78
|
|
|
|
|
|
|
# $algorithm = ordinary_patch || undef_keys_in_patch || store_nonexistent_keys_in_struct |
79
|
|
|
|
|
|
|
|
80
|
14
|
100
|
66
|
|
|
139
|
return if ref $struct ne 'HASH' and ref $patch ne 'HASH'; |
81
|
|
|
|
|
|
|
|
82
|
8
|
100
|
|
|
|
25
|
unless ( scalar keys %$struct ) { |
83
|
1
|
|
|
|
|
7
|
%$struct = %$patch; |
84
|
1
|
|
|
|
|
4
|
return; |
85
|
|
|
|
|
|
|
} |
86
|
|
|
|
|
|
|
|
87
|
7
|
|
|
|
|
27
|
my $algo_id = { |
88
|
|
|
|
|
|
|
ordinary_patch => 1, |
89
|
|
|
|
|
|
|
undef_keys_in_patch => 2, |
90
|
|
|
|
|
|
|
store_nonexistent_keys_in_struct => 3, |
91
|
|
|
|
|
|
|
}; |
92
|
|
|
|
|
|
|
|
93
|
7
|
|
|
|
|
17
|
foreach my $k (keys %$patch) { |
94
|
|
|
|
|
|
|
|
95
|
11
|
|
|
|
|
20
|
push @curr_patch_config_path, $k; |
96
|
|
|
|
|
|
|
|
97
|
11
|
100
|
66
|
|
|
76
|
if (! exists $struct->{$k}) { |
|
|
100
|
66
|
|
|
|
|
|
|
50
|
66
|
|
|
|
|
|
|
0
|
33
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
98
|
3
|
100
|
|
|
|
11
|
if ( $algo_id->{$algorithm} == 2 ) { |
|
|
100
|
|
|
|
|
|
99
|
1
|
|
|
|
|
3
|
$struct->{$k} = _recursive_undef_struct($patch->{$k}); |
100
|
|
|
|
|
|
|
} |
101
|
|
|
|
|
|
|
elsif ( $algo_id->{$algorithm} == 3 ) { |
102
|
1
|
|
|
|
|
4
|
_recursive_traverse_struct($patch->{$k}, join('.', @curr_patch_config_path)); |
103
|
|
|
|
|
|
|
} |
104
|
|
|
|
|
|
|
else { |
105
|
1
|
|
|
|
|
6
|
$struct->{$k} = $patch->{$k}; |
106
|
|
|
|
|
|
|
} |
107
|
|
|
|
|
|
|
|
108
|
|
|
|
|
|
|
} elsif ( |
109
|
|
|
|
|
|
|
(! ref $patch->{$k} && ! ref $struct->{$k}) |
110
|
|
|
|
|
|
|
|| (ref $patch->{$k} eq 'ARRAY' && (ref $struct->{$k} eq 'ARRAY')) |
111
|
|
|
|
|
|
|
|| (ref $patch->{$k} eq 'Regexp' && (ref $struct->{$k} eq 'Regexp')) |
112
|
|
|
|
|
|
|
) { |
113
|
7
|
100
|
|
|
|
18
|
if ( $algo_id->{$algorithm} == 2 ) { |
114
|
4
|
|
|
|
|
19
|
patch ($struct->{$k}, $patch->{$k}, $algorithm); |
115
|
|
|
|
|
|
|
} |
116
|
|
|
|
|
|
|
else { |
117
|
3
|
|
|
|
|
17
|
$struct->{$k} = $patch->{$k}; |
118
|
|
|
|
|
|
|
} |
119
|
|
|
|
|
|
|
} elsif (ref $patch->{$k} eq 'HASH' && (ref $struct->{$k} eq 'HASH')) { |
120
|
1
|
|
|
|
|
3
|
patch ($struct->{$k}, $patch->{$k}, $algorithm); |
121
|
|
|
|
|
|
|
} elsif (ref $patch->{$k} eq 'CODE' && (ref $struct->{$k} eq 'CODE' || ! defined $struct->{$k})) { |
122
|
0
|
|
|
|
|
0
|
$struct->{$k} = $patch->{$k}; |
123
|
|
|
|
|
|
|
} |
124
|
|
|
|
|
|
|
} |
125
|
|
|
|
|
|
|
} |
126
|
|
|
|
|
|
|
|
127
|
|
|
|
|
|
|
sub _recursive_undef_struct { |
128
|
21
|
|
|
21
|
|
18
|
my $data = shift; |
129
|
|
|
|
|
|
|
|
130
|
21
|
100
|
|
|
|
35
|
if ( ! ref $data ) { |
131
|
15
|
|
|
|
|
15
|
$data = undef; |
132
|
|
|
|
|
|
|
} |
133
|
|
|
|
|
|
|
else { |
134
|
6
|
100
|
|
|
|
20
|
if ( ref $data eq 'ARRAY' ) { |
|
|
50
|
|
|
|
|
|
135
|
3
|
|
|
|
|
5
|
@$data = map { _recursive_undef_struct($_) } @$data; |
|
11
|
|
|
|
|
20
|
|
136
|
|
|
|
|
|
|
} |
137
|
|
|
|
|
|
|
elsif ( ref $data eq 'HASH' ) { |
138
|
3
|
|
|
|
|
17
|
%$data = map { $_ => _recursive_undef_struct($data->{$_}) } keys %$data; |
|
6
|
|
|
|
|
19
|
|
139
|
|
|
|
|
|
|
} |
140
|
|
|
|
|
|
|
} |
141
|
|
|
|
|
|
|
|
142
|
21
|
|
|
|
|
57
|
return $data; |
143
|
|
|
|
|
|
|
} |
144
|
|
|
|
|
|
|
|
145
|
|
|
|
|
|
|
sub _recursive_traverse_struct { |
146
|
11
|
|
|
11
|
|
14
|
my $data = shift; |
147
|
11
|
|
|
|
|
11
|
my $name = shift; |
148
|
|
|
|
|
|
|
|
149
|
11
|
100
|
|
|
|
28
|
if ( ! ref $data) { |
|
|
100
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
150
|
6
|
|
|
|
|
14
|
$nonexistent_keys_in_config{$name} = 1; |
151
|
|
|
|
|
|
|
} |
152
|
|
|
|
|
|
|
elsif ( ref $data eq 'ARRAY' ) { |
153
|
1
|
|
|
|
|
2
|
foreach my $element (@$data) { |
154
|
4
|
100
|
|
|
|
8
|
if (! ref $element) { |
155
|
3
|
|
|
|
|
8
|
$nonexistent_keys_in_config{$name} = 'ARRAY of ' . scalar @$data . ' elements'; |
156
|
|
|
|
|
|
|
} |
157
|
|
|
|
|
|
|
else { |
158
|
1
|
|
|
|
|
2
|
_recursive_traverse_struct($element, $name); |
159
|
|
|
|
|
|
|
} |
160
|
|
|
|
|
|
|
} |
161
|
|
|
|
|
|
|
} |
162
|
|
|
|
|
|
|
elsif ( ref $data eq 'HASH' ) { |
163
|
4
|
|
|
|
|
9
|
foreach my $key ( keys %$data ) { |
164
|
8
|
100
|
|
|
|
37
|
$nonexistent_keys_in_config{"$name.$key"} = 1 if (! ref $data->{$key}); |
165
|
|
|
|
|
|
|
|
166
|
8
|
|
|
|
|
24
|
_recursive_traverse_struct($data->{$key}, "$name.$key"); |
167
|
|
|
|
|
|
|
} |
168
|
|
|
|
|
|
|
} |
169
|
|
|
|
|
|
|
|
170
|
11
|
|
|
|
|
24
|
return $data; |
171
|
|
|
|
|
|
|
} |
172
|
|
|
|
|
|
|
|
173
|
|
|
|
|
|
|
1; |