mirror of
https://github.com/espocrm/espocrm.git
synced 2026-06-28 15:06:06 +00:00
593 lines
16 KiB
PHP
593 lines
16 KiB
PHP
<?php
|
|
/************************************************************************
|
|
* This file is part of EspoCRM.
|
|
*
|
|
* EspoCRM - Open Source CRM application.
|
|
* Copyright (C) 2014-2023 Yurii Kuznietsov, Taras Machyshyn, Oleksii Avramenko
|
|
* Website: https://www.espocrm.com
|
|
*
|
|
* EspoCRM is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* EspoCRM 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
|
|
*
|
|
* The interactive user interfaces in modified source and object code versions
|
|
* of this program must display Appropriate Legal Notices, as required under
|
|
* Section 5 of the GNU General Public License version 3.
|
|
*
|
|
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
|
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
|
************************************************************************/
|
|
|
|
namespace Espo\Core\WebSocket;
|
|
|
|
use Ratchet\ConnectionInterface;
|
|
use Ratchet\Wamp\WampServerInterface;
|
|
|
|
use Symfony\Component\Process\PhpExecutableFinder;
|
|
|
|
use Exception;
|
|
use RuntimeException;
|
|
|
|
class Pusher implements WampServerInterface
|
|
{
|
|
/**
|
|
* @var string[]
|
|
*/
|
|
private $categoryList;
|
|
|
|
/**
|
|
* @var array<string,array<string,mixed>>
|
|
*/
|
|
protected $categoriesData;
|
|
|
|
protected bool $isDebugMode = false;
|
|
|
|
/**
|
|
* @var array<string,string>
|
|
*/
|
|
protected $connectionIdUserIdMap = [];
|
|
|
|
/**
|
|
* @var array<string,string[]>
|
|
*/
|
|
protected $userIdConnectionIdListMap = [];
|
|
|
|
/**
|
|
* @var array<string,string[]>
|
|
*/
|
|
protected $connectionIdTopicIdListMap = [];
|
|
|
|
/**
|
|
* @var array<string,ConnectionInterface>
|
|
*/
|
|
protected $connections = [];
|
|
|
|
/**
|
|
* @var array<string,\Ratchet\Wamp\Topic<object>>
|
|
*/
|
|
protected $topicHash = [];
|
|
|
|
private string $phpExecutablePath;
|
|
|
|
/**
|
|
* @param array<string,array<string,mixed>> $categoriesData
|
|
*/
|
|
public function __construct(
|
|
array $categoriesData = [],
|
|
?string $phpExecutablePath = null,
|
|
bool $isDebugMode = false
|
|
) {
|
|
$this->categoryList = array_keys($categoriesData);
|
|
$this->categoriesData = $categoriesData;
|
|
|
|
if (!$phpExecutablePath) {
|
|
$phpExecutablePath = (new PhpExecutableFinder)->find() ?: null;
|
|
}
|
|
|
|
if (!$phpExecutablePath) {
|
|
if ($isDebugMode) {
|
|
$this->log("Error: No php-executable-path.");
|
|
}
|
|
|
|
throw new RuntimeException("No php-executable-path.");
|
|
}
|
|
|
|
$this->phpExecutablePath = $phpExecutablePath;
|
|
$this->isDebugMode = $isDebugMode;
|
|
}
|
|
|
|
/**
|
|
* @param \Ratchet\Wamp\Topic<object> $topic
|
|
* @return void
|
|
*/
|
|
public function onSubscribe(ConnectionInterface $connection, $topic)
|
|
{
|
|
$topicId = $topic->getId();
|
|
|
|
if (!$topicId) {
|
|
return;
|
|
}
|
|
|
|
if (!$this->isTopicAllowed($topicId)) {
|
|
return;
|
|
}
|
|
|
|
/** @var string $connectionId */
|
|
/** @phpstan-ignore-next-line */
|
|
$connectionId = $connection->resourceId;
|
|
|
|
$userId = $this->getUserIdByConnection($connection);
|
|
|
|
if (!$userId) {
|
|
return;
|
|
}
|
|
|
|
if (!isset($this->connectionIdTopicIdListMap[$connectionId])) {
|
|
$this->connectionIdTopicIdListMap[$connectionId] = [];
|
|
}
|
|
|
|
$checkCommand = $this->getAccessCheckCommandForTopic($connection, $topic);
|
|
|
|
if ($checkCommand) {
|
|
$checkResult = shell_exec($checkCommand);
|
|
|
|
if ($checkResult !== 'true') {
|
|
if ($this->isDebugMode) {
|
|
$this->log("{$connectionId}: check access failed for topic {$topicId} for user {$userId}");
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if ($this->isDebugMode) {
|
|
$this->log("{$connectionId}: check access succeed for topic {$topicId} for user {$userId}");
|
|
}
|
|
}
|
|
|
|
if (!in_array($topicId, $this->connectionIdTopicIdListMap[$connectionId])) {
|
|
if ($this->isDebugMode) {
|
|
$this->log("{$connectionId}: add topic {$topicId} for user {$userId}");
|
|
}
|
|
|
|
$this->connectionIdTopicIdListMap[$connectionId][] = $topicId;
|
|
}
|
|
|
|
$this->topicHash[$topicId] = $topic;
|
|
}
|
|
|
|
/**
|
|
* @param \Ratchet\Wamp\Topic<object> $topic
|
|
* @return void
|
|
*/
|
|
public function onUnSubscribe(ConnectionInterface $connection, $topic)
|
|
{
|
|
$topicId = $topic->getId();
|
|
|
|
if (!$topicId) {
|
|
return;
|
|
}
|
|
|
|
if (!$this->isTopicAllowed($topicId)) {
|
|
return;
|
|
}
|
|
|
|
/** @var string $connectionId */
|
|
/** @phpstan-ignore-next-line */
|
|
$connectionId = $connection->resourceId;
|
|
|
|
$userId = $this->getUserIdByConnection($connection);
|
|
|
|
if (!$userId) {
|
|
return;
|
|
}
|
|
|
|
if (isset($this->connectionIdTopicIdListMap[$connectionId])) {
|
|
$index = array_search($topicId, $this->connectionIdTopicIdListMap[$connectionId]);
|
|
|
|
if ($index !== false) {
|
|
if ($this->isDebugMode) {
|
|
$this->log("{$connectionId}: remove topic {$topicId} for user {$userId}");
|
|
}
|
|
|
|
unset($this->connectionIdTopicIdListMap[$connectionId][$index]);
|
|
|
|
$this->connectionIdTopicIdListMap[$connectionId] = array_values(
|
|
$this->connectionIdTopicIdListMap[$connectionId]
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return array<string,mixed>
|
|
*/
|
|
protected function getCategoryData(string $topicId): array
|
|
{
|
|
$arr = explode('.', $topicId);
|
|
|
|
$category = $arr[0];
|
|
|
|
if (array_key_exists($category, $this->categoriesData)) {
|
|
$data = $this->categoriesData[$category];
|
|
}
|
|
else if (array_key_exists($topicId, $this->categoriesData)) {
|
|
$data = $this->categoriesData[$topicId];
|
|
}
|
|
else {
|
|
$data = [];
|
|
}
|
|
|
|
return $data;
|
|
}
|
|
|
|
/**
|
|
* @return array<string,mixed>
|
|
*/
|
|
protected function getParamsFromTopicId(string $topicId): array
|
|
{
|
|
$arr = explode('.', $topicId);
|
|
|
|
$data = $this->getCategoryData($topicId);
|
|
|
|
$params = [];
|
|
|
|
if (array_key_exists('paramList', $data)) {
|
|
foreach ($data['paramList'] as $i => $item) {
|
|
/** @var string $item */
|
|
if (isset($arr[$i + 1])) {
|
|
$params[$item] = $arr[$i + 1];
|
|
}
|
|
else {
|
|
$params[$item] = '';
|
|
}
|
|
}
|
|
}
|
|
|
|
return $params;
|
|
}
|
|
|
|
/**
|
|
* @param \Ratchet\Wamp\Topic<object> $topic
|
|
*/
|
|
protected function getAccessCheckCommandForTopic(ConnectionInterface $connection, $topic): ?string
|
|
{
|
|
$topicId = $topic->getId();
|
|
|
|
$params = $this->getParamsFromTopicId($topicId);
|
|
$params['userId'] = $this->getUserIdByConnection($connection);
|
|
|
|
if (!$params['userId']) {
|
|
$connection->close();
|
|
|
|
return null;
|
|
}
|
|
|
|
$data = $this->getCategoryData($topic->getId());
|
|
|
|
if (!array_key_exists('accessCheckCommand', $data)) {
|
|
return null;
|
|
}
|
|
|
|
$command = $this->phpExecutablePath . " command.php " . $data['accessCheckCommand'];
|
|
|
|
foreach ($params as $key => $value) {
|
|
$command = str_replace(
|
|
':' . $key,
|
|
escapeshellarg($value),
|
|
$command
|
|
);
|
|
}
|
|
|
|
return $command;
|
|
}
|
|
|
|
/**
|
|
* @param \Ratchet\Wamp\Topic<object> $topic
|
|
* @return string
|
|
*/
|
|
protected function getTopicCategory($topic)
|
|
{
|
|
list($category) = explode('.', $topic->getId());
|
|
|
|
return $category;
|
|
}
|
|
|
|
/**
|
|
* @param string $topicId
|
|
* @return bool
|
|
*/
|
|
protected function isTopicAllowed($topicId)
|
|
{
|
|
list($category) = explode('.', $topicId);
|
|
|
|
return in_array($topicId, $this->categoryList) || in_array($category, $this->categoryList);
|
|
}
|
|
|
|
/**
|
|
* @param string $userId
|
|
* @return string[]
|
|
*/
|
|
protected function getConnectionIdListByUserId($userId)
|
|
{
|
|
if (!isset($this->userIdConnectionIdListMap[$userId])) {
|
|
return [];
|
|
}
|
|
|
|
return $this->userIdConnectionIdListMap[$userId];
|
|
}
|
|
|
|
/**
|
|
* @return ?string
|
|
*/
|
|
protected function getUserIdByConnection(ConnectionInterface $connection)
|
|
{
|
|
/** @phpstan-ignore-next-line */
|
|
if (!isset($this->connectionIdUserIdMap[$connection->resourceId])) {
|
|
return null;
|
|
}
|
|
|
|
/** @phpstan-ignore-next-line */
|
|
return $this->connectionIdUserIdMap[$connection->resourceId];
|
|
}
|
|
|
|
/**
|
|
* @param string $userId
|
|
* @return void
|
|
*/
|
|
protected function subscribeUser(ConnectionInterface $connection, $userId)
|
|
{
|
|
/** @var string $resourceId */
|
|
/** @phpstan-ignore-next-line */
|
|
$resourceId = $connection->resourceId;
|
|
|
|
$this->connectionIdUserIdMap[$resourceId] = $userId;
|
|
|
|
if (!isset($this->userIdConnectionIdListMap[$userId])) {
|
|
$this->userIdConnectionIdListMap[$userId] = [];
|
|
}
|
|
|
|
if (!in_array($resourceId, $this->userIdConnectionIdListMap[$userId])) {
|
|
$this->userIdConnectionIdListMap[$userId][] = $resourceId;
|
|
}
|
|
|
|
$this->connections[$resourceId] = $connection;
|
|
|
|
if ($this->isDebugMode) {
|
|
$this->log("{$resourceId}: user {$userId} subscribed");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param string $userId
|
|
* @return void
|
|
*/
|
|
protected function unsubscribeUser(ConnectionInterface $connection, $userId)
|
|
{
|
|
/** @phpstan-ignore-next-line */
|
|
$resourceId = $connection->resourceId;
|
|
|
|
unset($this->connectionIdUserIdMap[$resourceId]);
|
|
|
|
if (isset($this->userIdConnectionIdListMap[$userId])) {
|
|
$index = array_search($resourceId, $this->userIdConnectionIdListMap[$userId]);
|
|
|
|
if ($index !== false) {
|
|
unset($this->userIdConnectionIdListMap[$userId][$index]);
|
|
$this->userIdConnectionIdListMap[$userId] = array_values($this->userIdConnectionIdListMap[$userId]);
|
|
}
|
|
}
|
|
|
|
if ($this->isDebugMode) {
|
|
$this->log("{$resourceId}: user {$userId} unsubscribed");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return void
|
|
*/
|
|
public function onOpen(ConnectionInterface $connection)
|
|
{
|
|
if ($this->isDebugMode) {
|
|
/** @phpstan-ignore-next-line */
|
|
$this->log("{$connection->resourceId}: open");
|
|
}
|
|
|
|
/** @phpstan-ignore-next-line */
|
|
$httpRequest = $connection->httpRequest;
|
|
|
|
$query = $httpRequest->getUri()->getQuery();
|
|
|
|
$params = \GuzzleHttp\Psr7\parse_query($query ?: '');
|
|
|
|
if (empty($params['userId']) || empty($params['authToken'])) {
|
|
$this->closeConnection($connection);
|
|
|
|
return;
|
|
}
|
|
|
|
$authToken = preg_replace('/[^a-zA-Z0-9]+/', '', $params['authToken']);
|
|
$userId = preg_replace('/[^a-zA-Z0-9]+/', '', $params['userId']);
|
|
|
|
$result = $this->getUserIdByAuthToken($authToken);
|
|
|
|
if (empty($result)) {
|
|
$this->closeConnection($connection);
|
|
|
|
return;
|
|
}
|
|
|
|
if ($result !== $userId) {
|
|
$this->closeConnection($connection);
|
|
|
|
return;
|
|
}
|
|
|
|
$this->subscribeUser($connection, $userId);
|
|
}
|
|
|
|
/**
|
|
* @param string $authToken
|
|
* @return string
|
|
*/
|
|
private function getUserIdByAuthToken($authToken)
|
|
{
|
|
/** @var string|null|false $result */
|
|
$result = shell_exec($this->phpExecutablePath . " command.php AuthTokenCheck " . $authToken);
|
|
|
|
if ($result === null || $result === false) {
|
|
return '';
|
|
}
|
|
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* @return void
|
|
*/
|
|
protected function closeConnection(ConnectionInterface $connection)
|
|
{
|
|
$userId = $this->getUserIdByConnection($connection);
|
|
|
|
if ($userId) {
|
|
$this->unsubscribeUser($connection, $userId);
|
|
}
|
|
|
|
$connection->close();
|
|
}
|
|
|
|
/**
|
|
* @return void
|
|
*/
|
|
public function onClose(ConnectionInterface $connection)
|
|
{
|
|
if ($this->isDebugMode) {
|
|
/** @phpstan-ignore-next-line */
|
|
$this->log("{$connection->resourceId}: close");
|
|
}
|
|
|
|
$userId = $this->getUserIdByConnection($connection);
|
|
|
|
if ($userId) {
|
|
$this->unsubscribeUser($connection, $userId);
|
|
}
|
|
|
|
/** @phpstan-ignore-next-line */
|
|
unset($this->connections[$connection->resourceId]);
|
|
}
|
|
|
|
/**
|
|
* @param string $id
|
|
* @param \Ratchet\Wamp\Topic<object> $topic
|
|
* @param array<string,mixed> $params
|
|
* @return void
|
|
*/
|
|
public function onCall(ConnectionInterface $connection, $id, $topic, array $params)
|
|
{
|
|
if (!method_exists($connection, 'callError')) {
|
|
return;
|
|
}
|
|
|
|
$connection
|
|
->callError($id, $topic, 'You are not allowed to make calls')
|
|
->close();
|
|
}
|
|
|
|
/**
|
|
* @param \Ratchet\Wamp\Topic<object> $topic
|
|
* @param string $event
|
|
* @param array<mixed,mixed> $exclude
|
|
* @param array<mixed,mixed> $eligible
|
|
* @return void
|
|
*/
|
|
public function onPublish(ConnectionInterface $connection, $topic, $event, array $exclude, array $eligible)
|
|
{
|
|
$topicId = $topic->getId();
|
|
|
|
$connection->close();
|
|
}
|
|
|
|
/**
|
|
* @return void
|
|
*/
|
|
public function onError(ConnectionInterface $connection, Exception $e)
|
|
{
|
|
}
|
|
|
|
public function onMessageReceive(string $message): void
|
|
{
|
|
$data = json_decode($message);
|
|
|
|
if (!property_exists($data, 'topicId')) {
|
|
return;
|
|
}
|
|
|
|
$userId = $data->userId ?? null;
|
|
$topicId = $data->topicId ?? null;
|
|
|
|
if (!$topicId) {
|
|
return;
|
|
}
|
|
|
|
if (!$this->isTopicAllowed($topicId)) {
|
|
return;
|
|
}
|
|
|
|
if ($userId) {
|
|
foreach ($this->getConnectionIdListByUserId($userId) as $connectionId) {
|
|
if (!isset($this->connections[$connectionId])) {
|
|
continue;
|
|
}
|
|
|
|
if (!isset($this->connectionIdTopicIdListMap[$connectionId])) {
|
|
continue;
|
|
}
|
|
|
|
/** @var \Ratchet\Wamp\WampConnection $connection */
|
|
$connection = $this->connections[$connectionId];
|
|
|
|
if (in_array($topicId, $this->connectionIdTopicIdListMap[$connectionId])) {
|
|
if ($this->isDebugMode) {
|
|
$this->log("send {$topicId} for connection {$connectionId}");
|
|
}
|
|
|
|
$connection->event($topicId, $data);
|
|
}
|
|
}
|
|
|
|
if ($this->isDebugMode) {
|
|
$this->log("message {$topicId} for user {$userId}");
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
$topic = $this->topicHash[$topicId] ?? null;
|
|
|
|
if ($topic) {
|
|
$topic->broadcast($data);
|
|
|
|
if ($this->isDebugMode) {
|
|
$this->log("send {$topicId} to all");
|
|
}
|
|
}
|
|
|
|
if ($this->isDebugMode) {
|
|
$this->log("message {$topicId} for all");
|
|
}
|
|
}
|
|
|
|
protected function log(string $msg): void
|
|
{
|
|
echo "[" . date('Y-m-d H:i:s') . "] " . $msg . "\n";
|
|
}
|
|
}
|