2017-10-20 04:19:32 +00:00
#!/usr/bin/perl
#
# This is the master daemon that manages all periodically run processes on Striker dashboards and Anvil!
# nodes.
#
2017-12-07 23:42:48 +00:00
# Exit codes;
# 0 = Normal exit
# 1 = md5sum of this program changed. Exited to reload.
2018-08-24 04:52:56 +00:00
# 2 = Unable to connect to any database, even after trying to initialize the local system.
2017-12-07 23:42:48 +00:00
#
2017-12-08 22:04:36 +00:00
# TODO:
2018-08-24 04:52:56 +00:00
# - Need to check what kind of machine this is and not prep the database unless its a dashboard.
2018-09-03 02:13:41 +00:00
# - Add a "running: pending,yes,done,dead" and show an appropriate icon beside jobs
2018-09-06 05:37:08 +00:00
# - Decide if holding before the main loop until 'systemctl is-system-running' returns 'running' is a good
# idea or not.
#
# NOTE:
# - For later; 'reboot --force --force' immediately kills the OS, like disabling ACPI on EL6 and hitting the power button. Might be useful in ScanCore down the road.
2017-10-20 04:19:32 +00:00
#
2017-12-08 22:04:36 +00:00
2017-10-20 04:19:32 +00:00
use strict;
use warnings;
use Anvil::Tools;
* 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
use Proc::Simple;
2018-09-10 06:17:02 +00:00
#use Time::HiRes qw ( time sleep );
2018-08-29 04:33:33 +00:00
use JSON;
2018-08-29 23:52:05 +00:00
use HTML::Strip;
use HTML::FromText;
2017-10-20 04:19:32 +00:00
2018-10-04 07:37:43 +00:00
use Data::Dumper;
2017-10-20 04:19:32 +00:00
my $THIS_FILE = ($0 =~ /^.*\/(.*)$/)[0];
my $running_directory = ($0 =~ /^(.*?)\/$THIS_FILE$/)[0];
if (($running_directory =~ /^\./) && ($ENV{PWD}))
{
$running_directory =~ s/^\./$ENV{PWD}/;
}
# Turn off buffering so that the pinwheel will display while waiting for the SSH call(s) to complete.
$| = 1;
* 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
my $anvil = Anvil::Tools->new({log_level => 2, log_secure => 1});
2018-09-07 05:29:43 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, secure => 0, key => "log_0115", variables => { program => $THIS_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
# Connect to the database(s). If we have no connections, we'll proceed anyway as one of the 'run_once' tasks
# is to setup the database server.
2018-10-02 07:31:42 +00:00
$anvil->Database->connect({debug => 3, check_if_configured => 1});
2018-09-10 06:17:02 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 3, secure => 0, key => "log_0132"});
2017-10-20 04:19:32 +00:00
2018-08-24 04:52:56 +00:00
# If I have no databases, sleep for a second and then exit (systemd will restart us).
if (not $anvil->data->{sys}{database}{connections})
{
# Try to configure the local database, and then try to connect again.
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, secure => 0, key => "log_0201"});
prep_database($anvil);
sleep 1;
# Try connecting again
$anvil->Database->connect();
2018-09-07 05:29:43 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, secure => 0, key => "log_0132"});
2018-08-24 04:52:56 +00:00
if (not $anvil->data->{sys}{database}{connections})
{
# Still nothing, sleep and exit.
print $anvil->Words->string({key => "error_0003"})."\n";
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 0, secure => 0, key => "error_0003"});
$anvil->nice_exit({exit_code => 2});
}
}
# Read switches
2018-09-03 06:35:25 +00:00
$anvil->data->{switches}{'refresh-json'} = "";
$anvil->data->{switches}{'run-once'} = 0;
$anvil->data->{switches}{'main-loop-only'} = 0;
2018-08-31 20:52:53 +00:00
$anvil->data->{switches}{'no-start'} = 0;
2018-09-06 05:37:08 +00:00
$anvil->data->{switches}{'startup-only'} = 0;
2018-08-24 04:52:56 +00:00
$anvil->Get->switches;
2018-09-03 06:35:25 +00:00
if ($anvil->data->{switches}{'refresh-json'})
{
$anvil->data->{switches}{'run-once'} = 1;
$anvil->data->{switches}{'main-loop-only'} = 1;
$anvil->data->{switches}{'no-start'} = 1;
}
2017-10-20 04:19:32 +00:00
# There are some things we only want to run on (re)start and don't need to always run.
2018-08-29 02:42:46 +00:00
run_once($anvil) if not $anvil->data->{switches}{'main-loop-only'};
2017-10-20 04:19:32 +00:00
2017-12-07 23:42:48 +00:00
# Calculate my sum so that we can exit if it changes later.
2017-12-08 22:04:36 +00:00
$anvil->Storage->record_md5sums;
2017-12-07 23:42:48 +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
# Disconnect. We'll reconnect inside the loop
2018-08-31 06:40:49 +00:00
$anvil->Database->disconnect();
2018-09-07 05:29:43 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 3, secure => 0, key => "log_0203"});
* 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
2018-09-01 04:59:14 +00:00
# This will prevent restarting while jobs are running.
$anvil->data->{sys}{jobs_running} = 0;
2018-09-10 06:17:02 +00:00
# Once a minute, we'll check the md5sums and see if we should restart. We don't check every loop as it places a
2018-10-02 07:31:42 +00:00
my $system_check = 60;
2018-09-10 06:17:02 +00:00
my $now_time = time;
2018-10-02 07:31:42 +00:00
my $next_check = $now_time + $system_check;
2018-09-10 06:17:02 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-10-02 07:31:42 +00:00
"s1:system_check" => $system_check,
2018-09-10 06:17:02 +00:00
"s2:now_time" => $now_time,
"s3:next_check" => $next_check,
}});
2018-10-02 07:31:42 +00:00
# When we periodically check if system files have changed, we'll also ask Database>connect() to check if it
# needs to be configured or updated. This is done periodically as it is expensive to run on every loop.
my $check_if_database_is_configured = 0;
2017-10-20 04:19:32 +00:00
# These are the things we always want running.
while(1)
{
2018-10-05 00:54:00 +00:00
# Reload defaults, re-read the config and then connect to the database(s)
2018-10-04 07:37:43 +00:00
$anvil->Storage->read_config({force_read => 1, file => $anvil->data->{path}{configs}{'anvil.conf'}});
2018-10-02 07:31:42 +00:00
$anvil->Database->connect({check_if_configured => $check_if_database_is_configured});
2018-09-07 05:29:43 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 3, secure => 0, key => "log_0132"});
2018-08-24 04:52:56 +00:00
2018-10-02 07:31:42 +00:00
# Mark that we don't want to check the database now.
$check_if_database_is_configured = 0;
2018-08-24 04:52:56 +00:00
if ($anvil->data->{sys}{database}{connections})
{
2018-09-10 06:17:02 +00:00
# Run the normal tasks
2018-08-24 04:52:56 +00:00
keep_running($anvil);
}
else
{
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 0, secure => 0, key => "log_0202"});
}
2017-10-20 04:19:32 +00:00
# Exit if called with '--run-once'
2017-10-20 15:13:00 +00:00
if ($anvil->data->{switches}{'run-once'})
2017-10-20 04:19:32 +00:00
{
2017-10-20 15:13:00 +00:00
$anvil->nice_exit({code => 0});
2017-10-20 04:19:32 +00:00
}
2017-12-08 22:04:36 +00:00
# Has the file on disk changed?
2018-09-10 06:17:02 +00:00
$now_time = time;
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
"s1:now_time" => $now_time,
"s2:next_check" => $next_check,
}});
2018-10-02 07:31:42 +00:00
if ($now_time >= $next_check)
2017-12-08 22:04:36 +00:00
{
2018-09-10 06:17:02 +00:00
# Even if it is time to check, don't if a job is running.
if ((not $anvil->data->{sys}{jobs_running}) && ($anvil->Storage->check_md5sums))
{
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 0, priority => "warn", key => "message_0014"});
$anvil->nice_exit({code => 1});
}
2018-10-02 07:31:42 +00:00
# Mark that we want to check the database config next time.
$check_if_database_is_configured = 1;
2018-09-10 06:17:02 +00:00
# Update the next check time.
2018-10-02 07:31:42 +00:00
$next_check = $now_time + $system_check;
2018-10-05 01:38:04 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-10-02 07:31:42 +00:00
"s1:system_check" => $system_check,
2018-09-10 06:17:02 +00:00
"s2:next_check" => $next_check,
}});
2017-12-08 22:04:36 +00:00
}
2018-08-24 04:52:56 +00:00
# Exit if 'run-once' selected.
if ($anvil->data->{switches}{'run-once'})
{
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 0, priority => "warn", key => "message_0055"});
$anvil->nice_exit({code => 0});
}
2018-08-31 06:40:49 +00:00
# Disconnect from the database(s) and sleep now.
2018-10-05 00:54:00 +00:00
$anvil->Database->disconnect();
2018-09-08 02:55:05 +00:00
sleep(1);
2017-10-20 04:19:32 +00:00
}
2017-10-20 15:13:00 +00:00
$anvil->nice_exit({code => 0});
2017-10-20 04:19:32 +00:00
2018-09-06 05:37:08 +00:00
2017-10-20 04:19:32 +00:00
#############################################################################################################
# Functions #
#############################################################################################################
* 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
# These are tools that don't need to constantly run. They'll typically run when the server starts up or the
# daemon is restarted or reloaded.
2017-10-20 04:19:32 +00:00
sub run_once
{
my ($anvil) = @_;
# Check that the database is ready.
2018-08-24 04:52:56 +00:00
prep_database($anvil);
2017-10-20 04:19:32 +00:00
2018-09-05 07:39:13 +00:00
# Check to see if we need to do boot-time tasks. We only run these if we've just booted
boot_time_tasks($anvil);
2018-09-06 05:37:08 +00:00
if ($anvil->data->{switches}{'startup-only'})
{
$anvil->nice_exit({code => 0});
}
2018-09-05 07:39:13 +00:00
return(0);
}
# This handles tasks that need to run on boot (if any)
sub boot_time_tasks
{
my ($anvil) = @_;
2018-08-22 06:16:56 +00:00
# If the uptime is less than ten minutes, clear the reboot flag.
2018-10-06 07:16:08 +00:00
my $uptime = $anvil->System->get_uptime;
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { uptime => $uptime }});
2018-08-22 06:16:56 +00:00
2018-10-06 07:16:08 +00:00
# Now find out if a reboot is listed as needed and when it was last changed.
2018-09-05 07:39:13 +00:00
my $reboot_needed = 0;
my $changed_seconds_ago = 0;
my $query = "
SELECT
variable_value,
2018-09-25 06:05:07 +00:00
(SELECT extract(epoch from now()) - extract(epoch from modified_date)) AS changed_seconds_ago
2018-09-05 07:39:13 +00:00
FROM
variables
WHERE
variable_source_table = 'hosts'
AND
variable_source_uuid = ".$anvil->data->{sys}{database}{use_handle}->quote($anvil->data->{sys}{host_uuid})."
AND
variable_name = 'reboot::needed'
;";
2018-10-06 07:16:08 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 2, key => "log_0124", variables => { query => $query }});
2018-09-05 07:39:13 +00:00
my $results = $anvil->Database->query({query => $query, source => $THIS_FILE, line => __LINE__});
my $count = @{$results};
2018-10-06 07:16:08 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
2018-09-05 07:39:13 +00:00
results => $results,
count => $count,
}});
if ($count)
{
$reboot_needed = $results->[0]->[0];
$changed_seconds_ago = $results->[0]->[1];
$changed_seconds_ago =~ s/^(\d+)\..*$/$1/;
2018-10-06 07:16:08 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
2018-09-05 07:39:13 +00:00
reboot_needed => $reboot_needed,
changed_seconds_ago => $changed_seconds_ago,
}});
}
# If a reboot is needed, see if the uptime is less than the time since the reboot needed flag was
# set. If the uptime is less, then the system rebooted since it was requested so clear it. h/t to
# Lisa Seelye (@thedoh) for this idea!
2018-10-06 07:16:08 +00:00
my $difference = ($changed_seconds_ago - $uptime);
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
"s1:reboot_needed" => $reboot_needed,
"s2:changed_seconds_ago" => $changed_seconds_ago,
"s3:uptime" => $uptime,
"s4:difference" => $difference,
2018-09-05 07:39:13 +00:00
}});
if (($reboot_needed) && ($uptime < $changed_seconds_ago))
2018-08-22 06:16:56 +00:00
{
# Clear the reboot request.
2018-10-06 07:16:08 +00:00
$reboot_needed = $anvil->System->reboot_needed({debug => 2, set => 0});
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { reboot_needed => $reboot_needed }});
2018-09-06 05:37:08 +00:00
# Check to see if there was a reboot job in progress. If so, finish it off.
2018-10-06 07:16:08 +00:00
my $job_uuid = $anvil->Job->get_job_uuid({debug => 2, program => "anvil-manage-power"});
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { job_uuid => $job_uuid }});
2018-09-06 05:37:08 +00:00
if ($job_uuid)
{
# Update the percentage to '100' and then clear the old PID.
my $date_time = $anvil->Get->date_and_time();
2018-10-06 07:16:08 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { date_time => $date_time }});
$anvil->Job->update_progress({
debug => 2,
progress => 100,
message => "message_0064,!!date_and_time!".$date_time."!!",
job_uuid => $job_uuid,
picked_up_by => 0,
});
2018-09-06 05:37:08 +00:00
}
2018-08-22 06:16:56 +00:00
}
2017-10-20 04:19:32 +00:00
return(0);
}
2018-08-24 04:52:56 +00:00
# Configure the local database, if needed.
sub prep_database
{
my ($anvil) = @_;
my $shell_call = $anvil->data->{path}{exe}{'anvil-prep-database'};
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { shell_call => $shell_call }});
my $database_output = $anvil->System->call({shell_call => $shell_call, source => $THIS_FILE, line => __LINE__});
if ($database_output)
{
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { database_output => $database_output }});
}
return(0);
}
2017-10-20 04:19:32 +00:00
# These are tools that need to keep running.
sub keep_running
{
my ($anvil) = @_;
* 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
# Check for jobs that were running and now exited.
if (exists $anvil->data->{processes})
{
foreach my $job_uuid (%{$anvil->data->{jobs}{handles}})
{
# If it's not a handle, delete it.
my $running = $anvil->data->{jobs}{handles}{$job_uuid}->poll();
2018-09-10 06:17:02 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
* 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
"jobs::handles::${job_uuid}" => $anvil->data->{jobs}{handles}{$job_uuid},
running => $running,
}});
# If it's not running, update the table to clear the 'job_picked_up_by' column.
if (not $running)
{
my $exit_status = $anvil->data->{jobs}{handles}{$job_uuid}->exit_status();
2018-09-07 05:29:43 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 1, list => {
job_uuid => $job_uuid,
exit_status => $exit_status,
}});
* 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
# Free up memory
$anvil->data->{jobs}{handles}{$job_uuid}->cleanup();
2018-09-06 05:37:08 +00:00
$anvil->Job->clear({job_uuid => $job_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
}
}
}
2018-09-08 02:55:05 +00:00
# Update hardware state files if the system isn't configured. Running it always is too intensive.
my $configured = $anvil->System->check_if_configured;
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { configured => $configured }});
if (not $configured)
{
update_state_file($anvil);
}
2017-10-20 04:19:32 +00:00
2018-03-07 08:11:55 +00:00
# Run any pending jobs by calling 'anvil-jobs' with the 'job_uuid' as a background process.
* 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
run_jobs($anvil);
return(0);
}
# This will check for any jobs that aren't at 100%. For each found, if 'picked_up_by' is set, a check is made
# to see if the PID is still alive. If it isn't, or if 'picked_up_by' is not set, the appropriate tool is
# invoked to handle it.
sub run_jobs
{
my ($anvil) = @_;
2018-09-01 04:59:14 +00:00
# This will be set to 1 if any jobs are not complete, preventing a restart of the daemon if it's
# changed on disk.
$anvil->data->{sys}{jobs_running} = 0;
2018-08-22 06:16:56 +00:00
# We'll also update the jobs.json file.
my $jobs_file = "{\"jobs\":[\n";
* 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
# Get a list of pending or incomplete jobs.
2018-09-28 05:50:38 +00:00
my $return = $anvil->Database->get_jobs({debug => 3, ended_within => 300});
2018-08-28 05:54:49 +00:00
my $count = @{$return};
2018-09-07 05:29:43 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-08-28 05:54:49 +00:00
'return' => $return,
count => $count,
* 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
}});
2018-08-28 05:54:49 +00:00
foreach my $hash_ref (@{$return})
* 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
{
2018-08-28 05:54:49 +00:00
my $job_uuid = $hash_ref->{job_uuid};
my $job_command = $hash_ref->{job_command};
my $job_data = $hash_ref->{job_data};
my $job_picked_up_by = $hash_ref->{job_picked_up_by};
my $job_picked_up_at = $hash_ref->{job_picked_up_at};
my $job_updated = $hash_ref->{job_updated};
my $job_name = $hash_ref->{job_name};
my $job_progress = $hash_ref->{job_progress};
my $job_title = $hash_ref->{job_title};
my $job_description = $hash_ref->{job_description};
my $job_status = $hash_ref->{job_status};
2018-08-24 04:52:56 +00:00
my $started_seconds_ago = $job_picked_up_at ? (time - $job_picked_up_at) : 0;
my $updated_seconds_ago = $job_updated ? (time - $job_updated) : 0;
2018-10-04 07:37:43 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-08-22 06:16:56 +00:00
job_uuid => $job_uuid,
job_command => $job_command,
job_data => $job_data,
job_picked_up_by => $job_picked_up_by,
job_picked_up_at => $job_picked_up_at,
job_updated => $job_updated,
2018-08-29 02:42:46 +00:00
job_name => $job_name,
2018-08-22 06:16:56 +00:00
job_progress => $job_progress,
2018-08-29 02:42:46 +00:00
job_title => $job_title,
2018-08-29 23:52:05 +00:00
job_description => $job_description,
job_status => $job_status,
2018-08-22 06:16:56 +00:00
started_seconds_ago => $started_seconds_ago,
updated_seconds_ago => $updated_seconds_ago,
* 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
}});
2018-09-01 04:59:14 +00:00
if ($job_progress ne "100")
{
$anvil->data->{sys}{jobs_running} = 1;
2018-10-04 07:37:43 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { "sys::jobs_running" => $anvil->data->{sys}{jobs_running} }});
2018-09-01 04:59:14 +00:00
}
2018-08-29 23:52:05 +00:00
# See if the job was picked up by a now-dead instance.
if ($job_picked_up_by)
{
# Check if the PID is still active.
$anvil->System->pids({ignore_me => 1});
### TODO: Add a check to verify the job isn't hung.
# Skip if this job is in progress.
if (not exists $anvil->data->{pids}{$job_picked_up_by})
{
2018-08-31 06:40:49 +00:00
# If the job is done, just clear the 'job_picked_up_by' and be done.
if ($job_progress ne "100")
{
2018-10-04 07:37:43 +00:00
# It's possible that the job updated to 100% and exited after we
# gathered the job data, so we won't restart until we've seen it not
# running and not at 100% after 5 loops.
if ((not exists $anvil->data->{lost_job_count}{$job_uuid}) or (not defined $anvil->data->{lost_job_count}{$job_uuid}))
{
$anvil->data->{lost_job_count}{$job_uuid} = 0;
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { "lost_job_count::${job_uuid}" => $anvil->data->{lost_job_count}{$job_uuid} }});
}
if ($anvil->data->{lost_job_count}{$job_uuid} > 5)
{
# The previous job is gone, but the job isn't finished. Start it again.
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, priority => "alert", key => "striker_warning_0007", variables => {
command => $job_command,
pid => $job_picked_up_by,
percent => $job_progress,
}});
# Clear some variables.
$job_progress = 0;
$job_status = "message_0056";
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
job_progress => $job_progress,
job_status => $job_status,
}});
# Clear the job.
$anvil->Job->clear({debug => 3, job_uuid => $job_uuid});
$anvil->data->{lost_job_count}{$job_uuid} = 0;
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { "lost_job_count::${job_uuid}" => $anvil->data->{lost_job_count}{$job_uuid} }});
}
else
{
$anvil->data->{lost_job_count}{$job_uuid}++;
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { "lost_job_count::${job_uuid}" => $anvil->data->{lost_job_count}{$job_uuid} }});
}
2018-08-31 06:40:49 +00:00
}
2018-08-29 23:52:05 +00:00
2018-08-31 06:40:49 +00:00
# Clear the PID
2018-08-29 23:52:05 +00:00
$job_picked_up_by = 0;
2018-10-04 07:37:43 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { job_picked_up_by => $job_picked_up_by }});
2018-08-29 23:52:05 +00:00
}
}
2018-08-29 02:42:46 +00:00
# Convert the double-banged strings into a proper message.
2018-09-07 05:29:43 +00:00
my $say_title = $job_title ? $anvil->Words->parse_banged_string({key_string => $job_title}) : "";
my $say_description = $job_description ? $anvil->Words->parse_banged_string({key_string => $job_description}) : "";
my $say_status = $job_status ? $anvil->Words->parse_banged_string({key_string => $job_status}) : "";
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-08-29 02:42:46 +00:00
job_title => $job_title,
say_description => $say_description,
say_status => $say_status,
}});
2018-08-22 06:16:56 +00:00
2018-08-29 23:52:05 +00:00
# Make the status HTML friendly. Strip any embedded HTML then encode the text string.
2018-08-31 06:40:49 +00:00
if ($say_status)
{
my $html_strip = HTML::Strip->new();
$say_status = $html_strip->parse($say_status);
2018-09-07 05:29:43 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { say_status => $say_status }});
2018-08-31 06:40:49 +00:00
# Now make the resulting text string HTML friendly
my $text_to_html = HTML::FromText->new({
urls => 1,
email => 1,
lines => 1,
});
$say_status = $text_to_html->parse($say_status);
2018-09-07 05:29:43 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { say_status => $say_status }});
2018-08-31 06:40:49 +00:00
}
2018-08-29 23:52:05 +00:00
2018-08-28 05:54:49 +00:00
# Add this to the jobs.json file
2018-08-29 04:33:33 +00:00
my $json_string = to_json ({
job_uuid => $job_uuid,
job_command => $job_command,
job_data => $job_data,
job_picked_up_at => $job_picked_up_at,
job_updated => $job_updated,
job_name => $job_name,
job_progress => $job_progress,
job_title => $say_title,
job_description => $say_description,
job_status => $say_status,
started_seconds_ago => $started_seconds_ago,
updated_seconds_ago => $updated_seconds_ago,
});
2018-09-07 05:29:43 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { json_string => $json_string }});
2018-08-31 20:52:53 +00:00
$jobs_file .= $json_string.",\n";
2018-08-22 06:16:56 +00:00
2018-08-29 02:42:46 +00:00
# If the job is done, move on.
next if $job_progress eq "100";
2018-08-29 23:52:05 +00:00
# If the job is not running, start it.
2018-09-28 05:50:38 +00:00
if ((not $job_picked_up_by) && ($job_progress ne "100") && (not $anvil->data->{switches}{'no-start'}))
2018-08-27 06:51:53 +00:00
{
# Start the job, appending '--job-uuid' to the command.
2018-09-07 05:29:43 +00:00
my $command = $job_command." --job-uuid ".$job_uuid;
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, secure => 0, key => "log_0210", variables => { command => $command }});
2018-08-27 06:51:53 +00:00
$anvil->data->{jobs}{handles}{$job_uuid} = $anvil->System->call({
2018-09-27 05:54:30 +00:00
debug => 3,
2018-08-27 06:51:53 +00:00
background => 1,
stdout_file => "/tmp/anvil.job.".$job_uuid.".stdout",
stderr_file => "/tmp/anvil.job.".$job_uuid.".stderr",
2018-09-07 05:29:43 +00:00
shell_call => $command,
2018-08-27 06:51:53 +00:00
source => $THIS_FILE,
line => __LINE__,
});
2018-09-27 05:54:30 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { "jobs::handles::${job_uuid}" => $anvil->data->{jobs}{handles}{$job_uuid} }});
2018-08-27 06:51:53 +00:00
2018-08-28 05:54:49 +00:00
# Log the PID (the job should update the database).
2018-08-27 06:51:53 +00:00
my $pid = $anvil->data->{jobs}{handles}{$job_uuid}->pid();
2018-09-27 05:54:30 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { pid => $pid }});
2018-08-27 06:51:53 +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
}
2018-04-02 05:03:28 +00:00
2018-08-22 06:16:56 +00:00
# Close the jobs file.
2018-08-31 20:52:53 +00:00
$jobs_file =~ s/,\n$/\n/ms;
2018-08-22 06:16:56 +00:00
$jobs_file .= "]}\n";
2018-09-07 05:29:43 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { jobs_file => $jobs_file }});
2018-08-24 04:52:56 +00:00
# Write the JSON file
my $output_json = $anvil->data->{path}{directories}{html}."/status/jobs.json";
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { output_xml => $output_json }});
$anvil->Storage->write_file({
file => $output_json,
body => $jobs_file,
overwrite => 1,
mode => "0644",
user => "apache",
group => "apache"
});
2018-08-22 06:16:56 +00:00
2017-10-20 04:19:32 +00:00
return(0);
}
2018-04-13 23:55:34 +00:00
# This calls 'anvil-update-states' which will scan the local machine's state (hardware and software) and
2017-10-20 04:19:32 +00:00
# record write it out to an HTML file
sub update_state_file
{
my ($anvil) = @_;
2018-09-07 05:29:43 +00:00
my $states_output = $anvil->System->call({debug => 3, shell_call => $anvil->data->{path}{exe}{'anvil-update-states'}, source => $THIS_FILE, line => __LINE__});
2017-10-20 04:19:32 +00:00
if ($states_output)
{
2017-12-08 22:04:36 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { states_output => $states_output }});
2017-10-20 04:19:32 +00:00
}
return(0);
}