<?php namespace App\Models; use Carbon\Carbon; use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\SoftDeletes; use Illuminate\Support\Arr; use Illuminate\Support\Collection; use Illuminate\Support\Facades\DB; use Illuminate\Support\Facades\Log; use App\Casts\{CollectionOrNull,CompressedStringOrNull,UTF8StringOrNull}; use App\Interfaces\Packet; use App\Pivots\ViaPivot; use App\Traits\{MessageAttributes,MsgID}; final class Netmail extends Model implements Packet { use SoftDeletes,MsgID,MessageAttributes; private const LOGKEY = 'MN-'; public const UPDATED_AT = NULL; private const PATH_REGEX = '/^([0-9]+:[0-9]+\/[0-9]+(\.?[0-9@a-zA-Z]*)?)\s+@([0-9.a-zA-Z]+)\s+(.*)$/'; /** * Kludges that we absorb in this model */ private const kludges = [ 'MSGID:'=>'msgid', 'REPLY:'=>'replyid', 'Via' => 'set_path', ]; protected $casts = [ 'to' => UTF8StringOrNull::class, 'from' => UTF8StringOrNull::class, 'subject' => UTF8StringOrNull::class, 'datetime' => 'datetime:Y-m-d H:i:s', 'kludges' => CollectionOrNull::class, 'msg' => CompressedStringOrNull::class, 'msg_src' => CompressedStringOrNull::class, 'sent_at' => 'datetime:Y-m-d H:i:s', ]; public function __get($key) { switch ($key) { case 'get_fftn': return $this->set->get('set_fftn') ?: $this->fftn->ftn; case 'get_tftn': return $this->set->get('set_tftn') ?: $this->tftn->ftn; case 'set_fftn': case 'set_tftn': case 'set_path': case 'set_pkt': case 'set_recvtime': case 'set_seenby': case 'set_sender': case 'set_tagline': case 'set_tearline': case 'set_origin': return $this->set->get($key); default: return parent::__get($key); } } public function __set($key,$value) { switch ($key) { case 'kludges': if (! count($value)) return; if (array_key_exists($value[0],self::kludges)) { $this->{self::kludges[$value[0]]} = $value[1]; } else { $this->kludges->put($value[0],$value[1]); } break; case 'set_fftn': case 'set_tftn': // Values that we pass to boot() to record how we got this netmail case 'set_pkt': case 'set_recvtime': case 'set_sender': case 'set_tagline': case 'set_tearline': case 'set_origin': $this->set->put($key,$value); break; // The path the netmail went through to get here case 'set_path': if (! $this->set->has($key)) $this->set->put($key,collect()); $this->set->get($key)->push($value); break; default: parent::__set($key,$value); } } public static function boot() { parent::boot(); static::creating(function($model) { if (isset($model->errors) && $model->errors->count()) throw new \Exception('Cannot save, validation errors exist'); if ($model->set->has('set_tagline')) { $x = Tagline::where('value',utf8_encode($model->set_tagline))->single(); if (! $x) { $x = new Tagline; $x->value = $model->set_tagline; $x->save(); } $model->tagline_id = $x->id; } if ($model->set->has('set_tearline')) { $x = Tearline::where('value',utf8_encode($model->set_tearline))->single(); if (! $x) { $x = new Tearline; $x->value = $model->set_tearline; $x->save(); } $model->tearline_id = $x->id; } if ($model->set->has('set_origin')) { // Make sure our origin contains our FTN $m = []; if ((preg_match('#^(.*)\s+\(([0-9]+:[0-9]+/[0-9]+.*)\)+\s*$#',$model->set_origin,$m)) && (Address::findFTN(sprintf('%s@%s',$m[2],$model->fftn->domain->name),TRUE,TRUE)?->id === $model->fftn_id)) { $x = Origin::where('value',utf8_encode($m[1]))->single(); if (! $x) { $x = new Origin; $x->value = $m[1]; $x->save(); } $model->origin_id = $x->id; } } // If we can rebuild the message content, then we can do away with msg_src if (md5($model->rebuildMessage()) === $model->msg_crc) { Log::debug(sprintf('%s:- Pruning message source, since we can rebuild the message [%s]',self::LOGKEY,$model->msgid)); $model->msg_src = NULL; } }); static::created(function($model) { $nodes = collect(); // Parse PATH // <FTN Address> @YYYYMMDD.HHMMSS[.Precise][.Time Zone] <Program Name> <Version> [Serial Number] if ($model->set->has('set_path')) { foreach ($model->set->get('set_path') as $line) { $m = []; if (preg_match(self::PATH_REGEX,$line,$m)) { // Address // @todo Do we need to add a domain here, since the path line may not include one $ao = Address::findFTN($m[1]); // Time $t = []; $datetime = ''; if (! preg_match('/^([0-9]+\.[0-9]+)(\.?(.*))?$/',$m[3],$t)) Log::alert(sprintf('%s:! Unable to determine time from [%s]',self::LOGKEY,$m[3])); else $datetime = Carbon::createFromFormat('Ymd.His',$t[1],$t[3] ?? ''); if (! $ao) { Log::alert(sprintf('%s:! Undefined Node [%s] in netmail path.',self::LOGKEY,$m[1])); //$rogue->push(['node'=>$m[1],'datetime'=>$datetime,'program'=>$m[4]]); } else { $nodes->push(['node'=>$ao,'datetime'=>$datetime,'program'=>$m[4]]); } } } // If there are no details (Mystic), we'll create a blank } elseif ($model->set->has('set_sender')) { $nodes->push(['node'=>$model->set->get('set_sender'),'datetime'=>Carbon::now(),'program'=>'Unknown']); } // Save the Path $ppoid = NULL; foreach ($nodes as $path) { $po = DB::select('INSERT INTO netmail_path (netmail_id,address_id,parent_id,datetime,program) VALUES (?,?,?,?,?) RETURNING id',[ $model->id, $path['node']->id, $ppoid, (string)$path['datetime'], $path['program'], ]); $ppoid = $po[0]->id; } // Our last node in the path is our sender if ($nodes->count() && $model->set->has('set_pkt') && $model->set->has('set_sender') && $model->set->has('set_recvtime')) { DB::update('UPDATE netmail_path set recv_pkt=?,recv_at=?,recv_id=? where address_id=? and netmail_id=?',[ $model->set->get('set_pkt'), $model->set->get('set_recvtime'), $model->set->get('set_sender')->id, Arr::get($nodes->last(),'node')->id, $model->id, ]); } $model->save(); }); } /* RELATIONS */ public function path() { return $this->belongsToMany(Address::class,'netmail_path') ->withPivot(['id','parent_id','datetime','program','recv_pkt','recv_id']) ->orderBy('netmail_path.id') ->using(ViaPivot::class); } public function sent() { return $this->belongsTo(Address::class); } public function tftn() { return $this ->belongsTo(Address::class) ->withTrashed(); } /* ATTRIBUTES */ /** * Enable rendering the path even if the model hasnt been saved * * @return Collection */ public function getPathAttribute(): Collection { return ((! $this->exists) && $this->set->has('set_path')) ? $this->set->get('set_path')->map(function($item) { $m = []; preg_match(self::PATH_REGEX,$item,$m); return $m[1]; }) : $this->getRelationValue('path'); } /** * Split the body of a message into a collection of lines. * * @return array */ public function getBodyLinesAttribute(): array { return explode("\r",$this->msg_src); } /* METHODS */ /** * Render the via line * * @param Address $ao * @return string * @throws \Exception */ public function via(Address $ao): string { if (! $ao->pivot) throw new \Exception('Cannot render the via line without an address record without a path pivot'); return sprintf('%s @%s.UTC %s', $ao->ftn3d, $ao->pivot->datetime->format('Ymd.His'), $ao->pivot->program); } }