package Anvil::Tools; # # This is the "root" package that manages the sub modules and controls access to their methods. # BEGIN { our $VERSION = "3.0.0"; # This suppresses the 'could not find ParserDetails.ini in /PerlApp/XML/SAX' warning message in # XML::Simple calls. #$ENV{HARNESS_ACTIVE} = 1; } use strict; use warnings; use Scalar::Util qw(weaken isweak); use Time::HiRes; use Data::Dumper; use CGI; my $THIS_FILE = "Tools.pm"; ### Methods; # data # environment # nice_exit # _add_hash_reference # _anvil_version # _hostname # _make_hash_reference # _set_defaults # _set_paths # _short_hostname use utf8; binmode(STDERR, ':encoding(utf-8)'); binmode(STDOUT, ':encoding(utf-8)'); # I intentionally don't use EXPORT, @ISA and the like because I want my "subclass"es to be accessed in a # somewhat more OO style. I know some may wish to strike me down for this, but I like the idea of accessing # methods via their containing module's name. (A La: C<< $anvil->Module->method >> rather than C<< $anvil->method >>). use Anvil::Tools::Account; use Anvil::Tools::Alert; use Anvil::Tools::Database; use Anvil::Tools::Convert; use Anvil::Tools::Get; use Anvil::Tools::Log; use Anvil::Tools::Remote; use Anvil::Tools::Storage; use Anvil::Tools::System; use Anvil::Tools::Template; use Anvil::Tools::Words; use Anvil::Tools::Validate; =pod =encoding utf8 =head1 NAME Anvil::Tools Provides a common oject handle to all Anvil::Tools::* module methods and handles invocation configuration. =head1 SYNOPSIS use Anvil::Tools; # Get a common object handle on all Anvil::Tools::* modules. my $anvil = Anvil::Tools->new(); # Again, but this time sets some initial values in the '$anvil->data' hash. my $anvil = Anvil::Tools->new( { data => { foo => "", bar => [], baz => {}, }, }); # This example gets the handle and also sets the default user and log # languages as Japanese, sets a custom log file and sets the log level to # '2'. my $anvil = Anvil::Tools->new( { 'Log' => { user_language => "jp", log_language => "jp" level => 2, }, }); =head1 DESCRIPTION The Anvil::Tools module and all sub-modules are designed for use by Alteeve-based applications. It can be used as a general framework by anyone interested. Core features are; * Supports per user, per logging language selection where translations from from XML-formatted "String" files that support UTF8 and variable substitutions. * Support for command-line and HTML output. Skinning support for HTML-based user interfaces. * Redundant database access, resynchronization and archiving. * Highly-native with minimal use of external perl modules and compiled code. =head1 METHODS Methods in the core module; =cut # The constructor through which all other module's methods will be accessed. sub new { my $class = shift; my $parameter = shift; my $self = { HANDLE => { ACCOUNT => Anvil::Tools::Account->new(), ALERT => Anvil::Tools::Alert->new(), DATABASE => Anvil::Tools::Database->new(), CONVERT => Anvil::Tools::Convert->new(), GET => Anvil::Tools::Get->new(), LOG => Anvil::Tools::Log->new(), REMOTE => Anvil::Tools::Remote->new(), STORAGE => Anvil::Tools::Storage->new(), SYSTEM => Anvil::Tools::System->new(), TEMPLATE => Anvil::Tools::Template->new(), WORDS => Anvil::Tools::Words->new(), VALIDATE => Anvil::Tools::Validate->new(), # This is to be removed before development ends. log_file => "", }, DATA => {}, ENV_VALUES => { ENVIRONMENT => 'cli', }, HOST => { # This is the host's UUID. It should never be manually set. UUID => "", ANVIL_VERSION => "", }, }; # Bless you! bless $self, $class; # This isn't needed, but it makes the code below more consistent with and portable to other modules. my $anvil = $self; weaken($anvil); # Helps avoid memory leaks. See Scalar::Utils # Get a handle on the various submodules $anvil->Account->parent($anvil); $anvil->Alert->parent($anvil); $anvil->Database->parent($anvil); $anvil->Convert->parent($anvil); $anvil->Get->parent($anvil); $anvil->Log->parent($anvil); $anvil->Remote->parent($anvil); $anvil->Storage->parent($anvil); $anvil->System->parent($anvil); $anvil->Template->parent($anvil); $anvil->Words->parent($anvil); $anvil->Validate->parent($anvil); # Set some system paths and system default variables $anvil->_set_paths(); $anvil->_set_defaults(); # Record the start time. $anvil->data->{ENV_VALUES}{START_TIME} = Time::HiRes::time; # Set passed parameters if needed. my $debug = 3; if (ref($parameter) eq "HASH") { # Local parameters... if ($parameter->{debug}) { $debug = $parameter->{debug}; } if ($parameter->{log_secure}) { $anvil->Log->secure({set => $parameter->{log_secure}}); } } elsif ($parameter) { # Um... print $THIS_FILE." ".__LINE__."; Anvil::Tools->new() invoked with an invalid parameter. Expected a hash reference, but got: [$parameter]\n"; exit(1); } # If the user passed a custom log level, sit it now. if ($parameter->{log_level}) { $anvil->Log->level({set => $parameter->{log_level}}); } # This will help clean up if we catch a signal. $SIG{INT} = sub { $anvil->catch_sig({signal => "INT"}); }; $SIG{TERM} = sub { $anvil->catch_sig({signal => "TERM"}); }; # This sets the environment this program is running in. if ($ENV{SERVER_NAME}) { $anvil->environment("html"); # There is no PWD environment variable, so we'll use 'DOCUMENT_ROOT' as 'PWD' $ENV{PWD} = $ENV{DOCUMENT_ROOT}; } else { $anvil->environment("cli"); } # Setup my '$anvil->data' hash right away so that I have a place to store the strings hash. $anvil->data($parameter->{data}) if $parameter->{data}; # Initialize the list of directories to seach. $anvil->Storage->search_directories({debug => $debug, initialize => 1}); # I need to read the initial words early. $anvil->Words->read({debug => $debug, file => $anvil->data->{path}{words}{'words.xml'}}); # If the local './tools.conf' file exists, read it in. if (-r $anvil->data->{path}{configs}{'anvil.conf'}) { $anvil->Storage->read_config({debug => $debug, file => $anvil->data->{path}{configs}{'anvil.conf'}}); ### TODO: Should anvil.conf override parameters? # Let parameters override config file values. if ($parameter->{log_level}) { $anvil->Log->level({set => $parameter->{log_level}}); } if ($parameter->{log_secure}) { $anvil->Log->secure({set => $parameter->{log_secure}}); } } # Get the local host UUID. $anvil->Get->host_uuid({debug => $debug}); # Read in any command line switches. $anvil->Get->switches({debug => $debug}); # Read in the local Anvil! version. #... return ($self); } ############################################################################################################# # Public methods # ############################################################################################################# =head2 data This is the method used to access the main hash reference that all user-accessible values are stored in. This includes words, configuration file variables and so forth. When called without an argument, it returns the existing '$anvil->data' hash reference. my $anvil = $anvil->data(); When called with a hash reference as the argument, it sets '$anvil->data' to the new hash. my $some_hash = {}; my $anvil = $anvil->data($some_hash); Data can be entered into or access by treating '$anvil->data' as a normal hash reference. my $anvil = Anvil::Tools->new( { data => { foo => "", bar => [6, 4, 12], baz => { animal => "Cat", thing => "Boat", }, }, }); # Copy the 'Cat' value into the $animal variable. my $animal = $anvil->data->{baz}{animal}; # Set 'A thing' in 'foo'. $anvil->data->{foo} = "A thing"; The C<< $anvil >> variable is set inside all modules and acts as shared storage for variables, values and references in all modules. It acts as the core storage for most applications using Anvil::Tools. =cut sub data { my ($anvil) = shift; # Pick up the passed in hash, if any. $anvil->{DATA} = shift if $_[0]; return ($anvil->{DATA}); } =head2 environment This is the method used to check or set whether the program is outputting to command line or a browser. When called without an argument, it returns the current environment. if ($anvil->environment() eq "cli") { # format for STDOUT } elsif ($anvil->environment() eq "html") { # Use the template system to output HTML } When called with a string as the argument, that string will be set as the environment string. $anvil->environment("cli"); Technically, any string can be used, however only 'cli' or 'html' are used by convention. =cut sub environment { my ($anvil) = shift; weaken($anvil); # Pick up the passed in delimiter, if any. if ($_[0]) { $anvil->{ENV_VALUES}{ENVIRONMENT} = shift; # Load the CGI stuff if we're in a browser if ($anvil->{ENV_VALUES}{ENVIRONMENT} eq "html") { CGI::Carp->import(qw(fatalsToBrowser)); } } return ($anvil->{ENV_VALUES}{ENVIRONMENT}); } =head2 nice_exit This is a simple method to exit cleanly, closing database connections and exiting with the set exit code. Parameters; =head3 exit_code (optional) If set, this will be the exit code. The default is to exit with code C<< 0 >>. =cut sub nice_exit { my $self = shift; my $parameter = shift; my $anvil = $self; my $exit_code = defined $parameter->{exit_code} ? $parameter->{exit_code} : 0; # Close database connections (if any). $anvil->Database->disconnect(); # Report the runtime. my $end_time = Time::HiRes::time; my $run_time = $end_time - $anvil->data->{ENV_VALUES}{START_TIME}; $anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { 's1:ENV_VALUES::START_TIME' => $anvil->data->{ENV_VALUES}{START_TIME}, 's2:end_time' => $end_time, 's3:run_time' => $run_time, }}); $anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, key => "log_0135", variables => { runtime => $run_time }}); # Close the log file. if ($anvil->{HANDLE}{log_file}) { close $anvil->{HANDLE}{log_file}; $anvil->{HANDLE}{log_file} = ""; } exit($exit_code); } ############################################################################################################# # Public methods used to access sub modules. # ############################################################################################################# =head1 Submodule Access Methods The methods below are used to access methods of submodules using 'C<< $anvil->Module->method() >>'. =cut =head2 Account Access the C methods via 'C<< $anvil->Alert->method >>'. =cut sub Account { my $self = shift; return ($self->{HANDLE}{ACCOUNT}); } =head2 Alert Access the C methods via 'C<< $anvil->Alert->method >>'. =cut sub Alert { my $self = shift; return ($self->{HANDLE}{ALERT}); } =head2 Database Access the C methods via 'C<< $anvil->Database->method >>'. =cut sub Database { my $self = shift; return ($self->{HANDLE}{DATABASE}); } =head2 Convert Access the C methods via 'C<< $anvil->Convert->method >>'. =cut sub Convert { my $self = shift; return ($self->{HANDLE}{CONVERT}); } =head2 Get Access the C methods via 'C<< $anvil->Get->method >>'. =cut sub Get { my $self = shift; return ($self->{HANDLE}{GET}); } =head2 Log Access the C methods via 'C<< $anvil->Log->method >>'. =cut sub Log { my $self = shift; return ($self->{HANDLE}{LOG}); } =head2 Remote Access the C methods via 'C<< $anvil->Remote->method >>'. =cut sub Remote { my $self = shift; return ($self->{HANDLE}{REMOTE}); } =head2 Storage Access the C methods via 'C<< $anvil->Storage->method >>'. =cut sub Storage { my $self = shift; return ($self->{HANDLE}{STORAGE}); } =head2 System Access the C methods via 'C<< $anvil->System->method >>'. =cut sub System { my $self = shift; return ($self->{HANDLE}{SYSTEM}); } =head2 Template Access the C methods via 'C<< $anvil->Template->method >>'. =cut sub Template { my $self = shift; return ($self->{HANDLE}{TEMPLATE}); } =head2 Words Access the C methods via 'C<< $anvil->Words->method >>'. =cut sub Words { my $self = shift; return ($self->{HANDLE}{WORDS}); } =head2 Validate Access the C methods via 'C<< $anvil->Validate->method >>'. =cut sub Validate { my $self = shift; return ($self->{HANDLE}{VALIDATE}); } =head1 Private Functions; These methods generally should never be called from a program using Anvil::Tools. However, we are not your boss. =cut ############################################################################################################# # Private methods # ############################################################################################################# =head2 _add_hash_reference This is a helper to the '$anvil->_make_hash_reference' method. It is called each time a new string is to be created as a new hash key in the passed hash reference. NOTE: Contributed by Shaun Fryer and Viktor Pavlenko by way of Toronto Perl Mongers. =cut sub _add_hash_reference { my $self = shift; my $href1 = shift; my $href2 = shift; for my $key (keys %$href2) { if (ref $href1->{$key} eq 'HASH') { $self->_add_hash_reference( $href1->{$key}, $href2->{$key} ); } else { $href1->{$key} = $href2->{$key}; } } } =head2 _anvil_version =cut sub _anvil_version { my $self = shift; my $anvil = $self; if ($anvil->{HOST}{ANVIL_VERSION} eq "") { # Try to read the local Anvil! version. $anvil->{HOST}{ANVIL_VERSION} = $anvil->Get->anvil_version(); } return($anvil->{HOST}{ANVIL_VERSION}); } =head2 _hostname This returns the (full) hostname for the machine this is running on. =cut sub _hostname { my $self = shift; my $anvil = $self; my $hostname = ""; if ($ENV{HOSTNAME}) { # We have an environment variable, so use it. $hostname = $ENV{HOSTNAME}; } else { # The environment variable isn't set. Call 'hostname' on the command line. $hostname = $anvil->System->call({shell_call => $anvil->data->{path}{exe}{hostname}}); } return($hostname); } =head2 _get_hash_reference This is called when we need to parse a double-colon separated string into two or more elements which represent keys in the 'C<< $anvil->data >>' hash. Once suitably split up, the value is read and returned. For example; $anvil->data->{foo}{bar} = "baz"; my $value = $anvil->_get_hash_reference({ key => "foo::bar" }); The 'C<< $value >>' now contains "C<< baz >>". NOTE: If the key is not found, 'C<< undef >>' is returned. Parameters; =head3 key (required) This is the key to return the value for. If it is not passed, or if it does not have 'C<< :: >>' in it, 'C<< undef >>' will be returned. =cut sub _get_hash_reference { # 'href' is the hash reference I am working on. my $self = shift; my $parameter = shift; my $anvil = $self; #print "$THIS_FILE ".__LINE__."; hash: [".$an."], key: [$parameter->{key}]\n"; die "$THIS_FILE ".__LINE__."; The hash key string: [$parameter->{key}] doesn't seem to be valid. It should be a string in the format 'foo::bar::baz'.\n" if $parameter->{key} !~ /::/; # Split up the keys. my $key = $parameter->{key} ? $parameter->{key} : ""; my $value = undef; # We return 'undef' so that the caller can tell the difference between an empty string versus nothing found. if ($key =~ /::/) { my @keys = split /::/, $key; my $last_key = pop @keys; # Re-order the array. my $current_hash_ref = $anvil->data; foreach my $key (@keys) { $current_hash_ref = $current_hash_ref->{$key}; } $value = $current_hash_ref->{$last_key}; } return ($value); } =head2 _make_hash_reference This takes a string with double-colon seperators and divides on those double-colons to create a hash reference where each element is a hash key. NOTE: Contributed by Shaun Fryer and Viktor Pavlenko by way of Toronto Perl Mongers. =cut sub _make_hash_reference { my $self = shift; my $href = shift; my $key_string = shift; my $value = shift; my @keys = split /::/, $key_string; my $last_key = pop @keys; my $_href = {}; $_href->{$last_key} = $value; while (my $key = pop @keys) { my $elem = {}; $elem->{$key} = $_href; $_href = $elem; } $self->_add_hash_reference($href, $_href); } =head2 _set_defaults This sets default variable values for the program. =cut sub _set_defaults { my ($anvil) = shift; $anvil->data->{sys} = { apache => { user => "admin", }, daemons => { restart_firewalld => 1, }, database => { archive => { compress => 1, count => 50000, directory => "/usr/local/anvil/archives/", division => 6000, trigger => 100000, }, # grep 'CREATE TABLE' tools/anvil.sql | grep -v history. | awk '{print $3}' | sort core_tables => [ "alerts", "alert_sent", "bonds", "bridges", "hosts", "host_variable", "ip_addresses", "jobs", "network_interfaces", "states", "updated", "users", "variables", ], local_lock_active => 0, locking_reap_age => 300, log_transactions => 0, maximum_batch_size => 25000, name => "anvil", test_table => "hosts", user => "admin", }, host_type => "", host_uuid => "", language => "en_CA", log_file => "/var/log/anvil.log", password => { algorithm => "sha512", hash_count => 500000, salt_length => 16, }, service => { postgresql => "postgresql", }, terminal => { columns => 80, stty => "", }, use_base2 => 1, user => { name => "admin", cookie_valid => 0, language => "en_CA", skin => "alteeve", }, # This is data filled from the active user's database table. users => { user_name => "", user_password_hash => "", user_session_salt => "", user_salt => "", user_algorithm => "", user_hash_count => "", user_language => "", user_is_admin => "", user_is_experienced => "", user_is_trusted => "", }, }; $anvil->data->{defaults} = { ## Network stuff... The second octet auto-increments to handle N-number of netowrks. As such, ## we need to use a wider spread between the BCNs, SNs and IFNs than we had ## in v2. network => { # BCN starts at 10.(0+n)/16 bcn => { subnet => "10.0.0.0", netmask => "255.255.0.0", pdu_octet3 => "1", ups_octet3 => "2", switch_octet3 => "3", striker_octet3 => "4", }, dns => "8.8.8.8, 8.8.4.4", # IFN starts at 10.(80+)/16 ifn => { subnet => "10.100.0.0", netmask => "255.255.0.0", striker_octet3 => "4", }, # SN starts at 10.(40+n)/16 sn => { subnet => "10.50.0.0", netmask => "255.255.0.0", }, }, database => { locking => { reap_age => 300, } }, language => { # Default language for all output shown to a user. output => 'en_CA', }, limits => { # This is the maximum number of times we're allow to loop when injecting variables # into a string being processed in Anvil::Tools::Words->string(); string_loops => 1000, }, 'log' => { db_transactions => 0, facility => "local0", language => "en_CA", level => 1, secure => 0, server => "", tag => "anvil", }, template => { html => "alteeve", }, }; return(0); } =head2 _set_paths This sets default paths to many system commands, checking to make sure the binary exists at the path and, if not, try to find it. =cut sub _set_paths { my ($anvil) = shift; # Executables $anvil->data->{path} = { configs => { 'anvil.conf' => "/etc/anvil/anvil.conf", 'anvil.version' => "/etc/anvil/anvil.version", 'firewalld.conf' => "/etc/firewalld/firewalld.conf", 'httpd.conf' => "/etc/httpd/conf/httpd.conf", 'journald_anvil' => "/etc/systemd/journald.conf.d/anvil.conf", 'pg_hba.conf' => "/var/lib/pgsql/data/pg_hba.conf", 'postgresql.conf' => "/var/lib/pgsql/data/postgresql.conf", ssh_config => "/etc/ssh/ssh_config", }, data => { group => "/etc/group", '.htpasswd' => "/etc/httpd/.htpasswd", host_uuid => "/etc/anvil/host.uuid", passwd => "/etc/passwd", }, directories => { backups => "/root/anvil-backups", 'cgi-bin' => "/var/www/cgi-bin", firewalld_services => "/usr/lib/firewalld/services", firewalld_zones => "/etc/firewalld/zones", html => "/var/www/html", ifcfg => "/etc/sysconfig/network-scripts", skins => "/var/www/html/skins", tools => "/usr/sbin", units => "/usr/lib/systemd/system", }, exe => { 'anvil-change-password' => "/usr/sbin/anvil-change-password", 'anvil-prep-database' => "/usr/sbin/anvil-prep-database", 'anvil-update-states' => "/usr/sbin/anvil-update-states", 'anvil-report-memory' => "/usr/sbin/anvil-report-memory", 'chmod' => "/usr/bin/chmod", 'chown' => "/usr/bin/chown", cp => "/usr/bin/cp", createdb => "/usr/bin/createdb", createuser => "/usr/bin/createuser", dmidecode => "/usr/sbin/dmidecode", echo => "/usr/bin/echo", ethtool => "/usr/sbin/ethtool", expect => "/usr/bin/expect", 'firewall-cmd' => "/usr/bin/firewall-cmd", gethostip => "/usr/bin/gethostip", 'grep' => "/usr/bin/grep", head => "/usr/bin/head", hostname => "/usr/bin/hostname", hostnamectl => "/usr/bin/hostnamectl", htpasswd => "/usr/bin/htpasswd", ifdown => "/sbin/ifdown", ifup => "/sbin/ifup", ip => "/usr/sbin/ip", 'iptables-save' => "/usr/sbin/iptables-save", journalctl => "/usr/bin/journalctl", logger => "/usr/bin/logger", md5sum => "/usr/bin/md5sum", 'mkdir' => "/usr/bin/mkdir", nmcli => "/bin/nmcli", openssl => "/usr/bin/openssl", passwd => "/usr/bin/passwd", ping => "/usr/bin/ping", pgrep => "/usr/bin/pgrep", ps => "/usr/bin/ps", psql => "/usr/bin/psql", 'postgresql-setup' => "/usr/bin/postgresql-setup", pwd => "/usr/bin/pwd", rsync => "/usr/bin/rsync", sed => "/usr/bin/sed", 'shutdown' => "/usr/sbin/shutdown", 'ssh-keyscan' => "/usr/bin/ssh-keyscan", strings => "/usr/bin/strings", stty => "/usr/bin/stty", su => "/usr/bin/su", systemctl => "/usr/bin/systemctl", timeout => "/usr/bin/timeout", touch => "/usr/bin/touch", tput => "/usr/bin/tput", 'tr' => "/usr/bin/tr", usermod => "/usr/sbin/usermod", uuidgen => "/usr/bin/uuidgen", virsh => "/usr/bin/virsh", }, 'lock' => { database => "/tmp/anvil-tools.database.lock", }, secure => { postgres_pgpass => "/var/lib/pgsql/.pgpass", }, sql => { 'anvil.sql' => "/usr/share/anvil/anvil.sql", }, urls => { skins => "/skins", }, words => { 'words.xml' => "/usr/share/anvil/words.xml", }, }; # Make sure we actually have the requested files. foreach my $type (sort {$a cmp $b} keys %{$anvil->data->{path}}) { # We don't look for urls because they're relative to the domain. We also don't look for # configs as we might find backups. next if $type eq "urls"; next if $type eq "configs"; foreach my $file (sort {$a cmp $b} keys %{$anvil->data->{path}{$type}}) { if (not -e $anvil->data->{path}{$type}{$file}) { my $full_path = $anvil->Storage->find({file => $file}); if (($full_path) && ($full_path ne "#!not_found!#")) { $anvil->data->{path}{$type}{$file} = $full_path; } } } }; return(0); } =head3 _short_hostname This returns the short hostname for the machine this is running on. That is to say, the hostname up to the first '.'. =cut sub _short_hostname { my $self = shift; my $anvil = $self; my $short_host_name = $anvil->_hostname; $short_host_name =~ s/\..*$//; return($short_host_name); } =head1 Exit Codes =head2 C<1> Anvil::Tools->new() passed something other than a hash reference. =head2 C<2> Failed to find the requested file in C<< Anvil::Tools::Storage->find >> and 'fatal' was set. =head1 Requirements The following packages are required on EL7. * C * C * C * C * C * C * C * C =head1 Recommended Packages The following packages provide non-critical functionality. * C =cut # This catches SIGINT and SIGTERM and fires out an email before shutting down. sub catch_sig { my $self = shift; my $parameter = shift; my $anvil = $self; my $signal = $parameter->{signal} ? $parameter->{signal} : ""; if ($signal) { print "Process with PID: [$$] exiting on SIG".$signal.".\n"; if ($anvil->data->{sys}{terminal}{stty}) { # Restore the terminal. print "Restoring the terminal\n"; $anvil->System->call({shell_call => $anvil->data->{path}{exe}{stty}." ".$anvil->data->{sys}{terminal}{stty}}); } } $anvil->nice_exit({code => 255}); } 1;