/usr/share/perl5/Config/Onion.pm is in libconfig-onion-perl 1.007-1.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
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 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 | package Config::Onion;
use strict;
use warnings;
our $VERSION = 1.007;
use Config::Any;
use Hash::Merge::Simple 'merge';
use Moo;
has cfg => ( is => 'lazy', clearer => '_reset_cfg' );
has prefix_key => ( is => 'rw' );
sub get { goto &cfg }
has [qw( default main local override )]
=> ( is => 'rwp', default => sub { {} } );
sub set_default {
my $self = shift;
$self = $self->new unless ref $self;
my $default = $self->default;
$default = merge $default, shift while ref $_[0] eq 'HASH';
$default = merge $default, { @_ } if @_;
$self->_set_default($default);
$self->_reset_cfg;
return $self;
}
sub set_override {
my $self = shift;
$self = $self->new unless ref $self;
my $override = $self->override;
$override = merge $override, shift while ref $_[0] eq 'HASH';
$override = merge $override, { @_ } if @_;
$self->_set_override($override);
$self->_reset_cfg;
return $self;
}
sub load {
my $self = shift;
$self = $self->new unless ref $self;
my $ca_opts = $self->_ca_opts;
# user passed in a hash ref as the last argument.
if (ref $_[$#_] eq "HASH") {
# allow user to override the default use_ext => 1
delete $ca_opts->{use_ext} unless exists $_[$#_]{use_ext};
# merge in any other options passed in
$ca_opts = merge $ca_opts, pop @_;
# ensure that flatten_to_hash feature that Config::Any supports is turned
# off
delete $ca_opts->{flatten_to_hash} if exists $ca_opts->{flatten_to_hash};
}
my $main = Config::Any->load_stems({ stems => \@_ , %$ca_opts });
my $local = Config::Any->load_stems({ stems => [ map { "$_.local" } @_ ],
%$ca_opts });
$self->_add_loaded($main, $local);
return $self;
}
sub load_glob {
my $self = shift;
$self = $self->new unless ref $self;
my $ca_opts = $self->_ca_opts;
# user passed in a hash ref as the last argument.
if (ref $_[$#_] eq "HASH") {
# allow user to override the default use_ext => 1
delete $ca_opts->{use_ext} unless exists $_[$#_]{use_ext};
# merge in any other options passed in
$ca_opts = merge $ca_opts, pop @_;
# ensure that flatten_to_hash feature that Config::Any supports is turned
# off
delete $ca_opts->{flatten_to_hash} if exists $ca_opts->{flatten_to_hash};
}
# if use_ext is on, we need to query Config::Any to see what extensions are
# allowed
my $ext_re = '';
if ($ca_opts->{use_ext}) {
my @exts = Config::Any->extensions();
$ext_re = '\.' . (shift @exts) . '$';
$ext_re .= "|\\.$_\$" foreach @exts;
}
my (@main_files, @local_files);
for my $globspec (@_) {
for (glob $globspec) {
next if $ca_opts->{use_ext} && !/$ext_re/;
if (/\.local\./) { push @local_files, $_ }
else { push @main_files, $_ }
}
}
my $main = Config::Any->load_files({ files => \@main_files, %$ca_opts });
my $local = Config::Any->load_files({ files => \@local_files, %$ca_opts });
$self->_add_loaded($main, $local);
return $self;
}
sub _add_loaded {
my $self = shift;
my ($main, $local) = @_;
my @main; @main = map { values %$_ } @$main if @$main;
my @local; @local = map { values %$_ } @$local if @$local;
if ($self->prefix_key) {
for my $cfg (@main, @local) {
$self->_replace_prefix_key($cfg) if exists $cfg->{$self->prefix_key};
}
}
$self->_set_main( merge $self->main, @main);
$self->_set_local(merge $self->local, @local);
$self->_reset_cfg;
}
sub _build_cfg {
my $self = shift;
my $cfg = merge $self->default, $self->main, $self->local, $self->override;
# remove any hash keys with a (merged) value of '!DELETE!'
my @nodes = $cfg;
while (my $curr = shift @nodes) {
for (keys %$curr) {
next unless defined $curr->{$_};
if ($curr->{$_} eq '!DELETE!') {
delete $curr->{$_};
} elsif (ref $curr->{$_} eq 'HASH') {
push @nodes, $curr->{$_};
}
}
}
return $cfg;
}
sub _ca_opts { { use_ext => 1 } }
sub _replace_prefix_key {
my $self = shift;
my $cfg = shift;
my $top_key;
my $root = $cfg->{$self->prefix_key};
while (1) {
die "Config::Onion prefix key structure may not branch" if keys %$root > 1;
$top_key ||= (keys %$root)[0];
my $child = (values %$root)[0];
unless ($child) {
my $key = (keys %$root)[0];
$root = $root->{$key} = {};
last;
}
$root = $child;
}
my $new = $cfg->{$self->prefix_key}{$top_key};
delete $cfg->{$self->prefix_key};
for (keys %$cfg) {
$root->{$_} = $cfg->{$_};
delete $cfg->{$_};
}
$cfg->{$top_key} = $new;
}
1;
=pod
=encoding UTF-8
=head1 NAME
Config::Onion - Layered configuration, because configs are like ogres
=head1 VERSION
version 1.007
=head1 SYNOPSIS
my $cfg = Config::Onion->new;
my $cfg = Config::Onion->set_default(db => {name => 'foo', password => 'bar'});
my $cfg = Config::Onion->load('/etc/myapp', './myapp');
my $cfg = Config::Onion->load('/etc/myapp', './myapp', {use_ext => 1, filter => \&filter});
my $cfg = Config::Onion->load_glob('./plugins/*');
my $cfg = Config::Onion->load_glob('./plugins/*', {force_plugins => ['Config::Any::YAML']});
$cfg->set_default(font => 'Comic Sans');
$cfg->load('config');
$cfg->load_glob('conf.d/myapp*');
$cfg->set_override(font => 'Arial');
my $dbname = $cfg->get->{db}{name};
my $plain_hashref_conf = $cfg->get;
my $dbpassword = $plain_hashref_conf->{db}{password};
=head1 DESCRIPTION
All too often, configuration is not a universal or one-time thing, yet most
configuration-handling treats it as such. Perhaps you can only load one config
file. If you can load more than one, you often have to load all of them at the
same time or each is stored completely independently, preventing one from being
able to override another. Config::Onion changes that.
Config::Onion stores all configuration settings in four layers: Defaults,
Main, Local, and Override. Each layer can be added to as many times as you
like. Within each layer, settings which are given multiple times will take the
last specified value, while those which are not repeated will remain untouched.
$cfg->set_default(name => 'Arthur Dent', location => 'Earth');
$cfg->set_default(location => 'Magrathea');
# In the Default layer, 'name' is still 'Arthur Dent', but 'location' has
# been changed to 'Magrathea'.
Regardless of the order in which they are set, values in Main will always
override values in the Default layer, the Local layer always overrides both
Default and Main, and the Override layer overrides all the others.
The design intent for each layer is:
=over 4
=item * Default
Hardcoded default values to be used when no further configuration is present
=item * Main
Values loaded from standard configuration files shipped with the application
=item * Local
Values loaded from local configuration files which are kept separate to prevent
them from being overwritten by application upgrades, etc.
=item * Override
Settings provided at run-time which take precendence over all configuration
files, such as settings provided via command line switches
=back
If a higher-priority layer wishes to completely remove a hash entry made by a
lower-priority layer (i.e., delete the hash key, not just set it to an empty
value), it can do so by setting the value to "!DELETE!". This only applies to
hash entries, not array values, as the entire array already needs to be
overwritten to make any changes to it. Also, if, for some reason, the
configuration contains objects, the contents of those objects will be ignored
for the sake of encapsulation. Only unblessed hashes are cleaned in this
manner.
=head1 METHODS
=head2 new
Returns a new, empty configuration object.
=head2 load(@file_stems)
=head2 load(@file\_stems, {...})
Loads files matching the given stems using C<< Config::Any->load_stems >> into
the Main layer. Also concatenates ".local" to each stem and loads matching
files into the Local layer. e.g., C<< $cfg->load('myapp') >> would load
C<myapp.yml> into Main and C<myapp.local.js> into Local. All filename
extensions supported by C<Config::Any> are recognized along with their
corresponding formats.
An optional hash ref final argument can be provided to override the default
option C<< use_ext => 1 >> passed to C<Config::Any>. All options supported by C<Config::Any>
are supported except flatten_to_hash. See C<< Config::Any->load_files >>
documentation for available options.
=head2 load_glob(@globs)
=head2 load_glob(@globs, {...})
Uses the Perl C<glob> function to expand each parameter into a list of
filenames and loads each file using C<Config::Any>. Files whose names contain
the string ".local." are loaded into the Local layer. All other files are
loaded into the Main layer.
An optional hash ref final argument can be provided to override the default
option C<< use_ext => 1 >> passed to C<Config::Any>. All options supported by C<Config::Any>
are supported except flatten_to_hash. See C<< Config::Any->load_files >>
documentation for available options.
=head2 set_default([\%settings,...,] %settings)
=head2 set_override([\%settings,...,] %settings)
Imports C<%settings> into the Default or Override layer. Accepts settings both
as a plain hash and as hash references, but, if the two are mixed, all hash
references must appear at the beginning of the parameter list, before any
non-hashref settings.
=head1 PROPERTIES
=head2 cfg
=head2 get
Returns the complete configuration as a hash reference.
=head2 default
=head2 main
=head2 local
=head2 override
These properties each return a single layer of the configuration. This is
not likely to be useful other than for debugging. For most other purposes,
you probably want to use C<get> instead.
=head2 prefix_key
If set, enables the Prefix Structures functionality described below when using
the C<load> or C<load_glob> methods. The value of C<prefix_key> specifies the
name of the key under which the prefix structure may be found.
Default value is C<undef>.
=head1 Prefix Structures
If you find that your configuration structure is becoming unwieldy due to
deeply-nested structures, you can define a file-specific "prefix structure"
and all other settings within that file will be loaded as children of the
prefix structure. For example, if your main program uses
$cfg = Config::Onion->new(prefix_key => '_prefix');
$cfg->load("myapp/config");
and C<myapp/config.yml> contains
_prefix:
foo:
bar:
baz: 1
then C<$cfg> will contain the configuration
foo:
bar:
baz: 1
Note that the top-level C<prefix_key> is removed.
There are some limitations on the prefix structure, in order to keep it sane
and deterministic. First, the prefix structure may only contain hashes.
Second, each hash must contain exactly one key. Finally, the value associated
with the final key must be left undefined.
=head1 BUGS AND LIMITATIONS
No bugs have been reported.
Please report any bugs or feature requests at
L<https://github.com/dsheroh/Config-Onion/issues>
=head1 AUTHOR
Dave Sherohman <dsheroh@cpan.org>
=head1 COPYRIGHT AND LICENSE
This software is copyright (c) 2012 by Lund University Library.
This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.
=cut
__END__
# ABSTRACT: Layered configuration, because configs are like ogres
|