当前位置:首页 > 文章列表 > Golang > Go问答 > 重新建立websocket连接的优雅方法

重新建立websocket连接的优雅方法

来源:stackoverflow 2024-02-20 19:45:26 0浏览 收藏

各位小伙伴们,大家好呀!看看今天我又给各位带来了什么文章?本文标题《重新建立websocket连接的优雅方法》,很明显是关于Golang的文章哈哈哈,其中内容主要会涉及到等等,如果能帮到你,觉得很不错的话,欢迎各位多多点评和分享!

问题内容

我正在使用 http 和 websockets 构建单页面应用程序。用户提交表单,我将响应流式传输给客户端。下面是一个片段客户端。

var html = `<!doctype html>

<meta charset="utf-8">
<head>
</head>
<body>

<script>
var ws = new websocket("ws://localhost:8000/ws")
ws.onmessage = function(e) {
  document.getelementbyid("output").innerhtml += e.data + "<br>"
}

function submitfunction() {
  document.getelementbyid("output").innerhtml += ""
  return false
}
</script>

<form
enctype="multipart/x-www-form-urlencoded"
action="http://localhost:8000/"
method="post"
>`

这是服务器。如果请求不是 post,我会编写/渲染 html (parseandexecute),以建立新的 websocket 连接。如果请求是 post(来自表单),那么我开始处理并最终写入 websocket。

func (c *config) servehtml(w http.responsewriter, r *http.request) {
    if r.method == http.methodpost {
        //process
        channel <- data
    }

    c.parseandexecute(w)
}

func (sh *sockethandler) servehttp(w http.responsewriter, r *http.request) {
    ws, err := upgrader.upgrade(w, r, nil)
    if err != nil {
        w.write([]byte(fmt.sprintf("", err)))
        return
    }
    //defer ws.close()

    // discard received messages
    go func(c *websocket.conn) {
        for {
            if _, _, err := c.nextreader(); err != nil {
                c.close()
                break
            }
        }
    }(ws)

    data <- channel

仅当我不刷新页面时,一切都会按我的预期进行。如果我不刷新,我可以继续提交表单并看到不同的输出逐行出现。澄清一下,它实际上仅在页面已经启动时才有效,因此永远不会调用 parseandexecute 。该函数解析并执行 html/template,创建一个新的 websocket 客户端。

任何页面刷新或最初浏览 localhost:8000 都会导致服务器上的 websocket: close sent

我不知道如何解决这个问题。服务器是否需要优雅地处理断开连接并允许重新连接?或者客户需要做些什么?看起来服务器应该升级 /ws 上的任何连接,因此创建多少个新的 websocket 客户端并不重要,但显然我的理解是错误的。

我不会关闭服务器上的 websocket 连接,因为只要程序运行,它就应该保持连接。当用户停止该程序时,我认为它会自动关闭。

完整的 sockethandler 代码:

func (sh *SocketHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
    ws, err := upgrader.Upgrade(w, r, nil)
    if err != nil {
        w.Write([]byte(fmt.Sprintf("", err)))
        return
    }

    // discard received messages
    go func(c *websocket.Conn) {
        for {
            if _, _, err := c.NextReader(); err != nil {
                c.Close()
                break
            }
        }
    }(ws)

    cmd := <-sh.cmdCh

    log.Printf("Executing")
    stdout, err := cmd.StdoutPipe()
    if err != nil {
        w.Write([]byte(err.Error()))
        return
    }
    defer stdout.Close()

    stderr, err := cmd.StderrPipe()
    if err != nil {
        w.Write([]byte(err.Error()))
        return
    }
    defer stderr.Close()

    if err := cmd.Start(); err != nil {
        w.Write([]byte(err.Error()))
        return
    }

    s := bufio.NewScanner(io.MultiReader(stdout, stderr))
    for s.Scan() {
        err := ws.WriteMessage(1, s.Bytes())
        if err != nil {
            log.Printf("Error writing to client: %v", err)
            ws.Close()
        }

    }

    if err := cmd.Wait(); err != nil {
        w.Write([]byte(err.Error()))
        return
    }

    log.Println("Done")
}

解决方案


websocket 服务器应用程序必须通过关闭连接并释放与连接关联的资源来处理连接上的错误。

刷新包含页面时,浏览器会关闭 websocket 连接。浏览器关闭连接后,服务器最终会收到读取或写入错误。

连接关闭是服务器可能遇到的几种错误之一。服务器应用程序应以相同的方式处理连接上的所有错误:关闭连接并释放与该连接关联的资源。

典型的应用程序设计是让客户端在页面加载时进行连接,并在发生错误后重新连接(使用退避)。服务器假设客户端会随着时间的推移而连接和断开连接。

可以通过添加一个通过退避重新连接的 onerror 处理程序来改进 js 代码。根据应用程序的不同,您可能还希望显示指示连接状态的 ui。

go代码不会在所有场景下关闭连接。正在运行的命令是与连接关联的资源。应用程序不会在连接错误时终止该程序。以下是一些修复:

升级成功后添加defer ws.close()。从 sockethandler.servehttp 中删除对 ws.close() 的其他直接调用。这确保了在所有场景中都会调用 ws.close()

在退出读写泵时终止该命令。命令启动后将读取泵移至。返回时杀死。

go func(c *websocket.conn, cmd *exec.command) {
    defer c.close()
    defer cmd.process.kill()
    for {
        if _, _, err := c.nextreader(); err != nil {
            break
        }
    }
}(ws, cmd)

退出写泵时终止命令:

s := bufio.NewScanner(io.MultiReader(stdout, stderr))
for s.Scan() {
    err := ws.WriteMessage(1, s.Bytes())
    if err != nil {
        break
    }
}
cmd.Process.Kill()

我尚未运行或测试此代码。可能有些细节是错误的,但这概述了关闭连接和释放资源的一般方法。

看一下 Gorilla command example。该示例展示了如何将 stdin 和 stdout 泵送到 websocket。该示例还处理更高级的功能,例如使用 ping/pong 检查连接的运行状况。

终于介绍完啦!小伙伴们,这篇关于《重新建立websocket连接的优雅方法》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

版本声明
本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
使用 Lambda 函数从 AWS S3 中提取 CSV 文件使用 Lambda 函数从 AWS S3 中提取 CSV 文件
上一篇
使用 Lambda 函数从 AWS S3 中提取 CSV 文件
在 Go 模板中使用嵌套模板和变量
下一篇
在 Go 模板中使用嵌套模板和变量
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    543次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    514次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    499次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • SEO  AI Mermaid 流程图:自然语言生成,文本驱动可视化创作
    AI Mermaid流程图
    SEO AI Mermaid 流程图工具:基于 Mermaid 语法,AI 辅助,自然语言生成流程图,提升可视化创作效率,适用于开发者、产品经理、教育工作者。
    707次使用
  • 搜获客笔记生成器:小红书医美爆款内容AI创作神器
    搜获客【笔记生成器】
    搜获客笔记生成器,国内首个聚焦小红书医美垂类的AI文案工具。1500万爆款文案库,行业专属算法,助您高效创作合规、引流的医美笔记,提升运营效率,引爆小红书流量!
    718次使用
  • iTerms:一站式法律AI工作台,智能合同审查起草与法律问答专家
    iTerms
    iTerms是一款专业的一站式法律AI工作台,提供AI合同审查、AI合同起草及AI法律问答服务。通过智能问答、深度思考与联网检索,助您高效检索法律法规与司法判例,告别传统模板,实现合同一键起草与在线编辑,大幅提升法律事务处理效率。
    740次使用
  • TokenPony:AI大模型API聚合平台,一站式接入,高效稳定高性价比
    TokenPony
    TokenPony是讯盟科技旗下的AI大模型聚合API平台。通过统一接口接入DeepSeek、Kimi、Qwen等主流模型,支持1024K超长上下文,实现零配置、免部署、极速响应与高性价比的AI应用开发,助力专业用户轻松构建智能服务。
    805次使用
  • 迅捷AIPPT:AI智能PPT生成器,高效制作专业演示文稿
    迅捷AIPPT
    迅捷AIPPT是一款高效AI智能PPT生成软件,一键智能生成精美演示文稿。内置海量专业模板、多样风格,支持自定义大纲,助您轻松制作高质量PPT,大幅节省时间。
    695次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码