0

0

Go语言WebSocket服务器:解决403 Origin校验错误与灵活配置

聖光之護

聖光之護

发布时间:2025-11-06 23:19:00

|

770人浏览过

|

来源于php中文网

原创

Go语言WebSocket服务器:解决403 Origin校验错误与灵活配置

本教程旨在解决go语言websocket服务器在处理客户端连接时常见的403 forbidden错误,该问题通常源于go.net/websocket包默认的origin头部校验机制。文章将深入解释错误原因,并提供使用websocket.server进行灵活配置的解决方案,确保服务器能够正确响应各类websocket客户端请求,构建稳定可靠的websocket服务。

Go语言中构建基础WebSocket服务

在Go语言中搭建WebSocket服务器,通常会利用code.google.com/p/go.net/websocket包(请注意,此包为早期版本,生产环境建议考虑github.com/gorilla/websocket或net/http的升级功能)。一个简单的WebSocket Echo服务器可以接收客户端发送的消息并回传。以下是一个基础的实现示例:

package main

import (
    "fmt"
    "code.google.com/p/go.net/websocket"
    "net/http"
)

// webHandler 处理WebSocket连接,接收并打印消息,然后回传
func webHandler(ws *websocket.Conn) {
    var s string
    // 从WebSocket连接读取数据
    if _, err := fmt.Fscan(ws, &s); err != nil {
        fmt.Printf("Error reading from websocket: %v\n", err)
        return
    }
    fmt.Println("Received: ", s)
    // 将消息回传给客户端
    if _, err := fmt.Fprint(ws, "Echo: "+s); err != nil {
        fmt.Printf("Error writing to websocket: %v\n", err)
    }
}

func main() {
    fmt.Println("Starting WebSocket server on :8080")

    // 使用websocket.Handler注册WebSocket处理函数
    // 默认情况下,websocket.Handler会执行Origin头部校验
    http.Handle("/echo", websocket.Handler(webHandler))

    // 启动HTTP服务器监听
    err := http.ListenAndServe(":8080", nil)
    if err != nil {
        panic("ListenAndServe: " + err.Error())
    }
}

配套的JavaScript客户端连接代码如下:

// 创建WebSocket连接
ws = new WebSocket("ws://localhost:8080/echo");

// 连接成功时触发
ws.onopen = function(event) {
    console.log("WebSocket connection opened.");
    // 连接建立后发送一条消息
    ws.send("Hello from client!");
};

// 收到服务器消息时触发
ws.onmessage = function(e) {
    console.log("WebSocket message received: " + e.data);
};

// 连接发生错误时触发
ws.onerror = function(error) {
    console.error("WebSocket error: ", error);
};

// 连接关闭时触发
ws.onclose = function(event) {
    console.log("WebSocket connection closed:", event.code, event.reason);
};

然而,当尝试使用上述配置的Go服务器与JavaScript客户端连接时,可能会在浏览器控制台看到类似WebSocket connection to 'ws://localhost:8080/echo' failed: Unexpected response code: 403的错误信息。

深入理解403 Origin校验错误

这个403 Forbidden错误并非表示认证失败,而是与code.google.com/p/go.net/websocket包内部的Origin头部校验机制密切相关。websocket.Handler在默认情况下,会检查HTTP请求中的Origin头部是否是一个有效的URL。

立即学习go语言免费学习笔记(深入)”;

Origin头部是Web浏览器在发起跨域请求(包括WebSocket连接)时自动添加的一个安全特性,它指示了请求发起的源(协议、域名和端口)。websocket.Handler默认会验证这个Origin头部,如果出现以下情况,它可能会拒绝连接:

  • Origin头部缺失(例如,非浏览器客户端或某些测试工具可能不会发送此头部)。
  • Origin头部不是一个有效的URL格式。
  • Origin头部虽然有效,但与服务器的预期或默认规则不符。

在上述任何一种情况下,websocket.Handler都会拒绝连接,并返回403状态码。这对于Web浏览器客户端而言是一种重要的安全保障,可以防止恶意网站发起WebSocket连接。然而,对于需要连接非浏览器客户端(如桌面应用、移动应用或服务器间通信)或在特定开发场景下,这种默认的严格校验可能会造成困扰。

viable
viable

基于GPT-4的AI非结构化数据分析平台

下载

解决方案:利用 websocket.Server 灵活控制 Origin 校验

为了解决Origin校验导致的403错误,我们需要更精细地控制WebSocket连接的握手过程。code.google.com/p/go.net/websocket包提供了websocket.Server结构体,它允许我们自定义握手逻辑,特别是对Origin头部的处理。

通过websocket.Server,我们可以选择禁用默认的Origin校验,或者实现自定义的校验逻辑。对于需要接受非浏览器客户端或Origin头部可能不符合默认校验规则的场景,使用websocket.Server是更灵活且强大的选择。

以下是修改后的Go服务器代码,它使用websocket.Server来处理WebSocket连接,从而绕过默认的Origin校验:

package main

import (
    "fmt"
    "code.google.com/p/go.net/websocket"
    "net/http"
)

// webHandler 保持不变,处理WebSocket连接的业务逻辑
func webHandler(ws *websocket.Conn) {
    var s string
    if _, err := fmt.Fscan(ws, &s); err != nil {
        fmt.Printf("Error reading from websocket: %v\n", err)
        return
    }
    fmt.Println("Received: ", s)
    // 示例:回传消息
    if _, err := fmt.Fprint(ws, "Echo: "+s); err != nil {
        fmt.Printf("Error writing to websocket: %v\n", err)
    }
}

func main() {
    fmt.Println("Starting WebSocket server on :8080 with custom Origin handling")

    // 使用http.HandleFunc注册一个普通的HTTP处理函数
    // 在这个处理函数内部,我们创建并使用websocket.Server实例
    http.HandleFunc("/echo",
        func (w http.ResponseWriter, req *http.Request) {
            // 创建websocket.Server实例
            // 将webHandler作为其Handler。
            // 关键点:当websocket.Server的Handshake字段为nil时,它不会执行Origin校验,
            // 而是接受所有连接。这与websocket.Handler的默认行为不同。
            s := websocket.Server{Handler: websocket.Handler(webHandler)}
            // 调用ServeHTTP来处理当前的HTTP请求,将其升级为WebSocket连接
            s.ServeHTTP(w, req)
        })

    err := http.ListenAndServe(":8080", nil)
    if err != nil {
        panic("ListenAndServe: " + err.Error())
    }
}

通过这种方式,websocket.Server在处理请求时,如果其Handshake字段为nil,它将不会执行默认的Origin校验,从而允许各种客户端(包括那些不发送Origin头部或发送不符合默认校验规则Origin头部的客户端)成功建立WebSocket连接。

关键注意事项与总结

  1. websocket.Handler 与 websocket.Server 的选择:
    • websocket.Handler 是一个便捷的封装,它返回一个http.Handler,适用于大多数浏览器客户端,并默认执行严格的Origin校验以增强安全性。
    • websocket.Server 提供了更底层的控制,允许开发者自定义握手过程,包括Origin校验。当需要接受非浏览器客户端、实现服务器间通信或有特殊Origin校验需求时,应优先考虑websocket.Server。
  2. 安全性考量: 禁用Origin校验会降低安全性。Origin头部是防止跨站请求伪造(CSRF)攻击的一种重要机制。在生产环境中,如果服务面向Web浏览器客户端,强烈建议启用或实现自定义的Origin校验,以确保只有来自受信任域的客户端才能连接。对于服务器到服务器的WebSocket通信,或者您能完全控制客户端来源的场景,禁用Origin校验可能是可接受的。
  3. 包版本说明: 本教程中使用的code.google.com/p/go.net/websocket是Go语言早期提供的WebSocket库。在现代Go应用开发中,更推荐使用社区维护的github.com/gorilla/websocket包,它功能更丰富,性能更好,且维护更活跃。或者,Go标准库net/http也提供了http.Hijacker接口,可以用于手动升级HTTP连接到WebSocket。尽管如此,本文提供的解决方案对于理解go.net/websocket的工作原理及其Origin校验问题仍然具有指导意义。

通过理解go.net/websocket包中Origin校验的机制,并恰当地利用websocket.Server,我们可以灵活地构建Go语言WebSocket服务器,以适应不同类型的客户端连接需求,避免常见的403错误,并根据实际场景平衡安全性和便利性。

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

536

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

372

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

706

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

470

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

388

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

989

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

652

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

535

2023.09.20

苹果官网入口直接访问
苹果官网入口直接访问

苹果官网直接访问入口是https://www.apple.com/cn/,该页面具备0.8秒首屏渲染、HTTP/3与Brotli加速、WebP+AVIF双格式图片、免登录浏览全参数等特性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

10

2025.12.24

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 2.9万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 1.7万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 2.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号