The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.

USING PERL TO ADMINISTER LINUX SORCERER

I am a confirmed Sorcerer Linux SA ( see http://sorcerer.wox.org ). For several years now I have been steadily building up my base of Sorcerer installations ( eat your heart out Billy G. ), and have found it to be fast, lean, reliable, and down right fun to maintain. My users are happy, so I am happy.

I have been seriously using Linux for about four years now, having come from the Microsoft world. Before I started using and administrating Linux I was one of Microsoft's minions and thought if the software didn't cost a lot of money and have the funny 'window' picture on the box, it was really just for amateurs and geeks. Not so, as I and many of you have discovered.

Perl, for me, is a different story. I have been a 'serious' perl monger for almost ten years, reaching back to my Windows days. I am still more productive in perl than bash. When I started writing perl scripts to help me administer Sorcerer, I ran into the problem that the environment variables on which Sorcerer so heavily depends, are 'sourced' into the bash environment from /etc/sorcery/config. It is very hard to access them from perl. And variables that are bash arrays, good luck! Try this:

  # . /etc/sorcery/config && \
        perl -e 'print "$_ => $ENV{$_}\n" for sort keys %ENV'

Look at the result. Yep, nary a Sorcerer variable appears. Now try:

  # . /etc/sorcery/config
  # export GRIMOIRE
  # perl -e 'print "$_ => $ENV{$_}\n" for sort keys %ENV'

Yep, GRIMORE appears in the output. So I had to write bash scripts to source /etc/sorcery/config, export the variables I wanted to use, and run my perl script. That got old quickly. And I still didn't have a good way to get to bash arrays.

I have developed a solution, based on the CPAN module Env::Bash. Please take a moment to glance at the documentation before continuing.

A SIMPLE EXAMPLE

I have written a small, more or less useless, perl script using Env::Bash with Sorcerer. The source is scripts/show-spell.pl in the module's distribution. The script can list the spells in the grimoire and/or display spell details:

  perl scripts/show-spell [-d] [-l] [<spell> ...]

where:

  -d      shows internal debugging information
  -l      lists spells in the grimoire
  <spell> display details of one or more spells.

If the script is started without any arguments, spell linux is displayed.

Here is a blow-by-blow description of show-spell.pl

Starting

  #!/usr/bin/perl
  
  use warnings;
  use strict;
  use Env::Bash;
  use Data::Dumper;
  use Getopt::Std;
  
  my %opt;
  unless (getopts('dl', \%opt)) {
      usage();
  }

Pretty standard; options are handled too.

tie a HASH to the Environment

  # tie a hash to /etc/sorcery/config, no ForceArray
  my %env = ();
  tie %env, "Env::Bash", Source => "/etc/sorcery/config",
      Debug => $opt{d};

This is the easiest way to interface to Env::Bash ( there is a simple, standard interface and an oo interface which are fully discussed in the module's documentation ). The tie statement says to interface to the environment through hash %env, with the option Source ( the script or list of scripts to source ), and conditionally set internal debugging. More on ForceArray below.

Find the GRIMOIRE directory

  # find the GRIMOIRE directory
  my $grimoire = $env{GRIMOIRE} || die "cannot find GRIMOIRE\n";

This is the first real use of the module; the grimoire directory ( defined in /etc/sorcery/config ), is returned.

Perl code to list the GRIMOIRE and display spells

  # display spells in the grimoire if option -l
  if( $opt{l} ) {
      print "---spells in grimoire-------------------------\n";
      my @spells = ();
      for my $spell( <$grimoire/*> ) {
          $spell =~ s,.*/,,;
          push @spells, $spell;
      }
      print "$_\n" for sort @spells;
  }
  
  # show spells on command line, or linux if none given
  show_spell( $_ ) for @ARGV;
  show_spell( 'linux' ) unless @ARGV || $opt{l};

Just perl code.

show-spell subroutine

  sub show_spell
  {
      my $spell = shift;
    
      # find the spell and DETAILS
      unless( -e "$grimoire/$spell" ) {
          warn "Spell '$spell' not found.\n";
          return;
      }
      my $details = -d _ ? "$grimoire/$spell/DETAILS" : "$grimoire/$spell";
      unless( -e "$details" ) {
          warn "Spell '$spell' DETAILS not found.\n";
          return;
      }
  
      # tie a hash to /etc/sorcery/config and DETAILS w/ForceArray
      my %env = ();
      tie %env, "Env::Bash",
      [],
      Source => [ "/etc/sorcery/config", $details ],
      Debug => $opt{d};
  
      print "---$spell-------------------------------------\n";
      show_detail( VERSION   => \%env );
      show_detail( CATEGORY  => \%env );
      show_detail( ATTRIBUTE => \%env );
      show_detail( SOURCE    => \%env );
      show_detail( URL       => \%env );
      show_detail( HOMEPAGE  => \%env );
      show_detail( REQ       => \%env );
      show_detail( PROTECT   => \%env );
      show_detail( ESTIMATE  => \%env );
      show_detail( DESC      => \%env );
  }

Here we do another tie for two reasons: 1) we want to source the spell's DETAILS script as well as /etc/sorcery/config, and 2) we want to be able to access any variables that are bash arrays - like VERSION. The Source option to the tie is a list reference - that's how Env::Bash knows how to use more than one source script. To access the environment, the module constructs a mini bash script, which in this case would be something like:

  #!/bin/sh
  . /etc/sorcery/config;. /var/state/sorcery/grimoire/<spell>/DETAILS;set

The script is run by forking via backtics and the set output is parsed to get a list of environment names. Turn on debugging ( -d ) if you really want to see the the script.

To access bash arrays, the ForceArray option ( shortcut: [] ) is specified. The module again creates and runs a bash script that returns all elements of any bash arrays; the results are stored in the tied hash as an array reference, whether or not the variable is a bash array. Again, for the more curious, look at what's happening by turning on debugging.

Show spell details

  sub show_detail
  {
      my( $name, $env ) = @_;
  
      # get the requested detail ( return is an array because the
      # the $env hash was tied with ForceArray ( [] ).
      my $values = $env->{$name};
  
      # print each detail
      my $eq = '=';
      for my $value( @$values ) {
          $value = join( "\n".' ' x 14, split /\n/, $value )
              if $value =~ /\n/s;
          printf "%12s%1s\"%s\"\n", $name, $eq, $value;
          $name = $eq = '';
      }
  }

Now we get the variable from the tied hash. As mentioned above, each hash element is an array reference ( because ForceArray was specified ). The resulting values are manipulated and printed.

usage subroutine

  sub usage
  {
      my $progname = $0;
      $progname =~ s,.*/,,;    # only basename left in progname
      die "Usage: $progname [-d] [-l] <spell> [<spell> ...]\n";
  }

Summary

OK, so you now have a script the demonstrates the use of Env::Bash. I admit it's slower and much harder than a few bash ls and cat commands, but that's not the point. Env::Bash has proved to open up Sorcerer to perl; I have written some perl scripts that are, in fact, useful; I will try to get them posted my my home page ( currently not available as I write this - December 2004 ) and let you know on the Sorcerer list. I hope you can find some use for Env::Bash.

AUTHOR

Beau E. Cox, <beaucox@hawaii.rr.com>.

COPYRIGHT AND LICENSE

Copyright (C) 2004 by Beau E. Cox.

This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself, either Perl version 5.8.6 or, at your option, any later version of Perl 5 you may have available.