mirror of
https://github.com/moudsen/mailGraph
synced 2025-10-28 16:17:39 +01:00
Compare commits
68 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
88862f69eb | ||
|
|
b31f686e24 | ||
|
|
bf7927a0c3 | ||
|
|
b43611588a | ||
|
|
b1f635ab36 | ||
|
|
dce6f57876 | ||
|
|
74e8a79e4c | ||
|
|
ba3e860fe0 | ||
|
|
2045601eca | ||
|
|
5f49628b8a | ||
|
|
a069dfa99d | ||
|
|
2c2561149a | ||
|
|
0847513eae | ||
|
|
62a3ee6149 | ||
|
|
a7196f8bc5 | ||
|
|
ededded428 | ||
|
|
624ce29a39 | ||
|
|
16729e0720 | ||
|
|
97dba8f197 | ||
|
|
288b0adaf7 | ||
|
|
2859d427f7 | ||
|
|
0cd169a9d2 | ||
|
|
8269195795 | ||
|
|
6da6da77fb | ||
|
|
00a67cbb65 | ||
|
|
bf970029ea | ||
|
|
9f3bfb5c69 | ||
|
|
504422f452 | ||
|
|
3ef16ae597 | ||
|
|
a597ce6ab3 | ||
|
|
a04050c3bc | ||
|
|
b81414e06d | ||
|
|
accb8f40d6 | ||
|
|
bf5bbcb778 | ||
|
|
702725e5f7 | ||
|
|
0725e01bc0 | ||
|
|
a8858784a5 | ||
|
|
39ada5d6ff | ||
|
|
32b8582757 | ||
|
|
9c42ee4b89 | ||
|
|
fa7fba040a | ||
|
|
3a4a7ab23e | ||
|
|
7075b40ad1 | ||
|
|
8bc476ed63 | ||
|
|
e80a2a56a3 | ||
|
|
4b28d3f066 | ||
|
|
93535d4006 | ||
|
|
b9622da72f | ||
|
|
56fb68ae9c | ||
|
|
fa51c3e59e | ||
|
|
4ae62b3a1f | ||
|
|
0731b87d44 | ||
|
|
bdf4b05a6b | ||
|
|
df16444665 | ||
|
|
bccd134523 | ||
|
|
27c1d64511 | ||
|
|
8597658670 | ||
|
|
b7e2062486 | ||
|
|
2955ffc404 | ||
|
|
2976a4af9c | ||
|
|
2882367308 | ||
|
|
8b4885038a | ||
|
|
f541719fbe | ||
|
|
9edab245d8 | ||
|
|
341bd561ab | ||
|
|
e2f657f3dd | ||
|
|
82a8fe4121 | ||
|
|
bee76b63c0 |
48
README.md
48
README.md
@@ -1,16 +1,42 @@
|
|||||||
## mailGraph (v1.28)
|
## Introduction ##
|
||||||
Zabbix Media module and scripts for sending e-mail alerts with graphs.
|
Over the years I've been using Zabbix as both personal and business solution for monitoring. One of the missing features in Zabbix itself was the ability to have graphs generated and associated to monitoring messages as well as an easy way to jump back into Zabbix when searching for monitoring information for specific events, problems or to have a direct link to the associated Zabbix item or host.
|
||||||
|
|
||||||
**Please use the Wiki for information on how to install, configure and use MailGraph in Zabbix.**
|
This initiated the development of mailGraph v1 in Zabbix 5.4 delivering an elementary solution for sending HTML enriched messages from Zabbix including graphs.
|
||||||
|
To facilitate in a very flexible way to setup and format messages including the graphs, Twig made its introduction towards mailGraph v2, the current release branch of mailGraph.
|
||||||
|
|
||||||
## UPGRADE NOTES
|
The below message is just an example of what MailGraph is capable of. The template engine in Twig allows for a fully customized message creation to your needs! It is also possible to add more Zabbix fields as any field is passed to Twig when accessible in Zabbix through the macro mechanism.
|
||||||
### v1.27
|
|
||||||
If you upgrade to v1.27 please be aware of the additional features for adding Tags to Trigger and Host to add additional graphs and the associated `html.template` updates that come alone with it (otherwise the new graphs will not show ...).
|
|
||||||
|
|
||||||
### v1.25 and higher
|
Note that all Zabbix host, item or screen related information is made available to Twig automatically.
|
||||||
Template data provisioning and code has fundamentally changed. If you upgrade from an earlier version as v1.25, make sure you understand the changes in templates/html.template (now making use of arrays for lists of items).
|
|
||||||
|
|
||||||
## Example message
|
Example message:
|
||||||
The below message is just an example of what MailGraph is capable of. The template engine used ("Twig") however allows for a fully customized message creation to your needs!
|
|
||||||
|
|
||||||
[](images/Example-mail-message-v122.png)
|

|
||||||
|
|
||||||
|
mailGraph is capable of adding several series of graphs into a single message delivering a unique experience when and how groups of graph images per the requested periods of time are added.
|
||||||
|
Currently mailGraph supports hosts, (one or more related) items and screens (applicable to Zabbix 5.4 only).
|
||||||
|
|
||||||
|
More information can be found in the Wiki.
|
||||||
|
|
||||||
|
## Installation ##
|
||||||
|
Please refer to the Wiki how to get mailGraph installed and configured on your system.
|
||||||
|
|
||||||
|
## Announcement - Zabbix 5.x maintenance for mailGraph end-of-life - mailGraph 3.x development in progress ##
|
||||||
|
_(2023/11/01)_
|
||||||
|
|
||||||
|
As per November 2023 the maintenance on mailGraph v2.x for Zabbix 5.x will stop in conjunction with the Zabbix lifecycle policy (https://www.zabbix.com/life_cycle_and_release_policy) as Zabbix 5 is nearing it's end of life.
|
||||||
|
|
||||||
|
In conjunction mailGraph v2.x is now frozen and I'm setting up a roadmap for mailGraph v3.x starting new development in Alfa mode as November 2023. This new code is not expected to be published until start of 2024.
|
||||||
|
|
||||||
|
Principal bug fixing on mailGraph v2.x (logic failure or similar) will continue but only for Zabbix 6.x onwards.
|
||||||
|
|
||||||
|
I'm open to new feature requests - please raise an issue for this in this Github space.
|
||||||
|
|
||||||
|
## Special thank you ##
|
||||||
|
I would like to express my gratitude to the following people that have actively contributed to bring bugs and improvements to my attention:
|
||||||
|
- [pqvindesland](https://github.com/pqvindesland)
|
||||||
|
- [BernardLinz](https://github.com/BernhardLinz)
|
||||||
|
- [WMP](https://github.com/WMP)
|
||||||
|
- [dima-online](https://github.com/dima-online)
|
||||||
|
|
||||||
|
## IMPORTANT NOTE for users of mailGraph v2 and older and Zabbix versions under 6.0 ##
|
||||||
|
As per November 2023, mailGraph is no longer maintained for Zabbix 5 and older.
|
||||||
|
|||||||
@@ -1,7 +1,5 @@
|
|||||||
{
|
{
|
||||||
"script_baseurl": "https:\/\/mydomain.com\/",
|
"script_baseurl": "https:\/\/mydomain.com\/",
|
||||||
"cli_itemId": 0,
|
|
||||||
"cli_triggerId": 0,
|
|
||||||
"cli_eventId": 0,
|
"cli_eventId": 0,
|
||||||
"cli_duration": 0,
|
"cli_duration": 0,
|
||||||
"cli_recipient": "recipient@mydomain.com",
|
"cli_recipient": "recipient@mydomain.com",
|
||||||
@@ -15,14 +13,19 @@
|
|||||||
"zabbix_user_pwd": "astrongpassword",
|
"zabbix_user_pwd": "astrongpassword",
|
||||||
"zabbix_api_user": "alogicalusername",
|
"zabbix_api_user": "alogicalusername",
|
||||||
"zabbix_api_pwd": "astrongpassword",
|
"zabbix_api_pwd": "astrongpassword",
|
||||||
"mail_from": "sender@mydomain.com",
|
|
||||||
"subject": "{{ HOST_NAME|raw }}: ({{ EVENT_SEVERITY }}) {{ EVENT_NAME|raw }}",
|
"subject": "{{ HOST_NAME|raw }}: ({{ EVENT_SEVERITY }}) {{ EVENT_NAME|raw }}",
|
||||||
"smtp_server": "localhost",
|
"smtp_server": "localhost",
|
||||||
"smtp_port": 25,
|
"smtp_port": 25,
|
||||||
"smtp_transport": "none",
|
"smtp_security": "none",
|
||||||
"smtp_strict": "yes",
|
"smtp_strict": "yes",
|
||||||
|
"smtp_from_address": "mailgraph@mydomain.com",
|
||||||
|
"smtp_from_name": "mailGraph",
|
||||||
|
"smtp_reply_address": "feedback@mydomain.com",
|
||||||
|
"smtp_reply_name": "mailGraph response mailbox",
|
||||||
"graph_match": "any",
|
"graph_match": "any",
|
||||||
"period": "20m",
|
"period": "20m",
|
||||||
"period_header": "Last 20 minutes",
|
"period_header": "Last 20 minutes",
|
||||||
|
"retention_images": 7,
|
||||||
|
"retention_logs": 14,
|
||||||
"debug": 0
|
"debug": 0
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,5 @@
|
|||||||
{
|
{
|
||||||
"script_baseurl": "https:\/\/mydomain.com\/",
|
"script_baseurl": "https:\/\/mydomain.com\/",
|
||||||
"cli_itemId": 0,
|
|
||||||
"cli_triggerId": 0,
|
|
||||||
"cli_eventId": 0,
|
"cli_eventId": 0,
|
||||||
"cli_duration": 0,
|
"cli_duration": 0,
|
||||||
"cli_recipient": "recipient@mydomain.com",
|
"cli_recipient": "recipient@mydomain.com",
|
||||||
@@ -15,12 +13,15 @@
|
|||||||
"zabbix_user_pwd": "astrongpassword",
|
"zabbix_user_pwd": "astrongpassword",
|
||||||
"zabbix_api_user": "alogicalusername",
|
"zabbix_api_user": "alogicalusername",
|
||||||
"zabbix_api_pwd": "astrongpassword",
|
"zabbix_api_pwd": "astrongpassword",
|
||||||
"mail_from": "sender@mydomain.com",
|
|
||||||
"subject": "{{ HOST_NAME|raw }}: ({{ EVENT_SEVERITY }}) {{ EVENT_NAME|raw }}",
|
"subject": "{{ HOST_NAME|raw }}: ({{ EVENT_SEVERITY }}) {{ EVENT_NAME|raw }}",
|
||||||
"smtp_server": "localhost",
|
"smtp_server": "localhost",
|
||||||
"smtp_port": 25,
|
"smtp_port": 25,
|
||||||
"smtp_transport": "none",
|
"smtp_security": "none",
|
||||||
"smtp_strict": "yes",
|
"smtp_strict": "yes",
|
||||||
|
"smtp_from_address": "mailgraph@mydomain.com",
|
||||||
|
"smtp_from_name": "mailGraph",
|
||||||
|
"smtp_reply_address": "feedback@mydomain.com",
|
||||||
|
"smtp_reply_name": "mailGraph response mailbox",
|
||||||
"graph_match": "any",
|
"graph_match": "any",
|
||||||
"periods": "10m,4h,2d,7d",
|
"periods": "10m,4h,2d,7d",
|
||||||
"periods_headers": "Last 10 minutes,Last 4 hours,Last 2 days,Last 7 days",
|
"periods_headers": "Last 10 minutes,Last 4 hours,Last 2 days,Last 7 days",
|
||||||
|
|||||||
@@ -27,7 +27,7 @@
|
|||||||
$content = file_get_contents($argv[1]);
|
$content = file_get_contents($argv[1]);
|
||||||
$data = json_decode($content,TRUE);
|
$data = json_decode($content,TRUE);
|
||||||
|
|
||||||
if ($data==NULL)
|
if (($data==NULL) && (sizeof($content)>2))
|
||||||
{
|
{
|
||||||
echo 'Invalid JSON format in config file?!'.$cCRLF;
|
echo 'Invalid JSON format in config file?!'.$cCRLF;
|
||||||
die;
|
die;
|
||||||
|
|||||||
BIN
images/Example-Zabbix-MediaType-Config.png
Normal file
BIN
images/Example-Zabbix-MediaType-Config.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 85 KiB |
BIN
images/Screenshot 2023-08-18 110118.png
Normal file
BIN
images/Screenshot 2023-08-18 110118.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 115 KiB |
1
javascript/READ.ME
Normal file
1
javascript/READ.ME
Normal file
@@ -0,0 +1 @@
|
|||||||
|
Issue #32 has been resolved. The javascript code now works for Zabbix 5.4, 6.0 LTS and 6.4.
|
||||||
@@ -1,7 +1,19 @@
|
|||||||
|
// mailGraph v2.16
|
||||||
|
|
||||||
|
// Function to test string
|
||||||
|
function isJSON(str) {
|
||||||
|
try {
|
||||||
|
JSON.stringify(JSON.parse(str));
|
||||||
|
return true;
|
||||||
|
} catch (e) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// Pickup parameters
|
// Pickup parameters
|
||||||
params = JSON.parse(value),
|
params = JSON.parse(value),
|
||||||
req = new CurlHttpRequest(),
|
req = new HttpRequest(),
|
||||||
fields = {},
|
fields = {},
|
||||||
resp = '',
|
resp = '',
|
||||||
result = { tags: {} };
|
result = { tags: {} };
|
||||||
@@ -13,14 +25,18 @@ try {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Declare output type
|
// Declare output type
|
||||||
req.AddHeader('Content-Type: application/json');
|
req.addHeader('Content-Type: application/json');
|
||||||
|
|
||||||
// Must have fields
|
// Pick up fields relevant for mailGraph API level call while parsing/casting fields that should be integer
|
||||||
fields.itemId = params.itemId;
|
fields.itemId = params.itemId * 1;
|
||||||
fields.eventId = params.eventId;
|
fields.eventId = params.eventId * 1;
|
||||||
fields.recipient = params.recipient;
|
fields.recipient = params.recipient;
|
||||||
fields.baseURL = params.baseURL;
|
fields.baseURL = params.baseURL;
|
||||||
fields.duration = params.duration;
|
fields.duration = params.duration * 1;
|
||||||
|
|
||||||
|
if (fields.recipient.charAt(0) == '{') {
|
||||||
|
throw '[MailGraph Webhook] Please define recipient for the test message!';
|
||||||
|
}
|
||||||
|
|
||||||
// Optional fields
|
// Optional fields
|
||||||
if (typeof params.graphWidth === 'string') { fields.graphWidth = params.graphWidth; }
|
if (typeof params.graphWidth === 'string') { fields.graphWidth = params.graphWidth; }
|
||||||
@@ -40,22 +56,44 @@ try {
|
|||||||
|
|
||||||
// Post information to the processing script
|
// Post information to the processing script
|
||||||
Zabbix.Log(4, '[MailGraph Webhook] Sending request: ' + params.URL + '?' + JSON.stringify(fields));
|
Zabbix.Log(4, '[MailGraph Webhook] Sending request: ' + params.URL + '?' + JSON.stringify(fields));
|
||||||
var resp = req.Post(params.URL,JSON.stringify(fields));
|
var resp = req.post(params.URL,JSON.stringify(fields));
|
||||||
Zabbix.Log(4, '[Mailgraph Webhook] Receiving response:' + resp);
|
Zabbix.Log(4, '[Mailgraph Webhook] Received response:' + resp);
|
||||||
|
|
||||||
// If there was an error, report it
|
// The response can be
|
||||||
if (req.Status() != 200) { throw JSON.parse(resp).errors[0]; }
|
// - did not receive status 200 as result (contains HTTP server response)
|
||||||
|
// - null (no response received at all)
|
||||||
|
// - empty string (likely no e-mail sent due to recipient issue)
|
||||||
|
// - not json (debugging message for troubleshooting or configuration hints)
|
||||||
|
// - json (contains the mail message ID sent)
|
||||||
|
|
||||||
// We expect the message id back from the processing script
|
if (req.getStatus() != 200) {
|
||||||
resp = JSON.parse(resp);
|
throw '[MailGraph Webhook] Processing of mailGraph.php failed: ' + resp;
|
||||||
result.tags.__message_id = resp.messageId;
|
}
|
||||||
|
if (resp==null) {
|
||||||
|
throw '[MailGraph Webhook] No response received from mailGraph.php? This should not occur (check URL and your webserver!)';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (resp=='') {
|
||||||
|
throw '[MailGraph Webhook] No data received from mailGraph - please check recipient address or mailGraph log and retry.';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if JSON was returned
|
||||||
|
if (!isJSON(resp)) {
|
||||||
|
throw '[MailGraph Webhook] An error has occurred during processing: ' + resp;
|
||||||
|
}
|
||||||
|
|
||||||
|
// We expect the message id back from the processing script response in JSON format
|
||||||
|
msg = JSON.parse(resp);
|
||||||
|
|
||||||
|
result.tags.__message_id = msg.messageId;
|
||||||
|
Zabbix.Log(4, '[MailGraph Webhook] Message sent with identification "' + msg.messageId + '"');
|
||||||
|
|
||||||
// Pass the result back to Zabbix
|
// Pass the result back to Zabbix
|
||||||
return JSON.stringify(result);
|
return JSON.stringify(result);
|
||||||
}
|
}
|
||||||
catch (error)
|
catch (error)
|
||||||
{
|
{
|
||||||
// In case something went wrong in the processing, pass the error back to Zabbix
|
// In case something else went wrong in the processing, pass the error back to Zabbix
|
||||||
Zabbix.Log(127, 'MailGraph notification failed: '+error);
|
Zabbix.Log(127, 'MailGraph notification failed: '+error);
|
||||||
throw 'MailGraph notification failed : '+error;
|
throw 'MailGraph notification failed : '+error;
|
||||||
}
|
}
|
||||||
|
|||||||
660
mailGraph.php
660
mailGraph.php
@@ -8,34 +8,39 @@
|
|||||||
// upon an alert message.
|
// upon an alert message.
|
||||||
//
|
//
|
||||||
// ------------------------------------------------------------------------------------------------------
|
// ------------------------------------------------------------------------------------------------------
|
||||||
// 1.00 2021/02/26 - Mark Oudsen - MVP version, ready for distribution
|
// Release 1 tested with Zabbix 5.4 and 6.0 LTS (history available on GitHub)
|
||||||
// 1.01 2021/02/27 - Mark Oudsen - Enhanced search for associated graphs to an item // bug fixes
|
// ------------------------------------------------------------------------------------------------------
|
||||||
// 1.10 2021/02/27 - Mark Oudsen - Moved all configuration outside code
|
// Release 2 tested with Zabbix 5.4, 6.0 LTS and 6.4 - tested with latest Composer libraries
|
||||||
// 1.11 2021/02/28 - Mark Oudsen - Bugfixes
|
// ------------------------------------------------------------------------------------------------------
|
||||||
// 1.12 2021/03/01 - Mark Oudsen - Bugfixes
|
// 2.00 2021/12/16 - Mark Oudsen - itemId not always provisioned by Zabbix
|
||||||
// Adding mail server configuration via config.json
|
// Several fixes on warning - several small bug fixes
|
||||||
// 1.13 2021/03/01 - Mark Oudsen - Added smtp options to encrypt none,ssl,tls
|
// 2.01 2021/12/16 - Mark Oudsen - Screens are no longer available - reverting to using Dashboards now
|
||||||
// 1.14 2021/03/01 - Mark Oudsen - Added smtp strict certificates yes|no via config.json
|
// 2.02 2022/01/30 - Mark Oudsen - Added cleanup routine for old logs and images
|
||||||
// 1.15 2021/03/01 - Mark Oudsen - Revised relevant graph locator; allowing other item graphs if current
|
// 2.10 2023/06/30 - Mark Oudsen - Refactored deprecated code - now compatible with Zabbix 6.0 LTS, 6.4
|
||||||
// item does not have a graph associated
|
// 2.11 2023/07/01 - Mark Oudsen - Refactored Zabbix javascript - now capturing obvious errors
|
||||||
// 1.16 2021/03/02 - Mark Oudsen - Found issue with graph.get not returning graphs to requested item ids
|
// Added ability to locate latest problems for testing purposes
|
||||||
// Workaround programmed (fetch host graphs, search for certain itemids)
|
// 2.12 2023/07/02 - Mark Oudsen - Replaced SwiftMailer with PHPMailer (based on AutoTLS)
|
||||||
// 1.17 2021/03/02 - Mark Oudsen - Added ability to specify period of time displayed in the graph
|
// 2.13 2023/07/03 - Mark Oudsen - Bugfixes speciifally on links into Zabbix (missing context or info)
|
||||||
// 1.18 2021/03/04 - Mark Oudsen - Added ability to specify Tags per trigger
|
// 2.14 2023/07/10 - Mark Oudsen - Adding ability to set 'From' and 'ReplyTo' addresses in configuration
|
||||||
// Shorten long "lastvalue" or "prevvalue"
|
// Adding ACK_URL for utilization in the template to point to Ack page
|
||||||
// 1.19 2021/03/05 - Mark Oudsen - Added ability to pass Zabbix 'infoXXX' parameters for TWIG template
|
// Small refactor on itemId variable processing (no longer mandatory)
|
||||||
// 1.20 2021/03/07 - Mark Oudsen - Production level version - leaving BETA from here on ...
|
// Additional logic added to random eventId to explain in case of issues
|
||||||
// 1.21 2021/03/09 - Mark Oudsen - Reverted graph.get code back to original code as it was not a bug but
|
// Fixed missing flag for fetching web url related items
|
||||||
// a wrongly typed requested (should be ARRAY, not comma separated)!
|
// 2.15 2023/08/16 - Mark Oudsen - Bugfix for Zabbix 5.4 where empty or zeroed variables are no longer
|
||||||
// 1.22 2021/03/10 - Mark Oudsen - Added ability to embed multiple periods (1-4) of the same graph
|
// passed by Zabbix (hence resulting in weird errors)
|
||||||
// 1.23 2021/03/12 - Mark Oudsen - Added graph support for 'Stacked', 'Pie' and 'Exploded'
|
// 2.16 2023/08/16 - Mark Oudsen - Adding ability to use ACKNOWLEDGE messages in the mail message
|
||||||
// 1.24 2021/03/12 - Mark Oudsen - Added support for HTTP proxy
|
// 2.17 2024/12/30 - Mark Oudsen - Fixed #47 mailData initializaton (wrong location) - BernardLinz
|
||||||
// 1.25 2021/03/16 - Mark Oudsen - Refactoring for optimized flow and relevant data retrieval
|
// Fixed #46 invalid GraphId on trigger tag - BernardLinz
|
||||||
// 1.26 2021/03/19 - Mark Oudsen - Bugfixes after refactor (wrong itemId and incorrect eventValue)
|
// Fixed #44 config.json.template wrong var name - WMP
|
||||||
// Suppressing Zabbix username-password in log
|
// Fixed #45 handling of international characters - Dima-online
|
||||||
// 1.27 2021/03/19 - Mark Oudsen - Added ability to define "mailGraph.screen" tag to embed graphs from
|
// Tested with latest PHPMailer (6.9.3) and TWIG (3.11.3), PHP 7.4
|
||||||
// Added PHP informational and warnings to log for easier debug/spotting
|
// Tested with PHP 8.3, TWIG (3.18.0)
|
||||||
// 1.28 2021/03/24 - Mark Oudsen - Added ability to specify username/password for TLS/SSL
|
// 2.18 2025/01/10 - Mark Oudsen - SCREEN tag information is only processed for Zabbix versions <= 5
|
||||||
|
// 2025/01/14 - Mark Oudsen - Fixed #51 SMTPS (implicit) or STARTTLS (explicit)
|
||||||
|
// ------------------------------------------------------------------------------------------------------
|
||||||
|
// Release 3 placeholder for Zabbix 7.0 LTS and 7.2+
|
||||||
|
// ------------------------------------------------------------------------------------------------------
|
||||||
|
// v2.18 Testing on Zabbix 7.0 LTS (in progress), Zabbix 7.2 (in progress)
|
||||||
// ------------------------------------------------------------------------------------------------------
|
// ------------------------------------------------------------------------------------------------------
|
||||||
//
|
//
|
||||||
// (C) M.J.Oudsen, mark.oudsen@puzzl.nl
|
// (C) M.J.Oudsen, mark.oudsen@puzzl.nl
|
||||||
@@ -44,40 +49,79 @@
|
|||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
//
|
||||||
|
// Notes
|
||||||
|
// -----
|
||||||
|
// 1) mailGraph is following the environmental requirements from Zabbix, supporting PHP 7.4-8.3 ad per
|
||||||
|
// - https://www.zabbix.com/documentation/6.0/en/manual/installation/requirements
|
||||||
|
// - https://www.zabbix.com/documentation/6.4/en/manual/installation/requirements
|
||||||
|
//
|
||||||
|
// 2) TWIG 3.18.0 is available on PHP 8 only (seemless upgrade when using composer update)
|
||||||
|
//
|
||||||
|
// 3) Testing of composer libraries updates is limited to every 6 to 12 months
|
||||||
|
// - In case you encounter an issue, please raise an issue on GitHub
|
||||||
|
// https://github.com/moudsen/mailGraph/issues
|
||||||
|
//
|
||||||
|
//
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
//
|
||||||
|
// Roadmap
|
||||||
|
// -------
|
||||||
|
// - Automatic setup and configuration of mailGraph
|
||||||
|
// - Automatic code updates (CLI triggered)
|
||||||
|
// - Add DASHBOARD facility (SCREEN was abandoned in Zabbix 5.4
|
||||||
|
// - Extract Graph API functionality to seperate code unit/object
|
||||||
|
//
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
//
|
//
|
||||||
// MAIN SEQUENCE
|
// MAIN SEQUENCE
|
||||||
// -------------
|
// -------------
|
||||||
// 1) Fetch trigger, item, host, graph, event information via Zabbix API via CURL
|
// 1) Fetch trigger, item, host, graph, event information via Zabbix API via CURL
|
||||||
// 2) Fetch Graph(s) associated to the item/trigger (if any) via Zabbix URL login via CURL
|
// 2) Fetch Graph(s) associated to the item/trigger (if any) via Zabbix URL via CURL
|
||||||
// 3) Build and send mail message from template using Swift/TWIG
|
// 3) Build and send mail message from template using PHPmailer//TWIG
|
||||||
//
|
//
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
// CONSTANTS
|
// CONSTANTS
|
||||||
|
$cVersion = 'v2.18';
|
||||||
$cVersion = 'v1.28';
|
|
||||||
$cCRLF = chr(10).chr(13);
|
$cCRLF = chr(10).chr(13);
|
||||||
$maskDateTime = 'Y-m-d H:i:s';
|
$maskDateTime = 'Y-m-d H:i:s';
|
||||||
$maxGraphs = 4;
|
$maxGraphs = 8;
|
||||||
|
|
||||||
// DEBUG SETTINGS
|
// DEBUG SETTINGS
|
||||||
// -- Should be FALSE for production level use
|
// -- Should be FALSE for production level use
|
||||||
|
|
||||||
$cDebug = TRUE; // Extended debug logging mode
|
$cDebug = TRUE; // Extended debug logging mode, switch to FALSE for production environment
|
||||||
$cDebugMail = FALSE; // If TRUE, includes log in the mail message (html and plain text attachments)
|
$cDebugMail = FALSE; // If TRUE, includes log in the mail message (html and plain text attachments)
|
||||||
$showLog = FALSE; // Display the log - !!! only use in combination with CLI mode !!!
|
$showLog = FALSE; // Display the log - !!! only use in combination with CLI mode !!!
|
||||||
|
|
||||||
|
// Limit server level output errors
|
||||||
|
|
||||||
|
error_reporting(E_ERROR | E_PARSE);
|
||||||
|
|
||||||
// INCLUDE REQUIRED LIBRARIES (Composer)
|
// INCLUDE REQUIRED LIBRARIES (Composer)
|
||||||
// (configure at same location as the script is running or load in your own central library)
|
// (configure at same location as the script is running or load in your own central library)
|
||||||
// -- swiftmailer/swiftmailer https://swiftmailer.symfony.com/docs/introduction.html
|
// -- phpmailer/phpmailer https://github.com/PHPMailer/PHPMailer
|
||||||
// -- twig/twig https://twig.symfony.com/doc/3.x/templates.html
|
// -- twig/twig https://twig.symfony.com/doc/3.x/templates.html
|
||||||
|
|
||||||
// Change only required if you decide to use a local/central library, otherwise leave as is
|
// Change only required if you decide to use a local/central library, otherwise leave as is
|
||||||
include(getcwd().'/vendor/autoload.php');
|
include(getcwd().'/vendor/autoload.php');
|
||||||
|
|
||||||
|
use PHPMailer\PHPMailer\PHPMailer;
|
||||||
|
use PHPMailer\PHPMailer\SMTP;
|
||||||
|
use PHPMailer\PHPMailer\Exception;
|
||||||
|
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// Fetch the HTML source of the given URL
|
// Fetch the HTML source of the given URL
|
||||||
@@ -99,23 +143,24 @@
|
|||||||
$ch = curl_init();
|
$ch = curl_init();
|
||||||
|
|
||||||
// Set options
|
// Set options
|
||||||
curl_setopt($ch, CURLOPT_USERAGENT, 'Zabbix-mailGraph - '.$cVersion);
|
|
||||||
|
|
||||||
if ((isset($HTTPProxy)) && ($HTTPProxy!=''))
|
if ((isset($HTTPProxy)) && ($HTTPProxy!=''))
|
||||||
{
|
{
|
||||||
_log('% Using proxy: '.$HTTPProxy);
|
if ($cDebug) { _log('% Using proxy: '.$HTTPProxy); }
|
||||||
curl_setopt($ch, CURLOPT_PROXY, $HTTPProxy);
|
curl_setopt($ch, CURLOPT_PROXY, $HTTPProxy);
|
||||||
}
|
}
|
||||||
|
|
||||||
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4);
|
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4);
|
||||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
|
|
||||||
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 20);
|
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 20);
|
||||||
|
|
||||||
|
curl_setopt($ch, CURLOPT_POST, TRUE);
|
||||||
|
curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($data));
|
||||||
|
|
||||||
curl_setopt($ch, CURLOPT_URL, $url);
|
curl_setopt($ch, CURLOPT_URL, $url);
|
||||||
curl_setopt($ch, CURLOPT_HTTPHEADER, array('Content-Type:application/json'));
|
curl_setopt($ch, CURLOPT_HTTPHEADER, array('Content-Type:application/json'));
|
||||||
curl_setopt($ch, CURLOPT_POST, TRUE);
|
curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
|
||||||
curl_setopt($ch, CURLOPT_POSTFIELDS,
|
|
||||||
json_encode($data,JSON_PRETTY_PRINT|JSON_NUMERIC_CHECK));
|
curl_setopt($ch, CURLOPT_USERAGENT, 'Zabbix-mailGraph - '.$cVersion);
|
||||||
|
|
||||||
// Execute Curl
|
// Execute Curl
|
||||||
$data = curl_exec($ch);
|
$data = curl_exec($ch);
|
||||||
@@ -124,9 +169,10 @@
|
|||||||
{
|
{
|
||||||
_log('! Failed: '.curl_error($ch));
|
_log('! Failed: '.curl_error($ch));
|
||||||
|
|
||||||
$data = 'An error occurred while retreiving the requested page.'.$cCRLF;
|
$data = array();
|
||||||
$data .= 'Requested page = '.$url.$cCRLF;
|
$data[] = 'An error occurred while retreiving the requested page.';
|
||||||
$data .= 'Error = '.curl_error($ch).$cCRLF;
|
$data[] .= 'Requested page = '.$url;
|
||||||
|
$data[] .= 'Error = '.curl_error($ch);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -238,8 +284,12 @@
|
|||||||
|
|
||||||
curl_close($ch);
|
curl_close($ch);
|
||||||
|
|
||||||
// Delete cookie
|
// Delete cookie (if exists)
|
||||||
|
if (file_exists($filename_cookie))
|
||||||
|
{
|
||||||
unlink($filename_cookie);
|
unlink($filename_cookie);
|
||||||
|
_log('- Removed cookie '.$filename_cookie);
|
||||||
|
}
|
||||||
|
|
||||||
// Write file
|
// Write file
|
||||||
$fp = fopen($image_name, 'w');
|
$fp = fopen($image_name, 'w');
|
||||||
@@ -377,6 +427,78 @@
|
|||||||
return($info);
|
return($info);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Cleanup 'path': remove files with 'ext' older than 'daysOld'
|
||||||
|
|
||||||
|
function cleanupDir($path, $ext, $daysOld)
|
||||||
|
{
|
||||||
|
_log('- Scanning "'.$path.'"');
|
||||||
|
|
||||||
|
$files = glob($path.'/*'.$ext);
|
||||||
|
$now = time();
|
||||||
|
|
||||||
|
$filesRemoved = 0;
|
||||||
|
$filesKept = 0;
|
||||||
|
|
||||||
|
foreach ($files as $file)
|
||||||
|
{
|
||||||
|
if (is_file($file))
|
||||||
|
{
|
||||||
|
if ($now - filemtime($file) >= (60 * 60 * 24 * $daysOld))
|
||||||
|
{
|
||||||
|
_log('> Removing "'.$file.'"');
|
||||||
|
unlink($file);
|
||||||
|
$filesRemoved++;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$filesKept++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_log(': Done. Cleaned up '.$filesRemoved.' file(s), kept '.$filesKept.' file(s)');
|
||||||
|
}
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Zabbix translator functions
|
||||||
|
|
||||||
|
function zabbixTStoString($linuxTime)
|
||||||
|
{
|
||||||
|
return date("Y-m-d H:i:s", $linuxTime);
|
||||||
|
}
|
||||||
|
|
||||||
|
function zabbixActionToString($actionMask)
|
||||||
|
{
|
||||||
|
$values = [];
|
||||||
|
|
||||||
|
if ($actionMask & 1) { $values[] = "Close problem"; }
|
||||||
|
if ($actionMask & 2) { $values[] = "Acknowledge event"; }
|
||||||
|
if ($actionMask & 4) { $values[] = "Add message"; }
|
||||||
|
if ($actionMask & 8) { $values[] = "Change severity"; }
|
||||||
|
if ($actionMask & 16) { $values[] = "Unacknowledge event"; }
|
||||||
|
if ($actionMask & 32) { $values[] = "Suppress event"; }
|
||||||
|
if ($actionMask & 64) { $values[] = "Unsuppress event"; }
|
||||||
|
if ($actionMask & 128) { $values[] = "Change event rank to cause"; }
|
||||||
|
if ($actionMask & 256) { $values[] = "Change event rank to sympton"; }
|
||||||
|
|
||||||
|
return implode(", ", $values);
|
||||||
|
}
|
||||||
|
|
||||||
|
function zabbixSeverityToString($severity)
|
||||||
|
{
|
||||||
|
switch ($severity) {
|
||||||
|
case 0: return('Not classified');
|
||||||
|
case 1: return('Information');
|
||||||
|
case 2: return('Warning');
|
||||||
|
case 3: return('Average');
|
||||||
|
case 4: return('High');
|
||||||
|
case 5: return('Disaster');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// Initialize ///////////////////////////////////////////////////////////////////////////////////////////
|
// Initialize ///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
@@ -389,6 +511,10 @@
|
|||||||
_log('# Configuration taken from config.json'.$cCRLF.
|
_log('# Configuration taken from config.json'.$cCRLF.
|
||||||
json_encode(maskOutputFields($config),JSON_PRETTY_PRINT|JSON_NUMERIC_CHECK));
|
json_encode(maskOutputFields($config),JSON_PRETTY_PRINT|JSON_NUMERIC_CHECK));
|
||||||
|
|
||||||
|
// --- MAIL DATA ---
|
||||||
|
|
||||||
|
$mailData = array();
|
||||||
|
|
||||||
// --- POST DATA ---
|
// --- POST DATA ---
|
||||||
|
|
||||||
// Read POST data
|
// Read POST data
|
||||||
@@ -404,18 +530,25 @@
|
|||||||
{
|
{
|
||||||
if (($argc>1) && ($argv[1]=='test'))
|
if (($argc>1) && ($argv[1]=='test'))
|
||||||
{
|
{
|
||||||
|
// Switch on CLI log display
|
||||||
|
$showLog = TRUE;
|
||||||
|
|
||||||
|
_log('<<< mailGraph '.$cVersion.' >>>');
|
||||||
_log('# Invoked from CLI');
|
_log('# Invoked from CLI');
|
||||||
|
|
||||||
// Assumes that config.json file has the correct information
|
// Assumes that config.json file has the correct information for MANDATORY information
|
||||||
|
|
||||||
|
// DEFAULTS
|
||||||
|
$problemData['eventId'] = 0;
|
||||||
|
$problemData['duration'] = 0;
|
||||||
|
|
||||||
// MANDATORY
|
// MANDATORY
|
||||||
$problemData['itemId'] = $config['cli_itemId'];
|
|
||||||
$problemData['eventId'] = $config['cli_eventId'];
|
|
||||||
$problemData['recipient'] = $config['cli_recipient'];
|
$problemData['recipient'] = $config['cli_recipient'];
|
||||||
$problemData['baseURL'] = $config['cli_baseURL'];
|
$problemData['baseURL'] = $config['cli_baseURL'];
|
||||||
$problemData['duration'] = $config['cli_duration'];
|
|
||||||
|
|
||||||
// OPTIONAL
|
// OPTIONAL
|
||||||
|
if (isset($config['cli_eventId'])) { $problemData['eventId'] = $config['cli_eventId']; }
|
||||||
|
if (isset($config['cli_duration'])) { $problemData['duration'] = $config['cli_duration']; }
|
||||||
if (isset($config['cli_subject'])) { $problemData['subject'] = $config['cli_subject']; }
|
if (isset($config['cli_subject'])) { $problemData['subject'] = $config['cli_subject']; }
|
||||||
if (isset($config['cli_period'])) { $problemData['period'] = $config['cli_period']; }
|
if (isset($config['cli_period'])) { $problemData['period'] = $config['cli_period']; }
|
||||||
if (isset($config['cli_period_header'])) { $problemData['period_header'] = $config['cli_period_header']; }
|
if (isset($config['cli_period_header'])) { $problemData['period_header'] = $config['cli_period_header']; }
|
||||||
@@ -424,8 +557,34 @@
|
|||||||
if (isset($config['cli_debug'])) { $problemData['debug'] = $config['cli_debug']; }
|
if (isset($config['cli_debug'])) { $problemData['debug'] = $config['cli_debug']; }
|
||||||
if (isset($config['cli_proxy'])) { $problemData['HTTPProxy'] = $config['cli_proxy']; }
|
if (isset($config['cli_proxy'])) { $problemData['HTTPProxy'] = $config['cli_proxy']; }
|
||||||
|
|
||||||
|
// BACKWARDS COMPATIBILITY - obsolete from Zabbix 6.2 onwards
|
||||||
|
$problemData['itemId'] = 0;
|
||||||
|
if (isset($config['cli_itemId'])) { $problemData['itemId'] = $config['cli_itemId']; }
|
||||||
|
}
|
||||||
|
|
||||||
|
if (($argc>1) && ($argv[1]=='cleanup'))
|
||||||
|
{
|
||||||
// Switch on CLI log display
|
// Switch on CLI log display
|
||||||
$showLog = TRUE;
|
$showLog = TRUE;
|
||||||
|
|
||||||
|
// Check for configuration of retention period for images and logs
|
||||||
|
_log('<<< mailGraph '.$cVersion.' >>>');
|
||||||
|
_log('# Invoked from CLI');
|
||||||
|
|
||||||
|
// Set defaults
|
||||||
|
$retImages = 30;
|
||||||
|
$retLogs = 14;
|
||||||
|
|
||||||
|
// Check if configured settings
|
||||||
|
if (isset($config['retention_images'])) { $retImages = intval($config['retention_images']); }
|
||||||
|
if (isset($config['retention_logs'])) { $retLogs = intval($config['retention_logs']); }
|
||||||
|
|
||||||
|
_log('Cleaning up IMAGES ('.$retImages.' days) and LOGS ('.$retLogs.' days)');
|
||||||
|
|
||||||
|
cleanupDir(getcwd().'/images', '.png', $retImages);
|
||||||
|
cleanupDir(getcwd().'/log', '.dump', $retLogs);
|
||||||
|
|
||||||
|
exit(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -433,10 +592,8 @@
|
|||||||
$cCRLF.json_encode($problemData,JSON_PRETTY_PRINT|JSON_NUMERIC_CHECK));
|
$cCRLF.json_encode($problemData,JSON_PRETTY_PRINT|JSON_NUMERIC_CHECK));
|
||||||
|
|
||||||
// --- CHECK AND SET P_ VARIABLES ---
|
// --- CHECK AND SET P_ VARIABLES ---
|
||||||
// FROM POST OR CLI DATA
|
|
||||||
|
|
||||||
if (!isset($problemData['itemId'])) { echo "Missing ITEM ID?\n"; die; }
|
// FROM POST OR CLI DATA
|
||||||
$p_itemId = intval($problemData['itemId']);
|
|
||||||
|
|
||||||
if (!isset($problemData['eventId'])) { echo "Missing EVENT ID?\n"; die; }
|
if (!isset($problemData['eventId'])) { echo "Missing EVENT ID?\n"; die; }
|
||||||
$p_eventId = intval($problemData['eventId']);
|
$p_eventId = intval($problemData['eventId']);
|
||||||
@@ -444,8 +601,8 @@
|
|||||||
if (!isset($problemData['recipient'])) { echo "Missing RECIPIENT?\n"; die; }
|
if (!isset($problemData['recipient'])) { echo "Missing RECIPIENT?\n"; die; }
|
||||||
$p_recipient = $problemData['recipient'];
|
$p_recipient = $problemData['recipient'];
|
||||||
|
|
||||||
if (!isset($problemData['duration'])) { echo "Missing DURATION?\n"; die; }
|
$p_duration = 0;
|
||||||
$p_duration = intval($problemData['duration']);
|
if (isset($problemData['duration'])) { $p_duration = intval($problemData['duration']); }
|
||||||
|
|
||||||
if (!isset($problemData['baseURL'])) { echo "Missing URL?\n"; die; }
|
if (!isset($problemData['baseURL'])) { echo "Missing URL?\n"; die; }
|
||||||
$p_URL = $problemData['baseURL'];
|
$p_URL = $problemData['baseURL'];
|
||||||
@@ -465,7 +622,7 @@
|
|||||||
$p_period = '48h';
|
$p_period = '48h';
|
||||||
if (isset($problemData['period'])) { $p_period = $problemData['period']; }
|
if (isset($problemData['period'])) { $p_period = $problemData['period']; }
|
||||||
|
|
||||||
if (isset($problemData['HTTPProxy'])) { $HTTPProxy = $problemData['HTTPPRoxy']; }
|
if (isset($problemData['HTTPProxy'])) { $HTTPProxy = $problemData['HTTPProxy']; }
|
||||||
|
|
||||||
// DYNAMIC VARIABLES FROM ZABBIX
|
// DYNAMIC VARIABLES FROM ZABBIX
|
||||||
|
|
||||||
@@ -482,19 +639,40 @@
|
|||||||
$p_smtp_port = 25;
|
$p_smtp_port = 25;
|
||||||
if (isset($config['smtp_port'])) { $p_smtp_port = $config['smtp_port']; }
|
if (isset($config['smtp_port'])) { $p_smtp_port = $config['smtp_port']; }
|
||||||
|
|
||||||
$p_smtp_transport = 'none';
|
|
||||||
if ((isset($config['smtp_transport'])) && ($config['smtp_transport']=='tls')) { $p_smtp_transport = 'tls'; }
|
|
||||||
if ((isset($config['smtp_transport'])) && ($config['smtp_transport']=='ssl')) { $p_smtp_transport = 'ssl'; }
|
|
||||||
|
|
||||||
$p_smtp_strict = 'yes';
|
$p_smtp_strict = 'yes';
|
||||||
if ((isset($config['smtp_strict'])) && ($config['smtp_strict']=='no')) { $p_smtp_strict = 'no'; }
|
if ((isset($config['smtp_strict'])) && ($config['smtp_strict']=='no')) { $p_smtp_strict = 'no'; }
|
||||||
|
|
||||||
|
$p_smtp_security = 'none';
|
||||||
|
if ((isset($config['smtp_security'])) && ($config['smtp_security']=='smtps')) { $p_smtp_security = 'smtps'; }
|
||||||
|
if ((isset($config['smtp_security'])) && ($config['smtp_security']=='starttls')) { $p_smtp_security = 'starttls'; }
|
||||||
|
|
||||||
$p_smtp_username = '';
|
$p_smtp_username = '';
|
||||||
if (isset($config['smtp_username'])) { $p_smtp_username = $config['smtp_username']; }
|
if (isset($config['smtp_username'])) { $p_smtp_username = $config['smtp_username']; }
|
||||||
|
|
||||||
$p_smtp_password = '';
|
$p_smtp_password = '';
|
||||||
if (isset($config['smtp_password'])) { $p_smtp_password = $config['smtp_password']; }
|
if (isset($config['smtp_password'])) { $p_smtp_password = $config['smtp_password']; }
|
||||||
|
|
||||||
|
$p_smtp_from_address = '';
|
||||||
|
if (isset($config['smtp_from_address'])) { $p_smtp_from_address = $config['smtp_from_address']; }
|
||||||
|
|
||||||
|
$p_smtp_from_name = 'mailGraph';
|
||||||
|
if (isset($config['smtp_from_name'])) { $p_smtp_from_name = $config['smtp_from_name']; }
|
||||||
|
|
||||||
|
$p_smtp_reply_address = '';
|
||||||
|
if (isset($config['smtp_reply_address'])) { $p_smtp_reply_address = $config['smtp_reply_address']; }
|
||||||
|
|
||||||
|
$p_smtp_reply_name = 'mailGraph feedback';
|
||||||
|
if (isset($config['smtp_reply_name'])) { $p_smtp_reply_name = $config['smtp_reply_name']; }
|
||||||
|
|
||||||
|
// >>> Backwards compatibility but smtp_from_address is leading (<v2.14)
|
||||||
|
$mailFrom = '';
|
||||||
|
if (isset($config['mail_from'])) { $mailFrom = $config['mail_from']; }
|
||||||
|
|
||||||
|
if (($p_smtp_from_address=='') && ($mailFrom!=''))
|
||||||
|
{
|
||||||
|
$p_smtp_from_address = $mailFrom;
|
||||||
|
}
|
||||||
|
|
||||||
$p_graph_match = 'any';
|
$p_graph_match = 'any';
|
||||||
if ((isset($config['graph_match'])) && ($config['graph_match']=='exact')) { $p_graph_match = 'exact'; }
|
if ((isset($config['graph_match'])) && ($config['graph_match']=='exact')) { $p_graph_match = 'exact'; }
|
||||||
|
|
||||||
@@ -511,17 +689,44 @@
|
|||||||
$z_tmp_cookies = $z_path.'tmp/';
|
$z_tmp_cookies = $z_path.'tmp/';
|
||||||
$z_log_path = $z_path.'log/';
|
$z_log_path = $z_path.'log/';
|
||||||
|
|
||||||
|
// If tmp, log, or images does not exist, create them
|
||||||
|
if (!is_dir($z_tmp_cookies))
|
||||||
|
{
|
||||||
|
mkdir($z_tmp_cookies);
|
||||||
|
_log('+ created TMP directory "'.$z_tmp_cookies.'"');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!is_dir($z_log_path))
|
||||||
|
{
|
||||||
|
mkdir($z_log_path);
|
||||||
|
_log('+ created LOG directory "'.$z_log_path.'"');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!is_dir($z_images_path))
|
||||||
|
{
|
||||||
|
mkdir($z_images_path);
|
||||||
|
_log('+ created IMAGES directory "'.$z_images_path.'"');
|
||||||
|
}
|
||||||
|
|
||||||
// Zabbix user (requires Super Admin access rights to access image generator script)
|
// Zabbix user (requires Super Admin access rights to access image generator script)
|
||||||
$z_user = $config['zabbix_user'];
|
$z_user = $config['zabbix_user'];
|
||||||
$z_pass = $config['zabbix_user_pwd'];
|
$z_pass = $config['zabbix_user_pwd'];
|
||||||
|
|
||||||
// Zabbix API user (requires Super Admin access rights)
|
// Zabbix API user (requires Super Admin access rights)
|
||||||
|
// --- Copy from Zabbix user and override when defined in configuration
|
||||||
// TODO: Check if information retreival can be done with less rigths
|
// TODO: Check if information retreival can be done with less rigths
|
||||||
$z_api_user = $config['zabbix_api_user'];
|
$z_api_user = $z_user;
|
||||||
$z_api_pass = $config['zabbix_api_pwd'];
|
$z_api_pass = $z_pass;
|
||||||
|
|
||||||
// Mail sender
|
if (isset($config['zabbix_api_user']))
|
||||||
$mailFrom = array($config['mail_from']=>'Zabbix Mailgraph');
|
{
|
||||||
|
$z_api_user = $config['zabbix_api_user'];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isset($config['zabbix_api_pwd']))
|
||||||
|
{
|
||||||
|
$z_api_pass = $config['zabbix_api_pwd'];
|
||||||
|
}
|
||||||
|
|
||||||
// Derived variables - do not change!
|
// Derived variables - do not change!
|
||||||
$z_server = $p_URL; // Zabbix server URL from config
|
$z_server = $p_URL; // Zabbix server URL from config
|
||||||
@@ -541,8 +746,6 @@
|
|||||||
// Fetch information via API ////////////////////////////////////////////////////////////////////////////
|
// Fetch information via API ////////////////////////////////////////////////////////////////////////////
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
$mailData = array();
|
|
||||||
|
|
||||||
$mailData['BASE_URL'] = $p_URL;
|
$mailData['BASE_URL'] = $p_URL;
|
||||||
$mailData['SUBJECT'] = $p_subject;
|
$mailData['SUBJECT'] = $p_subject;
|
||||||
|
|
||||||
@@ -554,7 +757,7 @@
|
|||||||
|
|
||||||
$request = array('jsonrpc'=>'2.0',
|
$request = array('jsonrpc'=>'2.0',
|
||||||
'method'=>'user.login',
|
'method'=>'user.login',
|
||||||
'params'=>array('user'=>$z_api_user,
|
'params'=>array('username'=>$z_api_user,
|
||||||
'password'=>$z_api_pass),
|
'password'=>$z_api_pass),
|
||||||
'id'=>nextRequestID(),
|
'id'=>nextRequestID(),
|
||||||
'auth'=>null);
|
'auth'=>null);
|
||||||
@@ -564,10 +767,79 @@
|
|||||||
$token = '';
|
$token = '';
|
||||||
if (isset($result['result'])) { $token = $result['result']; }
|
if (isset($result['result'])) { $token = $result['result']; }
|
||||||
|
|
||||||
if ($token=='') { echo 'Error logging in to Zabbix? ('.$z_url_api.')'; die; }
|
if ($token=='')
|
||||||
|
{
|
||||||
|
echo 'Error logging in to Zabbix? ('.$z_url_api.'): '.$cCRLF;
|
||||||
|
echo var_dump($request).$cCRLF;
|
||||||
|
echo var_dump($result).$cCRLF;
|
||||||
|
die;
|
||||||
|
}
|
||||||
|
|
||||||
_log('> Token = '.$token);
|
_log('> Token = '.$token);
|
||||||
|
|
||||||
|
// -----------------------
|
||||||
|
// --- LOG API VERSION ---
|
||||||
|
// -----------------------
|
||||||
|
|
||||||
|
_log('# Record Zabbix API version');
|
||||||
|
|
||||||
|
$request = array('jsonrpc'=>'2.0',
|
||||||
|
'method'=>'apiinfo.version',
|
||||||
|
'params'=>[],
|
||||||
|
'id'=>nextRequestID());
|
||||||
|
|
||||||
|
$result = postJSON($z_url_api, $request);
|
||||||
|
|
||||||
|
$apiVersion = $result['result'];
|
||||||
|
$apiVersionMajor = substr($apiVersion,0,01);
|
||||||
|
|
||||||
|
_log('> API version '.$apiVersion);
|
||||||
|
|
||||||
|
// -----------------------------------
|
||||||
|
// --- IF NO EVENT ID FETCH LATEST ---
|
||||||
|
// -----------------------------------
|
||||||
|
|
||||||
|
if ($p_eventId=="0")
|
||||||
|
{
|
||||||
|
_log('# No event ID given, picking up random event from Zabbix');
|
||||||
|
|
||||||
|
$request = array('jsonrpc'=>'2.0',
|
||||||
|
'method'=>'problem.get',
|
||||||
|
'params'=>array('output'=>'extend',
|
||||||
|
'recent'=>'true',
|
||||||
|
'limit'=>1),
|
||||||
|
'auth'=>$token,
|
||||||
|
'id'=>nextRequestID());
|
||||||
|
|
||||||
|
$thisProblems = postJSON($z_url_api, $request);
|
||||||
|
_log('> Problem data (recent)'.$cCRLF.json_encode($thisProblems,JSON_PRETTY_PRINT|JSON_NUMERIC_CHECK));
|
||||||
|
|
||||||
|
if (!isset($thisProblems['result'][0]))
|
||||||
|
{
|
||||||
|
_log('- No response data received. Retrying with less recent problems ... ');
|
||||||
|
|
||||||
|
$request = array('jsonrpc'=>'2.0',
|
||||||
|
'method'=>'problem.get',
|
||||||
|
'params'=>array('output'=>'extend',
|
||||||
|
'recent'=>'false',
|
||||||
|
'limit'=>1),
|
||||||
|
'auth'=>$token,
|
||||||
|
'id'=>nextRequestID());
|
||||||
|
|
||||||
|
$thisProblems = postJSON($z_url_api, $request);
|
||||||
|
_log('> Problem data (not recent)'.$cCRLF.json_encode($thisProblems,JSON_PRETTY_PRINT|JSON_NUMERIC_CHECK));
|
||||||
|
|
||||||
|
if (!isset($thisProblems['result'][0]))
|
||||||
|
{
|
||||||
|
_log('! Cannot continue: mailGraph is unable to pick a random event via the Zabbix API. It is highly likely that no active problems exist? Please retry or determine and set an event ID manually and retry.');
|
||||||
|
die;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$p_eventId = $thisProblems['result'][0]['eventid'];
|
||||||
|
_log('> Picked up random last event #'.$p_eventId);
|
||||||
|
}
|
||||||
|
|
||||||
// ------------------------------
|
// ------------------------------
|
||||||
// --- READ EVENT INFORMATION ---
|
// --- READ EVENT INFORMATION ---
|
||||||
// ------------------------------
|
// ------------------------------
|
||||||
@@ -579,7 +851,8 @@
|
|||||||
'params'=>array('eventids'=>$p_eventId,
|
'params'=>array('eventids'=>$p_eventId,
|
||||||
'output'=>'extend',
|
'output'=>'extend',
|
||||||
'selectRelatedObject'=>'extend',
|
'selectRelatedObject'=>'extend',
|
||||||
'selectSuppressionData'=>'extend'),
|
'selectSuppressionData'=>'extend',
|
||||||
|
'select_acknowledges'=>'extend'),
|
||||||
'auth'=>$token,
|
'auth'=>$token,
|
||||||
'id'=>nextRequestID());
|
'id'=>nextRequestID());
|
||||||
|
|
||||||
@@ -608,6 +881,17 @@
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// --- Collect and attach acknowledge messages for this event
|
||||||
|
if (sizeof($thisEvent['result'][0]['acknowledges']>0)) {
|
||||||
|
foreach($thisEvent['result'][0]['acknowledges'] as $aCount=>$anAck) {
|
||||||
|
$mailData['ACKNOWLEDGES'][$aCount] = $anAck;
|
||||||
|
$mailData['ACKNOWLEDGES'][$aCount]['_clock'] = zabbixTStoString($anAck['clock']);
|
||||||
|
$mailData['ACKNOWLEDGES'][$aCount]['_actions'] = zabbixActionToString($anAck['action']);
|
||||||
|
$mailData['ACKNOWLEDGES'][$aCount]['_old_severity'] = zabbixSeverityToString($anAck['old_severity']);
|
||||||
|
$mailData['ACKNOWLEDGES'][$aCount]['_new_severity'] = zabbixSeverityToString($anAck['new_severity']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$p_triggerId = $thisEvent['result'][0]['relatedObject']['triggerid'];
|
$p_triggerId = $thisEvent['result'][0]['relatedObject']['triggerid'];
|
||||||
|
|
||||||
// ------------------------
|
// ------------------------
|
||||||
@@ -621,10 +905,10 @@
|
|||||||
'params'=>array('triggerids'=>$p_triggerId,
|
'params'=>array('triggerids'=>$p_triggerId,
|
||||||
'output'=>'extend',
|
'output'=>'extend',
|
||||||
'selectFunctions'=>'extend',
|
'selectFunctions'=>'extend',
|
||||||
'selectTags'=>'extend'),
|
'selectTags'=>'extend',
|
||||||
'expandComment'=>1,
|
'expandComment'=>1,
|
||||||
'expandDescription'=>1,
|
'expandDescription'=>1,
|
||||||
'expandExpression'=>1,
|
'expandExpression'=>1),
|
||||||
'auth'=>$token,
|
'auth'=>$token,
|
||||||
'id'=>nextRequestID());
|
'id'=>nextRequestID());
|
||||||
|
|
||||||
@@ -694,18 +978,37 @@
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case 'mailGraph.screen':
|
case 'mailGraph.screen':
|
||||||
|
if ($apiVersionMajor<="5") {
|
||||||
$triggerScreen = intval($aTag['value']);
|
$triggerScreen = intval($aTag['value']);
|
||||||
_log('+ Trigger screen = '.$triggerScreen);
|
_log('+ Trigger screen = '.$triggerScreen);
|
||||||
|
} else {
|
||||||
|
_log('- Trigger screen value ignored');
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'mailGraph.screenPeriod':
|
case 'mailGraph.screenPeriod':
|
||||||
|
if ($apiVersionMajor<="5") {
|
||||||
$triggerScreenPeriod = $aTag['value'];
|
$triggerScreenPeriod = $aTag['value'];
|
||||||
_log('+ Trigger screen period = '.$triggerScreenPeriod);
|
_log('+ Trigger screen period = '.$triggerScreenPeriod);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'mailGraph.screenPeriodHeader':
|
case 'mailGraph.screenPeriodHeader':
|
||||||
|
if ($apiVersionMajor<="5") {
|
||||||
$triggerScreenPeriodHeader = $aTag['value'];
|
$triggerScreenPeriodHeader = $aTag['value'];
|
||||||
_log('+ Trigger screen header = '.$triggerScreenPeriodHeader);
|
_log('+ Trigger screen header = '.$triggerScreenPeriodHeader);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If no specific itemId is requested take the first item found on the items list from the host
|
||||||
|
if (!isset($p_itemId))
|
||||||
|
{
|
||||||
|
foreach($thisTrigger['result'][0]['functions'] as $aFunction)
|
||||||
|
{
|
||||||
|
$p_itemId = $aFunction['itemid'];
|
||||||
|
_log('- Item ID taken from trigger (first) function = '.$p_itemId);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -719,6 +1022,7 @@
|
|||||||
$request = array('jsonrpc'=>'2.0',
|
$request = array('jsonrpc'=>'2.0',
|
||||||
'method'=>'item.get',
|
'method'=>'item.get',
|
||||||
'params'=>array('itemids'=>$p_itemId,
|
'params'=>array('itemids'=>$p_itemId,
|
||||||
|
'webitems'=>'true',
|
||||||
'output'=>'extend'),
|
'output'=>'extend'),
|
||||||
'auth'=>$token,
|
'auth'=>$token,
|
||||||
'id'=>nextRequestID());
|
'id'=>nextRequestID());
|
||||||
@@ -766,7 +1070,7 @@
|
|||||||
|
|
||||||
$mailData['HOST_ID'] = $thisHost['result'][0]['hostid'];
|
$mailData['HOST_ID'] = $thisHost['result'][0]['hostid'];
|
||||||
$mailData['HOST_NAME'] = $thisHost['result'][0]['name'];
|
$mailData['HOST_NAME'] = $thisHost['result'][0]['name'];
|
||||||
$mailData['HOST_ERROR'] = $thisHost['result'][0]['error'];
|
if (isset($thisHost['result'][0]['error'])) { $mailData['HOST_ERROR'] = $thisHost['result'][0]['error']; }
|
||||||
$mailData['HOST_DESCRIPTION'] = $thisHost['result'][0]['description'];
|
$mailData['HOST_DESCRIPTION'] = $thisHost['result'][0]['description'];
|
||||||
|
|
||||||
// --- Custom settings?
|
// --- Custom settings?
|
||||||
@@ -1011,6 +1315,7 @@
|
|||||||
_log('> Graphs found = '.sizeof($triggerGraphs));
|
_log('> Graphs found = '.sizeof($triggerGraphs));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$hostGraphs = array();
|
||||||
|
|
||||||
if ($hostScreen>0)
|
if ($hostScreen>0)
|
||||||
{
|
{
|
||||||
@@ -1062,7 +1367,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Strip off any excessive elements from the end
|
// Strip off any excessive elements from the end (protection of graph generation overload on system)
|
||||||
|
|
||||||
while (sizeof($p_periods)>$maxGraphs) { array_pop($p_periods); }
|
while (sizeof($p_periods)>$maxGraphs) { array_pop($p_periods); }
|
||||||
while (sizeof($p_periods_headers)>$maxGraphs) { array_pop($p_periods_headers); }
|
while (sizeof($p_periods_headers)>$maxGraphs) { array_pop($p_periods_headers); }
|
||||||
@@ -1078,7 +1383,7 @@
|
|||||||
{
|
{
|
||||||
if ($forceGraph>0)
|
if ($forceGraph>0)
|
||||||
{
|
{
|
||||||
$theGraph = $forceGraphInfo;
|
$theGraph = $forceGraphInfo['result'][0];
|
||||||
$theType = 'Forced';
|
$theType = 'Forced';
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -1114,6 +1419,7 @@
|
|||||||
$graphFiles[] = $graphFile;
|
$graphFiles[] = $graphFile;
|
||||||
|
|
||||||
$mailData['GRAPHS_I'][$aKey]['PATH'] = $z_images_path . $graphFile;
|
$mailData['GRAPHS_I'][$aKey]['PATH'] = $z_images_path . $graphFile;
|
||||||
|
$mailData['GRAPHS_I'][$aKey]['CID'] = 'images/'.$graphFile;
|
||||||
$mailData['GRAPHS_I'][$aKey]['URL'] = $z_url_image . $graphFile;
|
$mailData['GRAPHS_I'][$aKey]['URL'] = $z_url_image . $graphFile;
|
||||||
$mailData['GRAPHS_I'][$aKey]['HEADER'] = $p_periods_headers[$aKey];
|
$mailData['GRAPHS_I'][$aKey]['HEADER'] = $p_periods_headers[$aKey];
|
||||||
}
|
}
|
||||||
@@ -1144,6 +1450,7 @@
|
|||||||
|
|
||||||
$mailData['GRAPHS_'.$varName][$aKey]['URL'] = $z_url_image . $graphFile;
|
$mailData['GRAPHS_'.$varName][$aKey]['URL'] = $z_url_image . $graphFile;
|
||||||
$mailData['GRAPHS_'.$varName][$aKey]['PATH'] = $z_images_path . $graphFile;
|
$mailData['GRAPHS_'.$varName][$aKey]['PATH'] = $z_images_path . $graphFile;
|
||||||
|
$mailData['GRAPHS_'.$varName][$aKey]['CID'] = 'images/'.$graphFile;
|
||||||
}
|
}
|
||||||
|
|
||||||
$mailData['GRAPHS_'.$varName.'_LINK'] = $z_server.'screens.php?elementid='.$info[0]['screen']['screenid'];
|
$mailData['GRAPHS_'.$varName.'_LINK'] = $z_server.'screens.php?elementid='.$info[0]['screen']['screenid'];
|
||||||
@@ -1190,7 +1497,10 @@
|
|||||||
$mailData['LOG_HTML'] = str_replace($cCRLF,'<br/>',$mailData['LOG_HTML']);
|
$mailData['LOG_HTML'] = str_replace($cCRLF,'<br/>',$mailData['LOG_HTML']);
|
||||||
$mailData['LOG_HTML'] = str_replace('<br/>','<br/>'.$cCRLF,$mailData['LOG_HTML']);
|
$mailData['LOG_HTML'] = str_replace('<br/>','<br/>'.$cCRLF,$mailData['LOG_HTML']);
|
||||||
|
|
||||||
$mailData['LOG_HTML'] = '<html lang="en"><head><meta http-equiv=Content-Type content="text/html; charset=UTF-8">'.$cCRLF.
|
$mailData['LOG_HTML'] = '<html lang="en"><head><meta http-equiv=Content-Type content="text/html; charset=UTF-8"></head>'.$cCRLF.
|
||||||
|
'<style type="text/css">'.$cCRLF.
|
||||||
|
'body { font-family: courier, courier new, serif; font-size: 12px; }'.$cCRLF.
|
||||||
|
'</style>'.$cCRLF.
|
||||||
'<body>'.$cCRLF.
|
'<body>'.$cCRLF.
|
||||||
$mailData['LOG_HTML'].$cCRLF.
|
$mailData['LOG_HTML'].$cCRLF.
|
||||||
'</body>'.$cCRLF.
|
'</body>'.$cCRLF.
|
||||||
@@ -1202,81 +1512,105 @@
|
|||||||
|
|
||||||
// Prepare others
|
// Prepare others
|
||||||
|
|
||||||
$mailData['TRIGGER_URL'] = $z_server.'triggers.php?form=update&triggerid='.$mailData['TRIGGER_ID'];
|
$mailData['TRIGGER_URL'] = $z_server.'triggers.php?form=update&triggerid='.$mailData['TRIGGER_ID'].'&context=host';
|
||||||
$mailData['ITEM_URL'] = $z_server.'items.php?form=update&hostid='.$mailData['HOST_ID'].'&itemid='.$mailData['ITEM_ID'];
|
$mailData['ITEM_URL'] = $z_server.'items.php?form=update&hostid='.$mailData['HOST_ID'].'&itemid='.$mailData['ITEM_ID'].'&context=host';
|
||||||
$mailData['HOST_URL'] = $z_server.'hosts.php?form=update&hostid='.$mailData['HOST_ID'];
|
$mailData['HOST_URL'] = $z_server.'hosts.php?form=update&hostid='.$mailData['HOST_ID'];
|
||||||
|
$mailData['ACK_URL'] = $z_server.'zabbix.php?action=popup&popup_action=acknowledge.edit&eventids[]='.$mailData['EVENT_ID'];
|
||||||
$mailData['EVENTDETAILS_URL'] = $z_server.'tr_events.php?triggerid='.$mailData['TRIGGER_ID'].'&eventid='.$mailData['EVENT_ID'];
|
$mailData['EVENTDETAILS_URL'] = $z_server.'tr_events.php?triggerid='.$mailData['TRIGGER_ID'].'&eventid='.$mailData['EVENT_ID'];
|
||||||
|
|
||||||
$mailData['EVENT_DURATION'] = $p_duration;
|
$mailData['EVENT_DURATION'] = $p_duration;
|
||||||
|
$mailData['HOST_PROBLEMS_URL'] = $z_server.'zabbix.php?show=1&name=&inventory%5B0%5D%5Bfield%5D=type&inventory%5B0%5D%5Bvalue%5D=&evaltype=0&tags%5B0%5D%5Btag%5D=&tags%5B0%5D%5Boperator%5D=0&tags%5B0%5D%5Bvalue%5D=&show_tags=3&tag_name_format=0&tag_priority=&show_opdata=0&show_timeline=1&filter_name=&filter_show_counter=0&filter_custom_time=0&sort=clock&sortorder=DESC&age_state=0&show_suppressed=0&unacknowledged=0&compact_view=0&details=0&highlight_row=0&action=problem.view&hostids%5B%5D='.$mailData['HOST_ID'];
|
||||||
|
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// Compose & Send Message ///////////////////////////////////////////////////////////////////////////////
|
// Compose & Send Message ///////////////////////////////////////////////////////////////////////////////
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
_log('# Setting up mailer');
|
_log('# Configuring Mailer');
|
||||||
|
|
||||||
// Do we need TLS or SSL?
|
$mail = new PHPMailer(true);
|
||||||
|
|
||||||
if (($p_smtp_transport=='tls') || ($p_smtp_transport=='ssl'))
|
try
|
||||||
{
|
{
|
||||||
$transport = (new Swift_SmtpTransport($p_smtp_server, $p_smtp_port, $p_smtp_transport));
|
// If debugging is required change to '1'
|
||||||
|
$mail->SMTPDebug = 0;
|
||||||
|
|
||||||
|
// Initialize for international characters
|
||||||
|
$mail->CharSet = "UTF-8";
|
||||||
|
$mail->Encoding = "base64";
|
||||||
|
|
||||||
|
// --- Inialize SMTP parameters
|
||||||
|
$mail->isSMTP();
|
||||||
|
$mail->Host = $p_smtp_server;
|
||||||
|
$mail->Port = $p_smtp_port;
|
||||||
|
|
||||||
|
// --- Initialize transport security
|
||||||
|
switch($p_smtp_security) {
|
||||||
|
case 'smtps':
|
||||||
|
_log('> Using SMTPS transport encryption method');
|
||||||
|
$mail->SMTPSecure = PHPMailer::ENCRYPTION_SMTPS;
|
||||||
|
break;
|
||||||
|
case 'starttls':
|
||||||
|
_log('> Using STARTTLS transport encryption method');
|
||||||
|
$mail->SMTPSecure = PHPMailer::ENCRYPTION_STARTTLS;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
_log('> Plain transport (no encryption)');
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- Disable strict certificate checking?
|
||||||
if ($p_smtp_strict=='no')
|
if ($p_smtp_strict=='no')
|
||||||
{
|
{
|
||||||
if ($transport instanceof \Swift_Transport_EsmtpTransport)
|
_log('> No strict TLS checking');
|
||||||
|
$mail->SMTPOptions = [
|
||||||
|
'ssl' => [ 'verify_peer' => false, 'verify_peer_name' => false, 'allow_self_signed' => true ]
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- Authentication required?
|
||||||
|
if ($p_smtp_username!="")
|
||||||
{
|
{
|
||||||
$transport->setStreamOptions([
|
$mail->SMTPAuth = true;
|
||||||
'ssl' => ['allow_self_signed' => true,
|
$mail->Username = $p_smtp_username;
|
||||||
'verify_peer' => false,
|
$mail->Password = $p_smtp_password;
|
||||||
'verify_peer_name' => false]
|
|
||||||
]);
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
else
|
// --- Define from
|
||||||
|
$mail->Sender = $p_smtp_from_address;
|
||||||
|
$mail->SetFrom($p_smtp_from_address, $p_smtp_from_name, FALSE);
|
||||||
|
|
||||||
|
// --- Define reply-to
|
||||||
|
if ($p_smtp_reply_address!='')
|
||||||
{
|
{
|
||||||
if ($transport instanceof \Swift_Transport_EsmtpTransport)
|
$mail->clearReplyTos();
|
||||||
|
$mail->addReplyTo($p_smtp_reply_address, $p_smtp_reply_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- Add recipient
|
||||||
|
$mail->addAddress($p_recipient);
|
||||||
|
|
||||||
|
// --- Prepare embedding of the graphs by attaching and generating "cid" (content-id) information
|
||||||
|
function embedGraphs($graphs,$varName,$type)
|
||||||
{
|
{
|
||||||
$transport->setStreamOptions([
|
global $mail;
|
||||||
'ssl' => ['allow_self_signed' => false,
|
global $mailData;
|
||||||
'verify_peer' => true,
|
|
||||||
'verify_peer_name' => true]
|
foreach($graphs as $aKey=>$anItem)
|
||||||
]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
{
|
||||||
$transport = (new Swift_SmtpTransport($p_smtp_server, $p_smtp_port));
|
$mail->AddEmbeddedImage($mailData['GRAPHS_'.$varName][$aKey]['PATH'],
|
||||||
|
$mailData['GRAPHS_'.$varName][$aKey]['CID']);
|
||||||
|
|
||||||
|
// Add content-id marker to the identifier for ease of use in Twig
|
||||||
|
$mailData['GRAPHS_'.$varName][$aKey]['CID'] = 'cid:'.$mailData['GRAPHS_'.$varName][$aKey]['CID'];
|
||||||
|
|
||||||
|
_log('> Embedded graph image ('.$type.') '.$mailData['GRAPHS_'.$varName][$aKey]['URL']);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Username/password?
|
embedGraphs($graphFiles,'I','ITEM');
|
||||||
|
embedGraphs($triggerGraphs,'T','TRIGGER');
|
||||||
if ($p_smtp_username!='') { $transport->setUsername($p_smtp_username); }
|
embedGraphs($hostGraphs,'H','HOST');
|
||||||
if ($p_smtp_password!='') { $transport->setPassword($p_smtp_password); }
|
|
||||||
|
|
||||||
// Start actual mail(er)
|
|
||||||
|
|
||||||
$mailer = new Swift_Mailer($transport);
|
|
||||||
|
|
||||||
$message = (new Swift_Message());
|
|
||||||
|
|
||||||
// --- Fetch mailer ID from this message (no Swift function available for it ...)
|
|
||||||
// --- "Message-ID: <...id...@swift.generated>"
|
|
||||||
|
|
||||||
$content = $message->toString();
|
|
||||||
$lines = explode(chr(10),$content);
|
|
||||||
$firstLine = $lines[0];
|
|
||||||
$idParts = explode(' ',$firstLine);
|
|
||||||
$messageId = substr($idParts[1],1,-2);
|
|
||||||
|
|
||||||
_log('# Message ID = '.$messageId);
|
|
||||||
|
|
||||||
// Build parts for HTML and PLAIN
|
|
||||||
|
|
||||||
_log('# Processing templates');
|
|
||||||
_log('+ '.$z_template_path.'html.template');
|
|
||||||
_log('+ '.$z_template_path.'plain.template');
|
|
||||||
|
|
||||||
|
// --- Render the content with TWIG for HTML, Plain text and the Subject of the message
|
||||||
$loader = new \Twig\Loader\ArrayLoader([
|
$loader = new \Twig\Loader\ArrayLoader([
|
||||||
'html' => file_get_contents($z_template_path.'html.template'),
|
'html' => file_get_contents($z_template_path.'html.template'),
|
||||||
'plain' => file_get_contents($z_template_path.'plain.template'),
|
'plain' => file_get_contents($z_template_path.'plain.template'),
|
||||||
@@ -1285,57 +1619,53 @@
|
|||||||
|
|
||||||
$twig = new \Twig\Environment($loader);
|
$twig = new \Twig\Environment($loader);
|
||||||
|
|
||||||
// --- Embed the images
|
|
||||||
|
|
||||||
function embedGraphs($graphs,$varName,$type)
|
|
||||||
{
|
|
||||||
global $message;
|
|
||||||
global $mailData;
|
|
||||||
|
|
||||||
foreach($graphs as $aKey=>$anItem)
|
|
||||||
{
|
|
||||||
$mailData['GRAPHS_'.$varName][$aKey]['CID'] = $message->embed(Swift_Image::fromPath($mailData['GRAPHS_'.$varName][$aKey]['PATH']));
|
|
||||||
_log('> Embedded graph image ('.$type.') '.$mailData['GRAPHS_'.$varName][$aKey]['PATH']);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
embedGraphs($graphFiles,'I','ITEM');
|
|
||||||
embedGraphs($triggerGraphs,'T','TRIGGER');
|
|
||||||
embedGraphs($hostGraphs,'H','HOST');
|
|
||||||
|
|
||||||
// --- Render the content
|
|
||||||
|
|
||||||
$bodyHTML = $twig->render('html', $mailData);
|
$bodyHTML = $twig->render('html', $mailData);
|
||||||
$bodyPlain = $twig->render('plain', $mailData);
|
$bodyPlain = $twig->render('plain', $mailData);
|
||||||
$mailSubject = $twig->render('subject', $mailData);
|
$mailSubject = $twig->render('subject', $mailData);
|
||||||
|
|
||||||
// Prepare message
|
// --- Attach debug log processing?
|
||||||
|
|
||||||
$message->setSubject($mailSubject)
|
|
||||||
->setFrom($mailFrom)
|
|
||||||
->setTo($p_recipient)
|
|
||||||
->setBody($bodyHTML, 'text/html')
|
|
||||||
->addPart($bodyPlain, 'text/plain');
|
|
||||||
|
|
||||||
if (($cDebugMail) || (isset($problemData['debug'])))
|
if (($cDebugMail) || (isset($problemData['debug'])))
|
||||||
{
|
{
|
||||||
_log('# Attaching logs to mail message');
|
_log('# Attaching logs to mail message');
|
||||||
|
|
||||||
$attachLogHTML = new Swift_Attachment($mailData['LOG_HTML'], 'log.html', 'text/html');
|
$mail->addStringAttachment($mailData['LOG_HTML'],'log.html');
|
||||||
$message->attach($attachLogHTML);
|
$mail->addStringAttachment($mailData['LOG_PLAIN'],'log.txt');
|
||||||
|
|
||||||
$attachLogPlain = new Swift_Attachment($mailData['LOG_PLAIN'], 'log.txt', '/text/plain');
|
|
||||||
$message->attach($attachLogPlain);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Send message
|
// ---Fill body and subject and mark as HTML while also supplying plain text option alternative
|
||||||
|
// Note: Not using PHPMailer option for automatic text/plain generation (by design)
|
||||||
|
$mail->Body = $bodyHTML;
|
||||||
|
$mail->isHTML(true);
|
||||||
|
$mail->AltBody = $bodyPlain;
|
||||||
|
$mail->Subject = $mailSubject;
|
||||||
|
|
||||||
$result = $mailer->send($message);
|
// --- Send the message
|
||||||
|
if (!$mail->send())
|
||||||
|
{
|
||||||
|
_log("! Failed to send mail message");
|
||||||
|
echo "! Failed to send mail message. Likely an issue with the recipient email address?".$cCRLF;
|
||||||
|
echo "+ Mailer error: ".$mail->ErrorInfo.$cCRLF;
|
||||||
|
}
|
||||||
|
|
||||||
// Return Event TAG information for Zabbix
|
// --- Obtain message ID
|
||||||
|
$messageId = $mail->getlastMessageID();
|
||||||
|
_log('# Message ID = '.$messageId);
|
||||||
|
|
||||||
$response = array('messageId.mailGraph'=>$messageId);
|
// --- Return Event TAG information for Zabbix to store with Zabbix event
|
||||||
|
$response = array('messageId'=>$messageId);
|
||||||
echo json_encode($response).$cCRLF;
|
echo json_encode($response).$cCRLF;
|
||||||
|
} catch (phpmailerException $e)
|
||||||
|
{
|
||||||
|
echo "! Failed to send message".$cCRLF;
|
||||||
|
echo "! phpMailer error message: ".$e->getMessage().$cCRLF;
|
||||||
|
_log("! phpMailer failed: ".$e->getMessage());
|
||||||
|
} catch (Exception $e)
|
||||||
|
{
|
||||||
|
echo "! Failed to send message".$cCRLF;
|
||||||
|
echo "! Error message: ".$e->getMessage().$cCRLF;
|
||||||
|
echo "! Check your mail server and/or transport settings!".$cCRLF;
|
||||||
|
_log("! Failed: ".$e->getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// Wrap up //////////////////////////////////////////////////////////////////////////////////////////////
|
// Wrap up //////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
@@ -1345,14 +1675,16 @@
|
|||||||
|
|
||||||
if (($cDebug) || (isset($problemData['debug'])))
|
if (($cDebug) || (isset($problemData['debug'])))
|
||||||
{
|
{
|
||||||
|
// Prevent duplicate dump of log information
|
||||||
unset($mailData['LOG_HTML']);
|
unset($mailData['LOG_HTML']);
|
||||||
unset($mailData['LOG_PLAIN']);
|
unset($mailData['LOG_PLAIN']);
|
||||||
|
|
||||||
|
// Attach the collected information
|
||||||
$content = implode(chr(10),$logging).$cCRLF.$cCRLF.'=== VALUES AVAILABLE FOR TWIG TEMPLATE ==='.$cCRLF.$cCRLF.json_encode($mailData,JSON_PRETTY_PRINT|JSON_NUMERIC_CHECK);
|
$content = implode(chr(10),$logging).$cCRLF.$cCRLF.'=== VALUES AVAILABLE FOR TWIG TEMPLATE ==='.$cCRLF.$cCRLF.json_encode($mailData,JSON_PRETTY_PRINT|JSON_NUMERIC_CHECK);
|
||||||
$content = str_replace(chr(13),'',$content);
|
$content = str_replace(chr(13),'',$content);
|
||||||
|
|
||||||
|
// Save to unique log file
|
||||||
$logName = 'log.'.$p_eventId.'.'.date('YmdHis').'.dump';
|
$logName = 'log.'.$p_eventId.'.'.date('YmdHis').'.dump';
|
||||||
|
|
||||||
file_put_contents($z_log_path.$logName,$content);
|
file_put_contents($z_log_path.$logName,$content);
|
||||||
_log('= Log stored to '.$z_log_path.$logName);
|
_log('= Log stored to '.$z_log_path.$logName);
|
||||||
}
|
}
|
||||||
|
|||||||
245
mailGraph.xml
245
mailGraph.xml
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<zabbix_export>
|
<zabbix_export>
|
||||||
<version>5.0</version>
|
<version>5.4</version>
|
||||||
<date>2021-03-17T12:55:43Z</date>
|
<date>2023-08-16T20:38:38Z</date>
|
||||||
<media_types>
|
<media_types>
|
||||||
<media_type>
|
<media_type>
|
||||||
<name>MailGraph</name>
|
<name>MailGraph</name>
|
||||||
@@ -9,7 +9,7 @@
|
|||||||
<parameters>
|
<parameters>
|
||||||
<parameter>
|
<parameter>
|
||||||
<name>baseURL</name>
|
<name>baseURL</name>
|
||||||
<value>https://mydomain.com/zabbix/</value>
|
<value>https://myzabbix.com/zabbix/</value>
|
||||||
</parameter>
|
</parameter>
|
||||||
<parameter>
|
<parameter>
|
||||||
<name>duration</name>
|
<name>duration</name>
|
||||||
@@ -35,10 +35,6 @@
|
|||||||
<name>infoTest</name>
|
<name>infoTest</name>
|
||||||
<value>Test</value>
|
<value>Test</value>
|
||||||
</parameter>
|
</parameter>
|
||||||
<parameter>
|
|
||||||
<name>itemId</name>
|
|
||||||
<value>{ITEM.ID}</value>
|
|
||||||
</parameter>
|
|
||||||
<parameter>
|
<parameter>
|
||||||
<name>periods</name>
|
<name>periods</name>
|
||||||
<value>10m,4h,1d,7d</value>
|
<value>10m,4h,1d,7d</value>
|
||||||
@@ -61,128 +57,157 @@
|
|||||||
</parameter>
|
</parameter>
|
||||||
<parameter>
|
<parameter>
|
||||||
<name>URL</name>
|
<name>URL</name>
|
||||||
<value>https://mydomain.com/mailGraph.php</value>
|
<value>https://myzabbix.com/mailGraph.php</value>
|
||||||
</parameter>
|
</parameter>
|
||||||
</parameters>
|
</parameters>
|
||||||
<script>try {
|
<script>// mailGraph v2.16
|
||||||
// Pickup parameters
|
|
||||||
params = JSON.parse(value),
|
// Function to test string
|
||||||
req = new CurlHttpRequest(),
|
function isJSON(str) {
|
||||||
fields = {},
|
try {
|
||||||
resp = '',
|
JSON.stringify(JSON.parse(str));
|
||||||
result = { tags: {} };
|
return true;
|
||||||
|
} catch (e) {
|
||||||
// Set HTTP proxy if required
|
return false;
|
||||||
if (typeof params.HTTPProxy === 'string' && params.HTTPProxy.trim() !== '') {
|
}
|
||||||
req.setProxy(params.HTTPProxy);
|
}
|
||||||
fields.HTTPProxy = params.HTTPProxy;
|
|
||||||
}
|
try {
|
||||||
|
// Pickup parameters
|
||||||
// Declare output type
|
params = JSON.parse(value),
|
||||||
req.AddHeader('Content-Type: application/json');
|
req = new HttpRequest(),
|
||||||
|
fields = {},
|
||||||
// Must have fields
|
resp = '',
|
||||||
fields.itemId = params.itemId;
|
result = { tags: {} };
|
||||||
fields.eventId = params.eventId;
|
|
||||||
fields.recipient = params.recipient;
|
// Set HTTP proxy if required
|
||||||
fields.baseURL = params.baseURL;
|
if (typeof params.HTTPProxy === 'string' && params.HTTPProxy.trim() !== '') {
|
||||||
fields.duration = params.duration;
|
req.setProxy(params.HTTPProxy);
|
||||||
|
fields.HTTPProxy = params.HTTPProxy;
|
||||||
// Optional fields
|
}
|
||||||
if (typeof params.graphWidth === 'string') { fields.graphWidth = params.graphWidth; }
|
|
||||||
if (typeof params.graphHeight === 'string') { fields.graphHeight = params.graphHeight; }
|
// Declare output type
|
||||||
if (typeof params.subject === 'string') { fields.subject = params.subject; }
|
req.addHeader('Content-Type: application/json');
|
||||||
if (typeof params.showLegend === 'string') { fields.showLegend = params.showLegend; }
|
|
||||||
if (typeof params.periods === 'string') { fields.periods = params.periods; }
|
// Pick up fields relevant for mailGraph API level call while parsing/casting fields that should be integer
|
||||||
if (typeof params.periods_headers === 'string') { fields.periods_headers = params.periods_headers; }
|
fields.itemId = params.itemId * 1;
|
||||||
if (typeof params.debug === 'string') { fields.debug = params.debug; }
|
fields.eventId = params.eventId * 1;
|
||||||
|
fields.recipient = params.recipient;
|
||||||
// Add generic fields
|
fields.baseURL = params.baseURL;
|
||||||
Object.keys(params).forEach(function(key) {
|
fields.duration = params.duration * 1;
|
||||||
if (key.substring(0, 4) == 'info') {
|
|
||||||
fields[key] = params[key];
|
if (fields.recipient.charAt(0) == '{') {
|
||||||
}
|
throw '[MailGraph Webhook] Please define recipient for the test message!';
|
||||||
});
|
}
|
||||||
|
|
||||||
// Post information to the processing script
|
// Optional fields
|
||||||
Zabbix.Log(4, '[MailGraph Webhook] Sending request: ' + params.URL + '?' + JSON.stringify(fields));
|
if (typeof params.graphWidth === 'string') { fields.graphWidth = params.graphWidth; }
|
||||||
var resp = req.Post(params.URL,JSON.stringify(fields));
|
if (typeof params.graphHeight === 'string') { fields.graphHeight = params.graphHeight; }
|
||||||
Zabbix.Log(4, '[Mailgraph Webhook] Receiving response:' + resp);
|
if (typeof params.subject === 'string') { fields.subject = params.subject; }
|
||||||
|
if (typeof params.showLegend === 'string') { fields.showLegend = params.showLegend; }
|
||||||
// If there was an error, report it
|
if (typeof params.periods === 'string') { fields.periods = params.periods; }
|
||||||
if (req.Status() != 200) { throw JSON.parse(resp).errors[0]; }
|
if (typeof params.periods_headers === 'string') { fields.periods_headers = params.periods_headers; }
|
||||||
|
if (typeof params.debug === 'string') { fields.debug = params.debug; }
|
||||||
// We expect the message id back from the processing script
|
|
||||||
resp = JSON.parse(resp);
|
// Add generic fields
|
||||||
result.tags.__message_id = resp.messageId;
|
Object.keys(params).forEach(function(key) {
|
||||||
|
if (key.substring(0, 4) == 'info') {
|
||||||
// Pass the result back to Zabbix
|
fields[key] = params[key];
|
||||||
return JSON.stringify(result);
|
}
|
||||||
}
|
});
|
||||||
catch (error)
|
|
||||||
{
|
// Post information to the processing script
|
||||||
// In case something went wrong in the processing, pass the error back to Zabbix
|
Zabbix.Log(4, '[MailGraph Webhook] Sending request: ' + params.URL + '?' + JSON.stringify(fields));
|
||||||
Zabbix.Log(127, 'MailGraph notification failed : '+error);
|
var resp = req.post(params.URL,JSON.stringify(fields));
|
||||||
throw 'MailGraph notification failed : '+error;
|
Zabbix.Log(4, '[Mailgraph Webhook] Received response:' + resp);
|
||||||
|
|
||||||
|
// The response can be
|
||||||
|
// - did not receive status 200 as result (contains HTTP server response)
|
||||||
|
// - null (no response received at all)
|
||||||
|
// - empty string (likely no e-mail sent due to recipient issue)
|
||||||
|
// - not json (debugging message for troubleshooting or configuration hints)
|
||||||
|
// - json (contains the mail message ID sent)
|
||||||
|
|
||||||
|
if (req.getStatus() != 200) {
|
||||||
|
throw '[MailGraph Webhook] Processing of mailGraph.php failed: ' + resp;
|
||||||
|
}
|
||||||
|
if (resp==null) {
|
||||||
|
throw '[MailGraph Webhook] No response received from mailGraph.php? This should not occur (check URL and your webserver!)';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (resp=='') {
|
||||||
|
throw '[MailGraph Webhook] No data received from mailGraph - please check recipient address or mailGraph log and retry.';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if JSON was returned
|
||||||
|
if (!isJSON(resp)) {
|
||||||
|
throw '[MailGraph Webhook] An error has occurred during processing: ' + resp;
|
||||||
|
}
|
||||||
|
|
||||||
|
// We expect the message id back from the processing script response in JSON format
|
||||||
|
msg = JSON.parse(resp);
|
||||||
|
|
||||||
|
result.tags.__message_id = msg.messageId;
|
||||||
|
Zabbix.Log(4, '[MailGraph Webhook] Message sent with identification "' + msg.messageId + '"');
|
||||||
|
|
||||||
|
// Pass the result back to Zabbix
|
||||||
|
return JSON.stringify(result);
|
||||||
|
}
|
||||||
|
catch (error)
|
||||||
|
{
|
||||||
|
// In case something else went wrong in the processing, pass the error back to Zabbix
|
||||||
|
Zabbix.Log(127, 'MailGraph notification failed: '+error);
|
||||||
|
throw 'MailGraph notification failed : '+error;
|
||||||
}</script>
|
}</script>
|
||||||
<process_tags>YES</process_tags>
|
<process_tags>YES</process_tags>
|
||||||
<description>The "URL" must point to the location of the processing script. If a proxy is required, define "HTTPProxy" for the proxy address.
|
<description>The "URL" must point to the location of the processing script. If a proxy is required, define "HTTPProxy" for the proxy address.
|
||||||
|
|
||||||
Customization:
|
Customization:
|
||||||
- "graphWidth" and "graphWidth" can be defined for the image size
|
- "graphWidth" and "graphWidth" can be defined for the image size
|
||||||
- "showLegend" can be defined to show or hide the legend of the graph
|
- "showLegend" can be defined to show or hide the legend of the graph
|
||||||
- "subject" can be defined for a customized subject for the mail message
|
- "subject" can be defined for a customized subject for the mail message
|
||||||
- "periods" and "periods_headers" can be defined for displaying multiple periods of the same graph, or
|
- "periods" and "periods_headers" can be defined for displaying multiple periods of the same graph, or
|
||||||
- "period" and "period_header" can be defined to display a single graph
|
- "period" and "period_header" can be defined to display a single graph
|
||||||
|
|
||||||
The html.template and plain.template files can be adjusted (TWIG format).
|
The html.template and plain.template files can be adjusted (TWIG format).
|
||||||
|
|
||||||
More details are available at https://github.com/moudsen/mailGraph</description>
|
More details are available at https://github.com/moudsen/mailGraph</description>
|
||||||
<message_templates>
|
<message_templates>
|
||||||
<message_template>
|
<message_template>
|
||||||
<event_source>TRIGGERS</event_source>
|
<event_source>TRIGGERS</event_source>
|
||||||
<operation_mode>PROBLEM</operation_mode>
|
<operation_mode>PROBLEM</operation_mode>
|
||||||
<subject>Problem: {EVENT.NAME}</subject>
|
<subject>Problem: {EVENT.NAME}</subject>
|
||||||
<message>Problem started at {EVENT.TIME} on {EVENT.DATE}
|
<message>Problem started at {EVENT.TIME} on {EVENT.DATE}
|
||||||
Problem name: {EVENT.NAME}
|
Problem name: {EVENT.NAME}
|
||||||
Host: {HOST.NAME}
|
Host: {HOST.NAME}
|
||||||
Severity: {EVENT.SEVERITY}
|
Severity: {EVENT.SEVERITY}
|
||||||
Operational data: {EVENT.OPDATA}
|
Operational data: {EVENT.OPDATA}
|
||||||
Original problem ID: {EVENT.ID}
|
Original problem ID: {EVENT.ID}
|
||||||
{TRIGGER.URL}
|
Trigger ID (/url): {TRIGGER.ID} {TRIGGER.URL}</message>
|
||||||
|
|
||||||
eventId: {EVENT.ID}
|
|
||||||
TriggerId: {TRIGGER.ID}
|
|
||||||
itemId: {ITEM.ID]</message>
|
|
||||||
</message_template>
|
</message_template>
|
||||||
<message_template>
|
<message_template>
|
||||||
<event_source>TRIGGERS</event_source>
|
<event_source>TRIGGERS</event_source>
|
||||||
<operation_mode>RECOVERY</operation_mode>
|
<operation_mode>RECOVERY</operation_mode>
|
||||||
<subject>Resolved in {EVENT.DURATION}: {EVENT.NAME}</subject>
|
<subject>Resolved in {EVENT.DURATION}: {EVENT.RECOVERY.NAME}</subject>
|
||||||
<message>Problem has been resolved at {EVENT.RECOVERY.TIME} on {EVENT.RECOVERY.DATE}
|
<message>Problem has been resolved at {EVENT.RECOVERY.TIME} on {EVENT.RECOVERY.DATE}
|
||||||
Problem name: {EVENT.NAME}
|
Problem name: {EVENT.RECOVERY.NAME}
|
||||||
Problem duration: {EVENT.DURATION}
|
Problem duration: {EVENT.DURATION}
|
||||||
Host: {HOST.NAME}
|
Host: {HOST.NAME}
|
||||||
Severity: {EVENT.SEVERITY}
|
Severity: {EVENT.SEVERITY}
|
||||||
Original problem ID: {EVENT.ID}
|
Original problem ID: {EVENT.ID}
|
||||||
{TRIGGER.URL}
|
Trigger ID: {TRIGGER.ID} {TRIGGER.URL}</message>
|
||||||
|
|
||||||
eventId: {EVENT.ID}
|
|
||||||
TriggerId: {TRIGGER.ID}
|
|
||||||
itemId: {ITEM.ID]</message>
|
|
||||||
</message_template>
|
</message_template>
|
||||||
<message_template>
|
<message_template>
|
||||||
<event_source>TRIGGERS</event_source>
|
<event_source>TRIGGERS</event_source>
|
||||||
<operation_mode>UPDATE</operation_mode>
|
<operation_mode>UPDATE</operation_mode>
|
||||||
<subject>Updated problem in {EVENT.AGE}: {EVENT.NAME}</subject>
|
<subject>Updated problem in {EVENT.AGE}: {EVENT.NAME}</subject>
|
||||||
<message>{USER.FULLNAME} {EVENT.UPDATE.ACTION} problem at {EVENT.UPDATE.DATE} {EVENT.UPDATE.TIME}.
|
<message>{USER.FULLNAME} {EVENT.UPDATE.ACTION} problem at {EVENT.UPDATE.DATE} {EVENT.UPDATE.TIME}.
|
||||||
{EVENT.UPDATE.MESSAGE}
|
{EVENT.UPDATE.MESSAGE}
|
||||||
|
|
||||||
Current problem status is {EVENT.STATUS}, age is {EVENT.AGE}, acknowledged: {EVENT.ACK.STATUS}.
|
Current problem status is {EVENT.STATUS}, age is {EVENT.AGE}, acknowledged: {EVENT.ACK.STATUS}.
|
||||||
|
|
||||||
eventId: {EVENT.ID}
|
Event ID: {EVENT.ID}
|
||||||
TriggerId: {TRIGGER.ID}
|
Trigger ID: {TRIGGER.ID}</message>
|
||||||
itemId: {ITEM.ID]</message>
|
|
||||||
</message_template>
|
</message_template>
|
||||||
</message_templates>
|
</message_templates>
|
||||||
</media_type>
|
</media_type>
|
||||||
|
|||||||
@@ -81,6 +81,9 @@
|
|||||||
.content {
|
.content {
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
}
|
}
|
||||||
|
.acknowledge {
|
||||||
|
font-size: 10px;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
@@ -112,6 +115,18 @@
|
|||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
|
{% if ACKNOWLEDGES|length > 0 %}
|
||||||
|
<br/>
|
||||||
|
{% for anAck in ACKNOWLEDGES %}
|
||||||
|
<table style="border:0; text-align:left;" cellpadding="5" cellspacing="0">
|
||||||
|
<tr>
|
||||||
|
<td class="acknowledge">
|
||||||
|
<b>{{ anAck._clock }}</b><br/><em>({{ anAck.username }}, {{ anAck.name }} {{ anAck.surname }})</em><br/>[{{ anAck._actions }}]<br/>{{ anAck.message }}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
{% endfor %}
|
||||||
|
{% endif %}
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
@@ -120,10 +135,12 @@
|
|||||||
Event ID: <a href="{{ EVENTDETAILS_URL }}">{{ EVENT_ID }}</a> //
|
Event ID: <a href="{{ EVENTDETAILS_URL }}">{{ EVENT_ID }}</a> //
|
||||||
Trigger ID: <a href="{{ TRIGGER_URL }}">{{ TRIGGER_ID }}</a> //
|
Trigger ID: <a href="{{ TRIGGER_URL }}">{{ TRIGGER_ID }}</a> //
|
||||||
Item ID: <a href="{{ ITEM_URL }}">{{ ITEM_ID }}</a> //
|
Item ID: <a href="{{ ITEM_URL }}">{{ ITEM_ID }}</a> //
|
||||||
Host ID: <a href="{{ HOST_URL }}">{{ HOST_ID }}</a>
|
Host ID: <a href="{{ HOST_URL }}">{{ HOST_ID }}</a> //
|
||||||
|
<a href="{{ HOST_PROBLEMS_URL }}">Problems</a>
|
||||||
{% if GRAPH_ZABBIXLINK|length > 0 %}
|
{% if GRAPH_ZABBIXLINK|length > 0 %}
|
||||||
// Graph ID: <a href="{{ GRAPH_ZABBIXLINK }}">{{ GRAPH_ID }}</a>
|
// Graph ID: <a href="{{ GRAPH_ZABBIXLINK }}">{{ GRAPH_ID }}</a>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
// <a href="{{ ACK_URL }}">Ack</a>
|
||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|||||||
Reference in New Issue
Block a user