完善百家乐买单补偿自动领取与聊天室播报

This commit is contained in:
2026-04-11 23:43:07 +08:00
parent e43dceab2c
commit f4a632a9c1
3 changed files with 268 additions and 1 deletions

View File

@@ -0,0 +1,97 @@
<?php
/**
* 文件功能AI小班长自动领取百家乐买单活动补偿任务
*
* 当买单活动进入“可领取”状态后,异步为 AI小班长检查并领取
* 本次活动中累计的补偿金币,确保金币入账和流水日志统一走正式服务。
*/
namespace App\Jobs;
use App\Models\BaccaratLossCoverEvent;
use App\Models\BaccaratLossCoverRecord;
use App\Models\User;
use App\Services\BaccaratLossCoverService;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Queue\Queueable;
use Illuminate\Support\Facades\Log;
class AiClaimBaccaratLossCoverJob implements ShouldQueue
{
use Queueable;
/**
* 最大重试次数。
*/
public int $tries = 3;
/**
* @param int $eventId 需要自动领取补偿的活动 ID
*/
public function __construct(
public readonly int $eventId,
) {}
/**
* 执行 AI 小班长自动领取补偿逻辑。
*/
public function handle(BaccaratLossCoverService $lossCoverService): void
{
$event = BaccaratLossCoverEvent::query()->find($this->eventId);
if (! $event) {
Log::channel('daily')->warning('AI小班长自动领取买单补偿失败活动不存在', [
'event_id' => $this->eventId,
]);
return;
}
// 只有活动进入可领取状态后,才允许自动发起补偿领取。
if (! $event->isClaimable()) {
Log::channel('daily')->info('AI小班长自动领取买单补偿跳过活动暂不可领取', [
'event_id' => $event->id,
'status' => $event->status,
]);
return;
}
$aiUser = User::query()->where('username', 'AI小班长')->first();
if (! $aiUser) {
Log::channel('daily')->warning('AI小班长自动领取买单补偿失败未找到 AI 用户', [
'event_id' => $event->id,
]);
return;
}
$record = BaccaratLossCoverRecord::query()
->where('event_id', $event->id)
->where('user_id', $aiUser->id)
->first();
// 没有补偿记录或本次没有待领取金额时,直接记日志后结束。
if (! $record || $record->compensation_amount <= 0 || $record->claim_status !== 'pending') {
Log::channel('daily')->info('AI小班长自动领取买单补偿跳过暂无待领取补偿', [
'event_id' => $event->id,
'user_id' => $aiUser->id,
'claim_status' => $record?->claim_status,
'compensation_amount' => $record?->compensation_amount,
]);
return;
}
$claimResult = $lossCoverService->claim($event, $aiUser);
// 统一记录自动领取结果,便于后续核对 AI 补偿发放情况。
Log::channel('daily')->info('AI小班长自动领取买单补偿结果', [
'event_id' => $event->id,
'user_id' => $aiUser->id,
'ok' => $claimResult['ok'],
'message' => $claimResult['message'],
'amount' => $claimResult['amount'] ?? 0,
]);
}
}

View File

@@ -11,6 +11,7 @@ namespace App\Services;
use App\Enums\CurrencySource;
use App\Events\MessageSent;
use App\Jobs\AiClaimBaccaratLossCoverJob;
use App\Jobs\SaveMessageJob;
use App\Models\BaccaratBet;
use App\Models\BaccaratLossCoverEvent;
@@ -198,7 +199,7 @@ class BaccaratLossCoverService
return ['ok' => false, 'message' => '当前活动暂未开放领取,或已超过领取时间。'];
}
return DB::transaction(function () use ($event, $user): array {
$result = DB::transaction(function () use ($event, $user): array {
$record = BaccaratLossCoverRecord::query()
->where('event_id', $event->id)
->where('user_id', $user->id)
@@ -247,6 +248,13 @@ class BaccaratLossCoverService
'amount' => $amount,
];
});
// 领取成功后,需要向聊天室广播一条同款百家乐风格消息,方便其他人快速领取。
if (($result['ok'] ?? false) === true && isset($result['amount'])) {
$this->pushClaimMessage($event->fresh(), $user, (int) $result['amount']);
}
return $result;
}
/**
@@ -396,6 +404,9 @@ class BaccaratLossCoverService
$this->pushRoomMessage($content, '#7c3aed');
$event->update(['ended_notice_sent_at' => now()]);
// 活动开放领取后,为 AI小班长补发一次自动领取任务。
$this->dispatchAiAutoClaimJob($event->fresh());
}
/**
@@ -419,4 +430,66 @@ class BaccaratLossCoverService
broadcast(new MessageSent(1, $message));
SaveMessageJob::dispatch($message);
}
/**
* 在用户领取补偿成功后,向聊天室广播领取播报。
*/
private function pushClaimMessage(?BaccaratLossCoverEvent $event, User $user, int $amount): void
{
if (! $event) {
return;
}
$formattedAmount = number_format($amount);
$button = $event->status === 'claimable'
? ' <button onclick="claimBaccaratLossCover('.$event->id.')" style="margin-left:8px;padding:3px 12px;background:#16a34a;color:#fff;border:none;border-radius:12px;cursor:pointer;font-size:12px;font-weight:bold;">领取补偿</button>'
: '';
// 领取成功的公屏格式复用百家乐参与播报风格,保证聊天室感知一致。
$content = "🌟 🎲 <b>{$user->username}</b> 领取了 <b>{$formattedAmount}</b> 金币补偿!✨{$button}";
$message = [
'id' => $this->chatState->nextMessageId(1),
'room_id' => 1,
'from_user' => '系统传音',
'to_user' => '大家',
'content' => $content,
'is_secret' => false,
'font_color' => '#d97706',
'action' => '',
'sent_at' => now()->toDateTimeString(),
];
$this->chatState->pushMessage(1, $message);
broadcast(new MessageSent(1, $message));
SaveMessageJob::dispatch($message);
}
/**
* 在活动可领取后,为 AI小班长派发自动领取补偿任务。
*/
private function dispatchAiAutoClaimJob(?BaccaratLossCoverEvent $event): void
{
if (! $event || $event->status !== 'claimable') {
return;
}
$aiUserId = User::query()->where('username', 'AI小班长')->value('id');
if (! $aiUserId) {
return;
}
$hasPendingRecord = BaccaratLossCoverRecord::query()
->where('event_id', $event->id)
->where('user_id', $aiUserId)
->where('claim_status', 'pending')
->where('compensation_amount', '>', 0)
->exists();
// 只有 AI小班长确实存在待领取补偿时才派发自动领取任务。
if (! $hasPendingRecord) {
return;
}
AiClaimBaccaratLossCoverJob::dispatch($event->id);
}
}

View File

@@ -10,11 +10,15 @@
namespace Tests\Feature;
use App\Enums\CurrencySource;
use App\Events\MessageSent;
use App\Jobs\AiClaimBaccaratLossCoverJob;
use App\Jobs\SaveMessageJob;
use App\Models\BaccaratLossCoverEvent;
use App\Models\BaccaratLossCoverRecord;
use App\Models\BaccaratRound;
use App\Models\GameConfig;
use App\Models\User;
use App\Services\BaccaratLossCoverService;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Illuminate\Support\Facades\Event;
use Illuminate\Support\Facades\Queue;
@@ -133,6 +137,9 @@ class BaccaratLossCoverControllerTest extends TestCase
*/
public function test_user_can_claim_baccarat_loss_cover_and_currency_log_is_written(): void
{
Event::fake([MessageSent::class]);
Queue::fake([SaveMessageJob::class]);
$user = User::factory()->create(['jjb' => 200]);
$event = BaccaratLossCoverEvent::factory()->create([
'status' => 'claimable',
@@ -171,6 +178,12 @@ class BaccaratLossCoverControllerTest extends TestCase
'amount' => 300,
'source' => CurrencySource::BACCARAT_LOSS_COVER_CLAIM->value,
]);
Event::assertDispatched(MessageSent::class);
Queue::assertPushed(SaveMessageJob::class, function (SaveMessageJob $job): bool {
return str_contains($job->messageData['content'], '领取了 <b>300</b> 金币补偿')
&& str_contains($job->messageData['content'], 'claimBaccaratLossCover');
});
}
/**
@@ -200,4 +213,88 @@ class BaccaratLossCoverControllerTest extends TestCase
$response->assertJsonPath('events.0.my_record.claim_status', 'claimed');
$response->assertJsonPath('events.0.my_record.claimed_amount', 400);
}
/**
* 验证活动进入可领取状态后会为 AI 小班长派发自动领取任务。
*/
public function test_claimable_event_dispatches_ai_auto_claim_job(): void
{
Queue::fake();
$aiUser = User::factory()->create([
'username' => 'AI小班长',
'jjb' => 1000,
]);
$event = BaccaratLossCoverEvent::factory()->create([
'status' => 'active',
'starts_at' => now()->subHour(),
'ends_at' => now()->subMinute(),
'claim_deadline_at' => now()->addHours(12),
]);
BaccaratLossCoverRecord::factory()->create([
'event_id' => $event->id,
'user_id' => $aiUser->id,
'compensation_amount' => 260,
'total_loss_amount' => 260,
'claim_status' => 'pending',
]);
app(BaccaratLossCoverService::class)->closeDueActiveEvents();
Queue::assertPushed(AiClaimBaccaratLossCoverJob::class, function (AiClaimBaccaratLossCoverJob $job) use ($event): bool {
return $job->eventId === $event->id;
});
$this->assertDatabaseHas('baccarat_loss_cover_events', [
'id' => $event->id,
'status' => 'claimable',
]);
}
/**
* 验证 AI 小班长自动领取任务会发放补偿并写入金币流水。
*/
public function test_ai_auto_claim_job_claims_compensation_and_writes_currency_log(): void
{
$aiUser = User::factory()->create([
'username' => 'AI小班长',
'jjb' => 200,
]);
$event = BaccaratLossCoverEvent::factory()->create([
'status' => 'claimable',
'claim_deadline_at' => now()->addHours(12),
'total_loss_amount' => 320,
]);
BaccaratLossCoverRecord::factory()->create([
'event_id' => $event->id,
'user_id' => $aiUser->id,
'total_bet_amount' => 600,
'total_loss_amount' => 320,
'compensation_amount' => 320,
'claim_status' => 'pending',
]);
$job = new AiClaimBaccaratLossCoverJob($event->id);
$job->handle(app(BaccaratLossCoverService::class));
$this->assertSame(520, (int) $aiUser->fresh()->jjb);
$this->assertDatabaseHas('baccarat_loss_cover_records', [
'event_id' => $event->id,
'user_id' => $aiUser->id,
'claim_status' => 'claimed',
'claimed_amount' => 320,
]);
$this->assertDatabaseHas('user_currency_logs', [
'user_id' => $aiUser->id,
'currency' => 'gold',
'amount' => 320,
'source' => CurrencySource::BACCARAT_LOSS_COVER_CLAIM->value,
]);
}
}