2021-05-13 22:40:21 +10:00
< ? php
namespace App\Models ;
2023-07-30 20:14:38 +10:00
use AgliPanci\LaravelCase\Facades\CaseBuilder ;
2021-11-20 17:58:46 +11:00
use Carbon\Carbon ;
2021-06-15 22:19:14 +10:00
use Illuminate\Database\Eloquent\Factories\HasFactory ;
2021-05-13 22:40:21 +10:00
use Illuminate\Database\Eloquent\Model ;
2021-10-26 23:19:55 +11:00
use Illuminate\Support\Collection ;
2023-11-25 00:10:21 +11:00
use Illuminate\Support\Facades\Auth ;
2024-04-20 22:03:47 +10:00
use Illuminate\Support\Facades\Cache ;
2022-01-01 16:59:35 +11:00
use Illuminate\Support\Facades\DB ;
2021-05-13 22:40:21 +10:00
2024-11-05 00:28:34 +11:00
use App\Models\Casts\CompressedStringOrNull ;
2024-05-11 09:10:00 +10:00
use App\Traits\ { QueryCacheableConfig , ScopeActive };
2021-05-13 22:40:21 +10:00
2024-11-05 00:28:34 +11:00
/**
* This represents an FTN Domain .
*
* We are either a member of the domain ( because we have an AKA in it ) or NOT .
*
* The assumption is , if we are a member of the domain , we receive / send mail to an uplink and possibly downlinks
*/
2021-05-13 22:40:21 +10:00
class Domain extends Model
{
2024-05-11 09:10:00 +10:00
use HasFactory , QueryCacheableConfig , ScopeActive ;
2022-01-01 16:59:35 +11:00
2021-11-26 16:58:50 +11:00
private const CACHE_TIME = 3600 ;
2021-12-03 11:24:23 +11:00
private const STATS_MONTHS = 6 ;
2021-06-14 15:46:18 +10:00
2022-10-30 23:42:30 +11:00
protected $casts = [
2024-06-01 10:46:02 +10:00
'homepage' => CompressedStringOrNull :: class ,
2022-10-30 23:42:30 +11:00
];
2021-06-14 21:33:18 +10:00
/* SCOPES */
/**
2024-11-05 00:28:34 +11:00
* A domain is public ( visible ), if the user is an admin or , the domain is marked public )
2021-06-14 21:33:18 +10:00
*/
public function scopePublic ( $query )
{
2023-11-25 00:10:21 +11:00
$user = Auth :: user ();
return $query
2024-11-05 00:28:34 +11:00
-> active ()
-> when (( ! $user ) || ( ! $user -> isAdmin ()),
fn ( $query ) => $query -> where ( 'public' , TRUE ));
2021-06-14 21:33:18 +10:00
}
2021-06-14 15:46:18 +10:00
/* RELATIONS */
2021-08-11 23:45:30 +10:00
public function echoareas ()
{
2024-11-02 00:10:38 +11:00
return $this -> hasMany ( Echoarea :: class )
2024-11-05 00:28:34 +11:00
-> orderBy ( 'name' );
2021-08-11 23:45:30 +10:00
}
public function fileareas ()
{
2024-11-02 00:10:38 +11:00
return $this -> hasMany ( Filearea :: class )
2024-11-05 00:28:34 +11:00
-> orderBy ( 'name' );
2021-08-11 23:45:30 +10:00
}
2022-11-04 17:20:22 +11:00
public function nodelist_filearea ()
{
return $this -> belongsTo ( Filearea :: class );
}
2024-11-05 00:28:34 +11:00
public function nodestatus_echoarea ()
2024-05-25 22:25:57 +10:00
{
return $this -> belongsTo ( Echoarea :: class , 'nodestatus_id' );
}
2021-06-14 15:46:18 +10:00
public function zones ()
{
2024-06-17 18:33:48 +09:30
return $this -> hasMany ( Zone :: class )
2024-11-05 00:28:34 +11:00
-> select ([ 'id' , 'zone_id' , 'domain_id' , 'active' ])
-> orderBy ( 'zone_id' );
2021-06-14 15:46:18 +10:00
}
2021-06-14 21:33:18 +10:00
2022-01-06 00:19:57 +11:00
/* ATTRIBUTES */
2021-06-14 21:33:18 +10:00
2024-11-05 00:28:34 +11:00
/**
* We can accept applications if we have an address in the domain
*
* @ return bool
* @ throws \Exception
*/
2024-04-26 16:18:40 +10:00
public function getCanAcceptAppAttribute () : bool
{
2024-11-05 00:28:34 +11:00
return $this -> isManaged ()
2024-04-26 16:18:40 +10:00
&& $this -> accept_app
2024-11-05 00:28:34 +11:00
&& Auth :: id ();
2024-04-26 16:18:40 +10:00
}
2022-12-01 23:51:43 +11:00
public function getHomePageAttribute ( ? string $value ) : string
2021-06-14 21:33:18 +10:00
{
2022-10-30 23:42:30 +11:00
return $this -> castAttribute ( 'homepage' , $value ) ? : 'No available information at the moment.' ;
2021-06-14 21:33:18 +10:00
}
2021-10-26 23:19:55 +11:00
/* METHODS */
2024-11-05 00:28:34 +11:00
/**
* Show count of messages by day / week / month / all stats for each echoarea in this domain
*
* @ return Collection
*/
2023-07-30 20:14:38 +10:00
public function echoarea_stats () : Collection
2021-10-26 23:19:55 +11:00
{
2024-10-24 17:00:14 +11:00
return Cache :: remember ( md5 ( sprintf ( '%d-%s' , $this -> id , __METHOD__ )), self :: CACHE_TIME , function () {
$dt = Carbon :: now () -> startOfday ();
$case = CaseBuilder :: whenRaw ( " datetime >= '?' " , $dt -> subDay () -> format ( 'Y-m-d' )) -> thenRaw ( " 'day' " )
-> whenRaw ( " datetime >= '?' " , $dt -> subDays ( 7 ) -> format ( 'Y-m-d' )) -> thenRaw ( " 'week' " )
-> whenRaw ( " datetime >= '?' " , $dt -> subMonth () -> format ( 'Y-m-d' )) -> thenRaw ( " 'month' " )
-> elseRaw ( " 'all' " );
return Echoarea :: select ([ 'echoareas.id' , 'name' , 'description' , 'active' , DB :: raw ( 'count(echomails.id) AS count' ), DB :: raw ( 'min(datetime) as first_message' ), DB :: raw ( 'max(datetime) as last_message' )])
-> selectRaw ( $case -> toRaw () . ' AS stats' )
-> join ( 'echomails' ,[ 'echomails.echoarea_id' => 'echoareas.id' ], NULL , NULL , 'left outer' )
-> where ( 'domain_id' , $this -> id )
-> groupBy ( 'echoareas.id' )
-> groupBy ( 'echoareas.name' )
-> groupBy ( 'stats' )
-> orderBy ( 'echoareas.name' )
-> orderBy ( 'last_message' , 'DESC' )
-> get ();
});
2021-10-26 23:19:55 +11:00
}
2022-11-25 17:44:03 +07:00
2024-11-05 00:28:34 +11:00
/**
* Show daily total of messages by echoarea in this domain
*
* @ param Collection | NULL $systems
* @ return Collection
*/
2024-04-20 22:03:47 +10:00
public function echoarea_total_daily ( Collection $systems = NULL ) : Collection
{
return Cache :: remember ( md5 ( sprintf ( '%d-%s' , $this -> id , $systems ? -> pluck ( 'id' ) -> join ( ',' ))), self :: CACHE_TIME , function () use ( $systems ) {
return DB :: query ()
2024-09-10 14:52:35 +10:00
-> select ([ 'echoareas.name' , 'echoareas.show' , DB :: raw ( 'COUNT(echoareas.name) AS count' ), DB :: raw ( 'datetime::date AS date' )])
2024-04-20 22:03:47 +10:00
-> from ( $this -> getTable ())
-> join ( 'echoareas' ,[ 'echoareas.domain_id' => 'domains.id' ])
-> join ( 'echomails' ,[ 'echomails.echoarea_id' => 'echoareas.id' ])
-> where ( 'domains.id' , $this -> id )
-> where ( 'echomails.datetime' , '>=' , Carbon :: now () -> subMonths ( self :: STATS_MONTHS ) -> startOfMonth ())
2024-11-05 00:28:34 +11:00
-> when ( $systems ? -> count (), fn ( $query ) => $query -> whereIn ( 'echomails.fftn_id' , $systems -> pluck ( 'addresses' ) -> flatten () -> pluck ( 'id' )))
2024-04-20 22:03:47 +10:00
-> groupBy ([ 'echoareas.name' , 'echoareas.show' , 'date' ])
-> orderBy ( 'echoareas.name' )
-> orderBy ( 'date' )
-> get ();
});
}
2024-11-05 00:28:34 +11:00
/**
* Show count of files by week / month / all status for each filearea in this domain
*/
2024-10-23 21:17:38 +11:00
public function filearea_stats ()
{
2024-10-24 17:00:14 +11:00
return Cache :: remember ( md5 ( sprintf ( '%d-%s' , $this -> id , __METHOD__ )), self :: CACHE_TIME , function () {
$dt = Carbon :: now () -> startOfday ();
$case = CaseBuilder :: whenRaw ( " datetime >= '?' " , $dt -> subDays ( 7 ) -> format ( 'Y-m-d' )) -> thenRaw ( " 'week' " )
-> whenRaw ( " datetime >= '?' " , $dt -> subMonth () -> format ( 'Y-m-d' )) -> thenRaw ( " 'month' " )
-> elseRaw ( " 'all' " );
return Filearea :: select ([ 'fileareas.id' , 'fileareas.name' , 'description' , 'active' , DB :: raw ( 'count(files.id) AS count' ), DB :: raw ( 'min(datetime) as first_file' ), DB :: raw ( 'max(datetime) as last_file' )])
-> selectRaw ( $case -> toRaw () . ' AS stats' )
-> join ( 'files' ,[ 'files.filearea_id' => 'fileareas.id' ], NULL , NULL , 'left outer' )
-> where ( 'domain_id' , $this -> id )
-> groupBy ( 'fileareas.id' )
-> groupBy ( 'fileareas.name' )
-> groupBy ( 'stats' )
-> orderBy ( 'fileareas.name' )
-> orderBy ( 'last_file' , 'DESC' )
-> get ();
});
2024-10-23 21:17:38 +11:00
}
2022-11-25 17:44:03 +07:00
/**
* Determine if this zone is managed by this host
*
* @ return bool
2024-05-29 19:13:28 +10:00
* @ throws \Exception
2022-11-25 17:44:03 +07:00
*/
2024-05-29 19:13:28 +10:00
public function isManaged () : bool
2022-11-25 17:44:03 +07:00
{
2024-11-05 00:28:34 +11:00
return our_address ( $this ) -> count () > 0 ;
2024-04-26 16:18:40 +10:00
}
2021-05-13 22:40:21 +10:00
}