2017-10-20 04:19:32 +00:00
package Anvil::Tools::Get ;
2017-06-05 12:02:14 +00:00
#
# This module contains methods used to handle access to frequently used data.
#
use strict ;
use warnings ;
2017-08-17 21:16:45 +00:00
use Scalar::Util qw( weaken isweak ) ;
2017-06-05 12:02:14 +00:00
use Data::Dumper ;
2017-09-04 15:10:02 +00:00
use Encode ;
2018-09-23 20:16:08 +00:00
use UUID::Tiny qw( :std ) ;
2018-10-31 05:42:24 +00:00
use Net::Netmask ;
2017-06-05 12:02:14 +00:00
our $ VERSION = "3.0.0" ;
my $ THIS_FILE = "Get.pm" ;
### Methods;
2018-01-04 09:29:05 +00:00
# anvil_version
2020-06-10 22:26:50 +00:00
# bridges
2017-09-04 15:10:02 +00:00
# cgi
2017-06-05 12:02:14 +00:00
# date_and_time
2020-06-10 22:26:50 +00:00
# free_memory
# host_type
2017-06-07 16:15:03 +00:00
# host_uuid
2017-12-07 23:42:48 +00:00
# md5sum
2020-06-10 22:26:50 +00:00
# os_type
2017-06-07 16:15:03 +00:00
# switches
2020-07-03 22:11:56 +00:00
# trusted_hosts
2020-06-10 22:26:50 +00:00
# uptime
2017-07-07 05:54:49 +00:00
# users_home
2017-06-23 02:57:51 +00:00
# uuid
2018-05-15 05:55:56 +00:00
# _salt
# _wrap_to
2017-06-05 12:02:14 +00:00
= pod
= encoding utf8
= head1 NAME
2017-10-20 04:19:32 +00:00
Anvil::Tools:: Get
2017-06-05 12:02:14 +00:00
2017-06-07 16:30:28 +00:00
Provides all methods related to getting access to frequently used data .
2017-06-05 12:02:14 +00:00
= head1 SYNOPSIS
2017-10-20 04:19:32 +00:00
use Anvil::Tools ;
2017-06-05 12:02:14 +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-06-05 12:02:14 +00:00
2017-10-20 04:19:32 +00:00
# Access to methods using '$anvil->Get->X'.
2017-06-05 12:02:14 +00:00
#
# Example using 'date_and_time()';
2018-09-06 05:37:08 +00:00
my $ date = $ anvil - > Get - > date_and_time ( { ... } ) ;
2017-06-05 12:02:14 +00:00
= head1 METHODS
Methods in this module ;
= cut
sub new
{
my $ class = shift ;
2017-06-07 16:15:03 +00:00
my $ self = {
HOST = > {
UUID = > "" ,
} ,
} ;
2017-06-05 12:02:14 +00:00
bless $ self , $ class ;
return ( $ self ) ;
}
2017-10-20 04:19:32 +00:00
# Get a handle on the Anvil::Tools object. I know that technically that is a sibling module, but it makes more
2017-06-05 12:02:14 +00:00
# sense in this case to think of it as a parent.
sub parent
{
my $ self = shift ;
my $ parent = shift ;
$ self - > { HANDLE } { TOOLS } = $ parent if $ parent ;
2017-08-17 21:16:45 +00:00
# Defend against memory leads. See Scalar::Util'.
if ( not isweak ( $ self - > { HANDLE } { TOOLS } ) )
{
2019-08-07 03:31:35 +00:00
weaken ( $ self - > { HANDLE } { TOOLS } ) ;
2017-08-17 21:16:45 +00:00
}
2017-06-05 12:02:14 +00:00
return ( $ self - > { HANDLE } { TOOLS } ) ;
}
#############################################################################################################
# Public methods #
#############################################################################################################
2018-01-04 09:29:05 +00:00
= head2 anvil_version
This reads to C << VERSION >> file of a local or remote machine . If the version file isn ' t found , C << 0 >> is returned .
Parameters ;
= head3 password ( optional )
This is the password to use when connecting to a remote machine . If not set , but C << target >> is , an attempt to connect without a password will be made .
= head3 port ( optional )
This is the TCP port to use when connecting to a remote machine . If not set , but C << target >> is , C << 22 >> will be used .
2018-04-30 15:43:30 +00:00
= head3 remote_user ( optional , default root )
If C << target >> is set , this will be the user we connect to the remote machine as .
2018-01-04 09:29:05 +00:00
= head3 target ( optional )
This is the IP or host name of the machine to read the version of . If this is not set , the local system ' s version is checked .
= cut
2018-12-04 09:16:38 +00:00
# NOTE: the version is set in anvil.spec by sed'ing the release and arch onto anvil.version in anvil-core's %post
2018-01-04 09:29:05 +00:00
sub anvil_version
{
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
2018-03-07 08:11:55 +00:00
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
2018-01-04 09:29:05 +00:00
2018-04-30 15:43:30 +00:00
my $ password = defined $ parameter - > { password } ? $ parameter - > { password } : "" ;
my $ port = defined $ parameter - > { port } ? $ parameter - > { port } : "" ;
my $ remote_user = defined $ parameter - > { remote_user } ? $ parameter - > { remote_user } : "root" ;
2019-10-19 04:57:33 +00:00
my $ target = defined $ parameter - > { target } ? $ parameter - > { target } : "" ;
2018-04-30 15:43:30 +00:00
my $ version = 0 ;
2018-01-04 09:29:05 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
2019-09-09 02:54:47 +00:00
password = > $ anvil - > Log - > is_secure ( $ password ) ,
2018-04-30 15:43:30 +00:00
port = > $ port ,
remote_user = > $ remote_user ,
target = > $ target ,
2018-01-04 09:29:05 +00:00
} } ) ;
# Is this a local call or a remote call?
2019-10-19 04:57:33 +00:00
if ( $ anvil - > Network - > is_local ( { host = > $ target } ) )
{
# Local.
$ version = $ anvil - > Storage - > read_file ( { file = > $ anvil - > data - > { path } { configs } { 'anvil.version' } } ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { version = > $ version } } ) ;
# Did we actually read a version?
if ( $ version eq "!!error!!" )
{
$ version = 0 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 1 , list = > { version = > $ version } } ) ;
}
}
else
2018-01-04 09:29:05 +00:00
{
2020-06-24 04:39:56 +00:00
### TODO: Remote calls are fragile. Move the version of dashboards into a variable to read from the database.
* Got the node/dr host initialization form to the point where it can test access and decide if it should show the Red Hat account form. Decided that for M3, node/dr host setup will now be a four-stage process; initial install (over PXE), initialization (install the proper anvil-{node,dr} RPM and connect to the database), setup/map the network, and then add to an Anvil! pair.
* Updated striker to no longer try to SSH to a remote machine. To enable this, we'd have to give apache a shell and an SSH key, which is dumb and dangerous when considered.
* Created tools/striker-get-peer-data which is meant to be invoked as the 'admin' user (via a setuid c-wrapper). It collects basic data about a target machine and reports what it finds on STDOUT. It gets the password for the target via the database.
* Updated anvil-daemon to check/create/update setuid c-wrapper(s), which for now is limited to call_striker-initialize-host.
* Created Anvil/Tools/Striker.pm to store Striker web-specific methods, including get_peer_data() which calls tools/striker-initialize-host via the setuid admin call_striker-initialize-host c-wrapper.
* In order to allow striker via apache to read a peer's anvil.version, which it can no longer do over SSH, any connection to a peer where the anvil.version is read is cached as /etc/anvil/anvil.<peer>.version. When Get->anvil_version is called as 'apache', this file is read instead.
* Updated Database->resync_databases() and ->_find_behind_databases() to ignore the 'states' table.
* Created tools/striker-initialize-host which will be called as a job to initialize a node/dr host.
Signed-off-by: Digimer <digimer@alteeve.ca>
2019-09-16 04:17:02 +00:00
# Remote call. If we're running as the apache user, we need to read the cached version for
# the peer. otherwise, after we read the version, will write the cached version.
my $ user = getpwuid ( $< ) ;
my $ cache_file = $ anvil - > data - > { path } { directories } { anvil } . "/anvil." . $ target . ".version" ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
cache_file = > $ cache_file ,
user = > $ user ,
} } ) ;
if ( $ user eq "apache" )
{
# Try to read the local cached version.
if ( - e $ cache_file )
{
# Read it in.
$ version = $ anvil - > Storage - > read_file ( { file = > $ cache_file } ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { version = > $ version } } ) ;
}
}
else
{
my $ shell_call = "
2018-01-17 21:10:07 +00:00
if [ - e ".$anvil->data->{path}{configs}{'anvil.version'}." ] ;
2018-01-04 09:29:05 +00:00
then
2018-01-17 21:10:07 +00:00
cat ".$anvil->data->{path}{configs}{'anvil.version'}." ;
2018-01-04 09:29:05 +00:00
else
echo 0 ;
fi ;
" ;
* Got the node/dr host initialization form to the point where it can test access and decide if it should show the Red Hat account form. Decided that for M3, node/dr host setup will now be a four-stage process; initial install (over PXE), initialization (install the proper anvil-{node,dr} RPM and connect to the database), setup/map the network, and then add to an Anvil! pair.
* Updated striker to no longer try to SSH to a remote machine. To enable this, we'd have to give apache a shell and an SSH key, which is dumb and dangerous when considered.
* Created tools/striker-get-peer-data which is meant to be invoked as the 'admin' user (via a setuid c-wrapper). It collects basic data about a target machine and reports what it finds on STDOUT. It gets the password for the target via the database.
* Updated anvil-daemon to check/create/update setuid c-wrapper(s), which for now is limited to call_striker-initialize-host.
* Created Anvil/Tools/Striker.pm to store Striker web-specific methods, including get_peer_data() which calls tools/striker-initialize-host via the setuid admin call_striker-initialize-host c-wrapper.
* In order to allow striker via apache to read a peer's anvil.version, which it can no longer do over SSH, any connection to a peer where the anvil.version is read is cached as /etc/anvil/anvil.<peer>.version. When Get->anvil_version is called as 'apache', this file is read instead.
* Updated Database->resync_databases() and ->_find_behind_databases() to ignore the 'states' table.
* Created tools/striker-initialize-host which will be called as a job to initialize a node/dr host.
Signed-off-by: Digimer <digimer@alteeve.ca>
2019-09-16 04:17:02 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , key = > "log_0166" , variables = > { shell_call = > $ shell_call , target = > $ target , remote_user = > $ remote_user } } ) ;
my ( $ output , $ error , $ return_code ) = $ anvil - > Remote - > call ( {
debug = > $ debug ,
shell_call = > $ shell_call ,
target = > $ target ,
port = > $ port ,
password = > $ password ,
remote_user = > $ remote_user ,
} ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
error = > $ error ,
output = > $ output ,
} } ) ;
$ version = defined $ output ? $ output : "" ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { version = > $ version } } ) ;
# Create/Update the cache file.
if ( $ version )
{
my $ update_cache = 1 ;
my $ old_version = "" ;
if ( - e $ cache_file )
{
$ old_version = $ anvil - > Storage - > read_file ( { file = > $ cache_file } ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { old_version = > $ old_version } } ) ;
if ( $ old_version eq $ version )
{
# No need to update
$ update_cache = 0 ;
}
else
{
}
}
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { update_cache = > $ update_cache } } ) ;
if ( $ update_cache )
{
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 1 , key = > "log_0437" , variables = > {
target = > $ target ,
file = > $ cache_file ,
} } ) ;
$ anvil - > Storage - > write_file ( {
debug = > $ debug ,
file = > $ cache_file ,
body = > $ version ,
mode = > "0666" ,
overwrite = > 1 ,
} ) ;
}
}
}
2018-01-04 09:29:05 +00:00
}
2018-12-04 09:16:38 +00:00
# Clear off any newline.
$ version =~ s/\n//gs ;
2018-01-04 09:29:05 +00:00
return ( $ version ) ;
}
2020-06-10 22:26:50 +00:00
= head2 bridges
This finds a list of bridges on the host . Bridges that are found are stored is '
= cut
sub bridges
{
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , key = > "log_0125" , variables = > { method = > "Get->bridges()" } } ) ;
my ( $ output , $ return_code ) = $ anvil - > System - > call ( { shell_call = > $ anvil - > data - > { path } { exe } { bridge } . " -json -details link show" } ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
output = > $ output ,
return_code = > $ return_code ,
} } ) ;
# Delete any previously known data
if ( exists $ anvil - > data - > { 'local' } { network } { bridges } )
{
delete $ anvil - > data - > { 'local' } { network } { bridges } ;
} ;
my $ json = JSON - > new - > allow_nonref ;
my $ bridge_data = $ json - > decode ( $ output ) ;
#print Dumper $bridge_data;
foreach my $ hash_ref ( @ { $ bridge_data } )
{
# If the ifname and master are the same, it's a bridge.
my $ type = "interface" ;
my $ interface = $ hash_ref - > { ifname } ;
my $ master_bridge = $ hash_ref - > { master } ;
if ( $ interface eq $ master_bridge )
{
$ type = "bridge" ;
$ anvil - > data - > { 'local' } { network } { bridges } { bridge } { $ interface } { found } = 1 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
"local::network::bridges::bridge::${interface}::found" = > $ anvil - > data - > { 'local' } { network } { bridges } { bridge } { $ interface } { found } ,
} } ) ;
}
else
{
# Store this interface under the bridge.
$ anvil - > data - > { 'local' } { network } { bridges } { bridge } { $ master_bridge } { connected_interface } { $ interface } = 1 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
"local::network::bridges::bridge::${master_bridge}::connected_interface::${interface}" = > $ anvil - > data - > { 'local' } { network } { bridges } { bridge } { $ master_bridge } { connected_interface } { $ interface } ,
} } ) ;
}
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
interface = > $ interface ,
master_bridge = > $ master_bridge ,
type = > $ type ,
} } ) ;
foreach my $ key ( sort { $ a cmp $ b } keys % { $ hash_ref } )
{
if ( ref ( $ hash_ref - > { $ key } ) eq "ARRAY" )
{
$ anvil - > data - > { 'local' } { network } { bridges } { $ type } { $ interface } { $ key } = [] ;
foreach my $ value ( sort { $ a cmp $ b } @ { $ hash_ref - > { $ key } } )
{
push @ { $ anvil - > data - > { 'local' } { network } { bridges } { $ type } { $ interface } { $ key } } , $ value ;
}
for ( my $ i = 0 ; $ i < @ { $ anvil - > data - > { 'local' } { network } { bridges } { $ type } { $ interface } { $ key } } ; $ i + + )
{
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
"local::network::bridges::${type}::${interface}::${key}->[$i]" = > $ anvil - > data - > { 'local' } { network } { bridges } { $ type } { $ interface } { $ key } - > [ $ i ] ,
} } ) ;
}
}
else
{
$ anvil - > data - > { 'local' } { network } { bridges } { $ type } { $ interface } { $ key } = $ hash_ref - > { $ key } ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
"local::network::bridges::${type}::${interface}::${key}" = > $ anvil - > data - > { 'local' } { network } { bridges } { $ type } { $ interface } { $ key } ,
} } ) ;
}
}
}
# Summary of found bridges.
foreach my $ interface ( sort { $ a cmp $ b } keys % { $ anvil - > data - > { 'local' } { network } { bridges } { bridge } } )
{
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
"local::network::bridges::bridge::${interface}::found" = > $ anvil - > data - > { 'local' } { network } { bridges } { bridge } { $ interface } { found } ,
} } ) ;
}
return ( 0 ) ;
}
2017-09-04 15:10:02 +00:00
= head2 cgi
This reads in the CGI variables passed in by a form or URL .
2019-09-09 02:54:47 +00:00
This method takes no parameters .
2017-09-05 08:38:25 +00:00
2017-09-04 15:10:02 +00:00
= cut
sub cgi
{
my $ self = shift ;
my $ parameter = shift ;
2017-10-20 04:19:32 +00:00
my $ anvil = $ self - > parent ;
2018-03-07 08:11:55 +00:00
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
2017-09-04 15:10:02 +00:00
# This will store all of the CGI variables.
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { sys } { cgi_string } = "?" ;
2017-09-04 15:10:02 +00:00
# Needed to read in passed CGI variables
my $ cgi = CGI - > new ( ) ;
my $ cgis = [] ;
my $ cgi_count = 0 ;
2019-02-08 08:21:51 +00:00
# Get the list of parameters coming in, if possible,
if ( exists $ cgi - > { param } )
2017-09-04 15:10:02 +00:00
{
2019-02-08 08:21:51 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { 'cgi->{param}' = > $ cgi - > { param } } } ) ;
foreach my $ variable ( sort { $ a cmp $ b } keys % { $ cgi - > { param } } )
2017-09-04 15:10:02 +00:00
{
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { variable = > $ variable } } ) ;
2017-09-04 15:10:02 +00:00
push @ { $ cgis } , $ variable ;
}
}
2018-06-27 22:13:16 +00:00
$ cgi_count = @ { $ cgis } ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { cgi_count = > $ cgi_count } } ) ;
2017-09-04 15:10:02 +00:00
# If we don't have at least one variable, we're done.
if ( $ cgi_count < 1 )
{
return ( 0 ) ;
}
# NOTE: Later, we will have another array for handling file uploads.
# Now read in the variables.
foreach my $ variable ( sort { $ a cmp $ b } @ { $ cgis } )
{
2019-09-09 02:54:47 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { variable = > $ variable } } ) ;
2017-09-04 15:10:02 +00:00
2019-02-08 08:21:51 +00:00
$ anvil - > data - > { cgi } { $ variable } { value } = "" ;
$ anvil - > data - > { cgi } { $ variable } { mime_type } = "string" ;
$ anvil - > data - > { cgi } { $ variable } { file_handle } = "" ;
$ anvil - > data - > { cgi } { $ variable } { file_name } = "" ;
$ anvil - > data - > { cgi } { $ variable } { alert } = 0 ; # This is set if a sanity check fails
2017-09-05 08:38:25 +00:00
2019-02-08 08:21:51 +00:00
# This is a special CGI key for download files (upload from the user's perspective)
if ( $ variable eq "upload_file" )
2017-09-05 08:38:25 +00:00
{
2019-02-08 08:21:51 +00:00
if ( not $ cgi - > upload ( 'upload_file' ) )
2017-09-05 08:38:25 +00:00
{
# Empty file passed, looks like the user forgot to select a file to upload.
2019-02-08 08:21:51 +00:00
$ anvil - > Log - > entry ( { log_level = > 2 , message_key = > "log_0242" , file = > $ THIS_FILE , line = > __LINE__ } ) ;
2017-09-05 08:38:25 +00:00
}
else
{
2019-02-08 08:21:51 +00:00
$ anvil - > data - > { cgi } { upload_file } { file_handle } = $ cgi - > upload ( 'upload_file' ) ;
my $ file = $ anvil - > data - > { cgi } { upload_file } { file_handle } ;
$ anvil - > data - > { cgi } { upload_file } { file_name } = $ file ;
$ anvil - > data - > { cgi } { upload_file } { mime_type } = $ cgi - > uploadInfo ( $ file ) - > { 'Content-Type' } ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 2 , list = > {
variable = > 'upload_file' ,
"cgi::${variable}::file_handle" = > $ anvil - > data - > { cgi } { upload_file } { file_handle } ,
"cgi::${variable}::file_handle->handle" = > $ anvil - > data - > { cgi } { upload_file } { file_handle } - > handle ,
"cgi::${variable}::file_name" = > $ anvil - > data - > { cgi } { upload_file } { file_name } ,
"cgi::${variable}::mime_type" = > $ anvil - > data - > { cgi } { upload_file } { mime_type } ,
"cgi->upload('upload_file')" = > $ cgi - > upload ( 'upload_file' ) ,
"cgi->upload('upload_file')->handle" = > $ cgi - > upload ( 'upload_file' ) - > handle ,
2017-09-05 08:38:25 +00:00
} } ) ;
}
}
2017-09-04 15:10:02 +00:00
if ( defined $ cgi - > param ( $ variable ) )
{
2017-09-05 08:38:25 +00:00
# Make this UTF8 if it isn't already.
if ( Encode:: is_utf8 ( $ cgi - > param ( $ variable ) ) )
{
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { cgi } { $ variable } { value } = $ cgi - > param ( $ variable ) ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { "cgi::${variable}::value" = > $ anvil - > data - > { cgi } { $ variable } { value } } } ) ;
2017-09-05 08:38:25 +00:00
}
else
{
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { cgi } { $ variable } { value } = Encode:: decode_utf8 ( $ cgi - > param ( $ variable ) ) ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { "cgi::${variable}::value" = > $ anvil - > data - > { cgi } { $ variable } { value } } } ) ;
2017-09-05 08:38:25 +00:00
}
2019-09-07 12:15:11 +00:00
# Append to 'sys::cgi_string', so long as the variable doesn't have 'passwd' or 'password' in it.
if ( ( $ variable !~ /password/ ) && ( $ variable !~ /passwd/ ) )
{
$ anvil - > data - > { sys } { cgi_string } . = "$variable=" . $ anvil - > data - > { cgi } { $ variable } { value } . "&" ;
}
2017-09-05 08:38:25 +00:00
}
}
# This is a pretty way of displaying the passed-in CGI variables. It loops through all we've got and
# sorts out the longest variable name. Then it loops again, appending '.' to shorter ones so that
2018-03-27 03:55:39 +00:00
# everything is lined up in the logs. This almost always prints, save for log level 0.
if ( $ anvil - > Log - > level >= 1 )
2017-09-05 08:38:25 +00:00
{
my $ longest_variable = 0 ;
2017-10-20 04:19:32 +00:00
foreach my $ variable ( sort { $ a cmp $ b } keys % { $ anvil - > data - > { cgi } } )
2017-09-05 08:38:25 +00:00
{
2017-10-20 04:19:32 +00:00
next if $ anvil - > data - > { cgi } { $ variable } eq "" ;
2017-09-05 08:38:25 +00:00
if ( length ( $ variable ) > $ longest_variable )
{
$ longest_variable = length ( $ variable ) ;
}
2017-09-04 15:10:02 +00:00
}
2019-09-09 02:54:47 +00:00
# Now loop again.
2017-09-05 08:38:25 +00:00
foreach my $ variable ( @ { $ cgis } )
2017-09-04 15:10:02 +00:00
{
2017-10-20 04:19:32 +00:00
next if $ anvil - > data - > { cgi } { $ variable } eq "" ;
2017-09-05 08:38:25 +00:00
my $ difference = $ longest_variable - length ( $ variable ) ;
my $ say_value = "value" ;
if ( $ difference == 0 )
{
# Do nothing
}
elsif ( $ difference == 1 )
{
$ say_value . = " " ;
}
elsif ( $ difference == 2 )
{
$ say_value . = " " ;
}
else
{
my $ dots = $ difference - 2 ;
$ say_value . = " " ;
for ( 1 .. $ dots )
{
$ say_value . = "." ;
}
$ say_value . = " " ;
}
2018-03-27 03:55:39 +00:00
# This is always '1' as the passed-in variables are what we want to see.
my $ censored_value = $ anvil - > data - > { cgi } { $ variable } { value } ;
if ( ( ( $ variable =~ /passwd/ ) or ( $ variable =~ /password/ ) ) && ( not $ anvil - > Log - > secure ) )
{
# This is a password and we're not logging sensitive data, obfuscate it.
2018-07-11 18:07:07 +00:00
$ censored_value = $ anvil - > Words - > string ( { key = > "log_0186" } ) ;
2018-03-27 03:55:39 +00:00
}
2019-09-09 02:54:47 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 1 , list = > {
2018-03-27 03:55:39 +00:00
"cgi::${variable}::$say_value" = > $ censored_value ,
2017-09-05 08:38:25 +00:00
} } ) ;
2017-09-04 15:10:02 +00:00
}
}
2017-09-05 08:38:25 +00:00
# Clear the last &
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { sys } { cgi_string } =~ s/&$// ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { "sys::cgi_string" = > $ anvil - > data - > { sys } { cgi_string } } } ) ;
2017-09-05 08:38:25 +00:00
2017-09-04 15:10:02 +00:00
return ( 0 ) ;
}
2017-06-05 12:02:14 +00:00
= head2 date_and_time
This method returns the date and / or time using either the current time , or a specified unix time .
NOTE: This only returns times in 24 - hour notation .
= head2 Parameters ;
= head3 date_only ( optional )
If set , only the date will be returned ( in C << yyyy /mm/ dd >> format ) .
= head3 file_name ( optional )
When set , the date and /or time returned in a string more useful in file names. Specifically, it will replace spaces with 'C<< _ >>' and 'C<< : >>' and 'C<< / >> ' for ' C << - >> '. This will result in a string in the format like ' C << yyyy - mm - dd_hh - mm - ss >> ' .
= head3 offset ( optional )
If set to a signed number , it will add or subtract the number of seconds from the 'C<< use_time >>' before processing .
= head3 use_time ( optional )
This can be set to a unix timestamp . If it is not set , the current time is used .
= head3 time_only ( optional )
If set , only the time will be returned ( in C << hh:mm:ss >> format ) .
2019-12-04 05:02:19 +00:00
= head3 use_utc ( optional )
If set , C << gmtime >> is used instead of C << localtime >> . The effect of this is that GMTime ( greenwhich mean time , UTC - 0 ) is used instead of the local system ' s time zone .
2017-06-05 12:02:14 +00:00
= cut
sub date_and_time
{
my $ self = shift ;
my $ parameter = shift ;
2017-10-20 04:19:32 +00:00
my $ anvil = $ self - > parent ;
2018-03-07 08:11:55 +00:00
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
2017-06-05 12:02:14 +00:00
my $ offset = defined $ parameter - > { offset } ? $ parameter - > { offset } : 0 ;
my $ use_time = defined $ parameter - > { use_time } ? $ parameter - > { use_time } : time ;
2019-12-04 05:02:19 +00:00
my $ use_utc = defined $ parameter - > { use_utc } ? $ parameter - > { use_utc } : 0 ;
2017-06-05 12:02:14 +00:00
my $ file_name = defined $ parameter - > { file_name } ? $ parameter - > { file_name } : 0 ;
my $ time_only = defined $ parameter - > { time_only } ? $ parameter - > { time_only } : 0 ;
my $ date_only = defined $ parameter - > { date_only } ? $ parameter - > { date_only } : 0 ;
2018-09-06 05:37:08 +00:00
### NOTE: This is used too early for normal error handling.
2017-06-05 12:02:14 +00:00
# Are things sane?
if ( $ use_time =~ /D/ )
{
die "Get->date_and_time() was called with 'use_time' set to: [$use_time]. Only a unix timestamp is allowed.\n" ;
}
if ( $ offset =~ /D/ )
{
die "Get->date_and_time() was called with 'offset' set to: [$offset]. Only real number is allowed.\n" ;
}
# Do my initial calculation.
my $ return_string = "" ;
my $ time = { } ;
my $ adjusted_time = $ use_time + $ offset ;
#print $THIS_FILE." ".__LINE__."; [ Debug ] - adjusted_time: [$adjusted_time]\n";
# Get the date and time pieces
2019-12-04 05:02:19 +00:00
if ( $ use_utc )
{
( $ time - > { sec } , $ time - > { min } , $ time - > { hour } , $ time - > { mday } , $ time - > { mon } , $ time - > { year } , $ time - > { wday } , $ time - > { yday } , $ time - > { isdst } ) = gmtime ( $ adjusted_time ) ;
#print $THIS_FILE." ".__LINE__."; [ Debug ] - time->{sec}: [".$time->{sec}."], time->{min}: [".$time->{min}."], time->{hour}: [".$time->{hour}."], time->{mday}: [".$time->{mday}."], time->{mon}: [".$time->{mon}."], time->{year}: [".$time->{year}."], time->{wday}: [".$time->{wday}."], time->{yday}: [".$time->{yday}."], time->{isdst}: [".$time->{isdst}."]\n";
}
else
{
( $ time - > { sec } , $ time - > { min } , $ time - > { hour } , $ time - > { mday } , $ time - > { mon } , $ time - > { year } , $ time - > { wday } , $ time - > { yday } , $ time - > { isdst } ) = localtime ( $ adjusted_time ) ;
#print $THIS_FILE." ".__LINE__."; [ Debug ] - time->{sec}: [".$time->{sec}."], time->{min}: [".$time->{min}."], time->{hour}: [".$time->{hour}."], time->{mday}: [".$time->{mday}."], time->{mon}: [".$time->{mon}."], time->{year}: [".$time->{year}."], time->{wday}: [".$time->{wday}."], time->{yday}: [".$time->{yday}."], time->{isdst}: [".$time->{isdst}."]\n";
}
2017-06-05 12:02:14 +00:00
# Process the raw data
$ time - > { pad_hour } = sprintf ( "%02d" , $ time - > { hour } ) ;
$ time - > { mon } + + ;
$ time - > { pad_min } = sprintf ( "%02d" , $ time - > { min } ) ;
$ time - > { pad_sec } = sprintf ( "%02d" , $ time - > { sec } ) ;
$ time - > { year } = ( $ time - > { year } + 1900 ) ;
$ time - > { pad_mon } = sprintf ( "%02d" , $ time - > { mon } ) ;
$ time - > { pad_mday } = sprintf ( "%02d" , $ time - > { mday } ) ;
#print $THIS_FILE." ".__LINE__."; [ Debug ] - time->{pad_hour}: [".$time->{pad_hour}."], time->{pad_min}: [".$time->{pad_min}."], time->{pad_sec}: [".$time->{pad_sec}."], time->{year}: [".$time->{year}."], time->{pad_mon}: [".$time->{pad_mon}."], time->{pad_mday}: [".$time->{pad_mday}."], time->{mon}: [".$time->{mon}."]\n";
# Now, the date and time separator depends on if 'file_name' is set.
my $ date_separator = $ file_name ? "-" : "/" ;
my $ time_separator = $ file_name ? "-" : ":" ;
my $ space_separator = $ file_name ? "_" : " " ;
if ( $ time_only )
{
$ return_string = $ time - > { pad_hour } . $ time_separator . $ time - > { pad_min } . $ time_separator . $ time - > { pad_sec } ;
#print $THIS_FILE." ".__LINE__."; [ Debug ] - return_string: [$return_string]\n";
}
elsif ( $ date_only )
{
$ return_string = $ time - > { year } . $ date_separator . $ time - > { pad_mon } . $ date_separator . $ time - > { pad_mday } ;
#print $THIS_FILE." ".__LINE__."; [ Debug ] - return_string: [$return_string]\n";
}
else
{
$ return_string = $ time - > { year } . $ date_separator . $ time - > { pad_mon } . $ date_separator . $ time - > { pad_mday } . $ space_separator . $ time - > { pad_hour } . $ time_separator . $ time - > { pad_min } . $ time_separator . $ time - > { pad_sec } ;
#print $THIS_FILE." ".__LINE__."; [ Debug ] - return_string: [$return_string]\n";
}
return ( $ return_string ) ;
}
2017-06-07 16:15:03 +00:00
2018-09-28 05:50:38 +00:00
= head2 host_name
This takes a host UUID and returns the host name ( as recorded in the C << hosts >> table ) . If the entry is not found , an empty string is returned .
Parameters ;
= head3 host_uuid ( required )
This is the C << host_uuid >> to translate into a host name .
= cut
sub host_name
{
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
my $ host_name = "" ;
my $ host_uuid = defined $ parameter - > { host_uuid } ? $ parameter - > { host_uuid } : "" ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { host_uuid = > $ host_uuid } } ) ;
my $ query = "
SELECT
host_name
FROM
hosts
WHERE
2019-03-06 06:49:59 +00:00
host_uuid = ".$anvil->Database->quote($host_uuid)." ;
2018-09-28 05:50:38 +00:00
" ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { query = > $ query } } ) ;
my $ results = $ anvil - > Database - > query ( { query = > $ query , source = > $ THIS_FILE , line = > __LINE__ } ) ;
my $ count = @ { $ results } ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
results = > $ results ,
count = > $ count ,
} } ) ;
if ( $ count == 1 )
{
# Found it
$ host_name = defined $ results - > [ 0 ] - > [ 0 ] ? $ results - > [ 0 ] - > [ 0 ] : "" ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { host_name = > $ host_name } } ) ;
}
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { host_name = > $ host_name } } ) ;
return ( $ host_name ) ;
}
2020-06-10 22:26:50 +00:00
= head2 free_memory
This returns , in bytes , host much free memory is available on the local system .
= cut
### TODO: Make this work on remote systems.
sub free_memory
{
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , key = > "log_0125" , variables = > { method = > "Get->free_memory()" } } ) ;
my $ available = 0 ;
my ( $ free_output , $ free_rc ) = $ anvil - > System - > call ( { shell_call = > $ anvil - > data - > { path } { exe } { free } . " --bytes" } ) ;
foreach my $ line ( split /\n/ , $ free_output )
{
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 3 , list = > { line = > $ line } } ) ;
if ( $ line =~ /Mem:\s+(\d+)\s+(\d+)\s+(\d+)\s+(\d+)\s+(\d+)\s+(\d+)$/ )
{
my $ total = $ 1 ;
my $ used = $ 2 ;
my $ free = $ 3 ;
my $ shared = $ 4 ;
my $ cache = $ 5 ;
$ available = $ 6 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
total = > $ total . " (" . $ anvil - > Convert - > bytes_to_human_readable ( { 'bytes' = > $ total } ) . "})" ,
used = > $ used . " (" . $ anvil - > Convert - > bytes_to_human_readable ( { 'bytes' = > $ used } ) . "})" ,
free = > $ free . " (" . $ anvil - > Convert - > bytes_to_human_readable ( { 'bytes' = > $ free } ) . "})" ,
shared = > $ shared . " (" . $ anvil - > Convert - > bytes_to_human_readable ( { 'bytes' = > $ shared } ) . "})" ,
cache = > $ cache . " (" . $ anvil - > Convert - > bytes_to_human_readable ( { 'bytes' = > $ cache } ) . "})" ,
available = > $ available . " (" . $ anvil - > Convert - > bytes_to_human_readable ( { 'bytes' = > $ available } ) . "})" ,
} } ) ;
}
}
return ( $ available ) ;
}
= head2 host_type
This method tries to determine the host type and returns a value suitable for use is the C << hosts >> table .
my $ type = $ anvil - > System - > host_type ( ) ;
First , it looks to see if C << sys:: host_type >> is set and , if so , uses that string as it is .
If that isn ' t set , it then looks to see if the file C << /etc/ anvil / type . X >> exists , where C << X >> is C << node >> , C << striker >> or C << dr >> . If found , the appropriate type is returned .
If that file doesn ' t exist , then it then checks to see which C << anvil - <type> >> rpm is installed . In order , it looks for C << anvil - striker >> , then C << anvil - node >> and finally C << anvil - dr >> . If one of them is found , the appropriate C << /etc/ anvil / type . X >> is created .
= cut
sub host_type
{
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , key = > "log_0125" , variables = > { method = > "Get->host_type()" } } ) ;
my $ host_type = "" ;
my $ host_name = $ anvil - > _short_host_name ;
$ host_type = "unknown" ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
host_type = > $ host_type ,
host_name = > $ host_name ,
"sys::host_type" = > $ anvil - > data - > { sys } { host_type } ,
} } ) ;
if ( $ anvil - > data - > { sys } { host_type } )
{
$ host_type = $ anvil - > data - > { sys } { host_type } ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { host_type = > $ host_type } } ) ;
}
else
{
# Can I determine it by seeing a file?
if ( - e $ anvil - > data - > { path } { configs } { 'type.node' } )
{
$ host_type = "node" ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { host_type = > $ host_type } } ) ;
}
elsif ( - e $ anvil - > data - > { path } { configs } { 'type.striker' } )
{
$ host_type = "striker" ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { host_type = > $ host_type } } ) ;
}
elsif ( - e $ anvil - > data - > { path } { configs } { 'type.dr' } )
{
$ host_type = "dr" ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { host_type = > $ host_type } } ) ;
}
else
{
# Last gasp here is to use 'rpm' to see which RPMs are installed. If we find one,
# we'll touch 'type.X' file
2020-07-18 02:49:25 +00:00
my $ check_types = {
'striker' = > 1 ,
'node' = > 1 ,
'dr' = > 1 ,
} ;
2020-06-10 22:26:50 +00:00
foreach my $ rpm ( "anvil-striker" , "anvil-node" , "anvil-dr" )
{
2020-07-18 02:49:25 +00:00
my ( $ output , $ return_code ) = $ anvil - > System - > call ( {
debug = > $ debug ,
shell_call = > $ anvil - > data - > { path } { exe } { rpm } . " -q " . $ rpm ,
} ) ;
2020-06-10 22:26:50 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { output = > $ output , return_code = > $ return_code } } ) ;
if ( $ return_code eq "0" )
{
# Found out what we are.
2020-07-18 02:49:25 +00:00
if ( $ output =~ /anvil-(.*?)-/ )
2020-06-10 22:26:50 +00:00
{
2020-07-18 02:49:25 +00:00
$ host_type = $ 1 ;
2020-06-10 22:26:50 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { host_type = > $ host_type } } ) ;
2020-07-18 02:49:25 +00:00
$ check_types - > { $ host_type } = 0 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { "check_types->{$host_type}" = > $ check_types - > { $ host_type } } } ) ;
2020-06-10 22:26:50 +00:00
}
my $ key = "type." . $ host_type ;
my $ file = $ anvil - > data - > { path } { configs } { $ key } ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
key = > $ key ,
file = > $ file ,
} } ) ;
# If we have a file and we're root, touch to the file.
2020-07-18 02:49:25 +00:00
if ( ( $< == 0 ) or ( $> == 0 ) )
2020-06-10 22:26:50 +00:00
{
2020-07-18 02:49:25 +00:00
foreach my $ test_type ( sort { $ a cmp $ b } keys % { $ check_types } )
{
my $ test_key = "type." . $ test_type ;
my $ test_file = $ anvil - > data - > { path } { configs } { $ test_key } ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
"check_types->{$test_type}" = > $ check_types - > { $ test_type } ,
test_file = > $ test_file ,
} } ) ;
if ( ( $ check_types - > { $ test_type } ) && ( - e $ test_file ) )
{
# Remove the old type.
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 1 , key = > "log_0497" , variables = > { file = > $ test_file } } ) ;
unlink $ test_file ;
}
}
if ( $ file )
{
my $ error = $ anvil - > Storage - > write_file ( {
debug = > $ debug ,
body = > "" ,
file = > $ file ,
} ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { error = > $ error } } ) ;
}
2020-06-10 22:26:50 +00:00
}
last ;
}
}
}
}
return ( $ host_type ) ;
}
2017-06-07 16:15:03 +00:00
= head2 host_uuid
2018-07-26 01:02:48 +00:00
This returns the local host 's system UUID (as reported by ' dmidecode '). If the host UUID isn' t available , and the program is not running with root priviledges , C << #!error!# >> is returned.
2017-06-07 16:15:03 +00:00
2017-10-20 04:19:32 +00:00
print "This host's UUID: [" . $ anvil - > Get - > host_uuid . "]\n" ;
2017-06-07 16:15:03 +00:00
It is possible to override the local UUID , though it is not recommended .
2017-10-20 04:19:32 +00:00
$ anvil - > Get - > host_uuid ( { set = > "720a0509-533d-406b-8fc1-03aca3e75fa7" } )
2017-06-07 16:15:03 +00:00
= cut
sub host_uuid
{
my $ self = shift ;
my $ parameter = shift ;
2017-10-20 04:19:32 +00:00
my $ anvil = $ self - > parent ;
2018-03-07 08:11:55 +00:00
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
2017-06-07 16:15:03 +00:00
2018-05-15 05:55:56 +00:00
my $ set = defined $ parameter - > { set } ? $ parameter - > { set } : "" ;
2019-07-17 06:41:05 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
set = > $ set ,
'HOST::UUID' = > $ anvil - > { HOST } { UUID } ,
} } ) ;
2017-06-07 16:15:03 +00:00
if ( $ set )
{
2019-07-17 06:41:05 +00:00
$ anvil - > { HOST } { UUID } = $ set ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { "HOST::UUID" = > $ anvil - > { HOST } { UUID } } } ) ;
2017-06-07 16:15:03 +00:00
}
2019-07-17 06:41:05 +00:00
elsif ( not $ anvil - > { HOST } { UUID } )
2017-06-07 16:15:03 +00:00
{
2019-07-17 06:41:05 +00:00
# Read /etc/anvil/host.uuid if it exists. If not, and if we're root, we'll create that file
# using the UUID from dmidecode.
2017-09-27 07:24:27 +00:00
my $ uuid = "" ;
2019-07-17 06:41:05 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
'$<' = > $< ,
'$>' = > $> ,
'path::data::host_uuid' = > $ anvil - > data - > { path } { data } { host_uuid } ,
} } ) ;
if ( - e $ anvil - > data - > { path } { data } { host_uuid } )
2017-06-07 16:15:03 +00:00
{
2019-07-17 06:41:05 +00:00
# Read the UUID in
$ uuid = $ anvil - > Storage - > read_file ( { debug = > $ debug , file = > $ anvil - > data - > { path } { data } { host_uuid } } ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { uuid = > $ uuid } } ) ;
}
elsif ( ( $< == 0 ) or ( $> == 0 ) )
{
# Create the UUID file.
2019-10-02 06:04:22 +00:00
( $ uuid , my $ return_code ) = $ anvil - > System - > call ( { debug = > $ debug , shell_call = > $ anvil - > data - > { path } { exe } { dmidecode } . " --string system-uuid" } ) ;
$ uuid = lc ( $ uuid ) ;
2019-07-17 06:41:05 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
uuid = > $ uuid ,
return_code = > $ return_code ,
} } ) ;
2017-09-27 07:24:27 +00:00
}
else
{
2019-07-17 06:41:05 +00:00
# Host UUID file doesn't exist and I'm Not running as root, I'm done.
# We're done.
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 0 , priority = > "err" , key = > "log_0187" } ) ;
return ( "#!error!#" ) ;
2017-06-07 16:15:03 +00:00
}
2020-07-07 05:18:38 +00:00
if ( $ anvil - > Validate - > uuid ( { uuid = > $ uuid } ) )
2017-06-07 16:15:03 +00:00
{
2019-07-17 06:41:05 +00:00
$ anvil - > { HOST } { UUID } = $ uuid ;
2017-10-20 04:19:32 +00:00
if ( not - e $ anvil - > data - > { path } { data } { host_uuid } )
2017-10-19 17:20:01 +00:00
{
### TODO: This will need to set the proper SELinux context.
# Apache run scripts can't call the system UUID, so we'll write it to a text
# file.
2017-10-20 04:19:32 +00:00
$ anvil - > Storage - > write_file ( {
* 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
debug = > $ debug ,
2017-10-20 04:19:32 +00:00
file = > $ anvil - > data - > { path } { data } { host_uuid } ,
2017-10-19 17:20:01 +00:00
body = > $ uuid ,
user = > "apache" ,
group = > "apache" ,
mode = > "0666" ,
overwrite = > 0 ,
} ) ;
2017-10-20 04:19:32 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 1 , key = > "message_0011" , variables = > { file = > $ anvil - > data - > { path } { configs } { 'postgresql.conf' } } } ) ;
2017-10-19 17:20:01 +00:00
}
2017-06-07 16:15:03 +00:00
}
2017-09-27 07:24:27 +00:00
else
{
# Bad UUID.
2019-07-17 06:41:05 +00:00
$ anvil - > { HOST } { UUID } = "" ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { "HOST::UUID" = > $ anvil - > { HOST } { UUID } } } ) ;
2017-10-20 04:19:32 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 0 , priority = > "err" , key = > "log_0134" , variables = > { uuid = > $ uuid } } ) ;
2019-07-17 06:41:05 +00:00
return ( "#!error!#" ) ;
2017-09-27 07:24:27 +00:00
}
2017-06-07 16:15:03 +00:00
}
2018-05-15 05:55:56 +00:00
# We'll also store the host UUID in a variable.
2019-07-17 06:41:05 +00:00
if ( ( not $ anvil - > data - > { sys } { host_uuid } ) && ( $ anvil - > { HOST } { UUID } ) )
2018-05-15 05:55:56 +00:00
{
2019-07-17 06:41:05 +00:00
$ anvil - > data - > { sys } { host_uuid } = $ anvil - > { HOST } { UUID } ;
2018-05-15 05:55:56 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { "sys::host_uuid" = > $ anvil - > data - > { sys } { host_uuid } } } ) ;
}
2019-07-17 06:41:05 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { "HOST::UUID" = > $ anvil - > { HOST } { UUID } } } ) ;
return ( $ anvil - > { HOST } { UUID } ) ;
2017-06-07 16:15:03 +00:00
}
2017-12-07 23:42:48 +00:00
= head2 md5sum
This returns the C << md5sum >> of a given file .
Parameters ;
= head3 file
This is the full or relative path to the file . If the file doesn ' t exist , an empty string is returned .
= cut
sub md5sum
{
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
2018-03-07 08:11:55 +00:00
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
2017-12-07 23:42:48 +00:00
my $ sum = "" ;
my $ file = defined $ parameter - > { file } ? $ parameter - > { file } : "" ;
2019-02-21 06:37:51 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { file = > $ file } } ) ;
2017-12-07 23:42:48 +00:00
if ( - e $ file )
{
my $ shell_call = $ anvil - > data - > { path } { exe } { md5sum } . " " . $ file ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { shell_call = > $ shell_call } } ) ;
2017-12-07 23:42:48 +00:00
2019-07-13 08:16:03 +00:00
my ( $ return , $ return_code ) = $ anvil - > System - > call ( { debug = > $ debug , shell_call = > $ shell_call } ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { 'return' = > $ return , return_code = > $ return_code } } ) ;
2017-12-08 22:04:36 +00:00
# split the sum off.
$ sum = ( $ return =~ /^(.*?)\s+$file$/ ) [ 0 ] ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { sum = > $ sum } } ) ;
2017-12-07 23:42:48 +00:00
}
return ( $ sum ) ;
}
2020-06-10 22:26:50 +00:00
= head2 os_type
This returns the operating system type and the system architecture as two separate string variables .
# Run on RHEL 8, on a 64-bit system
my ( $ os_type , $ os_arch ) = $ anvil - > Get - > os_type ( ) ;
# '$os_type' holds 'rhel8' ('rhel' or 'centos' + release version)
# '$os_arch' holds 'x86_64' (specifically, 'uname --hardware-platform')
If either can not be determined , C << unknown >> will be returned .
This method takes no parameters .
= cut
2020-06-11 20:18:32 +00:00
sub os_type
2020-06-10 22:26:50 +00:00
{
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , key = > "log_0125" , variables = > { method = > "Get->os_type()" } } ) ;
my $ os_type = "unknown" ;
my $ os_arch = "unknown" ;
### NOTE: Examples;
# Red Hat Enterprise Linux release 8.0 Beta (Ootpa)
# Red Hat Enterprise Linux Server release 7.5 (Maipo)
# CentOS Linux release 7.5.1804 (Core)
# Read in the /etc/redhat-release file
my $ release = $ anvil - > Storage - > read_file ( { file = > $ anvil - > data - > { path } { data } { 'redhat-release' } } ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { release = > $ release } } ) ;
if ( $ release =~ /Red Hat Enterprise Linux .* (\d+)\./ )
{
# RHEL, with the major version number appended
$ os_type = "rhel" . $ 1 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { os_type = > $ os_type } } ) ;
}
elsif ( $ release =~ /CentOS .*? (\d+)\./ )
{
# CentOS, with the major version number appended
$ os_type = "centos" . $ 1 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { os_type = > $ os_type } } ) ;
}
my ( $ output , $ return_code ) = $ anvil - > System - > call ( { shell_call = > $ anvil - > data - > { path } { exe } { uname } . " --hardware-platform" } ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { output = > $ output , return_code = > $ return_code } } ) ;
if ( $ output )
{
$ os_arch = $ output ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { os_arch = > $ os_arch } } ) ;
}
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
os_type = > $ os_type ,
os_arch = > $ os_arch ,
} } ) ;
return ( $ os_type , $ os_arch ) ;
}
2017-06-13 21:08:38 +00:00
= head2 switches
2017-06-07 16:15:03 +00:00
This reads in the command line switches used to invoke the parent program .
2017-10-20 04:19:32 +00:00
It takes no arguments , and data is stored in 'C<< $anvil->data->{switches}{x} >>' , where 'x' is the switch used .
2017-06-07 16:15:03 +00:00
2017-10-20 04:19:32 +00:00
Switches in the form 'C<< -x >>' and 'C<< --x >>' are treated the same and the corresponding 'C<< $anvil->data->{switches}{x} >>' will contain '#!set!#' .
2017-06-07 16:15:03 +00:00
2017-10-20 04:19:32 +00:00
Switches in the form 'C<< -x foo >>' , 'C<< --x foo >>' , 'C<< -x=foo >>' and 'C<< --x=foo >>' are treated the same and the corresponding 'C<< $anvil->data->{switches}{x} >>' will contain 'foo' .
2017-06-07 16:15:03 +00:00
The switches 'C<< -v >>' , 'C<< -vv >>' , 'C<< -vvv >>' and 'C<< -vvvv >>' will cause the active log level to automatically change to 1 , 2 , 3 or 4 respectively . Passing 'C<< -V >>' will set the log level to '0' .
Anything after 'C<< -- >>' is treated as a raw string and is not processed .
= cut
2020-07-30 04:23:47 +00:00
### TODO: This doesn't handle quoted values, System->parse_arguments() does. Switch to using it. Note that
### we'll still need to process '--raw' here (or make it work there)
2017-06-07 16:15:03 +00:00
sub switches
{
2018-03-07 08:11:55 +00:00
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
2017-06-07 16:15:03 +00:00
my $ last_argument = "" ;
foreach my $ argument ( @ ARGV )
{
if ( $ last_argument eq "raw" )
{
# Don't process anything.
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { switches } { raw } . = " $argument" ;
2017-06-07 16:15:03 +00:00
}
elsif ( $ argument =~ /^-/ )
{
# If the argument is just '--', appeand everything after it to 'raw'.
if ( $ argument eq "--" )
{
2020-07-30 04:23:47 +00:00
$ last_argument = "raw" ;
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { switches } { raw } = "" ;
2017-06-07 16:15:03 +00:00
}
else
{
( $ last_argument ) = ( $ argument =~ /^-{1,2}(.*)/ ) [ 0 ] ;
if ( $ last_argument =~ /=/ )
{
# Break up the variable/value.
( $ last_argument , my $ value ) = ( split /=/ , $ last_argument , 2 ) ;
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { switches } { $ last_argument } = $ value ;
2017-06-07 16:15:03 +00:00
}
else
{
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { switches } { $ last_argument } = "#!SET!#" ;
2017-06-07 16:15:03 +00:00
}
}
}
else
{
if ( $ last_argument )
{
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { switches } { $ last_argument } = $ argument ;
2020-07-30 04:23:47 +00:00
$ last_argument = "" ;
2017-06-07 16:15:03 +00:00
}
else
{
2018-02-13 02:58:37 +00:00
# Got a value without an argument, so just record it as '#!SET!#'.
$ anvil - > data - > { switches } { $ argument } = "#!SET!#" ;
2017-06-07 16:15:03 +00:00
}
}
}
2020-02-02 00:05:39 +00:00
2017-06-07 16:15:03 +00:00
# Clean up the initial space added to 'raw'.
2017-10-20 04:19:32 +00:00
if ( $ anvil - > data - > { switches } { raw } )
2017-06-07 16:15:03 +00:00
{
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { switches } { raw } =~ s/^ // ;
2017-06-07 16:15:03 +00:00
}
2017-06-23 02:57:51 +00:00
2017-06-07 16:15:03 +00:00
# Adjust the log level if requested.
2017-10-20 04:19:32 +00:00
$ anvil - > Log - > _adjust_log_level ( ) ;
2017-06-07 16:15:03 +00:00
return ( 0 ) ;
}
2020-07-03 22:11:56 +00:00
= head2 trusted_hosts
This returns an array reference containing host UUIDs of hosts this machine should trust . Specifically , any Striker dashboards this host uses , and if this host is in an Anvil ! , the peers . The array will include this host ' s UUID as well .
This method takes no parameters
= cut
sub trusted_hosts
{
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , key = > "log_0125" , variables = > { method = > "Get->uptime()" } } ) ;
my $ local_host_uuid = $ anvil - > Get - > host_uuid ;
my $ in_anvil = $ anvil - > data - > { hosts } { host_uuid } { $ local_host_uuid } { anvil_name } ;
my $ trusted_host_uuids = [ $ local_host_uuid ] ;
foreach my $ host_uuid ( keys % { $ anvil - > data - > { hosts } { host_uuid } } )
{
# Skip ourselves.
next if $ host_uuid eq $ anvil - > Get - > host_uuid ;
my $ host_name = $ anvil - > data - > { hosts } { host_uuid } { $ host_uuid } { host_name } ;
my $ host_type = $ anvil - > data - > { hosts } { host_uuid } { $ host_uuid } { host_type } ;
my $ anvil_name = $ anvil - > data - > { hosts } { host_uuid } { $ host_uuid } { anvil_name } ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
's1:host_uuid' = > $ host_uuid ,
's2:host_name' = > $ host_name ,
's3:host_type' = > $ host_type ,
's4:anvil_name' = > $ anvil_name ,
} } ) ;
if ( $ anvil - > Get - > host_type eq "striker" )
{
# Add all known machines
push @ { $ trusted_host_uuids } , $ host_uuid ;
}
elsif ( ( ( $ in_anvil ) && ( $ anvil_name eq $ in_anvil ) ) or ( exists $ anvil - > data - > { database } { $ host_uuid } ) )
{
# Add dashboards we use and peers
push @ { $ trusted_host_uuids } , $ host_uuid ;
}
}
return ( $ trusted_host_uuids )
}
2020-06-10 22:26:50 +00:00
= head2 uptime
This returns , in seconds , how long the host has been up and running for .
This method takes no parameters .
= cut
### TODO: Make this work on remote hosts
sub uptime
{
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , key = > "log_0125" , variables = > { method = > "Get->uptime()" } } ) ;
my $ uptime = $ anvil - > Storage - > read_file ( {
force_read = > 1 ,
cache = > 0 ,
file = > $ anvil - > data - > { path } { proc } { uptime } ,
} ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { uptime = > $ uptime } } ) ;
# Clean it up. We'll have gotten two numbers, the uptime in seconds (to two decimal places) and the
# total idle time. We only care about the int number.
$ uptime =~ s/^(\d+)\..*$/$1/ ;
$ uptime =~ s/\n//gs ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { uptime = > $ uptime } } ) ;
return ( $ uptime ) ;
}
2017-07-07 05:54:49 +00:00
= head2 users_home
This method takes a user 's name and returns the user' s home directory . If the home directory isn ' t found , C << 0 >> is returned .
Parameters ;
2018-07-25 06:34:47 +00:00
= head3 user ( optional , default is the user name of the real UID ( as stored in '$<' ) )
2017-07-07 05:54:49 +00:00
This is the user whose home directory you are looking for .
= cut
sub users_home
{
my $ self = shift ;
my $ parameter = shift ;
2017-10-20 04:19:32 +00:00
my $ anvil = $ self - > parent ;
2018-03-07 08:11:55 +00:00
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
2017-07-07 05:54:49 +00:00
my $ home_directory = 0 ;
2018-07-25 06:34:47 +00:00
my $ user = defined $ parameter - > { user } ? $ parameter - > { user } : getpwuid ( $< ) ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { user = > $ user } } ) ;
2017-07-07 05:54:49 +00:00
# Make sure the user is only one digit. Sometimes $< (and others) will return multiple IDs.
if ( $ user =~ /^\d+ \d$/ )
{
$ user =~ s/^(\d+)\s.*$/$1/ ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { user = > $ user } } ) ;
2017-07-07 05:54:49 +00:00
}
# If the user is numerical, convert it to a name.
if ( $ user =~ /^\d+$/ )
{
$ user = getpwuid ( $ user ) ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { user = > $ user } } ) ;
2017-07-07 05:54:49 +00:00
}
# Still don't have a name? fail...
if ( $ user eq "" )
{
# No user? No bueno...
2017-10-20 04:19:32 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 0 , priority = > "err" , key = > "log_0020" , variables = > { method = > "Get->users_home()" , parameter = > "user" } } ) ;
2017-07-07 05:54:49 +00:00
return ( $ home_directory ) ;
}
2017-10-20 04:19:32 +00:00
my $ body = $ anvil - > Storage - > read_file ( { file = > $ anvil - > data - > { path } { data } { passwd } } ) ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { body = > $ body } } ) ;
2017-07-07 05:54:49 +00:00
foreach my $ line ( split /\n/ , $ body )
{
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { line = > $ line } } ) ;
2017-07-07 05:54:49 +00:00
if ( $ line =~ /^$user:/ )
{
$ home_directory = ( split /:/ , $ line ) [ 5 ] ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { home_directory = > $ home_directory } } ) ;
2017-07-07 05:54:49 +00:00
last ;
}
}
# Do I have the a user's $HOME now?
if ( not $ home_directory )
{
2017-10-20 04:19:32 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 0 , priority = > "err" , key = > "log_0061" , variables = > { user = > $ user } } ) ;
2017-07-07 05:54:49 +00:00
}
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { home_directory = > $ home_directory } } ) ;
2017-07-07 05:54:49 +00:00
return ( $ home_directory ) ;
}
2017-06-23 02:57:51 +00:00
= head2 uuid
2019-02-11 08:36:41 +00:00
This method returns a new v4 UUID ( using 'UUID::Tiny' ) .
Parameters ;
= head3 short ( optional , default '0' )
This returns just the first 8 bytes of the uuid . For example , if the generated UUID is C << 9e4 b3f7c - 5 a98 - 40 b6 - 9 c34 - 84 fdb24ddd30 >> , only C << 9e4 b3f7c >> is returned .
2017-06-23 02:57:51 +00:00
= cut
sub uuid
{
2018-03-07 08:11:55 +00:00
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
2017-06-23 02:57:51 +00:00
2019-02-11 08:36:41 +00:00
my $ short = defined $ parameter - > { short } ? $ parameter - > { short } : 0 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
short = > $ short ,
} } ) ;
2018-09-23 20:16:08 +00:00
my $ uuid = create_uuid_as_string ( UUID_RANDOM ) ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { uuid = > $ uuid } } ) ;
2017-06-23 02:57:51 +00:00
2019-02-11 08:36:41 +00:00
if ( $ short )
{
$ uuid =~ s/^(\w+?)-.*$/$1/ ;
}
2017-06-23 02:57:51 +00:00
return ( $ uuid ) ;
}
2017-06-07 16:15:03 +00:00
# =head3
#
# Private Functions;
#
# =cut
#############################################################################################################
# Private functions #
#############################################################################################################
2018-05-15 05:55:56 +00:00
= head2 _salt
This generates a random salt string for use with internal Striker passwords .
= cut
sub _salt
{
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
my $ salt = "" ;
2018-05-15 21:45:52 +00:00
my $ salt_length = $ anvil - > data - > { sys } { password } { salt_length } =~ /^\d+$/ ? $ anvil - > data - > { sys } { password } { salt_length } : 16 ;
2018-05-15 05:55:56 +00:00
my @ seed = ( " " , "~" , "`" , "!" , "#" , "^" , "&" , "*" , "(" , ")" , "-" , "_" , "+" , "=" , "{" , "[" , "}" , "]" , "|" , ":" , ";" , "'" , "," , "<" , "." , ">" , "/" ) ;
my @ alpha = ( "a" , "b" , "c" , "d" , "e" , "f" , "g" , "h" , "i" , "j" , "k" , "l" , "m" , "n" , "o" , "p" , "q" , "r" , "s" , "t" , "u" , "v" , "w" , "x" , "y" , "z" , "A" , "B" , "C" , "D" , "E" , "F" , "G" , "H" , "I" , "J" , "K" , "L" , "M" , "N" , "O" , "P" , "Q" , "R" , "S" , "T" , "U" , "V" , "W" , "X" , "Y" , "Z" ) ;
my $ seed_count = @ seed ;
my $ alpha_count = @ alpha ;
my $ skip_count = 0 ;
for ( 1 .. $ salt_length )
{
# We want to have a little randomness in the salt length, but not skip tooooo many times.
if ( ( int ( rand ( 20 ) ) == 2 ) && ( $ skip_count <= 3 ) )
{
$ skip_count + + ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { skip_count = > $ skip_count } } ) ;
next ;
}
# What character will this string be?
my $ this_integer = int ( rand ( 3 ) ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { this_integer = > $ this_integer } } ) ;
if ( $ this_integer == 0 )
{
# Inject a random digit
$ salt . = int ( rand ( 10 ) ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { salt = > $ salt } } ) ;
}
elsif ( $ this_integer == 1 )
{
# Inject a random letter
$ salt . = $ alpha [ int ( rand ( $ alpha_count ) ) ] ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { salt = > $ salt } } ) ;
}
else
{
# Inject a random character
$ salt . = $ seed [ int ( rand ( $ seed_count ) ) ] ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { salt = > $ salt } } ) ;
}
}
return ( $ salt ) ;
}
= head2 _wrap_to
This determines how wide the user ' s terminal currently is and returns that width , as well as store it in C << sys::terminal:: columns >> .
This takes no parameters . If there is a problem reading the column width , C << 0 >> will be returned .
= cut
sub _wrap_to
{
my $ self = shift ;
my $ parameter = shift ;
my $ anvil = $ self - > parent ;
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
# Get the column width
2019-07-13 08:16:03 +00:00
my ( $ columns , $ return_code ) = $ anvil - > System - > call ( { debug = > $ debug , redirect_stderr = > 0 , shell_call = > $ anvil - > data - > { path } { exe } { tput } . " cols" } ) ;
2018-05-15 05:55:56 +00:00
if ( ( not defined $ columns ) or ( $ columns !~ /^\d+$/ ) )
{
# Set 0.
$ columns = 0 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { columns = > $ columns } } ) ;
}
else
{
# Got a good value
$ anvil - > data - > { sys } { terminal } { columns } = $ columns ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { 'sys::terminal::columns' = > $ anvil - > data - > { sys } { terminal } { columns } } } ) ;
}
return ( $ columns ) ;
}
1 ;