workerman使用
检测是否支持
% curl -Ss http://www.workerman.net/check.php | php
PHP Version >= 5.3.3 [OK]
Extension pcntl check [OK]
Extension posix check [OK]
composer 安装
% composer require workerman/workerman
Using version ^4.0 for workerman/workerman
./composer.json has been updated
Running composer update workerman/workerman
Loading composer repositories with package information
Updating dependencies
Lock file operations: 1 install, 0 updates, 0 removals
- Locking workerman/workerman (v4.0.19)
Writing lock file
Installing dependencies from lock file (including require-dev)
Package operations: 1 install, 0 updates, 0 removals
- Downloading workerman/workerman (v4.0.19)
- Installing workerman/workerman (v4.0.19): Extracting archive
1 package suggestions were added by new dependencies, use `composer suggest` to see details.
Generating autoload files
9 packages you are using are looking for funding.
Use the `composer fund` command to find out more!
核心代码
<?php
/**
* workerman
*/
namespace Index\Action;
use Common\Util\JsonUtil;
use Workerman\Worker;
class WsAction extends CommonAction
{
public function start()
{
global $worker;
$worker = new Worker('websocket://0.0.0.0:'.C('WS_PORT')); // 8000
// 这里进程数必须设置为1
$worker->count = 1;
// worker进程启动后建立一个内部通讯端口
$worker->onWorkerStart = function ($worker) {
// 开启一个内部端口,方便内部系统推送数据,Text协议格式 文本+换行符
$inner_text_worker = new Worker('Text://0.0.0.0:'.C('WS_INNER_PORT')); // 8001
$inner_text_worker->onMessage = function ($connection, $buffer) {
global $worker;
// $data数组格式,里面有uid,表示向那个uid的页面推送数据
$data = JsonUtil::jsonDecode($buffer);
$uid = $data['uid'];
// 发送数据
if ($uid) {
$res = $this->sendMessageByUid($uid, $data['data']);
} else {
$res = $this->broadcast($data['data']);
}
// 返回推送结果
$connection->send($res ? 'ok' : 'fail');
echo "inner text Msg $uid \n";
};
$inner_text_worker->listen();
echo "Start \n";
};
// 新增加一个属性,用来保存uid到connection的映射
$worker->uidConnections = array();
// 当有客户端发来消息时执行的回调函数
$worker->onMessage = function ($connection, $data) use ($worker) {
// 判断当前客户端是否已经验证,既是否设置了uid
if (!isset($connection->uid)) {
// 没验证的话把第一个包当做uid(这里为了方便演示,没做真正的验证)
$connection->uid = $data;
/* 保存uid到connection的映射,这样可以方便的通过uid查找connection,
* 实现针对特定uid推送数据
*/
$worker->uidConnections[$connection->uid] = $connection;
}
echo $connection->uid."msg \n";
};
//当客户端连接时
$worker->onConnect = function ($connection){
echo $connection->id."connect \n";
};
// 当有客户端连接断开时
$worker->onClose = function ($connection) use ($worker) {
global $worker;
if (isset($connection->uid)) {
// 连接断开时删除映射
unset($worker->uidConnections[$connection->uid]);
}
echo $connection->uid."close \n";
};
// 运行所有的worker(其实当前只定义了一个)
Worker::runAll();
}
// 向所有验证的用户推送数据
public function broadcast($message)
{
global $worker;
foreach ($worker->uidConnections as $connection) {
$connection->send($message);
}
return true;
}
// 针对uid推送数据
public function sendMessageByUid($uid, $message)
{
global $worker;
if (isset($worker->uidConnections[$uid])) {
$connection = $worker->uidConnections[$uid];
$connection->send($message);
return true;
}
return false;
}
}
启动服务,必须通过php指令启动
% php -f index.php Ws/start start
Workerman[index.php] start in DEBUG mode
------------------------------------------- WORKERMAN --------------------------------------------
Workerman version:4.0.19 PHP version:5.6.40
-------------------------------------------- WORKERS ---------------------------------------------
proto user worker listen processes status
tcp jiqing none websocket://0.0.0.0:8000 1 [OK]
--------------------------------------------------------------------------------------------------
Press Ctrl+C to stop. Start success.
Start
% php -f index.php Ws/start start -d
Workerman[index.php] start in DAEMON mode
------------------------------------------- WORKERMAN --------------------------------------------
Workerman version:4.0.19 PHP version:5.6.40
-------------------------------------------- WORKERS ---------------------------------------------
proto user worker listen processes status
tcp jiqing none websocket://0.0.0.0:8000 1 [OK]
--------------------------------------------------------------------------------------------------
Input "php index.php Ws/start stop" to stop. Start success.
% php -f index.php Ws/start stop
Workerman[index.php] stop
Workerman[index.php] is stopping ...
Workerman[index.php] stop success
客户端连接
ws = new WebSocket("ws://127.0.0.1:8000");
ws.onopen = function() {
alert("连接成功");
ws.send('uid1');
};
ws.onmessage = function(e) {
alert(e.data);
};
服务端主动发送消息
public function send()
{
// 建立socket连接到内部推送端口
$client = stream_socket_client('tcp://127.0.0.1:'.C('WS_INNER_PORT'), $errno, $errmsg, 1);
// 推送的数据,包含uid字段,表示是给这个uid推送
$data = array('uid' => '', 'data' => JsonUtil::jsonEncode(['name'=>'张三','age'=>15]));
// 发送数据,注意8001端口是Text协议的端口,Text协议需要在数据末尾加上换行符
fwrite($client, json_encode($data) . "\n");
// 读取推送结果
$res = trim(fread($client, 8192),"\n");
$this->json->ok($res);
}
升级,同时向多个相同uid,发送消息
<?php
/**
* workerman
*/
namespace AdminApi\Action;
use Common\Util\JsonUtil;
use Workerman\Worker;
class WsAction extends CommonAction
{
public function start()
{
global $worker;
$worker = new Worker('websocket://0.0.0.0:'.C('WS_PORT')); // 8000
// 这里进程数必须设置为1
$worker->count = 1;
// worker进程启动后建立一个内部通讯端口
$worker->onWorkerStart = function ($worker) {
// 开启一个内部端口,方便内部系统推送数据,Text协议格式 文本+换行符
$inner_text_worker = new Worker('Text://0.0.0.0:'.C('WS_INNER_PORT')); // 8001
$inner_text_worker->onMessage = function ($connection, $buffer) {
global $worker;
// $data数组格式,里面有uid,表示向那个uid的页面推送数据
$data = JsonUtil::jsonDecode($buffer);
$uid = $data['uid'];
// 发送数据
if ($uid) {
$res = $this->sendMessageByUid($uid, $data['data']);
} else {
$res = $this->broadcast($data['data']);
}
// 返回推送结果
$connection->send($res ? 'ok' : 'fail');
echo "inner text Msg $uid \n";
};
$inner_text_worker->listen();
echo "Start \n";
};
// 新增加一个属性,用来保存uid到connection的映射
$worker->uidConnections = array();
// 当有客户端发来消息时执行的回调函数
$worker->onMessage = function ($connection, $data) use ($worker) {
// 判断当前客户端是否已经验证,既是否设置了uid
if (!isset($connection->uid)) {
// 没验证的话把第一个包当做uid(这里为了方便演示,没做真正的验证)
$connection->uid = $data;
/**
* 保存uid到connection的映射,这样可以方便的通过uid查找connection,
* 实现针对特定uid推送数据
*/
$worker->uidConnections[$connection->uid][$connection->id] = $connection;
}
echo $connection->uid."-".$connection->id."msg \n";
};
//当客户端连接时
$worker->onConnect = function ($connection){
echo $connection->id."connect \n";
};
// 当有客户端连接断开时
$worker->onClose = function ($connection) use ($worker) {
global $worker;
if (isset($connection->uid)) {
// 连接断开时删除映射
unset($worker->uidConnections[$connection->uid][$connection->id]);
}
echo $connection->uid."-".$connection->id."close \n";
};
// 运行所有的worker(其实当前只定义了一个)
Worker::runAll();
}
// 向所有验证的用户推送数据
public function broadcast($message)
{
global $worker;
foreach ($worker->uidConnections as $connection) {
$connection->send($message);
}
return true;
}
// 针对uid推送数据
public function sendMessageByUid($uid, $message)
{
global $worker;
if (isset($worker->uidConnections[$uid])) {
foreach($worker->uidConnections[$uid] as $connection) {
$connection->send($message);
}
return true;
}
return false;
}
}