/frontd

心动网络游戏通用网关

Primary LanguageGoMIT LicenseMIT

Build Status Coverage Status

简介

根据心动开放技术文档 统一的Gateway 中的要求: 整体架构对公网可以仅暴露1个网关入口,即 1个IP 1个端口。 并要求这个网关在需要时可以水平扩展。 因此需要游戏开发和使用网关来与服务器后端进行通讯。

本项目提供一种通用方案,可以直接部署生产环境,也可以进行改造后部署。

主要特点

  • 高性能、高并发
  • 无状态,可水平扩展
  • 同时支持TCP和HTTP两种模式
  • 安全(无明文后端地址及端口信息)
    • 后端地址端口使用AES(aes-256-cbc)加密
  • 免配置免维护
    • 无论后端地址端口发生什么变化,本网关并不需要重新配置或维护
  • 可使用容器部署 官方镜像 https://hub.docker.com/r/tomasen/frontd/

注意事项

切勿将 Secret Passphrase 写入客户端代码!

  • Secret Passphrase 用来生成加密的地址信息
  • 加密后的地址信息密文可以存放在客户端或通过其他方式发送给客户端,但切忌将 Secret Passphrase 写入客户端代码!
  • 后端默认超时时间为5秒,如需延长,请配置环境变量 BACKEND_TIMEOUT(单位为秒)。 启动命令范例如: docker run -e "SECRET=SomePassphrase" -e "BACKEND_TIMEOUT=10" tomasen/frontd /go/bin/frontd

编译

go builddocker build

部署服务端

  1. 通过环境变量 SECRET 设置解密用的秘钥

  2. 可以使用官方 Docker 镜像 tomasen/frontd

    启动命令范例如下:

    docker run -e "SECRET=SomePassphrase" tomasen/frontd /go/bin/frontd

通讯协议

客户端建立TCP连接后,以文本形式发送 加密并的后端地址端口信息 + \n 换行符。之后开始正常通讯即可。

如果出现后端地址端口无法连接等错误,会根据下表返回4个字节的文本错误码,如果是HTTP/WebSocket模式则会根据标准返回HTTP/WebSocket状态码:

错误码 含义
4101 后端服务器超时
4102 无法连接后端服务器
4103 数据头读取失败
4104 获取后端地址密文失败
4106 后端地址解密失败
4107 HTTP后端地址解析失败
4108 没有后端地址的HTTP请求
4109 获取后端地址密文失败(二进制模式)
4100 不被允许的IP地址

接入方式

  1. 生成 Passphrase 。并保存在安全的文档中。
  2. 使用上述 Secret Passphrase 部署服务端
  3. 使用 AES 算法加密文本格式的后端地址,生成 base64 编码的密文。可以使用在线工具如 [http://tool.oschina.net/encrypt] 生成密文 。也可以使用 openssl 命令行如 echo -n "127.0.0.1:62863" | openssl enc -e -aes-256-cbc -a -salt -k "p0S8rX680*48" 生成密文。
    • 例:当后端地址为 127.0.0.1:62863 时,如 Passphrase=p0S8rX680*48 , 密文结果应类似 U2FsdGVkX19KIJ9OQJKT/yHGMrS+5SsBAAjetomptQ0=
      注:上述方式都会使用随机Salt——这也是建议的方式。其结果是每次加密得出的密文结果并不一样,但并不会影响解密
  4. frontd 同时支持多种连接建立方式
    • TCP网关模式-Base64密文

      客户端与网关建立连接后,将后端地址的密文文本加一个换行符发送给网关。 根据前例密文,应该先发送如下的45个字节: U2FsdGVkX19KIJ9OQJKT/yHGMrS+5SsBAAjetomptQ0=\n

    • TCP网关模式-二进制密文(更少通讯量)

      客户端与网关建立连接后,先发送值为0x00的一个字节(byte), 再发送值为二进制密文长度的一个字节(byte), 再将后端地址的二进制密文发送给网关。 根据前例密文,应该先发送如下的34个字节: 0x00 0x20 0x53 0x61 0x6c 0x74 0x65 0x64 0x5f 0x5f 0x4a 0x20 0x9f 0x4e 0x40 0x92 0x93 0xff 0x21 0xc6 0x32 0xb4 0xbe 0xe5 0x2b 0x01 0x00 0x08 0xde 0xb6 0x89 0xa9 0xb5 0x0d

    • HTTP网关模式

      在HTTP请求中加入Header X-Cipher-Origin 并以后端地址密文为值 根据前例密文,HTTP请求应如: > GET / HTTP/1.1 > Host: game101.xd.com > X-Cipher-Origin: U2FsdGVkX19KIJ9OQJKT/yHGMrS+5SsBAAjetomptQ0=\n > User-Agent: curl/7.43.0 > Accept: / 注1:默认支持最大HTTP尺寸为8k,如需更大可以启动时配置环境变量MAX_HTTP_HEADER_SIZE

Benchmark 基准测试数据指标

  • 测试环境

    Mac Mini(i7, 16G ram)

  • 测试数据

    当地址缓存命中时(正常情形) frontd 为后端带来的额外延迟为 0.57ms
    当地址缓存完全无命时(极端情况) frontd 为后端带来的额外延迟为 1.05ms

BenchmarkEncryptText-4 200000 7345 ns/op BenchmarkDecryptText-4 500000 4279 ns/op BenchmarkEcho-4 1000 1151079 ns/op BenchmarkLatency-4 1000 1727021 ns/op BenchmarkNoHitLatency-4 1000 2205162 ns/op ```

  • 测试方法

    go test -bench .

Profiling

如果启动时通过环境变量 PPROF_PORT,就会在该端口启动 pprof 。使用方法可以参考 [https://golang.org/pkg/net/http/pprof/]

启动命令范例如下:

`docker run -e "SECRET=SomePassphrase" -e "PPROF_PORT=4044" -p 4044 tomasen/frontd /go/bin/frontd`

设计说明

frontd 在设计上是安全性+性能+易于接入+易于维护的折中方案。其中:

  • 使用加密地址,首先是出于安全角度: 内网端口可以被扫描显然是绝对不能被接受的。所以我们使用最强的AES-256-CBC加密方法,攻击者就不能通过监听数据和篡改数据,篡改变造后端目标地址,来实现访问任意后端IP端口的情况。
  • 使用密文地址,而不是预配置地址表,是出于易维护端角度。 这样运维人员可以不必另外维护一份地址表,也不需要每次增减后端服务器时都再去更新 frontd 的配置。
  • 使用文本而不是二进制通讯,是方便客户端和服务器列表部分功能的接入 相信这对于繁忙研发侧,是最快接入的方式。
  • 无状态,不增加业务功能 这是为了适配各种各样的后端服务。基本上无论后端架构如何,基本不需要改动关键代码来配合 frontd 工作。

折中意味着牺牲。因此,我们并不反对项目组在本代码基础上进行优化,但不建议弱化安全性的部分。 也欢迎提交 Issue 和 Pull Request。

Developing

Pull request must pass:

TODO

  • Improve test coverage to over 90%
  • 支持 WebSocket
  • 提供高可用的健康监测接口
  • 队列化请求,并发整形
  • 支持更多加密解密算法
  • 支持 consul 服务发现