# -*- indent-tabs-mode: nil; -*- # vim:ft=perl:et:sw=4 # $Id$ # Sympa - SYsteme de Multi-Postage Automatique # # Copyright 2018, 2021 The Sympa Community. See the # AUTHORS.md file at the top-level directory of this distribution and at # . # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; either version 2 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . package Sympa::Config; use strict; use warnings; use Sympa::Language; use Sympa::Tools::Data; use Sympa::Tools::Text; sub new { my $class = shift; my $context = shift; my %options = @_; # The undef means list creation. # Empty hashref (default) means loading existing config. my $config; if (exists $options{config}) { $config = $options{config}; } else { $config = {}; } $config = Sympa::Tools::Data::clone_var($config) if $options{copy}; #FIXME:Should $config be sanitized? my $self = bless {context => $context, _config => $config, _changes => {}} => $class; my $pinfo = $self->_schema; foreach my $pname (_keys($pinfo)) { $self->__init_schema($pinfo->{$pname}, [$pname], %options); } $self->{_pinfo} = $pinfo; return $self; } #sub _schema; # Initialize default values. sub __init_schema { my $self = shift; my $phash = shift; my $pnames = shift; my %options = @_; my $subres; if (ref $phash->{format} eq 'HASH') { foreach my $pname (_keys($phash->{format})) { my $pii = $phash->{format}->{$pname}; $subres->{$pname} = $self->__init_schema($pii, [@$pnames, $pname], %options); } } return $self->_init_schema_item($phash, $pnames, $subres, %options); } sub _init_schema_item { my $self = shift; my $pitem = shift; my $pnames = shift; my $subres = shift; my %options = @_; return undef unless ref $pitem->{format} ne 'HASH' and exists $pitem->{default}; my $default = $pitem->{default}; if ($pitem->{occurrence} =~ /n$/) { # The set or the array of scalars if (ref $default) { ; } elsif (defined $default) { my $re = quotemeta($pitem->{split_char} || ','); $pitem->{default} = [split /\s*$re\s*/, $default]; } else { $pitem->{default} = []; } } elsif ($pitem->{scenario} or $pitem->{task}) { if (ref $default) { ; } elsif (defined $default) { $pitem->{default} = {name => $default}; } } return undef; } sub get { my $self = shift; my $ppath = shift; my @ppaths = split /[.]/, $ppath; return unless @ppaths; my @value = _get($self->{_config}, @ppaths); return unless @value; return $value[0] unless ref $value[0]; return Sympa::Tools::Data::clone_var($value[0]); } sub _get { my $cur = shift; my @ppaths = @_; while (1) { my $key = shift @ppaths; if ($key =~ /\A\d+\z/) { unless (ref $cur eq 'ARRAY' and exists $cur->[$key]) { return; } elsif (not @ppaths) { return ($cur->[$key]); } else { $cur = $cur->[$key]; } } else { unless (ref $cur eq 'HASH' and exists $cur->{$key}) { return; } elsif (not @ppaths) { return $cur->{$key}; } else { $cur = $cur->{$key}; } } } } sub get_change { my $self = shift; my $ppath = shift; my @ppaths = split /[.]/, $ppath; return unless @ppaths; my @value = _get_change($self->{_changes}, @ppaths); return unless @value; return $value[0] unless ref $value[0]; return Sympa::Tools::Data::clone_var($value[0]); } sub _get_change { my $new = shift; my @ppaths = @_; while (1) { my $key = shift @ppaths; unless (ref $new eq 'HASH' and exists $new->{$key}) { return; } elsif (not @ppaths) { return $new->{$key}; } else { $new = $new->{$key}; } } } sub get_changeset { my $self = shift; return $self->{_changes}; } # Apply default values, if elements are mandatory and are scalar. # The init option means list/node creation. sub _apply_defaults { my $self = shift; my $cur = shift; my $phash = shift; my %options = @_; foreach my $key (_keys($phash)) { my $pii = $phash->{$key}; if (exists $cur->{$key}) { next; } elsif (ref $pii->{format} eq 'HASH') { # Not a scalar next; } elsif (exists $pii->{default}) { if ($options{init} or $pii->{occurrence} =~ /^1/) { if (ref $pii->{default}) { $cur->{$key} = Sympa::Tools::Data::clone_var($pii->{default}); } else { $cur->{$key} = $pii->{default}; } } } } } sub get_schema { my $self = shift; return Sympa::Tools::Data::clone_var($self->{_pinfo}); } sub keys { my $self = shift; my $pname = shift; return _keys($self->{_pinfo}) unless $pname; my @pnames = split /[.]/, $pname; my $phash = $self->{_pinfo}; while (1) { my $key = shift @pnames; unless (ref $phash eq 'HASH' and exists $phash->{$key} and exists $phash->{$key}->{format}) { return; } elsif (not @pnames) { return _keys($phash->{$key}->{format}); } else { $phash = $phash->{$key}->{format}; } } } sub _keys { my $hash = shift; my $phash = shift || $hash; return sort { ($phash->{$a}->{order} || 999) <=> ($phash->{$b}->{order} || 999) } CORE::keys %$hash; } # Gets parameter name of node from list of parameter paths. sub _pname { my $ppaths = shift; return undef unless $ppaths and @$ppaths; [grep { !/\A\d+\z/ } @$ppaths]->[-1]; } # Gets full parameter name of node from list of parameter paths. sub _pfullname { my $ppaths = shift; return undef unless $ppaths and @$ppaths; return join '.', grep { !/\A\d+\z/ } @$ppaths; } sub submit { my $self = shift; my $new = shift; my $user = shift; my $errors = shift; my %options = @_; my $changes = $self->_sanitize_changes($new, $user); # Error if no parameter was edited. unless ($changes and %$changes) { $self->{_changes} = {}; push @$errors, ['notice', 'no_parameter_edited']; return ''; } $self->{_changes} = $changes; return $self->_validate_changes($changes, $errors, %options); } # Sanitizes parsed input including changes. # Parameters: # $new: Change information. # $user: Operating user. $param->{'user'}{'email'}. # Returns: # Sanitized input, where "owner.0.gecos" will be stores in # $hashref->{'owner'}{'0'}{'gecos'}. sub _sanitize_changes { my $self = shift; my $new = shift; my $user = shift; return undef unless ref $new eq 'HASH'; # Sanity check # Apply privileges: {privilege} will keep 'hidden', 'read' or 'write'. my $pinfo = $self->get_schema($user); # Undefined {_config} means list creation. # Empty hashref means loading existing config. my $init = (not defined $self->{_config}); my $loading = ($self->{_config} and not %{$self->{_config}}); my $cur = $init ? {} : Sympa::Tools::Data::clone_var($self->{_config}); $self->_apply_defaults($cur, $pinfo, init => ($init and not $loading)); my %ret = map { unless (exists $pinfo->{$_} and $pinfo->{$_}) { (); # Sanity check: unknown parameter } else { # Resolve alias. my ($k, $o) = ($_, $_); do { ($k, $o) = ($o, $pinfo->{$o}->{obsolete}); } while ($o and $pinfo->{$o}); unless ($k eq $_) { $new->{$k} = $new->{$_}; delete $new->{$_}; } my $pii = $pinfo->{$k}; my $ppi = [$k]; my $newi = $new->{$k}; my $curi = $cur->{$k}; my @r; if ($pii->{occurrence} =~ /n$/) { if (ref $pii->{format} eq 'ARRAY') { @r = $self->_sanitize_changes_set($curi, $newi, $pii, $ppi); } else { @r = $self->_sanitize_changes_array($curi, $newi, $pii, $ppi, loading => $loading); } } elsif (ref $pii->{format} eq 'HASH') { @r = $self->_sanitize_changes_paragraph( $curi, $newi, $pii, $ppi, init => (not defined $curi), loading => $loading ); } else { @r = $self->_sanitize_changes_leaf($curi, $newi, $pii, $ppi); } # Omit removal if current configuration is already empty. (@r and not defined $r[1] and not defined $curi) ? () : @r; } } _keys($new, $pinfo); return {%ret}; } # Sanitizes set. sub _sanitize_changes_set { my $self = shift; my $cur = shift || []; my $new = shift; my $pitem = shift; my $ppaths = shift; return () unless ref $new eq 'ARRAY'; # Sanity check return () if $pitem->{obsolete}; return () unless 'write' eq ($pitem->{privilege} // ''); # Resolve synonym. if (ref $pitem->{synonym} eq 'HASH') { @$new = map { if (defined $_) { my $synonym = $pitem->{synonym}->{$_}; (defined $synonym) ? $synonym : $_; } else { undef; } } @$new; } my $i = -1; my %updated = map { $i++; my $curi = $_; (grep { Sympa::Tools::Data::smart_eq($curi, $_) } @$new) ? () : ($i => undef); } @$cur; my %added = map { my $newi = $_; (grep { Sympa::Tools::Data::smart_eq($newi, $_) } @$cur) ? () : (++$i => $_); } @$new; my %ret = (%updated, %added); # If all children are removed, remove parent. while (my ($k, $v) = each %ret) { $cur->[$k] = $v; } return (_pname($ppaths) => undef) unless grep { defined $_ } @$cur; unless (%ret) { return (); # No valid changes } else { return (_pname($ppaths) => {%ret}); } } # Sanitizes array. sub _sanitize_changes_array { my $self = shift; my $cur = shift || []; my $new = shift; my $pitem = shift; my $ppaths = shift; my %options = @_; return () unless ref $new eq 'ARRAY'; # Sanity check return () if $pitem->{obsolete}; return () unless 'write' eq ($pitem->{privilege} // ''); my $i = -1; my %ret = map { $i++; my $curi = $cur->[$i]; my $ppi = [@$ppaths, $i]; my @r; if (ref $pitem->{format} eq 'HASH') { @r = $self->_sanitize_changes_paragraph( $curi, $_, $pitem, $ppi, init => (not defined $curi), loading => $options{loading} ); } else { @r = $self->_sanitize_changes_leaf($curi, $_, $pitem, $ppi); } # Omit removal if current configuration is already empty. (@r and not defined $r[1] and not defined $curi) ? () : (@r ? ($i => $r[1]) : ()); } @$new; # If all children are removed, remove parent. while (my ($k, $v) = each %ret) { $cur->[$k] = $v; } return (_pname($ppaths) => undef) unless grep { defined $_ } @$cur; unless (%ret) { return (); # No valid changes } else { return (_pname($ppaths) => {%ret}); } } # Sanitizes paragraph. # The init option means node creation. sub _sanitize_changes_paragraph { my $self = shift; my $cur = shift || {}; my $new = shift; my $pitem = shift; my $ppaths = shift; my %options = @_; return () unless ref $new eq 'HASH'; # Sanity check return () if $pitem->{obsolete}; return () unless 'write' eq ($pitem->{privilege} // ''); $self->_apply_defaults($cur, $pitem->{format}, init => ($options{init} and not $options{loading})); my %ret = map { unless (exists $pitem->{format}->{$_} and $pitem->{format}->{$_}) { (); # Sanity check: unknown parameter } else { # Resolve alias. my ($k, $o) = ($_, $_); do { ($k, $o) = ($o, $pitem->{format}->{$o}->{obsolete}); } while ($o and $pitem->{format}->{$o}); unless ($k eq $_) { $new->{$k} = $new->{$_}; delete $new->{$_}; } my $pii = $pitem->{format}->{$k}; my $ppi = [@$ppaths, $k]; my $newi = $new->{$k}; my $curi = $cur->{$k}; my @r; if ($pii->{occurrence} =~ /n$/) { if (ref $pii->{format} eq 'ARRAY') { @r = $self->_sanitize_changes_set($curi, $newi, $pii, $ppi); } else { @r = $self->_sanitize_changes_array($curi, $newi, $pii, $ppi, loading => $options{loading}); } } elsif (ref $pii->{format} eq 'HASH') { @r = $self->_sanitize_changes_paragraph( $curi, $newi, $pii, $ppi, init => (not defined $curi), loading => $options{loading} ); } else { @r = $self->_sanitize_changes_leaf($curi, $newi, $pii, $ppi); } # Omit removal if current configuration is already empty. (@r and not defined $r[1] and not defined $curi) ? () : @r; } } _keys($new, $pitem->{format}); while (my ($k, $v) = each %ret) { $cur->{$k} = $v; } # As soon as a required component is found to be removed, # the whole parameter instance is removed. return (_pname($ppaths) => undef) if grep { not $pitem->{format}->{$_}->{obsolete} and $pitem->{format}->{$_}->{occurrence} =~ /^1/ and not defined $cur->{$_} } _keys($pitem->{format}); # If all children are removed, remove parent. return (_pname($ppaths) => undef) unless grep { defined $_ } values %$cur; unless (%ret) { return (); # No valid changes } else { return (_pname($ppaths) => {%ret}); } } my %filters = ( canonic_domain => sub { my $self = shift; my $new = shift; return lc $new; #FIXME:how about i18n'ed domains? }, canonic_email => sub { my $self = shift; my $new = shift; return Sympa::Tools::Text::canonic_email($new); }, canonic_lang => sub { my $self = shift; my $new = shift; $new = Sympa::Language::canonic_lang($new); # be scalar return $new; }, lc => sub { my $self = shift; my $new = shift; return lc $new; }, ); # Sanitizes leaf. sub _sanitize_changes_leaf { my $self = shift; my $cur = shift; my $new = shift; my $pitem = shift; my $ppaths = shift; return () if ref $new eq 'ARRAY'; # Sanity check: Hashref or scalar return () if $pitem->{obsolete}; return () unless 'write' eq ($pitem->{privilege} // ''); # If the parameter corresponds to a scenario or a task, mark it # as changed if its name was changed. Example: 'subscribe'. if ($pitem->{scenario} or $pitem->{task}) { return () unless ref($new || {}) eq 'HASH'; # Sanity check $cur = ($cur || {})->{name}; $new = ($new || {})->{name}; } # Resolve synonym. if (defined $new and ref $pitem->{synonym} eq 'HASH') { my $synonym = $pitem->{synonym}->{$new}; $new = $synonym if defined $synonym; } # Apply filters. # Note: Erroneous values are overlooked and _not_ eliminated in this step. # We should eliminate them in the step of validation. if (defined $new) { my $f_new = $new; foreach my $filter (@{$pitem->{filters} || []}) { next unless ref $filters{$filter} eq 'CODE'; $f_new = $filters{$filter}->($self, $f_new); last unless defined $f_new; } $new = $f_new if defined $f_new; } if (Sympa::Tools::Data::smart_eq($cur, $new)) { return (); # Not changed } if ($pitem->{scenario} or $pitem->{task}) { return (_pname($ppaths) => {name => $new}); } else { return (_pname($ppaths) => $new); } } # Global validations examine the entire configuration for semantic errors or # requirements that can't be detected within a single paragraph. # # Error data is returned in a hashref with the usual keys. # sub _global_validations { {} } # Validates changes on list configuration. # # Parameters: # - $new: Hashref including changes. # - $errors: Error information, initially may be empty arrayref. # Returns: # - 'valid' if changes are valid; 'invalid' otherwise; # '' if no changes necessary; undef if internal error occurred. # - $new may be modified, if there are any omittable changes. # - Error information will be added to $errors. sub _validate_changes { my $self = shift; my $new = shift; my $errors = shift; my %options = @_; my $pinfo = $self->{_pinfo}; my $ret = 'valid'; foreach my $pname (_keys($new, $pinfo)) { my $newi = $new->{$pname}; my $pii = $pinfo->{$pname}; my $ppi = [$pname]; my $r; if ($pii->{occurrence} =~ /n$/) { $r = $self->_validate_changes_multiple($newi, $pii, $ppi, $errors); } elsif (ref $pii->{format} eq 'HASH') { $r = $self->_validate_changes_paragraph($newi, $pii, $ppi, $errors); } else { $r = $self->_validate_changes_leaf($newi, $pii, $ppi, $errors); } return undef unless defined $r; delete $new->{$pname} if $r eq 'omit'; $ret = 'invalid' if $r eq 'invalid'; } my %global_validations = %{$self->_global_validations || {}} unless $options{no_global_validations}; # review the entire new configuration as a whole foreach my $validation (CORE::keys %global_validations) { next unless ref $global_validations{$validation} eq 'CODE'; my ($error, $err_info) = $global_validations{$validation}->($self, $new); next unless $error; push @$errors, ['user', $error, $err_info]; $ret = 'invalid'; } return '' unless %$new; return $ret; } # Validates array or set. sub _validate_changes_multiple { my $self = shift; my $new = shift; my $pitem = shift; my $ppaths = shift; my $errors = shift; if (not defined $new and $pitem->{occurrence} =~ /^1/) { push @$errors, [ 'user', 'mandatory_parameter', {p_info => $pitem, p_paths => $ppaths} ]; return 'omit'; } my $ret = 'valid'; if (defined $new) { foreach my $i (sort { $a <=> $b } CORE::keys %$new) { my $newi = $new->{$i}; my $ppi = [@$ppaths, $i]; if (defined $newi) { my $r; if (ref $pitem->{format} eq 'HASH') { $r = $self->_validate_changes_paragraph($newi, $pitem, $ppi, $errors); } else { $r = $self->_validate_changes_leaf($newi, $pitem, $ppi, $errors); } return undef unless defined $r; delete $new->{$i} if $r eq 'omit'; $ret = 'invalid' if $r eq 'invalid'; } } return 'omit' unless %$new; } return $ret; } # Validates paragraph. sub _validate_changes_paragraph { my $self = shift; my $new = shift; my $pitem = shift; my $ppaths = shift; my $errors = shift; if (not defined $new and $pitem->{occurrence} =~ /^1/) { push @$errors, [ 'user', 'mandatory_parameter', {p_info => $pitem, p_paths => $ppaths} ]; return 'omit'; } my $ret = 'valid'; if (defined $new) { foreach my $key (_keys($new, $pitem->{format})) { my $pii = $pitem->{format}->{$key}; my $ppi = [@$ppaths, $key]; my $newi = $new->{$key}; my $r; if ($pii->{occurrence} =~ /n$/) { $r = $self->_validate_changes_multiple($newi, $pii, $ppi, $errors); } elsif (ref $pii->{format} eq 'HASH') { $r = $self->_validate_changes_paragraph($newi, $pii, $ppi, $errors); } else { $r = $self->_validate_changes_leaf($newi, $pii, $ppi, $errors); } return undef unless defined $r; delete $new->{$key} if $r eq 'omit'; $ret = 'invalid' if $r eq 'invalid'; } return 'omit' unless %$new; } return $ret; } sub _local_validations { {} } # Validates leaf. sub _validate_changes_leaf { my $self = shift; my $new = shift; my $pitem = shift; my $ppaths = shift; my $errors = shift; # If the parameter corresponds to a scenario or a task, mark it # as changed if its name was changed. Example: 'subscribe'. if ($pitem->{scenario} or $pitem->{task}) { $new = $new->{name} if defined $new; } if (not defined $new and $pitem->{occurrence} =~ /^1/) { push @$errors, [ 'user', 'mandatory_parameter', {p_info => $pitem, p_paths => $ppaths} ]; return 'omit'; } # Check that the new values have the right syntax. if (defined $new) { my $format = $pitem->{format}; if (ref $format eq 'ARRAY' and not grep { $new eq $_ } @$format) { push @$errors, [ 'user', 'syntax_errors', {p_info => $pitem, p_paths => $ppaths, value => $new} ]; return 'invalid'; } elsif (ref $format ne 'ARRAY' and not $new =~ /^$format$/) { push @$errors, [ 'user', 'syntax_errors', {p_info => $pitem, p_paths => $ppaths, value => $new} ]; return 'invalid'; } my %validations = %{$self->_local_validations || {}}; foreach my $validation (@{$pitem->{validations} || []}) { next unless ref $validations{$validation} eq 'CODE'; my ($error, $validity) = $validations{$validation}->($self, $new, $pitem, $ppaths); next unless $error; push @$errors, [ 'user', $error, {p_info => $pitem, p_paths => $ppaths, value => $new} ]; return $validity || 'invalid'; } } return 'valid'; } sub commit { my $self = shift; my $errors = shift || []; my $pinfo = $self->{_pinfo}; # Undefined {_config} means list creation. # Empty hashref means loading existing config. my $init = (not defined $self->{_config}); my $loading = ($self->{_config} and not %{$self->{_config}}); my $cur = $init ? {} : $self->{_config}; $self->_apply_defaults($cur, $pinfo, init => ($init and not $loading)); foreach my $pname (_keys($self->{_changes}, $pinfo)) { my $curi = $cur->{$pname}; my $newi = $self->{_changes}->{$pname}; my $pii = $pinfo->{$pname}; unless (defined $newi) { delete $cur->{$pname}; } elsif ($pii->{occurrence} =~ /n$/) { $curi = $cur->{$pname} = [] unless defined $curi; $self->_merge_changes_multiple($curi, $newi, $pii, loading => $loading); } elsif (ref $pii->{format} eq 'HASH') { my $init = (not defined $curi); $curi = $cur->{$pname} = {} if $init; $self->_merge_changes_paragraph( $curi, $newi, $pii, init => $init, loading => $loading ); } else { $cur->{$pname} = $newi; } } $self->{_config} = $cur if $init; # Update 'defaults' item to indicate default settings, for compatibility. #FIXME:Multiple levels of keys should be possible. foreach my $pname (_keys($self->{_changes}, $pinfo)) { if (defined $self->{_changes}->{$pname} or $pinfo->{$pname}->{internal}) { delete $self->{_config}->{defaults}->{$pname}; } else { $self->{_config}->{defaults}->{$pname} = 1; } } } sub _merge_changes_multiple { my $self = shift; my $cur = shift; my $new = shift; my $pitem = shift; my %options = @_; foreach my $i (reverse sort { $a <=> $b } CORE::keys %$new) { my $curi = $cur->[$i]; my $newi = $new->{$i}; unless (defined $new->{$i}) { splice @$cur, $i, 1; } elsif (ref $pitem->{format} eq 'HASH') { my $init = (not defined $curi); $curi = $cur->[$i] = {} if $init; $self->_merge_changes_paragraph( $curi, $newi, $pitem, init => $init, loading => $options{loading} ); } else { $cur->[$i] = $newi; } } # The set: Dedupe and sort. if (ref $pitem->{format} eq 'ARRAY') { my %elements = map { ($_ => 1) } grep { defined $_ } @$cur; @$cur = sort(CORE::keys %elements); } } # Merges changes on paragraph node. # The init option means node creation. sub _merge_changes_paragraph { my $self = shift; my $cur = shift; my $new = shift; my $pitem = shift; my %options = @_; $self->_apply_defaults($cur, $pitem->{format}, init => ($options{init} and not $options{loading})); foreach my $key (_keys($new, $pitem->{format})) { my $curi = $cur->{$key}; my $newi = $new->{$key}; my $pii = $pitem->{format}->{$key}; unless (defined $newi) { delete $cur->{$key}; } elsif ($pii->{occurrence} =~ /n$/) { $curi = $cur->{$key} = [] unless defined $curi; $self->_merge_changes_multiple($curi, $newi, $pii, loading => $options{loading}); } elsif (ref $pii->{format} eq 'HASH') { my $init = (not defined $curi); $curi = $cur->{$key} = {} if $init; $self->_merge_changes_paragraph( $curi, $newi, $pii, init => $init, loading => $options{loading} ); } else { $cur->{$key} = $newi; } } } sub get_id { my $that = shift->{context}; (ref $that eq 'Sympa::List') ? $that->get_id : (defined $that) ? $that : ''; } 1; __END__ =encoding utf-8 =head1 NAME Sympa::Config - List configuration =head1 SYNOPSIS use base qw(Sympa::Config); sub _schema { {...} } =head1 DESCRIPTION =head2 Methods =over =item new ( $that, [ config =E $initial_config ], [ copy =E 1 ], [ I =E I, ... ] ) I. Creates new instance of L object. Parameters: =over =item $that Context. An instance of L class, Robot or Site. =item config =E $initial_config Initial configuration. =over =item * When the context object will be initially created, C must be specified explicitly so that default parameter values will be completed. =item * When existing list will be instantiated and config will be loaded, C<{}> (default) would be specified so that default parameter values except optional ones (with occurrence C<'0-1'> or C<'0-n'>) will be completed. =item * Otherwise, default parameter values are completed only when the new paragraph node will be added by submit(). =back Note that initial configuration will never be sanitized. =item copy =E 1 Uses deep copy of initial configuration (see L) instead of real reference. =back =item get ( $ppath ) I. Gets copy of current value of parameter. Parameter: =over =item $ppath Parameter path, e.g.: C<'owner.0.email'> specifies "email" parameter of the first "owner" paragraph; C<'owner.0'> specifies the first "owner" paragraph; C<'owner'> specifies the array of all "owner" paragraph. =back Returns: Value of parameter. If parameter or value does not exist, returns C in scalar context and an empty list in array context. =item get_change ( $ppath ) I. Gets copy of submitted change on parameter. Parameter: =over =item $ppath Parameter path. See also get(). =back Returns: If value won't be changed, returns empty list in array context and C in scalar context. If value would be deleted, returns C. Changes on the array are given by hashref with keys as affected indexes of the array. =item get_changeset ( ) I. Gets all submitted changes. Note that returned value is the real reference to internal information. Any modifications might break it. =item get_schema ( [ I, ... ] ) I. Get configuration schema as hashref. =item keys ( [ $pname ] ) I. Gets parameter keys in order defined by schema. Parameter: =over =item $pname Full parameter name, e.g. C<'owner'>. If omitted or false value, returns keys of top-level parameters. =back Returns: List of keys. If parameter does not exist or it does not have sub-parameters, i.e. it is not the paragraph, empty list. =item submit ( $new, $user, \@errors, [ no_global_validations =E 1 ] ) I. Submits change and verifies it. Submission is done by: =over =item * Sanitizing changes: Omits unknown parameters, resolves parameter aliases, omits malformed change information, omits obsoleted parameters, omits changes on unwritable parameters, removes nodes under which required children nodes will be removed, resolves synonym of input values, canonicalizes inputs (see L), and omits identical changes. =item * Verifying changes: Omits removal of mandatory parameters, checks format of inputs, and performs additional validations (see L). =back Parameters: =over =item $new Changes to be submitted, hashref. =item $user Email of the user requesting submission. =item \@errors If errors occur, they will be pushed in this arrayref. Each element is arrayref C<[ I, I, I ]>: =item no_global_validations =E 1 If set, global validations are disabled. Global validations examine the entire configuration for semantic errors or requirements that can't be detected within a single paragraph. See also L. =over =item I One of C<'user'> (failure), C<'intern'> (internal failure) and C<'notice'> (successful notice). =item I A keyword to determine error. =item I Optional hashref with keys: C for schema item of parameter; C for elements of parameter path; C for erroneous value (optional). =back =back Returns: If no changes found (or all changes were omitted), an empty string C<''>. If any errors found in input, C<'invalid'>. Otherwise, C<'valid'>. In case any changes are submitted, changeset may be accessible by get_change() or get_changeset(). =item commit ( [ \@errors ] ) I. Merges changes set by sbumit() into actual configuration. Parameter: =over =item \@errors Arrayref. See \@errors in submit(). =back Returns: None. Errors will be stored in arrayref. =back =head3 Methods child classes should implement =over =item _schema ( ) I, I. TBD. =item _init_schema_item ( $pitem, $pnames, $subres, [ I =E I, ... ] ) I. TBD. =item _global_validations ( ) I. TBD. =item _local_validations ( ) I. TBD. =back =head2 Attribute Instance of L has following attribute. =over =item {context} Context, L instance. =back =head2 Structure of configuration Configuration on the memory is represented by a hashref, with its keys as node names and values as node values. =head3 Node types Each node of configuration has one of following four types. Some of them can include other type of nodes recursively. =over =item Set (multiple enumerated values) Arrayref. In the schema, defined with: =over =item * {occurrence}: C<'0-n'> or C<'1-n'>. =item * {format}: Arrayref. =back List of unique items not considering order. Items are scalars, and cannot be special values (scenario or task). The set cannot contain paragraphs, sets or arrays. =item Array (multiple values) Arrayref. In the schema, defined with: =over =item * {occurrence}: C<'0-n'> or C<'1-n'>. =item * {format}: Regexp or hashref. =back List of the same type of nodes in order. Type of all nodes can be one of paragraph, scalar or special value (scenario or task). The array cannot contain sets or arrays. =item Paragraph (structured value) Hashref. In the schema, defined with: =over =item * {occurrence}: If the node is an item of array, C<'0-n'> or C<'1-n'>. Otherwise, C<'0-1'> or C<'1'>. =item * {format}: Hashref. =back Compound node of one or more named nodes. Paragraph can contain any type of nodes, and each of their names and types are defined as member of {format} item in schema. =item Leaf (simple value) Scalar, or hashref for special value (scenario or task). In the schema, defined with: =over =item * {occurrence}: If the node is an item of array, C<'0-n'> or C<'1-n'>. Otherwise, C<'0-1'> or C<'1'>. =item * {format}: If the node is an item of array, regexp. Otherwise, regexp or arrayref. =back Scalar or special value (scenario or task). Leaf cannot contain any other nodes. =back =head2 Filters TBD. =head2 Validations TBD. =head1 SEE ALSO L. =head1 HISTORY L appeared on Sympa 6.2.33b.2. =cut