2023-03-07 09:57:42 +00:00
|
|
|
#!rsc by RouterOS
|
|
|
|
# RouterOS script: mod/notification-email
|
2024-01-01 14:25:25 +00:00
|
|
|
# Copyright (c) 2013-2024 Christian Hesse <mail@eworm.de>
|
2023-03-07 09:57:42 +00:00
|
|
|
# https://git.eworm.de/cgit/routeros-scripts/about/COPYING.md
|
2023-04-04 17:23:03 +00:00
|
|
|
#
|
2024-04-04 19:45:02 +00:00
|
|
|
# requires RouterOS, version=7.13
|
2023-10-06 21:37:54 +00:00
|
|
|
#
|
2023-04-04 17:23:03 +00:00
|
|
|
# send notifications via e-mail
|
|
|
|
# https://git.eworm.de/cgit/routeros-scripts/about/doc/mod/notification-email.md
|
2023-03-07 09:57:42 +00:00
|
|
|
|
2024-03-12 20:52:18 +00:00
|
|
|
:global EMailGenerateFrom;
|
2023-03-07 09:57:42 +00:00
|
|
|
:global FlushEmailQueue;
|
|
|
|
:global LogForwardFilterLogForwarding;
|
|
|
|
:global NotificationEMailSubject;
|
|
|
|
:global NotificationFunctions;
|
2023-08-06 14:25:06 +00:00
|
|
|
:global PurgeEMailQueue;
|
2023-03-07 09:57:42 +00:00
|
|
|
:global QuotedPrintable;
|
|
|
|
:global SendEMail;
|
|
|
|
:global SendEMail2;
|
|
|
|
|
2024-03-12 20:52:18 +00:00
|
|
|
# generate from-property with display name
|
|
|
|
:set EMailGenerateFrom do={
|
|
|
|
:global Identity;
|
|
|
|
|
2024-03-12 21:03:04 +00:00
|
|
|
:global CleanName;
|
|
|
|
|
2024-03-12 20:52:18 +00:00
|
|
|
:local From [ /tool/e-mail/get from ];
|
|
|
|
|
|
|
|
:if ($From ~ "<.*>\$") do={
|
|
|
|
:return $From;
|
|
|
|
}
|
|
|
|
|
2024-03-12 21:03:04 +00:00
|
|
|
:return ([ $CleanName $Identity ] . " via routeros-scripts <" . $From . ">");
|
2024-03-12 20:52:18 +00:00
|
|
|
}
|
|
|
|
|
2023-03-07 09:57:42 +00:00
|
|
|
# flush e-mail queue
|
|
|
|
:set FlushEmailQueue do={
|
|
|
|
:global EmailQueue;
|
|
|
|
|
|
|
|
:global EitherOr;
|
2024-03-12 20:52:18 +00:00
|
|
|
:global EMailGenerateFrom;
|
2023-03-07 09:57:42 +00:00
|
|
|
:global IsDNSResolving;
|
|
|
|
:global IsTimeSync;
|
2024-03-08 11:45:38 +00:00
|
|
|
:global LogPrint;
|
2023-03-07 09:57:42 +00:00
|
|
|
|
|
|
|
:local AllDone true;
|
|
|
|
:local QueueLen [ :len $EmailQueue ];
|
2023-11-14 21:34:45 +00:00
|
|
|
:local Scheduler [ /system/scheduler/find where name="_FlushEmailQueue" ];
|
2024-03-04 19:40:53 +00:00
|
|
|
|
2023-03-07 09:57:42 +00:00
|
|
|
:if ([ :len $Scheduler ] > 0 && [ /system/scheduler/get $Scheduler interval ] < 1m) do={
|
|
|
|
/system/scheduler/set interval=1m comment="Doing initial checks..." $Scheduler;
|
|
|
|
}
|
|
|
|
|
|
|
|
:if ([ /tool/e-mail/get last-status ] = "in-progress") do={
|
2024-03-08 11:45:38 +00:00
|
|
|
$LogPrint debug $0 ("Sending mail is currently in progress, not flushing.");
|
2023-03-07 09:57:42 +00:00
|
|
|
:return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
:if ([ $IsTimeSync ] = false) do={
|
2024-03-08 11:45:38 +00:00
|
|
|
$LogPrint debug $0 ("Time is not synced, not flushing.");
|
2023-03-07 09:57:42 +00:00
|
|
|
:return false;
|
|
|
|
}
|
|
|
|
|
2023-10-06 17:48:30 +00:00
|
|
|
:local EMailSettings [ /tool/e-mail/get ];
|
2023-10-06 21:37:54 +00:00
|
|
|
:if ([ :typeof [ :toip ($EMailSettings->"server") ] ] != "ip" && [ $IsDNSResolving ] = false) do={
|
2024-03-08 11:45:38 +00:00
|
|
|
$LogPrint debug $0 ("Server address is a DNS name and resolving fails, not flushing.");
|
2023-03-07 09:57:42 +00:00
|
|
|
:return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
:if ([ :len $Scheduler ] > 0 && $QueueLen = 0) do={
|
2024-03-08 11:45:38 +00:00
|
|
|
$LogPrint warning $0 ("Flushing E-Mail messages from scheduler, but queue is empty.");
|
2023-03-07 09:57:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/system/scheduler/set interval=([ $EitherOr $QueueLen 1 ] . "m") comment="Sending..." $Scheduler;
|
|
|
|
|
|
|
|
:foreach Id,Message in=$EmailQueue do={
|
|
|
|
:if ([ :typeof $Message ] = "array" ) do={
|
|
|
|
:local Attach ({});
|
|
|
|
:while ([ /tool/e-mail/get last-status ] = "in-progress") do={ :delay 1s; }
|
|
|
|
:foreach File in=[ :toarray [ $EitherOr ($Message->"attach") "" ] ] do={
|
|
|
|
:if ([ :len [ /file/find where name=$File ] ] = 1) do={
|
|
|
|
:set Attach ($Attach, $File);
|
|
|
|
} else={
|
2024-03-08 11:45:38 +00:00
|
|
|
$LogPrint warning $0 ("File '" . $File . "' does not exist, can not attach.");
|
2023-03-07 09:57:42 +00:00
|
|
|
}
|
|
|
|
}
|
2024-03-12 20:52:18 +00:00
|
|
|
/tool/e-mail/send from=[ $EMailGenerateFrom ] to=($Message->"to") cc=($Message->"cc") \
|
|
|
|
subject=($Message->"subject") body=($Message->"body") file=$Attach;
|
2023-03-07 09:57:42 +00:00
|
|
|
:local Wait true;
|
|
|
|
:do {
|
|
|
|
:delay 1s;
|
|
|
|
:local Status [ /tool/e-mail/get last-status ];
|
|
|
|
:if ($Status = "succeeded") do={
|
|
|
|
:set ($EmailQueue->$Id);
|
|
|
|
:set Wait false;
|
|
|
|
:if (($Message->"remove-attach") = true) do={
|
|
|
|
:foreach File in=$Attach do={
|
|
|
|
/file/remove $File;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
:if ($Status = "failed") do={
|
|
|
|
:set AllDone false;
|
|
|
|
:set Wait false;
|
|
|
|
}
|
|
|
|
} while=($Wait = true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
:if ($AllDone = true && $QueueLen = [ :len $EmailQueue ]) do={
|
|
|
|
/system/scheduler/remove $Scheduler;
|
|
|
|
:set EmailQueue;
|
|
|
|
} else={
|
|
|
|
/system/scheduler/set interval=1m comment="Waiting for retry..." $Scheduler;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# generate filter for log-forward
|
|
|
|
:set LogForwardFilterLogForwarding do={
|
|
|
|
:global EscapeForRegEx;
|
|
|
|
:global NotificationEMailSubject;
|
|
|
|
:global SymbolForNotification;
|
|
|
|
|
|
|
|
:return ("^Error sending e-mail <(" . \
|
|
|
|
[ $EscapeForRegEx [ $NotificationEMailSubject ([ $SymbolForNotification \
|
|
|
|
"memo" ] . "Log Forwarding") ] ] . "|" . \
|
|
|
|
[ $EscapeForRegEx [ $NotificationEMailSubject ([ $SymbolForNotification \
|
|
|
|
"warning-sign" ] . "Log Forwarding") ] ] . ")>:");
|
|
|
|
}
|
|
|
|
|
|
|
|
# generate the e-mail subject
|
|
|
|
:set NotificationEMailSubject do={
|
|
|
|
:global Identity;
|
|
|
|
:global IdentityExtra;
|
|
|
|
|
|
|
|
:global QuotedPrintable;
|
|
|
|
|
|
|
|
:return [ $QuotedPrintable ("[" . $IdentityExtra . $Identity . "] " . $1) ];
|
|
|
|
}
|
|
|
|
|
|
|
|
# send notification via e-mail - expects one array argument
|
|
|
|
:set ($NotificationFunctions->"email") do={
|
|
|
|
:local Notification $1;
|
|
|
|
|
|
|
|
:global EmailGeneralTo;
|
|
|
|
:global EmailGeneralToOverride;
|
|
|
|
:global EmailGeneralCc;
|
|
|
|
:global EmailGeneralCcOverride;
|
|
|
|
:global EmailQueue;
|
|
|
|
|
|
|
|
:global EitherOr;
|
|
|
|
:global IfThenElse;
|
2023-12-04 21:21:51 +00:00
|
|
|
:global NotificationEMailSignature;
|
2023-03-07 09:57:42 +00:00
|
|
|
:global NotificationEMailSubject;
|
|
|
|
|
|
|
|
:local To [ $EitherOr ($EmailGeneralToOverride->($Notification->"origin")) $EmailGeneralTo ];
|
|
|
|
:local Cc [ $EitherOr ($EmailGeneralCcOverride->($Notification->"origin")) $EmailGeneralCc ];
|
|
|
|
|
|
|
|
:local EMailSettings [ /tool/e-mail/get ];
|
2023-10-06 21:37:54 +00:00
|
|
|
:if ([ :len $To ] = 0 || ($EMailSettings->"server") = "0.0.0.0" || ($EMailSettings->"from") = "<>") do={
|
2023-03-07 09:57:42 +00:00
|
|
|
:return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
:if ([ :typeof $EmailQueue ] = "nothing") do={
|
|
|
|
:set EmailQueue ({});
|
|
|
|
}
|
2023-12-04 21:21:51 +00:00
|
|
|
:local Signature [ $EitherOr [ $NotificationEMailSignature ] [ /system/note/get note ] ];
|
2023-03-07 09:57:42 +00:00
|
|
|
:set ($EmailQueue->[ :len $EmailQueue ]) {
|
|
|
|
to=$To; cc=$Cc;
|
|
|
|
subject=[ $NotificationEMailSubject ($Notification->"subject") ];
|
|
|
|
body=(($Notification->"message") . \
|
|
|
|
[ $IfThenElse ([ :len ($Notification->"link") ] > 0) ("\n\n" . ($Notification->"link")) "" ] . \
|
|
|
|
[ $IfThenElse ([ :len $Signature ] > 0) ("\n-- \n" . $Signature) "" ]); \
|
|
|
|
attach=($Notification->"attach"); remove-attach=($Notification->"remove-attach") };
|
2023-11-14 21:34:45 +00:00
|
|
|
:if ([ :len [ /system/scheduler/find where name="_FlushEmailQueue" ] ] = 0) do={
|
|
|
|
/system/scheduler/add name="_FlushEmailQueue" interval=1s start-time=startup \
|
2023-03-07 09:57:42 +00:00
|
|
|
comment="Queuing new mail..." on-event=(":global FlushEmailQueue; \$FlushEmailQueue;");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-06 14:25:06 +00:00
|
|
|
# purge the e-mail queue
|
|
|
|
:set PurgeEMailQueue do={
|
|
|
|
:global EmailQueue;
|
|
|
|
|
2023-11-14 21:34:45 +00:00
|
|
|
/system/scheduler/remove [ find where name="_FlushEmailQueue" ];
|
2023-08-06 14:25:06 +00:00
|
|
|
:set EmailQueue;
|
|
|
|
}
|
|
|
|
|
2023-03-07 09:57:42 +00:00
|
|
|
# convert string to quoted-printable
|
|
|
|
:global QuotedPrintable do={
|
|
|
|
:local Input [ :tostr $1 ];
|
|
|
|
|
2023-12-04 08:44:32 +00:00
|
|
|
:global CharacterMultiply;
|
|
|
|
|
2023-03-07 09:57:42 +00:00
|
|
|
:if ([ :len $Input ] = 0) do={
|
|
|
|
:return $Input;
|
|
|
|
}
|
|
|
|
|
|
|
|
:local Return "";
|
2023-12-04 08:44:32 +00:00
|
|
|
:local Chars ( \
|
|
|
|
"\00\01\02\03\04\05\06\07\08\09\0A\0B\0C\0D\0E\0F\10\11\12\13\14\15\16\17\18\19\1A\1B\1C\1D\1E\1F" . \
|
2023-12-04 10:45:28 +00:00
|
|
|
[ $CharacterMultiply ("\00") 29 ] . "=\00?" . [ $CharacterMultiply ("\00") 63 ] . "\7F" . \
|
2023-12-04 08:44:32 +00:00
|
|
|
"\80\81\82\83\84\85\86\87\88\89\8A\8B\8C\8D\8E\8F\90\91\92\93\94\95\96\97\98\99\9A\9B\9C\9D\9E\9F" . \
|
|
|
|
"\A0\A1\A2\A3\A4\A5\A6\A7\A8\A9\AA\AB\AC\AD\AE\AF\B0\B1\B2\B3\B4\B5\B6\B7\B8\B9\BA\BB\BC\BD\BE\BF" . \
|
|
|
|
"\C0\C1\C2\C3\C4\C5\C6\C7\C8\C9\CA\CB\CC\CD\CE\CF\D0\D1\D2\D3\D4\D5\D6\D7\D8\D9\DA\DB\DC\DD\DE\DF" . \
|
|
|
|
"\E0\E1\E2\E3\E4\E5\E6\E7\E8\E9\EA\EB\EC\ED\EE\EF\F0\F1\F2\F3\F4\F5\F6\F7\F8\F9\FA\FB\FC\FD\FE\FF");
|
|
|
|
:local Hex "0123456789ABCDEF";
|
2023-03-07 09:57:42 +00:00
|
|
|
|
|
|
|
:for I from=0 to=([ :len $Input ] - 1) do={
|
|
|
|
:local Char [ :pick $Input $I ];
|
|
|
|
:local Replace [ :find $Chars $Char ];
|
|
|
|
|
|
|
|
:if ([ :typeof $Replace ] = "num") do={
|
2023-12-04 08:44:32 +00:00
|
|
|
:set Char ("=" . [ :pick $Hex ($Replace / 16)] . [ :pick $Hex ($Replace % 16) ]);
|
2023-03-07 09:57:42 +00:00
|
|
|
}
|
|
|
|
:set Return ($Return . $Char);
|
|
|
|
}
|
|
|
|
|
|
|
|
:if ($Input = $Return) do={
|
|
|
|
:return $Input;
|
|
|
|
}
|
|
|
|
|
2023-06-26 09:18:12 +00:00
|
|
|
:return ("=?utf-8?Q?" . $Return . "?=");
|
2023-03-07 09:57:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
# send notification via e-mail - expects at least two string arguments
|
|
|
|
:set SendEMail do={
|
|
|
|
:global SendEMail2;
|
|
|
|
|
2024-04-09 21:59:00 +00:00
|
|
|
$SendEMail2 ({ origin=$0; subject=$1; message=$2; link=$3 });
|
2023-03-07 09:57:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
# send notification via e-mail - expects one array argument
|
|
|
|
:set SendEMail2 do={
|
|
|
|
:local Notification $1;
|
|
|
|
|
|
|
|
:global NotificationFunctions;
|
|
|
|
|
|
|
|
($NotificationFunctions->"email") ("\$NotificationFunctions->\"email\"") $Notification;
|
|
|
|
}
|