2017-10-20 04:19:32 +00:00
package Anvil::Tools ;
2017-04-13 21:04:03 +00:00
#
# This is the "root" package that manages the sub modules and controls access to their methods.
#
BEGIN
{
2017-05-27 05:23:34 +00:00
our $ VERSION = "3.0.0" ;
2017-04-13 21:04:03 +00:00
# This suppresses the 'could not find ParserDetails.ini in /PerlApp/XML/SAX' warning message in
# XML::Simple calls.
2017-05-27 05:23:34 +00:00
#$ENV{HARNESS_ACTIVE} = 1;
2017-04-13 21:04:03 +00:00
}
use strict ;
use warnings ;
2017-08-17 21:16:45 +00:00
use Scalar::Util qw( weaken isweak ) ;
2017-10-04 06:11:03 +00:00
use Time::HiRes ;
2017-05-02 04:41:12 +00:00
use Data::Dumper ;
2017-12-08 22:04:36 +00:00
use CGI ;
2017-04-13 21:04:03 +00:00
my $ THIS_FILE = "Tools.pm" ;
2017-05-27 05:23:34 +00:00
### Methods;
# data
# environment
2017-08-02 01:04:35 +00:00
# nice_exit
2017-05-27 05:23:34 +00:00
# _add_hash_reference
2018-01-04 09:29:05 +00:00
# _anvil_version
2017-07-07 05:54:49 +00:00
# _hostname
2017-05-27 05:23:34 +00:00
# _make_hash_reference
# _set_defaults
# _set_paths
2017-07-07 05:54:49 +00:00
# _short_hostname
2017-05-27 05:23:34 +00:00
2017-04-13 21:04:03 +00:00
use utf8 ;
2017-05-27 05:23:34 +00:00
binmode ( STDERR , ':encoding(utf-8)' ) ;
binmode ( STDOUT , ':encoding(utf-8)' ) ;
2017-04-13 21:04:03 +00:00
2017-05-02 04:41:12 +00:00
# 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
2017-10-20 04:19:32 +00:00
# methods via their containing module's name. (A La: C<< $anvil->Module->method >> rather than C<< $anvil->method >>).
2018-05-15 05:55:56 +00:00
use Anvil::Tools::Account ;
2017-10-20 04:19:32 +00:00
use Anvil::Tools::Alert ;
use Anvil::Tools::Database ;
use Anvil::Tools::Convert ;
use Anvil::Tools::Get ;
2018-09-04 22:57:09 +00:00
use Anvil::Tools::Job ;
2017-10-20 04:19:32 +00:00
use Anvil::Tools::Log ;
2018-04-23 02:16:10 +00:00
use Anvil::Tools::Remote ;
2017-10-20 04:19:32 +00:00
use Anvil::Tools::Storage ;
use Anvil::Tools::System ;
use Anvil::Tools::Template ;
use Anvil::Tools::Words ;
use Anvil::Tools::Validate ;
2017-05-02 04:41:12 +00:00
2017-04-13 21:04:03 +00:00
= pod
= encoding utf8
= head1 NAME
2017-10-20 04:19:32 +00:00
Anvil:: Tools
2017-04-13 21:04:03 +00:00
2017-10-20 04:19:32 +00:00
Provides a common oject handle to all Anvil::Tools:: * module methods and handles invocation configuration .
2017-04-13 21:04:03 +00:00
= head1 SYNOPSIS
2017-10-20 04:19:32 +00:00
use Anvil::Tools ;
2017-04-13 21:04:03 +00:00
2017-10-20 04:19:32 +00:00
# Get a common object handle on all Anvil::Tools::* modules.
my $ anvil = Anvil::Tools - > new ( ) ;
2017-05-27 05:23:34 +00:00
2017-10-20 04:19:32 +00:00
# Again, but this time sets some initial values in the '$anvil->data' hash.
my $ anvil = Anvil::Tools - > new (
2017-04-13 21:04:03 +00:00
{
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'.
2017-10-20 04:19:32 +00:00
my $ anvil = Anvil::Tools - > new (
2017-04-13 21:04:03 +00:00
{
'Log' = > {
2017-05-27 05:23:34 +00:00
user_language = > "jp" ,
2017-04-13 21:04:03 +00:00
log_language = > "jp"
level = > 2 ,
2017-05-27 05:23:34 +00:00
} ,
2017-04-13 21:04:03 +00:00
} ) ;
= head1 DESCRIPTION
2017-10-20 04:19:32 +00:00
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 .
2017-04-13 21:04:03 +00:00
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 .
2017-05-02 04:41:12 +00:00
= head1 METHODS
2017-04-13 21:04:03 +00:00
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 = > {
2018-05-15 05:55:56 +00:00
ACCOUNT = > Anvil::Tools::Account - > new ( ) ,
2017-10-20 04:19:32 +00:00
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 ( ) ,
2018-09-04 22:57:09 +00:00
JOB = > Anvil::Tools::Job - > new ( ) ,
2018-04-23 02:16:10 +00:00
REMOTE = > Anvil::Tools::Remote - > new ( ) ,
2017-10-20 04:19:32 +00:00
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 ( ) ,
2018-03-07 08:48:48 +00:00
# This is to be removed before development ends.
2018-12-29 08:23:58 +00:00
'log' = > {
main = > "" ,
} ,
2017-04-13 21:04:03 +00:00
} ,
DATA = > { } ,
ENV_VALUES = > {
ENVIRONMENT = > 'cli' ,
} ,
2017-06-07 16:15:03 +00:00
HOST = > {
# This is the host's UUID. It should never be manually set.
UUID = > "" ,
2018-01-04 09:29:05 +00:00
ANVIL_VERSION = > "" ,
2017-06-07 16:15:03 +00:00
} ,
2017-04-13 21:04:03 +00:00
} ;
2017-08-17 21:16:45 +00:00
2017-04-13 21:04:03 +00:00
# Bless you!
bless $ self , $ class ;
2017-08-17 21:16:45 +00:00
2017-04-13 21:04:03 +00:00
# This isn't needed, but it makes the code below more consistent with and portable to other modules.
2017-10-20 04:19:32 +00:00
my $ anvil = $ self ;
weaken ( $ anvil ) ; # Helps avoid memory leaks. See Scalar::Utils
2017-05-02 04:41:12 +00:00
# Get a handle on the various submodules
2018-05-15 05:55:56 +00:00
$ anvil - > Account - > parent ( $ anvil ) ;
2017-10-20 04:19:32 +00:00
$ anvil - > Alert - > parent ( $ anvil ) ;
$ anvil - > Database - > parent ( $ anvil ) ;
$ anvil - > Convert - > parent ( $ anvil ) ;
$ anvil - > Get - > parent ( $ anvil ) ;
$ anvil - > Log - > parent ( $ anvil ) ;
2018-09-04 22:57:09 +00:00
$ anvil - > Job - > parent ( $ anvil ) ;
2018-04-23 02:16:10 +00:00
$ anvil - > Remote - > parent ( $ anvil ) ;
2017-10-20 04:19:32 +00:00
$ anvil - > Storage - > parent ( $ anvil ) ;
$ anvil - > System - > parent ( $ anvil ) ;
$ anvil - > Template - > parent ( $ anvil ) ;
$ anvil - > Words - > parent ( $ anvil ) ;
$ anvil - > Validate - > parent ( $ anvil ) ;
2017-08-17 21:16:45 +00:00
2017-04-13 21:04:03 +00:00
# Set some system paths and system default variables
* Fixed a bug where setting the debug level to 3 caused a deep recursion and a system hang.
* Update Anvil::Tools->new() to access the parameters 'log_level', 'log_secure' and 'debug', streamlining the frequent calls to $anvil->Log->level and ->secure in program startup, and allowing the values to take effect during the ->new constructor.
* Passed 'debug' to child method calls in more places (still more to do though).
* Fixed a bug where 'test_table' wasn't set in the right place, causing the database to try to initialize repeatedly.
* Made Database->archive_database only run if called with root access.
* Now the number of database connections are stored in 'sys::db_connections' instead of checking the returned number, and that is cleared on disconnect.
* Started working more on 'anvil-daemon', including adding support for System->call being taking 'background', 'stderr_file' and 'stdout_file' paramters which, when set, used Proc::Simple to background the process.
* Did some more work on database archiving, though still far from done.
Signed-off-by: Digimer <digimer@alteeve.ca>
2018-08-01 06:06:16 +00:00
$ anvil - > _set_defaults ( ) ;
2019-01-18 08:19:36 +00:00
$ anvil - > _set_paths ( ) ;
* Fixed a bug where setting the debug level to 3 caused a deep recursion and a system hang.
* Update Anvil::Tools->new() to access the parameters 'log_level', 'log_secure' and 'debug', streamlining the frequent calls to $anvil->Log->level and ->secure in program startup, and allowing the values to take effect during the ->new constructor.
* Passed 'debug' to child method calls in more places (still more to do though).
* Fixed a bug where 'test_table' wasn't set in the right place, causing the database to try to initialize repeatedly.
* Made Database->archive_database only run if called with root access.
* Now the number of database connections are stored in 'sys::db_connections' instead of checking the returned number, and that is cleared on disconnect.
* Started working more on 'anvil-daemon', including adding support for System->call being taking 'background', 'stderr_file' and 'stdout_file' paramters which, when set, used Proc::Simple to background the process.
* Did some more work on database archiving, though still far from done.
Signed-off-by: Digimer <digimer@alteeve.ca>
2018-08-01 06:06:16 +00:00
# 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 } } ) ;
}
2017-08-17 21:16:45 +00:00
# This will help clean up if we catch a signal.
2017-10-20 04:19:32 +00:00
$ SIG { INT } = sub { $ anvil - > catch_sig ( { signal = > "INT" } ) ; } ;
$ SIG { TERM } = sub { $ anvil - > catch_sig ( { signal = > "TERM" } ) ; } ;
2017-08-17 21:16:45 +00:00
2017-06-11 07:00:35 +00:00
# This sets the environment this program is running in.
if ( $ ENV { SERVER_NAME } )
{
2017-10-20 04:19:32 +00:00
$ anvil - > environment ( "html" ) ;
2017-06-11 07:00:35 +00:00
# There is no PWD environment variable, so we'll use 'DOCUMENT_ROOT' as 'PWD'
$ ENV { PWD } = $ ENV { DOCUMENT_ROOT } ;
}
else
{
2017-10-20 04:19:32 +00:00
$ anvil - > environment ( "cli" ) ;
2017-06-11 07:00:35 +00:00
}
2017-08-17 21:16:45 +00:00
2017-10-20 04:19:32 +00:00
# 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 } ;
2017-05-27 05:23:34 +00:00
2017-06-05 12:02:14 +00:00
# Initialize the list of directories to seach.
* Fixed a bug where setting the debug level to 3 caused a deep recursion and a system hang.
* Update Anvil::Tools->new() to access the parameters 'log_level', 'log_secure' and 'debug', streamlining the frequent calls to $anvil->Log->level and ->secure in program startup, and allowing the values to take effect during the ->new constructor.
* Passed 'debug' to child method calls in more places (still more to do though).
* Fixed a bug where 'test_table' wasn't set in the right place, causing the database to try to initialize repeatedly.
* Made Database->archive_database only run if called with root access.
* Now the number of database connections are stored in 'sys::db_connections' instead of checking the returned number, and that is cleared on disconnect.
* Started working more on 'anvil-daemon', including adding support for System->call being taking 'background', 'stderr_file' and 'stdout_file' paramters which, when set, used Proc::Simple to background the process.
* Did some more work on database archiving, though still far from done.
Signed-off-by: Digimer <digimer@alteeve.ca>
2018-08-01 06:06:16 +00:00
$ anvil - > Storage - > search_directories ( { debug = > $ debug , initialize = > 1 } ) ;
2017-08-17 21:16:45 +00:00
2017-04-13 21:04:03 +00:00
# I need to read the initial words early.
2018-12-28 23:06:26 +00:00
$ anvil - > Words - > read ( { debug = > $ debug } ) ;
2017-06-07 15:36:06 +00:00
# If the local './tools.conf' file exists, read it in.
2017-10-20 04:19:32 +00:00
if ( - r $ anvil - > data - > { path } { configs } { 'anvil.conf' } )
2017-06-07 15:36:06 +00:00
{
* Fixed a bug where setting the debug level to 3 caused a deep recursion and a system hang.
* Update Anvil::Tools->new() to access the parameters 'log_level', 'log_secure' and 'debug', streamlining the frequent calls to $anvil->Log->level and ->secure in program startup, and allowing the values to take effect during the ->new constructor.
* Passed 'debug' to child method calls in more places (still more to do though).
* Fixed a bug where 'test_table' wasn't set in the right place, causing the database to try to initialize repeatedly.
* Made Database->archive_database only run if called with root access.
* Now the number of database connections are stored in 'sys::db_connections' instead of checking the returned number, and that is cleared on disconnect.
* Started working more on 'anvil-daemon', including adding support for System->call being taking 'background', 'stderr_file' and 'stdout_file' paramters which, when set, used Proc::Simple to background the process.
* Did some more work on database archiving, though still far from done.
Signed-off-by: Digimer <digimer@alteeve.ca>
2018-08-01 06:06:16 +00:00
$ 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 } } ) ;
}
2017-06-07 15:36:06 +00:00
}
2017-06-07 16:15:03 +00:00
2018-05-15 05:55:56 +00:00
# Get the local host UUID.
* Fixed a bug where setting the debug level to 3 caused a deep recursion and a system hang.
* Update Anvil::Tools->new() to access the parameters 'log_level', 'log_secure' and 'debug', streamlining the frequent calls to $anvil->Log->level and ->secure in program startup, and allowing the values to take effect during the ->new constructor.
* Passed 'debug' to child method calls in more places (still more to do though).
* Fixed a bug where 'test_table' wasn't set in the right place, causing the database to try to initialize repeatedly.
* Made Database->archive_database only run if called with root access.
* Now the number of database connections are stored in 'sys::db_connections' instead of checking the returned number, and that is cleared on disconnect.
* Started working more on 'anvil-daemon', including adding support for System->call being taking 'background', 'stderr_file' and 'stdout_file' paramters which, when set, used Proc::Simple to background the process.
* Did some more work on database archiving, though still far from done.
Signed-off-by: Digimer <digimer@alteeve.ca>
2018-08-01 06:06:16 +00:00
$ anvil - > Get - > host_uuid ( { debug = > $ debug } ) ;
2018-05-15 05:55:56 +00:00
2017-06-07 16:15:03 +00:00
# Read in any command line switches.
* Fixed a bug where setting the debug level to 3 caused a deep recursion and a system hang.
* Update Anvil::Tools->new() to access the parameters 'log_level', 'log_secure' and 'debug', streamlining the frequent calls to $anvil->Log->level and ->secure in program startup, and allowing the values to take effect during the ->new constructor.
* Passed 'debug' to child method calls in more places (still more to do though).
* Fixed a bug where 'test_table' wasn't set in the right place, causing the database to try to initialize repeatedly.
* Made Database->archive_database only run if called with root access.
* Now the number of database connections are stored in 'sys::db_connections' instead of checking the returned number, and that is cleared on disconnect.
* Started working more on 'anvil-daemon', including adding support for System->call being taking 'background', 'stderr_file' and 'stdout_file' paramters which, when set, used Proc::Simple to background the process.
* Did some more work on database archiving, though still far from done.
Signed-off-by: Digimer <digimer@alteeve.ca>
2018-08-01 06:06:16 +00:00
$ anvil - > Get - > switches ( { debug = > $ debug } ) ;
2017-06-07 16:15:03 +00:00
2018-01-04 09:29:05 +00:00
# Read in the local Anvil! version.
2018-07-26 01:02:48 +00:00
#...
2017-08-17 21:16:45 +00:00
2017-04-13 21:04:03 +00:00
return ( $ self ) ;
}
2017-05-02 04:41:12 +00:00
#############################################################################################################
# Public methods #
#############################################################################################################
2017-04-13 21:04:03 +00:00
= 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 .
2017-10-20 04:19:32 +00:00
When called without an argument , it returns the existing '$anvil->data' hash reference .
2017-04-13 21:04:03 +00:00
2017-10-20 04:19:32 +00:00
my $ anvil = $ anvil - > data ( ) ;
2017-04-13 21:04:03 +00:00
2017-10-20 04:19:32 +00:00
When called with a hash reference as the argument , it sets '$anvil->data' to the new hash .
2017-04-13 21:04:03 +00:00
my $ some_hash = { } ;
2017-10-20 04:19:32 +00:00
my $ anvil = $ anvil - > data ( $ some_hash ) ;
2017-04-13 21:04:03 +00:00
2017-10-20 04:19:32 +00:00
Data can be entered into or access by treating '$anvil->data' as a normal hash reference .
2017-04-13 21:04:03 +00:00
2017-10-20 04:19:32 +00:00
my $ anvil = Anvil::Tools - > new (
2017-04-13 21:04:03 +00:00
{
data = > {
foo = > "" ,
bar = > [ 6 , 4 , 12 ] ,
baz = > {
animal = > "Cat" ,
thing = > "Boat" ,
} ,
} ,
} ) ;
# Copy the 'Cat' value into the $animal variable.
2017-10-20 04:19:32 +00:00
my $ animal = $ anvil - > data - > { baz } { animal } ;
2017-04-13 21:04:03 +00:00
# Set 'A thing' in 'foo'.
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { foo } = "A thing" ;
2017-04-13 21:04:03 +00:00
* Fixed a bug where setting the debug level to 3 caused a deep recursion and a system hang.
* Update Anvil::Tools->new() to access the parameters 'log_level', 'log_secure' and 'debug', streamlining the frequent calls to $anvil->Log->level and ->secure in program startup, and allowing the values to take effect during the ->new constructor.
* Passed 'debug' to child method calls in more places (still more to do though).
* Fixed a bug where 'test_table' wasn't set in the right place, causing the database to try to initialize repeatedly.
* Made Database->archive_database only run if called with root access.
* Now the number of database connections are stored in 'sys::db_connections' instead of checking the returned number, and that is cleared on disconnect.
* Started working more on 'anvil-daemon', including adding support for System->call being taking 'background', 'stderr_file' and 'stdout_file' paramters which, when set, used Proc::Simple to background the process.
* Did some more work on database archiving, though still far from done.
Signed-off-by: Digimer <digimer@alteeve.ca>
2018-08-01 06:06:16 +00:00
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 .
2017-04-13 21:04:03 +00:00
= cut
sub data
{
2017-10-20 04:19:32 +00:00
my ( $ anvil ) = shift ;
2017-04-13 21:04:03 +00:00
# Pick up the passed in hash, if any.
2017-10-20 04:19:32 +00:00
$ anvil - > { DATA } = shift if $ _ [ 0 ] ;
2017-04-13 21:04:03 +00:00
2017-10-20 04:19:32 +00:00
return ( $ anvil - > { DATA } ) ;
2017-04-13 21:04:03 +00:00
}
= 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 .
2017-10-20 04:19:32 +00:00
if ( $ anvil - > environment ( ) eq "cli" )
2017-04-13 21:04:03 +00:00
{
# format for STDOUT
}
2017-10-20 04:19:32 +00:00
elsif ( $ anvil - > environment ( ) eq "html" )
2017-04-13 21:04:03 +00:00
{
# Use the template system to output HTML
}
When called with a string as the argument , that string will be set as the environment string .
2017-10-20 04:19:32 +00:00
$ anvil - > environment ( "cli" ) ;
2017-04-13 21:04:03 +00:00
Technically , any string can be used , however only 'cli' or 'html' are used by convention .
= cut
sub environment
{
2017-10-20 04:19:32 +00:00
my ( $ anvil ) = shift ;
weaken ( $ anvil ) ;
2017-08-17 21:16:45 +00:00
2017-04-13 21:04:03 +00:00
# Pick up the passed in delimiter, if any.
2017-09-04 15:10:02 +00:00
if ( $ _ [ 0 ] )
{
2018-08-13 06:22:46 +00:00
$ anvil - > data - > { ENV_VALUES } { ENVIRONMENT } = shift ;
2017-12-08 22:04:36 +00:00
# Load the CGI stuff if we're in a browser
2018-08-13 06:22:46 +00:00
if ( $ anvil - > data - > { ENV_VALUES } { ENVIRONMENT } eq "html" )
2017-09-04 15:10:02 +00:00
{
2017-12-08 22:04:36 +00:00
CGI::Carp - > import ( qw( fatalsToBrowser ) ) ;
2017-09-04 15:10:02 +00:00
}
}
2017-05-02 04:41:12 +00:00
2018-08-13 06:22:46 +00:00
return ( $ anvil - > data - > { ENV_VALUES } { ENVIRONMENT } ) ;
2017-05-02 04:41:12 +00:00
}
2017-08-02 01:04:35 +00:00
= 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 ;
2018-09-06 05:37:08 +00:00
my $ anvil = $ self ;
2018-09-07 05:29:43 +00:00
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
2017-08-02 01:04:35 +00:00
my $ exit_code = defined $ parameter - > { exit_code } ? $ parameter - > { exit_code } : 0 ;
# Close database connections (if any).
2018-09-07 05:29:43 +00:00
$ anvil - > Database - > disconnect ( { debug = > $ debug } ) ;
2017-08-02 01:04:35 +00:00
2017-10-04 06:11:03 +00:00
# Report the runtime.
my $ end_time = Time::HiRes:: time ;
2017-10-20 04:19:32 +00:00
my $ run_time = $ end_time - $ anvil - > data - > { ENV_VALUES } { START_TIME } ;
2018-09-06 05:37:08 +00:00
my $ caller = ( $ 0 =~ /^.*\/(.*)$/ ) [ 0 ] ;
2018-09-07 05:29:43 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
2017-10-20 04:19:32 +00:00
's1:ENV_VALUES::START_TIME' = > $ anvil - > data - > { ENV_VALUES } { START_TIME } ,
2017-10-04 06:11:03 +00:00
's2:end_time' = > $ end_time ,
's3:run_time' = > $ run_time ,
2018-09-06 05:37:08 +00:00
's4:caller' = > $ caller ,
} } ) ;
2018-09-07 05:29:43 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , key = > "log_0135" , variables = > { 'caller' = > $ caller , runtime = > $ run_time } } ) ;
2018-09-06 05:37:08 +00:00
my ( $ package , $ filename , $ line ) = caller ;
2018-09-07 05:29:43 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
2018-09-06 05:37:08 +00:00
's1:package' = > $ package ,
's2:filename' = > $ filename ,
's3:line' = > $ line ,
2017-10-04 06:11:03 +00:00
} } ) ;
2018-03-07 08:48:48 +00:00
# Close the log file.
2018-12-29 08:23:58 +00:00
if ( $ anvil - > data - > { HANDLE } { 'log' } { main } )
2018-03-07 08:48:48 +00:00
{
2018-12-29 08:23:58 +00:00
close $ anvil - > data - > { HANDLE } { 'log' } { main } ;
$ anvil - > data - > { HANDLE } { 'log' } { main } = "" ;
2018-03-07 08:48:48 +00:00
}
2017-08-02 01:04:35 +00:00
exit ( $ exit_code ) ;
}
2017-05-02 04:41:12 +00:00
#############################################################################################################
# Public methods used to access sub modules. #
#############################################################################################################
= head1 Submodule Access Methods
2017-10-20 04:19:32 +00:00
The methods below are used to access methods of submodules using 'C<< $anvil->Module->method() >>' .
2017-05-02 04:41:12 +00:00
= cut
2018-05-15 05:55:56 +00:00
= head2 Account
Access the C <Acount.pm> methods via 'C<< $anvil->Alert->method >>' .
= cut
sub Account
{
my $ self = shift ;
return ( $ self - > { HANDLE } { ACCOUNT } ) ;
}
2017-05-02 04:41:12 +00:00
= head2 Alert
2017-10-20 04:19:32 +00:00
Access the C <Alert.pm> methods via 'C<< $anvil->Alert->method >>' .
2017-05-02 04:41:12 +00:00
= cut
sub Alert
{
my $ self = shift ;
return ( $ self - > { HANDLE } { ALERT } ) ;
}
2017-06-13 21:08:38 +00:00
= head2 Database
2017-10-20 04:19:32 +00:00
Access the C <Database.pm> methods via 'C<< $anvil->Database->method >>' .
2017-06-13 21:08:38 +00:00
= cut
sub Database
{
my $ self = shift ;
return ( $ self - > { HANDLE } { DATABASE } ) ;
}
= head2 Convert
2017-10-20 04:19:32 +00:00
Access the C <Convert.pm> methods via 'C<< $anvil->Convert->method >>' .
2017-06-13 21:08:38 +00:00
= cut
sub Convert
{
my $ self = shift ;
return ( $ self - > { HANDLE } { CONVERT } ) ;
}
2017-06-05 12:02:14 +00:00
= head2 Get
2017-10-20 04:19:32 +00:00
Access the C <Get.pm> methods via 'C<< $anvil->Get->method >>' .
2017-06-05 12:02:14 +00:00
= cut
sub Get
{
my $ self = shift ;
return ( $ self - > { HANDLE } { GET } ) ;
}
2018-09-04 22:57:09 +00:00
= head2 Job
Access the C <Job.pm> methods via 'C<< $anvil->Log->method >>' .
= cut
sub Job
{
my $ self = shift ;
return ( $ self - > { HANDLE } { JOB } ) ;
}
2017-06-05 12:02:14 +00:00
= head2 Log
2017-10-20 04:19:32 +00:00
Access the C <Log.pm> methods via 'C<< $anvil->Log->method >>' .
2017-06-05 12:02:14 +00:00
= cut
sub Log
{
my $ self = shift ;
return ( $ self - > { HANDLE } { LOG } ) ;
}
2018-04-23 02:16:10 +00:00
= head2 Remote
Access the C <Remote.pm> methods via 'C<< $anvil->Remote->method >>' .
= cut
sub Remote
{
my $ self = shift ;
return ( $ self - > { HANDLE } { REMOTE } ) ;
}
2017-05-02 04:41:12 +00:00
= head2 Storage
2017-10-20 04:19:32 +00:00
Access the C <Storage.pm> methods via 'C<< $anvil->Storage->method >>' .
2017-05-02 04:41:12 +00:00
= cut
sub Storage
{
my $ self = shift ;
2017-04-13 21:04:03 +00:00
2017-05-02 04:41:12 +00:00
return ( $ self - > { HANDLE } { STORAGE } ) ;
2017-04-13 21:04:03 +00:00
}
2017-05-02 04:41:12 +00:00
2017-06-10 06:38:04 +00:00
= head2 System
2017-10-20 04:19:32 +00:00
Access the C <System.pm> methods via 'C<< $anvil->System->method >>' .
2017-06-10 06:38:04 +00:00
= cut
sub System
{
my $ self = shift ;
return ( $ self - > { HANDLE } { SYSTEM } ) ;
}
2017-06-09 08:29:25 +00:00
= head2 Template
2017-10-20 04:19:32 +00:00
Access the C <Template.pm> methods via 'C<< $anvil->Template->method >>' .
2017-06-09 08:29:25 +00:00
= cut
sub Template
{
my $ self = shift ;
return ( $ self - > { HANDLE } { TEMPLATE } ) ;
}
2017-05-10 19:01:36 +00:00
= head2 Words
2017-10-20 04:19:32 +00:00
Access the C <Words.pm> methods via 'C<< $anvil->Words->method >>' .
2017-05-10 19:01:36 +00:00
= cut
sub Words
{
my $ self = shift ;
return ( $ self - > { HANDLE } { WORDS } ) ;
}
2017-06-07 16:30:28 +00:00
= head2 Validate
2017-10-20 04:19:32 +00:00
Access the C <Validate.pm> methods via 'C<< $anvil->Validate->method >>' .
2017-06-07 16:30:28 +00:00
= cut
sub Validate
{
my $ self = shift ;
return ( $ self - > { HANDLE } { VALIDATE } ) ;
}
2017-05-02 04:41:12 +00:00
= head1 Private Functions ;
2017-10-20 04:19:32 +00:00
These methods generally should never be called from a program using Anvil:: Tools . However , we are not your boss .
2017-05-02 04:41:12 +00:00
= cut
#############################################################################################################
# Private methods #
#############################################################################################################
2017-05-27 05:23:34 +00:00
= head2 _add_hash_reference
2017-10-20 04:19:32 +00:00
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 .
2017-05-02 04:41:12 +00:00
2017-05-27 05:23:34 +00:00
NOTE: Contributed by Shaun Fryer and Viktor Pavlenko by way of Toronto Perl Mongers .
2017-05-02 04:41:12 +00:00
= cut
2017-05-27 05:23:34 +00:00
sub _add_hash_reference
2017-05-02 04:41:12 +00:00
{
2017-05-27 05:23:34 +00:00
my $ self = shift ;
my $ href1 = shift ;
my $ href2 = shift ;
2017-05-02 04:41:12 +00:00
2017-05-27 05:23:34 +00:00
for my $ key ( keys %$ href2 )
2017-05-02 04:41:12 +00:00
{
2017-05-27 05:23:34 +00:00
if ( ref $ href1 - > { $ key } eq 'HASH' )
2017-05-02 04:41:12 +00:00
{
2017-05-27 05:23:34 +00:00
$ self - > _add_hash_reference ( $ href1 - > { $ key } , $ href2 - > { $ key } ) ;
2017-05-02 04:41:12 +00:00
}
2017-05-27 05:23:34 +00:00
else
2017-05-02 04:41:12 +00:00
{
2017-05-27 05:23:34 +00:00
$ href1 - > { $ key } = $ href2 - > { $ key } ;
2017-05-02 04:41:12 +00:00
}
}
2017-05-27 05:23:34 +00:00
}
2018-01-04 09:29:05 +00:00
= head2 _anvil_version
= cut
sub _anvil_version
{
my $ self = shift ;
my $ anvil = $ self ;
2018-09-13 21:08:50 +00:00
$ anvil - > data - > { HOST } { ANVIL_VERSION } = "" if not defined $ anvil - > data - > { HOST } { ANVIL_VERSION } ;
2018-08-13 06:22:46 +00:00
if ( $ anvil - > data - > { HOST } { ANVIL_VERSION } eq "" )
2018-01-04 09:29:05 +00:00
{
# Try to read the local Anvil! version.
2018-08-13 06:22:46 +00:00
$ anvil - > data - > { HOST } { ANVIL_VERSION } = $ anvil - > Get - > anvil_version ( ) ;
2018-01-04 09:29:05 +00:00
}
2018-08-13 06:22:46 +00:00
return ( $ anvil - > data - > { HOST } { ANVIL_VERSION } ) ;
2018-01-04 09:29:05 +00:00
}
2017-07-07 05:54:49 +00:00
= head2 _hostname
This returns the ( full ) hostname for the machine this is running on .
= cut
sub _hostname
{
2018-01-04 09:29:05 +00:00
my $ self = shift ;
my $ anvil = $ self ;
2017-07-07 05:54:49 +00:00
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.
2017-10-20 04:19:32 +00:00
$ hostname = $ anvil - > System - > call ( { shell_call = > $ anvil - > data - > { path } { exe } { hostname } } ) ;
2017-07-07 05:54:49 +00:00
}
return ( $ hostname ) ;
}
2017-06-04 04:40:17 +00:00
= head2 _get_hash_reference
2017-10-20 04:19:32 +00:00
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 .
2017-06-04 04:40:17 +00:00
For example ;
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { foo } { bar } = "baz" ;
my $ value = $ anvil - > _get_hash_reference ( { key = > "foo::bar" } ) ;
2017-06-04 04:40:17 +00:00
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 ;
2017-10-20 04:19:32 +00:00
my $ anvil = $ self ;
2017-06-04 04:40:17 +00:00
2019-03-22 04:57:53 +00:00
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 } !~ /::/ ;
2017-06-04 04:40:17 +00:00
# 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.
2017-10-20 04:19:32 +00:00
my $ current_hash_ref = $ anvil - > data ;
2017-06-04 04:40:17 +00:00
foreach my $ key ( @ keys )
{
$ current_hash_ref = $ current_hash_ref - > { $ key } ;
}
$ value = $ current_hash_ref - > { $ last_key } ;
}
return ( $ value ) ;
}
2017-05-27 05:23:34 +00:00
= 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
{
2017-10-20 04:19:32 +00:00
my ( $ anvil ) = shift ;
2017-05-27 05:23:34 +00:00
2018-12-28 22:17:59 +00:00
$ anvil - > data - > { scancore } = {
2018-12-18 23:20:49 +00:00
timing = > {
# Delay between DB connection attempts when no databases are available?
2018-12-28 22:17:59 +00:00
agent_runtime = > 30 ,
2018-12-18 23:20:49 +00:00
db_retry_interval = > 2 ,
# Delay between scans?
run_interval = > 30 ,
} ,
} ;
$ anvil - > data - > { sys } = {
2018-05-10 05:41:45 +00:00
apache = > {
user = > "admin" ,
} ,
2018-11-01 07:16:07 +00:00
daemon = > {
dhcpd = > "dhcpd.service" ,
firewalld = > "firewalld.service" ,
httpd = > "httpd.service" ,
postgresql = > "postgresql.service" ,
tftp = > "tftp.socket" ,
} ,
2017-08-17 17:40:15 +00:00
daemons = > {
restart_firewalld = > 1 ,
} ,
2017-08-02 01:36:57 +00:00
database = > {
2017-08-23 06:45:10 +00:00
archive = > {
compress = > 1 ,
count = > 50000 ,
2017-10-20 04:19:32 +00:00
directory = > "/usr/local/anvil/archives/" ,
2017-08-23 06:45:10 +00:00
division = > 6000 ,
trigger = > 100000 ,
} ,
2018-08-15 20:57:57 +00:00
connections = > 0 ,
2018-09-22 04:54:37 +00:00
### WARNING: The order the tables are resync'ed is important! Any table that has a
### foreign key needs to resync *AFTER* the tables with the primary keys.
# NOTE: Check that this list is complete with;
2019-02-06 07:36:19 +00:00
# grep 'CREATE TABLE' share/anvil.sql | grep -v history. | awk '{print $3}'
2017-08-18 06:09:35 +00:00
core_tables = > [
2018-09-22 04:54:37 +00:00
"hosts" , # Always has to be first.
2019-02-06 07:36:19 +00:00
"host_keys" ,
2018-09-22 04:54:37 +00:00
"users" ,
"host_variable" ,
"sessions" , # Has to come after users and hosts
2019-02-06 07:36:19 +00:00
"anvils" ,
2017-08-18 06:09:35 +00:00
"alerts" ,
2019-02-06 07:36:19 +00:00
"recipients" ,
"notifications" ,
"mail_servers" ,
"host_mail_servers" ,
2018-09-22 04:54:37 +00:00
"variables" ,
"jobs" ,
"network_interfaces" ,
2017-10-20 04:19:32 +00:00
"bonds" ,
"bridges" ,
2018-06-04 18:12:32 +00:00
"ip_addresses" ,
2019-02-06 07:36:19 +00:00
"files" ,
"file_locations" ,
"servers" ,
"definitions" ,
2017-08-19 06:40:14 +00:00
"updated" ,
2018-09-22 04:54:37 +00:00
"alert_sent" ,
"states" ,
2017-08-18 06:09:35 +00:00
] ,
2018-10-10 23:43:23 +00:00
failed_connection_log_level = > 1 ,
2017-08-02 01:36:57 +00:00
local_lock_active = > 0 ,
2018-08-15 20:57:57 +00:00
local_uuid = > "" ,
2017-08-02 01:36:57 +00:00
locking_reap_age = > 300 ,
log_transactions = > 0 ,
maximum_batch_size = > 25000 ,
2018-04-13 23:55:34 +00:00
name = > "anvil" ,
2018-08-15 20:57:57 +00:00
read_uuid = > "" ,
* Fixed a bug where setting the debug level to 3 caused a deep recursion and a system hang.
* Update Anvil::Tools->new() to access the parameters 'log_level', 'log_secure' and 'debug', streamlining the frequent calls to $anvil->Log->level and ->secure in program startup, and allowing the values to take effect during the ->new constructor.
* Passed 'debug' to child method calls in more places (still more to do though).
* Fixed a bug where 'test_table' wasn't set in the right place, causing the database to try to initialize repeatedly.
* Made Database->archive_database only run if called with root access.
* Now the number of database connections are stored in 'sys::db_connections' instead of checking the returned number, and that is cleared on disconnect.
* Started working more on 'anvil-daemon', including adding support for System->call being taking 'background', 'stderr_file' and 'stdout_file' paramters which, when set, used Proc::Simple to background the process.
* Did some more work on database archiving, though still far from done.
Signed-off-by: Digimer <digimer@alteeve.ca>
2018-08-01 06:06:16 +00:00
test_table = > "hosts" ,
2018-08-15 20:57:57 +00:00
timestamp = > "" ,
2018-04-13 23:55:34 +00:00
user = > "admin" ,
2018-08-15 20:57:57 +00:00
use_handle = > "" ,
2017-08-02 01:36:57 +00:00
} ,
host_type = > "" ,
2018-05-15 05:55:56 +00:00
host_uuid = > "" ,
2018-05-17 04:45:20 +00:00
language = > "en_CA" ,
2018-12-29 08:23:58 +00:00
'log' = > {
date = > 1 ,
# Stores the '-v|...|-vvv' so that shell calls can be run at the same level as the
# avtive program when set by the user at the command line.
level = > "" ,
} ,
2018-12-11 07:27:55 +00:00
manage = > {
firewall = > 1 ,
} ,
2018-05-15 21:45:52 +00:00
password = > {
algorithm = > "sha512" ,
hash_count = > 500000 ,
salt_length = > 16 ,
} ,
2018-05-15 05:55:56 +00:00
terminal = > {
columns = > 80 ,
stty = > "" ,
} ,
2017-08-02 01:36:57 +00:00
use_base2 = > 1 ,
2018-06-27 05:28:13 +00:00
user = > {
name = > "admin" ,
cookie_valid = > 0 ,
language = > "en_CA" ,
skin = > "alteeve" ,
} ,
2018-07-04 21:57:25 +00:00
# This is data filled from the active user's database table.
users = > {
user_name = > "" ,
user_password_hash = > "" ,
user_salt = > "" ,
user_algorithm = > "" ,
user_hash_count = > "" ,
user_language = > "" ,
user_is_admin = > "" ,
user_is_experienced = > "" ,
user_is_trusted = > "" ,
} ,
2017-08-02 01:36:57 +00:00
} ;
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { defaults } = {
2018-08-22 06:16:56 +00:00
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" ,
} ,
2018-11-03 06:55:35 +00:00
# NOTE: These are here to allow foreign users to override western defaults in anvil.conf.
kickstart = > {
keyboard = > "--vckeymap=us --xlayouts='us'" ,
password = > "Initial1" ,
timezone = > "Etc/GMT --isUtc" ,
} ,
2019-01-24 00:54:01 +00:00
# See 'striker' -> 'sub generate_ip()' function comments for details on how m3 IPs are handled.
2018-06-15 06:17:20 +00:00
network = > {
2019-01-24 00:54:01 +00:00
# BCN starts at 10.200(+n)/16
2018-06-15 06:17:20 +00:00
bcn = > {
2019-01-24 00:54:01 +00:00
subnet = > "10.200.0.0" ,
netmask = > "255.255.0.0" ,
switch_octet3 = > "1" ,
pdu_octet3 = > "2" ,
ups_octet3 = > "3" ,
striker_octet3 = > "4" ,
striker_ipmi_octet3 = > "5" ,
2018-06-15 06:17:20 +00:00
} ,
2018-06-17 22:25:56 +00:00
dns = > "8.8.8.8, 8.8.4.4" ,
2019-01-24 00:54:01 +00:00
# The IFN will not be under our control. So for suggestion to the user purpose only,
# IFN starts at 10.255/16
2018-06-15 06:17:20 +00:00
ifn = > {
2019-01-24 00:54:01 +00:00
subnet = > "10.255.0.0" ,
2018-06-15 06:17:20 +00:00
netmask = > "255.255.0.0" ,
striker_octet3 = > "4" ,
} ,
2019-01-24 00:54:01 +00:00
# SN starts at 10.100(+n)/16
2018-06-15 06:17:20 +00:00
sn = > {
2019-01-24 00:54:01 +00:00
subnet = > "10.100.0.0" ,
2018-06-15 06:17:20 +00:00
netmask = > "255.255.0.0" ,
} ,
} ,
2017-06-09 08:29:25 +00:00
template = > {
html = > "alteeve" ,
} ,
2017-05-27 05:23:34 +00:00
} ;
2017-05-02 04:41:12 +00:00
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
{
2017-10-20 04:19:32 +00:00
my ( $ anvil ) = shift ;
2017-05-02 04:41:12 +00:00
# Executables
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { path } = {
2017-08-02 01:04:35 +00:00
configs = > {
2018-08-13 06:22:46 +00:00
'anvil.conf' = > "/etc/anvil/anvil.conf" ,
'anvil.version' = > "/etc/anvil/anvil.version" ,
2018-11-01 07:16:07 +00:00
'autoindex.conf' = > "/etc/httpd/conf.d/autoindex.conf" ,
'dhcpd.conf' = > "/etc/dhcp/dhcpd.conf" ,
'dnf.conf' = > "/etc/dnf/dnf.conf" ,
2018-08-13 06:22:46 +00:00
'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" ,
2018-10-29 23:50:43 +00:00
pxe_default = > "/var/lib/tftpboot/pxelinux.cfg/default" ,
2018-08-13 06:22:46 +00:00
ssh_config = > "/etc/ssh/ssh_config" ,
2017-07-07 05:54:49 +00:00
} ,
2017-08-02 01:04:35 +00:00
data = > {
2018-08-13 06:22:46 +00:00
group = > "/etc/group" ,
2019-02-27 05:23:03 +00:00
host_ssh_key = > "/etc/ssh/ssh_host_ecdsa_key.pub" ,
2018-08-13 06:22:46 +00:00
'.htpasswd' = > "/etc/httpd/.htpasswd" ,
host_uuid = > "/etc/anvil/host.uuid" ,
passwd = > "/etc/passwd" ,
2018-10-23 16:55:19 +00:00
'redhat-release' = > "/etc/redhat-release" ,
2017-06-20 02:32:35 +00:00
} ,
2017-08-02 01:04:35 +00:00
directories = > {
2018-08-13 06:22:46 +00:00
backups = > "/root/anvil-backups" ,
'cgi-bin' = > "/var/www/cgi-bin" ,
firewalld_services = > "/usr/lib/firewalld/services" ,
2019-01-24 00:54:01 +00:00
#firewalld_zones => "/etc/firewalld/zones",
firewalld_zones = > "/usr/lib/firewalld/zones" ,
2018-08-13 06:22:46 +00:00
html = > "/var/www/html" ,
ifcfg = > "/etc/sysconfig/network-scripts" ,
2018-12-28 22:17:59 +00:00
scan_agents = > "/usr/sbin/scancore-agents" ,
2019-02-07 08:26:12 +00:00
shared = > {
archives = > "/mnt/shared/archives" ,
2019-02-22 04:57:26 +00:00
base = > "/mnt/shared" ,
2019-02-07 08:26:12 +00:00
definitions = > "/mnt/shared/definitions" ,
files = > "/mnt/shared/files" ,
incoming = > "/mnt/shared/incoming" ,
2019-03-15 06:40:44 +00:00
temp = > "/mnt/shared/temp" ,
2019-02-07 08:26:12 +00:00
} ,
2018-08-13 06:22:46 +00:00
skins = > "/var/www/html/skins" ,
2018-11-02 06:06:04 +00:00
syslinux = > "/usr/share/syslinux" ,
tftpboot = > "/var/lib/tftpboot" ,
2018-08-13 06:22:46 +00:00
tools = > "/usr/sbin" ,
units = > "/usr/lib/systemd/system" ,
2017-06-10 06:38:04 +00:00
} ,
2017-08-02 01:04:35 +00:00
exe = > {
2018-08-13 06:22:46 +00:00
'anvil-change-password' = > "/usr/sbin/anvil-change-password" ,
2018-12-04 09:16:38 +00:00
'anvil-daemon' = > "/usr/sbin/anvil-daemon" ,
2019-04-04 16:27:27 +00:00
'anvil-download-file' = > "/usr/sbin/anvil-download-file" ,
2019-03-12 06:27:31 +00:00
'anvil-file-details' = > "/usr/sbin/anvil-file-details" ,
2018-12-04 09:16:38 +00:00
'anvil-maintenance-mode' = > "/usr/sbin/anvil-maintenance-mode" ,
2018-12-11 07:27:55 +00:00
'anvil-manage-firewall' = > "/usr/sbin/anvil-manage-firewall" ,
2018-12-04 09:16:38 +00:00
'anvil-manage-power' = > "/usr/sbin/anvil-manage-power" ,
2018-08-13 06:22:46 +00:00
'anvil-report-memory' = > "/usr/sbin/anvil-report-memory" ,
2019-02-14 09:25:52 +00:00
'anvil-update-files' = > "/usr/sbin/anvil-update-files" ,
2018-12-04 09:16:38 +00:00
'anvil-update-states' = > "/usr/sbin/anvil-update-states" ,
2018-08-13 06:22:46 +00:00
'chmod' = > "/usr/bin/chmod" ,
'chown' = > "/usr/bin/chown" ,
cp = > "/usr/bin/cp" ,
createdb = > "/usr/bin/createdb" ,
2018-11-05 08:33:43 +00:00
createrepo = > "/usr/bin/createrepo" ,
2018-08-13 06:22:46 +00:00
createuser = > "/usr/bin/createuser" ,
dmidecode = > "/usr/sbin/dmidecode" ,
2018-08-14 07:45:36 +00:00
dnf = > "/usr/bin/dnf" ,
2018-08-13 06:22:46 +00:00
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" ,
2019-02-21 06:37:51 +00:00
mv = > "/usr/bin/mv" ,
2018-08-13 06:22:46 +00:00
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" ,
2018-10-24 06:21:41 +00:00
rpm = > "/usr/bin/rpm" ,
2018-08-13 06:22:46 +00:00
rsync = > "/usr/bin/rsync" ,
sed = > "/usr/bin/sed" ,
'shutdown' = > "/usr/sbin/shutdown" ,
2019-02-27 05:23:03 +00:00
'ssh-keygen' = > "/usr/bin/ssh-keygen" ,
2018-08-13 06:22:46 +00:00
'ssh-keyscan' = > "/usr/bin/ssh-keyscan" ,
strings = > "/usr/bin/strings" ,
2018-12-06 09:14:58 +00:00
'striker-configure-host' = > "/usr/sbin/striker-configure-host" ,
'striker-manage-install-target' = > "/usr/sbin/striker-manage-install-target" ,
2019-02-06 07:36:19 +00:00
'striker-manage-peers' = > "/usr/sbin/striker-manage-peers" ,
'striker-prep-database' = > "/usr/sbin/striker-prep-database" ,
2018-08-13 06:22:46 +00:00
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" ,
2018-10-23 16:55:19 +00:00
uname = > "/usr/bin/uname" ,
2018-08-13 06:22:46 +00:00
usermod = > "/usr/sbin/usermod" ,
uuidgen = > "/usr/bin/uuidgen" ,
virsh = > "/usr/bin/virsh" ,
2019-03-29 06:51:47 +00:00
wget = > "/usr/bin/wget" ,
2017-06-20 02:32:35 +00:00
} ,
2019-02-11 08:36:41 +00:00
json = > {
files = > "files.json" ,
} ,
2017-08-02 01:04:35 +00:00
'lock' = > {
2018-08-13 06:22:46 +00:00
database = > "/tmp/anvil-tools.database.lock" ,
2017-08-02 01:04:35 +00:00
} ,
2018-12-29 08:23:58 +00:00
'log' = > {
2019-02-06 07:36:19 +00:00
file = > "/var/log/anvil.log" ,
2018-12-29 08:23:58 +00:00
} ,
2018-08-22 06:16:56 +00:00
proc = > {
uptime = > "/proc/uptime" ,
} ,
2017-06-20 02:32:35 +00:00
secure = > {
2018-08-13 06:22:46 +00:00
postgres_pgpass = > "/var/lib/pgsql/.pgpass" ,
2017-05-10 19:01:36 +00:00
} ,
2017-07-20 06:00:40 +00:00
sql = > {
2018-08-13 06:22:46 +00:00
'anvil.sql' = > "/usr/share/anvil/anvil.sql" ,
2017-07-20 06:00:40 +00:00
} ,
2018-11-01 07:16:07 +00:00
systemd = > {
httpd_enabled_symlink = > "/etc/systemd/system/multi-user.target.wants/httpd.service" ,
tftp_enabled_symlink = > "/etc/systemd/system/sockets.target.wants/tftp.socket" ,
} ,
2018-09-05 07:39:13 +00:00
urls = > {
2018-08-13 06:22:46 +00:00
skins = > "/skins" ,
2017-06-12 18:39:20 +00:00
} ,
2018-09-05 07:39:13 +00:00
words = > {
2018-08-13 06:22:46 +00:00
'words.xml' = > "/usr/share/anvil/words.xml" ,
2017-05-10 19:01:36 +00:00
} ,
2017-05-02 04:41:12 +00:00
} ;
2017-05-10 19:01:36 +00:00
# Make sure we actually have the requested files.
2017-10-20 04:19:32 +00:00
foreach my $ type ( sort { $ a cmp $ b } keys % { $ anvil - > data - > { path } } )
2017-05-02 04:41:12 +00:00
{
2018-07-14 06:51:22 +00:00
# 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.
2017-06-12 18:39:20 +00:00
next if $ type eq "urls" ;
2018-07-14 06:51:22 +00:00
next if $ type eq "configs" ;
2017-10-20 04:19:32 +00:00
foreach my $ file ( sort { $ a cmp $ b } keys % { $ anvil - > data - > { path } { $ type } } )
2017-05-02 04:41:12 +00:00
{
2017-10-20 04:19:32 +00:00
if ( not - e $ anvil - > data - > { path } { $ type } { $ file } )
2017-05-02 04:41:12 +00:00
{
2017-10-20 04:19:32 +00:00
my $ full_path = $ anvil - > Storage - > find ( { file = > $ file } ) ;
2017-06-20 02:32:35 +00:00
if ( ( $ full_path ) && ( $ full_path ne "#!not_found!#" ) )
2017-05-10 19:01:36 +00:00
{
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { path } { $ type } { $ file } = $ full_path ;
2017-05-10 19:01:36 +00:00
}
2017-05-02 04:41:12 +00:00
}
}
2017-06-09 08:29:25 +00:00
} ;
2017-05-02 04:41:12 +00:00
return ( 0 ) ;
}
2017-07-07 05:54:49 +00:00
= 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
{
2017-10-20 04:19:32 +00:00
my $ self = shift ;
my $ anvil = $ self ;
2017-07-07 05:54:49 +00:00
2017-10-20 04:19:32 +00:00
my $ short_host_name = $ anvil - > _hostname ;
2017-07-07 05:54:49 +00:00
$ short_host_name =~ s/\..*$// ;
return ( $ short_host_name ) ;
}
2017-05-02 04:41:12 +00:00
= head1 Exit Codes
= head2 C <1>
2017-10-20 04:19:32 +00:00
Anvil::Tools - > new ( ) passed something other than a hash reference .
2017-05-02 04:41:12 +00:00
= head2 C <2>
2017-10-20 04:19:32 +00:00
Failed to find the requested file in C << Anvil::Tools::Storage - > find >> and 'fatal' was set .
2017-05-02 04:41:12 +00:00
= head1 Requirements
The following packages are required on EL7 .
* C <expect>
* C <httpd>
* C <mailx>
* C <perl-Test-Simple>
* C <policycoreutils-python>
* C <postgresql>
* C <syslinux>
2017-05-27 05:23:34 +00:00
* C <perl-XML-Simple>
2017-05-02 04:41:12 +00:00
= head1 Recommended Packages
The following packages provide non - critical functionality .
* C <subscription-manager>
= cut
2017-08-17 21:16:45 +00:00
# This catches SIGINT and SIGTERM and fires out an email before shutting down.
sub catch_sig
{
my $ self = shift ;
my $ parameter = shift ;
2017-10-20 04:19:32 +00:00
my $ anvil = $ self ;
2017-08-17 21:16:45 +00:00
my $ signal = $ parameter - > { signal } ? $ parameter - > { signal } : "" ;
if ( $ signal )
{
2018-09-06 05:37:08 +00:00
print "\n\nProcess with PID: [$$] exiting on SIG" . $ signal . ".\n" ;
2018-04-26 16:41:03 +00:00
2018-05-15 05:55:56 +00:00
if ( $ anvil - > data - > { sys } { terminal } { stty } )
2018-04-26 16:41:03 +00:00
{
# Restore the terminal.
print "Restoring the terminal\n" ;
2018-05-15 05:55:56 +00:00
$ anvil - > System - > call ( { shell_call = > $ anvil - > data - > { path } { exe } { stty } . " " . $ anvil - > data - > { sys } { terminal } { stty } } ) ;
2018-04-26 16:41:03 +00:00
}
2017-08-17 21:16:45 +00:00
}
2017-10-20 04:19:32 +00:00
$ anvil - > nice_exit ( { code = > 255 } ) ;
2017-08-17 21:16:45 +00:00
}
2017-05-02 04:41:12 +00:00
1 ;