You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

294 lines
9.3 KiB

<?php
declare(strict_types=1);
/**
* @copyright Copyright (c) 2019 Joas Schilling <coding@schilljs.com>
*
* @license GNU AGPL version 3 or any later version
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
namespace OCA\Talk\Middleware;
use OCA\Talk\Controller\AEnvironmentAwareController;
use OCA\Talk\Exceptions\ParticipantNotFoundException;
use OCA\Talk\Exceptions\PermissionsException;
use OCA\Talk\Exceptions\RoomNotFoundException;
use OCA\Talk\Manager;
use OCA\Talk\Middleware\Exceptions\LobbyException;
use OCA\Talk\Middleware\Exceptions\NotAModeratorException;
use OCA\Talk\Middleware\Exceptions\ReadOnlyException;
use OCA\Talk\Model\Attendee;
use OCA\Talk\Participant;
use OCA\Talk\Room;
use OCA\Talk\Service\ParticipantService;
use OCA\Talk\TalkSession;
use OCA\Talk\Webinary;
use OCP\AppFramework\Controller;
use OCP\AppFramework\Http;
use OCP\AppFramework\Http\Response;
use OCP\AppFramework\Http\RedirectToDefaultAppResponse;
use OCP\AppFramework\Middleware;
use OCP\AppFramework\OCS\OCSException;
use OCP\AppFramework\OCSController;
use OCP\AppFramework\Utility\IControllerMethodReflector;
use OCP\IRequest;
use OCP\Security\Bruteforce\IThrottler;
class InjectionMiddleware extends Middleware {
protected IRequest $request;
protected IControllerMethodReflector $reflector;
protected ParticipantService $participantService;
protected TalkSession $talkSession;
protected Manager $manager;
protected IThrottler $throttler;
protected ?string $userId;
public function __construct(IRequest $request,
IControllerMethodReflector $reflector,
ParticipantService $participantService,
TalkSession $talkSession,
Manager $manager,
IThrottler $throttler,
?string $userId) {
$this->request = $request;
$this->reflector = $reflector;
$this->participantService = $participantService;
$this->talkSession = $talkSession;
$this->manager = $manager;
$this->throttler = $throttler;
$this->userId = $userId;
}
/**
* @param Controller $controller
* @param string $methodName
* @throws RoomNotFoundException
* @throws ParticipantNotFoundException
* @throws NotAModeratorException
* @throws ReadOnlyException
* @throws LobbyException
*/
public function beforeController($controller, $methodName): void {
if (!$controller instanceof AEnvironmentAwareController) {
return;
}
$apiVersion = $this->request->getParam('apiVersion');
$controller->setAPIVersion((int) substr($apiVersion, 1));
if ($this->reflector->hasAnnotation('RequireLoggedInParticipant')) {
$this->getLoggedIn($controller, false);
}
if ($this->reflector->hasAnnotation('RequireLoggedInModeratorParticipant')) {
$this->getLoggedIn($controller, true);
}
if ($this->reflector->hasAnnotation('RequireParticipant')) {
$this->getLoggedInOrGuest($controller, false);
}
if ($this->reflector->hasAnnotation('RequireModeratorParticipant')) {
$this->getLoggedInOrGuest($controller, true);
}
if ($this->reflector->hasAnnotation('RequireRoom')) {
$this->getRoom($controller);
}
if ($this->reflector->hasAnnotation('RequireReadWriteConversation')) {
$this->checkReadOnlyState($controller);
}
if ($this->reflector->hasAnnotation('RequireModeratorOrNoLobby')) {
$this->checkLobbyState($controller);
}
$requiredPermissions = $this->reflector->getAnnotationParameter('RequirePermissions', 'permissions');
if ($requiredPermissions) {
$this->checkPermissions($controller, $requiredPermissions);
}
}
/**
* @param AEnvironmentAwareController $controller
*/
protected function getRoom(AEnvironmentAwareController $controller): void {
$token = $this->request->getParam('token');
$room = $this->manager->getRoomByToken($token);
$controller->setRoom($room);
}
/**
* @param AEnvironmentAwareController $controller
* @param bool $moderatorRequired
* @throws NotAModeratorException
*/
protected function getLoggedIn(AEnvironmentAwareController $controller, bool $moderatorRequired): void {
$token = $this->request->getParam('token');
$sessionId = $this->talkSession->getSessionForRoom($token);
$room = $this->manager->getRoomForUserByToken($token, $this->userId, $sessionId);
$controller->setRoom($room);
$participant = $this->participantService->getParticipant($room, $this->userId, $sessionId);
$controller->setParticipant($participant);
if ($moderatorRequired && !$participant->hasModeratorPermissions(false)) {
throw new NotAModeratorException();
}
}
/**
* @param AEnvironmentAwareController $controller
* @param bool $moderatorRequired
* @throws NotAModeratorException
* @throws ParticipantNotFoundException
*/
protected function getLoggedInOrGuest(AEnvironmentAwareController $controller, bool $moderatorRequired): void {
$room = $controller->getRoom();
if (!$room instanceof Room) {
$token = $this->request->getParam('token');
$sessionId = $this->talkSession->getSessionForRoom($token);
$room = $this->manager->getRoomForUserByToken($token, $this->userId, $sessionId);
$controller->setRoom($room);
}
$participant = $controller->getParticipant();
if (!$participant instanceof Participant) {
$participant = null;
if ($sessionId !== null) {
try {
$participant = $this->participantService->getParticipantBySession($room, $sessionId);
} catch (ParticipantNotFoundException $e) {
// ignore and fall back in case a concurrent request might have
// invalidated the session
}
}
if ($participant === null) {
$participant = $this->participantService->getParticipant($room, $this->userId);
}
$controller->setParticipant($participant);
}
if ($moderatorRequired && !$participant->hasModeratorPermissions()) {
throw new NotAModeratorException();
}
}
/**
* @param AEnvironmentAwareController $controller
* @throws ReadOnlyException
*/
protected function checkReadOnlyState(AEnvironmentAwareController $controller): void {
$room = $controller->getRoom();
if (!$room instanceof Room || $room->getReadOnly() === Room::READ_ONLY) {
throw new ReadOnlyException();
}
}
/**
* @param AEnvironmentAwareController $controller
* @throws PermissionsException
*/
protected function checkPermissions(AEnvironmentAwareController $controller, string $permissions): void {
$textPermissions = explode(',', $permissions);
$participant = $controller->getParticipant();
if (!$participant instanceof Participant) {
throw new PermissionsException();
}
foreach ($textPermissions as $textPermission) {
if ($textPermission === 'chat' && !($participant->getPermissions() & Attendee::PERMISSIONS_CHAT)) {
throw new PermissionsException();
}
if ($textPermission === 'call-start' && !($participant->getPermissions() & Attendee::PERMISSIONS_CALL_START)) {
throw new PermissionsException();
}
}
}
/**
* @param AEnvironmentAwareController $controller
* @throws LobbyException
*/
protected function checkLobbyState(AEnvironmentAwareController $controller): void {
try {
$this->getLoggedInOrGuest($controller, true);
return;
} catch (NotAModeratorException $e) {
} catch (ParticipantNotFoundException $e) {
}
$participant = $controller->getParticipant();
if ($participant instanceof Participant &&
$participant->getPermissions() & Attendee::PERMISSIONS_LOBBY_IGNORE) {
return;
}
$room = $controller->getRoom();
if (!$room instanceof Room || $room->getLobbyState() !== Webinary::LOBBY_NONE) {
throw new LobbyException();
}
}
/**
* @param Controller $controller
* @param string $methodName
* @param \Exception $exception
* @throws \Exception
* @return Response
*/
public function afterException($controller, $methodName, \Exception $exception): Response {
if ($exception instanceof RoomNotFoundException ||
$exception instanceof ParticipantNotFoundException) {
if ($controller instanceof OCSController) {
$isBruteForceProtected = $this->reflector->hasAnnotation('BruteForceProtection');
if ($isBruteForceProtected) {
$ip = $this->request->getRemoteAddress();
$action = 'talkRoomToken';
$this->throttler->sleepDelay($ip, $action);
$this->throttler->registerAttempt($action, $ip);
}
throw new OCSException('', Http::STATUS_NOT_FOUND);
}
return new RedirectToDefaultAppResponse();
}
if ($exception instanceof LobbyException) {
if ($controller instanceof OCSController) {
throw new OCSException('', Http::STATUS_PRECONDITION_FAILED);
}
return new RedirectToDefaultAppResponse();
}
if ($exception instanceof NotAModeratorException ||
$exception instanceof ReadOnlyException ||
$exception instanceof PermissionsException) {
if ($controller instanceof OCSController) {
throw new OCSException('', Http::STATUS_FORBIDDEN);
}
return new RedirectToDefaultAppResponse();
}
throw $exception;
}
}