ctrlpanel/app/Models/User.php
2023-01-10 19:45:19 +01:00

306 lines
6.8 KiB
PHP

<?php
namespace App\Models;
use App\Classes\Pterodactyl;
use App\Notifications\Auth\QueuedVerifyEmail;
use App\Notifications\WelcomeMessage;
use Illuminate\Contracts\Auth\MustVerifyEmail;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
use Illuminate\Database\Eloquent\Relations\HasMany;
use Illuminate\Database\Eloquent\Relations\HasOne;
use Illuminate\Foundation\Auth\User as Authenticatable;
use Illuminate\Notifications\Notifiable;
use Spatie\Activitylog\LogOptions;
use Spatie\Activitylog\Traits\CausesActivity;
use Spatie\Activitylog\Traits\LogsActivity;
/**
* Class User
*/
class User extends Authenticatable implements MustVerifyEmail
{
use HasFactory, Notifiable, LogsActivity, CausesActivity;
/**
* @var string[]
*/
protected static $logAttributes = ['name', 'email'];
/**
* @var string[]
*/
protected static $ignoreChangedAttributes = [
'remember_token',
'credits',
'updated_at',
'server_limit',
'last_seen',
'ip',
'pterodactyl_id',
];
/**
* The attributes that are mass assignable.
*
* @var array
*/
protected $fillable = [
'name',
'ip',
'mac',
'last_seen',
'role',
'credits',
'email',
'server_limit',
'password',
'pterodactyl_id',
'discord_verified_at',
'avatar',
'suspended',
'referral_code',
];
/**
* The attributes that should be hidden for arrays.
*
* @var array
*/
protected $hidden = [
'password',
'remember_token',
];
/**
* The attributes that should be cast to native types.
*
* @var array
*/
protected $casts = [
'email_verified_at' => 'datetime',
'last_seen' => 'datetime',
'credits' => 'float',
'server_limit' => 'float',
];
public static function boot()
{
parent::boot();
static::created(function (User $user) {
$user->notify(new WelcomeMessage($user));
});
static::deleting(function (User $user) {
$user->servers()->chunk(10, function ($servers) {
foreach ($servers as $server) {
$server->delete();
}
});
$user->payments()->chunk(10, function ($payments) {
foreach ($payments as $payment) {
$payment->delete();
}
});
$user->tickets()->chunk(10, function ($tickets) {
foreach ($tickets as $ticket) {
$ticket->delete();
}
});
$user->ticketBlackList()->delete();
$user->vouchers()->detach();
$user->discordUser()->delete();
Pterodactyl::client()->delete("/application/users/{$user->pterodactyl_id}");
});
}
/**
* @return HasMany
*/
public function servers()
{
return $this->hasMany(Server::class);
}
/**
* @return HasMany
*/
public function payments()
{
return $this->hasMany(Payment::class);
}
/**
* @return HasMany
*/
public function tickets()
{
return $this->hasMany(Ticket::class);
}
/**
* @return HasMany
*/
public function ticketBlackList()
{
return $this->hasMany(TicketBlacklist::class);
}
/**
* @return BelongsToMany
*/
public function vouchers()
{
return $this->belongsToMany(Voucher::class);
}
/**
* @return HasOne
*/
public function discordUser()
{
return $this->hasOne(DiscordUser::class);
}
public function sendEmailVerificationNotification()
{
$this->notify(new QueuedVerifyEmail);
}
/**
* @return string
*/
public function credits()
{
return number_format($this->credits, 2, '.', '');
}
/**
* @return bool
*/
public function isSuspended()
{
return $this->suspended;
}
/**
* @throws Exception
*/
public function suspend()
{
foreach ($this->servers as $server) {
$server->suspend();
}
$this->update([
'suspended' => true,
]);
return $this;
}
/**
* @throws Exception
*/
public function unSuspend()
{
foreach ($this->getServersWithProduct() as $server) {
if ($this->credits >= $server->product->getHourlyPrice()) {
$server->unSuspend();
}
}
$this->update([
'suspended' => false,
]);
return $this;
}
private function getServersWithProduct()
{
return $this->servers()
->with('product')
->get();
}
/**
* @return string
*/
public function getAvatar()
{
//TODO loading the images to confirm they exist is causing to much load time. alternative has to be found :) maybe onerror tag on the <img tags>
// if ($this->discordUser()->exists()) {
// if(@getimagesize($this->discordUser->getAvatar())) {
// $avatar = $this->discordUser->getAvatar();
// } else {
// $avatar = "https://www.gravatar.com/avatar/" . md5(strtolower(trim($this->email)));
// }
// } else {
// $avatar = "https://www.gravatar.com/avatar/" . md5(strtolower(trim($this->email)));
// }
return 'https://www.gravatar.com/avatar/'.md5(strtolower(trim($this->email)));
}
/**
* @return string
*/
public function creditUsage()
{
$usage = 0;
foreach ($this->getServersWithProduct() as $server) {
$usage += $server->product->price;
}
return number_format($usage, 2, '.', '');
}
/**
* @return array|string|string[]
*/
public function getVerifiedStatus()
{
$status = '';
if ($this->hasVerifiedEmail()) {
$status .= 'email ';
}
if ($this->discordUser()->exists()) {
$status .= 'discord';
}
$status = str_replace(' ', '/', $status);
return $status;
}
public function verifyEmail()
{
$this->forceFill([
'email_verified_at' => now(),
])->save();
}
public function reVerifyEmail()
{
$this->forceFill([
'email_verified_at' => null,
])->save();
}
public function getActivitylogOptions(): LogOptions
{
return LogOptions::defaults()
-> logOnly(['role', 'name', 'server_limit', 'pterodactyl_id', 'email'])
-> logOnlyDirty()
-> dontSubmitEmptyLogs();
}
}