2019-04-27 23:57:39 +10:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Models;
|
|
|
|
|
2021-09-06 23:39:32 +10:00
|
|
|
use Carbon\Carbon;
|
2022-01-01 16:59:35 +11:00
|
|
|
use Illuminate\Database\Eloquent\Model;
|
|
|
|
use Illuminate\Database\Eloquent\SoftDeletes;
|
2022-01-22 23:08:46 +11:00
|
|
|
use Illuminate\Support\Collection;
|
2021-08-30 00:44:20 +10:00
|
|
|
use Illuminate\Support\Facades\DB;
|
2021-09-06 23:39:32 +10:00
|
|
|
use Illuminate\Support\Facades\Log;
|
2022-01-06 00:19:57 +11:00
|
|
|
use Rennokki\QueryCache\Traits\QueryCacheable;
|
2019-04-27 23:57:39 +10:00
|
|
|
|
2022-10-30 23:42:30 +11:00
|
|
|
use App\Casts\{CollectionOrNull,CompressedString};
|
2021-07-31 00:35:52 +10:00
|
|
|
use App\Classes\FTN\Message;
|
|
|
|
use App\Interfaces\Packet;
|
2022-01-01 16:59:35 +11:00
|
|
|
use App\Traits\{EncodeUTF8,MsgID};
|
2021-07-31 00:35:52 +10:00
|
|
|
|
2021-09-06 23:39:32 +10:00
|
|
|
final class Echomail extends Model implements Packet
|
2019-04-27 23:57:39 +10:00
|
|
|
{
|
2022-01-06 00:19:57 +11:00
|
|
|
use SoftDeletes,EncodeUTF8,MsgID,QueryCacheable;
|
2021-08-11 23:45:30 +10:00
|
|
|
|
2021-09-06 23:39:32 +10:00
|
|
|
private const LOGKEY = 'ME-';
|
2022-10-30 23:42:30 +11:00
|
|
|
private Collection $set_seenby;
|
|
|
|
private Collection $set_path;
|
2023-08-05 21:32:45 +10:00
|
|
|
private Carbon $set_recvtime;
|
2023-07-15 22:10:05 +10:00
|
|
|
private string $set_pkt;
|
2022-01-05 09:25:36 +11:00
|
|
|
private bool $no_export = FALSE;
|
2022-01-01 16:59:35 +11:00
|
|
|
|
|
|
|
protected $casts = [
|
2023-06-26 12:32:38 +12:00
|
|
|
'datetime' => 'datetime:Y-m-d H:i:s',
|
2022-10-30 23:42:30 +11:00
|
|
|
'kludges' => CollectionOrNull::class,
|
|
|
|
'msg' => CompressedString::class,
|
|
|
|
'msg_src' => CompressedString::class,
|
2023-06-26 12:32:38 +12:00
|
|
|
'rogue_seenby' => CollectionOrNull::class,
|
|
|
|
'rogue_path' => CollectionOrNull::class,
|
2022-01-01 16:59:35 +11:00
|
|
|
];
|
2021-09-14 23:14:13 +10:00
|
|
|
|
2021-08-11 23:45:30 +10:00
|
|
|
private const cast_utf8 = [
|
2021-12-01 22:45:51 +11:00
|
|
|
'to',
|
|
|
|
'from',
|
2021-08-26 22:32:32 +10:00
|
|
|
'subject',
|
|
|
|
'msg',
|
2022-02-19 16:33:14 +11:00
|
|
|
'msg_src',
|
2021-09-12 23:06:17 +10:00
|
|
|
'origin',
|
|
|
|
'tearline',
|
|
|
|
'tagline',
|
2021-08-11 23:45:30 +10:00
|
|
|
];
|
2019-04-27 23:57:39 +10:00
|
|
|
|
2022-11-01 22:24:36 +11:00
|
|
|
public function __set($key,$value)
|
2022-01-01 16:59:35 +11:00
|
|
|
{
|
|
|
|
switch ($key) {
|
2022-01-05 09:25:36 +11:00
|
|
|
case 'no_export':
|
2022-01-01 16:59:35 +11:00
|
|
|
case 'set_path':
|
2023-07-15 22:10:05 +10:00
|
|
|
case 'set_pkt':
|
2023-08-05 21:32:45 +10:00
|
|
|
case 'set_recvtime':
|
2022-01-01 16:59:35 +11:00
|
|
|
case 'set_seenby':
|
|
|
|
$this->{$key} = $value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
parent::__set($key,$value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-06 23:39:32 +10:00
|
|
|
public static function boot()
|
|
|
|
{
|
|
|
|
parent::boot();
|
|
|
|
|
2021-11-29 21:12:44 +11:00
|
|
|
// @todo if the message is updated with new SEEN-BY's from another route, we'll delete the pending export for systems (if there is one)
|
2021-09-06 23:39:32 +10:00
|
|
|
static::created(function($model) {
|
|
|
|
if (! $model->echoarea_id) {
|
2022-11-01 22:24:36 +11:00
|
|
|
Log::alert(sprintf('%s:- Message has no echoarea, not exporting',self::LOGKEY,$model->id));
|
2021-09-06 23:39:32 +10:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-01-01 16:59:35 +11:00
|
|
|
// Save the seenby
|
2023-07-15 22:10:05 +10:00
|
|
|
$model->seenby()->sync($model->set_seenby);
|
2021-09-06 23:39:32 +10:00
|
|
|
|
2022-01-01 16:59:35 +11:00
|
|
|
// Save the Path
|
|
|
|
$ppoid = NULL;
|
|
|
|
foreach ($model->set_path as $aoid) {
|
|
|
|
$po = DB::select('INSERT INTO echomail_path (echomail_id,address_id,parent_id) VALUES (?,?,?) RETURNING id',[
|
|
|
|
$model->id,
|
|
|
|
$aoid,
|
|
|
|
$ppoid,
|
2021-09-06 23:39:32 +10:00
|
|
|
]);
|
2022-01-01 16:59:35 +11:00
|
|
|
|
|
|
|
$ppoid = $po[0]->id;
|
2021-09-06 23:39:32 +10:00
|
|
|
}
|
|
|
|
|
2023-07-15 22:10:05 +10:00
|
|
|
// Our last node in the path is our sender
|
2023-08-05 21:32:45 +10:00
|
|
|
if (isset($model->set_pkt) && isset($model->set_recvtime)) {
|
|
|
|
DB::update('UPDATE echomail_path set recv_pkt=?,recv_at=? where address_id=? and echomail_id=?',[
|
2023-07-15 22:10:05 +10:00
|
|
|
$model->set_pkt,
|
2023-08-05 21:32:45 +10:00
|
|
|
$model->set_recvtime,
|
2023-07-15 22:10:05 +10:00
|
|
|
$model->set_path->last(),
|
|
|
|
$model->id,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2022-01-01 16:59:35 +11:00
|
|
|
// See if we need to export this message.
|
2023-07-29 13:17:36 +10:00
|
|
|
if ($model->echoarea->sec_read) {
|
|
|
|
$exportto = ($x=$model
|
|
|
|
->echoarea
|
|
|
|
->addresses
|
|
|
|
->filter(function($item) use ($model) { return $item->security >= $model->echoarea->sec_read; }))
|
|
|
|
->pluck('id')
|
|
|
|
->diff($model->set_seenby);
|
|
|
|
|
|
|
|
if ($exportto->count()) {
|
|
|
|
if ($model->no_export) {
|
2023-09-08 21:11:53 +10:00
|
|
|
Log::alert(sprintf('%s:- NOT processing exporting of message by configuration [%s] to [%s]',self::LOGKEY,$model->id,$exportto->join(',')));
|
2023-07-29 13:17:36 +10:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-09-08 21:11:53 +10:00
|
|
|
Log::info(sprintf('%s:- Exporting message [%s] to [%s]',self::LOGKEY,$model->id,$exportto->join(',')));
|
2023-07-29 13:17:36 +10:00
|
|
|
|
|
|
|
// Save the seenby for the exported systems
|
|
|
|
$model->seenby()->syncWithPivotValues($exportto,['export_at'=>Carbon::now()],FALSE);
|
2022-01-05 09:25:36 +11:00
|
|
|
}
|
2022-01-01 16:59:35 +11:00
|
|
|
}
|
2021-09-06 23:39:32 +10:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-07-31 00:35:52 +10:00
|
|
|
/* RELATIONS */
|
2019-04-27 23:57:39 +10:00
|
|
|
|
2021-09-06 23:39:32 +10:00
|
|
|
public function echoarea()
|
|
|
|
{
|
|
|
|
return $this->belongsTo(Echoarea::class);
|
|
|
|
}
|
|
|
|
|
2021-07-31 00:35:52 +10:00
|
|
|
public function fftn()
|
2019-04-27 23:57:39 +10:00
|
|
|
{
|
2022-01-01 16:59:35 +11:00
|
|
|
return $this->belongsTo(Address::class)
|
2021-07-31 00:35:52 +10:00
|
|
|
->withTrashed();
|
2019-04-27 23:57:39 +10:00
|
|
|
}
|
|
|
|
|
2022-01-01 16:59:35 +11:00
|
|
|
public function seenby()
|
2021-08-29 11:48:27 +10:00
|
|
|
{
|
2022-01-01 16:59:35 +11:00
|
|
|
return $this->belongsToMany(Address::class,'echomail_seenby')
|
2023-09-19 17:28:25 +10:00
|
|
|
->FTN2DOrder();
|
2021-08-29 11:48:27 +10:00
|
|
|
}
|
|
|
|
|
2022-01-01 16:59:35 +11:00
|
|
|
public function path()
|
2021-08-29 11:48:27 +10:00
|
|
|
{
|
2022-01-22 23:08:46 +11:00
|
|
|
return $this->belongsToMany(Address::class,'echomail_path')
|
|
|
|
->withPivot(['id','parent_id']);
|
2021-08-29 11:48:27 +10:00
|
|
|
}
|
|
|
|
|
2021-07-31 00:35:52 +10:00
|
|
|
/* METHODS */
|
|
|
|
|
2022-11-04 23:14:47 +11:00
|
|
|
public function init(): void
|
|
|
|
{
|
|
|
|
$this->set_path = collect();
|
|
|
|
$this->set_seenby = collect();
|
|
|
|
}
|
|
|
|
|
2021-08-11 23:45:30 +10:00
|
|
|
public function jsonSerialize(): array
|
|
|
|
{
|
|
|
|
return $this->encode();
|
|
|
|
}
|
|
|
|
|
2021-07-31 00:35:52 +10:00
|
|
|
/**
|
|
|
|
* Return this model as a packet
|
|
|
|
*/
|
|
|
|
public function packet(Address $ao): Message
|
2019-04-27 23:57:39 +10:00
|
|
|
{
|
2023-09-08 21:11:53 +10:00
|
|
|
Log::info(sprintf('%s:+ Bundling [%s]',self::LOGKEY,$this->id));
|
2021-09-06 23:39:32 +10:00
|
|
|
|
2021-07-31 00:35:52 +10:00
|
|
|
// @todo Dont bundle mail to nodes that have been disabled, or addresses that have been deleted
|
|
|
|
$o = new Message;
|
|
|
|
|
|
|
|
$o->header = [
|
|
|
|
'onode' => $this->fftn->node_id,
|
|
|
|
'dnode' => $ao->node_id,
|
|
|
|
'onet' => $this->fftn->host_id,
|
|
|
|
'dnet' => $ao->host_id,
|
2023-07-15 22:10:05 +10:00
|
|
|
'flags' => 0,
|
2021-07-31 00:35:52 +10:00
|
|
|
'cost' => 0,
|
|
|
|
'date'=>$this->datetime->format('d M y H:i:s'),
|
|
|
|
];
|
|
|
|
|
|
|
|
$o->tzutc = $this->datetime->utcOffset($this->tzoffset)->getOffsetString('');
|
|
|
|
$o->user_to = $this->to;
|
|
|
|
$o->user_from = $this->from;
|
|
|
|
$o->subject = $this->subject;
|
2021-09-06 23:39:32 +10:00
|
|
|
$o->echoarea = $this->echoarea->name;
|
2021-07-31 00:35:52 +10:00
|
|
|
$o->flags = $this->flags;
|
|
|
|
|
2021-09-14 23:14:13 +10:00
|
|
|
if ($this->kludges)
|
2022-01-15 13:06:15 +11:00
|
|
|
$o->kludge = collect($this->kludges);
|
2021-09-14 23:14:13 +10:00
|
|
|
|
2022-01-01 16:59:35 +11:00
|
|
|
$o->kludge->put('dbid',$this->id);
|
2021-07-31 00:35:52 +10:00
|
|
|
|
|
|
|
$o->msgid = $this->msgid;
|
2022-01-01 16:59:35 +11:00
|
|
|
if ($this->replyid)
|
|
|
|
$o->replyid = $this->replyid;
|
2021-07-31 00:35:52 +10:00
|
|
|
|
|
|
|
$o->message = $this->msg;
|
|
|
|
|
|
|
|
if ($this->tagline)
|
|
|
|
$o->tagline = $this->tagline;
|
|
|
|
|
|
|
|
if ($this->tearline)
|
|
|
|
$o->tearline = $this->tearline;
|
|
|
|
|
|
|
|
if ($this->origin)
|
|
|
|
$o->origin = $this->origin;
|
|
|
|
|
2023-07-15 22:10:05 +10:00
|
|
|
$sysaddress = Setup::findOrFail(config('app.id'))->system->match($this->fftn->zone)->first();
|
2023-07-30 12:11:08 +10:00
|
|
|
|
|
|
|
if (! $sysaddress)
|
|
|
|
throw new \Exception(sprintf('%s:! We dont have an address in this network? (%s)',self::LOGKEY,$this->fftn->zone->domain->name));
|
|
|
|
|
2023-07-15 22:10:05 +10:00
|
|
|
$o->seenby = $this->seenby->push($sysaddress)->unique()->pluck('ftn2d');
|
|
|
|
|
|
|
|
// Add our address to the path and seenby
|
|
|
|
$o->path = $this->pathorder()->merge($sysaddress->ftn2d);
|
2021-07-31 00:35:52 +10:00
|
|
|
|
2021-09-06 23:39:32 +10:00
|
|
|
$o->packed = TRUE;
|
|
|
|
|
2021-07-31 00:35:52 +10:00
|
|
|
return $o;
|
2019-04-27 23:57:39 +10:00
|
|
|
}
|
2022-01-22 23:08:46 +11:00
|
|
|
|
|
|
|
public function pathorder(string $display='ftn2d',int $start=NULL): Collection
|
|
|
|
{
|
|
|
|
$result = collect();
|
|
|
|
|
|
|
|
if ($x=$this->path->firstWhere('pivot.parent_id',$start)) {
|
|
|
|
$result->push($x->$display);
|
|
|
|
$result->push($this->pathorder($display,$x->pivot->id));
|
2023-06-27 19:39:11 +12:00
|
|
|
}
|
2022-01-22 23:08:46 +11:00
|
|
|
|
|
|
|
return $result->flatten()->filter();
|
|
|
|
}
|
2019-04-27 23:57:39 +10:00
|
|
|
}
|