mirror of
https://gitee.com/ShopeX/OMS
synced 2026-03-23 19:05:34 +08:00
246 lines
6.4 KiB
PHP
246 lines
6.4 KiB
PHP
<?php
|
|
/**
|
|
* Copyright 2012-2026 ShopeX (https://www.shopex.cn)
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
class prism_notify {
|
|
|
|
private $client;
|
|
private $connected = false;
|
|
private $sock;
|
|
private $_frames = array();
|
|
private $messages = array();
|
|
private $last_buf = '';
|
|
private $consuming = false;
|
|
const TextFrame = 0x01;
|
|
const BinaryFrame = 0x02;
|
|
const CloseFrame = 0x08;
|
|
const PingFrame = 0x09;
|
|
const PongFrame = 0x09;
|
|
|
|
const action_publish = 1;
|
|
const action_ack = 3;
|
|
const action_consume = 2;
|
|
|
|
function __construct(&$client){
|
|
$this->client = $client;
|
|
$this->client->register_handler(101, array($this, 'handle_upgrade'));
|
|
$this->connect();
|
|
}
|
|
|
|
/**
|
|
* pub
|
|
* @param mixed $routing_key routing_key
|
|
* @param mixed $message message
|
|
* @return mixed 返回值
|
|
*/
|
|
public function pub($routing_key, $message){
|
|
$size_routing_key = strlen($routing_key);
|
|
$size_message = strlen($message);
|
|
$this->send(self::action_publish, pack("na*Na*", $size_routing_key,
|
|
$routing_key, $size_message, $message));
|
|
}
|
|
|
|
/**
|
|
* close
|
|
* @return mixed 返回值
|
|
*/
|
|
public function close(){
|
|
if($this->connected){
|
|
fwrite($this->sock, $this->encode(self::CloseFrame));
|
|
fclose($this->sock);
|
|
$this->connected = false;
|
|
}
|
|
}
|
|
|
|
private function consume(){
|
|
if(!$this->consuming){
|
|
$this->send(self::action_consume);
|
|
$this->consuming = true;
|
|
}
|
|
}
|
|
|
|
private function send($type, $message=""){
|
|
if(!$this->connected || !@fwrite($this->sock,
|
|
$this->encode(self::BinaryFrame, pack("ca*", $type , $message)))){
|
|
$this->connected = false;
|
|
throw new prism_exception("websocket is not connected");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* ack
|
|
* @param mixed $tid ID
|
|
* @return mixed 返回值
|
|
*/
|
|
public function ack($tid){
|
|
return $this->send(prism_notify::action_ack, $tid);
|
|
}
|
|
|
|
/**
|
|
* 获取
|
|
* @return mixed 返回结果
|
|
*/
|
|
public function get(){
|
|
if(!$this->consuming){
|
|
$this->consume();
|
|
}
|
|
if (!$this->messages)
|
|
$this->recv_message();
|
|
|
|
return new prism_message($this, array_shift($this->messages));
|
|
}
|
|
|
|
/**
|
|
* handle_upgrade
|
|
* @param mixed $c c
|
|
* @param mixed $sock sock
|
|
* @return mixed 返回值
|
|
*/
|
|
public function handle_upgrade($c, $sock){
|
|
$this->client->log("connected");
|
|
$this->connected = true;
|
|
$this->consuming = false;
|
|
$this->sock = $sock;
|
|
register_shutdown_function(array($this, 'close'));
|
|
}
|
|
|
|
private function connect(){
|
|
$headers = array('Upgrade'=>'websocket',
|
|
'Sec-Websocket-Key' => $this->wskey(),
|
|
'Sec-WebSocket-Version' => 13,
|
|
'Sec-WebSocket-Protocol' => 'chat',
|
|
'Origin' => $this->client->base_url,
|
|
'Connection'=>'Upgrade');
|
|
$error = $this->client->get('notify', array(), $headers);
|
|
if($error){
|
|
$this->client->log("websocket handshake error: ".$error);
|
|
}
|
|
}
|
|
|
|
private function recv_message(){
|
|
$raw = fread($this->sock, 8192);
|
|
$raw = $this->last_buf . $raw;
|
|
$i = 0;
|
|
|
|
while($raw){
|
|
$i ++;
|
|
$len = ord($raw[1]) & ~128;
|
|
$raw = substr($raw, 2);
|
|
|
|
if ($len == 126) {
|
|
$arr = unpack("n", $raw);
|
|
$len = array_pop($arr);
|
|
$raw = substr($raw, 2);
|
|
} elseif ($len == 127) {
|
|
list(, $h, $l) = unpack('N2', $raw);
|
|
$len = ($l + ($h * 0x0100000000));
|
|
$raw = substr($raw, 8);
|
|
}
|
|
|
|
if(strlen($raw)>=$len){
|
|
$this->last_buf = '';
|
|
array_push($this->messages, substr($raw, 0, $len));
|
|
$raw = substr($raw, $len);
|
|
}else{
|
|
$this->last_buf .= $raw;
|
|
}
|
|
}
|
|
return $i;
|
|
}
|
|
|
|
private function encode($type, $data='') {
|
|
$b1 = 0x80 | ($type & 0x0f);
|
|
$length = strlen($data);
|
|
|
|
if($length <= 125)
|
|
$header = pack('CC', $b1, 128 + $length);
|
|
elseif($length > 125 && $length < 65536)
|
|
$header = pack('CCS', $b1, 128 + 126, $length);
|
|
elseif($length >= 65536)
|
|
$header = pack('CCN', $b1, 128 + 127, $length);
|
|
|
|
$key = 0;
|
|
$key = pack("N", rand(0, pow(255, 4) - 1));
|
|
$header .= $key;
|
|
|
|
return $header.$this->rotMask($data, $key);
|
|
}
|
|
|
|
private function wskey(){
|
|
return base64_encode(time());
|
|
}
|
|
|
|
private function rotMask($data, $key, $offset = 0) {
|
|
$res = '';
|
|
for ($i = 0; $i < strlen($data); $i++) {
|
|
$j = ($i + $offset) % 4;
|
|
$res .= chr(ord($data[$i]) ^ ord($key[$j]));
|
|
}
|
|
|
|
return $res;
|
|
}
|
|
|
|
}
|
|
|
|
class prism_exception extends Exception{
|
|
|
|
}
|
|
|
|
class prism_command{
|
|
var $type;
|
|
var $data;
|
|
|
|
function __construct($type, $data=""){
|
|
$this->type = $type;
|
|
$this->data = $data;
|
|
}
|
|
|
|
function __toString(){
|
|
return json_encode(array(
|
|
'type'=> &$this->type,
|
|
'data'=> &$this->data,
|
|
));
|
|
}
|
|
}
|
|
|
|
class prism_message{
|
|
|
|
public $body;
|
|
|
|
private $conn;
|
|
private $raw;
|
|
private $tid;
|
|
|
|
function __construct($conn, $raw){
|
|
$this->raw = $raw;
|
|
$this->data = json_decode($raw);
|
|
$this->conn = &$conn;
|
|
if($this->data){
|
|
$this->body = &$this->data->Body;
|
|
$this->tid = &$this->data->DeliveryTag;
|
|
}else{
|
|
$this->body = &$this->raw;
|
|
}
|
|
}
|
|
|
|
function ack(){
|
|
return $this->conn->ack($this->tid);
|
|
}
|
|
|
|
function __toString(){
|
|
return (string)$this->body;
|
|
}
|
|
} |