2022-11-01 22:24:36 +11:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Models;
|
|
|
|
|
|
|
|
use Carbon\Carbon;
|
|
|
|
use Illuminate\Database\Eloquent\Model;
|
|
|
|
use Illuminate\Database\Eloquent\SoftDeletes;
|
|
|
|
use Illuminate\Support\Facades\DB;
|
|
|
|
use Illuminate\Support\Facades\Log;
|
2022-11-02 21:20:02 +11:00
|
|
|
use Illuminate\Support\Facades\Storage;
|
2022-11-01 22:24:36 +11:00
|
|
|
use Rennokki\QueryCache\Traits\QueryCacheable;
|
|
|
|
|
|
|
|
use App\Casts\{CollectionOrNull,CompressedString};
|
|
|
|
use App\Traits\EncodeUTF8;
|
|
|
|
|
|
|
|
class File extends Model
|
|
|
|
{
|
|
|
|
use SoftDeletes,EncodeUTF8,QueryCacheable;
|
|
|
|
|
|
|
|
private const LOGKEY = 'MF-';
|
|
|
|
private bool $no_export = FALSE;
|
|
|
|
|
|
|
|
protected $casts = [
|
|
|
|
'kludges' => CollectionOrNull::class,
|
2023-06-26 12:32:38 +12:00
|
|
|
'datetime' => 'datetime:Y-m-d H:i:s',
|
2022-11-01 22:24:36 +11:00
|
|
|
'desc' => CompressedString::class,
|
|
|
|
'ldesc' => CompressedString::class,
|
2023-06-26 12:32:38 +12:00
|
|
|
'rogue_seenby' => CollectionOrNull::class,
|
|
|
|
'rogue_path' => CollectionOrNull::class,
|
2022-11-01 22:24:36 +11:00
|
|
|
'size' => 'int',
|
|
|
|
];
|
|
|
|
|
|
|
|
private const cast_utf8 = [
|
|
|
|
'desc',
|
|
|
|
'ldesc',
|
|
|
|
];
|
|
|
|
|
|
|
|
public function __set($key,$value)
|
|
|
|
{
|
|
|
|
switch ($key) {
|
|
|
|
case 'fullname':
|
|
|
|
case 'replaces':
|
|
|
|
case 'no_export':
|
|
|
|
case 'set_path':
|
|
|
|
case 'set_seenby':
|
|
|
|
$this->{$key} = $value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
parent::__set($key,$value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function boot()
|
|
|
|
{
|
|
|
|
parent::boot();
|
|
|
|
|
|
|
|
static::creating(function($model) {
|
2022-11-02 21:20:02 +11:00
|
|
|
Log::debug(sprintf('%s:- Storing file [%s] in [%s]',self::LOGKEY,$model->fullname,$model->full_storage_path));
|
2022-11-01 22:24:36 +11:00
|
|
|
|
|
|
|
// Store file
|
2022-11-02 21:20:02 +11:00
|
|
|
if (Storage::put($model->full_storage_path,Storage::disk('local')->get($model->fullname),'public')) {
|
2022-11-03 22:05:49 +11:00
|
|
|
unlink(Storage::disk('local')->path($model->fullname));
|
2022-11-02 21:20:02 +11:00
|
|
|
} else {
|
|
|
|
throw new \Exception(sprintf('Unable to move file [%s] to [%s]',$model->fullname,$model->full_storage_path));
|
2023-06-27 19:39:11 +12:00
|
|
|
}
|
2022-11-01 22:24:36 +11:00
|
|
|
|
|
|
|
// Delete anything being replaced
|
2022-11-02 21:20:02 +11:00
|
|
|
// @todo implement replace
|
2022-11-01 22:24:36 +11:00
|
|
|
});
|
|
|
|
|
|
|
|
// @todo if the file is updated with new SEEN-BY's from another route, we'll delete the pending export for systems (if there is one)
|
|
|
|
static::created(function($model) {
|
|
|
|
if (! $model->filearea_id) {
|
|
|
|
Log::alert(sprintf('%s:- File has no filearea, not exporting',self::LOGKEY,$model->id));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$so = Setup::findOrFail(config('app.id'));
|
|
|
|
|
|
|
|
// Our address
|
|
|
|
$ftns = $so
|
|
|
|
->system
|
|
|
|
->match($model->fftn->zone,Address::NODE_ACTIVE|Address::NODE_PVT|Address::NODE_HOLD);
|
|
|
|
|
|
|
|
// Add our address to the seenby;
|
|
|
|
$model->set_seenby = $model->set_seenby->merge($ftns->pluck('id'))->unique();
|
|
|
|
$model->set_path = $model->set_path->merge([[
|
|
|
|
'address'=>$ftns->first(),
|
|
|
|
'datetime'=>($x=Carbon::now())->timestamp,
|
|
|
|
'extra'=>sprintf('%s %s (%s)',$x->toRfc7231String(),$so::PRODUCT_NAME,$so->version),
|
|
|
|
]]);
|
|
|
|
|
2022-11-03 22:05:49 +11:00
|
|
|
// Make sure all the path is in the seenby
|
|
|
|
$model->set_seenby = $model->set_seenby->merge($model->set_path->pluck('address.id'))->unique();
|
|
|
|
|
2022-11-01 22:24:36 +11:00
|
|
|
// Save the seenby
|
|
|
|
$model->seenby()->sync($model->set_seenby);
|
|
|
|
|
|
|
|
// Save the Path
|
|
|
|
$ppoid = NULL;
|
|
|
|
foreach ($model->set_path as $path) {
|
|
|
|
$po = DB::select('INSERT INTO file_path (file_id,address_id,parent_id,datetime,extra) VALUES (?,?,?,?,?) RETURNING id',[
|
|
|
|
$model->id,
|
|
|
|
$path['address']->id,
|
|
|
|
$ppoid,
|
|
|
|
Carbon::createFromTimestamp($path['datetime']),
|
|
|
|
$path['extra'],
|
|
|
|
]);
|
|
|
|
|
|
|
|
$ppoid = $po[0]->id;
|
|
|
|
}
|
|
|
|
|
|
|
|
// See if we need to export this message.
|
|
|
|
$exportto = $model->filearea->addresses->pluck('id')->diff($model->set_seenby);
|
|
|
|
|
|
|
|
if ($exportto->count()) {
|
|
|
|
if ($model->no_export) {
|
|
|
|
Log::debug(sprintf('%s:- NOT processing exporting of message by configuration [%s] to [%s]',self::LOGKEY,$model->id,$exportto->join(',')));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Log::debug(sprintf('%s:- Exporting file [%s] to [%s]',self::LOGKEY,$model->id,$exportto->join(',')));
|
|
|
|
|
|
|
|
// Save the seenby for the exported systems
|
|
|
|
$model->seenby()->syncWithPivotValues($exportto,['export_at'=>Carbon::now()],FALSE);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/* RELATIONS */
|
|
|
|
|
|
|
|
public function filearea()
|
|
|
|
{
|
|
|
|
return $this->belongsTo(Filearea::class);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function fftn()
|
|
|
|
{
|
|
|
|
return $this->belongsTo(Address::class)
|
|
|
|
->withTrashed();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function seenby()
|
|
|
|
{
|
|
|
|
return $this->belongsToMany(Address::class,'file_seenby')
|
|
|
|
->ftnOrder();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function path()
|
|
|
|
{
|
|
|
|
return $this->belongsToMany(Address::class,'file_path')
|
2023-06-22 17:36:22 +10:00
|
|
|
->withPivot(['id','parent_id','datetime','extra']);
|
2022-11-01 22:24:36 +11:00
|
|
|
}
|
|
|
|
|
2022-11-02 21:20:02 +11:00
|
|
|
/* ATTRIBUTES */
|
|
|
|
|
|
|
|
public function getFullStoragePathAttribute(): string
|
|
|
|
{
|
2023-06-23 17:33:47 +10:00
|
|
|
return sprintf('%04X/%s',$this->filearea_id,$this->name);
|
2022-11-02 21:20:02 +11:00
|
|
|
}
|
|
|
|
|
2022-11-01 22:24:36 +11:00
|
|
|
/* METHODS */
|
|
|
|
|
|
|
|
public function jsonSerialize(): array
|
|
|
|
{
|
|
|
|
return $this->encode();
|
|
|
|
}
|
|
|
|
}
|