'testroom']); $user = User::factory()->create(); $response = $this->actingAs($user)->get(route('chat.room', $room->id)); $response->assertStatus(200); $response->assertViewIs('chat.frame'); // Assert user was added to room in redis $this->assertEquals(1, Redis::hexists("room:{$room->id}:users", $user->username)); } /** * 测试用户可以发送普通文本消息。 */ public function test_can_send_message() { $room = Room::create(['room_name' => 'test_send']); $user = User::factory()->create(); // 进房 $this->actingAs($user)->get(route('chat.room', $room->id)); $response = $this->actingAs($user)->postJson(route('chat.send', $room->id), [ 'to_user' => '大家', 'content' => '测试消息', 'is_secret' => false, 'font_color' => '#000000', 'action' => 'say', ]); $response->assertStatus(200); $response->assertJson(['status' => 'success']); // 查看 Redis 里的消息记录 $messages = Redis::lrange("room:{$room->id}:messages", 0, -1); $this->assertNotEmpty($messages); $found = false; foreach ($messages as $msgJson) { $msg = json_decode($msgJson, true); if ($msg['from_user'] === $user->username && $msg['content'] === '测试消息') { $found = true; break; } } $this->assertTrue($found, 'Message not found in Redis'); } /** * 测试文本内容为字符串 0 时仍可正常发送。 */ public function test_can_send_zero_message_content(): void { $room = Room::create(['room_name' => 'send0']); $user = User::factory()->create(); $this->actingAs($user)->get(route('chat.room', $room->id)); $response = $this->actingAs($user)->postJson(route('chat.send', $room->id), [ 'to_user' => '大家', 'content' => '0', 'is_secret' => false, 'font_color' => '#000000', 'action' => '', ]); $response->assertOk(); $response->assertJson(['status' => 'success']); $messages = Redis::lrange("room:{$room->id}:messages", 0, -1); $found = false; foreach ($messages as $msgJson) { $msg = json_decode($msgJson, true); if ($msg['from_user'] === $user->username && $msg['content'] === '0') { $found = true; break; } } $this->assertTrue($found, 'Zero message not found in Redis'); } /** * 测试用户可以发送带缩略图的图片消息。 */ public function test_can_send_image_message(): void { Storage::fake('public'); $room = Room::create(['room_name' => 'imgsend']); $user = User::factory()->create(); $this->actingAs($user)->get(route('chat.room', $room->id)); $response = $this->actingAs($user)->post(route('chat.send', $room->id), [ 'to_user' => '大家', 'content' => '图片说明', 'font_color' => '#000000', 'action' => '', 'image' => UploadedFile::fake()->image('chat-picture.png', 1280, 960), ], [ 'Accept' => 'application/json', ]); $response->assertOk(); $response->assertJson(['status' => 'success']); $messages = Redis::lrange("room:{$room->id}:messages", 0, -1); $payload = collect($messages) ->map(fn (string $item) => json_decode($item, true)) ->first(fn (array $item) => ($item['from_user'] ?? null) === $user->username && ($item['message_type'] ?? null) === 'image'); $this->assertNotNull($payload); $this->assertSame('image', $payload['message_type'] ?? null); $this->assertNotEmpty($payload['image_path'] ?? null); $this->assertNotEmpty($payload['image_thumb_path'] ?? null); $this->assertNotEmpty($payload['image_url'] ?? null); $this->assertNotEmpty($payload['image_thumb_url'] ?? null); Storage::disk('public')->assertExists($payload['image_path']); Storage::disk('public')->assertExists($payload['image_thumb_path']); } /** * 测试超过保留期的聊天图片会被命令清理并改成过期占位消息。 */ public function test_purge_command_cleans_expired_chat_images(): void { Storage::fake('public'); Storage::disk('public')->put('chat-images/2026-04-08/sample_original.png', 'original'); Storage::disk('public')->put('chat-images/2026-04-08/sample_thumb.png', 'thumb'); $message = \App\Models\Message::create([ 'room_id' => 1, 'from_user' => 'tester', 'to_user' => '大家', 'content' => '历史图片', 'is_secret' => false, 'font_color' => '#000000', 'action' => '', 'message_type' => 'image', 'image_path' => 'chat-images/2026-04-08/sample_original.png', 'image_thumb_path' => 'chat-images/2026-04-08/sample_thumb.png', 'image_original_name' => 'sample.png', 'sent_at' => now()->subDays(4), ]); $this->artisan('messages:purge', [ '--days' => 30, '--image-days' => 3, ])->assertExitCode(0); $message->refresh(); $this->assertSame('expired_image', $message->message_type); $this->assertNull($message->image_path); $this->assertNull($message->image_thumb_path); $this->assertNull($message->image_original_name); $this->assertStringContainsString('图片已过期', $message->content); Storage::disk('public')->assertMissing('chat-images/2026-04-08/sample_original.png'); Storage::disk('public')->assertMissing('chat-images/2026-04-08/sample_thumb.png'); } /** * 测试心跳接口可以正常返回成功响应。 */ public function test_can_trigger_heartbeat() { $room = Room::create(['room_name' => 'test_hb']); $user = User::factory()->create(['exp_num' => 0]); $response = $this->actingAs($user)->postJson(route('chat.heartbeat', $room->id)); $response->assertStatus(200); $response->assertJsonFragment(['status' => 'success']); $user->refresh(); $this->assertGreaterThanOrEqual(0, $user->exp_num); // Might be 1 depending on sysparam } /** * 测试显式退房会清理 Redis 在线状态。 */ public function test_can_leave_room() { $room = Room::create(['room_name' => 'test_leave']); $user = User::factory()->create(); // 进房 $this->actingAs($user)->get(route('chat.room', $room->id)); $this->assertEquals(1, Redis::hexists("room:{$room->id}:users", $user->username)); // 显式退房 $response = $this->actingAs($user)->postJson(route('chat.leave', $room->id).'?explicit=1'); $response->assertStatus(200); // 缓存中被移除 $this->assertEquals(0, Redis::hexists("room:{$room->id}:users", $user->username)); } /** * 测试签名退房链接同样可以正常清理在线状态。 */ public function test_can_leave_room_through_signed_expired_route(): void { $room = Room::create(['room_name' => 'leave2']); $user = User::factory()->create(); $this->actingAs($user)->get(route('chat.room', $room->id)); $this->assertEquals(1, Redis::hexists("room:{$room->id}:users", $user->username)); $url = URL::temporarySignedRoute('chat.leave.expired', now()->addMinutes(5), [ 'id' => $room->id, 'user' => $user->id, ]); $response = $this->getJson($url); $response->assertStatus(200); $this->assertEquals(0, Redis::hexists("room:{$room->id}:users", $user->username)); } /** * 测试会员用户首次进房时会把专属欢迎主题写入历史消息。 */ public function test_vip_user_join_message_uses_presence_theme_payload(): void { $room = Room::create(['room_name' => 'viproom']); $vipLevel = \App\Models\VipLevel::factory()->create([ 'join_effect' => 'lightning', 'join_banner_style' => 'storm', 'allow_custom_messages' => true, ]); $user = User::factory()->create([ 'vip_level_id' => $vipLevel->id, 'hy_time' => now()->addDays(30), 'custom_join_message' => '{username} 带着风暴王座闪耀降临', 'has_received_new_gift' => true, ]); $response = $this->actingAs($user)->get(route('chat.room', $room->id)); $response->assertStatus(200); $history = $response->viewData('historyMessages'); $presenceMessage = collect($history)->first(fn (array $message) => ($message['action'] ?? '') === 'vip_presence'); $this->assertNotNull($presenceMessage); $this->assertSame('join', $presenceMessage['presence_type']); $this->assertSame('lightning', $presenceMessage['presence_effect']); $this->assertSame('storm', $presenceMessage['presence_banner_style']); $this->assertStringContainsString($user->username, $presenceMessage['presence_text']); } /** * 测试可以获取所有房间的在线人数状态。 */ public function test_can_get_rooms_online_status() { $user = User::factory()->create(); $room1 = Room::create(['room_name' => 'room1']); $room2 = Room::create(['room_name' => 'room2']); $this->actingAs($user)->get(route('chat.room', $room1->id)); $response = $this->actingAs($user)->getJson(route('chat.rooms-online-status')); $response->assertStatus(200); // Assert room1 has 1 online, room2 has 0 $response->assertJsonFragment([ 'id' => $room1->id, 'online' => 1, ]); $response->assertJsonFragment([ 'id' => $room2->id, 'online' => 0, ]); } /** * 测试管理员可以设置房间公告。 */ public function test_can_set_announcement() { $user = User::factory()->create(['user_level' => 100]); // superadmin $room = Room::create(['room_name' => 'test_ann', 'room_owner' => 'someone']); $response = $this->actingAs($user)->postJson(route('chat.announcement', $room->id), [ 'announcement' => 'This is a new test announcement', ]); $response->assertStatus(200); $room->refresh(); $this->assertStringContainsString('This is a new test announcement', $room->announcement); } /** * 测试无权限用户不能设置房间公告。 */ public function test_cannot_set_announcement_without_permission() { $user = User::factory()->create(['user_level' => 0]); $room = Room::create(['room_name' => 'test_ann2', 'room_owner' => 'someone']); $response = $this->actingAs($user)->postJson(route('chat.announcement', $room->id), [ 'announcement' => 'This is a new test announcement', ]); $response->assertStatus(403); } }