2017-10-20 04:19:32 +00:00
package Anvil::Tools::Alert ;
2017-05-02 04:41:12 +00:00
#
# This module contains methods used to handle alerts and errors.
#
use strict ;
use warnings ;
2017-08-17 21:16:45 +00:00
use Scalar::Util qw( weaken isweak ) ;
2017-05-02 04:41:12 +00:00
our $ VERSION = "3.0.0" ;
my $ THIS_FILE = "Alert.pm" ;
### Methods;
2017-07-20 06:00:40 +00:00
# check_alert_sent
2020-10-23 05:28:21 +00:00
# check_condition_age
2017-05-02 04:41:12 +00:00
# error
2018-12-29 08:23:58 +00:00
# register
2017-05-02 04:41:12 +00:00
= pod
= encoding utf8
= head1 NAME
2017-10-20 04:19:32 +00:00
Anvil::Tools:: Alert
2017-05-02 04:41:12 +00:00
2018-05-15 05:55:56 +00:00
Provides all methods related to warnings and alerts .
2017-05-02 04:41:12 +00:00
= head1 SYNOPSIS
2017-10-20 04:19:32 +00:00
use Anvil::Tools ;
2017-05-02 04:41:12 +00:00
2017-10-20 04:19:32 +00:00
# Get a common object handle on all Anvil::Tools modules.
my $ anvil = Anvil::Tools - > new ( ) ;
2017-05-02 04:41:12 +00:00
2017-10-20 04:19:32 +00:00
# Access to methods using '$anvil->Alert->X'. Example using 'find';
my $ foo_path = $ anvil - > Storage - > find ( { file = > "foo" } ) ;
2017-05-02 04:41:12 +00:00
= head1 METHODS
Methods in the core module ;
= cut
sub new
{
my $ class = shift ;
my $ self = { } ;
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-05-02 04:41:12 +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-05-02 04:41:12 +00:00
return ( $ self - > { HANDLE } { TOOLS } ) ;
}
#############################################################################################################
# Public methods #
#############################################################################################################
2017-07-20 06:00:40 +00:00
= head2 check_alert_sent
2020-09-09 05:48:51 +00:00
This method is used to see if an event that might last some time has had an alert send already to recipients .
2020-09-05 05:18:42 +00:00
This is used by programs , usually scancore scan agents , that need to track whether an alert was sent when a sensor dropped below / rose above a set alert threshold . For example , if a sensor alerts at 20 ° C and clears at 25 ° C , this will be called when either value is passed . When passing the warning threshold , the alert is registered and sent to the user . Once set , no further warning alerts are sent . When the value passes over the clear threshold , this is checked and if an alert was previously registered , it is removed and an "all clear" message is sent . In this way , multiple alerts will not go out if a sensor floats around the warning threshold and a "cleared" message won ' t be sent unless a "warning" message was previously sent .
2017-07-20 06:00:40 +00:00
2017-08-02 01:04:35 +00:00
If there is a problem , C << ! ! error ! ! >> is returned .
2017-07-20 06:00:40 +00:00
Parameters ;
= head3 record_locator
This is a record locator , which generally allows a given alert to be tied to a given source . For example , an alert related to a temperature might use C << an - a01n01 . alteeve . com:cpu1_temperature >> .
= head3 set_by ( required )
This is a string , usually the name of the program , that set the alert . Usuall this is simple C << $ THIS_FILE >> or C << $ 0 >> .
2020-09-15 06:16:35 +00:00
= head3 clear ( optional , default '0' )
2017-07-20 06:00:40 +00:00
2020-09-15 06:16:35 +00:00
If set to C << 0 >> ( set the alert ) , C << 1 >> will be returned if this is the first time we ' ve tried to set this alert . If the alert was set before , C << 0 >> is returned .
2017-07-20 06:00:40 +00:00
2020-09-15 06:16:35 +00:00
If set to C << 1 >> ( clear the alert ) , C << 1 >> will be returned if this is the alert existed and was cleared . If the alert didn 't exist (and thus didn' t need to be cleared ) , C << 0 >> is returned .
2017-07-20 06:00:40 +00:00
= cut
sub check_alert_sent
{
my $ self = shift ;
my $ parameter = shift ;
2018-03-07 08:11:55 +00:00
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
2017-10-20 04:19:32 +00:00
my $ anvil = $ self - > parent ;
2020-09-06 05:52:03 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , key = > "log_0125" , variables = > { method = > "Alert->check_alert_sent()" } } ) ;
2017-07-20 06:00:40 +00:00
2017-12-27 17:01:58 +00:00
my $ record_locator = defined $ parameter - > { record_locator } ? $ parameter - > { record_locator } : "" ;
my $ set_by = defined $ parameter - > { set_by } ? $ parameter - > { set_by } : "" ;
2020-09-15 06:16:35 +00:00
my $ clear = defined $ parameter - > { clear } ? $ parameter - > { clear } : 0 ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
2017-07-20 06:00:40 +00:00
record_locator = > $ record_locator ,
set_by = > $ set_by ,
2020-09-15 06:16:35 +00:00
clear = > $ clear ,
2017-07-20 06:00:40 +00:00
} } ) ;
# Do we have an record locator?
if ( not $ record_locator )
{
# Nope
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 = > "Alert->check_alert_sent()" , parameter = > "record_locator" } } ) ;
2017-08-02 01:04:35 +00:00
return ( "!!error!!" ) ;
2017-07-20 06:00:40 +00:00
}
# Do we know who is setting this??
if ( not $ set_by )
{
# Nope
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 = > "Alert->check_alert_sent()" , parameter = > "set_by" } } ) ;
2017-08-02 01:04:35 +00:00
return ( "!!error!!" ) ;
2017-07-20 06:00:40 +00:00
}
# This will get set to '1' if an alert is added or removed.
2020-09-09 05:48:51 +00:00
my $ changed = 0 ;
2017-07-20 06:00:40 +00:00
my $ query = "
SELECT
2017-12-27 17:01:58 +00:00
alert_sent_uuid
2017-07-20 06:00:40 +00:00
FROM
alert_sent
WHERE
2020-09-15 05:18:36 +00:00
alert_sent_host_uuid = ".$anvil->Database->quote($anvil->Get->host_uuid)."
2017-07-20 06:00:40 +00:00
AND
2019-03-06 06:49:59 +00:00
alert_set_by = ".$anvil->Database->quote($set_by)."
2017-07-20 06:00:40 +00:00
AND
2019-03-06 06:49:59 +00:00
alert_record_locator = ".$anvil->Database->quote($record_locator)."
2017-07-20 06:00:40 +00:00
; " ;
2017-12-27 17:01:58 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { query = > $ query } } ) ;
2017-07-20 06:00:40 +00:00
2020-09-15 06:16:35 +00:00
# Now, if this is clear = 0, register the alert if it doesn't exist. If it is clear = 1, remove the
2017-08-02 07:00:08 +00:00
# alert if it exists.
2017-12-27 17:01:58 +00:00
my $ alert_sent_uuid = $ anvil - > Database - > query ( { query = > $ query , source = > $ THIS_FILE , line = > __LINE__ } ) - > [ 0 ] - > [ 0 ] ;
$ alert_sent_uuid = "" if not defined $ alert_sent_uuid ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
2020-09-15 06:16:35 +00:00
clear = > $ clear ,
2017-12-27 17:01:58 +00:00
alert_sent_uuid = > $ alert_sent_uuid ,
2017-07-20 06:00:40 +00:00
} } ) ;
2020-09-15 06:16:35 +00:00
if ( ( not $ clear ) && ( not $ alert_sent_uuid ) )
2017-07-20 06:00:40 +00:00
{
### New alert
# Make sure this host is in the database... It might not be on the very first run of ScanCore
# before the peer exists (tried to connect to the peer, fails, tries to send an alert, but
# this host hasn't been added because it is the very first attempt to connect...)
2017-10-20 04:19:32 +00:00
if ( not $ anvil - > data - > { sys } { host_is_in_db } )
2017-07-20 06:00:40 +00:00
{
my $ query = "
SELECT
COUNT ( * )
FROM
hosts
WHERE
2020-09-15 05:18:36 +00:00
host_uuid = ".$anvil->Database->quote($anvil->Get->host_uuid)."
2017-07-20 06:00:40 +00:00
; " ;
2017-12-27 17:01:58 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { query = > $ query } } ) ;
2017-07-20 06:00:40 +00:00
2017-10-20 04:19:32 +00:00
my $ count = $ anvil - > Database - > query ( { query = > $ query , source = > $ THIS_FILE , line = > __LINE__ } ) - > [ 0 ] - > [ 0 ] ;
2017-12-27 17:01:58 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { count = > $ count } } ) ;
2017-07-20 06:00:40 +00:00
if ( not $ count )
{
# Too early, we can't set an alert.
2017-10-20 04:19:32 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 0 , priority = > "alert" , key = > "log_0098" , variables = > {
2020-09-15 06:16:35 +00:00
clear = > $ clear ,
2017-08-04 05:45:56 +00:00
set_by = > $ set_by ,
record_locator = > $ record_locator ,
2017-07-20 06:00:40 +00:00
} } ) ;
2017-08-02 01:04:35 +00:00
return ( "!!error!!" ) ;
2017-07-20 06:00:40 +00:00
}
else
{
2017-10-20 04:19:32 +00:00
$ anvil - > data - > { sys } { host_is_in_db } = 1 ;
2017-12-27 17:01:58 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { 'sys::host_is_in_db' = > $ anvil - > data - > { sys } { host_is_in_db } } } ) ;
2017-07-20 06:00:40 +00:00
}
}
2020-09-09 05:48:51 +00:00
$ changed = 1 ;
my $ query = "
2017-07-20 06:00:40 +00:00
INSERT INTO
alert_sent
(
2017-12-27 17:01:58 +00:00
alert_sent_uuid ,
2017-07-20 06:00:40 +00:00
alert_sent_host_uuid ,
alert_set_by ,
alert_record_locator ,
2018-09-25 06:05:07 +00:00
modified_date
2017-07-20 06:00:40 +00:00
) VALUES (
2019-03-06 06:49:59 +00:00
".$anvil->Database->quote($anvil->Get->uuid)." ,
2020-09-15 05:18:36 +00:00
".$anvil->Database->quote($anvil->Get->host_uuid)." ,
2019-03-06 06:49:59 +00:00
".$anvil->Database->quote($set_by)." ,
".$anvil->Database->quote($record_locator)." ,
2021-06-08 19:23:15 +00:00
".$anvil->Database->quote($anvil->Database->refresh_timestamp)."
2017-07-20 06:00:40 +00:00
) ;
" ;
2017-12-27 17:01:58 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
2020-09-09 05:48:51 +00:00
query = > $ query ,
changed = > $ changed ,
2017-07-20 06:00:40 +00:00
} } ) ;
2017-10-20 04:19:32 +00:00
$ anvil - > Database - > write ( { query = > $ query , source = > $ THIS_FILE , line = > __LINE__ } ) ;
2017-07-20 06:00:40 +00:00
}
2020-09-15 06:16:35 +00:00
elsif ( ( $ clear ) && ( $ alert_sent_uuid ) )
2017-07-20 06:00:40 +00:00
{
# Alert previously existed, clear it.
2020-09-09 05:48:51 +00:00
$ changed = 1 ;
my $ query = "
2017-07-20 06:00:40 +00:00
DELETE FROM
alert_sent
WHERE
2019-03-06 06:49:59 +00:00
alert_sent_uuid = ".$anvil->Database->quote($alert_sent_uuid)."
2017-07-20 06:00:40 +00:00
; " ;
2017-12-27 17:01:58 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
2020-09-09 05:48:51 +00:00
query = > $ query ,
changed = > $ changed ,
2017-07-20 06:00:40 +00:00
} } ) ;
2017-10-20 04:19:32 +00:00
$ anvil - > Database - > write ( { query = > $ query , source = > $ THIS_FILE , line = > __LINE__ } ) ;
2017-07-20 06:00:40 +00:00
}
2020-09-09 05:48:51 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { changed = > $ changed } } ) ;
return ( $ changed ) ;
2017-07-20 06:00:40 +00:00
}
2020-10-23 05:28:21 +00:00
= head2 check_condition_age
This checks to see how long ago a given condition ( variable , really ) has been set . This is generally used when a program , often a scan agent , wants to wait to see if a given state persists before sending an alert and / or taking an action .
A common example is seeing how long power has been lost , if a lost sensor is going to return , etc .
The age of the condition is returned , in seconds . If there is a problem , C << ! ! error ! ! >> is returned .
Parameters ;
= head3 clear ( optional )
When set to C << 1 >> , if the condition exists , it is cleared . If the condition does not exist , nothing happens .
= head3 name ( required )
This is the name of the condition being set . It ' s a free - form string , but generally in a format like C << <scan_agent_name> :: <condition_name> >> .
= head3 host_uuid ( optional )
If a condition is host - specific , this can be set to the caller ' s C << host_uuid >> . Generally this is needed , save for conditions related to hosted servers that are not host - bound .
= cut
sub check_condition_age
{
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 = > "Database->check_condition_age()" } } ) ;
my $ clear = defined $ parameter - > { clear } ? $ parameter - > { clear } : 0 ;
my $ name = defined $ parameter - > { name } ? $ parameter - > { name } : "" ;
2022-08-29 21:30:52 +00:00
my $ host_uuid = defined $ parameter - > { host_uuid } ? $ parameter - > { host_uuid } : "" ;
2020-10-23 05:28:21 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
clear = > $ clear ,
name = > $ name ,
host_uuid = > $ host_uuid ,
} } ) ;
if ( not $ name )
{
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 0 , priority = > "err" , key = > "log_0020" , variables = > { method = > "Database->check_condition_age()" , parameter = > "name" } } ) ;
return ( "!!error!!" ) ;
}
my $ age = 0 ;
my $ source_table = $ host_uuid ? "hosts" : "" ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { source_table = > $ source_table } } ) ;
# See if this variable has been set yet.
my ( $ variable_value , $ variable_uuid , $ epoch_modified_date , $ modified_date ) = $ anvil - > Database - > read_variable ( {
2022-01-14 02:07:25 +00:00
debug = > $ debug ,
2020-10-23 05:28:21 +00:00
variable_name = > $ name ,
variable_source_table = > $ source_table ,
variable_source_uuid = > $ host_uuid ,
} ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
variable_value = > $ variable_value ,
variable_uuid = > $ variable_uuid ,
epoch_modified_date = > $ epoch_modified_date ,
modified_date = > $ modified_date ,
} } ) ;
if ( $ variable_uuid )
{
# Are we clearing?
if ( $ clear )
{
# Yup
$ variable_uuid = $ anvil - > Database - > insert_or_update_variables ( {
debug = > $ debug ,
variable_uuid = > $ variable_uuid ,
variable_value = > "clear" ,
update_value_only = > 1 ,
} ) ;
}
2022-01-14 02:07:25 +00:00
# if the 'clear' parameter isn't set, and the value is 'clear', change it to 'set'.
if ( ( $ variable_value eq "clear" ) && ( not $ clear ) )
2020-10-23 05:28:21 +00:00
{
# Set it.
$ variable_uuid = $ anvil - > Database - > insert_or_update_variables ( {
debug = > $ debug ,
variable_uuid = > $ variable_uuid ,
variable_value = > "set" ,
update_value_only = > 1 ,
} ) ;
}
else
{
# How old is it?
$ age = time - $ epoch_modified_date ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { age = > $ age } } ) ;
return ( $ age ) ;
}
}
elsif ( not $ clear )
{
# New, set it.
my $ variable_uuid = $ anvil - > Database - > insert_or_update_variables ( {
debug = > $ debug ,
variable_name = > $ name ,
variable_value = > "set" ,
variable_default = > "set" ,
variable_description = > "striker_0278" ,
variable_section = > "conditions" ,
variable_source_uuid = > $ host_uuid ,
variable_source_table = > $ source_table ,
} ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { variable_uuid = > $ variable_uuid } } ) ;
}
return ( $ age ) ;
}
2018-12-29 08:23:58 +00:00
= head2 register
2017-07-20 06:00:40 +00:00
2020-09-05 05:18:42 +00:00
This registers an alert to be sent later by C << Email - > send_alerts >> .
2017-07-20 06:00:40 +00:00
2020-09-05 05:18:42 +00:00
The C << alert_uuid >> is returned on success . If anything goes wrong , C << ! ! error ! ! >> is returned . If there are no recipients who would receive the alert , it will not be recorded and an empty string is returned .
2018-12-18 23:20:49 +00:00
Parameters ;
= head3 alert_level ( required )
2020-09-05 05:18:42 +00:00
This assigns an severity level to the alert . Any recipient listening to this level or higher will receive this alert . This value can be set as a numeric value or as a string .
2018-12-18 23:20:49 +00:00
2020-09-05 05:18:42 +00:00
= head4 1 / critical
2018-12-18 23:20:49 +00:00
Alerts at this level will go to all recipients , except for those ignoring the source system entirely .
This is reserved for alerts that could lead to imminent service interruption or unexpected loss of redundancy .
Alerts at this level should trigger alarm systems for all administrators as well as management who may be impacted by service interruptions .
2020-09-05 05:18:42 +00:00
= head4 2 / warning
2018-12-18 23:20:49 +00:00
This is used for alerts that require attention from administrators . Examples include intentional loss of redundancy caused by load shedding , hardware in pre - failure , loss of input power , temperature anomalies , etc .
Alerts at this level should trigger alarm systems for administrative staff .
2020-09-05 05:18:42 +00:00
= head4 3 / notice
2018-12-18 23:20:49 +00:00
This is used for alerts that are generally safe to ignore , but might provide early warnings of developing issues or insight into system behaviour .
Alerts at this level should not trigger alarm systems . Periodic review is sufficient .
2020-09-05 05:18:42 +00:00
= head4 4 / info
2018-12-18 23:20:49 +00:00
This is used for alerts that are almost always safe to ignore , but may be useful in testing and debugging .
2018-12-29 08:23:58 +00:00
= head3 clear_alert ( optional , default '0' )
If set , this indicate that the alert has returned to an OK state . Alert level is still honoured for notification target delivery decisions , but some internal values are adjusted .
2018-12-18 23:20:49 +00:00
= head3 message ( required )
This is the message body of the alert . It is expected to be in the format C << <string_key> >> . If variables are to be injected into the C << string_key >> , a comma - separated list in the format C << ! ! variable_name1 ! value1 ! ! [ , ! ! variable_nameN ! valueN ! ! ] >> is used .
Example with a message alone ; C << foo_0001 >> .
Example with two variables ; C << foo_0002 , ! ! bar ! abc ! ! , ! ! baz ! 123 ! ! >> .
2020-11-23 16:18:00 +00:00
B << Note >> : See C << variables >> for an alternate method of passing variables
2020-09-15 05:18:36 +00:00
2018-12-18 23:20:49 +00:00
= head3 set_by ( required )
This is the name of the program that registered this alert . Usually this is simply the caller ' s C << $ THIS_FILE >> or C << $ 0 >> variable .
= head3 show_header ( optional , default '1' )
When set to C << 0 >> , only the alert message body is shown , and the title is omitted . This can be useful when a set of alerts are sorted under a common title .
= head3 sort_position ( optional , default '9999' )
This is used to keep a set of alerts in a certain order when converted to an message body . By default , all alerts have a default value of '9999' , so they will be sorted using their severity level , and then the time they were entered into the system . If this is set to a number lower than this , then the value here will sort /prioritize messages over the severity/ time values . If two or more alerts have the same sort position , then severity and then time stamps will be used .
In brief ; alert messages are sorted in this order ;
1 . C << sort_position >>
2 . c << alert_level >>
3 . C << timestamp >>
2019-01-06 01:57:44 +00:00
NOTE: All C << sort_position >> values are automatically zero - padded ( ie: C << 12 >> - > C << 0012 >> ) to ensure accurate sorting . If you plan to use values greater than C << 9999 >> , be sure to manually zero - pad your numbers . ( Or , better , find a way to make shorter alerts ... ) .
2018-12-18 23:20:49 +00:00
NOTE: The timestamp is generally set for a given program or agent run ( set when connecting to the database ) , NOT by the real time of the database insert . For this reason , relying on the timestamp alone will not generally give the desired results , and why C << sort_position >> exists .
= head3 title ( optional )
2020-09-05 05:18:42 +00:00
NOTE: If not set and C << show_header >> is set to C << 1 >> , a generic title will be added based on the C << alert_level >> and if C << clear_alert >> is set or not .
2018-12-18 23:20:49 +00:00
This is the title of the alert . It is expected to be in the format C << <string_key> >> . If variables are to be injected into the C << string_key >> , a comma - separated list in the format C << ! ! variable_name1 ! value1 ! ! [ , ! ! variable_nameN ! valueN ! ! ] >> is used .
Example with a message alone ; C << foo_0001 >> .
Example with two variables ; C << foo_0002 , ! ! bar ! abc ! ! , ! ! baz ! 123 ! ! >> .
2017-07-20 06:00:40 +00:00
2020-11-23 16:18:00 +00:00
= head3 variables ( optional )
This can be set as a hash reference containing key / variable pairs to inject into the message key . the C << variable = > value >> pairs will be appended to the C << message >> key automatically . This is meant to simplify when an alert is also being longed , or when a large number of variables are being injected into the string .
2017-07-20 06:00:40 +00:00
= cut
2018-12-29 08:23:58 +00:00
sub register
2017-07-20 06:00:40 +00:00
{
my $ self = shift ;
my $ parameter = shift ;
2017-10-20 04:19:32 +00:00
my $ anvil = $ self - > parent ;
2018-12-18 23:20:49 +00:00
my $ debug = defined $ parameter - > { debug } ? $ parameter - > { debug } : 3 ;
2020-09-06 05:52:03 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , key = > "log_0125" , variables = > { method = > "Alert->register()" } } ) ;
2017-07-20 06:00:40 +00:00
2020-11-23 16:18:00 +00:00
my $ alert_level = defined $ parameter - > { alert_level } ? $ parameter - > { alert_level } : 0 ;
my $ clear_alert = defined $ parameter - > { clear_alert } ? $ parameter - > { clear_alert } : 0 ;
my $ message = defined $ parameter - > { message } ? $ parameter - > { message } : "" ;
my $ set_by = defined $ parameter - > { set_by } ? $ parameter - > { set_by } : "" ;
my $ show_header = defined $ parameter - > { show_header } ? $ parameter - > { show_header } : 1 ;
my $ sort_position = defined $ parameter - > { sort_position } ? $ parameter - > { sort_position } : 9999 ;
my $ title = defined $ parameter - > { title } ? $ parameter - > { title } : "" ;
my $ variables = defined $ parameter - > { variables } ? $ parameter - > { variables } : "" ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
2020-11-23 16:18:00 +00:00
show_header = > $ show_header ,
clear_alert = > $ clear_alert ,
alert_level = > $ alert_level ,
message = > $ message ,
set_by = > $ set_by ,
sort_position = > $ sort_position ,
title = > $ title ,
variables = > ref ( $ variables ) ,
2017-07-20 06:00:40 +00:00
} } ) ;
2020-09-05 05:18:42 +00:00
# Missing parameters?
2018-12-18 23:20:49 +00:00
if ( not $ alert_level )
{
2018-12-29 08:23:58 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 0 , priority = > "err" , key = > "log_0020" , variables = > { method = > "Alert->register()" , parameter = > "alert_level" } } ) ;
2018-12-18 23:20:49 +00:00
return ( "!!error!!" ) ;
}
2017-07-20 06:00:40 +00:00
if ( not $ set_by )
{
2018-12-29 08:23:58 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 0 , priority = > "err" , key = > "log_0020" , variables = > { method = > "Alert->register()" , parameter = > "set_by" } } ) ;
2017-08-02 01:04:35 +00:00
return ( "!!error!!" ) ;
2017-07-20 06:00:40 +00:00
}
2018-12-18 23:20:49 +00:00
if ( not $ message )
2017-07-20 06:00:40 +00:00
{
2018-12-29 08:23:58 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 0 , priority = > "err" , key = > "log_0020" , variables = > { method = > "Alert->register()" , parameter = > "message" } } ) ;
2017-08-02 01:04:35 +00:00
return ( "!!error!!" ) ;
2017-07-20 06:00:40 +00:00
}
2020-09-05 05:18:42 +00:00
2020-11-23 16:18:00 +00:00
if ( ref ( $ variables ) eq "HASH" )
2020-09-15 05:18:36 +00:00
{
2020-11-23 16:18:00 +00:00
foreach my $ variable ( sort { $ a cmp $ b } keys % { $ variables } )
2020-09-15 05:18:36 +00:00
{
2020-11-23 16:18:00 +00:00
my $ value = defined $ variables - > { $ variable } ? $ variables - > { $ variable } : "undefined:" . $ variable ;
2020-09-15 05:18:36 +00:00
$ message . = ",!!" . $ variable . "!" . $ value . "!!" ;
}
}
2021-07-09 05:20:25 +00:00
# Sort the alerts as they come in, if the sort_position was not set.
if ( $ sort_position == 9999 )
{
if ( not defined $ anvil - > data - > { sys } { sort_position } { $ alert_level } )
{
if ( ( $ alert_level eq "critical" ) or ( $ alert_level eq "1" ) )
{
$ anvil - > data - > { sys } { sort_position } { $ alert_level } = 0 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
"sys::sort_position::${alert_level}" = > $ anvil - > data - > { sys } { sort_position } { $ alert_level } ,
} } ) ;
}
elsif ( ( $ alert_level eq "warning" ) or ( $ alert_level eq "2" ) )
{
$ anvil - > data - > { sys } { sort_position } { $ alert_level } = 20 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
"sys::sort_position::${alert_level}" = > $ anvil - > data - > { sys } { sort_position } { $ alert_level } ,
} } ) ;
}
elsif ( ( $ alert_level eq "notice" ) or ( $ alert_level eq "3" ) )
{
$ anvil - > data - > { sys } { sort_position } { $ alert_level } = 100 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
"sys::sort_position::${alert_level}" = > $ anvil - > data - > { sys } { sort_position } { $ alert_level } ,
} } ) ;
}
else
{
$ anvil - > data - > { sys } { sort_position } { $ alert_level } = 1000 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
"sys::sort_position::${alert_level}" = > $ anvil - > data - > { sys } { sort_position } { $ alert_level } ,
} } ) ;
}
}
else
{
$ anvil - > data - > { sys } { sort_position } { $ alert_level } + + ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
"sys::sort_position::${alert_level}" = > $ anvil - > data - > { sys } { sort_position } { $ alert_level } ,
} } ) ;
}
$ sort_position = $ anvil - > data - > { sys } { sort_position } { $ alert_level } ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { sort_position = > $ sort_position } } ) ;
}
2020-09-05 05:18:42 +00:00
# If the alert level was a string, convert it to the numerical version. Also check that we've got a
# sane alert level at all.
if ( lc ( $ alert_level ) eq "critical" )
{
$ alert_level = 1 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { alert_level = > $ alert_level } } ) ;
}
elsif ( lc ( $ alert_level ) eq "warning" )
{
$ alert_level = 2 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { alert_level = > $ alert_level } } ) ;
}
elsif ( lc ( $ alert_level ) eq "notice" )
{
$ alert_level = 3 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { alert_level = > $ alert_level } } ) ;
}
elsif ( lc ( $ alert_level ) eq "info" )
{
$ alert_level = 4 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { alert_level = > $ alert_level } } ) ;
}
elsif ( ( $ alert_level =~ /\D/ ) or ( $ alert_level < 1 ) or ( $ alert_level > 4 ) )
{
# Invalid
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > 0 , priority = > "err" , key = > "error_0142" , variables = > { alert_level = > $ alert_level } } ) ;
return ( "!!error!!" ) ;
}
# Do we need to generate a header?
2018-12-18 23:20:49 +00:00
if ( ( $ show_header ) && ( not $ title ) )
2017-07-20 06:00:40 +00:00
{
2018-12-29 08:23:58 +00:00
# Set it based on the alert_level.
2020-09-08 05:07:23 +00:00
if ( $ alert_level == 1 ) { $ title = $ clear_alert ? "title_0005" : "title_0001" ; } # Critical (or Critical Cleared)
elsif ( $ alert_level == 2 ) { $ title = $ clear_alert ? "title_0006" : "title_0002" ; } # Warning (or Warning Cleared)
elsif ( $ alert_level == 3 ) { $ title = $ clear_alert ? "title_0007" : "title_0003" ; } # Notice (or Notice Cleared)
elsif ( $ alert_level == 4 ) { $ title = $ clear_alert ? "title_0008" : "title_0004" ; } # Info (or Info Cleared)
2018-12-29 08:23:58 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { title = > $ title } } ) ;
2017-07-20 06:00:40 +00:00
}
# zero-pad sort numbers so that they sort properly.
2018-12-18 23:20:49 +00:00
$ sort_position = sprintf ( "%04d" , $ sort_position ) ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { sort_position = > $ sort_position } } ) ;
2019-01-06 01:57:44 +00:00
# Before we actually record the alert, see if there are any recipients listening. For example, very
# rarely is anyone listening to alert level 4 (info), so skipping recording it saves unnecessary
# growth of the history.alerts table.
2020-09-05 05:18:42 +00:00
my $ proceed = 0 ;
$ anvil - > Database - > get_recipients ( { debug = > $ debug } ) ;
foreach my $ recipient_uuid ( keys % { $ anvil - > data - > { recipients } { recipient_uuid } } )
2017-07-20 06:00:40 +00:00
{
2020-09-05 05:18:42 +00:00
my $ recipient_email = $ anvil - > data - > { recipients } { recipient_uuid } { $ recipient_uuid } { recipient_email } ;
my $ recipient_level = $ anvil - > data - > { recipients } { recipient_uuid } { $ recipient_uuid } { level_on_host } ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > {
's1:recipient_uuid' = > $ recipient_uuid ,
's2:recipient_level' = > $ recipient_level ,
's3:recipient_email' = > $ recipient_email ,
} } ) ;
2017-07-20 06:00:40 +00:00
2020-09-05 05:18:42 +00:00
if ( $ recipient_level >= $ alert_level )
2017-07-20 06:00:40 +00:00
{
2020-09-05 05:18:42 +00:00
# Someone wants to hear about this.
$ proceed = 1 ;
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { proceed = > $ proceed } } ) ;
last ;
2017-07-20 06:00:40 +00:00
}
}
2020-09-05 05:18:42 +00:00
if ( not $ proceed )
2017-07-20 06:00:40 +00:00
{
2020-09-05 05:18:42 +00:00
# No one is listening, ignore.
return ( "" ) ;
2017-07-20 06:00:40 +00:00
}
# Always INSERT. ScanCore removes them as they're acted on (copy is left in history.alerts).
2020-09-05 05:18:42 +00:00
my $ alert_uuid = $ anvil - > Get - > uuid ( ) ;
2017-07-20 06:00:40 +00:00
my $ query = "
INSERT INTO
alerts
(
alert_uuid ,
alert_host_uuid ,
alert_set_by ,
alert_level ,
2018-12-18 23:20:49 +00:00
alert_title ,
alert_message ,
alert_sort_position ,
alert_show_header ,
2018-09-25 06:05:07 +00:00
modified_date
2017-07-20 06:00:40 +00:00
) VALUES (
2020-09-05 05:18:42 +00:00
".$anvil->Database->quote($alert_uuid)." ,
".$anvil->Database->quote($anvil->Get->host_uuid())." ,
2019-03-06 06:49:59 +00:00
".$anvil->Database->quote($set_by)." ,
2020-09-05 05:18:42 +00:00
".$anvil->Database->quote($alert_level)." ,
2019-03-06 06:49:59 +00:00
".$anvil->Database->quote($title)." ,
".$anvil->Database->quote($message)." ,
".$anvil->Database->quote($sort_position)." ,
".$anvil->Database->quote($show_header)." ,
2021-06-08 19:23:15 +00:00
".$anvil->Database->quote($anvil->Database->refresh_timestamp)."
2017-07-20 06:00:40 +00:00
) ;
" ;
2018-03-07 08:11:55 +00:00
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { query = > $ query } } ) ;
2017-10-20 04:19:32 +00:00
$ anvil - > Database - > write ( { query = > $ query , source = > $ THIS_FILE , line = > __LINE__ } ) ;
2017-07-20 06:00:40 +00:00
2020-09-05 05:18:42 +00:00
### TODO: Add an optional 'send_now' parameter to causes us to call 'Email->send_alerts'
$ anvil - > Log - > variables ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , list = > { alert_uuid = > $ alert_uuid } } ) ;
return ( $ alert_uuid ) ;
2017-07-20 06:00:40 +00:00
}
2020-11-12 05:35:51 +00:00
2019-09-03 18:07:39 +00:00
### TODO: Write this, maybe? Or remove it and ->warning()?
2017-05-02 04:41:12 +00:00
= head2 error
= cut
# Later, this will support all the translation and logging methods. For now, just print the error and exit.
sub error
{
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 ;
2020-09-06 05:52:03 +00:00
$ anvil - > Log - > entry ( { source = > $ THIS_FILE , line = > __LINE__ , level = > $ debug , key = > "log_0125" , variables = > { method = > "Alert->error()" } } ) ;
2017-05-02 04:41:12 +00:00
}
1 ;