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.
 
 
 
 
 

1029 lines
33 KiB

<?php
declare(strict_types=1);
/**
* @copyright Copyright (c) 2016 Lukas Reschke <lukas@statuscode.ch>
* @copyright Copyright (c) 2016 Joas Schilling <coding@schilljs.com>
*
* @author Lukas Reschke <lukas@statuscode.ch>
* @author 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\Spreed;
use OCA\Spreed\Exceptions\InvalidPasswordException;
use OCA\Spreed\Exceptions\ParticipantNotFoundException;
use OCA\Spreed\Exceptions\UnauthorizedException;
use OCP\AppFramework\Utility\ITimeFactory;
use OCP\Comments\IComment;
use OCP\DB\QueryBuilder\IQueryBuilder;
use OCP\IDBConnection;
use OCP\IUser;
use OCP\Security\IHasher;
use OCP\Security\ISecureRandom;
use Symfony\Component\EventDispatcher\EventDispatcherInterface;
use Symfony\Component\EventDispatcher\GenericEvent;
class Room {
public const UNKNOWN_CALL = -1;
public const ONE_TO_ONE_CALL = 1;
public const GROUP_CALL = 2;
public const PUBLIC_CALL = 3;
public const PARTICIPANT_REMOVED = 'remove';
public const PARTICIPANT_LEFT = 'leave';
/** @var Manager */
private $manager;
/** @var IDBConnection */
private $db;
/** @var ISecureRandom */
private $secureRandom;
/** @var EventDispatcherInterface */
private $dispatcher;
/** @var ITimeFactory */
private $timeFactory;
/** @var IHasher */
private $hasher;
/** @var int */
private $id;
/** @var int */
private $type;
/** @var string */
private $token;
/** @var string */
private $name;
/** @var string */
private $password;
/** @var int */
private $activeGuests;
/** @var \DateTime|null */
private $activeSince;
/** @var \DateTime|null */
private $lastActivity;
/** @var IComment|null */
private $lastMessage;
/** @var string */
private $objectType;
/** @var string */
private $objectId;
/** @var string */
protected $currentUser;
/** @var Participant */
protected $participant;
public function __construct(Manager $manager,
IDBConnection $db,
ISecureRandom $secureRandom,
EventDispatcherInterface $dispatcher,
ITimeFactory $timeFactory,
IHasher $hasher,
int $id,
int $type,
string $token,
string $name,
string $password,
int $activeGuests,
\DateTime $activeSince = null,
\DateTime $lastActivity = null,
IComment $lastMessage = null,
string $objectType = '',
string $objectId = '') {
$this->manager = $manager;
$this->db = $db;
$this->secureRandom = $secureRandom;
$this->dispatcher = $dispatcher;
$this->timeFactory = $timeFactory;
$this->hasher = $hasher;
$this->id = $id;
$this->type = $type;
$this->token = $token;
$this->name = $name;
$this->password = $password;
$this->activeGuests = $activeGuests;
$this->activeSince = $activeSince;
$this->lastActivity = $lastActivity;
$this->lastMessage = $lastMessage;
$this->objectType = $objectType;
$this->objectId = $objectId;
}
public function getId(): int {
return $this->id;
}
public function getType(): int {
return $this->type;
}
public function getToken(): string {
return $this->token;
}
public function getName(): string {
return $this->name;
}
public function getDisplayName(string $userId): string {
return $this->manager->resolveRoomDisplayName($this, $userId);
}
public function getActiveGuests(): int {
return $this->activeGuests;
}
public function getActiveSince(): ?\DateTime {
return $this->activeSince;
}
public function getLastActivity(): ?\DateTime {
return $this->lastActivity;
}
public function getLastMessage(): ?IComment {
return $this->lastMessage;
}
public function getObjectType(): string {
return $this->objectType;
}
public function getObjectId(): string {
return $this->objectId;
}
public function hasPassword(): bool {
return $this->password !== '';
}
public function getPassword(): string {
return $this->password;
}
public function setParticipant(?string $userId, Participant $participant): void {
$this->currentUser = $userId;
$this->participant = $participant;
}
/**
* @param string|null $userId
* @return Participant
* @throws ParticipantNotFoundException When the user is not a participant
*/
public function getParticipant(?string $userId): Participant {
if (!is_string($userId) || $userId === '') {
throw new ParticipantNotFoundException('Not a user');
}
if ($this->currentUser === $userId && $this->participant instanceof Participant) {
return $this->participant;
}
$query = $this->db->getQueryBuilder();
$query->select('*')
->from('talk_participants')
->where($query->expr()->eq('user_id', $query->createNamedParameter($userId)))
->andWhere($query->expr()->eq('room_id', $query->createNamedParameter($this->getId())));
$result = $query->execute();
$row = $result->fetch();
$result->closeCursor();
if ($row === false) {
throw new ParticipantNotFoundException('User is not a participant');
}
if ($this->currentUser === $userId) {
$this->participant = $this->manager->createParticipantObject($this, $row);
return $this->participant;
}
return $this->manager->createParticipantObject($this, $row);
}
/**
* @param string|null $sessionId
* @return Participant
* @throws ParticipantNotFoundException When the user is not a participant
*/
public function getParticipantBySession(?string $sessionId): Participant {
if (!is_string($sessionId) || $sessionId === '' || $sessionId === '0') {
throw new ParticipantNotFoundException('Not a user');
}
$query = $this->db->getQueryBuilder();
$query->select('*')
->from('talk_participants')
->where($query->expr()->eq('session_id', $query->createNamedParameter($sessionId)))
->andWhere($query->expr()->eq('room_id', $query->createNamedParameter($this->getId())));
$result = $query->execute();
$row = $result->fetch();
$result->closeCursor();
if ($row === false) {
throw new ParticipantNotFoundException('User is not a participant');
}
return $this->manager->createParticipantObject($this, $row);
}
public function deleteRoom(): void {
$participants = $this->getParticipantsLegacy();
$this->dispatcher->dispatch(self::class . '::preDeleteRoom', new GenericEvent($this, [
'participants' => $participants,
]));
$query = $this->db->getQueryBuilder();
// Delete all participants
$query->delete('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
$query->execute();
// Delete room
$query->delete('talk_rooms')
->where($query->expr()->eq('id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
$query->execute();
$this->dispatcher->dispatch(self::class . '::postDeleteRoom', new GenericEvent($this, [
'participants' => $participants,
]));
}
/**
* @param string $newName Currently it is only allowed to rename: self::GROUP_CALL, self::PUBLIC_CALL
* @return bool True when the change was valid, false otherwise
*/
public function setName(string $newName): bool {
$oldName = $this->getName();
if ($newName === $oldName) {
return true;
}
if ($this->getType() === self::ONE_TO_ONE_CALL) {
return false;
}
$this->dispatcher->dispatch(self::class . '::preSetName', new GenericEvent($this, [
'newName' => $newName,
'oldName' => $oldName,
]));
$query = $this->db->getQueryBuilder();
$query->update('talk_rooms')
->set('name', $query->createNamedParameter($newName))
->where($query->expr()->eq('id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
$query->execute();
$this->name = $newName;
$this->dispatcher->dispatch(self::class . '::postSetName', new GenericEvent($this, [
'newName' => $newName,
'oldName' => $oldName,
]));
return true;
}
/**
* @param string $password Currently it is only allowed to have a password for Room::PUBLIC_CALL
* @return bool True when the change was valid, false otherwise
*/
public function setPassword(string $password): bool {
if ($this->getType() !== self::PUBLIC_CALL) {
return false;
}
$hash = $password !== '' ? $this->hasher->hash($password) : '';
$this->dispatcher->dispatch(self::class . '::preSetPassword', new GenericEvent($this, [
'password' => $password,
]));
$query = $this->db->getQueryBuilder();
$query->update('talk_rooms')
->set('password', $query->createNamedParameter($hash))
->where($query->expr()->eq('id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
$query->execute();
$this->password = $hash;
$this->dispatcher->dispatch(self::class . '::postSetPassword', new GenericEvent($this, [
'password' => $password,
]));
return true;
}
/**
* @param \DateTime $now
* @return bool
*/
public function setLastActivity(\DateTime $now): bool {
$query = $this->db->getQueryBuilder();
$query->update('talk_rooms')
->set('last_activity', $query->createNamedParameter($now, 'datetime'))
->where($query->expr()->eq('id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
$query->execute();
$this->lastActivity = $now;
return true;
}
/**
* @param \DateTime $since
* @param bool $isGuest
* @return bool
*/
public function setActiveSince(\DateTime $since, bool $isGuest): bool {
if ($isGuest && $this->getType() === self::PUBLIC_CALL) {
$query = $this->db->getQueryBuilder();
$query->update('talk_rooms')
->set('active_guests', $query->createFunction($query->getColumnName('active_guests') . ' + 1'))
->where($query->expr()->eq('id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
$query->execute();
$this->activeGuests++;
}
if ($this->activeSince instanceof \DateTime) {
return false;
}
$query = $this->db->getQueryBuilder();
$query->update('talk_rooms')
->set('active_since', $query->createNamedParameter($since, 'datetime'))
->where($query->expr()->eq('id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->isNull('active_since'));
$query->execute();
$this->activeSince = $since;
return true;
}
public function setLastMessage(IComment $message): void {
$query = $this->db->getQueryBuilder();
$query->update('talk_rooms')
->set('last_message', $query->createNamedParameter((int) $message->getId()))
->where($query->expr()->eq('id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
$query->execute();
}
public function resetActiveSince(): void {
$query = $this->db->getQueryBuilder();
$query->update('talk_rooms')
->set('active_guests', $query->createNamedParameter(0))
->set('active_since', $query->createNamedParameter(null, 'datetime'))
->where($query->expr()->eq('id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
$query->execute();
}
/**
* @param int $newType Currently it is only allowed to change to: self::GROUP_CALL, self::PUBLIC_CALL
* @return bool True when the change was valid, false otherwise
*/
public function changeType(int $newType): bool {
$newType = (int) $newType;
if ($newType === $this->getType()) {
return true;
}
if (!in_array($newType, [self::GROUP_CALL, self::PUBLIC_CALL], true)) {
return false;
}
$oldType = $this->getType();
$this->dispatcher->dispatch(self::class . '::preChangeType', new GenericEvent($this, [
'newType' => $newType,
'oldType' => $oldType,
]));
$query = $this->db->getQueryBuilder();
$query->update('talk_rooms')
->set('type', $query->createNamedParameter($newType, IQueryBuilder::PARAM_INT))
->where($query->expr()->eq('id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
$query->execute();
$this->type = $newType;
if ($oldType === self::PUBLIC_CALL) {
// Kick all guests and users that were not invited
$query = $this->db->getQueryBuilder();
$query->delete('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->in('participant_type', $query->createNamedParameter([Participant::GUEST, Participant::USER_SELF_JOINED], IQueryBuilder::PARAM_INT_ARRAY)));
$query->execute();
}
$this->dispatcher->dispatch(self::class . '::postChangeType', new GenericEvent($this, [
'newType' => $newType,
'oldType' => $oldType,
]));
return true;
}
/**
* @param array[] ...$participants
*/
public function addUsers(array ...$participants): void {
$this->dispatcher->dispatch(self::class . '::preAddUsers', new GenericEvent($this, [
'users' => $participants,
]));
$query = $this->db->getQueryBuilder();
$query->insert('talk_participants')
->values(
[
'user_id' => $query->createParameter('user_id'),
'session_id' => $query->createParameter('session_id'),
'participant_type' => $query->createParameter('participant_type'),
'room_id' => $query->createNamedParameter($this->getId()),
'last_ping' => $query->createNamedParameter(0, IQueryBuilder::PARAM_INT),
]
);
foreach ($participants as $participant) {
$query->setParameter('user_id', $participant['userId'])
->setParameter('session_id', $participant['sessionId'] ?? '0')
->setParameter('participant_type', $participant['participantType'] ?? Participant::USER, IQueryBuilder::PARAM_INT);
$query->execute();
}
$this->dispatcher->dispatch(self::class . '::postAddUsers', new GenericEvent($this, [
'users' => $participants,
]));
}
/**
* @param string $participant
* @param int $participantType
*/
public function setParticipantType(string $participant, int $participantType): void {
$this->dispatcher->dispatch(self::class . '::preSetParticipantType', new GenericEvent($this, [
'user' => $participant,
'newType' => $participantType,
]));
$query = $this->db->getQueryBuilder();
$query->update('talk_participants')
->set('participant_type', $query->createNamedParameter($participantType, IQueryBuilder::PARAM_INT))
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->eq('user_id', $query->createNamedParameter($participant)));
$query->execute();
$this->dispatcher->dispatch(self::class . '::postSetParticipantType', new GenericEvent($this, [
'user' => $participant,
'newType' => $participantType,
]));
}
/**
* @param Participant $participant
* @param int $participantType
*/
public function setParticipantTypeBySession(Participant $participant, int $participantType): void {
$this->dispatcher->dispatch(self::class . '::preSetParticipantTypeBySession', new GenericEvent($this, [
'participant' => $participant,
'newType' => $participantType,
]));
$query = $this->db->getQueryBuilder();
$query->update('talk_participants')
->set('participant_type', $query->createNamedParameter($participantType, IQueryBuilder::PARAM_INT))
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->eq('session_id', $query->createNamedParameter($participant->getSessionId())));
$query->execute();
$this->dispatcher->dispatch(self::class . '::postSetParticipantTypeBySession', new GenericEvent($this, [
'participant' => $participant,
'newType' => $participantType,
]));
}
/**
* @param IUser $user
* @param string $reason
*/
public function removeUser(IUser $user, string $reason): void {
try {
$participant = $this->getParticipant($user->getUID());
} catch (ParticipantNotFoundException $e) {
return;
}
$this->dispatcher->dispatch(self::class . '::preRemoveUser', new GenericEvent($this, [
'user' => $user,
'participant' => $participant,
'reason' => $reason,
]));
$query = $this->db->getQueryBuilder();
$query->delete('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->eq('user_id', $query->createNamedParameter($user->getUID())));
$query->execute();
$this->dispatcher->dispatch(self::class . '::postRemoveUser', new GenericEvent($this, [
'user' => $user,
'participant' => $participant,
'reason' => $reason,
]));
}
/**
* @param Participant $participant
* @param string $reason
*/
public function removeParticipantBySession(Participant $participant, string $reason): void {
$this->dispatcher->dispatch(self::class . '::preRemoveBySession', new GenericEvent($this, [
'participant' => $participant,
'reason' => $reason,
]));
$query = $this->db->getQueryBuilder();
$query->delete('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->eq('session_id', $query->createNamedParameter($participant->getSessionId())));
$query->execute();
$this->dispatcher->dispatch(self::class . '::postRemoveBySession', new GenericEvent($this, [
'participant' => $participant,
'reason' => $reason,
]));
}
/**
* @param IUser $user
* @param string $password
* @param bool $passedPasswordProtection
* @return string
* @throws InvalidPasswordException
* @throws UnauthorizedException
*/
public function joinRoom(IUser $user, string $password, bool $passedPasswordProtection = false): string {
$event = new GenericEvent($this, [
'userId' => $user->getUID(),
'password' => $password,
'passedPasswordProtection' => $passedPasswordProtection,
]);
$this->dispatcher->dispatch(self::class . '::preJoinRoom', $event);
if ($event->hasArgument('cancel') && $event->getArgument('cancel') === true) {
$this->removeUser($user, self::PARTICIPANT_LEFT);
throw new UnauthorizedException('Participant is not allowed to join');
}
$query = $this->db->getQueryBuilder();
$query->update('talk_participants')
->set('session_id', $query->createParameter('session_id'))
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->eq('user_id', $query->createNamedParameter($user->getUID())));
$sessionId = $this->secureRandom->generate(255);
$query->setParameter('session_id', $sessionId);
$result = $query->execute();
if ($result === 0) {
if (!$passedPasswordProtection && !$this->verifyPassword($password)['result']) {
throw new InvalidPasswordException();
}
// User joining a public room, without being invited
$this->addUsers([
'userId' => $user->getUID(),
'participantType' => Participant::USER_SELF_JOINED,
'sessionId' => $sessionId,
]);
}
while (!$this->isSessionUnique($sessionId)) {
$sessionId = $this->secureRandom->generate(255);
$query->setParameter('session_id', $sessionId);
$query->execute();
}
$this->dispatcher->dispatch(self::class . '::postJoinRoom', new GenericEvent($this, [
'userId' => $user->getUID(),
'password' => $password,
'passedPasswordProtection' => $passedPasswordProtection,
]));
return $sessionId;
}
/**
* @param string $userId
* @param string|null $sessionId
*/
public function leaveRoom(string $userId, ?string $sessionId = null): void {
try {
$participant = $this->getParticipant($userId);
} catch (ParticipantNotFoundException $e) {
return;
}
$this->dispatcher->dispatch(self::class . '::preUserDisconnectRoom', new GenericEvent($this, [
'userId' => $userId,
'sessionId' => $sessionId,
'participant' => $participant,
]));
// Reset session when leaving a normal room
$query = $this->db->getQueryBuilder();
$query->update('talk_participants')
->set('session_id', $query->createNamedParameter('0'))
->set('in_call', $query->createNamedParameter(0, IQueryBuilder::PARAM_INT))
->where($query->expr()->eq('user_id', $query->createNamedParameter($userId)))
->andWhere($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->neq('participant_type', $query->createNamedParameter(Participant::USER_SELF_JOINED, IQueryBuilder::PARAM_INT)));
if (!empty($sessionId)) {
$query->andWhere($query->expr()->eq('session_id', $query->createNamedParameter($sessionId)));
}
$query->execute();
// And kill session when leaving a self joined room
$query = $this->db->getQueryBuilder();
$query->delete('talk_participants')
->where($query->expr()->eq('user_id', $query->createNamedParameter($userId)))
->andWhere($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->eq('participant_type', $query->createNamedParameter(Participant::USER_SELF_JOINED, IQueryBuilder::PARAM_INT)));
if (!empty($sessionId)) {
$query->andWhere($query->expr()->eq('session_id', $query->createNamedParameter($sessionId)));
}
$selfJoined = (bool) $query->execute();
$this->dispatcher->dispatch(self::class . '::postUserDisconnectRoom', new GenericEvent($this, [
'userId' => $userId,
'sessionId' => $sessionId,
'participant' => $participant,
'selfJoin' => $selfJoined,
]));
}
/**
* @param string $password
* @param bool $passedPasswordProtection
* @return string
* @throws InvalidPasswordException
* @throws UnauthorizedException
*/
public function joinRoomGuest(string $password, bool $passedPasswordProtection = false): string {
$event = new GenericEvent($this);
$this->dispatcher->dispatch(self::class . '::preJoinRoomGuest', $event);
if ($event->hasArgument('cancel') && $event->getArgument('cancel') === true) {
throw new UnauthorizedException('Participant is not allowed to join');
}
if (!$passedPasswordProtection && !$this->verifyPassword($password)['result']) {
throw new InvalidPasswordException();
}
$sessionId = $this->secureRandom->generate(255);
while (!$this->db->insertIfNotExist('*PREFIX*talk_participants', [
'user_id' => '',
'room_id' => $this->getId(),
'last_ping' => 0,
'session_id' => $sessionId,
'participant_type' => Participant::GUEST,
], ['session_id'])) {
$sessionId = $this->secureRandom->generate(255);
}
$this->dispatcher->dispatch(self::class . '::postJoinRoomGuest', new GenericEvent($this));
return $sessionId;
}
public function changeInCall(string $sessionId, int $flags): void {
if ($flags !== Participant::FLAG_DISCONNECTED) {
$this->dispatcher->dispatch(self::class . '::preSessionJoinCall', new GenericEvent($this, [
'sessionId' => $sessionId,
'flags' => $flags,
]));
} else {
$this->dispatcher->dispatch(self::class . '::preSessionLeaveCall', new GenericEvent($this, [
'sessionId' => $sessionId,
]));
}
$query = $this->db->getQueryBuilder();
$query->update('talk_participants')
->set('in_call', $query->createNamedParameter($flags, IQueryBuilder::PARAM_INT))
->where($query->expr()->eq('session_id', $query->createNamedParameter($sessionId)))
->andWhere($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
$query->execute();
if ($flags !== Participant::FLAG_DISCONNECTED) {
$this->dispatcher->dispatch(self::class . '::postSessionJoinCall', new GenericEvent($this, [
'sessionId' => $sessionId,
'flags' => $flags,
]));
} else {
$this->dispatcher->dispatch(self::class . '::postSessionLeaveCall', new GenericEvent($this, [
'sessionId' => $sessionId,
]));
}
}
/**
* @param string $password
* @return array
*/
public function verifyPassword(string $password): array {
$event = new GenericEvent($this, [
'password' => $password
]);
$this->dispatcher->dispatch(self::class . '::verifyPassword', $event);
if ($event->hasArgument('result')) {
$result = $event->getArgument('result');
return [
'result' => $result['result'] ?? false,
'url' => $result['url'] ?? ''
];
}
return [
'result' => !$this->hasPassword() || $this->hasher->verify($password, $this->password),
'url' => ''
];
}
/**
* @param string $sessionId
* @return bool
*/
protected function isSessionUnique(string $sessionId): bool {
$query = $this->db->getQueryBuilder();
$query->selectAlias($query->createFunction('COUNT(*)'), 'num_sessions')
->from('talk_participants')
->where($query->expr()->eq('session_id', $query->createNamedParameter($sessionId)));
$result = $query->execute();
$numSessions = (int) $result->fetchColumn();
$result->closeCursor();
return $numSessions === 1;
}
public function cleanGuestParticipants(): void {
$this->dispatcher->dispatch(self::class . '::preCleanGuests', new GenericEvent($this));
$query = $this->db->getQueryBuilder();
$query->delete('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->emptyString('user_id'))
->andWhere($query->expr()->lte('last_ping', $query->createNamedParameter($this->timeFactory->getTime() - 100, IQueryBuilder::PARAM_INT)));
$query->execute();
$this->dispatcher->dispatch(self::class . '::postCleanGuests', new GenericEvent($this));
}
/**
* @param int $lastPing When the last ping is older than the given timestamp, the user is ignored
* @return Participant[]
*/
public function getParticipants(int $lastPing = 0): array {
$query = $this->db->getQueryBuilder();
$query->select('*')
->from('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
if ($lastPing > 0) {
$query->andWhere($query->expr()->gt('last_ping', $query->createNamedParameter($lastPing, IQueryBuilder::PARAM_INT)));
}
$result = $query->execute();
$participants = [];
while ($row = $result->fetch()) {
$participants[] = $this->manager->createParticipantObject($this, $row);
}
$result->closeCursor();
return $participants;
}
/**
* @param int $lastPing When the last ping is older than the given timestamp, the user is ignored
* @return array[] Array of users with [users => [userId => [lastPing, sessionId]], guests => [[lastPing, sessionId]]]
* @deprecated Use self::getParticipants() instead
*/
public function getParticipantsLegacy(int $lastPing = 0): array {
$query = $this->db->getQueryBuilder();
$query->select('*')
->from('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
if ($lastPing > 0) {
$query->andWhere($query->expr()->gt('last_ping', $query->createNamedParameter($lastPing, IQueryBuilder::PARAM_INT)));
}
$result = $query->execute();
$users = $guests = [];
while ($row = $result->fetch()) {
if ($row['user_id'] !== '' && $row['user_id'] !== null) {
$users[$row['user_id']] = [
'inCall' => (int) $row['in_call'],
'lastPing' => (int) $row['last_ping'],
'sessionId' => $row['session_id'],
'participantType' => (int) $row['participant_type'],
];
} else {
$guests[] = [
'inCall' => (int) $row['in_call'],
'lastPing' => (int) $row['last_ping'],
'participantType' => (int) $row['participant_type'],
'sessionId' => $row['session_id'],
];
}
}
$result->closeCursor();
return [
'users' => $users,
'guests' => $guests,
];
}
/**
* @param int $lastPing When the last ping is older than the given timestamp, the user is ignored
* @return string[]
*/
public function getParticipantUserIds(int $lastPing = 0): array {
$query = $this->db->getQueryBuilder();
$query->select('user_id')
->from('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->nonEmptyString('user_id'));
if ($lastPing > 0) {
$query->andWhere($query->expr()->gt('last_ping', $query->createNamedParameter($lastPing, IQueryBuilder::PARAM_INT)));
}
$result = $query->execute();
$users = [];
while ($row = $result->fetch()) {
$users[] = $row['user_id'];
}
$result->closeCursor();
return $users;
}
/**
* @param int $notificationLevel
* @return Participant[] Array of participants
*/
public function getParticipantsByNotificationLevel(int $notificationLevel): array {
$query = $this->db->getQueryBuilder();
$query->select('*')
->from('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->eq('notification_level', $query->createNamedParameter($notificationLevel, IQueryBuilder::PARAM_INT)));
$result = $query->execute();
$participants = [];
while ($row = $result->fetch()) {
$participants[] = $this->manager->createParticipantObject($this, $row);
}
$result->closeCursor();
return $participants;
}
/**
* @return string[]
*/
public function getActiveSessions(): array {
$query = $this->db->getQueryBuilder();
$query->select('session_id')
->from('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->neq('session_id', $query->createNamedParameter('0')));
$result = $query->execute();
$sessions = [];
while ($row = $result->fetch()) {
$sessions[] = $row['session_id'];
}
$result->closeCursor();
return $sessions;
}
/**
* Get all user ids which are participants in a room but currently not in the call
* @return string[]
*/
public function getNotInCallUserIds(): array {
$query = $this->db->getQueryBuilder();
$query->select('user_id')
->from('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->nonEmptyString('user_id'))
->andWhere($query->expr()->eq('in_call', $query->createNamedParameter(0, IQueryBuilder::PARAM_INT)));
$result = $query->execute();
$userIds = [];
while ($row = $result->fetch()) {
$userIds[] = $row['user_id'];
}
$result->closeCursor();
return $userIds;
}
/**
* @return bool
*/
public function hasSessionsInCall(): bool {
$query = $this->db->getQueryBuilder();
$query->select('session_id')
->from('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->neq('in_call', $query->createNamedParameter(Participant::FLAG_DISCONNECTED, IQueryBuilder::PARAM_INT)))
->setMaxResults(1);
$result = $query->execute();
$row = $result->fetch();
$result->closeCursor();
return (bool) $row;
}
/**
* @param bool $ignoreGuests
* @param int $lastPing When the last ping is older than the given timestamp, the user is ignored
* @return int
*/
public function getNumberOfParticipants(bool $ignoreGuests = true, int $lastPing = 0): int {
$query = $this->db->getQueryBuilder();
$query->selectAlias($query->createFunction('COUNT(*)'), 'num_participants')
->from('talk_participants')
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
if ($lastPing > 0) {
$query->andWhere($query->expr()->gt('last_ping', $query->createNamedParameter($lastPing, IQueryBuilder::PARAM_INT)));
}
if ($ignoreGuests) {
$query->andWhere($query->expr()->notIn('participant_type', $query->createNamedParameter([
Participant::GUEST,
Participant::USER_SELF_JOINED,
], IQueryBuilder::PARAM_INT_ARRAY)));
}
$result = $query->execute();
$row = $result->fetch();
$result->closeCursor();
return isset($row['num_participants']) ? (int) $row['num_participants'] : 0;
}
public function markUsersAsMentioned(array $userIds, \DateTime $time): void {
$query = $this->db->getQueryBuilder();
$query->update('talk_participants')
->set('last_mention', $query->createNamedParameter($time, 'datetime'))
->where($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)))
->andWhere($query->expr()->in('user_id', $query->createNamedParameter($userIds, IQueryBuilder::PARAM_STR_ARRAY)));
$query->execute();
}
/**
* @param string|null $userId
* @param string $sessionId
* @param int $timestamp
*/
public function ping(?string $userId, string $sessionId, int $timestamp): void {
$query = $this->db->getQueryBuilder();
$query->update('talk_participants')
->set('last_ping', $query->createNamedParameter($timestamp, IQueryBuilder::PARAM_INT))
->where($query->expr()->eq('user_id', $query->createNamedParameter((string) $userId)))
->andWhere($query->expr()->eq('session_id', $query->createNamedParameter($sessionId)))
->andWhere($query->expr()->eq('room_id', $query->createNamedParameter($this->getId(), IQueryBuilder::PARAM_INT)));
$query->execute();
}
}