line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
# |
2
|
|
|
|
|
|
|
# This file is part of Config-Model |
3
|
|
|
|
|
|
|
# |
4
|
|
|
|
|
|
|
# This software is Copyright (c) 2005-2022 by Dominique Dumont. |
5
|
|
|
|
|
|
|
# |
6
|
|
|
|
|
|
|
# This is free software, licensed under: |
7
|
|
|
|
|
|
|
# |
8
|
|
|
|
|
|
|
# The GNU Lesser General Public License, Version 2.1, February 1999 |
9
|
|
|
|
|
|
|
# |
10
|
|
|
|
|
|
|
package Config::Model::Instance 2.153; # TRIAL |
11
|
|
|
|
|
|
|
|
12
|
|
|
|
|
|
|
#use Scalar::Util qw(weaken) ; |
13
|
59
|
|
|
59
|
|
471
|
use strict; |
|
59
|
|
|
|
|
176
|
|
|
59
|
|
|
|
|
2234
|
|
14
|
|
|
|
|
|
|
|
15
|
59
|
|
|
59
|
|
797
|
use 5.10.1; |
|
59
|
|
|
|
|
238
|
|
16
|
59
|
|
|
59
|
|
379
|
use Mouse; |
|
59
|
|
|
|
|
157
|
|
|
59
|
|
|
|
|
575
|
|
17
|
59
|
|
|
59
|
|
30270
|
use Mouse::Util::TypeConstraints; |
|
59
|
|
|
|
|
202
|
|
|
59
|
|
|
|
|
619
|
|
18
|
59
|
|
|
59
|
|
7270
|
use MouseX::StrictConstructor; |
|
59
|
|
|
|
|
225
|
|
|
59
|
|
|
|
|
498
|
|
19
|
|
|
|
|
|
|
with "Config::Model::Role::NodeLoader"; |
20
|
|
|
|
|
|
|
|
21
|
59
|
|
|
59
|
|
13277
|
use File::Path; |
|
59
|
|
|
|
|
172
|
|
|
59
|
|
|
|
|
4741
|
|
22
|
59
|
|
|
59
|
|
50266
|
use Path::Tiny; |
|
59
|
|
|
|
|
826969
|
|
|
59
|
|
|
|
|
3966
|
|
23
|
59
|
|
|
59
|
|
605
|
use Log::Log4perl qw(get_logger :levels); |
|
59
|
|
|
|
|
173
|
|
|
59
|
|
|
|
|
484
|
|
24
|
|
|
|
|
|
|
|
25
|
59
|
|
|
59
|
|
35129
|
use Config::Model::TypeConstraints; |
|
59
|
|
|
|
|
187
|
|
|
59
|
|
|
|
|
2002
|
|
26
|
59
|
|
|
59
|
|
29256
|
use Config::Model::Exception; |
|
59
|
|
|
|
|
270
|
|
|
59
|
|
|
|
|
3115
|
|
27
|
59
|
|
|
59
|
|
40972
|
use Config::Model::Node; |
|
59
|
|
|
|
|
417
|
|
|
59
|
|
|
|
|
2776
|
|
28
|
59
|
|
|
59
|
|
528
|
use Config::Model::Loader; |
|
59
|
|
|
|
|
142
|
|
|
59
|
|
|
|
|
1516
|
|
29
|
59
|
|
|
59
|
|
30839
|
use Config::Model::SearchElement; |
|
59
|
|
|
|
|
236
|
|
|
59
|
|
|
|
|
2509
|
|
30
|
59
|
|
|
59
|
|
29103
|
use Config::Model::Iterator; |
|
59
|
|
|
|
|
183
|
|
|
59
|
|
|
|
|
2023
|
|
31
|
59
|
|
|
59
|
|
467
|
use Config::Model::ObjTreeScanner; |
|
59
|
|
|
|
|
147
|
|
|
59
|
|
|
|
|
1164
|
|
32
|
|
|
|
|
|
|
|
33
|
59
|
|
|
59
|
|
311
|
use warnings ; |
|
59
|
|
|
|
|
172
|
|
|
59
|
|
|
|
|
2034
|
|
34
|
|
|
|
|
|
|
|
35
|
59
|
|
|
59
|
|
362
|
use Carp qw/carp croak confess cluck/; |
|
59
|
|
|
|
|
167
|
|
|
59
|
|
|
|
|
230784
|
|
36
|
|
|
|
|
|
|
|
37
|
|
|
|
|
|
|
my $logger = get_logger("Instance"); |
38
|
|
|
|
|
|
|
my $change_logger = get_logger("Anything::Change"); |
39
|
|
|
|
|
|
|
my $user_logger = get_logger("User"); |
40
|
|
|
|
|
|
|
|
41
|
|
|
|
|
|
|
has [qw/root_class_name/] => ( is => 'ro', isa => 'Str', required => 1 ); |
42
|
|
|
|
|
|
|
|
43
|
139
|
|
|
139
|
0
|
1074
|
sub location { return "in instance" } |
44
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
has config_model => ( |
46
|
|
|
|
|
|
|
is => 'ro', |
47
|
|
|
|
|
|
|
isa => 'Config::Model', |
48
|
|
|
|
|
|
|
weak_ref => 1, |
49
|
|
|
|
|
|
|
required => 1 |
50
|
|
|
|
|
|
|
); |
51
|
|
|
|
|
|
|
|
52
|
|
|
|
|
|
|
has check => ( |
53
|
|
|
|
|
|
|
is => 'ro', |
54
|
|
|
|
|
|
|
isa => 'Str', |
55
|
|
|
|
|
|
|
default => 'yes', |
56
|
|
|
|
|
|
|
reader => 'read_check', |
57
|
|
|
|
|
|
|
); |
58
|
|
|
|
|
|
|
|
59
|
|
|
|
|
|
|
# used by cme -create option |
60
|
|
|
|
|
|
|
has auto_create => ( |
61
|
|
|
|
|
|
|
is => 'ro', |
62
|
|
|
|
|
|
|
isa => 'Bool', |
63
|
|
|
|
|
|
|
default => 0, |
64
|
|
|
|
|
|
|
); |
65
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
# a unique (instance wise) placeholder for various tree objects |
67
|
|
|
|
|
|
|
# to store information |
68
|
|
|
|
|
|
|
has _safe => ( |
69
|
|
|
|
|
|
|
is => 'rw', |
70
|
|
|
|
|
|
|
isa => 'HashRef', |
71
|
|
|
|
|
|
|
traits => ['Hash'], |
72
|
|
|
|
|
|
|
default => sub { {} }, |
73
|
|
|
|
|
|
|
handles => { |
74
|
|
|
|
|
|
|
data => 'accessor', |
75
|
|
|
|
|
|
|
}, |
76
|
|
|
|
|
|
|
); |
77
|
|
|
|
|
|
|
|
78
|
|
|
|
|
|
|
has appli_info => ( |
79
|
|
|
|
|
|
|
is => 'rw', |
80
|
|
|
|
|
|
|
isa => 'HashRef', |
81
|
|
|
|
|
|
|
traits => ['Hash'], |
82
|
|
|
|
|
|
|
default => sub { {} }, |
83
|
|
|
|
|
|
|
handles => { |
84
|
|
|
|
|
|
|
get_appli_info => 'get', |
85
|
|
|
|
|
|
|
# currying See Moose::Manual::Delegation |
86
|
|
|
|
|
|
|
get_support_info => [qw/get support_info/], |
87
|
|
|
|
|
|
|
}, |
88
|
|
|
|
|
|
|
); |
89
|
|
|
|
|
|
|
|
90
|
|
|
|
|
|
|
|
91
|
|
|
|
|
|
|
# preset mode: to load values found by HW scan or other automatic scheme |
92
|
|
|
|
|
|
|
# layered mode: to load values found in included files (e.g. a la multistrap) |
93
|
|
|
|
|
|
|
# canonical mode: write config data back using model order instead of user order |
94
|
|
|
|
|
|
|
has [qw/preset layered canonical/] => ( |
95
|
|
|
|
|
|
|
is => 'ro', |
96
|
|
|
|
|
|
|
isa => 'Bool', |
97
|
|
|
|
|
|
|
default => 0, |
98
|
|
|
|
|
|
|
); |
99
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
has changes => ( |
101
|
|
|
|
|
|
|
is => 'ro', |
102
|
|
|
|
|
|
|
isa => 'ArrayRef', |
103
|
|
|
|
|
|
|
traits => ['Array'], |
104
|
|
|
|
|
|
|
default => sub { [] }, |
105
|
|
|
|
|
|
|
handles => { |
106
|
|
|
|
|
|
|
add_change => 'push', |
107
|
|
|
|
|
|
|
c_count => 'count', |
108
|
|
|
|
|
|
|
has_changes => 'count', |
109
|
|
|
|
|
|
|
|
110
|
|
|
|
|
|
|
#needs_save => 'count' , |
111
|
|
|
|
|
|
|
clear_changes => 'clear', |
112
|
|
|
|
|
|
|
} ); |
113
|
|
|
|
|
|
|
|
114
|
|
|
|
|
|
|
sub needs_save { |
115
|
65
|
|
|
65
|
1
|
13793
|
my $self = shift; |
116
|
65
|
|
|
|
|
123
|
my $arg = shift; |
117
|
65
|
50
|
|
|
|
191
|
if ( defined $arg ) { |
118
|
0
|
0
|
|
|
|
0
|
if ($arg) { |
119
|
0
|
|
|
|
|
0
|
croak "replace needs_save(1) call with add_change"; |
120
|
0
|
|
|
|
|
0
|
$self->add_change(); # may not work |
121
|
|
|
|
|
|
|
} |
122
|
|
|
|
|
|
|
else { |
123
|
0
|
|
|
|
|
0
|
croak "replace needs_save(0) call with clear_changes"; |
124
|
0
|
|
|
|
|
0
|
$self->clear_changes; |
125
|
|
|
|
|
|
|
} |
126
|
|
|
|
|
|
|
} |
127
|
65
|
|
|
|
|
253
|
return $self->c_count; |
128
|
|
|
|
|
|
|
} |
129
|
|
|
|
|
|
|
|
130
|
|
|
|
|
|
|
has errors => ( |
131
|
|
|
|
|
|
|
is => 'ro', |
132
|
|
|
|
|
|
|
isa => 'HashRef', |
133
|
|
|
|
|
|
|
traits => ['Hash'], |
134
|
|
|
|
|
|
|
default => sub { {} }, |
135
|
|
|
|
|
|
|
handles => { |
136
|
|
|
|
|
|
|
_set_error => 'set', |
137
|
|
|
|
|
|
|
cancel_error => 'delete', |
138
|
|
|
|
|
|
|
has_error => 'count', |
139
|
|
|
|
|
|
|
clear_errors => 'clear', |
140
|
|
|
|
|
|
|
error_paths => 'keys' |
141
|
|
|
|
|
|
|
} ); |
142
|
|
|
|
|
|
|
|
143
|
|
|
|
|
|
|
sub add_error { |
144
|
21
|
|
|
21
|
0
|
52
|
my $self = shift; |
145
|
21
|
|
|
|
|
84
|
$self->_set_error( shift, '' ); |
146
|
|
|
|
|
|
|
} |
147
|
|
|
|
|
|
|
|
148
|
|
|
|
|
|
|
sub error_messages { |
149
|
9
|
|
|
9
|
1
|
7271
|
my $self = shift; |
150
|
9
|
|
|
|
|
40
|
my @errs = map { "$_: " . $self->config_root->grab($_)->error_msg } $self->error_paths; |
|
13
|
|
|
|
|
160
|
|
151
|
9
|
50
|
|
|
|
67
|
return wantarray ? @errs : join( "\n", @errs ); |
152
|
|
|
|
|
|
|
} |
153
|
|
|
|
|
|
|
|
154
|
|
|
|
|
|
|
sub has_warning { |
155
|
3
|
|
|
3
|
1
|
551
|
my $self = shift; |
156
|
|
|
|
|
|
|
|
157
|
|
|
|
|
|
|
my $count_leaf_warnings = sub { |
158
|
75
|
|
|
75
|
|
465
|
my ( $scanner, $data_ref, $node, $element_name, $index, $leaf_object ) = @_; |
159
|
75
|
|
|
|
|
202
|
$$data_ref += $leaf_object->has_warning; |
160
|
3
|
|
|
|
|
18
|
}; |
161
|
|
|
|
|
|
|
|
162
|
|
|
|
|
|
|
my $count_list_warnings = sub { |
163
|
9
|
|
|
9
|
|
20
|
my ( $scanner, $data_ref, $node, $element_name, $index, $leaf_object ) = @_; |
164
|
9
|
|
|
|
|
27
|
$$data_ref += $node->fetch_element($element_name)->has_warning; |
165
|
3
|
|
|
|
|
13
|
}; |
166
|
|
|
|
|
|
|
|
167
|
3
|
|
|
|
|
28
|
my $scan = Config::Model::ObjTreeScanner->new( |
168
|
|
|
|
|
|
|
leaf_cb => $count_leaf_warnings, |
169
|
|
|
|
|
|
|
list_element_hook => $count_list_warnings, |
170
|
|
|
|
|
|
|
hash_element_hook => $count_list_warnings, |
171
|
|
|
|
|
|
|
); |
172
|
|
|
|
|
|
|
|
173
|
3
|
|
|
|
|
11
|
my $result = 0; |
174
|
3
|
|
|
|
|
21
|
$scan->scan_node( \$result, $self->config_root ); |
175
|
|
|
|
|
|
|
|
176
|
3
|
|
|
|
|
73
|
return $result; |
177
|
|
|
|
|
|
|
} |
178
|
|
|
|
|
|
|
|
179
|
|
|
|
|
|
|
has on_change_cb => ( |
180
|
|
|
|
|
|
|
is => 'rw', |
181
|
|
|
|
|
|
|
traits => ['Code'], |
182
|
|
|
|
|
|
|
isa => 'CodeRef', |
183
|
|
|
|
|
|
|
default => sub { sub { } }, |
184
|
|
|
|
|
|
|
); |
185
|
|
|
|
|
|
|
|
186
|
|
|
|
|
|
|
has on_message_cb => ( |
187
|
|
|
|
|
|
|
traits => ['Code'], |
188
|
|
|
|
|
|
|
is => 'rw', |
189
|
|
|
|
|
|
|
isa => 'CodeRef', |
190
|
|
|
|
|
|
|
default => sub { sub { say @_; } }, |
191
|
|
|
|
|
|
|
handles => { |
192
|
|
|
|
|
|
|
show_message => 'execute', |
193
|
|
|
|
|
|
|
}, |
194
|
|
|
|
|
|
|
); |
195
|
|
|
|
|
|
|
|
196
|
|
|
|
|
|
|
# initial_load mode: when data is loaded the first time |
197
|
|
|
|
|
|
|
has initial_load => ( |
198
|
|
|
|
|
|
|
is => 'rw', |
199
|
|
|
|
|
|
|
isa => 'Bool', |
200
|
|
|
|
|
|
|
default => 0, |
201
|
|
|
|
|
|
|
trigger => \&_trace_initial_load, |
202
|
|
|
|
|
|
|
traits => [qw/Bool/], |
203
|
|
|
|
|
|
|
handles => { |
204
|
|
|
|
|
|
|
initial_load_start => 'set', |
205
|
|
|
|
|
|
|
initial_load_stop => 'unset', |
206
|
|
|
|
|
|
|
} ); |
207
|
|
|
|
|
|
|
|
208
|
|
|
|
|
|
|
sub _trace_initial_load { |
209
|
206
|
|
|
206
|
|
8420
|
my ( $self, $n, $o ) = @_; |
210
|
206
|
|
|
|
|
878
|
$logger->debug("switched to $n"); |
211
|
|
|
|
|
|
|
} |
212
|
|
|
|
|
|
|
|
213
|
|
|
|
|
|
|
# This array holds a set of sub ref that will be invoked when |
214
|
|
|
|
|
|
|
# the user requires to write all configuration tree in their |
215
|
|
|
|
|
|
|
# backend storage. |
216
|
|
|
|
|
|
|
has _write_back => ( |
217
|
|
|
|
|
|
|
is => 'ro', |
218
|
|
|
|
|
|
|
isa => 'HashRef', |
219
|
|
|
|
|
|
|
traits => ['Hash'], |
220
|
|
|
|
|
|
|
handles => { |
221
|
|
|
|
|
|
|
count_write_back => 'count', # mostly for tests |
222
|
|
|
|
|
|
|
has_no_write_back => 'is_empty', |
223
|
|
|
|
|
|
|
nodes_to_write_back => 'keys', |
224
|
|
|
|
|
|
|
write_back_node_info => 'get', |
225
|
|
|
|
|
|
|
delete_write_back => 'delete', |
226
|
|
|
|
|
|
|
clear_write_back => 'clear', |
227
|
|
|
|
|
|
|
}, |
228
|
|
|
|
|
|
|
default => sub { {} }, |
229
|
|
|
|
|
|
|
); |
230
|
|
|
|
|
|
|
|
231
|
|
|
|
|
|
|
sub register_write_back { |
232
|
93
|
|
|
93
|
1
|
326
|
my ($self, $path, $backend, $wb) = @_; |
233
|
93
|
|
100
|
|
|
172
|
push @{ $self->_write_back->{$path} //= [] }, [$backend, $wb]; |
|
93
|
|
|
|
|
1001
|
|
234
|
|
|
|
|
|
|
} |
235
|
|
|
|
|
|
|
|
236
|
|
|
|
|
|
|
# used for auto_read auto_write feature |
237
|
|
|
|
|
|
|
has [qw/name application backend_arg backup/] => ( |
238
|
|
|
|
|
|
|
is => 'ro', |
239
|
|
|
|
|
|
|
isa => 'Maybe[Str]', |
240
|
|
|
|
|
|
|
); |
241
|
|
|
|
|
|
|
|
242
|
|
|
|
|
|
|
has 'root_dir' => ( |
243
|
|
|
|
|
|
|
is => 'ro', |
244
|
|
|
|
|
|
|
isa => 'Config::Model::TypeContraints::Path', |
245
|
|
|
|
|
|
|
coerce => 1 |
246
|
|
|
|
|
|
|
); |
247
|
|
|
|
|
|
|
|
248
|
|
|
|
|
|
|
has root_path => ( |
249
|
|
|
|
|
|
|
is => 'ro', |
250
|
|
|
|
|
|
|
isa => 'Path::Tiny', |
251
|
|
|
|
|
|
|
lazy_build => 1, |
252
|
|
|
|
|
|
|
); |
253
|
|
|
|
|
|
|
|
254
|
|
|
|
|
|
|
sub _build_root_path { |
255
|
2
|
|
|
2
|
|
57
|
my $self = shift; |
256
|
2
|
|
100
|
|
|
54
|
my $root_dir = $self->root_dir // ''; |
257
|
2
|
100
|
|
|
|
21
|
return $root_dir ? path($root_dir) : Path::Tiny->cwd; |
258
|
|
|
|
|
|
|
} |
259
|
|
|
|
|
|
|
|
260
|
|
|
|
|
|
|
has [qw/config_dir config_file/] => ( |
261
|
|
|
|
|
|
|
is => 'ro', |
262
|
|
|
|
|
|
|
isa => 'Config::Model::TypeContraints::Path', |
263
|
|
|
|
|
|
|
coerce => 1 |
264
|
|
|
|
|
|
|
); |
265
|
|
|
|
|
|
|
|
266
|
|
|
|
|
|
|
has tree => ( |
267
|
|
|
|
|
|
|
is => 'ro', |
268
|
|
|
|
|
|
|
isa => 'Config::Model::Node', |
269
|
|
|
|
|
|
|
builder => '_build_tree', |
270
|
|
|
|
|
|
|
lazy => 1, |
271
|
|
|
|
|
|
|
clearer => '_clear_config', |
272
|
|
|
|
|
|
|
reader => 'config_root', |
273
|
|
|
|
|
|
|
handles => [qw/apply_fixes deep_check grab grab_value/], |
274
|
|
|
|
|
|
|
); |
275
|
|
|
|
|
|
|
|
276
|
|
|
|
|
|
|
sub reset_config { |
277
|
1
|
|
|
1
|
1
|
477
|
my $self = shift; |
278
|
1
|
|
|
|
|
6
|
$self->_clear_config; |
279
|
1
|
|
|
|
|
5
|
$self->clear_changes; |
280
|
1
|
|
|
|
|
23
|
return $self->config_root; |
281
|
|
|
|
|
|
|
} |
282
|
|
|
|
|
|
|
|
283
|
|
|
|
|
|
|
sub _build_tree { |
284
|
139
|
|
|
139
|
|
39231
|
my $self = shift; |
285
|
|
|
|
|
|
|
|
286
|
|
|
|
|
|
|
return $self->load_node ( |
287
|
|
|
|
|
|
|
config_class_name => $self->{root_class_name}, |
288
|
|
|
|
|
|
|
instance => $self, |
289
|
|
|
|
|
|
|
container => $self, |
290
|
|
|
|
|
|
|
config_file => $self->{config_file}, |
291
|
139
|
|
|
|
|
1158
|
); |
292
|
|
|
|
|
|
|
} |
293
|
|
|
|
|
|
|
|
294
|
|
|
|
|
|
|
sub preset_start { |
295
|
5
|
|
|
5
|
1
|
3915
|
my $self = shift; |
296
|
5
|
|
|
|
|
33
|
$logger->info("Starting preset mode"); |
297
|
|
|
|
|
|
|
carp "Cannot start preset mode during layered mode" |
298
|
5
|
50
|
|
|
|
77
|
if $self->{layered}; |
299
|
5
|
|
|
|
|
21
|
$self->{preset} = 1; |
300
|
|
|
|
|
|
|
} |
301
|
|
|
|
|
|
|
|
302
|
|
|
|
|
|
|
sub preset_stop { |
303
|
5
|
|
|
5
|
1
|
35
|
my $self = shift; |
304
|
5
|
|
|
|
|
24
|
$logger->info("Stopping preset mode"); |
305
|
5
|
|
|
|
|
50
|
$self->{preset} = 0; |
306
|
|
|
|
|
|
|
} |
307
|
|
|
|
|
|
|
|
308
|
|
|
|
|
|
|
sub preset_clear { |
309
|
1
|
|
|
1
|
1
|
4
|
my $self = shift; |
310
|
|
|
|
|
|
|
|
311
|
|
|
|
|
|
|
my $leaf_cb = sub { |
312
|
42
|
|
|
42
|
|
84
|
my ( $scanner, $data_ref, $node, $element_name, $index, $leaf_object ) = @_; |
313
|
42
|
|
100
|
|
|
164
|
$$data_ref ||= $leaf_object->clear_preset; |
314
|
1
|
|
|
|
|
7
|
}; |
315
|
|
|
|
|
|
|
|
316
|
1
|
|
|
|
|
6
|
$self->_stuff_clear($leaf_cb); |
317
|
|
|
|
|
|
|
} |
318
|
|
|
|
|
|
|
|
319
|
|
|
|
|
|
|
sub layered_start { |
320
|
15
|
|
|
15
|
1
|
678
|
my $self = shift; |
321
|
15
|
|
|
|
|
87
|
$logger->info("Starting layered mode"); |
322
|
|
|
|
|
|
|
carp "Cannot start layered mode during preset mode" |
323
|
15
|
50
|
|
|
|
219
|
if $self->{preset}; |
324
|
15
|
|
|
|
|
67
|
$self->{layered} = 1; |
325
|
|
|
|
|
|
|
} |
326
|
|
|
|
|
|
|
|
327
|
|
|
|
|
|
|
sub layered_stop { |
328
|
14
|
|
|
14
|
1
|
103
|
my $self = shift; |
329
|
14
|
|
|
|
|
65
|
$logger->info("Stopping layered mode"); |
330
|
14
|
|
|
|
|
153
|
$self->{layered} = 0; |
331
|
|
|
|
|
|
|
} |
332
|
|
|
|
|
|
|
|
333
|
|
|
|
|
|
|
sub layered_clear { |
334
|
10
|
|
|
10
|
1
|
28
|
my $self = shift; |
335
|
|
|
|
|
|
|
|
336
|
|
|
|
|
|
|
my $leaf_cb = sub { |
337
|
137
|
|
|
137
|
|
298
|
my ( $scanner, $data_ref, $node, $element_name, $index, $leaf_object ) = @_; |
338
|
137
|
|
100
|
|
|
2458
|
$$data_ref ||= $leaf_object->clear_layered; |
339
|
10
|
|
|
|
|
55
|
}; |
340
|
|
|
|
|
|
|
|
341
|
10
|
|
|
|
|
51
|
$self->_stuff_clear($leaf_cb); |
342
|
|
|
|
|
|
|
} |
343
|
|
|
|
|
|
|
|
344
|
|
|
|
|
|
|
sub get_data_mode { |
345
|
1665
|
|
|
1665
|
1
|
3283
|
my $self = shift; |
346
|
|
|
|
|
|
|
return |
347
|
|
|
|
|
|
|
$self->{layered} ? 'layered' |
348
|
1665
|
100
|
|
|
|
6062
|
: $self->{preset} ? 'preset' |
|
|
100
|
|
|
|
|
|
349
|
|
|
|
|
|
|
: 'normal'; |
350
|
|
|
|
|
|
|
} |
351
|
|
|
|
|
|
|
|
352
|
|
|
|
|
|
|
sub _stuff_clear { |
353
|
11
|
|
|
11
|
|
27
|
my ( $self, $leaf_cb ) = @_; |
354
|
|
|
|
|
|
|
|
355
|
|
|
|
|
|
|
# this sub may remove hash keys that were entered by user if the |
356
|
|
|
|
|
|
|
# corresponding hash value has no data. |
357
|
|
|
|
|
|
|
# it also clear auto_created ids if there's no data in there |
358
|
|
|
|
|
|
|
my $h_cb = sub { |
359
|
48
|
|
|
48
|
|
125
|
my ( $scanner, $data_ref, $node, $element_name, @keys ) = @_; |
360
|
48
|
|
|
|
|
131
|
my $obj = $node->fetch_element($element_name); |
361
|
|
|
|
|
|
|
|
362
|
|
|
|
|
|
|
# Since remove method uses splice(array) on list elements, the |
363
|
|
|
|
|
|
|
# removal must be done in reverse order to avoid messing up |
364
|
|
|
|
|
|
|
# the indexes of the array (i.e. the last indexes becomes |
365
|
|
|
|
|
|
|
# greater than the length of the array). |
366
|
48
|
|
|
|
|
208
|
foreach my $k (reverse @keys) { |
367
|
65
|
|
|
|
|
105
|
my $has_data = 0; |
368
|
65
|
|
|
|
|
205
|
$scanner->scan_hash( \$has_data, $node, $element_name, $k ); |
369
|
65
|
100
|
|
|
|
166
|
$obj->remove($k) unless $has_data; |
370
|
65
|
|
100
|
|
|
420
|
$$data_ref ||= $has_data; |
371
|
|
|
|
|
|
|
} |
372
|
11
|
|
|
|
|
49
|
}; |
373
|
|
|
|
|
|
|
|
374
|
11
|
|
|
|
|
94
|
my $wiper = Config::Model::ObjTreeScanner->new( |
375
|
|
|
|
|
|
|
fallback => 'all', |
376
|
|
|
|
|
|
|
auto_vivify => 0, |
377
|
|
|
|
|
|
|
check => 'skip', |
378
|
|
|
|
|
|
|
leaf_cb => $leaf_cb, |
379
|
|
|
|
|
|
|
hash_element_cb => $h_cb, |
380
|
|
|
|
|
|
|
list_element_cb => $h_cb, |
381
|
|
|
|
|
|
|
); |
382
|
|
|
|
|
|
|
|
383
|
11
|
|
|
|
|
81
|
$wiper->scan_node( undef, $self->config_root ); |
384
|
|
|
|
|
|
|
|
385
|
|
|
|
|
|
|
} |
386
|
|
|
|
|
|
|
|
387
|
|
|
|
|
|
|
sub modify { |
388
|
2
|
|
|
2
|
1
|
52
|
my $self = shift ; |
389
|
2
|
50
|
|
|
|
15
|
my %args = @_ eq 1 ? ( step => $_[0] ) : @_; |
390
|
2
|
|
33
|
|
|
13
|
my $force = delete $args{force_save} || delete $args{force}; |
391
|
2
|
|
|
|
|
5
|
my $quiet = delete $args{quiet}; |
392
|
2
|
|
|
|
|
13
|
$self->load(%args); |
393
|
2
|
50
|
|
|
|
28
|
$self->say_changes() unless $quiet; |
394
|
2
|
|
|
|
|
9
|
$self->write_back( force => $force ); |
395
|
2
|
|
|
|
|
39
|
return $self; |
396
|
|
|
|
|
|
|
} |
397
|
|
|
|
|
|
|
|
398
|
|
|
|
|
|
|
sub load { |
399
|
3
|
|
|
3
|
1
|
1845
|
my $self = shift; |
400
|
3
|
|
|
|
|
56
|
my $loader = Config::Model::Loader->new( start_node => $self->config_root ); |
401
|
3
|
50
|
|
|
|
160
|
my %args = @_ eq 1 ? ( step => $_[0] ) : @_; |
402
|
3
|
|
|
|
|
28
|
$loader->load( %args ); |
403
|
3
|
|
|
|
|
33
|
return $self; |
404
|
|
|
|
|
|
|
} |
405
|
|
|
|
|
|
|
|
406
|
|
|
|
|
|
|
sub search_element { |
407
|
0
|
|
|
0
|
0
|
0
|
my $self = shift; |
408
|
0
|
|
|
|
|
0
|
$self->config_root->search_element(@_); |
409
|
|
|
|
|
|
|
} |
410
|
|
|
|
|
|
|
|
411
|
|
|
|
|
|
|
sub wizard_helper { |
412
|
0
|
|
|
0
|
1
|
0
|
carp __PACKAGE__, "::wizard_helper helped is deprecated. Call iterator instead"; |
413
|
0
|
|
|
|
|
0
|
goto &iterator; |
414
|
|
|
|
|
|
|
} |
415
|
|
|
|
|
|
|
|
416
|
|
|
|
|
|
|
sub iterator { |
417
|
1
|
|
|
1
|
1
|
2
|
my $self = shift; |
418
|
1
|
|
|
|
|
5
|
my @args = @_; |
419
|
|
|
|
|
|
|
|
420
|
1
|
|
|
|
|
5
|
my $tree_root = $self->config_root; |
421
|
|
|
|
|
|
|
|
422
|
1
|
|
|
|
|
11
|
return Config::Model::Iterator->new( root => $tree_root, @args ); |
423
|
|
|
|
|
|
|
} |
424
|
|
|
|
|
|
|
|
425
|
|
|
|
|
|
|
sub read_directory { |
426
|
0
|
|
|
0
|
0
|
0
|
carp "read_directory is deprecated"; |
427
|
0
|
|
|
|
|
0
|
return shift->root_dir; |
428
|
|
|
|
|
|
|
} |
429
|
|
|
|
|
|
|
|
430
|
|
|
|
|
|
|
sub write_directory { |
431
|
0
|
|
|
0
|
0
|
0
|
my $self = shift; |
432
|
0
|
|
|
|
|
0
|
carp "write_directory is deprecated"; |
433
|
0
|
|
|
|
|
0
|
return $self->root_dir; |
434
|
|
|
|
|
|
|
} |
435
|
|
|
|
|
|
|
|
436
|
|
|
|
|
|
|
sub write_root_dir { |
437
|
0
|
|
|
0
|
0
|
0
|
my $self = shift; |
438
|
0
|
|
|
|
|
0
|
carp "deprecated"; |
439
|
0
|
|
|
|
|
0
|
return $self->root_dir; |
440
|
|
|
|
|
|
|
} |
441
|
|
|
|
|
|
|
|
442
|
|
|
|
|
|
|
# FIXME: record changes to implement undo/redo ? |
443
|
|
|
|
|
|
|
sub notify_change { |
444
|
1520
|
|
|
1520
|
1
|
2531
|
my $self = shift; |
445
|
1520
|
|
|
|
|
6890
|
my %args = @_; |
446
|
1520
|
100
|
|
|
|
3996
|
if ( $change_logger->is_debug ) { |
447
|
95
|
|
|
|
|
618
|
$change_logger->debug( "in instance ", $self->name, ' for path ', $args{path} ); |
448
|
|
|
|
|
|
|
} |
449
|
|
|
|
|
|
|
|
450
|
1520
|
|
|
|
|
10514
|
foreach my $obsolete (qw/note_only msg/) { |
451
|
3040
|
50
|
|
|
|
8001
|
if ( my $m = delete $args{$obsolete} ) { |
452
|
0
|
|
|
|
|
0
|
carp "notify_change: param $obsolete is obsolete ($m)"; |
453
|
0
|
|
0
|
|
|
0
|
$args{note} //=''; |
454
|
0
|
|
|
|
|
0
|
$args{note} .= $m; |
455
|
|
|
|
|
|
|
} |
456
|
|
|
|
|
|
|
} |
457
|
|
|
|
|
|
|
|
458
|
1520
|
|
|
|
|
6126
|
$self->add_change( \%args ); |
459
|
1520
|
|
|
|
|
29460
|
$self->on_change_cb->( %args ); |
460
|
|
|
|
|
|
|
} |
461
|
|
|
|
|
|
|
|
462
|
|
|
|
|
|
|
sub _truncate { |
463
|
37
|
|
|
37
|
|
91
|
my @lines = @_; |
464
|
37
|
|
|
|
|
79
|
foreach my $l (@lines) { |
465
|
74
|
100
|
|
|
|
140
|
next unless defined $l; |
466
|
52
|
|
|
|
|
103
|
$l =~ s/\n/ /g; |
467
|
52
|
50
|
|
|
|
113
|
substr ($l, 60) = '[...]' if length $l > 60; # limit string length |
468
|
|
|
|
|
|
|
} |
469
|
37
|
|
|
|
|
117
|
return @lines; |
470
|
|
|
|
|
|
|
} |
471
|
|
|
|
|
|
|
|
472
|
|
|
|
|
|
|
sub list_changes { |
473
|
19
|
|
|
19
|
1
|
5454
|
my $self = shift; |
474
|
19
|
|
|
|
|
77
|
my $l = $self->changes; |
475
|
19
|
|
|
|
|
46
|
my @all; |
476
|
|
|
|
|
|
|
|
477
|
19
|
|
|
|
|
58
|
foreach my $c (@$l) { |
478
|
37
|
|
|
|
|
79
|
my $path = $c->{path} ; |
479
|
|
|
|
|
|
|
|
480
|
37
|
|
100
|
|
|
110
|
my $vt = $c->{value_type} || ''; |
481
|
37
|
|
|
|
|
116
|
my ( $o, $n ) = _truncate( $c->{old}, $c->{new} ); |
482
|
|
|
|
|
|
|
|
483
|
37
|
100
|
|
|
|
131
|
my $note = $c->{note} ? " # $c->{note}" : ''; |
484
|
|
|
|
|
|
|
|
485
|
37
|
100
|
100
|
|
|
322
|
if ( defined $n and not defined $o ) { |
|
|
100
|
100
|
|
|
|
|
|
|
100
|
66
|
|
|
|
|
|
|
50
|
|
|
|
|
|
486
|
2
|
|
|
|
|
10
|
push @all, "$path has new value: '$n'$note"; |
487
|
|
|
|
|
|
|
} |
488
|
|
|
|
|
|
|
elsif ( not defined $n and defined $o) { |
489
|
6
|
|
|
|
|
22
|
push @all, "$path deleted value: '$o'$note"; |
490
|
|
|
|
|
|
|
} |
491
|
|
|
|
|
|
|
elsif ( defined $o and defined $n ) { |
492
|
22
|
|
|
|
|
109
|
push @all, "$path: '$o' -> '$n'$note"; |
493
|
|
|
|
|
|
|
} |
494
|
|
|
|
|
|
|
elsif ( defined $c->{note} ) { |
495
|
7
|
|
|
|
|
37
|
push @all, "$path: ".$c->{note}; |
496
|
|
|
|
|
|
|
} |
497
|
|
|
|
|
|
|
else { |
498
|
|
|
|
|
|
|
# something's unexpected with the call to notify_change |
499
|
0
|
|
|
|
|
0
|
push @all, "changed ".join(' ', each %$c); |
500
|
|
|
|
|
|
|
} |
501
|
|
|
|
|
|
|
} |
502
|
|
|
|
|
|
|
|
503
|
19
|
100
|
|
|
|
221
|
return wantarray ? @all : join( "\n", @all ); |
504
|
|
|
|
|
|
|
} |
505
|
|
|
|
|
|
|
|
506
|
|
|
|
|
|
|
sub say_changes { |
507
|
2
|
|
|
2
|
1
|
6
|
my $self = shift; |
508
|
2
|
|
|
|
|
10
|
my @changes = $self->list_changes; |
509
|
2
|
50
|
|
|
|
9
|
return $self unless @changes; |
510
|
|
|
|
|
|
|
|
511
|
2
|
|
33
|
|
|
23
|
my $msg = "\n" . |
512
|
|
|
|
|
|
|
join( "\n- ", "Changes applied to " . ($self->application // $self->name) . " configuration:", @changes ) . |
513
|
|
|
|
|
|
|
"\n"; |
514
|
|
|
|
|
|
|
|
515
|
2
|
|
|
|
|
11
|
$user_logger->info($msg); |
516
|
2
|
|
|
|
|
56
|
return $self; |
517
|
|
|
|
|
|
|
} |
518
|
|
|
|
|
|
|
|
519
|
|
|
|
|
|
|
sub write_back { |
520
|
39
|
|
|
39
|
1
|
13824
|
my $self = shift; |
521
|
39
|
50
|
|
|
|
267
|
my %args = |
|
|
100
|
|
|
|
|
|
522
|
|
|
|
|
|
|
scalar @_ > 1 ? @_ |
523
|
|
|
|
|
|
|
: scalar @_ == 1 ? ( config_dir => $_[0] ) |
524
|
|
|
|
|
|
|
: (); |
525
|
|
|
|
|
|
|
|
526
|
39
|
|
100
|
|
|
216
|
my $force_write = delete $args{force} || 0; |
527
|
|
|
|
|
|
|
|
528
|
39
|
50
|
|
|
|
162
|
if (delete $args{root}) { |
529
|
0
|
|
|
|
|
0
|
say "write_back: root argument is no longer supported"; |
530
|
|
|
|
|
|
|
} |
531
|
|
|
|
|
|
|
|
532
|
|
|
|
|
|
|
# make sure that root node is loaded |
533
|
39
|
|
|
|
|
348
|
$self->config_root->init; |
534
|
|
|
|
|
|
|
|
535
|
39
|
100
|
|
|
|
187
|
if ($force_write) { |
536
|
|
|
|
|
|
|
# make sure that the whole tree is loaded |
537
|
20
|
|
|
|
|
97
|
my $dump = $self->config_root->dump_tree; |
538
|
|
|
|
|
|
|
} |
539
|
|
|
|
|
|
|
|
540
|
39
|
|
|
|
|
217
|
foreach my $k ( keys %args ) { |
541
|
2
|
50
|
|
|
|
13
|
if ($k eq 'config_dir') { |
|
|
50
|
|
|
|
|
|
542
|
0
|
|
0
|
|
|
0
|
$args{$k} ||= ''; |
543
|
0
|
0
|
0
|
|
|
0
|
$args{$k} .= '/' if $args{$k} and $args{$k} !~ m(/$); |
544
|
|
|
|
|
|
|
} |
545
|
|
|
|
|
|
|
elsif ( $k ne 'config_file' ) { |
546
|
0
|
|
|
|
|
0
|
croak "write_back: wrong parameters $k"; |
547
|
|
|
|
|
|
|
} |
548
|
|
|
|
|
|
|
} |
549
|
|
|
|
|
|
|
|
550
|
39
|
50
|
|
|
|
251
|
if ($self->has_no_write_back ) { |
551
|
0
|
0
|
|
|
|
0
|
my $info = $self->application ? "the model of application ".$self->application |
552
|
|
|
|
|
|
|
: "model ".$self->root_class_name ; |
553
|
0
|
|
|
|
|
0
|
croak "Don't know how to save data of $self->{name} instance. ", |
554
|
|
|
|
|
|
|
"Either $info has no configured ", |
555
|
|
|
|
|
|
|
"read/write backend or no node containing a backend was loaded. ", |
556
|
|
|
|
|
|
|
"Try with -force option or add read/write backend to $info\n"; |
557
|
|
|
|
|
|
|
} |
558
|
|
|
|
|
|
|
|
559
|
39
|
|
|
|
|
643
|
foreach my $path ( sort $self->nodes_to_write_back ) { |
560
|
52
|
|
|
|
|
769
|
$logger->info("write_back called on node $path"); |
561
|
|
|
|
|
|
|
|
562
|
52
|
50
|
66
|
|
|
583
|
if ( $path and $self->{config_file} ) { |
563
|
0
|
|
|
|
|
0
|
$logger->warn("write_back: cannot override config_file in non root node ($path)"); |
564
|
|
|
|
|
|
|
delete $self->{config_file} |
565
|
0
|
|
|
|
|
0
|
} |
566
|
|
|
|
|
|
|
|
567
|
52
|
|
|
|
|
234
|
$self->_write_back_node(%args, path => $path, force_write => $force_write) ; |
568
|
|
|
|
|
|
|
} |
569
|
39
|
|
|
|
|
508
|
$self->clear_changes; |
570
|
|
|
|
|
|
|
} |
571
|
|
|
|
|
|
|
|
572
|
|
|
|
|
|
|
sub _write_back_node { |
573
|
52
|
|
|
52
|
|
139
|
my $self = shift; |
574
|
52
|
|
|
|
|
218
|
my %args = @_; |
575
|
|
|
|
|
|
|
|
576
|
52
|
|
|
|
|
161
|
my $path = delete $args{path}; |
577
|
52
|
|
|
|
|
117
|
my $force_write = delete $args{force_write}; |
578
|
|
|
|
|
|
|
|
579
|
52
|
|
|
|
|
360
|
my $node = $self->config_root->grab( |
580
|
|
|
|
|
|
|
step => $path, |
581
|
|
|
|
|
|
|
type => 'node', |
582
|
|
|
|
|
|
|
mode => 'loose', |
583
|
|
|
|
|
|
|
autoadd => 0, |
584
|
|
|
|
|
|
|
); |
585
|
|
|
|
|
|
|
|
586
|
52
|
|
|
|
|
149
|
foreach my $wb_info (@{ $self->write_back_node_info($path) }) { |
|
52
|
|
|
|
|
346
|
|
587
|
52
|
|
|
|
|
997
|
my ($backend, $cb) = @$wb_info; |
588
|
|
|
|
|
|
|
|
589
|
|
|
|
|
|
|
my @wb_args = ( |
590
|
|
|
|
|
|
|
%args, |
591
|
|
|
|
|
|
|
config_file => $self->{config_file}, |
592
|
52
|
|
|
|
|
391
|
force => $force_write, |
593
|
|
|
|
|
|
|
backup => $self->backup, |
594
|
|
|
|
|
|
|
); |
595
|
|
|
|
|
|
|
|
596
|
52
|
50
|
66
|
|
|
531
|
if (defined $node and ($node->needs_save or $force_write)) { |
|
|
|
100
|
|
|
|
|
597
|
51
|
|
|
|
|
131
|
my $dir = $args{config_dir}; |
598
|
51
|
50
|
33
|
|
|
203
|
mkpath( $dir, 0, oct(755) ) if $dir and not -d $dir; |
599
|
|
|
|
|
|
|
|
600
|
|
|
|
|
|
|
# exit when write is successfull |
601
|
51
|
|
|
|
|
217
|
my $res = $cb->(@wb_args); |
602
|
51
|
50
|
|
|
|
569
|
$logger->info( "write_back called with $backend backend, result is ", |
603
|
|
|
|
|
|
|
defined $res ? $res : '<undef>' ); |
604
|
|
|
|
|
|
|
} |
605
|
|
|
|
|
|
|
|
606
|
52
|
100
|
|
|
|
826
|
if (not defined $node) { |
607
|
1
|
|
|
|
|
14
|
$logger->debug("deleting file for deleted node $path"); |
608
|
1
|
|
|
|
|
15
|
$cb->(@wb_args, force_delete => 1); |
609
|
1
|
|
|
|
|
7
|
$self->delete_write_back($path); |
610
|
|
|
|
|
|
|
} |
611
|
|
|
|
|
|
|
} |
612
|
|
|
|
|
|
|
|
613
|
52
|
|
|
|
|
354
|
$logger->trace( "write_back on node '$path' done" ); |
614
|
|
|
|
|
|
|
} |
615
|
|
|
|
|
|
|
|
616
|
|
|
|
|
|
|
sub save { |
617
|
2
|
|
|
2
|
1
|
392
|
goto &write_back; |
618
|
|
|
|
|
|
|
} |
619
|
|
|
|
|
|
|
|
620
|
|
|
|
|
|
|
sub update { |
621
|
0
|
|
|
0
|
1
|
0
|
my ($self, %args) = @_; |
622
|
|
|
|
|
|
|
|
623
|
0
|
|
|
|
|
0
|
my @msgs ; |
624
|
|
|
|
|
|
|
my $hook = sub { |
625
|
0
|
|
|
0
|
|
0
|
my ($scanner, $data_ref,$node,@element_list) = @_; |
626
|
0
|
0
|
|
|
|
0
|
if ($node->can('update')) { |
627
|
0
|
|
|
|
|
0
|
my $loc = $node->location; |
628
|
0
|
0
|
0
|
|
|
0
|
say "Calling update on node '$loc'" if $loc and not $args{quiet}; |
629
|
0
|
|
|
|
|
0
|
push (@msgs, $node->update(%args)) |
630
|
|
|
|
|
|
|
} ; |
631
|
0
|
|
|
|
|
0
|
}; |
632
|
|
|
|
|
|
|
|
633
|
0
|
|
|
|
|
0
|
my $root = $self->config_root ; |
634
|
|
|
|
|
|
|
|
635
|
|
|
|
|
|
|
Config::Model::ObjTreeScanner->new( |
636
|
|
|
|
|
|
|
node_content_hook => $hook, |
637
|
|
|
|
|
|
|
check => ($args{quiet} ? 'no' : 'yes'), |
638
|
|
|
|
0
|
|
|
leaf_cb => sub { } |
639
|
0
|
0
|
|
|
|
0
|
)->scan_node( \@msgs, $root ); |
640
|
|
|
|
|
|
|
|
641
|
0
|
|
|
|
|
0
|
return @msgs; |
642
|
|
|
|
|
|
|
} |
643
|
|
|
|
|
|
|
|
644
|
|
|
|
|
|
|
sub DEMOLISH { |
645
|
121
|
|
|
121
|
1
|
3356003
|
my $self = shift; |
646
|
121
|
|
|
|
|
651
|
$self->clear_write_back; # avoid reference loops |
647
|
|
|
|
|
|
|
} |
648
|
|
|
|
|
|
|
|
649
|
|
|
|
|
|
|
__PACKAGE__->meta->make_immutable; |
650
|
|
|
|
|
|
|
|
651
|
|
|
|
|
|
|
1; |
652
|
|
|
|
|
|
|
|
653
|
|
|
|
|
|
|
# ABSTRACT: Instance of configuration tree |
654
|
|
|
|
|
|
|
|
655
|
|
|
|
|
|
|
__END__ |
656
|
|
|
|
|
|
|
|
657
|
|
|
|
|
|
|
=pod |
658
|
|
|
|
|
|
|
|
659
|
|
|
|
|
|
|
=encoding UTF-8 |
660
|
|
|
|
|
|
|
|
661
|
|
|
|
|
|
|
=head1 NAME |
662
|
|
|
|
|
|
|
|
663
|
|
|
|
|
|
|
Config::Model::Instance - Instance of configuration tree |
664
|
|
|
|
|
|
|
|
665
|
|
|
|
|
|
|
=head1 VERSION |
666
|
|
|
|
|
|
|
|
667
|
|
|
|
|
|
|
version 2.153 |
668
|
|
|
|
|
|
|
|
669
|
|
|
|
|
|
|
=head1 SYNOPSIS |
670
|
|
|
|
|
|
|
|
671
|
|
|
|
|
|
|
use Config::Model; |
672
|
|
|
|
|
|
|
use File::Path ; |
673
|
|
|
|
|
|
|
|
674
|
|
|
|
|
|
|
# setup a dummy popcon conf file |
675
|
|
|
|
|
|
|
my $wr_dir = '/tmp/etc/'; |
676
|
|
|
|
|
|
|
my $conf_file = "$wr_dir/popularity-contest.conf" ; |
677
|
|
|
|
|
|
|
|
678
|
|
|
|
|
|
|
unless (-d $wr_dir) { |
679
|
|
|
|
|
|
|
mkpath($wr_dir, { mode => 0755 }) |
680
|
|
|
|
|
|
|
|| die "can't mkpath $wr_dir: $!"; |
681
|
|
|
|
|
|
|
} |
682
|
|
|
|
|
|
|
open(my $conf,"> $conf_file" ) || die "can't open $conf_file: $!"; |
683
|
|
|
|
|
|
|
$conf->print( qq!MY_HOSTID="aaaaaaaaaaaaaaaaaaaa"\n!, |
684
|
|
|
|
|
|
|
qq!PARTICIPATE="yes"\n!, |
685
|
|
|
|
|
|
|
qq!USEHTTP="yes" # always http\n!, |
686
|
|
|
|
|
|
|
qq!DAY="6"\n!); |
687
|
|
|
|
|
|
|
$conf->close ; |
688
|
|
|
|
|
|
|
|
689
|
|
|
|
|
|
|
my $model = Config::Model->new; |
690
|
|
|
|
|
|
|
|
691
|
|
|
|
|
|
|
# PopCon model is provided. Create a new Config::Model::Instance object |
692
|
|
|
|
|
|
|
my $inst = $model->instance (root_class_name => 'PopCon', |
693
|
|
|
|
|
|
|
root_dir => '/tmp', |
694
|
|
|
|
|
|
|
); |
695
|
|
|
|
|
|
|
my $root = $inst -> config_root ; |
696
|
|
|
|
|
|
|
|
697
|
|
|
|
|
|
|
print $root->describe; |
698
|
|
|
|
|
|
|
|
699
|
|
|
|
|
|
|
=head1 DESCRIPTION |
700
|
|
|
|
|
|
|
|
701
|
|
|
|
|
|
|
This module provides an object that holds a configuration tree. |
702
|
|
|
|
|
|
|
|
703
|
|
|
|
|
|
|
=head1 CONSTRUCTOR |
704
|
|
|
|
|
|
|
|
705
|
|
|
|
|
|
|
An instance object is created by calling L<instance |
706
|
|
|
|
|
|
|
method|Config::Model/"Configuration instance"> on an existing |
707
|
|
|
|
|
|
|
model. This model can be specified by its application name: |
708
|
|
|
|
|
|
|
|
709
|
|
|
|
|
|
|
my $inst = $model->instance ( |
710
|
|
|
|
|
|
|
# run 'cme list' to get list of applications |
711
|
|
|
|
|
|
|
application => 'foo', |
712
|
|
|
|
|
|
|
# optional |
713
|
|
|
|
|
|
|
instance_name => 'test1' |
714
|
|
|
|
|
|
|
); |
715
|
|
|
|
|
|
|
|
716
|
|
|
|
|
|
|
my $inst = $model->instance ( |
717
|
|
|
|
|
|
|
root_class_name => 'SomeRootClass', |
718
|
|
|
|
|
|
|
instance_name => 'test1' |
719
|
|
|
|
|
|
|
); |
720
|
|
|
|
|
|
|
|
721
|
|
|
|
|
|
|
The directory (or directories) holding configuration files is |
722
|
|
|
|
|
|
|
specified within the configuration model. For test purpose you can |
723
|
|
|
|
|
|
|
change the "root" directory with C<root_dir> parameter. |
724
|
|
|
|
|
|
|
|
725
|
|
|
|
|
|
|
Constructor parameters are: |
726
|
|
|
|
|
|
|
|
727
|
|
|
|
|
|
|
=over |
728
|
|
|
|
|
|
|
|
729
|
|
|
|
|
|
|
=item root_dir |
730
|
|
|
|
|
|
|
|
731
|
|
|
|
|
|
|
Pseudo root directory where to read I<and> write configuration |
732
|
|
|
|
|
|
|
files (L<Path::Tiny> object or string). Configuration directory |
733
|
|
|
|
|
|
|
specified in model or with C<config_dir> option is appended to this |
734
|
|
|
|
|
|
|
root directory |
735
|
|
|
|
|
|
|
|
736
|
|
|
|
|
|
|
=item root_path |
737
|
|
|
|
|
|
|
|
738
|
|
|
|
|
|
|
L<Path::Tiny> object created with C<root_dir> value or with current |
739
|
|
|
|
|
|
|
directory if C<root_dir> is empty. |
740
|
|
|
|
|
|
|
|
741
|
|
|
|
|
|
|
=item config_dir |
742
|
|
|
|
|
|
|
|
743
|
|
|
|
|
|
|
Directory to read or write configuration file. This parameter must be |
744
|
|
|
|
|
|
|
supplied if not provided by the configuration model. (string) |
745
|
|
|
|
|
|
|
|
746
|
|
|
|
|
|
|
=item backend_arg |
747
|
|
|
|
|
|
|
|
748
|
|
|
|
|
|
|
Specify a backend argument that may be retrieved by some |
749
|
|
|
|
|
|
|
backend. Instance is used as a relay and does not use this data. |
750
|
|
|
|
|
|
|
|
751
|
|
|
|
|
|
|
=item check |
752
|
|
|
|
|
|
|
|
753
|
|
|
|
|
|
|
Specify whether to check value while reading config files. Either: |
754
|
|
|
|
|
|
|
|
755
|
|
|
|
|
|
|
=over |
756
|
|
|
|
|
|
|
|
757
|
|
|
|
|
|
|
=item yes |
758
|
|
|
|
|
|
|
|
759
|
|
|
|
|
|
|
Check value and throws an error for bad values. |
760
|
|
|
|
|
|
|
|
761
|
|
|
|
|
|
|
=item skip |
762
|
|
|
|
|
|
|
|
763
|
|
|
|
|
|
|
Check value and skip bad value. |
764
|
|
|
|
|
|
|
|
765
|
|
|
|
|
|
|
=item no |
766
|
|
|
|
|
|
|
|
767
|
|
|
|
|
|
|
Do not check. |
768
|
|
|
|
|
|
|
|
769
|
|
|
|
|
|
|
=back |
770
|
|
|
|
|
|
|
|
771
|
|
|
|
|
|
|
=item canonical |
772
|
|
|
|
|
|
|
|
773
|
|
|
|
|
|
|
When true: write config data back using model order. By default, write |
774
|
|
|
|
|
|
|
items back using the order found in the configuration file. This |
775
|
|
|
|
|
|
|
feature is experimental and not supported by all backends. |
776
|
|
|
|
|
|
|
|
777
|
|
|
|
|
|
|
=item on_change_cb |
778
|
|
|
|
|
|
|
|
779
|
|
|
|
|
|
|
Call back this function whenever C<notify_change> is called. Called with |
780
|
|
|
|
|
|
|
arguments: C<< name => <root node element name>, index => <index_value> >> |
781
|
|
|
|
|
|
|
|
782
|
|
|
|
|
|
|
=item on_message_cb |
783
|
|
|
|
|
|
|
|
784
|
|
|
|
|
|
|
Call back this function when L<show_message> is called. By default, |
785
|
|
|
|
|
|
|
messages are displayed on STDOUT. |
786
|
|
|
|
|
|
|
|
787
|
|
|
|
|
|
|
=item error_paths |
788
|
|
|
|
|
|
|
|
789
|
|
|
|
|
|
|
Returns a list of tree items that currently have an error. |
790
|
|
|
|
|
|
|
|
791
|
|
|
|
|
|
|
=item error_messages |
792
|
|
|
|
|
|
|
|
793
|
|
|
|
|
|
|
Returns a list of error messages from the tree content. |
794
|
|
|
|
|
|
|
|
795
|
|
|
|
|
|
|
=back |
796
|
|
|
|
|
|
|
|
797
|
|
|
|
|
|
|
Note that the root directory specified within the configuration model |
798
|
|
|
|
|
|
|
is overridden by C<root_dir> parameter. |
799
|
|
|
|
|
|
|
|
800
|
|
|
|
|
|
|
If you need to load configuration data that are not correct, you can |
801
|
|
|
|
|
|
|
use C<< force_load => 1 >>. Then, wrong data are discarded (equivalent to |
802
|
|
|
|
|
|
|
C<< check => 'no' >> ). |
803
|
|
|
|
|
|
|
|
804
|
|
|
|
|
|
|
=head1 METHODS |
805
|
|
|
|
|
|
|
|
806
|
|
|
|
|
|
|
=head2 Manage configuration data |
807
|
|
|
|
|
|
|
|
808
|
|
|
|
|
|
|
=head2 modify |
809
|
|
|
|
|
|
|
|
810
|
|
|
|
|
|
|
Calls L</"load"> and then L</save>. |
811
|
|
|
|
|
|
|
|
812
|
|
|
|
|
|
|
Takes the same parameter as C<load> plus: |
813
|
|
|
|
|
|
|
|
814
|
|
|
|
|
|
|
=over |
815
|
|
|
|
|
|
|
|
816
|
|
|
|
|
|
|
=item C<force_write> |
817
|
|
|
|
|
|
|
|
818
|
|
|
|
|
|
|
Force saving configuration file even if no value was modified |
819
|
|
|
|
|
|
|
(default is 0) |
820
|
|
|
|
|
|
|
|
821
|
|
|
|
|
|
|
=item C<quiet> |
822
|
|
|
|
|
|
|
|
823
|
|
|
|
|
|
|
Do no display the changes brought by the modification steps |
824
|
|
|
|
|
|
|
|
825
|
|
|
|
|
|
|
=back |
826
|
|
|
|
|
|
|
|
827
|
|
|
|
|
|
|
=head2 load |
828
|
|
|
|
|
|
|
|
829
|
|
|
|
|
|
|
Load configuration tree with configuration data. See |
830
|
|
|
|
|
|
|
L<Config::Model::Loader/"load"> for parameters. |
831
|
|
|
|
|
|
|
Returns <$self>. |
832
|
|
|
|
|
|
|
|
833
|
|
|
|
|
|
|
=head2 save |
834
|
|
|
|
|
|
|
|
835
|
|
|
|
|
|
|
Save the content of the configuration tree to |
836
|
|
|
|
|
|
|
configuration files. (See L</write_back> for more details) |
837
|
|
|
|
|
|
|
|
838
|
|
|
|
|
|
|
Use C<< force => 1 >> option to force saving configuration data. |
839
|
|
|
|
|
|
|
|
840
|
|
|
|
|
|
|
=head2 config_root |
841
|
|
|
|
|
|
|
|
842
|
|
|
|
|
|
|
Returns the L<root object|Config::Model::Node> of the configuration tree. |
843
|
|
|
|
|
|
|
|
844
|
|
|
|
|
|
|
=head2 apply_fixes |
845
|
|
|
|
|
|
|
|
846
|
|
|
|
|
|
|
Scan the tree and apply fixes that are attached to warning specifications. |
847
|
|
|
|
|
|
|
See C<warn_if_match> or C<warn_unless_match> in L<Config::Model::Value/>. |
848
|
|
|
|
|
|
|
|
849
|
|
|
|
|
|
|
=head2 deep_check |
850
|
|
|
|
|
|
|
|
851
|
|
|
|
|
|
|
Scan the tree and deep check on all elements that support this. Currently only hash or |
852
|
|
|
|
|
|
|
list element have this feature. |
853
|
|
|
|
|
|
|
|
854
|
|
|
|
|
|
|
=head2 needs_save |
855
|
|
|
|
|
|
|
|
856
|
|
|
|
|
|
|
Returns 1 (or more) if the instance contains data that needs to be |
857
|
|
|
|
|
|
|
saved. I.e some change were done in the tree that needs to be saved. |
858
|
|
|
|
|
|
|
|
859
|
|
|
|
|
|
|
=head2 has_changes |
860
|
|
|
|
|
|
|
|
861
|
|
|
|
|
|
|
Returns true if the instance contains unsasved changes. |
862
|
|
|
|
|
|
|
|
863
|
|
|
|
|
|
|
=head2 list_changes |
864
|
|
|
|
|
|
|
|
865
|
|
|
|
|
|
|
In list context, returns a array ref of strings describing the changes. |
866
|
|
|
|
|
|
|
In scalar context, returns a big string. Useful to print. |
867
|
|
|
|
|
|
|
|
868
|
|
|
|
|
|
|
=head2 say_changes |
869
|
|
|
|
|
|
|
|
870
|
|
|
|
|
|
|
Print all changes on STDOUT and return C<$self>. |
871
|
|
|
|
|
|
|
|
872
|
|
|
|
|
|
|
=head2 clear_changes |
873
|
|
|
|
|
|
|
|
874
|
|
|
|
|
|
|
Clear list of changes. Note that changes pending in the configuration |
875
|
|
|
|
|
|
|
tree is not affected. This clears only the list shown to user. Use |
876
|
|
|
|
|
|
|
only for tests. |
877
|
|
|
|
|
|
|
|
878
|
|
|
|
|
|
|
=head2 has_warning |
879
|
|
|
|
|
|
|
|
880
|
|
|
|
|
|
|
Returns the number of warning found in the elements of this configuration instance. |
881
|
|
|
|
|
|
|
|
882
|
|
|
|
|
|
|
=head2 update |
883
|
|
|
|
|
|
|
|
884
|
|
|
|
|
|
|
Parameters: C<< ( quiet => (0|1), %args ) >> |
885
|
|
|
|
|
|
|
|
886
|
|
|
|
|
|
|
Try to run update command on all nodes of the configuration tree. Node |
887
|
|
|
|
|
|
|
without C<update> method are ignored. C<update> prints a message |
888
|
|
|
|
|
|
|
otherwise (unless C<quiet> is true). |
889
|
|
|
|
|
|
|
|
890
|
|
|
|
|
|
|
=head2 grab |
891
|
|
|
|
|
|
|
|
892
|
|
|
|
|
|
|
Use the steps parameter to retrieve and returns an object from the |
893
|
|
|
|
|
|
|
configuration tree. Forwarded to L<Config::Model::Role::Grab/grab> |
894
|
|
|
|
|
|
|
|
895
|
|
|
|
|
|
|
=head2 grab_value |
896
|
|
|
|
|
|
|
|
897
|
|
|
|
|
|
|
Use the steps parameter to retrieve and returns the value of a leaf |
898
|
|
|
|
|
|
|
object from the configuration tree. Forwarded to |
899
|
|
|
|
|
|
|
L<Config::Model::Role::Grab/grab_value> |
900
|
|
|
|
|
|
|
|
901
|
|
|
|
|
|
|
=head2 searcher |
902
|
|
|
|
|
|
|
|
903
|
|
|
|
|
|
|
Returns an object dedicated to search an element in the configuration |
904
|
|
|
|
|
|
|
model. |
905
|
|
|
|
|
|
|
|
906
|
|
|
|
|
|
|
This method returns a L<Config::Model::Searcher> object. See |
907
|
|
|
|
|
|
|
L<Config::Model::Searcher> for details on how to handle a search. |
908
|
|
|
|
|
|
|
|
909
|
|
|
|
|
|
|
=head2 iterator |
910
|
|
|
|
|
|
|
|
911
|
|
|
|
|
|
|
This method returns a L<Config::Model::Iterator> object. See |
912
|
|
|
|
|
|
|
L<Config::Model::Iterator> for details. |
913
|
|
|
|
|
|
|
|
914
|
|
|
|
|
|
|
Arguments are explained in L<Config::Model::Iterator> |
915
|
|
|
|
|
|
|
L<constructor arguments|Config::Model::Iterator/"Creating an iterator">. |
916
|
|
|
|
|
|
|
|
917
|
|
|
|
|
|
|
=head2 application |
918
|
|
|
|
|
|
|
|
919
|
|
|
|
|
|
|
Returns the application name of the instance. (E.g C<popcon>, C<dpkg> ...) |
920
|
|
|
|
|
|
|
|
921
|
|
|
|
|
|
|
=head2 wizard_helper |
922
|
|
|
|
|
|
|
|
923
|
|
|
|
|
|
|
Deprecated. Call L</iterator> instead. |
924
|
|
|
|
|
|
|
|
925
|
|
|
|
|
|
|
=head1 Internal methods |
926
|
|
|
|
|
|
|
|
927
|
|
|
|
|
|
|
=head2 name |
928
|
|
|
|
|
|
|
|
929
|
|
|
|
|
|
|
Returns the instance name. |
930
|
|
|
|
|
|
|
|
931
|
|
|
|
|
|
|
=head2 read_check |
932
|
|
|
|
|
|
|
|
933
|
|
|
|
|
|
|
Returns which kind of check is performed while reading configuration |
934
|
|
|
|
|
|
|
files. (see C<check> parameter in L</CONSTRUCTOR> section) |
935
|
|
|
|
|
|
|
|
936
|
|
|
|
|
|
|
=head2 show_message |
937
|
|
|
|
|
|
|
|
938
|
|
|
|
|
|
|
Parameters: C<( string )> |
939
|
|
|
|
|
|
|
|
940
|
|
|
|
|
|
|
Display the message on STDOUT unless a custom function was passed to |
941
|
|
|
|
|
|
|
C<on_message_cb> parameter. |
942
|
|
|
|
|
|
|
|
943
|
|
|
|
|
|
|
=head2 reset_config |
944
|
|
|
|
|
|
|
|
945
|
|
|
|
|
|
|
Destroy current configuration tree (with data) and returns a new tree with |
946
|
|
|
|
|
|
|
data (and annotations) loaded from disk. |
947
|
|
|
|
|
|
|
|
948
|
|
|
|
|
|
|
=head2 config_model |
949
|
|
|
|
|
|
|
|
950
|
|
|
|
|
|
|
Returns the model (L<Config::Model> object) of the configuration tree. |
951
|
|
|
|
|
|
|
|
952
|
|
|
|
|
|
|
=head2 annotation_saver |
953
|
|
|
|
|
|
|
|
954
|
|
|
|
|
|
|
Returns the object loading and saving annotations. See |
955
|
|
|
|
|
|
|
L<Config::Model::Annotation> for details. |
956
|
|
|
|
|
|
|
|
957
|
|
|
|
|
|
|
=head2 preset_start |
958
|
|
|
|
|
|
|
|
959
|
|
|
|
|
|
|
All values stored in preset mode are shown to the user as default |
960
|
|
|
|
|
|
|
values. This feature is useful to enter configuration data entered by |
961
|
|
|
|
|
|
|
an automatic process (like hardware scan) |
962
|
|
|
|
|
|
|
|
963
|
|
|
|
|
|
|
=head2 preset_stop |
964
|
|
|
|
|
|
|
|
965
|
|
|
|
|
|
|
Stop preset mode |
966
|
|
|
|
|
|
|
|
967
|
|
|
|
|
|
|
=head2 preset |
968
|
|
|
|
|
|
|
|
969
|
|
|
|
|
|
|
Get preset mode |
970
|
|
|
|
|
|
|
|
971
|
|
|
|
|
|
|
=head2 preset_clear |
972
|
|
|
|
|
|
|
|
973
|
|
|
|
|
|
|
Clear all preset values stored. |
974
|
|
|
|
|
|
|
|
975
|
|
|
|
|
|
|
=head2 layered_start |
976
|
|
|
|
|
|
|
|
977
|
|
|
|
|
|
|
All values stored in layered mode are shown to the user as default |
978
|
|
|
|
|
|
|
values. This feature is useful to enter configuration data entered by |
979
|
|
|
|
|
|
|
an automatic process (like hardware scan) |
980
|
|
|
|
|
|
|
|
981
|
|
|
|
|
|
|
=head2 layered_stop |
982
|
|
|
|
|
|
|
|
983
|
|
|
|
|
|
|
Stop layered mode |
984
|
|
|
|
|
|
|
|
985
|
|
|
|
|
|
|
=head2 layered |
986
|
|
|
|
|
|
|
|
987
|
|
|
|
|
|
|
Get layered mode |
988
|
|
|
|
|
|
|
|
989
|
|
|
|
|
|
|
=head2 layered_clear |
990
|
|
|
|
|
|
|
|
991
|
|
|
|
|
|
|
Clear all layered values stored. |
992
|
|
|
|
|
|
|
|
993
|
|
|
|
|
|
|
=head2 get_data_mode |
994
|
|
|
|
|
|
|
|
995
|
|
|
|
|
|
|
Returns 'normal' or 'preset' or 'layered'. Does not take into account |
996
|
|
|
|
|
|
|
initial_load. |
997
|
|
|
|
|
|
|
|
998
|
|
|
|
|
|
|
=head2 initial_load_start |
999
|
|
|
|
|
|
|
|
1000
|
|
|
|
|
|
|
Start initial_load mode. This mode tracks the first modifications of |
1001
|
|
|
|
|
|
|
the tree done with data read from the configuration file. |
1002
|
|
|
|
|
|
|
|
1003
|
|
|
|
|
|
|
Instance is built with initial_load as 1. Read backend clears this |
1004
|
|
|
|
|
|
|
value once the first read is done. |
1005
|
|
|
|
|
|
|
|
1006
|
|
|
|
|
|
|
Other modifications, when initial_load is zero, are assumed to be user |
1007
|
|
|
|
|
|
|
modifications. |
1008
|
|
|
|
|
|
|
|
1009
|
|
|
|
|
|
|
=head2 initial_load_stop |
1010
|
|
|
|
|
|
|
|
1011
|
|
|
|
|
|
|
Stop initial_load mode. Instance is built with initial_load as 1. Read backend |
1012
|
|
|
|
|
|
|
clears this value once the first read is done. |
1013
|
|
|
|
|
|
|
|
1014
|
|
|
|
|
|
|
=head2 initial_load |
1015
|
|
|
|
|
|
|
|
1016
|
|
|
|
|
|
|
Get initial_load mode |
1017
|
|
|
|
|
|
|
|
1018
|
|
|
|
|
|
|
=head2 data |
1019
|
|
|
|
|
|
|
|
1020
|
|
|
|
|
|
|
This method provides a way to store some arbitrary data in the |
1021
|
|
|
|
|
|
|
instance object. |
1022
|
|
|
|
|
|
|
|
1023
|
|
|
|
|
|
|
E.g: |
1024
|
|
|
|
|
|
|
|
1025
|
|
|
|
|
|
|
$instance->data(foo => 'bar'); |
1026
|
|
|
|
|
|
|
|
1027
|
|
|
|
|
|
|
Later: |
1028
|
|
|
|
|
|
|
|
1029
|
|
|
|
|
|
|
my $foo = $instance->data('foo'); # $foo contains 'bar' |
1030
|
|
|
|
|
|
|
|
1031
|
|
|
|
|
|
|
=head1 Read and write backend features |
1032
|
|
|
|
|
|
|
|
1033
|
|
|
|
|
|
|
Usually, a program based on config model must first create the |
1034
|
|
|
|
|
|
|
configuration model, then load all configuration data. |
1035
|
|
|
|
|
|
|
|
1036
|
|
|
|
|
|
|
This feature enables you to declare with the model a way to load |
1037
|
|
|
|
|
|
|
configuration data (and to write it back). See |
1038
|
|
|
|
|
|
|
L<Config::Model::BackendMgr> for details. |
1039
|
|
|
|
|
|
|
|
1040
|
|
|
|
|
|
|
=head2 backend_arg |
1041
|
|
|
|
|
|
|
|
1042
|
|
|
|
|
|
|
Get L<cme> command line argument that may be used by the backend to |
1043
|
|
|
|
|
|
|
get the configuration file. These method is typically used in the read |
1044
|
|
|
|
|
|
|
and write method of a backend to know where is the configuration file |
1045
|
|
|
|
|
|
|
to edit. |
1046
|
|
|
|
|
|
|
|
1047
|
|
|
|
|
|
|
=head2 root_dir |
1048
|
|
|
|
|
|
|
|
1049
|
|
|
|
|
|
|
Returns a L<Path::Tiny> object for the root directory where |
1050
|
|
|
|
|
|
|
configuration data is read from or written to. |
1051
|
|
|
|
|
|
|
|
1052
|
|
|
|
|
|
|
=head2 root_path |
1053
|
|
|
|
|
|
|
|
1054
|
|
|
|
|
|
|
Same as C<root_dir> |
1055
|
|
|
|
|
|
|
|
1056
|
|
|
|
|
|
|
=head2 register_write_back |
1057
|
|
|
|
|
|
|
|
1058
|
|
|
|
|
|
|
Parameters: C<( node_location )> |
1059
|
|
|
|
|
|
|
|
1060
|
|
|
|
|
|
|
Register a node path that is called back with |
1061
|
|
|
|
|
|
|
C<write_back> method. |
1062
|
|
|
|
|
|
|
|
1063
|
|
|
|
|
|
|
=head2 notify_change |
1064
|
|
|
|
|
|
|
|
1065
|
|
|
|
|
|
|
Notify that some data has changed in the tree. See |
1066
|
|
|
|
|
|
|
L<Config::Model::AnyThing/notify_change> for more details. |
1067
|
|
|
|
|
|
|
|
1068
|
|
|
|
|
|
|
=head2 write_back |
1069
|
|
|
|
|
|
|
|
1070
|
|
|
|
|
|
|
In summary, save the content of the configuration tree to |
1071
|
|
|
|
|
|
|
configuration files. |
1072
|
|
|
|
|
|
|
|
1073
|
|
|
|
|
|
|
In more details, C<write_back> tries to run all subroutines registered |
1074
|
|
|
|
|
|
|
with C<register_write_back> to write the configuration information. |
1075
|
|
|
|
|
|
|
(See L<Config::Model::BackendMgr> for details). |
1076
|
|
|
|
|
|
|
|
1077
|
|
|
|
|
|
|
You can specify here another config directory to write configuration |
1078
|
|
|
|
|
|
|
data back with C<config_dir> parameter. This overrides the model |
1079
|
|
|
|
|
|
|
specifications. |
1080
|
|
|
|
|
|
|
|
1081
|
|
|
|
|
|
|
C<write_back> croaks if no write call-back are known. |
1082
|
|
|
|
|
|
|
|
1083
|
|
|
|
|
|
|
Use C<< force => 1 >> option to force saving configuration data. This |
1084
|
|
|
|
|
|
|
is useful to write back a file even no change are done at semantic |
1085
|
|
|
|
|
|
|
level, i.e. to reformat a file or remove unnecessary data. |
1086
|
|
|
|
|
|
|
|
1087
|
|
|
|
|
|
|
=head1 AUTHOR |
1088
|
|
|
|
|
|
|
|
1089
|
|
|
|
|
|
|
Dominique Dumont, (ddumont at cpan dot org) |
1090
|
|
|
|
|
|
|
|
1091
|
|
|
|
|
|
|
=head1 SEE ALSO |
1092
|
|
|
|
|
|
|
|
1093
|
|
|
|
|
|
|
L<Config::Model>, |
1094
|
|
|
|
|
|
|
L<Config::Model::Node>, |
1095
|
|
|
|
|
|
|
L<Config::Model::Loader>, |
1096
|
|
|
|
|
|
|
L<Config::Model::Searcher>, |
1097
|
|
|
|
|
|
|
L<Config::Model::Value>, |
1098
|
|
|
|
|
|
|
|
1099
|
|
|
|
|
|
|
=head1 AUTHOR |
1100
|
|
|
|
|
|
|
|
1101
|
|
|
|
|
|
|
Dominique Dumont |
1102
|
|
|
|
|
|
|
|
1103
|
|
|
|
|
|
|
=head1 COPYRIGHT AND LICENSE |
1104
|
|
|
|
|
|
|
|
1105
|
|
|
|
|
|
|
This software is Copyright (c) 2005-2022 by Dominique Dumont. |
1106
|
|
|
|
|
|
|
|
1107
|
|
|
|
|
|
|
This is free software, licensed under: |
1108
|
|
|
|
|
|
|
|
1109
|
|
|
|
|
|
|
The GNU Lesser General Public License, Version 2.1, February 1999 |
1110
|
|
|
|
|
|
|
|
1111
|
|
|
|
|
|
|
=cut |