Config-Dot
view release on metacpan or search on metacpan
package Config::Dot;
use strict;
use warnings;
use Class::Utils qw(set_params);
use Config::Utils qw(hash);
use English qw(-no_match_vars);
use Error::Pure qw(err);
use Readonly;
# Constants.
Readonly::Scalar my $EMPTY_STR => q{};
our $VERSION = 0.10;
# Constructor.
sub new {
my ($class, @params) = @_;
my $self = bless {}, $class;
# Callback.
$self->{'callback'} = undef;
# Config hash.
$self->{'config'} = {};
# Set conflicts detection as error.
$self->{'set_conflicts'} = 1;
# Process params.
set_params($self, @params);
# Check config hash.
if (! $self->_check($self->{'config'})) {
err 'Bad \'config\' parameter.';
}
# Check callback.
if (defined $self->{'callback'} && ref $self->{'callback'} ne 'CODE') {
err 'Parameter \'callback\' isn\'t code reference.';
}
# Count of lines.
$self->{'count'} = 0;
# Stack.
$self->{'stack'} = [];
# Object.
return $self;
}
# Parse text or array of texts.
sub parse {
my ($self, $string_or_array_ref) = @_;
my @text;
if (ref $string_or_array_ref eq 'ARRAY') {
@text = @{$string_or_array_ref};
} else {
@text = split m/$INPUT_RECORD_SEPARATOR/sm,
$string_or_array_ref;
}
foreach my $line (@text) {
$self->{'count'}++;
$self->_parse($line);
}
return $self->{'config'};
}
# Reset content.
sub reset {
my $self = shift;
$self->{'config'} = {};
$self->{'count'} = 0;
return;
}
# Serialize.
sub serialize {
my $self = shift;
return join $INPUT_RECORD_SEPARATOR,
$self->_serialize($self->{'config'});
}
# Check structure.
sub _check {
my ($self, $config_hr) = @_;
if (ref $config_hr eq 'HASH') {
foreach my $key (sort keys %{$config_hr}) {
if (ref $config_hr->{$key} ne ''
&& ! $self->_check($config_hr->{$key})) {
return 0;
}
}
return 1;
} else {
return 0;
}
}
# Parse string.
sub _parse {
my ($self, $string) = @_;
# Remove comments on single line.
$string =~ s/^\s*#.*$//sm;
# Blank space.
if ($string =~ m/^\s*$/sm) {
return 0;
}
# Split.
my ($key, $val) = split m/=/sm, $string, 2;
# Not a key.
if (length $key < 1) {
return 0;
}
# Bad key.
if ($key !~ m/^[-\w\.:,]+\+?$/sm) {
err "Bad key '$key' in string '$string' at line ".
"'$self->{'count'}'.";
}
my @tmp = split m/\./sm, $key;
hash($self, \@tmp, $val);
# Ok.
return 1;
}
# Serialize.
sub _serialize {
my ($self, $config_hr) = @_;
my @ret;
foreach my $key (sort keys %{$config_hr}) {
if (ref $config_hr->{$key} eq 'HASH') {
my @subkey = $self->_serialize($config_hr->{$key});
foreach my $subkey (@subkey) {
push @ret, $key.'.'.$subkey;
}
} else {
if ($config_hr->{$key} =~ m/\n/ms) {
err 'Unsupported stay with newline in value.';
}
push @ret, $key.'='.$config_hr->{$key};
}
}
return @ret;
}
1;
__END__
=pod
=encoding utf8
=head1 NAME
Config::Dot - Module for simple configure file parsing.
=head1 SYNOPSIS
use Config::Dot;
my $cnf = Config::Dot->new(%params);
my $struct_hr = $cnf->parse($string);
$cnf->reset;
my $serialized = $cnf->serialize;
=head1 METHODS
=head2 C<new>
my $cnf = Config::Dot->new(%params);
Constructor.
=over 8
=item * C<callback>
Callback code for adding parameter.
( run in 0.775 second using v1.01-cache-2.11-cpan-71847e10f99 )