Skynet服务器框架(五) 使用pbc(protobuf)

引言:

假如我们要建立的skynet服务器与客户端的连接方式为长连接,且选择了Google的Protobuf来定制我们的网络协议,那么,接下来我们要解决的问题就是:如何在skynet框架中使用socket+protobuf

API

几个常用的skynet接口:

*   输出错误信息:          skynet.error(...)
*   获取本地服务句柄方式:   skynet.localname(...)
*   设置定时器方式:        skynet.timeout(...)
*   skynet强制退出方式:    skyname.abort()
*   服务开始方式:          skynet.start(...)
*   服务注销方式:          skynet.exit()
*   发送原始文本消息方式:   skynet.core.send(...)

更多skynet.lua的API解析可以参考:skynet API:skynet.lua

protobuf在skynet中使用:

由于protobuf的lua版本的支持存在着部分缺陷,为了避免踩坑,这里我们直接使用云风博客中推荐的pbc动态proto解析库。

资源下载:

下载pbc:跟下载skynet源码一样,通过gitpbc的源码克隆到本地:

bogon:project linshuhe$ git clone https://github.com/cloudwu/pbc.git
Cloning into ‘pbc‘...
remote: Counting objects: 1156, done.
remote: Total 1156 (delta 0), reused 0 (delta 0), pack-reused 1156
Receiving objects: 100% (1156/1156), 302.95 KiB | 310.00 KiB/s, done.
Resolving deltas: 100% (682/682), done.
Checking connectivity... done.

编译并合入项目:

  • 项目编译:

    可以在pbc根目录下运行make指令编译项目:

    [email protected]:/application/pbc$ sudo make
    cd build && ar rc libpbc.a ../build/o/context.o ../build/o/varint.o ../build/o/array.o ../build/o/pattern.o ../build/o/register.o ../build/o/proto.o ../build/o/map.o ../build/o/alloc.o ../build/o/rmessage.o ../build/o/wmessage.o ../build/o/bootstrap.o ../build/o/stringpool.o ../build/o/decode.o
    cd build && gcc -O2 -fPIC -Wall -I.. -L. -o addressbook ../test/addressbook.c -lpbc
    cd build && gcc -O2 -fPIC -Wall -I.. -L. -o pattern ../test/pattern.c -lpbc
    cd build && gcc -O2 -fPIC -Wall -I.. -L. -o pbc ../test/pbc.c -lpbc
    cd build && gcc -O2 -fPIC -Wall -I.. -L. -o float ../test/float.c -lpbc
    cd build && gcc -O2 -fPIC -Wall -I.. -L. -o map ../test/map.c -lpbc
    cd build && gcc -O2 -fPIC -Wall -I.. -L. -o test ../test/test.c -lpbc
    cd build && gcc -O2 -fPIC -Wall -I.. -L. -o decode ../test/decode.c -lpbc
    protoc -obuild/addressbook.pb test/addressbook.proto
    protoc -obuild/descriptor.pb test/descriptor.proto
    protoc -obuild/float.pb test/float.proto
    protoc -obuild/test.pb test/test.proto

    假如编译结果报错了:

    make: protoc:命令未找到
    make: *** [build/addressbook.pb] 错误 127

    这是因为当前环境还没安装`protobuf,安装步骤如下:

    • 指令安装:

      sudo apt-get install protobuf-c-compiler protobuf-compiler
    • 查询版本验证完成:
      linsh@ubuntu:/application/pbc$ protoc --version
      libprotoc 2.5.0
  • 工具编译:

    用终端进入pbc项目pbc/binding/lua目录下面编译出protobuf.so:

    cd pbc/binding/lua
    sudo make

    编译成功的话,将protobuf.so放在config文件中lua_cpath项配置的目录下面,同时将protobuf.lua放在config文件lua_path配置的目录下,就可以调用protobuf中的库方法。我当前项目这两项的配置如下:

    lua_path = root.."lualib/?.lua;"..root.."lualib/?/init.lua"
    lua_cpath = root .. "luaclib/?.so"

    则移动文件命令可以如下:

    sudo cp protobuf.so /application/skynet/luaclib
    sudo cp protobuf.lua /application/skynet/lualib

编译报错:

  • pbc/binding/lua目录下编译,出现以下错误:

    [email protected]:/application/pbc/binding/lua$ sudo make
    gcc -O2 -fPIC -Wall -shared -o protobuf.so -I../.. -I/usr/local/include -L../../build pbc-lua.c -lpbc
    pbc-lua.c:4:17: fatal error: lua.h: 没有那个文件或目录
    
    #include "lua.h"
    
                 ^
    compilation terminated.
    make: *** [protobuf.so] 错误 1

    报错原因:这里因为没有安装 lua5.3,不能是lua5.2或是lua5.1等低版本,否则会报错。因为pbc用到了lua_rotate这是lua5.3新增的类型。

    解决方案:安装一下 lua5.3 即可解决,步骤如下:

    • lua官网下载lua5.3的安装包:

      lua-5.3.0.tar.gz

      也可以使用命令行下载工具 axel

      [email protected]:/application/pbc/binding$ sudo axel http://www.lua.org/ftp/lua-5.3.0.tar.gz
      初始化下载: http://www.lua.org/ftp/lua-5.3.0.tar.gz
      File size: 278045 bytes
      打开输出文件 lua-5.3.0.tar.gz
      开始下载
      
      [  0%]  .......... .......... .......... .......... ..........  [  53.8KB/s]
      [ 18%]  .......... .......... .......... .......... ..........  [  73.1KB/s]
      [ 36%]  .......... .......... .......... .......... ....
      连接 3 结束
              ,,,,,,,,,, ,,,,,,,,,, ,,,,,,,,,, ,,,,,,,,,, ,,,,......  [  96.9KB/s]
      [ 55%]  .......... .......... .......... .......... ..........  [  92.5KB/s]
      [ 73%]  .......... .......... .......... ......
      连接 0 结束
              ,,,,,,,,,, ,,,,,,,,,, ,,,,,,,,,, ,,,,,,.... ........
      连接 1 结束
              ,,,,,,,,,, ,,,,,,,,,, ,,,,,,,,,, ,,,,,,,,,, ,,,,,,,,..  [  80.5KB/s]
      [ 92%]  .......... .......... .
      
      271.5 千字节 已下载,用时 6 秒。(43.52 千字节/秒)
    • 使用压缩包进行解压和安装:

      先将下载好的文件拷贝到usr/local/src目录下

      sudo cp lua-5.3.0.tar.gz /usr/local/src

      依次执行以下指令:

      sudo tar zxf lua-5.3.0.tar.gz
      cd lua-5.3.0/
      sudo make linux test

      创建软链接,是lua可以在当前环境下随处可用:

      sudo ln -s /usr/local/src/lua-5.3.0/src/lua  /usr/bin/lua

      测试一下软连接是否成功:

      linsh@ubuntu:/usr/local/src/lua-5.3.0/src$ lua
      程序 ‘lua‘ 已包含在下列软件包中:
      * lua5.2
      * lua5.1
      * lua50
      请尝试:sudo apt-get install <选定的软件包>

      我也出现过这个问题,最后查到原因是因为添加软链接时第一个地址(lua的安装地址)/usr/local/src/lua-5.3.0/src/lua被我写成了/usr/local/src/lua5.3.0/src/lua,解决问题和可以看到:

      linsh@ubuntu:/usr/bin$ lua
      Lua 5.3.0  Copyright (C) 1994-2015 Lua.org, PUC-Rio
      > 
    • 修改pbc/binding/lua/Makefile中lua库的地址配置信息LUADIR为当前lua安装的地址:
      LUADIR = /usr/local/src/lua-5.3.0/src
    • 再次运行编译指令,并查询当前目录下文件的变化:
      linsh@ubuntu:/application/pbc/binding/lua$ sudo make
      gcc -O2 -fPIC -Wall -shared -o protobuf.so -I../.. -I/usr/local/src/lua-5.3.0/src -L../../build pbc-lua.c -lpbc
      linsh@ubuntu:/application/pbc/binding/lua$ ls
      build_ios.sh  Makefile  parser.lua  pbc-lua.c  protobuf.lua  protobuf.so  README.md  test2.lua  test.lua  testparser.lua

      可以看到多出了一个protobuf.so输出文件。

测试:

  • 先在项目根目录下创建一个protos文件夹,用来存放协议文件,创建一个Person.proto协议文件,内容如下:

    sudo mkdir protos
    cd protos
    sudo vi Person.proto

    协议文件的内容如下:

    packae cs;
    message Person {
    required string name = 1;
    required int32 id = 2;        // Unique ID number for this person.
    optional string email = 3;
    enum PhoneType {
    MOBILE = 0;
    HOME = 1;
    WORK = 2;
    }
    message PhoneNumber {
    required string number = 1;
    optional PhoneType type = 2 [default = HOME];
    }
    repeated PhoneNumber phone = 4;
    }
  • 将协议文件到处为.pb格式:
    linsh@ubuntu:/application/skynet/protos$ sudo protoc --descriptor_set_out Person.pb Person.proto
    linsh@ubuntu:/application/skynet/protos$ ls
    Person.pb  Person.proto

    目录下多出了一个对应的.pb文件。

在lua中注册对应的协议文件:

  • 引入protobuf.lua

    local pb = require "protobuf"
  • 注册.proto协议文件所对应的.pb文件,注册方法有两种:
    • 方法一:直接注册文件:

      pb.register_file "Person.pb"
    • 方法二:通过io读取文件,然后再获取文本内容进行注册:
      file = io.open("Person.pb","rb")
      buffer = file:read "*a"
      file:close()
      pb.register(buffer)
  • 通过 encodedecode 两个接口来实现编码和解码,完整测试脚本:
    local skynet = require "skynet"
    local protobuf = require "protobuf"
    
    skynet.start(function()
        protobuf.register_file "./protos/Person.pb"
        skynet.error("注册协议文件:Person.pb")
        stringbuffer = protobuf.encode("cs.Person",
        {
            name = "linsh",
            id = 1,
        })
        local data = protobuf.decode("cs.Person",stringbuffer)
        skynet.error("数据编码:name="..data.name..",id="..data.id)
    end)

    运行正确的结果:

    [:0100000c] 注册协议文件:Person.pb
    [:0100000c] 数据编码:name=linsh,id=1

除外,云风还自定义了一套协议格式sproto,据说比protobuf还要简单。

参考:

时间: 2024-10-02 09:08:54

Skynet服务器框架(五) 使用pbc(protobuf)的相关文章

Skynet服务器框架(一) Linux下的安装和启动

根据云风博客的描述,Skynet 的核心功能就是解决一个问题: 把一个符合规范的 C 模块,从 动态库(so文件)中启动起来,绑定一个永不重复(即使模块退出)的数字id做为其 handle.模块 被称为 服务(Service),服务间可以自由发送消息. 每个 模块 可以向 Skynet 框架注册一个 callback 函数,用来接收发给它的消息: 每个服务都是被一个个 消息包 驱动,当没有包到来的时候,它们就会处于 挂起状态,此状态对 CPU 资源零消耗.如果需要自主逻辑,则可以利用 Skyne

Skynet服务器框架(二) C源码剖析启动流程

引言: 之前我们已经完成了在Linux下配置安装 skynet 的环境,并成功启动了 skynet 服务框架,为了从底层更好地理解整个框架的实现过程,我们有必要剖析一下源码,由于底层的源码都是用C语言写的,lua脚本基本是用来进行业务层开发,所以我们从C源码开始解读框架.打开下载包的 skynet-src 目录,这里是skynet框架的核心C源码,接下来我们就要来解读 skynet_main.c 和 skynet_start.c 这两个与skynet启动相关的C源码. 1.入口函数和初始化: 我

游戏服务器框架:Leaf/go

Leaf 是一个使用 Go 语言开发的开源游戏服务器框架,注重运行效率并追求极致的开发效率.Leaf 适用于几乎所有的游戏类型.其主要的特性: 良好的使用体验.Leaf 总是尽可能的提供简洁和易用的接口,尽可能的提升开发的效率 稳定性.Leaf 总是尽可能的恢复运行过程中的错误,避免崩溃 多核支持.Leaf 通过模块机制和 leaf/go 尽可能的利用多核资源,同时又尽量避免各种副作用 良好的模块支持. 一个 Leaf 开发的游戏服务器由多个模块组成(例如 LeafServer),模块有以下特点

协作半驻留式服务器程序开发框架 --- 基于 Postfix 服务器框架改造

一.概述 现在大家在和Java, PHP, .net写应用程序时,都会用到一些成熟的服务框架,所以开发效率是比较高的.而在用C/C++写服务器程序时,用的就五花八门了,有些人用ACE, 有些人用ICE(号称比ACE强许多),等等,这类服务器框架及库比较丰富,但入门门槛比较高,所以更多的人是自己直接写服务器程序,初始写时觉得比较简 单,可时间久了,便会觉得难以扩展,性能低,容易出错.其实,Postfix 作者为我们提供了一个高效.稳定.安全的服务器框架模型,虽然Postfix主要用作邮件系统的 m

Leaf - 一个由 Go 语言编写的开发效率和执行效率并重的开源游戏服务器框架

转自:https://toutiao.io/posts/0l7l7n/preview Leaf 游戏服务器框架简介 Leaf 是一个由 Go 语言(golang)编写的开发效率和执行效率并重的开源游戏服务器框架.Leaf 适用于各类游戏服务器的开发,包括 H5(HTML5)游戏服务器. Leaf 的关注点: 良好的使用体验.Leaf 总是尽可能的提供简洁和易用的接口,尽可能的提升开发的效率 稳定性.Leaf 总是尽可能的恢复运行过程中的错误,避免崩溃 多核支持.Leaf 通过模块机制和 leaf

服务器框架思路(一)

本章主要说明服务器框架构成,有参考网狐服务器框架,仅记录自己想法,勿喷.. 一.登陆服务器,主要实现玩家账号注册,登陆,验证等功能. 二.大厅服务器,当玩家登陆验证通过之后,考虑到一些数据需要跟玩家实时的交互,比如邮件,比赛倒计时,聊天,公告,以及活动倒计时等,需要一个跟客户端长链接的服务器,即登陆上的玩家,都会链接到大厅服务器 三.工具服务器,游戏UI会有很多功能,比如修改昵称,签到,完成任务等:这些功能与服务端数据交互不频繁,但数据种类繁多,因此增加工具服务器,用来处理这些请求,使用短链接,

流行的python服务器框架有哪些

今天给大家推荐5款目前比较流行的python开发http://www.maiziedu.com/course/python/服务器框架,并向大家简单介绍下服务器框架的作用吧: 1.tonardo---- 多并发.轻量级应用, "非阻塞"的web 容器.类似tomcat.这个大家太熟悉了,就不多说了. 2.Twisted---- Twisted 是一个Python 应用程序和库文件的集成套件.其中包括全套页面服务器应用程序和基于文本模式的游戏引擎.还有一些诸如对数据流进行处理的模块.是一

PHP socket 服务器框架集

1.Swoole:重新定义PHP PHP语言的高性能网络通信框架,提供了PHP语言的异步多线程服务器,异步TCP/UDP网络客户端,异步MySQL,数据库连接池,AsyncTask,消息队列,毫秒定时器,异步文件读写,异步DNS查询.Swoole可以广泛应用于互联网.移动通信.企业软件.网络游戏.物联网.车联网.智能家庭等领域. 使用PHP+Swoole作为网络通信框架,可以使企业IT研发团队的效率大大提升,更加专注于开发创新产品. 2.workerman workerman是一个高性能的PHP

跨平台网络通信与服务器框架 acl 3.2.0 发布

acl 3.2.0 版本发布了,acl 是 one advanced C/C++ library 的简称,主要包括网络通信库以及服务器框架库等功能,支持 Linux/Windows/Solaris/FreeBsd/MacOS 平台:整个 acl 项目主要包含三个函数库:lib_acl(纯C开发的基础库,主要包含网络通信及服务器编程框架以及其它丰富的功能).lib_protocol(包含 HTTP/PING/SMTP 通信协议的C语言实现).lib_fiber(网络协程库).lib_acl_cpp