The Perl Toolchain Summit 2025 Needs You: You can help 🙏 Learn more

use 5.010;
use strict;
use Carp;
use feature ":5.10";
use Config;
use File::Spec::Functions qw(catdir rel2abs splitpath);
our $VERSION = '0.19';
use constant DFLT_TOCOPY_SECTION => "__TOCOPY__";
use constant FLD_KEY_PREFIX => __PACKAGE__ . ' __ ';
use constant {EXPANDED => FLD_KEY_PREFIX . 'EXPANDED',
CMNT_VL => FLD_KEY_PREFIX . 'CMNT_VL',
TOCOPY_SECTION => FLD_KEY_PREFIX . 'TOCOPY_SECTION',
CURR_TOCP_SECTION => FLD_KEY_PREFIX . 'CURR_TOCP_SECTION',
TOCOPY_VARS => FLD_KEY_PREFIX . 'TOCOPY_VARS',
NOT_TOCOPY => FLD_KEY_PREFIX . 'NOT_TOCOPY',
SECTIONS => FLD_KEY_PREFIX . 'SECTIONS',
SECTIONS_H => FLD_KEY_PREFIX . 'SECTIONS_H',
SRC_NAME => FLD_KEY_PREFIX . 'SRC_NAME',
VARIABLES => FLD_KEY_PREFIX . 'VARIABLES',
GLOBAL_VARS => FLD_KEY_PREFIX . 'GLOBAL_VARS',
GLOBAL_MODE => FLD_KEY_PREFIX . 'GLOBAL_MODE',
VREF_RE => FLD_KEY_PREFIX . 'VREF_RE',
SEPARATOR => FLD_KEY_PREFIX . 'SEPARATOR',
BACKUP => FLD_KEY_PREFIX . 'BACKUP',
VARNAME_CHK_RE => FLD_KEY_PREFIX . 'VARNAME_CHK_RE',
};
my %Globals = ('=:' => catdir("", "",),
'=::' => $Config{path_sep},
'=VERSION' => $VERSION);
# Match punctuation chars, but not the underscores.
my $Modifier_Char = '[^_[:^punct:]]';
my ($_look_up, $_x_var_name, $_expand_vars);
my $_check_tocopy_vars = sub {
my ($self, $tocopy_vars, $set) = @_;
croak("'tocopy_vars': expected HASH ref") if ref($tocopy_vars) ne 'HASH';
$tocopy_vars = { %$tocopy_vars };
while (my ($var, $value) = each(%$tocopy_vars)) {
croak("'tocopy_vars': value of '$var' is a ref, expected scalar") if ref($value);
if (!defined($value)) {
carp("'tocopy_vars': value '$var' is undef - treated as empty string");
$tocopy_vars->{$var} = "";
}
croak("'tocopy_vars': variable '$var': name is not permitted")
if ($var =~ /^\s*$/ || $var =~ /^[[=;]/);
}
# @{$self->{+TOCOPY_VARS}}{keys(%$tocopy_vars)} = values(%$tocopy_vars) if $set;
$self->{+TOCOPY_VARS} = {%$tocopy_vars} if $set;
return $tocopy_vars;
};
my $_check_not_tocopy = sub {
my ($self, $not_tocopy, $set) = @_;
my $ref = ref($not_tocopy);
if ($ref eq 'ARRAY') {
foreach my $v (@$not_tocopy) {
croak("'not_tocopy': undefined value in array") if !defined($v);
croak("'not_tocopy': unexpected ref value in array") if ref($v);
}
$not_tocopy = {map {$_ => undef} @$not_tocopy};
}
elsif ($ref eq 'HASH') {
$not_tocopy = %{$not_tocopy};
}
else {
croak("'not_tocopy': unexpected type: must be ARRAY or HASH ref");
}
$self->{+NOT_TOCOPY}= $not_tocopy if $set;
return $not_tocopy;
};
sub new {
my ($class, %args) = @_;
state $allowed_keys = {map {$_ => undef} qw(tocopy_section tocopy_vars not_tocopy global_mode
separator cmnt_vl varname_chk_re)};
_check_args(\%args, $allowed_keys);
my $self = {};
croak("'tocopy_section': must not be a reference") if ref($args{tocopy_section});
if (exists($args{separator})) {
state $allowed_sep_chars = "#!%&',./:~\\";
my $sep = $args{separator};
croak("'separator': unexpected ref type, must be a scalar") if ref($sep);
croak("'separator': invalid value. Allowed chars: $allowed_sep_chars")
if $sep !~ m{^[\Q$allowed_sep_chars\E]+$};
$self->{+SEPARATOR} = $sep;
$self->{+VREF_RE} = qr/^(.*?)(?:\Q$sep\E)(.*)$/;
}
else {
$self->{+VREF_RE} = qr/^\[\s*(.*?)\s*\](.*)$/;
}
$self->{+CMNT_VL} = $args{cmnt_vl};
$self->{+TOCOPY_SECTION} = $args{tocopy_section} // DFLT_TOCOPY_SECTION;
$self->$_check_tocopy_vars($args{tocopy_vars}, 1) if exists($args{tocopy_vars});
$self->$_check_not_tocopy($args{not_tocopy}, 1) if exists($args{not_tocopy});
$self->{+GLOBAL_MODE} = !!$args{global_mode};
if (exists($args{varname_chk_re})) {
croak("'varname_chk_re': must be a compiled regex") if ref($args{varname_chk_re}) ne 'Regexp';
$self->{+VARNAME_CHK_RE} = $args{varname_chk_re};
}
return bless($self, $class);
}
my $_expand_value = sub {
return $_[0]->$_expand_vars($_[1], undef, $_[2]);
};
#
# We assume that this is called when the target section is still empty and if
# tocopy vars exist.
#
my $_cp_tocopy_vars = sub {
my ($self, $to_sect_name) = @_;
my $comm_sec = $self->{+VARIABLES}{$self->{+TOCOPY_SECTION}} // die("no tocopy vars");
my $not_tocopy = $self->{+NOT_TOCOPY};
my $to_sec = $self->{+VARIABLES}{$to_sect_name} //= {};
my $expanded = $self->{+EXPANDED};
foreach my $comm_var (keys(%$comm_sec)) {
next if exists($not_tocopy->{$comm_var});
$to_sec->{$comm_var} = $comm_sec->{$comm_var};
my $comm_x_var_name = "[$comm_sec]$comm_var"; # see _x_var_name()
$expanded->{"[$to_sect_name]$comm_var"} = undef if exists($expanded->{$comm_x_var_name});
}
};
my $_parse_ini = sub {
my ($self, $src) = @_;
my $src_name;
if (ref($src)) {
croak("Internal error: argument is not an ARRAY ref") if ref($src) ne 'ARRAY';
$src_name = $self->{+SRC_NAME};
}
else {
$src_name = $src;
$src = [do { local (*ARGV); @ARGV = ($src_name); <> }];
}
my $curr_section;
my $cmnt_vl = $self->{+CMNT_VL};
my $sections = $self->{+SECTIONS};
my $sections_h = $self->{+SECTIONS_H};
my $expanded = $self->{+EXPANDED};
my $variables = $self->{+VARIABLES};
my $tocopy_sec = $self->{+TOCOPY_SECTION};
my $tocopy_vars = $variables->{$tocopy_sec}; # hash key need not to exist!
my $global_mode = $self->{+GLOBAL_MODE};
my $vnm_chk_re = $self->{+VARNAME_CHK_RE};
my $tocopy_sec_declared;
my $i; # index in for() loop
my $_fatal = sub { croak("'$src_name': ", $_[0], " at line ", $i + 1); };
my $set_curr_section = sub {
$curr_section = shift;
if ($curr_section eq $tocopy_sec) {
$_fatal->("tocopy section '$tocopy_sec' must be first section") if @$sections;
$tocopy_vars = $variables->{$tocopy_sec} = {} if !$tocopy_vars;
$tocopy_sec_declared = 1;
}
elsif ($tocopy_vars && !$global_mode) {
$self->$_cp_tocopy_vars($curr_section);
}
else {
$variables->{$curr_section} = {};
}
$_fatal->("'$curr_section': duplicate header") if exists($sections_h->{$curr_section});
$sections_h->{$curr_section} = @$sections; # Index!
push(@$sections, $curr_section);
};
for ($i = 0; $i < @$src; ++$i) {
my $line = $src->[$i];
if (index($line, ";!") == 0 || index($line, "=") == 0) {
$_fatal->("directives are not yet supported");
}
$line =~ s/^\s+//;
next if $line eq "" || $line =~ /^[;#]/;
$line =~ s/\s+$//;
# section header
if (index($line, "[") == 0) {
$line =~ s/\s*[#;][^\]]*$//;
$line =~ /^\[\s*(.*?)\s*\]$/ or $_fatal->("invalid section header");
$set_curr_section->($1);
next;
}
# var = val
$line =~ s/\s+;.*$// if $cmnt_vl;
$set_curr_section->($tocopy_sec) if !defined($curr_section);
$line =~ /^(.*?)\s*($Modifier_Char*?)=(?:\s*)(.*)/ or
$_fatal->("neither section header nor key definition");
my ($var_name, $modifier, $value) = ($1, $2, $3);
if ($vnm_chk_re) {
croak("'$var_name': var name does not match varname_chk_re") if $var_name !~ $vnm_chk_re;
}
my $x_var_name = $self->$_x_var_name($curr_section, $var_name);
my $exp_flag = exists($expanded->{$x_var_name});
$_fatal->("empty variable name") if $var_name eq "";
my $sect_vars = $variables->{$curr_section} //= {};
if ($modifier eq "") {
delete $expanded->{$x_var_name} if $exp_flag;
$sect_vars->{$var_name} = $value;
}
elsif ($modifier eq '?') {
$sect_vars->{$var_name} = $value if !exists($sect_vars->{$var_name});
}
elsif ($modifier eq '??') {
$sect_vars->{$var_name} = $value if (!exists($sect_vars->{$var_name})
|| $sect_vars->{$var_name} eq "");
}
elsif ($modifier eq '+') {
if (exists($sect_vars->{$var_name})) {
$sect_vars->{$var_name} .= " "
. ($exp_flag ? $self->$_expand_value($curr_section, $value) : $value);
}
else {
$sect_vars->{$var_name} = $value;
}
}
elsif ($modifier eq '.') {
$sect_vars->{$var_name} = ($sect_vars->{$var_name} // "")
. ($exp_flag ? $self->$_expand_value($curr_section, $value) : $value);
}
elsif ($modifier eq ':') {
delete $expanded->{$x_var_name} if $exp_flag; # Needed to make _expand_vars corectly!
$sect_vars->{$var_name} = $self->$_expand_vars($curr_section, $var_name, $value, undef, 1);
}
elsif ($modifier eq '+>') {
if (exists($sect_vars->{$var_name})) {
$sect_vars->{$var_name} =
($exp_flag ? $self->$_expand_value($curr_section, $value) : $value)
. ' ' . $sect_vars->{$var_name};
}
else {
$sect_vars->{$var_name} = $value;
}
}
elsif ($modifier eq '.>') {
$sect_vars->{$var_name} =
($exp_flag ? $self->$_expand_value($curr_section, $value) : $value)
. ($sect_vars->{$var_name} // "");
}
else {
$_fatal->("'$modifier': unsupported modifier");
}
}
return ($tocopy_sec_declared, $curr_section);
};
sub parse_ini {
my $self = shift;
my %args = (cleanup => 1,
@_ );
state $allowed_keys = {map {$_ => undef} qw(cleanup src src_name
tocopy_section tocopy_vars not_tocopy)};
state $dflt_src_name = "INI data";
_check_args(\%args, $allowed_keys);
foreach my $scalar_arg (qw(tocopy_section src_name)) {
croak("'$scalar_arg': must not be a reference") if ref($args{$scalar_arg});
}
delete $self->{+SRC_NAME} if exists($self->{+SRC_NAME});
$self->{+SRC_NAME} = $args{src_name} if exists($args{src_name});
my ( $cleanup, $src, $tocopy_section, $tocopy_vars, $not_tocopy) =
@args{qw(cleanup src tocopy_section tocopy_vars not_tocopy)};
croak("'src': missing mandatory argument") if !defined($src);
my $backup = $self->{+BACKUP} //= {};
if (defined($tocopy_section)) {
$backup->{tocopy_section} = $self->{+TOCOPY_SECTION};
$self->{+TOCOPY_SECTION} = $tocopy_section;
}
else {
$tocopy_section = $self->{+TOCOPY_SECTION};
}
$self->{+CURR_TOCP_SECTION} = $tocopy_section;
$Globals{'=TO_CP_SEC'} = $tocopy_section;
if ($tocopy_vars) {
$backup->{tocopy_vars} = $self->{+TOCOPY_VARS};
$self->$_check_tocopy_vars($tocopy_vars, 1);
}
if ($not_tocopy) {
$backup->{not_tocopy} = $self->{+NOT_TOCOPY};
$self->$_check_not_tocopy($not_tocopy, 1)
}
$self->{+SECTIONS} = [];
$self->{+SECTIONS_H} = {};
$self->{+EXPANDED} = {};
$self->{+VARIABLES} =
{$tocopy_section => ($self->{+TOCOPY_VARS} ? {%{$self->{+TOCOPY_VARS}}} : {})};
my $global_vars = $self->{+GLOBAL_VARS} = {%Globals};
my $variables = $self->{+VARIABLES};
my $tocopy_sec_vars = $variables->{$tocopy_section};
if (my $ref_src = ref($src)) {
$self->{+SRC_NAME} = $dflt_src_name if !exists($self->{+SRC_NAME});
if ($ref_src eq 'ARRAY') {
$src = [@$src];
foreach my $entry (@$src) {
croak("'src': unexpected ref type in array") if ref($entry);
if (!defined($entry)) {
carp("'src': undef entry - treated as empty string");
$entry = "";
}
}
}
else {
croak("'src': $ref_src: ref type not allowed");
}
}
else {
if (index($src, "\n") < 0) {
my $path = $src;
$src = [do { local (*ARGV); @ARGV = ($path); <> }];
$self->{+SRC_NAME} = $path if !exists($self->{+SRC_NAME});
my ($vol, $dirs, $file) = splitpath(rel2abs($path));
@{$global_vars}{'=INIfile', '=INIdir'} = ($file, catdir(length($vol // "") ? $vol : (),
$dirs));
}
else {
$src = [split(/\n/, $src)];
$self->{+SRC_NAME} = $dflt_src_name if !exists($self->{+SRC_NAME});
}
}
$global_vars->{'=srcname'} = $self->{+SRC_NAME};
my ($tocopy_sec_declared, undef) = $self->$_parse_ini($src);
my @sections = (exists($self->{+SECTIONS_H}{$tocopy_section}) ? () : $tocopy_section,
@{$self->{+SECTIONS}}
);
foreach my $section (@sections) {
my $sec_vars = $variables->{$section};
while (my ($variable, $value) = each(%$sec_vars)) {
$sec_vars->{$variable} = $self->$_expand_vars($section, $variable, $value);
}
}
if ($cleanup) {
while (my ($section, $sec_vars) = each(%{$variables})) {
foreach my $var (keys(%$sec_vars)) {
delete $sec_vars->{$var} if index($var, '=') >= 0;
}
}
delete $variables->{$self->{+TOCOPY_SECTION}} if (!$tocopy_sec_declared && !%$tocopy_sec_vars);
}
else {
if ($self->{+GLOBAL_MODE}) {
while (my ($section, $sec_vars) = each(%{$variables})) {
$sec_vars->{'='} = $section;
}
@{$tocopy_sec_vars}{keys(%$global_vars)} = values(%$global_vars);
}
else {
while (my ($section, $sec_vars) = each(%{$variables})) {
$sec_vars->{'='} = $section;
@{$sec_vars}{keys(%$global_vars)} = values(%$global_vars);
}
}
}
$self->{+TOCOPY_SECTION} = $backup->{tocopy_section} if exists($backup->{tocopy_section});
$self->{+TOCOPY_VARS} = $backup->{tocopy_vars} if exists($backup->{tocopy_vars});
$self->{+NOT_TOCOPY} = $backup->{not_tocopy} if exists($backup->{not_tocopy});
$backup = {};
return $self;
}
sub current_tocopy_section {$_[0]->{+CURR_TOCP_SECTION}}
sub tocopy_section {$_[0]->{+TOCOPY_SECTION}}
sub global_mode {$_[0]->{+GLOBAL_MODE}}
sub sections { defined($_[0]->{+SECTIONS}) ? [@{$_[0]->{+SECTIONS}}] : undef}
sub sections_h { defined($_[0]->{+SECTIONS_H}) ? +{ %{$_[0]->{+SECTIONS_H}} } : undef }
sub separator {$_[0]->{+SEPARATOR}}
sub src_name {$_[0]->{+SRC_NAME}}
sub variables { my $vars = $_[0]->{+VARIABLES} // return undef;
return {map {$_ => {%{$vars->{$_}}}} keys(%$vars)};
}
$_look_up = sub {
my ($self, $curr_sect, $variable) = @_;
my $matched = $variable =~ $self->{+VREF_RE};
my ($v_section, $v_basename) = $matched ? ($1, $2) : ($curr_sect, $variable);
my $v_value;
my $variables = $self->{+VARIABLES};
my $tocopy_section = $self->{+TOCOPY_SECTION};
if (!exists($variables->{$v_section})) {
$v_value = "";
}
elsif (exists($variables->{$v_section}{$v_basename})) {
$v_value = $variables->{$v_section}{$v_basename};
}
elsif ($v_basename !~ /\S/) {
$v_value = $v_basename;
}
elsif ($v_basename eq '=') {
$v_value = $v_section;
}
elsif ($v_basename =~ /^=(?:ENV|env):\s*(.*)$/) {
$v_value = $ENV{$1} // "";
}
elsif ($v_basename =~ /^=CONFIG:\s*(.*)$/) {
$v_value = $Config{$1} // "";
}
elsif (exists($self->{+GLOBAL_VARS}{$v_basename})) {
$v_value = $self->{+GLOBAL_VARS}{$v_basename};
}
elsif ($self->{+GLOBAL_MODE} && exists($variables->{$tocopy_section}{$v_basename})) {
if (!$matched && $curr_sect ne $tocopy_section && exists($self->{+NOT_TOCOPY}{$v_basename})) {
$v_value = "";
}
else {
$v_value = $variables->{$tocopy_section}{$v_basename};
}
}
else {
$v_value = "";
}
die("Internal error") if !defined($v_value);
return wantarray ? ($v_section, $v_basename, $v_value) : $v_value;
};
# extended var name
$_x_var_name = sub {
my ($self, $curr_sect, $variable) = @_;
if ($variable =~ $self->{+VREF_RE}) {
return ($2, "[$1]$2");
}
else {
return ($variable, "[$curr_sect]$variable");
}
};
$_expand_vars = sub {
my ($self, $curr_sect, $variable, $value, $seen, $not_seen) = @_;
my $top = !$seen;
my @result = ("");
my $level = 0;
my $x_variable_name;
if (defined($variable)) {
((my $var_basename), $x_variable_name) = $self->$_x_var_name($curr_sect, $variable);
return $self->$_look_up($curr_sect, $variable) if (exists($self->{+EXPANDED}{$x_variable_name})
|| $var_basename =~ /^=(?:ENV|CONFIG):/);
croak("recursive variable '", $x_variable_name, "' references itself")
if exists($seen->{$x_variable_name});
$seen->{$x_variable_name} = undef if !$not_seen;
}
foreach my $token (split(/(\$\(|\))/, $value)) {
if ($token eq '$(') {
++$level;
}
elsif ($token eq ')' && $level) {
# Now $result[$level] contains the name of a referenced variable.
if ($result[$level] eq '==') {
$result[$level - 1] .= $variable;
}
else {
$result[$level - 1] .=
$self->$_expand_vars($self->$_look_up($curr_sect, $result[$level]), $seen);
}
pop(@result);
--$level;
}
else {
$result[$level] .= $token;
}
}
croak("'$x_variable_name': unterminated variable reference") if $level;
$value = $result[0];
if ($x_variable_name) {
$self->{+EXPANDED}{$x_variable_name} = undef if $top;
delete $seen->{$x_variable_name};
}
return $value;
};
#
# This is a function, not a method!
#
sub _check_args {
my ($args, $allowed_args) = @_;
foreach my $key (keys(%$args)) {
croak("'$key': unsupported argument") if !exists($allowed_args->{$key});
}
delete @{$args}{ grep { !defined($args->{$_}) } keys(%$args) };
}
1; # End of Config::INI::RefVars
__END__
=pod
=head1 NAME
Config::INI::RefVars - INI file reader that supports make-style variable references and multiple assignment operators.
=head1 VERSION
Version 0.19
=head1 SYNOPSIS
use Config::INI::RefVars;
my $ini_reader = Config::INI::RefVars->new();
$ini_reader->parse_ini(src => $my_ini_file);
my $variables = $ini_reader->variables;
while (my ($section, $section_vars) = each(%$variables)) {
# ...
}
If the INI file contains:
section = $(=)
[sec A]
foo=Variable $(==) in section $(section)
bar:=$(foo)
foo.=!
[sec B]
baz = from $(=): ref foo from sec A: $([sec A]foo)
then C<< $ini_reader->variables >> returns:
{
'__TOCOPY__' => {
'section' => '__TOCOPY__'
},
'sec A' => {
'bar' => 'Variable foo in section sec A',
'foo' => 'Variable foo in section sec A!',
'section' => 'sec A'
},
'sec B' => {
'baz' => 'from sec B: ref foo from sec A: Variable foo in section sec A!',
'section' => 'sec B'
}
}
=head1 DESCRIPTION
=head2 INTRODUCTION
This module provides an INI file reader that allows INI variables and
environment variables to be referenced within the INI file. It also supports
some additional assignment operators.
Minimum version of perl required to use this module: C<v5.10.1>.
=head2 OVERVIEW
A line in an INI file should not start with an C<=> or the sequence
C<;!>. These are reserved for future extensions. Otherwise the parser throws a
"Directives are not yet supported" exception. Apart from these special cases,
the following rules apply:
=over
=item *
Spaces at the beginning and end of each line are ignored.
=item *
If the first non-white character of a line is a C<;> or a C<#>, then the line
is a comment line.
=item *
Comments can also be specified to the right of a section declaration (in this
case, the comment must not contain closing square brackets).
=item *
In a section header, spaces to the right of the opening square
bracket and to the left of the closing square bracket are ignored, i.e. a
section name always begins and ends with a non-white character. B<But>: As a
special case, the name of a section heading can be an empty character string.
=item *
Section name must be unique.
=item *
The order of the sections is retained: The C<sections> method returns an array
of sections in the order in which they appear in the INI file.
=item *
A variable name cannot be empty.
=item *
The sequence C<$(...)> is used to reference INI variables or environment
variables.
=item *
Spaces around the assignment operator are ignored. Note that there are several
assignment operators, not just C<=>.
=item *
If you want to define a variable whose name ends with an punctuation character other
than an underscore, there must be at least one space between the variable name
and the assignment operator.
=item *
The source to be parsed (argument C<src> of the method C<parse_ini>) does not
have to be a file, but can also be a string or an array.
=item *
There is no escape character.
=back
You will find further details in the following sections.
=head2 SECTIONS
A section begins with a section header:
[section]
A line contains a section heading if the first non-blank character is a C<[>
and the last non-blank character is a C<]>. The character string in between is
the name of the section, whereby spaces to the right of C<[> and to the left
of C<]> are ignored.
[ The name of the section ]
This sets the section name to C<The name of the section>.
As a special case, C<[]> or C<[ ]> are permitted, which results in a section
name that is just an empty string.
Section names must be unique.
An INI file does not have to start with a section header, it can also start
with variable definitions. In this case, the variables are added to the
I<tocopy> section (default name: C<__TOCOPY__>). You can explicitly specify
the I<tocopy> section heading, but then this must be the first active line in
your INI file.
=head2 VARIABLES AND ASSIGNMENT OPERATORS
There are several assignment operators, the basic one is the C<=>, the others
are formed by a C<=> preceded by one or more punctuation characters. Thus, if
you want to define a variable whose name ends with an punctuation character,
there must be at least one space between the variable name and the assignment
operator.
B<Note>: Since the use of the underscore in identifiers is so common, it is
not treated as a punctuation character here.
=over
=item C<=>
The standard assignment operator. Note: A second assignment to the same
variable simply overwrites the first.
=item C<?=>
Works like the corresponding operator of GNU Make: the assignment is only
executed if the variable is not yet defined.
=item C<??=>
This works similar to C<?=>: the assignment is only executed if the variable
is not yet defined or if its current, non-expanded value is an empty string.
This allows you to set a default value for an environment variable:
env_var:=$(=ENV:ENV_VAR)
env_var??=the default
If the environment variable C<ENV_VAR> is not defined or is empty, then
C<env_var> has the value C<the default>. This would not work with
C<?=>. Please note that you must also use C<:=>!
=item C<:=>
Works like the corresponding operator of GNU Make: all references to other
variables are expanded when the variable is defined. See section L</"REFERENCING VARIABLES">
=item C<.=>
The right-hand side is appended to the value of the variable. If the variable
is not yet defined, this does the same as a simple C<=>.
Example:
var=abc
var.=123
Now C<var> has the value C<abc123>.
=item C<+=>
Works like the corresponding operator of GNU Make: the right-hand side is
appended to the value of the variable, separated by a space. If the right-hand
side is empty, a space is appended. If the variable is not yet defined, this
has the same effect as a simple C<=>.
Example:
var=abc
var+=123
Now C<var> has the value C<abc 123>.
=item C<< .>= >>
The right-hand side is placed in front of the value of the variable. If the
variable is not yet defined, this has the same effect as a simple C<=>.
Example:
var=abc
var.>=123
Now C<var> has the value C<123abc>.
=item C<< +>= >>
The right-hand side is placed in front the value of the variable, separated by
a space. If the right-hand side is empty, a space is placed in front of the
variable value. If the variable is not yet defined, this has the same effect
as a simple C<=>.
Example:
var=abc
var+>=123
Now C<var> has the value C<123 abc>.
=back
=head2 REFERENCING VARIABLES
=head3 Basic Referencing
The referencing of variables is similar but not identical to that in B<make>,
you use C<$(I<VARIABLE>)>. Example:
a=hello
b=world
c=$(a) $(b)
Variable C<c> has the value C<hello world>. As with B<make>, lazy evaluation
is used, i.e. you would achieve exactly the same result with this:
c=$(a) $(b)
a=hello
b=world
But the following would result in C<c> containing only one space:
c:=$(a) $(b)
a=hello
b=world
Unlike in B<make>, the round brackets cannot be omitted for variables with
only one letter!
You can nest variable references:
foo=the foo value
var 1=fo
var 2=o
bar=$($(var 1)$(var 2))
Now the variable C<bar> has the value C<the foo value>.
A reference to a non-existent variable is always expanded to an empty
character string.
If you need a literal C<$(...)> sequence, e.g. C<$(FOO)>, as part of a
variable value, you can write:
var = $$()(FOO)
This results in the variable C<var> having the value C<$(FOO)>. It works
because C<$()> always expands to an empty string (see section L</"PREDEFINED
VARIABLES">).
Recursive references are not possible, an attempt to do so leads to a fatal
error. However, you can do this with the C<:=> assignment:
a=omethin
a:=s$(a)g
C<a> has the value C<something>. However, due to the way C<:=> works, this is
not really a recursive reference.
=head3 Referencing Variables of other Sections
By default, you can reference a variable in another section by writing the
name of the section in square brackets, followed by the name of the variable:
[sec A]
foo=Referencing a variable from section: $([sec B]bar)
[sec B]
bar=Referenced!
You can switch to a different notation by specifying the constructor argument
C<separator>.
A more complex example:
[A]
a var = 1234567
[B]
b var = a var
nested = $([$([C]c var)]$(b var))
[C]
c var = A
Variable C<nested> in section C<B> has the value C<1234567>:
=over
=item
C<$([C]c var)> expands to C<A>,
=item
C<$(b var)> expands to C<a var>,
=item
We therefore have C<$([A]a var)> which leads to C<1234567>.
=back
=head2 PREDEFINED VARIABLES
=head3 Variables related to Section and Variable Names
=over
=item C<=>
C<$(=)> expands to the name of the current section.
=item C<==>
C<$(==)> expands to the name of the variable that is currently being defined.
Think of this as a pseudo-variable, something like $([SECTION]==) always
results in an empty string.
=back
Example:
[A]
foo=variable $(==) of section $(=)
ref=Reference to foo of section B: $([B]foo)
[B]
foo=variable $(==) of section $(=)
bar=$(foo)
The hash returned by the C<variables> method is then:
{
'A' => {
'foo' => 'variable foo of section A',
'ref' => 'Reference to foo of section B: variable foo of section B'
},
'B' => {
'foo' => 'variable foo of section B'
'bar' => 'variable foo of section B',
}
}
=head3 Variables related to the Source
=over
=item C<=srcname>
Name of the INI source. If the source is a file, this corresponds to the value
that you have passed to C<parse_ini> via the C<src> argument, otherwise it is
set to "INI data". The value can be overwritten with the argument C<src_name>.
=item C<=INIdir>, C<=INIfile>
Directory (absolute path) and file name of the INI file. These variables are
only present if the source is a file, otherwise they are not defined.
=back
=head3 Variables related to the OS
=over
=item C<=:>
The directory separator, C<\> on Windows, C</> on Linux. Note: This is not
always sufficient to create a path, e.g. on VMS.
=item C<=::>
Path separator, which is used in the environment variable C<PATH>, for
example.
=back
=head3 Space Variables
C<$()> always expands to an empty string, C<$(E<nbsp>)>, C<$(E<nbsp>E<nbsp>)>
with any number of spaces within the parens expand to exactly these spaces. So
there are several ways to define variables with heading or trailing spaces:
foo = abc $()
bar = $( )abc
The value of C<foo> has three spaces at the end, the value of C<bar> has three
spaces at the beginning. A special use case for C<$()> is the avoidance of
unwanted variable expansion:
var=hello!
x=$(var)
y=$$()(var)
With these settings, C<x> has the value C<Hello!>, but C<y> has the value
C<$(var)>.
=head3 Other Variables
=over
=item C<=TO_CP_SEC>
Name of the I<tocopy> section, see L</"THE I<TOCOPY> SECTION">.
=item C<=VERSION>
Version of the C<Config::INI::RefVars> module.
=back
=head3 Custom predefined Variables
Currently, custom predefined variables are not supported. But you can do
something very similar, see argument C<tocopy_vars> (of C<new> and
C<parse_ini>), see also L</"THE I<TOCOPY> SECTION">. With this argument you
can also define variables whose names contain a C<=>, which is obviously
impossible in an INI file.
=head3 Predefined Variables in resulting Hash
By default, all variables whose names contain a C<=> are removed from the
resulting hash. This means that the variables discussed above are not normally
included in the result. This behavior can be changed with the C<parse_ini>
argument C<cleanup>. The variable C<==> can of course not be included in the
result.
=head2 ACCESSING ENVIRONMENT AND CONFIG VARIABLES
You can access environment variables with this C<$(=ENV:...)> or this
C<$(=env:...)> notation. Example:
path = $(=ENV:PATH)
C<path> now contains the content of your environment variable C<PATH>.
The results of C<$(=ENV:...)> and C<$(=env:...)> are almost always the
same. The difference is that the parser always leaves the value of
C<$(=ENV:...)> unchanged, but tries to expand the value of C<$(=env:...)>.
For example, let's assume you have an environment variable C<FOO> with the
value C<$(var)> and you write this in your INI file:
var=hello!
x=$(=ENV:FOO)
y=$(=env:FOO)
This results in C<x> having the value C<$(var)>, while C<y> has the value C<hello!>.
You can access configuration variables of Perl's L<Config> module with this C<$(=CONFIG:...)> notation. Example:
the archlib=$(=CONFIG:archlib)
This gives the variable C<the archlib> the value of C<$Config{archlib}>.
Note: In contrast to C<$(=ENV:...)>, there is no lower-case counterpart to
C<$(=CONFIG:...)>, as this would not make sense.
=head2 THE I<TOCOPY> SECTION
=head3 Default Behavior
If specified, the method C<parse_ini> copies the variables of the I<tocopy>
section (default name: C<__TOCOPY__>) to any other section when the INI file
is read (default, this behavior can be changed by the constructor argument
C<global_mode>). For example this
[__TOCOPY__]
some var=some value
section info=$(=)
[A]
[B]
is exactly the same as this:
[__TOCOPY__]
some var=some
section info=$(=)
[A]
some var=some
section info=$(=)
[B]
some var=some
section info=$(=)
Of course, you can change or overwrite a variable copied from the C<tocopy>
section locally within a section at any time without any side effects. In this
case, you can access the original value as follows:
$([__TOCOPY__]some var)
or - more generally - like this:
$([$(=TO_CP_SEC)]some var)
You can exclude variables with the argument C<not_tocopy> from copying
(methods C<new> and C<parse_ini>), but there is currently no notation to do
this in the INI file.
The I<tocopy section> is optional. If it is specified, it must be the first
section. By default, its name is C<__TOCOPY__>, this can be changed with the
argument C<tocopy_section> (methods C<new> and C<parse_ini>). You can omit the
C<[__TOCOPY__]> header and simply start your INI file with variable
definitions. These then simply become the I<tocopy section>. So this:
[__TOCOPY__]
a=this
b=that
[sec]
x=y
is exactly the same as this:
a=this
b=that
[sec]
x=y
You can also add I<tocopy> variables via the argument C<tocopy_vars> (methods
C<new> and C<parse_ini>), these are treated as if they were at the very
beginning of the C<tocopy> section.
=head3 Global Mode
If you specify the constructor argument C<global_mode> with a I<true> value,
the variables of the I<tocopy> section are not copied, but behave like global
variables. Variables that you specify with the argument C<not_tocopy> are not
treated as global.
Consequently, there is almost no difference in the referencing of variables if
you use the global mode. The advantage of this mode is that you do not clutter
your sections with unwanted variables.
Example:
[__TOCOPY__]
a=this
b=that
[sec]
x=y
This would lead to this result by default:
{
__TOCOPY__ => {a => 'this', b => 'that'},
sec => {a => 'this', b => 'that', x => 'y'}
}
But in global mode the result is:
{
__TOCOPY__ => {a => 'this', b => 'that'},
sec => {x => 'y'}
}
For a local copy of a global variable, use assignment operator C<:=>.
B<NOTE:>
In some special cases, variables have different values in standard mode than in global mode.
section=$(=)
x=GLOBAL
x_val=$(x)
[local-sec]
var_1 := $(section)
var_2 = $(section)
x=LOCAL
x_1 := $(x_val)
x_2 = $(x_val)
By default, you will get:
{
'__TOCOPY__' => {
'section' => '__TOCOPY__',
'x' => 'GLOBAL',
'x_val' => 'GLOBAL'
},
'local-sec' => {
'section' => 'local-sec',
'var_1' => 'local-sec',
'var_2' => 'local-sec',
'x' => 'LOCAL',
'x_1' => 'LOCAL',
'x_2' => 'LOCAL',
'x_val' => 'LOCAL'
}
}
But in global mode, the result is:
{
'__TOCOPY__' => {
'section' => '__TOCOPY__',
'x' => 'GLOBAL',
'x_val' => 'GLOBAL'
},
'local-sec' => {
'var_1' => 'local-sec',
'var_2' => '__TOCOPY__',
'x' => 'LOCAL',
'x_1' => 'LOCAL',
'x_2' => 'GLOBAL'
}
}
Note the different values for C<var_2> and C<x_2>. When the assignment C<x_1
:= $(x_val)> is reached, the right-hand side is evaluated immediately, so that
C<$(x_val)> becomes C<$(x)>, which in turn leads to C<LOCAL>, since the
definition of C<x> in C<[local-sec]> shadows the global C<x>.
In contrast, the value of C<x_2> is evaluated after the file has been
completely read. This value is C<$(x_val)> and the variable C<x_val> was in
turn previously evaluated in the global section and has the value C<GLOBAL>,
which then becomes the value of C<x_2>.
In standard mode, C<x_val=$(x)> is copied to C<[local-sec]> and C<x_2> is
given the value C<LOCAL> due to the local definition of C<x>.
A corresponding explanation applies to the different values of C<var_2>.
=head2 COMMENTS
As said, if the first non-white character of a line is a C<;> or a C<#>, then the line
is a comment line.
# This is a comment
; This is also a comment
;! a comment, but: avoid ";!" at the very beginning of a line!
var = value ; this is not a comment but part of the value.
Avoid C<;!> at the very beginning of a line, otherwise you will get an
error. The reason for this is that this sequence is reserved for future
extensions. However, you can use it if you precede it with spaces.
You cannot append a comment to the right of a variable definition, as your
comment would otherwise become part of the variable value. But you can append
a comment to the right of a header declaration:
[section] ; My fancy section
B<Attention>: if you do this, the comment must not contain a C<]> character!
=head2 PITFALLS
In most cases, the keys in the hash returned by C<variables> are the same as
the keys in the hash returned by the C<sections_h> method and the entries in
the array returned by the C<sections> method. In special cases, however, there
may be a difference with regard to the I<tocopy> section. Example:
[A]
a=1
[B]
b=2
If you parse this INI source like this:
my $obj = Config::INI::RefVars->new();
$obj->parse_ini(src => $src, tocopy_vars => {'foo' => 'xyz'});
then the C<variables> method returns this:
'A' => {
'a' => '1',
'foo' => 'xyz'
},
'B' => {
'b' => '2',
'foo' => 'xyz'
},
'__TOCOPY__' => {
'foo' => 'xyz'
}
but C<sections_h> returns
{ 'A' => '0',
'B' => '1' }
and C<sections> returns
['A', 'B']
No C<__TOCOPY__>. The reason for this is that the return values of
C<sections_h> and C<sections> refer to what is contained in the source, and in
this case C<__TOCOPY__> is not contained in the source, but comes from a
method argument.
=head2 METHODS
=head3 new
The constructor takes the following optional named arguments:
=over
=item C<cmnt_vl>
Optional, a Boolean value. If this value is set to I<true>, comments are
permitted in variable lines. The comment character is a semicolon preceded by
one or more spaces.
Example:
[section]
var 1=val 1 ; comment
var 2=val 2 ; ; ; comment
var 3=val 3; no comment
var 4=val 4 $(); no comment
After parsing, the C<variables> method returns:
section => {'var 1' => 'val 1',
'var 2' => 'val 2',
'var 3' => 'val 3; no comment',
'var 4' => 'val 4 ; no comment',
}
Default is I<false> (C<undef>).
=item C<global_mode>
Optional, a boolean. Changes handling of the I<tocopy> section, see section
L</"Global Mode">. See also the accessor method of the same name.
=item C<not_tocopy>
Optional, a reference to a hash or an array of strings. The hash keys or array
entries specify a list of variables that should not be copied from the
I<tocopy> section to the other sections. It does not matter whether these
variables actually occur in the I<tocopy> section or not.
Default is C<undef>.
=item C<separator>
Optional, a string. If specified, an alternative notation can be used for
referencing variables in another section. Example:
my $obj = Config::INI::RefVars->new(separator => '::');
Then you can write:
[A]
y=27
[B]
a var=$(A::y)
This gives the variable C<a var> the value C<27>.
The following characters are permitted for C<separator>:
#!%&',./:~\
See also the accessor method of the same name.
=item C<tocopy_section>
Optional, a string. Specifies a different name for the I<tocopy>
section. Default is C<__TOCOPY__>. See accessor C<tocopy_section>.
=item C<tocopy_vars>
Optional, a hash reference. If specified, its keys become variables of the
I<tocopy> section, the hash values become the corresponding variable values. This
allows you to specify variables that you cannot specify in the INI file,
e.g. variables with a C<=> in the name.
Keys with C<=> or C<;> as the first character are not permitted.
Default is C<undef>.
=item C<varname_chk_re>
Optional, a compiled regex. If specified, each variable name defined in the
INI source must match this regex.
Example:
my $obj = Config::INI::RefVars->new(varname_chk_re => qr/^[A-Z]/);
my $src = <<'EOT';
[the section]
A=the value
xYZ=123
Z1=z2
Y=
EOT
$obj->parse_ini(src => $src);
This will result in an exception with the message C<'xYZ': var name does not
match varname_chk_re>.
=back
=head3 current_tocopy_section
Returns the name of the section I<tocopy> that was used the last time
C<parse_ini> was called. Please note that the section does not have to be
present in the data.
See also method C<tocopy_section>.
=head3 global_mode
Returns a boolean value indicating whether the global mode is activated or
not. See constructor argument of the same name, see also section L</"Global
Mode">.
=head3 parse_ini
Parses an INI source. The method takes the following optional named arguments:
=over
=item C<src>
Mandatory, a string or an array reference. This specifies the source to
parse. If it is a character string that does not contain a newline character,
it is treated as the name of an INI file. Otherwise, its content is parsed
directly.
=item C<cleanup>
Optional, a boolean. If this value is set to I<false>, variables with a C<=>
in their name are not removed from the resulting hash that is returned by the
C<variables> method. But in global mode, most of this variables will not be
contained, see section L</"Global Mode">.
Default is 1 (I<true)>
=item C<tocopy_section>
Optional, a string. Specifies a different name for the I<tocopy> section for
this run only. The previous value is restored before the method
returns. Default is the string returned by accessor C<tocopy_section>.
See constructor argument of the same name.
=item C<tocopy_vars>
Optional, overwrites the corresponding setting saved in the object for this
run only. The previous setting is restored before the method returns.
See constructor argument of the same name.
=item C<not_tocopy>
Optional, overwrites the corresponding setting saved in the object for this
run only. The previous setting is restored before the method returns.
See constructor argument of the same name.
=item C<src_name>
Optional, overwrites the corresponding setting saved in the object for this
run only. The previous setting is restored before the method returns.
See constructor argument of the same name, see also the accessor of the same
name.
=back
=head3 sections
Returns a reference to an array of section names from the INI source, in the
order in which they appear there.
=head3 sections_h
Returns a reference to a hash whose keys are the section names from the INI
source, the values are the corresponding indices in the array returned by
C<sections>.
=head3 separator
Returns the value that was passed to the constructor via the argument of the
same name, or C<undef> .
=head3 src_name
Returns the name of the INI source (file name that you have passed to
C<parse_ini> via the argument C<src>, or the one that you have passed via the
argument C<src_name>, or "C<INI data>", see section L</"Variables in relation
to the source">.
=head3 tocopy_section
Returns the name of the I<tocopy> section that will be used as the default for
the next call to C<parse_ini>.
See also method C<current_tocopy_section>.
=head3 variables
Returns a reference to a hash of hashes. The keys are the section names, each
value is the corresponding hash of variables (key: variable name, value:
variable value). By default, variables with a C<=> in their name are not
included; this can be changed with the C<cleanup> argument.
=head2 EXAMPLES
You can parse INI files as described here L<$(section\name) syntax for INI
file
as follows:
my $obj = Config::INI::RefVars->new(separator => "\\",
cmnt_vl => 1,
tocopy_section => 'Settings',
global_mode => 1);
my $src = <<'EOT';
[Settings]
BaseDir="d:\dhcpsrv" ; dhcpsrv.exe resides here
IPBIND_1=192.168.17.2
IPPOOL_1=$(Settings\IPBIND_1)-50
AssociateBindsToPools=1
Trace=1
TraceFile="$(BaseDir)\dhcptrc.txt" ; trace file
[DNS-Settings]
EnableDNS=1
[General]
SUBNETMASK=255.255.255.0
DNS_1=$(IPBIND_1)
[TFTP-Settings]
EnableTFTP=1
Root="$(BaseDir)\wwwroot" ; use wwwroot for http and tftp
[HTTP-Settings]
EnableHTTP=1
Root="$(BaseDir)\wwwroot" ; use wwwroot for http and tftp
EOT
$obj->parse_ini(src => $src);
=head1 SEE ALSO
This one allows also referencing variables: L<Config::IOD::Reader>.
Other modules handling INI files:
L<Config::INI>,
L<Config::INI::Tiny>,
L<Config::IniFiles>, and many more.
=head1 AUTHOR
Abdul al Hazred, C<< <451 at gmx.eu> >>
=head1 BUGS
Please report any bugs or feature requests to C<bug-config-ini-accvars at
rt.cpan.org>, or through the web interface at
be notified, and then you'll automatically be notified of progress on your bug
as I make changes.
=head1 SUPPORT
You can find documentation for this module with the perldoc command.
perldoc Config::INI::RefVars
You can also look for information at:
=over 4
=item * RT: CPAN's request tracker (report bugs here)
=item * Search CPAN
=item * GitHub Repository
=back
=head1 LICENSE AND COPYRIGHT
This software is copyright (c) 2023 by Abdul al Hazred.
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