* @license https://www.gnu.org/licenses/agpl-3.0.en.html/ GNU Affero General Public License v3.0 */ namespace App\Http\Controllers\API; use App\Bots\NerdBot; use App\Bots\SystemBot; use App\Events\Chatter; use App\Http\Controllers\Controller; use App\Http\Resources\BotResource; use App\Http\Resources\ChatMessageResource; use App\Http\Resources\ChatRoomResource; use App\Http\Resources\UserAudibleResource; use App\Http\Resources\UserEchoResource; use App\Models\Bot; use App\Models\Message; use App\Models\User; use App\Models\UserAudible; use App\Models\UserEcho; use App\Repositories\ChatRepository; use Illuminate\Http\Request; use Illuminate\Support\Carbon; /** * @see \Tests\Feature\Http\Controllers\API\ChatControllerTest */ class ChatController extends Controller { /** * ChatController Constructor. */ public function __construct(private readonly ChatRepository $chatRepository) { } /* STATUSES */ public function statuses(): \Illuminate\Http\JsonResponse { return response()->json($this->chatRepository->statuses()); } /* ECHOES */ public function echoes(Request $request): \Illuminate\Http\Resources\Json\AnonymousResourceCollection { $user = $request->user()->load(['echoes']); if ($user->echoes->isEmpty()) { $userEcho = new UserEcho(); $userEcho->user_id = $request->user()->id; $userEcho->room_id = 1; $userEcho->save(); } return UserEchoResource::collection($this->chatRepository->echoes($request->user()->id)); } /* AUDIBLES */ public function audibles(Request $request): \Illuminate\Http\Resources\Json\AnonymousResourceCollection { $user = $request->user()->load(['audibles']); if ($user->audibles->isEmpty()) { $userAudible = new UserAudible(); $userAudible->user_id = $request->user()->id; $userAudible->room_id = 1; $userAudible->status = true; $userAudible->save(); } return UserAudibleResource::collection($this->chatRepository->audibles($request->user()->id)); } /* BOTS */ public function bots(): \Illuminate\Http\Resources\Json\AnonymousResourceCollection { return BotResource::collection($this->chatRepository->bots()); } /* ROOMS */ public function rooms(): \Illuminate\Http\Resources\Json\AnonymousResourceCollection { return ChatRoomResource::collection($this->chatRepository->rooms()); } public function config(): \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response { return response($this->chatRepository->config()); } /* MESSAGES */ public function messages(int $roomId): \Illuminate\Http\Resources\Json\AnonymousResourceCollection { return ChatMessageResource::collection($this->chatRepository->messages($roomId)); } /* MESSAGES */ public function privateMessages(Request $request, int $targetId): \Illuminate\Http\Resources\Json\AnonymousResourceCollection { return ChatMessageResource::collection($this->chatRepository->privateMessages($request->user()->id, $targetId)); } /* MESSAGES */ public function botMessages(Request $request, int $botId): \Illuminate\Http\Resources\Json\AnonymousResourceCollection { $runbot = null; $bot = Bot::findOrFail($botId); if ($bot->is_systembot) { $runbot = new SystemBot($this->chatRepository); } elseif ($bot->is_nerdbot) { $runbot = new NerdBot($this->chatRepository); } $runbot->process('message', $request->user(), '', 0); return ChatMessageResource::collection($this->chatRepository->botMessages($request->user()->id, $bot->id)); } public function createMessage(Request $request): \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response|bool|ChatMessageResource { $bot = null; $user = $request->user(); $userId = $user->id; $receiverId = $request->input('receiver_id'); $roomId = $request->input('chatroom_id'); $botId = $request->input('bot_id'); $message = $request->input('message'); $targeted = $request->input('targeted'); $save = $request->get('save'); if (!($user->can_chat ?? $user->group->can_chat)) { return response('error', 401); } $bots = cache()->remember('bots', 3600, fn () => Bot::where('active', '=', 1)->orderByDesc('position')->get()); $which = null; $target = null; $runbot = null; $trip = 'msg'; if ($message && str_starts_with((string) $message, '/'.$trip)) { $which = 'skip'; $command = @explode(' ', (string) $message); if (\array_key_exists(1, $command)) { $receiverId = User::where('username', 'like', $command[1])->sole()->id; $clone = $command; array_shift($clone); array_shift($clone); $message = trim(implode(' ', $clone)); } $botId = 1; } $trip = 'gift'; if ($message && str_starts_with((string) $message, '/'.$trip)) { $which = 'echo'; $target = 'system'; $message = '/bot gift'.substr((string) $message, \strlen($trip) + 1, \strlen((string) $message)); } if ($target == 'system') { $runbot = new SystemBot($this->chatRepository); } if ($which == null) { foreach ($bots as $bot) { if ($message && str_starts_with((string) $message, '/'.$bot->command)) { $which = 'echo'; } elseif ($message && str_starts_with((string) $message, '!'.$bot->command)) { $which = 'public'; } elseif ($message && str_starts_with((string) $message, '@'.$bot->command)) { $message = substr((string) $message, 1 + \strlen((string) $bot->command), \strlen((string) $message)); $which = 'private'; } elseif ($message && $receiverId == 1 && $bot->id == $botId) { if (str_starts_with((string) $message, '/'.$bot->command)) { $message = substr((string) $message, 1 + \strlen((string) $bot->command), \strlen((string) $message)); } if ($message && str_starts_with((string) $message, '!'.$bot->command)) { $message = substr((string) $message, 1 + \strlen((string) $bot->command), \strlen((string) $message)); } if ($message && str_starts_with((string) $message, '@'.$bot->command)) { $message = substr((string) $message, 1 + \strlen((string) $bot->command), \strlen((string) $message)); } $which = 'message'; } if ($which != null) { break; } } } if ($which != null && $which != 'skip' && !$runbot) { if ($bot->is_systembot) { $runbot = new SystemBot($this->chatRepository); } elseif ($bot->is_nerdbot) { $runbot = new NerdBot($this->chatRepository); } } if ($runbot !== null) { return $runbot->process($which ?? '', $request->user(), $message, 0); } $echo = false; if ($receiverId && $receiverId > 0) { // Create echo for both users if missing foreach ([[$userId, $receiverId], [$receiverId, $userId]] as [$user1Id, $user2Id]) { $echoes = cache()->remember( 'user-echoes'.$user1Id, 3600, fn () => UserEcho::with(['room', 'target', 'bot'])->where('user_id', '=', $user1Id)->get() ); if ($echoes->doesntContain(fn ($echo) => $echo->target_id == $user2Id)) { UserEcho::create([ 'user_id' => $user1Id, 'target_id' => $user2Id, ]); $echoes = UserEcho::with(['room', 'target', 'bot'])->where('user_id', '=', $user1Id)->get(); cache()->put('user-echoes'.$user1Id, $echoes, 3600); Chatter::dispatch('echo', $user1Id, UserEchoResource::collection($echoes)); } } // Create audible for both users if missing foreach ([[$userId, $receiverId], [$receiverId, $userId]] as [$user1Id, $user2Id]) { $audibles = cache()->remember( 'user-audibles'.$user1Id, 3600, fn () => UserAudible::with(['room', 'target', 'bot'])->where('user_id', '=', $user1Id)->get() ); if ($audibles->doesntContain(fn ($audible) => $audible->target_id == $user2Id)) { UserAudible::create([ 'user_id' => $user1Id, 'target_id' => $user2Id, 'status' => false, ]); $audibles = UserAudible::with(['room', 'target', 'bot'])->where('user_id', '=', $user1Id)->get(); cache()->put('user-audibles'.$user1Id, $audibles, 3600); Chatter::dispatch('audible', $user1Id, UserAudibleResource::collection($audibles)); } } $roomId = 0; $ignore = $botId > 0 && $receiverId == 1 ? true : null; $save = true; $echo = true; $message = $this->chatRepository->privateMessage($userId, $roomId, $message, $receiverId, null, $ignore); } else { $receiverId = null; $botId = null; $message = $this->chatRepository->message($userId, $roomId, $message, $receiverId, $botId); } if (!$save) { $message->delete(); } if ($save && $echo) { return new ChatMessageResource($message); } return response('success'); } public function deleteMessage(Request $request, int $id): \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response { $message = Message::findOrFail($id); abort_unless($request->user()->id === $message->user_id || $request->user()->group->is_modo, 403); $changedByStaff = $request->user()->id !== $message->user_id; abort_if($changedByStaff && !$request->user()->group->is_owner && $request->user()->group->level <= $message->user->group->level, 403); $this->chatRepository->deleteMessage($id); return response('success'); } public function deleteRoomEcho(Request $request): \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response { $user = $request->user(); UserEcho::where('user_id', '=', $user->id)->where('room_id', '=', $request->input('room_id'))->delete(); $user->load(['chatStatus', 'chatroom', 'group', 'echoes']); $room = $this->chatRepository->roomFindOrFail($request->input('room_id')); $user->chatroom()->dissociate(); $user->chatroom()->associate($room); $user->save(); $senderEchoes = UserEcho::with(['room', 'target', 'bot'])->where('user_id', $user->id)->get(); $expiresAt = Carbon::now()->addMinutes(60); cache()->put('user-echoes'.$user->id, $senderEchoes, $expiresAt); event(new Chatter('echo', $user->id, UserEchoResource::collection($senderEchoes))); /** * @see https://github.com/laravel/framework/blob/48246da2320c95a17bfae922d36264105a917906/src/Illuminate/Http/Response.php#L56 * @phpstan-ignore-next-line Laravel automatically converts models to json */ return response($user); } public function deleteTargetEcho(Request $request): \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response { $user = $request->user(); UserEcho::where('user_id', '=', $user->id)->where('target_id', '=', $request->input('target_id'))->delete(); $user->load(['chatStatus', 'chatroom', 'group', 'echoes']); $senderEchoes = UserEcho::with(['room', 'target', 'bot'])->where('user_id', $user->id)->get(); $expiresAt = Carbon::now()->addMinutes(60); cache()->put('user-echoes'.$user->id, $senderEchoes, $expiresAt); event(new Chatter('echo', $user->id, UserEchoResource::collection($senderEchoes))); /** * @see https://github.com/laravel/framework/blob/48246da2320c95a17bfae922d36264105a917906/src/Illuminate/Http/Response.php#L56 * @phpstan-ignore-next-line Laravel automatically converts models to json */ return response($user); } public function deleteBotEcho(Request $request): \Illuminate\Http\JsonResponse { $user = $request->user(); UserEcho::where('user_id', '=', $user->id)->where('bot_id', '=', $request->input('bot_id'))->delete(); $user->load(['chatStatus', 'chatroom', 'group', 'echoes']); $senderEchoes = UserEcho::with(['room', 'target', 'bot'])->where('user_id', $user->id)->get(); $expiresAt = Carbon::now()->addMinutes(60); cache()->put('user-echoes'.$user->id, $senderEchoes, $expiresAt); event(new Chatter('echo', $user->id, UserEchoResource::collection($senderEchoes))); return response()->json($user); } public function toggleRoomAudible(Request $request): \Illuminate\Http\JsonResponse { $user = $request->user(); $echo = UserAudible::where('user_id', '=', $user->id)->where('room_id', '=', $request->input('room_id'))->sole(); $echo->status = !$echo->status; $echo->save(); $user->load(['chatStatus', 'chatroom', 'group', 'audibles', 'audibles']); $senderAudibles = UserAudible::with(['room', 'target', 'bot'])->where('user_id', $user->id)->get(); $expiresAt = Carbon::now()->addMinutes(60); cache()->put('user-audibles'.$user->id, $senderAudibles, $expiresAt); event(new Chatter('audible', $user->id, UserAudibleResource::collection($senderAudibles))); return response()->json($user); } public function toggleTargetAudible(Request $request): \Illuminate\Http\JsonResponse { $user = $request->user(); $echo = UserAudible::where('user_id', '=', $user->id)->where('target_id', '=', $request->input('target_id'))->sole(); $echo->status = !$echo->status; $echo->save(); $user->load(['chatStatus', 'chatroom', 'group', 'audibles', 'audibles']); $senderAudibles = UserAudible::with(['target', 'room', 'bot'])->where('user_id', $user->id)->get(); $expiresAt = Carbon::now()->addMinutes(60); cache()->put('user-audibles'.$user->id, $senderAudibles, $expiresAt); event(new Chatter('audible', $user->id, UserAudibleResource::collection($senderAudibles))); return response()->json($user); } public function toggleBotAudible(Request $request): \Illuminate\Http\JsonResponse { $user = $request->user(); $echo = UserAudible::where('user_id', '=', $user->id)->where('bot_id', '=', $request->input('bot_id'))->sole(); $echo->status = !$echo->status; $echo->save(); $user->load(['chatStatus', 'chatroom', 'group', 'audibles', 'audibles'])->findOrFail($user->id); $senderAudibles = UserAudible::with(['bot', 'room', 'bot'])->where('user_id', $user->id)->get(); $expiresAt = Carbon::now()->addMinutes(60); cache()->put('user-audibles'.$user->id, $senderAudibles, $expiresAt); event(new Chatter('audible', $user->id, UserAudibleResource::collection($senderAudibles))); return response()->json($user); } /* USERS */ public function updateUserChatStatus(Request $request): \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response { $user = $request->user(); $user->load(['chatStatus', 'chatroom', 'group', 'echoes']); $status = $this->chatRepository->statusFindOrFail($request->input('status_id')); $this->chatRepository->systemMessage('[url=/users/'.$user->username.']'.$user->username.'[/url] has updated their status to [b]'.$status->name.'[/b]'); $user->chatStatus()->dissociate(); $user->chatStatus()->associate($status); $user->save(); return response('success'); } public function updateUserRoom(Request $request): \Illuminate\Http\JsonResponse { $user = $request->user(); $user->load(['chatStatus', 'chatroom', 'group', 'echoes']); $room = $this->chatRepository->roomFindOrFail($request->input('room_id')); $user->chatroom()->dissociate(); $user->chatroom()->associate($room); $user->save(); // Create echo for user if missing $echoes = cache()->remember( 'user-echoes'.$user->id, 3600, fn () => UserEcho::with(['room', 'target', 'bot'])->where('user_id', '=', $user->id)->get(), ); if ($echoes->doesntContain(fn ($echo) => $echo->room_id == $room->id)) { UserEcho::create([ 'user_id' => $user->id, 'room_id' => $room->id, ]); $echoes = UserEcho::with(['room', 'target', 'bot'])->where('user_id', '=', $user->id)->get(); cache()->put('user-echoes'.$user->id, $echoes, 3600); Chatter::dispatch('echo', $user->id, UserEchoResource::collection($echoes)); } return response()->json($user); } public function updateUserTarget(Request $request): \Illuminate\Http\JsonResponse { $user = $request->user()->load(['chatStatus', 'chatroom', 'group', 'echoes']); return response()->json($user); } }