324 lines
9.2 KiB
PHP
324 lines
9.2 KiB
PHP
<?php
|
|
|
|
namespace App\Models;
|
|
|
|
use App\Exceptions\User\BalanceNotEnoughException;
|
|
use Carbon\Exceptions\InvalidFormatException;
|
|
use GeneaLabs\LaravelModelCaching\CachedBuilder;
|
|
use GeneaLabs\LaravelModelCaching\Traits\Cachable;
|
|
use Illuminate\Contracts\Auth\MustVerifyEmail;
|
|
use Illuminate\Contracts\Encryption\DecryptException;
|
|
use Illuminate\Database\Eloquent\Builder;
|
|
use Illuminate\Database\Eloquent\Casts\Attribute;
|
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
|
use Illuminate\Database\Eloquent\Prunable;
|
|
use Illuminate\Database\Eloquent\Relations\BelongsTo;
|
|
use Illuminate\Database\Eloquent\Relations\HasMany;
|
|
use Illuminate\Database\Eloquent\SoftDeletes;
|
|
use Illuminate\Foundation\Auth\User as Authenticatable;
|
|
use Illuminate\Notifications\Notifiable;
|
|
use Illuminate\Support\Arr;
|
|
use Illuminate\Support\Facades\Cache;
|
|
use Illuminate\Support\Facades\Crypt;
|
|
use Illuminate\Support\Str;
|
|
use Laravel\Sanctum\HasApiTokens;
|
|
|
|
class User extends Authenticatable implements MustVerifyEmail
|
|
{
|
|
use HasApiTokens, HasFactory, Notifiable, SoftDeletes, Prunable, Cachable;
|
|
|
|
public array $publics = [
|
|
'id',
|
|
'uuid',
|
|
'name',
|
|
'email',
|
|
'real_name',
|
|
'balance',
|
|
'user_group_id'
|
|
];
|
|
|
|
/**
|
|
* The attributes that are mass assignable.
|
|
*
|
|
* @var array<int, string>
|
|
*/
|
|
protected $fillable = [
|
|
'uuid',
|
|
'name',
|
|
'email',
|
|
'password',
|
|
];
|
|
|
|
/**
|
|
* The attributes that should be hidden for serialization.
|
|
*
|
|
* @var array<int, string>
|
|
*/
|
|
protected $hidden = [
|
|
'password',
|
|
'remember_token',
|
|
'real_name',
|
|
'id_card',
|
|
];
|
|
|
|
protected $casts = [
|
|
'email_verified_at' => 'datetime',
|
|
'real_name_verified_at' => 'datetime',
|
|
'balance' => 'decimal:4',
|
|
'banned_at' => 'datetime',
|
|
'birthday_at' => 'date:Y-m-d',
|
|
];
|
|
|
|
protected $dates = [
|
|
'email_verified_at',
|
|
'real_name_verified_at',
|
|
'banned_at',
|
|
'birthday_at',
|
|
];
|
|
|
|
protected static function boot()
|
|
{
|
|
parent::boot();
|
|
|
|
static::creating(function (self $user) {
|
|
$user->email_md5 = md5($user->email);
|
|
$user->uuid = Str::uuid();
|
|
});
|
|
|
|
static::updating(function (self $user) {
|
|
if ($user->isDirty('banned_at')) {
|
|
if ($user->banned_at) {
|
|
$user->tokens()->delete();
|
|
$user->hosts()->update(['status' => 'suspended', 'suspended_at' => now()]);
|
|
} else {
|
|
$user->hosts()->update(['status' => 'stopped']);
|
|
}
|
|
}
|
|
|
|
if ($user->isDirty('email')) {
|
|
$user->email_md5 = md5($user->email);
|
|
}
|
|
|
|
if ($user->isDirty('id_card')) {
|
|
$user->id_card = Crypt::encryptString($user->id_card);
|
|
}
|
|
|
|
if ($user->isDirty('id_card') || $user->isDirty('real_name')) {
|
|
if (empty($user->id_card) || empty($user->real_name)) {
|
|
$user->real_name_verified_at = null;
|
|
} else {
|
|
$user->real_name_verified_at = now();
|
|
|
|
// 更新生日
|
|
try {
|
|
$user->birthday_at = $user->getBirthdayFromIdCard();
|
|
} catch (InvalidFormatException) {
|
|
$user->birthday_at = null;
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
static::deleting(function (self $user) {
|
|
$user->tokens()->delete();
|
|
$user->hosts()->update(['status' => 'suspended', 'suspended_at' => now()]);
|
|
});
|
|
}
|
|
|
|
public function hosts(): HasMany
|
|
{
|
|
return $this->hasMany(Host::class);
|
|
}
|
|
|
|
private function getBirthdayFromIdCard(): string
|
|
{
|
|
$idCard = $this->id_card;
|
|
|
|
$bir = substr($idCard, 6, 8);
|
|
$year = (int) substr($bir, 0, 4);
|
|
$month = (int) substr($bir, 4, 2);
|
|
$day = (int) substr($bir, 6, 2);
|
|
|
|
return $year.'-'.$month.'-'.$day;
|
|
}
|
|
|
|
public function hasBalance(string $amount = '0.01'): bool
|
|
{
|
|
return bccomp($this->balance, $amount, 4) >= 0;
|
|
}
|
|
|
|
public function isAdult(): bool
|
|
{
|
|
// 如果 birthday_at 为空,那么就返回 false
|
|
return $this->birthday_at?->diffInYears(now()) >= 18;
|
|
}
|
|
|
|
public function isRealNamed(): bool
|
|
{
|
|
return $this->real_name_verified_at !== null;
|
|
}
|
|
|
|
public function user_group(): BelongsTo
|
|
{
|
|
return $this->belongsTo(UserGroup::class);
|
|
}
|
|
|
|
public function scopeBirthday(): Builder|CachedBuilder
|
|
{
|
|
/** @noinspection PhpUndefinedMethodInspection */
|
|
return $this->select(['id', 'name', 'birthday_at', 'email_md5', 'created_at'])->whereMonth('birthday_at', now()->month)
|
|
->whereDay('birthday_at', now()->day)->whereNull('banned_at');
|
|
}
|
|
|
|
public function selectPublic(): self|Builder|CachedBuilder
|
|
{
|
|
// 仅需选择公开的
|
|
return $this->select($this->publics);
|
|
}
|
|
|
|
public function getOnlyPublic($appened_excepts = [], $display = []): array
|
|
{
|
|
if ($display) {
|
|
$this->publics = array_merge($this->publics, $display);
|
|
}
|
|
if ($appened_excepts) {
|
|
$this->publics = array_diff($this->publics, $appened_excepts);
|
|
}
|
|
|
|
return Arr::only($this->toArray(), $this->publics);
|
|
}
|
|
|
|
public function prunable(): self|Builder|CachedBuilder
|
|
{
|
|
return static::where('deleted_at', '<=', now()->subWeek());
|
|
}
|
|
|
|
public function startTransfer(self $to, string $amount, string|null $description)
|
|
{
|
|
$description_from = "转账给 $to->name($to->email)";
|
|
$description_to = "收到 $this->name($this->email) 的转账";
|
|
|
|
if ($description) {
|
|
$description_from .= ",备注:$description";
|
|
$description_to .= ",备注:$description";
|
|
}
|
|
|
|
$this->reduce($amount, $description_from, true);
|
|
|
|
$to->charge($amount, 'transfer', $description_to);
|
|
|
|
return $this->balance;
|
|
}
|
|
|
|
/**
|
|
* 扣除费用
|
|
*
|
|
* @param string|null $amount
|
|
* @param string $description
|
|
* @param bool $fail
|
|
* @param array $options
|
|
* @return string
|
|
*/
|
|
public function reduce(string|null $amount = '0', string $description = '消费', bool $fail = false, array $options = []): string
|
|
{
|
|
if ($amount === null || $amount === '') {
|
|
return $this->balance;
|
|
}
|
|
|
|
Cache::lock('user_balance_'.$this->id, 10)->block(10, function () use ($amount, $fail, $description, $options) {
|
|
$this->refresh();
|
|
|
|
if ($this->balance < $amount) {
|
|
if ($fail) {
|
|
throw new BalanceNotEnoughException();
|
|
}
|
|
}
|
|
|
|
$this->balance = bcsub($this->balance, $amount, 4);
|
|
$this->save();
|
|
|
|
$data = [
|
|
'user_id' => $this->id,
|
|
'amount' => $amount,
|
|
'description' => $description,
|
|
'payment' => 'balance',
|
|
'type' => 'payout',
|
|
];
|
|
|
|
if ($options) {
|
|
$data = array_merge($data, $options);
|
|
}
|
|
|
|
(new Transaction)->create($data);
|
|
});
|
|
|
|
return $this->balance;
|
|
}
|
|
|
|
/**
|
|
* 增加余额
|
|
*
|
|
* @param string|null $amount
|
|
* @param string $payment
|
|
* @param string $description
|
|
* @param array $options
|
|
* @return string
|
|
*/
|
|
public function charge(string|null $amount = '0', string $payment = 'console', string $description = '充值', array $options = []): string
|
|
{
|
|
if ($amount === null || $amount === '') {
|
|
return $this->balance;
|
|
}
|
|
|
|
Cache::lock('user_balance_'.$this->id, 10)->block(10, function () use ($amount, $description, $payment, $options) {
|
|
$this->refresh();
|
|
$this->balance = bcadd($this->balance, $amount, 4);
|
|
$this->save();
|
|
|
|
$data = [
|
|
'user_id' => $this->id,
|
|
'amount' => $amount,
|
|
'payment' => $payment,
|
|
'description' => $description,
|
|
'type' => 'income',
|
|
];
|
|
|
|
if ($options) {
|
|
$data = array_merge($data, $options);
|
|
}
|
|
|
|
(new Transaction)->create($data);
|
|
|
|
if (isset($options['add_balances_log']) && $options['add_balances_log'] === true) {
|
|
(new Balance)->create([
|
|
'user_id' => $this->id,
|
|
'amount' => $amount,
|
|
'payment' => $payment,
|
|
'description' => $description,
|
|
'paid_at' => now(),
|
|
]);
|
|
}
|
|
});
|
|
|
|
return $this->balance;
|
|
}
|
|
|
|
/**
|
|
* 获取用户的身份证号
|
|
*
|
|
* @return Attribute
|
|
*/
|
|
protected function idCard(): Attribute
|
|
{
|
|
return Attribute::make(
|
|
function ($value) {
|
|
try {
|
|
return Crypt::decryptString($value);
|
|
} catch (DecryptException) {
|
|
return $value;
|
|
}
|
|
}
|
|
);
|
|
}
|
|
}
|