GIN+GORILLA=A GOLANG WEBSOCKET SERVER
鉴于聊天已然成为大部分app的基础功能,而大部分app用户基数有没有辣么大,常用的聊天server架构如xmpp或者消息队列实现之类的用起来还挺麻烦的,有比较难跟网页端做交互,加之H5标准落地,所以websocket已然成为一个轻巧可用性高的聊天server实现方法;
websocket的server常见的是用nodejs或者java的netty框架实现,netty相对重一点,direct buffer的内存泄露调起来比较麻烦,试了一下go,轻巧,稳定性不错,性能不错,所以用go实现了一下;
websocket的协议标准和基本概念网上一搜一片,这里不赘述;
http server用gin来做,websocket的handler则用gorilla,由于不重复造轮子,所以整个搭建的过程很快;
import (
"util"
"os"
"fmt"
"github.com/DeanThompson/ginpprof"
"github.com/gin-gonic/gin"
"runtime"
)
var (
logger * util.LogHelper
)
func main() {
runtime.GOMAXPROCS(runtime.NumCPU())
logFile,err := os.OpenFile("/var/log/gows.log",os.O_CREATE|os.O_RDWR,0777)
if err!=nil {
fmt.Println(err.Error())
os.Exit(0)
}
defer logFile.Close()
logger = util.NewLogger(logFile)
logger.Info("Starting system...")
wsHandler := new(WebSocketHandler)
gin.SetMode(gin.ReleaseMode)
r := gin.Default()
r.GET("/", func(c *gin.Context) {
wsHandler.HandleConn(c.Writer, c.Request)
})
ginpprof.Wrapper(r)//调试用 可以看到堆栈状态和所有goroutine状态
//err = r.Run(listenPath, certPath, keyPath) 这样可以支持wss
err = r.Run("127.0.0.1:8888")
if err != nil {
fmt.Println(err)
}
}
这样我们的入口就有了~
websocket的模式大概是 onopen onmessage onerror onclose四个callback来覆盖整个通信流程
所以我们来看下简易版本的websockethandler的实现
package main import (
"bytes"
"compress/gzip"
"encoding/json"
"errors"
"net/http"
"strconv"
"time"
"util"
"github.com/gorilla/websocket"
) var (
ctxHashMap = util.NewConcurrentMap()
)
//用来升级http协议到ws协议
type WebSocketHandler struct {
wsupgrader websocket.Upgrader
} func (wsh *WebSocketHandler) NewWebSocketHandler() {
wsh.wsupgrader = websocket.Upgrader{
ReadBufferSize: 4096,
WriteBufferSize: 4096,
}
} func (wsh *WebSocketHandler) onMessage(conn *websocket.Conn, ctx *ConnContext, msg []byte, msgType int) {
//处理文本消息 或者 2进制消息 2进制通常是些 gzip的文本 语音或者图片视频之类的一般会用其他云服务不然带宽会爆掉
if msgType == websocket.TextMessage {
wsh.processIncomingTextMsg(conn, ctx, msg)
}
if msgType == websocket.BinaryMessage { }
} func (wsh *WebSocketHandler) onOpen(conn *websocket.Conn, r *http.Request) (ctx *ConnContext, err error) {
if err := r.ParseForm(); err != nil {
return nil, errors.New("参数校验错误")
}
specialKey := r.FormValue("specialKey")
supportGzip := r.FormValue("support_gzip") ctx = &ConnContext{specialKey, supportGzip}
//用来标识一个tcp链接
keyString := ctx.AsHashKey() if oldConn, ok := ctxHashMap.Get(keyString); ok {
wsh.onClose(oldConn.(*websocket.Conn), ctx)
oldConn.(*websocket.Conn).Close()
}
ctxHashMap.Set(keyString, conn)
return ctx, nil
} func (wsh *WebSocketHandler) onClose(conn *websocket.Conn, ctx *ConnContext) {
logger.Info("client close itself as " + ctx.String())
wsh.closeConnWithCtx(ctx)
return
} func (wsh *WebSocketHandler) onError(errMsg string) {
logger.Error(errMsg)
}
func (wsh *WebSocketHandler) HandleConn(w http.ResponseWriter, r *http.Request) {
wsh.wsupgrader.CheckOrigin = func(r *http.Request) bool { return true }
conn, err := wsh.wsupgrader.Upgrade(w, r, nil)
if err != nil {
logger.Error("Failed to set websocket upgrade: " + err.Error())
return
}
defer conn.Close()
if ctx, err := wsh.onOpen(conn, r); err != nil {
logger.Error("Open connection failed " + err.Error() + r.URL.RawQuery)
return
} else {
conn.SetPingHandler(func(message string) error {
conn.WriteControl(websocket.PongMessage, []byte(message), time.Now().Add(time.Second))
return nil
})
for {
t, msg, err := conn.ReadMessage()
if err != nil {
logger.Error("READ ERR FROM " + ctx.String() + " ERR " + err.Error())
wsh.onClose(conn, ctx)
return
} switch t {
case websocket.TextMessage, websocket.BinaryMessage:
wsh.onMessage(conn, ctx, msg, t)
case websocket.CloseMessage:
wsh.onClose(conn, ctx)
return
case websocket.PingMessage:
case websocket.PongMessage:
} }
}
} func (wsh *WebSocketHandler) closeConnWithCtx(ctx *ConnContext) {
keyString := ctx.AsHashKey()
ctxHashMap.Remove(keyString)
return
}
func (wsh *WebSocketHandler) processIncomingTextMsg(conn *websocket.Conn, ctx *ConnContext, msg []byte) {
logger.Debug("CLIENT SAID " + string(msg))
sendMessageToAll(msg)
} func (wsh *WebSocketHandler) sendMessageToAll(msg []byte]) {
var gzMsg bytes.Buffer
gzWriter := gzip.NewWriter(&gzMsg)
gzWriter.Write(msg)
gzWriter.Flush()
gzWriter.Close()
for key, conn := range ctxHashMap.Items() {
if ctx, err := HashKeyAsCtx(key.(string)); err != nil {
wsh.onError(err.Error())
} else {
if ctx.supportGzip == "1" {
err = conn.(*websocket.Conn).WriteMessage(websocket.BinaryMessage, gzMsg.Bytes())
logger.Debug("send binary msg to " + ctx.String())
} else {
err = conn.(*websocket.Conn).WriteMessage(websocket.TextMessage, []byte(msg))
logger.Debug("send text msg to " + ctx.String())
}
if err != nil {
wsh.onClose(conn.(*websocket.Conn), ctx)
conn.(*websocket.Conn).Close()
wsh.onError("WRITE ERR TO " + key.(string) + " ERR:" + err.Error())
}
}
} }
因为删了一些线上代码的敏感信息 所以未必编译的过,不过差不多一个意思,主要看气质
里面的一个莫名其妙的叫做ctx的东西出现了很多次其实是connectionContext的缩写,一般链接形如ws://ip:port/?param=value¶m1=value1之类的形式,当然会加密,所以在onopen的时候会对url做一次基础校验,并且回记录url的一些关键参数标记,以用来确认消息到底要发送给谁
一个简单connContext实现如下
// connContext.go
package main import (
"errors"
"strings"
"util"
) type ConnContext struct {
specialKey string
supportGzip string
}
func HashKeyAsCtx(hashKey string) (*ConnContext,error){
values := strings.Split(hashKey,":")
if(len(values)!=2){
return nil,errors.New("艾玛 key不对: "+hashKey)
}else{
return &ConnContext{values[0],values[1]},nil
}
}
func (ctx *ConnContext) AsHashKey() string{
return strings.Join([]string{ctx.specialKey, ctx.supportGzip},":")
}
func (ctx * ConnContext) String () string{
return util.NewStringBuilder("specialkey: ",ctx.specialkey, " gzip ",ctx.supportGzip).String()
}
以上 一个简易的websocket server 就这样完成了 可喜可贺
有事儿寻这儿
GIN+GORILLA=A GOLANG WEBSOCKET SERVER的更多相关文章
- 使用Jetty搭建Java Websocket Server,实现图像传输
https://my.oschina.net/yushulx/blog/298140 How to Implement a Java WebSocket Server for Image Transm ...
- SpringBoot报错:Failed to load ApplicationContext(javax.websocket.server.ServerContainer not available)
引起条件: WebSocket+单元测试,单元测试报错! 解决方法: SpringBootTest增加webEnvironment参数. https://docs.spring.io/spring-b ...
- Golang websocket推送
Golang websocket推送 在工作用主要使用的是Java,也做过IM(后端用的netty websocket).最近想通过Golang重写下,于是通过websocket撸了一个聊天室. 项目 ...
- io.undertow.websockets.jsr.ServerWebSocketContainer cannot be cast to org.apache.tomcat.websocket.server.WsServerContainer
Caused by: java.lang.ClassCastException: io.undertow.websockets.jsr.ServerWebSocketContainer cannot ...
- springboot整合websocket后运行测试类报错:javax.websocket.server.ServerContainer not available
springboot项目添加websocket依赖后运行测试类报如下错误: org.springframework.beans.factory.BeanCreationException: Error ...
- ClassCastException: org.apache.tomcat.websocket.server.WsServerContainer cannot be cast to javax.websocket.server.ServerContainer
21:09:22.221 [MessageBroker-3] INFO c.t.s.s.impl.StockNewsServiceImpl - [2017-12-16 21:09:22] execut ...
- Caused by: java.lang.IllegalStateException: javax.websocket.server.ServerContainer not available
java.lang.IllegalStateException: Failed to load ApplicationContext at org.springframework.test.co ...
- swoole之建立 websocket server
一.代码部分 <?php /** * 为什么用WebSocket? * HTTP的通信只能由客户端发起 * * WebSocket 协议是基于TCP的一种新的网络协议.实现了浏览器与服务器全双工 ...
- springboot整合websocket后打包报错:javax.websocket.server.ServerContainer not available
项目整合了websocket以后,打包多次都没有成功,原来是报错了,报错内容如下: Error starting ApplicationContext. To display the conditio ...
随机推荐
- HDU 6464 权值线段树 && HDU 6468 思维题
免费送气球 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)Total Submi ...
- 2017 CCPC 哈尔滨站 题解
题目链接 2017 CCPC Harbin Problem A Problem B Problem D Problem F Problem L 考虑二分答案. 设当前待验证的答案为x 我们可以把第二 ...
- jenkins集群节点构建maven(几乎是坑最多的)
业务量变大时,单台的jenkins进行自动化构建部署,就显得没那么灵活,jenkins的集群并非像web服务器.mysql集群那样,jenkins的集群无需在额外的主机安装jenkins,但是用于ja ...
- CountDownLatch、CyclicBarrier、Samephore浅谈三大机制
CountDownLatch.CyclieBarrier与SamePhore都可用来控制线程的执行,那么他们之间有什么区别呢 CountDownLatch CountDowenlatch可以看成一个线 ...
- BZOJ1003物流運輸 DP + SPFA
@[DP, SPFA] Description 物流公司要把一批货物从码头A运到码头B.由于货物量比较大,需要\(n\)天才能运完.货物运输过程中一般要转 停好几个码头.物流公司通常会设计一条固定的运 ...
- 358. Rearrange String k Distance Apart
/* * 358. Rearrange String k Distance Apart * 2016-7-14 by Mingyang */ public String rearrangeString ...
- Java EE官方文档汇总
Java EE是一个开发规范标准,各个容器厂商根据标准去实现,比如Tomcat等,其中Oracle通过标准用GlassFish去实现. 5:https://docs.oracle.com/javaee ...
- 3D投影
3D投影方式的几大种类: 1.快门式 主动快门式即时分式,不过我们通常用前面的叫法,快门式3D眼镜(3D Shutter Glasses,也称作LC shutter glassesor active ...
- 关于Oracle中sysoper这个系统权限的问题
我们都知道Oracle数据库安装完之后.默认的会有这样几个系统角色或权限.nomal,sysdba,sysoper等等,之前每次登录Oracle的时候.都是直接以conn / as sysdba 的身 ...
- HDU 4403 A very hard Aoshu problem (DFS暴力)
题意:给你一个数字字符串.问在字符串中间加'='.'+'使得'='左右两边相等. 1212 : 1+2=1+2, 12=12. 12345666 : 12+3+45+6=66. 1+2+3+4 ...