php使用workman框架实现socket服务以及连接客户端

  • 1. 解决什么问题,为什么要用workman  socket服务

都知道游戏安装包很大,渠道推广时,需要对游戏进行分包处理,而PHP命令模式是单进程,一次只能分一次包,故这里用workman实现socket服务开启多进程,对游戏进行分包处理(一个进程处理一个分包,多个进程可以同时处理多个分包)

  • 2. 服务端代码

server.php

<?php
/**
 * 分包程序.切记不能有die或exit出现.
 *
 * User: yzm
 * Data: 2018/1/16
 */

require_once './vendor/workerman/workerman/Autoloader.php';
require_once './Lib/Function.php';

require_once __DIR__ . '/Lib/Db.php';
require_once __DIR__ . '/Lib/DbConnection.php';
require_once __DIR__ . '/Config/Db.php';

use Workerman\Worker;

// #### create socket and listen 1234 port ####
$tcp_worker = new Worker("tcp://0.0.0.0:9998");

/**
 * 定义常量.
 */
define('REP_SUCCESS', 0); // 成功
define('REP_FAIL', -1); // 失败
define('REP_FAIL_NO_COMPLETED', 1); // 文件未上传完成

// 16 processes,与cpu个数相同
$tcp_worker->count = 16;
$msg = '';

define('ORGPKG', '/Volumes/VMware\ Shared\ Folders/orgpkg/');
define('DISTPKG', '/Volumes/VMware\ Shared\ Folders/');
//define('SYS_IP', '39.108.223.28');
define('SYS_IP', '120.92.142.115');
define('IOS_URL','http://ios.package.tonguu.cn/');

// Emitted when new connection come
$tcp_worker->onConnect = function ($connection) {
    $connection->sized = 0;

    // xcode调用脚本
    $certMobile = '/mnt/www/DIVIDE_PKG/Cert/%d/mslabEnt.mobileprovision'; // 证书文件
    $shell = "/mnt/www/DIVIDE_PKG/Lib/dividePkg/resign  sign -ipapath  %s  -destpath %s  -pppath %s -agentid %s";

    $connection->shell = $shell;
    $connection->pppath = $certMobile;

    echo date("Y-m-d H:i:s") . " connect!" . getclientip() . PHP_EOL;

};

/**
 * 响应结果.
 *
 * @author yzm
 */
function resonse($conn, $msg, $error = REP_FAIL, $data = [])
{
    $res = ['msg' => $msg, 'error' => intval($error)];
    if (!empty($data)) {
        $res['content'] = $data;
    }

    debug($res);

    // 返回JSON数据格式到客户端 包含状态信息
    $rst = json_encode($res);

    $conn->send($rst);
}

// Emitted when data received
$tcp_worker->onMessage = function ($connection, $data) {
    set_time_limit(0);
    ini_set('memory_limit', -1);

    $db = \Lib\Db::instance('btmox');
    $data = @json_decode($data, true);

    try{
        if (empty($data['authId'])) {
            throw new \Exception('授权文件参数错误');
        }

        //1. 查询所有待分包的ios渠道包
        $iosPkg = $db
            ->select('a.id,a.vid,a.filename,a.agent,d.pinyin,b.name,c.package_name')
            ->from('cy_ct_ios_package a')
            ->where("a.status=0 AND c.is_send=1")
            ->leftJoin('cy_ct_ios_mobileversion b','b.id=a.m_v_id')
            ->rightJoin('cy_ct_ios_version c','c.id=a.vid')
            ->leftJoin('cy_game d','d.id=c.game_id')
            ->orderByASC(['a.create_time'])->query();

        if(empty($iosPkg)) throw new \Exception('没有需要待分包的数据'.PHP_EOL);

        //2. 分包
        foreach($iosPkg as $one){
            try{
                //对当前正要分的包把状态改为‘分包中'
                $db->update('cy_ct_ios_package')->cols([
                    'status' => 2,
                ])->where("id=".$one['id'])->query();

                $filename = $one['pinyin'];
                // 渠道分包
                $verId = @$one['vid'];
                $agent = @$one['agent'];
                $location = isset($data['location']) ? $data['location'] : 1;
                $authId = @intval($data['authId']); // 授权文件

                if (empty($verId) || empty($agent)) {
                    throw new \Exception("分包失败:".$one['id']."版本、渠道为空\r\n");
                }

                // 替换\,否则PHP验证不文件是否存在
                $orgPkg = str_replace('\\', '', ORGPKG) . "{$filename}.ipa";

                debug($one['id'].'原包:' . $orgPkg);

                debug($one['id'].'是否是文件:' . is_file($orgPkg));

                if (!is_file($orgPkg)) {
                    throw new \Exception("分包失败:".$one['id']."母包不存在-$orgPkg\r\n");
                }

                // 从新拼接文件
                $orgPkg = ORGPKG . "{$filename}.ipa";

                // 获取目标包存放路径
                $distPkgPath = getDistPkgPath($location);

                $distPkg = $distPkgPath . "$filename/vers_{$verId}/{$filename}_$agent.ipa";
                debug('渠道分包地址:' . $distPkg);
                if (file_exists($filename)) {
                    @unlink($filename);
                }

                // 替换授权文件
                $certMobile = sprintf($connection->pppath, $authId);

                // 渠道分包
                list($msg, $code) = dividePkg($connection->shell, $orgPkg, $distPkg, $agent, $certMobile);

                debug('$code' . $code);

                if ($code != 0) {
                    throw new \Exception("分包失败:".$msg."\r\n");
                }

                $distPkg = str_replace($distPkgPath, '', $distPkg);

            }catch (\Exception $ex){
                debug($ex->getMessage());
                $code = -1;
                $msg = $ex->getMessage();
            }

            //3. 分包后更新分包结果,状态,下载地址
            $status = $code == 0 ? 1 : 2;
            $sdata['status'] = $status;
            $sdata['message'] = $msg;
            if($status == 1){
                $sdata['url'] = IOS_URL.$distPkg;
            }
            $db->update('cy_ct_ios_package')->cols($sdata)->where("id=".$one['id'])->query();
        }

        resonse($connection, $msg,$code);
    }catch (\Exception  $ex){
        resonse($connection, $ex->getMessage());
    }
};

// Emitted when new connection come
$tcp_worker->onClose = function ($connection) {
    echo date("Y-m-d H:i:s") . " closed!" . PHP_EOL;
};

Worker::runAll();
  • 3. 客户端代码

client.php

<?php

/**
 * 读取socket数据.
 *
 * @author yzm
 *
 * @param $socket
 * @param bool|true $isDividePkg
 * @return array|null|string
 */
function socketRead($socket, $isDividePkg = true)
{
    $rst = null;

    $buf = socket_read($socket, 8192);
    if ($isDividePkg) {
        $_buf = @json_decode($buf, true);
        $rst = !empty($_buf) ? [$_buf['error'], $_buf['msg'], @$_buf['content']] : $buf;
    } else {
        $rst = $buf;
    }

    return $rst;
}

/**
 * 向物理机发起socket请求.
 *
 * @param $args 参数
 * @return bool
 * @throws \Exception
 */
function sendSocket($args)
{
    set_time_limit(0);
    ini_set('memory_limit', -1);

    $type = isset($args['type']) ? $args['type'] : 0;

    if (!$type) throw new \Exception('类型参数错误');

    $port = 9998;
    $ip = "127.0.0.1";

    // 创建socket
    $socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);

    if ($socket <= 0) throw new \Exception('创建socket失败,REASON:' . socket_strerror($socket));

    try {

        // 连接服务器
        $result = socket_connect($socket, $ip, $port);
        if ($result < 0 || is_null($result) || !$result) throw new \Exception('连接失败,REASON:' . socket_strerror($result));

        $in = json_encode($args);

        // 写入文件信息
        if (!socket_write($socket, $in, strlen($in))) throw new \Exception('消息发送失败,REASON:' . socket_strerror($socket));

        // 读取socket返回的数据
        list($error, $msg, $data) = socketRead($socket);

        if ($type != 3 && $error != 0) throw new \Exception('104服务器异常,REASON:' . $msg);

        // 关闭socket
        socket_close($socket);

        switch ($type) {
            case 2: // 分包
                $rst = $data['url'];
                break;
            case 3: // 检测文件
                if ($error == -1) {
                    throw new \Exception('检测文件失败,REASON:' . $msg);
                }

                $rst = $error;
                break;
            default:
                $rst = true;
                break;
        }

    } catch (\Exception $ex) {

        // 关闭socket
        @socket_close($socket);

        throw new \Exception($ex->getMessage());
    }

    return $rst;
}

/**
 * 分包程序.切记不能有die或exit出现.
 *
 * User: yzm
 * Data: 2018/1/16
 */
require_once './Lib/Function.php';

$i=0;
while ($i<30){
    try{
        $data['type'] = 1;
        $data['authId'] = 2;
        $data['location'] = 1;
        sendSocket($data);
    }catch (\Exception $ex){
        echo $ex->getMessage();
    }
    $i++;
    sleep(5);
}
 
  • 4. 使用

    a. 开启服务

php server.php start  //可以看到开启了多个进程

b. 客户端连接

php client.php  //从代码知道,里头用了循环,可以多次连接服务,同时发送数据,服务端会把结果返回

到此这篇关于php使用workman框架实现socket服务以及连接客户端的文章就介绍到这了,更多相关php使用workman内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • PHP实现两种排课方式

    两种排课方式: 固定每周的固定时间上课(例:共上20节,每周六.周日早上8点-10点上课.假如今天周六凌晨1点,那么排课也需要从今天开始)总共上几个周,每周上课时间比较个性化(例:共上三周,第一周周一周二早上8点-10点上课:第二周周三周四下午8点-10点上课:第三周周日中午11点-12点上课.) 第一种排课比较好实现,简要代码如下: /** * 生成日期列表 * * @param int $startDate 开始日期 时间戳格式 * @param array $timeList 课时计划列表

  • php生成用户密码的两种方式

    目录 一.md5密码 二.hash密码 PS:php生成随机密码的几种方法 方法一: 方法二: 方法三: 方法四: 在用户系统中,生成用户的密码是很重要的,而简单的密码必然给一些不法用户开了一些门户,这里列出几种常见的密码生成方式 一.md5密码 这种应该算是最常见的密码加密方式了 md5是属于非对称加密中的一种,这种密码方式其实说安全也很安全,而因为哈希碰撞的存在,会导致可能会出现漏洞 最好是在加密的时候,加入混淆字符串,如下所示 <?php //加密 function md5_passwor

  • php类中static与self的使用区别浅析

    使用 self:: 或者 __CLASS__ 对当前类的静态引用,取决于定义当前方法所在的类: 使用 static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的.也可以称之为"静态绑定",因为它可以用于(但不限于)静态方法的调用. 静态绑定是PHP 5.3.0,增加的一个功能 用于在继承范围内引用静态调用的类 php类中static和self的区别,php类的定义中经常能看到self和static,在运行的时候经常发现结果并没有区别,但是肯定不是没有区别,因为没有区别为

  • 详细分析PHP7与PHP5区别

    1.php标量类型和返回类型声明 #主要分为两种模式,强制性模式和严格模式 declare(strict_types=1) #1表示严格类型校验模式,作用于函数调用和返回语句:0表示弱类型校验模式. 2.NULL合并运算符 $site = isset($_GET['site']) ? $_GET['site'] : 'wo'; #简写成 $site = $_GET['site'] ??'wo'; 3.组合预算符 // 整型比较 print( 1 <=> 1);print(PHP_EOL); p

  • php如何用PDO操作大数据对象

    目录 什么是大数据对象 "大"通常意味着"大约 4kb 或以上",尽管某些数据库在数据达到"大"之前可以轻松地处理多达 32kb 的数据.大对象本质上可能是文本或二进制形式的,我们在 PDOStatement::bindParam() 或 PDOStatement::bindColumn() 调用中使用 PDO::PARAM_LOB 类型码可以让 PDO 使用大数据类型.PDO::PARAM_LOB 告诉 PDO 作为流来映射数据,以便能使用 P

  • php使用workman框架实现socket服务以及连接客户端

    1. 解决什么问题,为什么要用workman  socket服务 都知道游戏安装包很大,渠道推广时,需要对游戏进行分包处理,而PHP命令模式是单进程,一次只能分一次包,故这里用workman实现socket服务开启多进程,对游戏进行分包处理(一个进程处理一个分包,多个进程可以同时处理多个分包) 2. 服务端代码 server.php <?php /** * 分包程序.切记不能有die或exit出现. * * User: yzm * Data: 2018/1/16 */ require_once

  • nodejs socket服务端和客户端简单通信功能

    本文实例讲述了通过node.js的net模块实现nodejs socket服务端和客户端简单通信功能,可以用作客户端对服务端的端口监听以及事件回执. server端代码 var net = require('net'); //模块引入 var listenPort = 8080;//监听端口 var server = net.createServer(function(socket){ // 创建socket服务端 console.log('connect: ' + socket.remoteA

  • Golang 实现Socket服务端和客户端使用TCP协议通讯

    Socket服务器是网络服务中常用的服务器.使用go语言实现这个业务场景是很容易的. 这样的网络通讯,需要一个服务端和至少一个客户端. 我们计划构建一个这样的通讯工程.服务端启动后等待客户端的访问.客户端发送一段信息给服务端.服务端接收到信息后,再回馈给客户端一段信息. 首先要建立服务端.服务端最先要做的事情就是"建立Socket端口监听". netListen, err := net.Listen("tcp", "localhost:1024"

  • Android Socket服务端与客户端用字符串的方式互相传递图片的方法

    发送图片: 首先找到具体传递的图片: <span style="font-family: comic sans ms,sans-serif; font-size: 16px;">private Bitmap getimage(String srcPath) { BitmapFactory.Options newOpts = new BitmapFactory.Options(); // 开始读入图片,此时把options.inJustDecodeBounds 设回true了

  • Python socket如何实现服务端和客户端数据传输(TCP)

    目录 socket服务端和客户端数据传输(TCP) socket服务端,客户端互相通信 socket服务端和客户端数据传输(TCP) 服务器端: import socket #创建一个socket对象 socket_server = socket.socket(socket.AF_INET, socket.SOCK_STREAM) host = "127.0.0.1" port = 9999 #绑定地址 socket_server.bind((host, port)) #设置监听 so

  • 使用socket进行服务端与客户端传文件的方法

    逻辑: 1.客户端将需要查找的文件名以流的形式传给服务端 2.服务端接受客户端的连接,把流转化为字符串,进行一个目录的遍历,查找是否存在需要的文件,若未找到,则输出未找到,若找到,则将文件转化为流,传给客户端 3.客户端准备接受,将服务端传过来的流转化为文件,存储下载. 4,至此,完成一个简单的客户端与服务端传输文件的小栗子~ Client.Java package com.ysk; import java.io.BufferedReader; import java.io.FileOutput

  • PHP基于socket实现的简单客户端和服务端通讯功能示例

    本文实例讲述了PHP基于socket实现的简单客户端和服务端通讯功能.分享给大家供大家参考,具体如下: 服务器端: <?php set_time_limit(0); $host="localhost"; $port=1001; //创建一个连接 $socket=socket_create(AF_INET,SOCK_STREAM,SOL_TCP)or die("cannot create socket\n"); //绑定socket到端口 $result=soc

  • nodejs socket实现的服务端和客户端功能示例

    本文实例讲述了nodejs socket实现的服务端和客户端功能.分享给大家供大家参考,具体如下: 使用node.js的net模块能很快的开发出基于TCP的服务端和客户端.直接贴代码. server.js /** * Created with JetBrains WebStorm. * User: Administrator * Date: 12-10-26 * Time: 下午3:44 * To change this template use File | Settings | File T

  • Java基于socket服务实现UDP协议的方法

    本文实例讲述了Java基于socket服务实现UDP协议的方法.分享给大家供大家参考.具体如下: 示例1: 接收类: package com.socket.demo; import java.io.IOException; import java.net.DatagramPacket; import java.net.DatagramSocket; public class UDPReceiveDemo { public static void main(String[] args) throw

  • Android socket实现原理详解 服务端和客户端如何搭建

    本文实例为大家分享了Android socket的实现原理,供大家参考,具体内容如下 Socket套接字 是网络上具有唯一标识的IP地址和端口号组合在一起才能构成唯一能识别的标识符套接字. socket实现的原理机制: 1.通信的两端都有Socket 2.网络通信其实就是Socket间的通信 3.数据在两个Socket间通过IO传输 建立Socket(客户端)和ServerSocket(服务器端) 建立连接后,通过Socket中的IO流进行数据的传输 关闭socket 同样,客户端与服务器端是两

随机推荐