当前位置:首页 > 文章列表 > 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基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    511次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    498次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 千音漫语:智能声音创作助手,AI配音、音视频翻译一站搞定!
    千音漫语
    千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
    100次使用
  • MiniWork:智能高效AI工具平台,一站式工作学习效率解决方案
    MiniWork
    MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
    92次使用
  • NoCode (nocode.cn):零代码构建应用、网站、管理系统,降低开发门槛
    NoCode
    NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
    110次使用
  • 达医智影:阿里巴巴达摩院医疗AI影像早筛平台,CT一扫多筛癌症急慢病
    达医智影
    达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
    102次使用
  • 智慧芽Eureka:更懂技术创新的AI Agent平台,助力研发效率飞跃
    智慧芽Eureka
    智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
    102次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码