Пример 1
(требуется версия Workerman >= 3.3.0)
Система оповещения на основе Worker с многопоточностью (распределённый кластер), групповая рассылка и широковещательная передача в кластере.
start_channel.php
Вся система может иметь только один сервис start_channel. Предположим, он работает на 192.168.1.1.
<?php
use Workerman\Worker;
require_once __DIR__ . '/vendor/autoload.php';
// Инициализация сервера Channel
$channel_server = new Channel\Server('0.0.0.0', 2206);
Worker::runAll();
start_ws.php
Вся система может иметь несколько сервисов start_ws, предположим, что они работают на серверах 192.168.1.2 и 192.168.1.3.
<?php
use Workerman\Worker;
use Workerman\Connection\TcpConnection;
require_once __DIR__ . '/vendor/autoload.php';
// Сервер websocket
$worker = new Worker('websocket://0.0.0.0:4236');
$worker->count=2;
$worker->name = 'pusher';
$worker->onWorkerStart = function($worker)
{
// Клиент Channel подключается к серверу Channel
Channel\Client::connect('192.168.1.1', 2206);
// Используйте свой идентификатор процесса как имя события
$event_name = $worker->id;
// Подписка на событие worker->id и регистрация функции обработки события
Channel\Client::on($event_name, function($event_data)use($worker){
$to_connection_id = $event_data['to_connection_id'];
$message = $event_data['content'];
if(!isset($worker->connections[$to_connection_id]))
{
echo "Соединение не существует\n";
return;
}
$to_connection = $worker->connections[$to_connection_id];
$to_connection->send($message);
});
// Подписка на широковещательное событие
$event_name = '广播';
// При получении широковещательного события отправьте данные всем клиентским соединениям в текущем процессе
Channel\Client::on($event_name, function($event_data)use($worker){
$message = $event_data['content'];
foreach($worker->connections as $connection)
{
$connection->send($message);
}
});
};
$worker->onConnect = function(TcpConnection $connection)use($worker)
{
$msg = "workerID:{$worker->id} connectionID:{$connection->id} подключен\n";
echo $msg;
$connection->send($msg);
};
Worker::runAll();
start_http.php
Вся система может иметь несколько сервисов start_http, предположим, что они работают на серверах 192.168.1.4 и 192.168.1.5.
<?php
use Workerman\Worker;
use Workerman\Connection\TcpConnection;
require_once __DIR__ . '/vendor/autoload.php';
// Обработка http-запросов, для отправки данных любому клиенту, нужно передать workerID и connectionID
$http_worker = new Worker('http://0.0.0.0:4237');
$http_worker->name = 'publisher';
$http_worker->onWorkerStart = function()
{
Channel\Client::connect('192.168.1.1', 2206);
};
$http_worker->onMessage = function(TcpConnection $connection, $request)
{
// Совместимость с workerman 4.x 5.x
if (!is_array($request)) {
$get = $request->get();
}
$connection->send('ok');
if(empty($get['content'])) return;
// Это отправка данных конкретному соединению в определенном worker процессе
if(isset($get['to_worker_id']) && isset($get['to_connection_id']))
{
$event_name = $get['to_worker_id'];
$to_connection_id = $get['to_connection_id'];
$content = $get['content'];
Channel\Client::publish($event_name, array(
'to_connection_id' => $to_connection_id,
'content' => $content
));
}
// Это глобальная широковещательная передача
else
{
$event_name = '广播';
$content = $get['content'];
Channel\Client::publish($event_name, array(
'content' => $content
));
}
};
Worker::runAll();
Тестирование
-
Запустите сервисы на каждом из серверов.
-
Подключите клиента к серверу.
Откройте браузер Chrome, нажмите F12, чтобы открыть консоль разработчика, введите в строку Console (или поместите нижеприведённый код в html-страницу и выполните с помощью js):
// Также можно подключиться к ws://192.168.1.3:4236
ws = new WebSocket("ws://192.168.1.2:4236");
ws.onmessage = function(e) {
alert("Получено сообщение от сервера: " + e.data);
};
- Отправьте уведомление, вызвав http-интерфейс.
URL для обращения http://192.168.1.4:4237/?content={$content} или http://192.168.1.5:4237/?content={$content} для отправки данных $content всем клиентским соединениям.
URL для обращения http://192.168.1.4:4237/?to_worker_id={$worker_id}&to_connection_id={$connection_id}&content={$content} или http://192.168.1.5:4237/?to_worker_id={$worker_id}&to_connection_id={$connection_id}&content={$content} для отправки данных $content конкретному клиентскому соединению в определённом worker процессе.
Обратите внимание: во время тестирования замените {$worker_id}, {$connection_id} и {$content} на реальные значения.