Files
UNIT3D-Community-Edition/app/Http/Controllers/API/ChatController.php

516 lines
20 KiB
PHP

<?php
declare(strict_types=1);
/**
* NOTICE OF LICENSE.
*
* UNIT3D Community Edition is open-sourced software licensed under the GNU Affero General Public License v3.0
* The details is bundled with this project in the file LICENSE.txt.
*
* @project UNIT3D Community Edition
*
* @author HDVinnie <hdinnovations@protonmail.com>
* @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\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response
{
/**
* @phpstan-ignore-next-line Laravel automatically converts models to json
* @see https://github.com/laravel/framework/blob/48246da2320c95a17bfae922d36264105a917906/src/Illuminate/Http/Response.php#L56
*/
return response($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);
}
// Temp Fix For HTMLPurifier
if ($message === '<') {
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::find($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)));
/**
* @phpstan-ignore-next-line Laravel automatically converts models to json
* @see https://github.com/laravel/framework/blob/48246da2320c95a17bfae922d36264105a917906/src/Illuminate/Http/Response.php#L56
*/
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)));
/**
* @phpstan-ignore-next-line Laravel automatically converts models to json
* @see https://github.com/laravel/framework/blob/48246da2320c95a17bfae922d36264105a917906/src/Illuminate/Http/Response.php#L56
*/
return response($user);
}
public function deleteBotEcho(Request $request): \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response
{
$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)));
/**
* @phpstan-ignore-next-line Laravel automatically converts models to json
* @see https://github.com/laravel/framework/blob/48246da2320c95a17bfae922d36264105a917906/src/Illuminate/Http/Response.php#L56
*/
return response($user);
}
public function toggleRoomAudible(Request $request): \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response
{
$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)));
/**
* @phpstan-ignore-next-line Laravel automatically converts models to json
* @see https://github.com/laravel/framework/blob/48246da2320c95a17bfae922d36264105a917906/src/Illuminate/Http/Response.php#L56
*/
return response($user);
}
public function toggleTargetAudible(Request $request): \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response
{
$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)));
/**
* @phpstan-ignore-next-line Laravel automatically converts models to json
* @see https://github.com/laravel/framework/blob/48246da2320c95a17bfae922d36264105a917906/src/Illuminate/Http/Response.php#L56
*/
return response($user);
}
public function toggleBotAudible(Request $request): \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response
{
$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)));
/**
* @phpstan-ignore-next-line Laravel automatically converts models to json
* @see https://github.com/laravel/framework/blob/48246da2320c95a17bfae922d36264105a917906/src/Illuminate/Http/Response.php#L56
*/
return response($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();
/**
* @phpstan-ignore-next-line Laravel automatically converts models to json
* @see https://github.com/laravel/framework/blob/48246da2320c95a17bfae922d36264105a917906/src/Illuminate/Http/Response.php#L56
*/
return response($user);
}
public function updateUserRoom(Request $request): \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response
{
$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));
}
/**
* @phpstan-ignore-next-line Laravel automatically converts models to json
* @see https://github.com/laravel/framework/blob/48246da2320c95a17bfae922d36264105a917906/src/Illuminate/Http/Response.php#L56
*/
return response($user);
}
public function updateUserTarget(Request $request): \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\Response
{
$user = $request->user()->load(['chatStatus', 'chatroom', 'group', 'echoes']);
/**
* @phpstan-ignore-next-line Laravel automatically converts models to json
* @see https://github.com/laravel/framework/blob/48246da2320c95a17bfae922d36264105a917906/src/Illuminate/Http/Response.php#L56
*/
return response($user);
}
}