Lae/app/Models/Host.php

472 lines
12 KiB
PHP
Raw Permalink Normal View History

2022-08-16 10:44:16 +00:00
<?php
namespace App\Models;
2023-03-07 16:45:29 +00:00
use App\Exceptions\User\BalanceNotEnoughException;
2023-01-13 14:13:46 +00:00
use App\Jobs\Host\HostJob;
use App\Jobs\Host\UpdateOrDeleteHostJob;
2022-11-06 11:28:22 +00:00
use GeneaLabs\LaravelModelCaching\Traits\Cachable;
2022-11-20 14:35:53 +00:00
use Illuminate\Database\Eloquent\Collection;
2022-11-06 11:28:22 +00:00
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo as BelongsToAlias;
use Illuminate\Support\Facades\Cache;
2022-08-16 10:44:16 +00:00
class Host extends Model
{
2022-12-12 03:55:08 +00:00
use Cachable;
2022-08-16 10:44:16 +00:00
protected $table = 'hosts';
protected $fillable = [
'name',
'module_id',
'user_id',
'price',
2023-02-12 18:22:12 +00:00
'managed_price',
2022-08-16 10:44:16 +00:00
'configuration',
'status',
2022-11-18 09:16:30 +00:00
'suspended_at',
2023-03-07 16:45:29 +00:00
'trial_ends_at',
'billing_cycle',
'cancel_at_period_end',
'last_paid',
'last_paid_at',
'expired_at',
2022-08-16 10:44:16 +00:00
];
protected $casts = [
2023-01-14 10:35:13 +00:00
'price' => 'decimal:2',
2023-03-07 16:45:29 +00:00
'last_paid' => 'decimal:2',
2023-01-14 10:35:13 +00:00
'managed_price' => 'decimal:2',
2023-02-12 18:22:12 +00:00
'configuration' => 'array',
'suspended_at' => 'datetime',
2023-03-07 16:45:29 +00:00
'last_paid_at' => 'datetime',
'trial_ends_at' => 'datetime',
'expired_at' => 'datetime',
2022-08-16 10:44:16 +00:00
];
2023-01-10 13:42:27 +00:00
/** @noinspection PhpUndefinedMethodInspection */
2022-11-20 14:35:53 +00:00
public function getUserHosts($user_id = null): array|Collection
{
2022-11-06 11:28:22 +00:00
return $this->where('user_id', $user_id)->with('module', function ($query) {
$query->select(['id', 'name']);
})->get();
2022-09-10 04:03:20 +00:00
}
2023-02-19 16:10:18 +00:00
public function module(): BelongsToAlias
{
2023-02-19 16:10:18 +00:00
return $this->belongsTo(Module::class);
2022-08-16 10:44:16 +00:00
}
2023-02-19 16:10:18 +00:00
public function scopeActive($query)
{
2023-02-19 16:10:18 +00:00
return $query->whereIn('status', ['running', 'stopped']);
2022-08-16 10:44:16 +00:00
}
2022-11-20 13:10:46 +00:00
// public function workOrders(): HasManyAlias
// {
// return $this->hasMany(WorkOrder::class);
// }
2022-08-16 10:44:16 +00:00
2023-02-12 18:22:12 +00:00
public function scopeDraft($query)
{
return $query->where('status', 'draft');
}
public function scopeExpiring($query)
{
return $query->where('status', 'running')->where('next_due_at', '<=', now()->addDays(7));
}
public function scopeSuspended($query)
{
return $query->where('status', 'suspended');
}
2022-11-20 14:35:53 +00:00
public function scopeThisUser($query, $module = null)
{
if ($module) {
return $query->where('user_id', auth()->id())->where('module_id', $module);
} else {
return $query->where('user_id', auth()->id());
}
}
2022-08-19 15:27:57 +00:00
2023-02-12 18:22:12 +00:00
public function isDraft(): bool
{
return $this->status === 'draft';
}
public function isRunning(): bool
{
return $this->status === 'running';
}
public function isStopped(): bool
{
return $this->status === 'stopped';
}
2023-02-13 07:26:33 +00:00
public function isUnavailable(): bool
{
return $this->status === 'unavailable';
}
2023-02-12 18:46:32 +00:00
public function getPrice(): float
2023-02-12 18:22:12 +00:00
{
2023-02-12 18:46:32 +00:00
return $this->managed_price ?? $this->price;
}
2023-02-12 19:12:26 +00:00
public function isSuspended(): bool
{
return $this->status === 'suspended';
}
2023-03-07 16:45:29 +00:00
public function isExpired(): bool
{
return $this->expired_at && $this->expired_at->isPast();
}
public function safeDelete(): bool
{
2023-03-07 16:45:29 +00:00
if ($this->isHourly()) {
// 如果创建时间大于 1 小时
if ($this->created_at->diffInHours(now()) > 1) {
// 如果当前时间比扣费时间小,则说明没有扣费。执行扣费。
if (now()->minute < $this->minute_at) {
$this->cost();
}
}
} elseif ($this->isMonthly() && $this->last_paid && ! $this->isExpired()) {
// 根据扣费时间,计算出退款金额
$refund = $this->getRefundAmount();
if ($refund) {
// 如果有退款金额,则退款
$this->module?->reduce($refund, 'module_balance', '主机 '.$this->name.' 退款。', [
'host_id' => $this->id,
'module_id' => $this->module_id,
]);
$this->user->charge($refund, 'balance', '主机 '.$this->name.' 退款。', [
'host_id' => $this->id,
'module_id' => $this->module_id,
]);
// 退款后,更新扣费时间
$this->update([
'last_paid_at' => null,
'last_paid' => 0,
]);
2022-11-23 02:39:35 +00:00
}
2022-11-22 11:31:41 +00:00
}
2022-12-28 13:19:40 +00:00
dispatch(new HostJob($this, 'delete'));
2023-01-30 16:14:07 +00:00
return true;
}
2023-03-07 16:45:29 +00:00
public function getRefundAmount(): string|null
{
if (! $this->last_paid_at) {
return null;
}
// 如果是月付,则按比例
$days = $this->last_paid_at->daysInMonth;
// 本月已经过的天数
$passed_days = $this->last_paid_at->day;
// 本月还剩下的天数
$left_days = $days - $passed_days;
// 计算
return bcmul($this->last_paid, bcdiv($left_days, $days, 2), 2);
}
public function isTrial(): bool
{
return $this->trial_ends_at !== null;
}
public function isMonthly(): bool
{
return $this->billing_cycle === 'monthly';
}
public function isHourly(): bool
{
return $this->billing_cycle === 'hourly';
}
public function isNextMonthCancel(): bool
{
2023-03-07 16:53:59 +00:00
if ($this->isHourly()) {
return false;
}
if ($this->isMonthly()) {
return $this->cancel_at_period_end;
}
return false;
2023-03-07 16:45:29 +00:00
}
2023-02-13 09:05:07 +00:00
public function cost(
string $amount = null, $auto = true, $description = null
): bool {
2023-03-08 00:25:52 +00:00
if ($this->isTrial() && ! $this->trial_ends_at->isPast()) {
return true;
}
2023-03-07 16:45:29 +00:00
2022-11-19 14:42:47 +00:00
$this->load('user');
2022-11-26 13:52:30 +00:00
$user = $this->user;
$user->load('user_group');
$user_group = $user->user_group;
if ($user_group) {
if ($user_group->exempt) {
return true;
}
}
2022-11-19 14:42:47 +00:00
$real_price = $amount ?? $this->price;
2023-02-12 19:12:26 +00:00
if (! $amount) {
2022-11-19 14:42:47 +00:00
if ($this->managed_price) {
$real_price = $this->managed_price;
}
}
2022-11-26 14:01:24 +00:00
$append_description = '';
2022-11-26 13:52:30 +00:00
if ($user_group) {
if ($user_group->discount !== 100 && $user_group->discount !== null) {
2023-02-12 18:22:12 +00:00
$real_price = $user_group->getCostPrice($real_price);
2023-01-16 20:36:43 +00:00
2023-02-12 19:12:26 +00:00
$append_description = ' (折扣 '.$user_group->discount.'%)';
2022-11-26 13:52:30 +00:00
}
}
2023-03-07 16:45:29 +00:00
if ($auto && $this->isHourly()) {
2022-11-20 12:34:13 +00:00
// 获取本月天数
$days = now()->daysInMonth;
// 本月每天的每小时的价格
2023-01-16 20:36:43 +00:00
// 使用 bcmath 函数,解决浮点数计算精度问题
2023-01-17 16:07:31 +00:00
$real_price = bcdiv($real_price, $days, 4);
$real_price = bcdiv($real_price, 24, 4);
2022-11-20 12:34:13 +00:00
}
if ($real_price == 0) {
2023-02-12 19:12:26 +00:00
echo '价格为 0不扣费'.PHP_EOL;
2023-01-30 16:14:07 +00:00
2022-11-19 14:42:47 +00:00
return true;
}
2022-12-05 05:00:48 +00:00
// 如果太小,则重置为 0.0001
2022-11-24 00:13:38 +00:00
if ($real_price < 0.0001) {
2022-11-24 01:26:33 +00:00
$real_price = 0.0001;
2022-11-23 13:27:38 +00:00
}
2023-01-16 20:36:43 +00:00
$real_price = bcdiv($real_price, 1, 4);
2022-11-19 06:04:42 +00:00
$month = now()->month;
2023-02-12 19:12:26 +00:00
$month_cache_key = 'user_'.$this->user_id.'_month_'.$month.'_hosts_balances';
2022-11-23 04:04:56 +00:00
$hosts_balances = Cache::get($month_cache_key, []);
2022-11-19 06:04:42 +00:00
2022-11-19 14:42:47 +00:00
// 统计 Host 消耗的 Balance
2022-11-23 04:04:56 +00:00
if (isset($hosts_balances[$this->id])) {
$hosts_balances[$this->id] += $real_price;
2022-11-19 06:04:42 +00:00
} else {
2022-11-23 04:04:56 +00:00
$hosts_balances[$this->id] = $real_price;
2022-11-19 06:04:42 +00:00
}
2023-01-16 20:36:43 +00:00
$hosts_balances[$this->id] = bcdiv($hosts_balances[$this->id], 1, 4);
2022-11-23 04:04:56 +00:00
Cache::put($month_cache_key, $hosts_balances, 604800);
2022-11-19 06:04:42 +00:00
2023-03-07 16:45:29 +00:00
$description = '主机: '.$this->name.', '.$description;
if ($auto && $this->isHourly()) {
$description .= '小时计费。';
} elseif ($auto && $this->isMonthly()) {
$description .= '月度计费。';
} else {
$description .= '扣费。';
2023-02-10 18:18:34 +00:00
}
2022-11-19 14:42:47 +00:00
2023-03-07 16:45:29 +00:00
if ($this->isTrial() && $this->trial_ends_at->isPast()) {
$description .= '试用已过期。';
$this->trial_ends_at = null;
2022-11-19 14:42:47 +00:00
}
2022-11-19 06:04:42 +00:00
2022-11-26 14:01:24 +00:00
if ($append_description) {
$description .= $append_description;
}
2023-01-16 20:36:43 +00:00
$data = [
'host_id' => $this->id,
'module_id' => $this->module_id,
];
2023-03-07 16:45:29 +00:00
$this->last_paid = $real_price;
$this->last_paid_at = now();
if ($this->isMonthly()) {
$this->expired_at = now()->addMonth();
}
try {
$left = $user->reduce($real_price, $description, ! $this->isHourly(), $data)->user_remain;
} catch (BalanceNotEnoughException) {
$this->changeStatus('suspended');
return false;
}
2022-11-19 06:04:42 +00:00
2023-01-17 17:09:15 +00:00
$this->addLog($real_price);
2022-11-19 06:04:42 +00:00
if ($left < 0) {
2023-02-13 09:05:07 +00:00
$this->changeStatus('suspended');
2022-11-19 06:04:42 +00:00
}
2023-03-07 16:45:29 +00:00
$this->save();
2022-11-19 06:04:42 +00:00
return true;
}
2022-11-20 12:34:13 +00:00
2023-02-19 16:10:18 +00:00
public function addLog(string $amount = '0'): bool
{
if ($amount === '0') {
return false;
}
/** 统计收益开始 */
$current_month = now()->month;
$current_year = now()->year;
$cache_key = 'module_earning_'.$this->module_id;
// 应支付的提成
$commission = config('settings.billing.commission');
$should_amount = bcmul($amount, $commission, 4);
// 应得的余额
$should_balance = bcsub($amount, $should_amount, 4);
// 如果太小,则重置为 0.0001
if ($should_balance < 0.0001) {
$should_balance = 0.0001;
}
$earnings = Cache::get($cache_key, []);
if (! isset($earnings[$current_year])) {
$earnings[$current_year] = [];
}
if (isset($earnings[$current_year][$current_month])) {
$earnings[$current_year][$current_month]['balance'] = bcadd($earnings[$current_year][$current_month]['balance'], $amount, 4);
$earnings[$current_year][$current_month]['should_balance'] = bcadd($earnings[$current_year][$current_month]['should_balance'], $should_balance, 4);
} else {
$earnings[$current_year][$current_month] = [
'balance' => $amount,
// 应得(交了手续费)
'should_balance' => $should_balance,
];
}
// 删除 前 3 年的数据
if (count($earnings) > 3) {
$earnings = array_slice($earnings, -3, 3, true);
}
$this->module->charge($amount, 'balance', null);
// 保存 1 年
Cache::forever($cache_key, $earnings);
/** 统计收益结束 */
return true;
}
2023-02-13 09:05:07 +00:00
public function changeStatus(
string $status
): bool {
$user = auth()->guard('sanctum')->user() ?? auth()->guard('web')->user();
2022-11-20 12:34:13 +00:00
2023-02-13 09:05:07 +00:00
if ($user) {
2023-02-13 07:50:19 +00:00
if ($this->isPending() || $this->isOverdue() || $this->status === 'locked' || $this->status === 'unavailable') {
return false;
}
2023-02-13 09:05:07 +00:00
2023-03-07 16:45:29 +00:00
if ($this->isMonthly()) {
if (! $user->hasBalance($this->price)) {
return false;
}
} elseif (! $user->hasBalance('0.5')) {
2023-02-13 09:05:07 +00:00
return false;
}
2023-02-13 07:50:19 +00:00
}
if ($status === 'running') {
return $this->run();
2023-02-19 16:10:18 +00:00
} elseif (($status === 'suspended' || $status === 'suspend')) {
2023-02-13 07:50:19 +00:00
return $this->suspend();
} elseif ($status === 'stopped') {
return $this->stop();
}
return false;
}
2023-02-19 16:10:18 +00:00
public function user(): BelongsToAlias
{
return $this->belongsTo(User::class);
}
2023-02-13 07:50:19 +00:00
public function isPending(): bool
{
return $this->status === 'pending';
}
public function isOverdue(): bool
{
return now()->gt($this->next_due_at);
2022-11-20 12:34:13 +00:00
}
2023-01-13 14:14:34 +00:00
2023-02-19 16:10:18 +00:00
public function run(): bool
{
$this->update([
'status' => 'running',
]);
return true;
}
2023-02-12 18:46:32 +00:00
public function suspend(): bool
2023-01-13 14:14:34 +00:00
{
2023-02-12 18:46:32 +00:00
$this->update([
'status' => 'suspended',
]);
2023-01-13 14:14:34 +00:00
return true;
}
2023-02-13 07:50:19 +00:00
public function stop(): bool
{
$this->update([
'status' => 'stopped',
]);
return true;
}
public function updateOrDelete(): bool
{
dispatch(new UpdateOrDeleteHostJob($this));
return true;
}
2022-08-16 10:44:16 +00:00
}