From 28a99d64417d9c9ccf4c56b97c1e3f52a24fc47e Mon Sep 17 00:00:00 2001 From: James Adams Date: Fri, 8 Jun 2018 15:13:51 +0100 Subject: [PATCH 1/3] ncm-sysconfig: Fix mixed source indentation --- ncm-sysconfig/src/main/perl/sysconfig.pm | 187 +++++++++++------------ 1 file changed, 93 insertions(+), 94 deletions(-) diff --git a/ncm-sysconfig/src/main/perl/sysconfig.pm b/ncm-sysconfig/src/main/perl/sysconfig.pm index 6e6c4d1d05..07df2dd516 100755 --- a/ncm-sysconfig/src/main/perl/sysconfig.pm +++ b/ncm-sysconfig/src/main/perl/sysconfig.pm @@ -11,100 +11,99 @@ use File::Basename; sub Configure { - - my ($self, $config) = @_; - - # Define paths for convenience. - my $base = "/software/components/sysconfig"; - - # The base directory for sysconfig files. - my $sysconfigdir = "/etc/sysconfig"; - - # Load configuration into a hash - my $sysconfig_config = $config->getElement($base)->getTree(); - - # Ensure that sysconfig directory exists. - mkpath($sysconfigdir,0,0755) unless (-e $sysconfigdir); - if (! -d $sysconfigdir) { - $self->error("$sysconfigdir isn't a directory or can't be created"); - return 1; - } - - # This will be a list of the configuration files managed by this component. - my %newcfg; - - # Read first the list of sysconfig files which have been - # previously managed by this component. These will have to - # be deleted if no longer in the configuration. - my %oldcfg; - if (-f "$sysconfigdir/ncm-sysconfig") { - open CONF, "<", "$sysconfigdir/ncm-sysconfig"; - while () { - chomp; - $oldcfg{$_} = 1; - } - } - - # Loop over all of the defined files, writing each as necessary. - if ( $sysconfig_config->{files} ) { - for my $file (sort keys %{$sysconfig_config->{files}}) { - - my $pairs = $sysconfig_config->{files}->{$file}; - - # Start with an empty file. - my $contents = ''; - - # Add the prologue if it exists. - if (defined($pairs->{"prologue"})) { - $contents .= $pairs->{"prologue"} . "\n"; - } - - # Loop over the pairs adding the information to the file. - for my $key (sort keys %{$pairs}) { - if ($key ne 'prologue' && $key ne 'epilogue') { - $contents .= $key . '=' . $pairs->{$key} . "\n"; - } - } - - # Add the epilogue if it exists. - if (defined($pairs->{"epilogue"})) { - $contents .= $pairs->{"epilogue"} . "\n"; - } - - # Now actually update the file, if needed. - - my $result = LC::Check::file("$sysconfigdir/$file", - backup => ".old", - contents => $contents, - ); - unless ( $result >= 0 ) { - $self->error("Error updating file $sysconfigdir/$file"); - } - - # Remove this file from the list of old configuration - # files add to the new configuration files. - delete($oldcfg{"$sysconfigdir/$file"}); - $newcfg{"$sysconfigdir/$file"} = 1; - } - } - - # Remove any old configuration files which haven't been updated. - for my $file (keys %oldcfg) { - unlink $file if (-e $file); - } - - # Write the list of managed configuration files. - if(open CONF, ">", "$sysconfigdir/ncm-sysconfig") { - for my $file (keys %newcfg) { - print CONF $file . "\n"; - } - close CONF; - } else { - $self->error("error writing file $sysconfigdir/ncm-sysconfig"); - return 1; - } - - return 0; + my ($self, $config) = @_; + + # Define paths for convenience. + my $base = "/software/components/sysconfig"; + + # The base directory for sysconfig files. + my $sysconfigdir = "/etc/sysconfig"; + + # Load configuration into a hash + my $sysconfig_config = $config->getElement($base)->getTree(); + + # Ensure that sysconfig directory exists. + mkpath($sysconfigdir,0,0755) unless (-e $sysconfigdir); + if (! -d $sysconfigdir) { + $self->error("$sysconfigdir isn't a directory or can't be created"); + return 1; + } + + # This will be a list of the configuration files managed by this component. + my %newcfg; + + # Read first the list of sysconfig files which have been + # previously managed by this component. These will have to + # be deleted if no longer in the configuration. + my %oldcfg; + if (-f "$sysconfigdir/ncm-sysconfig") { + open CONF, "<", "$sysconfigdir/ncm-sysconfig"; + while () { + chomp; + $oldcfg{$_} = 1; + } + } + + # Loop over all of the defined files, writing each as necessary. + if ( $sysconfig_config->{files} ) { + for my $file (sort keys %{$sysconfig_config->{files}}) { + + my $pairs = $sysconfig_config->{files}->{$file}; + + # Start with an empty file. + my $contents = ''; + + # Add the prologue if it exists. + if (defined($pairs->{"prologue"})) { + $contents .= $pairs->{"prologue"} . "\n"; + } + + # Loop over the pairs adding the information to the file. + for my $key (sort keys %{$pairs}) { + if ($key ne 'prologue' && $key ne 'epilogue') { + $contents .= $key . '=' . $pairs->{$key} . "\n"; + } + } + + # Add the epilogue if it exists. + if (defined($pairs->{"epilogue"})) { + $contents .= $pairs->{"epilogue"} . "\n"; + } + + # Now actually update the file, if needed. + + my $result = LC::Check::file("$sysconfigdir/$file", + backup => ".old", + contents => $contents, + ); + unless ( $result >= 0 ) { + $self->error("Error updating file $sysconfigdir/$file"); + } + + # Remove this file from the list of old configuration + # files add to the new configuration files. + delete($oldcfg{"$sysconfigdir/$file"}); + $newcfg{"$sysconfigdir/$file"} = 1; + } + } + + # Remove any old configuration files which haven't been updated. + for my $file (keys %oldcfg) { + unlink $file if (-e $file); + } + + # Write the list of managed configuration files. + if(open CONF, ">", "$sysconfigdir/ncm-sysconfig") { + for my $file (keys %newcfg) { + print CONF $file . "\n"; + } + close CONF; + } else { + $self->error("error writing file $sysconfigdir/ncm-sysconfig"); + return 1; + } + + return 0; } 1; # Required for PERL modules From 77e22cedb2e970c96061367721d0e0eba5e60cc2 Mon Sep 17 00:00:00 2001 From: James Adams Date: Fri, 8 Jun 2018 17:08:57 +0100 Subject: [PATCH 2/3] ncm-sysconfig: Modernise and add unit tests * Switch to CAF for all file interaction and declare NoAction support. * Use logger with FileWriters * Declare no-action support * Factor out filelist read/write to seperate functions. * Add a set of basic Configure unit tests. * Move pod into main source module * Switch nlists to dicts in pod * Move example to schema and remove redundant pod --- ncm-sysconfig/src/main/perl/sysconfig.pm | 130 ++++++++++-------- ncm-sysconfig/src/main/perl/sysconfig.pod | 46 ------- ncm-sysconfig/src/test/perl/configure.t | 29 ++++ .../src/test/resources/configure.pan | 6 + 4 files changed, 106 insertions(+), 105 deletions(-) delete mode 100755 ncm-sysconfig/src/main/perl/sysconfig.pod create mode 100644 ncm-sysconfig/src/test/perl/configure.t create mode 100644 ncm-sysconfig/src/test/resources/configure.pan diff --git a/ncm-sysconfig/src/main/perl/sysconfig.pm b/ncm-sysconfig/src/main/perl/sysconfig.pm index 07df2dd516..97fa11d385 100755 --- a/ncm-sysconfig/src/main/perl/sysconfig.pm +++ b/ncm-sysconfig/src/main/perl/sysconfig.pm @@ -1,52 +1,78 @@ #${PMcomponent} -use parent qw(NCM::Component); +=head1 NAME + +sysconfig: management of sysconfig files + +=head1 DESCRIPTION + +The I component manages system configuration files in +C<< /etc/sysconfig >> . These are files which contain key-value pairs. +However there is the possibility to add verbatim text either before or after the key-value pair definitions. + +=cut + +use parent qw(NCM::Component CAF::Path); + our $EC = LC::Exception::Context->new->will_store_all; -use LC::Check; +our $NoActionSupported = 1; -use File::Path; -use File::Basename; +use Readonly; +Readonly my $QUOTE => "\""; +Readonly my $SYSCONFIGDIR => "/etc/sysconfig"; # The base directory for sysconfig files. -sub Configure +sub filelist_read { - my ($self, $config) = @_; + my ($self) = @_; + + # Read first the list of sysconfig files which have been + # previously managed by this component. These will have to + # be deleted if no longer in the configuration. + my %filelist; + my $fh = CAF::FileReader->open("$SYSCONFIGDIR/ncm-sysconfig", log => $self); + while (my $line = <$fh>) { + chomp($line); + $filelist{$line} = 1; + } + $fh->close(); - # Define paths for convenience. - my $base = "/software/components/sysconfig"; + return %filelist; +} - # The base directory for sysconfig files. - my $sysconfigdir = "/etc/sysconfig"; +sub filelist_write +{ + my ($self, %filelist) = @_; + + # Write the list of managed configuration files. + my $fh = CAF::FileWriter->open("$SYSCONFIGDIR/ncm-sysconfig", log => $self); + for my $file (keys %filelist) { + print $fh "$file\n"; + } + $fh->close(); + + return 1; +} + +sub Configure +{ + my ($self, $config) = @_; # Load configuration into a hash - my $sysconfig_config = $config->getElement($base)->getTree(); + my $sysconfig_config = $config->getTree($self->prefix()); # Ensure that sysconfig directory exists. - mkpath($sysconfigdir,0,0755) unless (-e $sysconfigdir); - if (! -d $sysconfigdir) { - $self->error("$sysconfigdir isn't a directory or can't be created"); - return 1; - } + $self->directory($SYSCONFIGDIR, owner=>0, group=>0, mode=>0755); # This will be a list of the configuration files managed by this component. - my %newcfg; + my %files_managed; - # Read first the list of sysconfig files which have been - # previously managed by this component. These will have to - # be deleted if no longer in the configuration. - my %oldcfg; - if (-f "$sysconfigdir/ncm-sysconfig") { - open CONF, "<", "$sysconfigdir/ncm-sysconfig"; - while () { - chomp; - $oldcfg{$_} = 1; - } - } + my %files_previous = $self->filelist_read(); # Loop over all of the defined files, writing each as necessary. if ( $sysconfig_config->{files} ) { - for my $file (sort keys %{$sysconfig_config->{files}}) { + foreach my $file (sort keys %{$sysconfig_config->{files}}) { my $pairs = $sysconfig_config->{files}->{$file}; @@ -54,56 +80,42 @@ sub Configure my $contents = ''; # Add the prologue if it exists. - if (defined($pairs->{"prologue"})) { - $contents .= $pairs->{"prologue"} . "\n"; + if (defined($pairs->{prologue})) { + $contents .= "$pairs->{prologue}\n"; } # Loop over the pairs adding the information to the file. - for my $key (sort keys %{$pairs}) { + for my $key (sort keys %$pairs) { if ($key ne 'prologue' && $key ne 'epilogue') { - $contents .= $key . '=' . $pairs->{$key} . "\n"; + $contents .= "$key=$pairs->{$key}\n"; } } # Add the epilogue if it exists. - if (defined($pairs->{"epilogue"})) { - $contents .= $pairs->{"epilogue"} . "\n"; + if (defined($pairs->{epilogue})) { + $contents .= "$pairs->{epilogue}\n"; } # Now actually update the file, if needed. - - my $result = LC::Check::file("$sysconfigdir/$file", - backup => ".old", - contents => $contents, - ); - unless ( $result >= 0 ) { - $self->error("Error updating file $sysconfigdir/$file"); - } + my $fh = CAF::FileWriter->open("$SYSCONFIGDIR/$file", backup => ".old", log => $self); + print $fh $contents; + $fh->close(); # Remove this file from the list of old configuration # files add to the new configuration files. - delete($oldcfg{"$sysconfigdir/$file"}); - $newcfg{"$sysconfigdir/$file"} = 1; + delete($files_previous{"$SYSCONFIGDIR/$file"}); + $files_managed{"$SYSCONFIGDIR/$file"} = 1; } } # Remove any old configuration files which haven't been updated. - for my $file (keys %oldcfg) { - unlink $file if (-e $file); + for my $file (keys %files_previous) { + $self->cleanup($file); } - # Write the list of managed configuration files. - if(open CONF, ">", "$sysconfigdir/ncm-sysconfig") { - for my $file (keys %newcfg) { - print CONF $file . "\n"; - } - close CONF; - } else { - $self->error("error writing file $sysconfigdir/ncm-sysconfig"); - return 1; - } + $self->filelist_write(%files_managed); - return 0; + return 1; } -1; # Required for PERL modules +1; # Required for PERL modules diff --git a/ncm-sysconfig/src/main/perl/sysconfig.pod b/ncm-sysconfig/src/main/perl/sysconfig.pod deleted file mode 100755 index adf9e68bf0..0000000000 --- a/ncm-sysconfig/src/main/perl/sysconfig.pod +++ /dev/null @@ -1,46 +0,0 @@ -# ${license-info} -# ${developer-info} -# ${author-info} - - -=head1 NAME - -sysconfig: management of sysconfig files - -=head1 DESCRIPTION - -The I component manages system configuration files in -C<< /etc/sysconfig >> . These are files which contain key-value pairs. -However there is the possibility to add verbatim text either -before or after the key-value pair definitions. - -=head1 FILE DEFINITIONS - -=over - -=item * C<< /software/components/sysconfig/files >> - -This is an nlist which has the file name (unescaped) as the key, and -the content information as the value. The value is an nlist. - -=item * C<< /software/components/sysconfig/files// >> - -This is a nlist containing key-value pairs. Both are strings. -There are two special keys C<< prologue >> and C<< epilogue >> which contain -text which will be copied verbatim into the file before or after -the pair definitions, respectively. - -=back - -=head1 EXAMPLE - - "/software/components/sysconfig/files/scfg" = - nlist("epilogue", "export LANG=C", - "KEY", "VALUE"); - -This will create the file C<< /etc/sysconfig/scfg >> which contains: - - KEY=VALUE - export LANG=C - -=cut diff --git a/ncm-sysconfig/src/test/perl/configure.t b/ncm-sysconfig/src/test/perl/configure.t new file mode 100644 index 0000000000..5d075a7b9b --- /dev/null +++ b/ncm-sysconfig/src/test/perl/configure.t @@ -0,0 +1,29 @@ +use strict; +use warnings; +use Test::More; +use Test::Quattor qw(configure); +use NCM::Component::sysconfig; + +$CAF::Object::NoAction = 1; + +my $cmp = NCM::Component::sysconfig->new('sysconfig'); +my $cfg = get_config_for_profile('configure'); + +# Set up list of previously managed files +set_file_contents('/etc/sysconfig/ncm-sysconfig', "/etc/sysconfig/delete_me\n/etc/sysconfig/examplefile\n"); + +# Set up existing files with garbage contents +set_file_contents('/etc/sysconfig/delete_me', "DELETE_ME=delete_me\n"); +set_file_contents('/etc/sysconfig/examplefile', "NOPE=nope_nope\n"); + +is($cmp->Configure($cfg), 1, "Does the component run correctly with a test profile?"); + +isa_ok(get_file('/etc/sysconfig/ncm-sysconfig'), "CAF::FileWriter", "Is the file-handle for the list of managed files the correct class?"); +is(get_file_contents('/etc/sysconfig/ncm-sysconfig'), "/etc/sysconfig/examplefile\n", "Is the list of managed files updated correctly?"); + +is(get_file_contents('/etc/sysconfig/delete_me'), undef, "Has the previously managed file been deleted?"); + +isa_ok(get_file('/etc/sysconfig/examplefile'), "CAF::FileWriter", "Is the file-handle for the example file the correct class?"); +is(get_file_contents('/etc/sysconfig/examplefile'), "key1=testvalue\nkey2=valuetest\n", "Does the example file have the correct contents?"); + +done_testing(); diff --git a/ncm-sysconfig/src/test/resources/configure.pan b/ncm-sysconfig/src/test/resources/configure.pan new file mode 100644 index 0000000000..8ce521239a --- /dev/null +++ b/ncm-sysconfig/src/test/resources/configure.pan @@ -0,0 +1,6 @@ +object template configure; + +prefix '/software/components/sysconfig'; + +'files/examplefile/key1' = 'testvalue'; +'files/examplefile/key2' = 'valuetest'; From aa5205529059f177db2f07e88b4fc3c4639618d1 Mon Sep 17 00:00:00 2001 From: James Adams Date: Mon, 11 Jun 2018 17:30:09 +0100 Subject: [PATCH 3/3] ncm-sysconfig: Update and clarify schema * Break apart schema into sub-types for clarification * Disallow empty values * Make use of schema in tests * Add annotations based on paraphrased description from pod. --- .../main/pan/components/sysconfig/schema.pan | 26 ++++++++++++++++--- .../src/test/resources/configure.pan | 2 ++ 2 files changed, 24 insertions(+), 4 deletions(-) diff --git a/ncm-sysconfig/src/main/pan/components/sysconfig/schema.pan b/ncm-sysconfig/src/main/pan/components/sysconfig/schema.pan index a2740866fc..6a462e35d4 100755 --- a/ncm-sysconfig/src/main/pan/components/sysconfig/schema.pan +++ b/ncm-sysconfig/src/main/pan/components/sysconfig/schema.pan @@ -2,16 +2,34 @@ # ${developer-info} # ${author-info} - declaration template components/sysconfig/schema; include 'quattor/schema'; +@documentation{ +Contents of a sysconfig file modelled as a dict of key-value pairs. + +Two reserved keys `prologue` and `epilogue` are treated specially, +their values will be copied verbatim into the file before or after the key-pair definitions. + +Example: + +'/software/components/sysconfig/files/scfg' = dict( + 'epilogue', 'export LANG=C', + 'KEY', 'VALUE', +); + +This will create the file `/etc/sysconfig/scfg` which contains: + +KEY=VALUE +export LANG=C +} +type component_sysconfig_file = string(1..){}; + type component_sysconfig = { include structure_component - 'files' ? string{}{} + @{ dict of dicts with a file name as the first key and the contents of each file as the child dict. } + 'files' ? component_sysconfig_file{} }; bind '/software/components/sysconfig' = component_sysconfig; - - diff --git a/ncm-sysconfig/src/test/resources/configure.pan b/ncm-sysconfig/src/test/resources/configure.pan index 8ce521239a..4a41d6130f 100644 --- a/ncm-sysconfig/src/test/resources/configure.pan +++ b/ncm-sysconfig/src/test/resources/configure.pan @@ -1,5 +1,7 @@ object template configure; +include 'components/sysconfig/schema'; + prefix '/software/components/sysconfig'; 'files/examplefile/key1' = 'testvalue';