2021-07-16 00:54:23 +10:00
< ? php
namespace App\Jobs ;
2021-10-18 23:10:15 +11:00
use Carbon\Carbon ;
2021-07-16 00:54:23 +10:00
use Illuminate\Bus\Queueable ;
use Illuminate\Contracts\Queue\ShouldQueue ;
use Illuminate\Foundation\Bus\Dispatchable ;
use Illuminate\Queue\InteractsWithQueue ;
use Illuminate\Queue\SerializesModels ;
2021-07-24 00:53:35 +10:00
use Illuminate\Support\Facades\Log ;
2023-07-23 17:27:52 +10:00
use Illuminate\Support\Facades\Notification ;
2021-07-16 00:54:23 +10:00
2023-07-23 17:27:52 +10:00
use App\Classes\FTN\Message ;
2024-05-19 23:28:45 +10:00
use App\Models\ { Echomail , Netmail , User };
use App\Notifications\Netmails\ { EchoareaNotExist , EchoareaNotSubscribed , EchoareaNoWrite , NetmailForward , NetmailHubNoUser };
2024-06-01 10:46:02 +10:00
use App\Traits\ParseAddresses ;
2021-07-16 00:54:23 +10:00
2021-09-06 23:39:32 +10:00
class MessageProcess implements ShouldQueue
2021-07-16 00:54:23 +10:00
{
2021-09-12 00:07:02 +10:00
private const LOGKEY = 'JMP' ;
2021-08-19 00:20:34 +10:00
2024-06-01 10:46:02 +10:00
use Dispatchable , InteractsWithQueue , Queueable , SerializesModels , ParseAddresses ;
2021-07-16 00:54:23 +10:00
2024-05-17 22:10:54 +10:00
private Echomail | Netmail | string $mo ;
2021-09-11 00:38:11 +10:00
private bool $skipbot ;
2021-07-16 00:54:23 +10:00
2024-05-17 22:10:54 +10:00
/**
* Process a message from a packet
*
* @ param Echomail | Netmail $mo The message object
* @ param bool $skipbot Dont trigger bot actions
*/
public function __construct ( Echomail | Netmail $mo , bool $skipbot = FALSE )
2021-07-16 00:54:23 +10:00
{
2024-05-17 22:10:54 +10:00
// @todo We need to serialize this model here, because laravel has an error unserializing it (Model Not Found)
2024-06-05 21:57:16 +10:00
$this -> mo = utf8_encode ( serialize ( $mo ));
2023-07-15 10:46:19 +10:00
$this -> skipbot = $skipbot ;
2021-07-16 00:54:23 +10:00
}
2023-09-05 19:42:41 +12:00
public function __get ( $key ) : mixed
{
switch ( $key ) {
2024-05-21 21:10:58 +10:00
case 'jobname' :
$mo = unserialize ( $this -> mo );
return sprintf ( '%s-%s-%s' , $mo -> set -> get ( 'set_pkt' ), $mo -> set -> get ( 'set_sender' ) -> ftn , $mo -> msgid );
2023-09-05 19:42:41 +12:00
default :
return NULL ;
}
}
2021-07-16 00:54:23 +10:00
/**
2024-05-17 22:10:54 +10:00
* At this point , we know that the packet is from a system we know about , and the packet is to us :
* + From a system that is configured with us , and the password has been validated
* + From a system that is not configured with us , and it may have netmails for us
2021-07-16 00:54:23 +10:00
*/
public function handle ()
{
2024-06-05 21:57:16 +10:00
$this -> mo = unserialize ( utf8_decode ( $this -> mo ));
2024-05-17 22:10:54 +10:00
2021-07-16 00:54:23 +10:00
// Load our details
2023-12-18 15:13:16 +11:00
$ftns = our_address ();
2021-07-16 00:54:23 +10:00
// If we are a netmail
2024-05-17 22:10:54 +10:00
if ( $this -> mo instanceof Netmail ) {
// @todo generate exception when netmail to system that doesnt exist (node/point) and its this host's responsibility
2023-07-15 10:46:19 +10:00
Log :: info ( sprintf ( '%s:- Processing Netmail [%s] to (%s) [%s] from (%s) [%s].' ,
2023-01-25 16:26:10 +11:00
self :: LOGKEY ,
2024-05-17 22:10:54 +10:00
$this -> mo -> msgid ,
$this -> mo -> to , $this -> mo -> tftn -> ftn ,
$this -> mo -> from , $this -> mo -> fftn -> ftn ,
2023-01-25 16:26:10 +11:00
));
2021-07-16 00:54:23 +10:00
// @todo Enable checks to reject old messages
2023-07-15 10:46:19 +10:00
// Check for duplicate messages
// FTS-0009.001
2024-05-17 22:10:54 +10:00
if ( $this -> mo -> msgid ) {
Log :: debug ( sprintf ( '%s:- Checking for duplicate from host [%s].' , self :: LOGKEY , $this -> mo -> fftn -> ftn ));
$o = Netmail :: where ( 'msgid' , $this -> mo -> msgid )
-> where ( 'fftn_id' , $this -> mo -> fftn -> id )
2023-07-15 10:46:19 +10:00
-> where ( 'datetime' , '>' , Carbon :: now () -> subYears ( 3 ))
-> single ();
if ( $o ) {
2024-05-17 22:10:54 +10:00
Log :: alert ( sprintf ( '%s:! Duplicate netmail #%d [%s] from (%s) [%s] to (%s) - ignoring.' ,
2023-07-15 10:46:19 +10:00
self :: LOGKEY ,
2024-05-17 22:10:54 +10:00
$o -> id ,
$this -> mo -> msgid ,
$this -> mo -> from , $this -> mo -> fftn -> ftn ,
$this -> mo -> to ,
2023-07-15 10:46:19 +10:00
));
return ;
}
}
2021-07-24 00:53:35 +10:00
// @todo Enable checks to see if this is a file request or file send
2021-07-16 00:54:23 +10:00
2023-07-20 20:04:41 +10:00
// Strip any local/transit flags
2024-05-17 22:10:54 +10:00
$this -> mo -> flags &= ~ ( Message :: FLAG_LOCAL | Message :: FLAG_INTRANSIT );
2023-01-24 22:37:41 +11:00
2021-07-16 00:54:23 +10:00
// Determine if the message is to this system, or in transit
2024-05-17 22:10:54 +10:00
if ( $ftns -> contains ( $this -> mo -> tftn )) {
2021-07-16 00:54:23 +10:00
$processed = FALSE ;
// If the message is to a bot, we'll process it
2021-09-11 00:38:11 +10:00
if ( ! $this -> skipbot )
foreach ( config ( 'process.robots' ) as $class ) {
2024-05-17 22:10:54 +10:00
if ( $processed = $class :: handle ( $this -> mo )) {
$this -> mo -> flags |= Message :: FLAG_RECD ;
$this -> mo -> save ();
2023-01-25 16:35:58 +11:00
2023-07-17 16:36:53 +10:00
Log :: info ( sprintf ( '%s:= Netmail [%s] from (%s:%s) - was processed by us internally [%d]' ,
2023-07-15 22:10:05 +10:00
self :: LOGKEY ,
2024-05-17 22:10:54 +10:00
$this -> mo -> msgid ,
$this -> mo -> from ,
$this -> mo -> fftn -> ftn ,
$this -> mo -> id ,
2023-07-15 22:10:05 +10:00
));
2024-05-17 22:10:54 +10:00
2021-09-11 00:38:11 +10:00
break ;
}
2021-07-16 00:54:23 +10:00
}
2023-08-02 22:42:59 +10:00
if ( ! $processed ) {
// Check if the netmail is to a user, with netmail forwarding enabled
$uo = User :: active ()
-> where ( function ( $query ) {
2024-06-01 10:46:02 +10:00
return $query -> whereRaw ( sprintf ( " LOWER(name)='%s' " , strtolower ( utf8_encode ( $this -> mo -> to ))))
-> orWhereRaw ( sprintf ( " LOWER(alias)='%s' " , strtolower ( utf8_encode ( $this -> mo -> to ))));
2023-08-02 22:42:59 +10:00
})
-> whereNotNull ( 'system_id' )
-> single ();
2024-05-17 22:10:54 +10:00
if ( $uo && ( $ao = $uo -> system -> match ( $this -> mo -> tftn -> zone ) ? -> pop ())) {
2023-08-02 22:42:59 +10:00
$note = " +--[ FORWARDED MESSAGE ]----------------------------------+ \r " ;
$note .= " + This message has been forwarded to you, it was originally sent to you \r " ;
2024-05-17 22:10:54 +10:00
$note .= sprintf ( " + at [%s] \r " , $this -> mo -> tftn -> ftn );
2023-08-02 22:42:59 +10:00
$note .= " +---------------------------------------------------------+ \r \r " ;
2024-05-17 22:10:54 +10:00
$this -> mo -> msg = $note . $this -> mo -> content ;
$this -> mo -> tftn_id = $ao -> id ;
$this -> mo -> flags |= Message :: FLAG_INTRANSIT ;
$this -> mo -> save ();
2023-08-02 22:42:59 +10:00
$processed = TRUE ;
// Dont send an advisement to an areabot
2024-05-17 22:10:54 +10:00
if ( ! in_array ( strtolower ( $this -> mo -> from ), config ( 'fido.areabots' )))
Notification :: route ( 'netmail' , $this -> mo -> fftn ) -> notify ( new NetmailForward ( $this -> mo , $ao ));
2023-08-02 22:42:59 +10:00
// We'll ignore messages from *fix users
2024-05-17 22:10:54 +10:00
} elseif ( in_array ( strtolower ( $this -> mo -> from ), config ( 'fido.areabots' ))) {
$this -> mo -> flags |= Message :: FLAG_RECD ;
$this -> mo -> save ();
2023-08-02 22:42:59 +10:00
Log :: alert ( sprintf ( '%s:! Ignoring Netmail [%s] to the Hub from (%s:%s) - its from a bot [%d]' ,
self :: LOGKEY ,
2024-05-17 22:10:54 +10:00
$this -> mo -> msgid ,
$this -> mo -> from ,
$this -> mo -> fftn -> ftn ,
$this -> mo -> id ,
2023-08-02 22:42:59 +10:00
));
$processed = TRUE ;
}
2021-08-15 00:21:04 +10:00
}
2021-07-16 00:54:23 +10:00
// If not processed, no users here!
if ( ! $processed ) {
2024-05-17 22:10:54 +10:00
Log :: alert ( sprintf ( '%s:! Netmail to the Hub from (%s) [%s] but no users here.' , self :: LOGKEY , $this -> mo -> from , $this -> mo -> fftn -> ftn ));
2021-07-24 00:53:35 +10:00
2024-05-19 23:28:45 +10:00
Notification :: route ( 'netmail' , $this -> mo -> fftn ) -> notify ( new NetmailHubNoUser ( $this -> mo ));
2021-07-16 00:54:23 +10:00
}
// If in transit, store for collection
} else {
// @todo In transit loop checking
// @todo In transit TRACE response
2024-05-17 22:10:54 +10:00
$this -> mo -> flags |= Message :: FLAG_INTRANSIT ;
$this -> mo -> save ();
2023-01-24 22:37:41 +11:00
2023-07-15 10:46:19 +10:00
Log :: info ( sprintf ( '%s:= Netmail [%s] in transit to (%s:%s) from (%s:%s) [%d].' ,
2023-01-24 22:37:41 +11:00
self :: LOGKEY ,
2024-05-17 22:10:54 +10:00
$this -> mo -> msgid ,
$this -> mo -> to , $this -> mo -> tftn -> ftn ,
$this -> mo -> from , $this -> mo -> fftn -> ftn ,
$this -> mo -> id ,
2023-01-24 22:37:41 +11:00
));
2021-07-16 00:54:23 +10:00
}
// Else we are echomail
} else {
2024-05-17 22:10:54 +10:00
// The packet sender
$sender = $this -> mo -> set -> get ( 'set_sender' );
2023-01-01 13:50:12 +11:00
2024-05-17 22:10:54 +10:00
// @todo Check that this does evaulate to true if a message has been rescanned
$rescanned = $this -> mo -> kludges -> get ( 'RESCANNED' , FALSE );
2023-01-01 13:50:12 +11:00
2024-05-17 22:10:54 +10:00
// Echoarea doesnt exist, cant import the message
if ( ! $this -> mo -> echoarea ) {
Log :: alert ( sprintf ( '%s:! Echoarea [%s] doesnt exist for zone [%d@%s]' , self :: LOGKEY , $this -> mo -> set -> get ( 'set_echoarea' ), $sender -> zone -> zone_id , $sender -> zone -> domain -> name ));
2021-08-11 23:45:30 +10:00
2024-05-17 22:10:54 +10:00
Notification :: route ( 'netmail' , $sender ) -> notify ( new EchoareaNotExist ( $this -> mo ));
2022-02-12 10:21:46 +11:00
return ;
}
2024-05-17 22:10:54 +10:00
Log :: debug ( sprintf ( '%s:- Processing echomail [%s] in [%s] from [%s].' , self :: LOGKEY , $this -> mo -> msgid , $this -> mo -> echoarea -> name , $sender -> ftn ));
2021-11-24 22:34:40 +11:00
2024-05-17 22:10:54 +10:00
// Message from zone is incorrect for echoarea
if ( ! $this -> mo -> echoarea -> domain -> zones -> contains ( $this -> mo -> fftn -> zone )) {
2023-09-20 20:29:23 +10:00
Log :: alert ( sprintf ( '%s:! The message [%s] is from a different zone [%d] than the packet sender [%d] - not importing' ,
2023-09-17 15:46:09 +10:00
self :: LOGKEY ,
2024-05-17 22:10:54 +10:00
$this -> mo -> msgid ,
$this -> mo -> fftn -> zone -> zone_id ,
$this -> mo -> fftn -> zone -> zone_id ));
2023-09-20 20:29:23 +10:00
return ;
2023-09-17 15:46:09 +10:00
}
2021-08-19 00:20:34 +10:00
// Check for duplicate messages
2021-10-18 23:10:15 +11:00
// FTS-0009.001
2024-05-17 22:10:54 +10:00
if ( $this -> mo -> msgid ) {
$o = Echomail :: where ( 'msgid' , $this -> mo -> msgid )
-> where ( 'fftn_id' , $this -> mo -> fftn -> id )
2024-06-28 15:36:48 +10:00
-> where ( 'datetime' , '>=' , $this -> mo -> datetime -> subYears ( 3 ))
-> where ( 'datetime' , '<=' , $this -> mo -> datetime )
2021-10-18 23:10:15 +11:00
-> single ();
2021-08-19 00:20:34 +10:00
2024-05-17 22:10:54 +10:00
Log :: debug ( sprintf ( '%s:- Checking for duplicate from host id [%d].' , self :: LOGKEY , $this -> mo -> fftn -> id ));
2021-11-24 22:34:40 +11:00
2021-08-19 00:20:34 +10:00
if ( $o ) {
2024-05-17 22:10:54 +10:00
// @todo Actually update seenby
2021-11-24 22:34:40 +11:00
Log :: alert ( sprintf ( '%s:! Duplicate echomail [%s] in [%s] from (%s) [%s] to (%s) - updating seenby.' ,
2021-08-19 00:20:34 +10:00
self :: LOGKEY ,
2024-05-17 22:10:54 +10:00
$this -> mo -> msgid ,
$this -> mo -> echoarea -> name ,
$this -> mo -> from , $this -> mo -> fftn -> ftn ,
$this -> mo -> to ,
2021-08-19 00:20:34 +10:00
));
2024-05-17 22:10:54 +10:00
//$o->save();
2021-11-25 21:22:36 +11:00
// @todo This duplicate message may have gone via a different path, be nice to record it.
2024-05-17 22:10:54 +10:00
/*
2023-09-20 20:29:23 +10:00
// If we didnt get the path on the original message, we'll override it
if ( ! $o -> path -> count ()) {
$dummy = collect ();
2024-05-17 22:10:54 +10:00
$path = $this -> parseAddresses ( 'path' , $this -> mo -> path , $sender -> zone , $dummy );
2023-11-17 12:18:55 +11:00
2023-09-20 20:29:23 +10:00
$ppoid = NULL ;
foreach ( $path as $aoid ) {
$po = DB :: select ( 'INSERT INTO echomail_path (echomail_id,address_id,parent_id) VALUES (?,?,?) RETURNING id' ,[
$o -> id ,
$aoid ,
$ppoid ,
]);
$ppoid = $po [ 0 ] -> id ;
}
}
2024-05-17 22:10:54 +10:00
*/
2023-09-20 20:29:23 +10:00
2022-01-15 13:06:15 +11:00
// @todo if we have an export for any of the seenby addresses, remove it
2024-05-17 22:10:54 +10:00
2024-05-19 23:28:45 +10:00
//$seenby = $this->parseAddresses('seenby',$this->mo->seenby,$sender->zone,$o->rogue_seenby);
//$this->mo->seenby()->syncWithoutDetaching($seenby);
2023-09-20 20:29:23 +10:00
// In case our rogue_seenby changed
2024-05-19 23:28:45 +10:00
//$this->mo->save();
2021-11-24 22:34:40 +11:00
2021-08-19 00:20:34 +10:00
return ;
}
}
2021-11-24 22:34:40 +11:00
// Find another message with the same msg_crc
2024-05-17 22:10:54 +10:00
if ( $this -> mo -> msg_crc ) {
$o = Echomail :: where ( 'msg_crc' , $xx = md5 ( $this -> mo -> msg_crc ))
-> where ( 'fftn_id' , $this -> mo -> fftn -> id )
2021-11-24 22:34:40 +11:00
-> where ( 'datetime' , '>' , Carbon :: now () -> subWeek ())
-> get ();
if ( $o -> count ())
Log :: alert ( sprintf ( '%s:! Duplicate message CRC [%s] in [%s].' ,
self :: LOGKEY ,
$xx ,
$o -> pluck ( 'id' ) -> join ( '|' )
));
}
2023-07-29 13:17:36 +10:00
// Can the system send messages to this area?
2024-05-17 22:10:54 +10:00
if ( ! $this -> mo -> echoarea -> can_write ( $sender -> security )) {
Log :: alert ( sprintf ( '%s:! FTN [%s] is not allowed to post [%s] to [%s].' , self :: LOGKEY , $sender -> ftn , $this -> mo -> msgid , $this -> mo -> echoarea -> name ));
if ( ! $rescanned )
Notification :: route ( 'netmail' , $sender ) -> notify ( new EchoareaNoWrite ( $this -> mo ));
2023-09-17 15:46:09 +10:00
2023-07-29 13:17:36 +10:00
return ;
}
2024-05-19 23:28:45 +10:00
// If the node is not subscribed, we'll accept it, but let them know
if ( ! $sender -> echoareas -> contains ( $this -> mo -> echoarea )) {
Log :: alert ( sprintf ( '%s:! FTN [%s] is not subscribed to [%s] for [%s].' , self :: LOGKEY , $sender -> ftn , $this -> mo -> echoarea -> name , $this -> mo -> msgid ));
if ( ! $rescanned )
Notification :: route ( 'netmail' , $sender ) -> notify ( new EchoareaNotSubscribed ( $this -> mo ));
}
2021-07-16 00:54:23 +10:00
// We know about this area, store it
2024-05-17 22:10:54 +10:00
$this -> mo -> save ();
2021-07-31 00:35:52 +10:00
2023-07-15 10:46:19 +10:00
Log :: info ( sprintf ( '%s:= Echomail [%s] in [%s] from (%s) [%s] to (%s) - [%s].' ,
2021-08-19 23:35:48 +10:00
self :: LOGKEY ,
2024-05-17 22:10:54 +10:00
$this -> mo -> msgid ,
$this -> mo -> echoarea -> name ,
$this -> mo -> from , $this -> mo -> fftn -> ftn ,
$this -> mo -> to ,
$this -> mo -> id ,
2021-08-19 23:35:48 +10:00
));
2021-09-13 23:02:39 +10:00
// If the message is to a bot, but not rescanned, or purposely skipbot set, we'll process it
2024-05-17 22:10:54 +10:00
if (( ! $this -> skipbot ) && ( ! $rescanned ))
2021-09-11 00:38:11 +10:00
foreach ( config ( 'process.echomail' ) as $class ) {
2024-05-17 22:10:54 +10:00
if ( $class :: handle ( $this -> mo )) {
2021-09-11 00:38:11 +10:00
break ;
}
2021-07-31 00:35:52 +10:00
}
2021-07-16 00:54:23 +10:00
}
}
}