go语言实现websocket数据推送

主要分享下go语言实现的websocket推送。
有这样一个应用场景,服务端产生了一条很重要的数据,需要实时推送到所有客户端。
数据源采用redis 订阅方式,有数据产生就pub,然后服务sub读取数据,最后用 websocket模式推送数据到所有客户端。
这是基本设计思想。

首先 我们用第三方的库 websocket和一个生成uuid的库还有一个redis

go get -u github.com/gorilla/websocket
go get -u github.com/satori/go.uuid
go get -u github.com/go-redis/redis

开始服务端编程

server.go

 package main

import (
    "encoding/json"
    "fmt"
    "net/http"
    "io"

    "github.com/gorilla/websocket"
    "github.com/satori/go.uuid"
    "github.com/go-redis/redis"
)

type ClientManager struct {
    clients    map[*Client]bool
    broadcast  chan []byte
    register   chan *Client
    unregister chan *Client
}

type Client struct {
    id     string
    socket *websocket.Conn
    send   chan []byte
}

type Message struct {
    Sender    string `json:"sender,omitempty"`
    Recipient string `json:"recipient,omitempty"`
    Content   string `json:"content,omitempty"`
}

var manager = ClientManager{
    broadcast:  make(chan []byte),
    register:   make(chan *Client),
    unregister: make(chan *Client),
    clients:    make(map[*Client]bool),
}

func (manager *ClientManager) start() {
    for {
        select {
        case conn := <-manager.register: //新客户端加入
            manager.clients[conn] = true
            jsonMessage, _ := json.Marshal(&Message{Content: " a new socket has connected."})
            manager.send(jsonMessage, conn) //调用发送
        case conn := <-manager.unregister:
            if _, ok := manager.clients[conn]; ok {
                close(conn.send)
                delete(manager.clients, conn)
                jsonMessage, _ := json.Marshal(&Message{Content: "a socket has disconnected."})
                manager.send(jsonMessage, conn)
            }
        case message := <-manager.broadcast: //读到广播管道数据后的处理
        fmt.Println(string(message))
            for conn := range manager.clients {
                fmt.Println("每个客户端",conn.id)

                select {
                    case conn.send <- message: //调用发送给全体客户端
                    default:
                        fmt.Println("要关闭连接啊")
                        close(conn.send)
                        delete(manager.clients, conn)
                }
            }
        }
    }
}

func (manager *ClientManager) send(message []byte, ignore *Client) {
    for conn := range manager.clients {
        if conn != ignore {
            conn.send <- message //发送的数据写入所有的 websocket 连接 管道
        }
    }
}
//客户端写入后 激活这里读取
//想改成 读取redis(已废弃)
func (c *Client) read() {
    //pubsub := c.cache.Subscribe("mychannel1")
    defer func() {
        manager.unregister <- c
        c.socket.Close()
        fmt.Println("读关闭")
    }()

    for {
        
        _, message, err := c.socket.ReadMessage()
        //msg,err := c.getRedis()
        fmt.Println("是在不停的读吗?")
        if err != nil {
            manager.unregister <- c
            c.socket.Close()
            //c.cache.Close()
            fmt.Println("读不到数据就关闭?")
            break
        }
        jsonMessage, _ := json.Marshal(&Message{Sender: c.id, Content: string(message)})
        manager.broadcast <- jsonMessage  //激活start 程序 入广播管道
        fmt.Println("发送数据到广播")
    }
}
//写入管道后激活这个进程
func (c *Client) write() {
    defer func() {
        manager.unregister <- c
        c.socket.Close()
        fmt.Println("写关闭了")
    }()

    for {
        select {
        case message, ok := <-c.send: //这个管道有了数据 写这个消息出去
            if !ok {
                c.socket.WriteMessage(websocket.CloseMessage, []byte{})
                fmt.Println("发送关闭提示")
                return
            }

            err := c.socket.WriteMessage(websocket.TextMessage, message)
            if err != nil {
                manager.unregister <- c
                c.socket.Close()
                fmt.Println("写不成功数据就关闭了")
                break
            }
            fmt.Println("写数据")
        }
    }
}

func main() {
    fmt.Println("Starting application...")
    go manager.start()
    go manager.getRedisData()
    http.HandleFunc("/ws", wsPage)
    http.ListenAndServe(":12345", nil)
}

func wsPage(res http.ResponseWriter, req *http.Request) {
    //解析一个连接
    conn, error := (&websocket.Upgrader{CheckOrigin: func(r *http.Request) bool { return true }}).Upgrade(res, req, nil)
    if error != nil {
        //http.NotFound(res, req)
        //http 请求一个输出
        io.WriteString(res,"这是一个websocket,不是网站.")
        return
    }
    
    uid,_ := uuid.NewV4()
    sha1 := uid.String()
    
    //初始化一个客户端对象
    client := &Client{id: sha1, socket: conn,send: make(chan []byte)}
    //把这个对象发送给 管道
    manager.register <- client

    //go client.read()
    go client.write()
}

func (manager *ClientManager) getRedisData(){
    redisClient := redis.NewClient(&redis.Options{
        Addr:     "20.10.1.31:6381",
        Password: "", // no password set
        DB:       0,  // use default DB
    })
    redisSubscript :=redisClient.Subscribe("mychannel1")
    for{
        msg,err := redisSubscript.ReceiveMessage()
        if err != nil{
            redisClient.Close()
        }
        //manager.redisData<- msg.String()
        fmt.Println("重新读数据吧")
        jsonMessage, _ := json.Marshal(&Message{Sender: "hi", Content: msg.String()})
        manager.broadcast <- jsonMessage  //激活start 程序 入广播管道

    }
}

客户端

client.go

package main

import (
    "flag"
    "fmt"
    "net/url"
    "time"

    "github.com/gorilla/websocket"
)

var addr = flag.String("addr", "localhost:12345", "http service address")

func main() {
    u := url.URL{Scheme: "ws", Host: *addr, Path: "/ws"}
    var dialer *websocket.Dialer

    conn, _, err := dialer.Dial(u.String(), nil)
    if err != nil {
        fmt.Println(err)
        return
    }

    //go timeWriter(conn)

    for {
        _, message, err := conn.ReadMessage()
        if err != nil {
            fmt.Println("read:", err)
            return
        }

        fmt.Printf("received: %s\n", message)
    }
}

一个网页测试

js核心代码,起一个websocket连接然后去读取数据

<script type="text/javascript">

    if (window["WebSocket"]) {
        conn = new WebSocket("ws://localhost:12345/ws");
        conn.onclose = function(evt) {
            appendLog($("<div><b>Connection Closed.</b></div>"))
        }
        conn.onmessage = function(evt) {
            appendLog($("<div/>").text(evt.data))
        }
    } else {
        appendLog($("<div><b>WebSockets Not Support.</b></div>"))
    }
    });

</script>

最后在redis端写publish mychannel1 6

然后在客户端和web客户端都收到 received:

{"sender":"hi","content":"Message\u003cmychannel1: 6\u003e"}

基本思路

主线程里起两个并行协程,一个处理各项数据,一个循环读取redis订阅,当用户客户端连接上socket,再起一个这个客户端写的协程,用户处理当客户端退出后释放资源。然后用管道传递数据循环发送消息到客户端。

资料参考

https://blog.csdn.net/wangshubo1989/article/details/78250790

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,425评论 4 361
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,058评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,186评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,848评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,249评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,554评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,830评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,536评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,239评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,505评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,004评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,346评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,999评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,060评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,821评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,574评论 2 271
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,480评论 2 267

推荐阅读更多精彩内容