0

0

如何在 JSON-RPC 方法中获取 WebSocket 连接信息

聖光之護

聖光之護

发布时间:2026-02-07 17:29:39

|

994人浏览过

|

来源于php中文网

原创

如何在 JSON-RPC 方法中获取 WebSocket 连接信息

go 的 `jsonrpc` + `websocket` 组合中,标准 rpc 方法无法直接访问底层 `*websocket.conn`。本文介绍一种基于 `context.context` 和自定义 `servercodec` 的安全、可扩展方案,使 rpc 处理函数能获取连接元数据(如远程地址、header 等)。

Go 标准库的 net/rpc/jsonrpc 是面向无状态通信设计的,其 ServeConn 接口将 *websocket.Conn 封装后完全隔离,导致 RPC 方法(如 Multiply)无法感知调用来源。强行打破这一抽象虽可行,但会牺牲可维护性与兼容性。推荐采用 上下文注入(Context Injection) 方式,在不修改 RPC 协议语义的前提下,安全传递连接信息。

✅ 核心思路:通过自定义 ServerCodec 注入 context.Context

我们需实现一个符合 rpc.ServerCodec 接口的编码器,在反序列化请求参数时,自动将携带连接信息的 context.Context 注入到参数结构体的 Context 字段中。

1. 定义支持上下文的参数结构体

import (
    "context"
    "net/http"
    "net/rpc"
    "net/rpc/jsonrpc"
    "reflect"
    "sync"
)

type Args struct {
    A       int
    B       int
    Context context.Context // 新增字段,用于接收注入的上下文
}

2. 实现自定义 ServerCodec

以下是一个轻量级 WebSocketContextCodec,它包装原始 jsonrpc.ServerCodec,并在 ReadRequestBody 阶段注入 context.WithValue:

type WebSocketContextCodec struct {
    codec rpc.ServerCodec
    ctx   context.Context
}

func (c *WebSocketContextCodec) ReadRequestHeader(r *rpc.Request) error {
    return c.codec.ReadRequestHeader(r)
}

func (c *WebSocketContextCodec) ReadRequestBody(x interface{}) error {
    err := c.codec.ReadRequestBody(x)
    if err != nil {
        return err
    }

    // 使用反射向 x 中的 Context 字段注入上下文
    v := reflect.ValueOf(x)
    if v.Kind() == reflect.Ptr {
        v = v.Elem()
        if v.Kind() == reflect.Struct {
            ctxField := v.FieldByName("Context")
            if ctxField.IsValid() && ctxField.CanSet() && ctxField.Type() == reflect.TypeOf((*context.Context)(nil)).Elem().Elem() {
                ctxField.Set(reflect.ValueOf(c.ctx))
            }
        }
    }
    return nil
}

func (c *WebSocketContextCodec) WriteResponse(r *rpc.Response, body interface{}) error {
    return c.codec.WriteResponse(r, body)
}

func (c *WebSocketContextCodec) Close() error {
    return c.codec.Close()
}

3. 在 WebSocket 处理中创建带上下文的 Codec

修改 serve 函数,为每个连接创建专属 context.Context(例如包含远程地址、握手 Header 等):

Palette
Palette

在线生成整套UI调色板

下载
import "code.google.com/p/go.net/websocket"

func serve(ws *websocket.Conn) {
    // 构建连接上下文:可携带 ws.RemoteAddr(), ws.Config().Origin, 或自定义元数据
    connCtx := context.WithValue(
        context.Background(),
        "websocket.conn",
        ws,
    )
    connCtx = context.WithValue(connCtx, "remote_addr", ws.RemoteAddr().String())

    // 创建自定义 codec
    codec := &WebSocketContextCodec{
        codec: jsonrpc.NewServerCodec(ws),
        ctx:   connCtx,
    }

    // 使用自定义 codec 启动 RPC 服务
    rpc.ServeCodec(codec)
}

4. 在 RPC 方法中使用连接信息

现在 Multiply 可直接访问连接上下文:

func (t *Arith) Multiply(args *Args, reply *int) error {
    *reply = args.A * args.B

    // ✅ 安全获取 WebSocket 连接对象和元数据
    if conn, ok := args.Context.Value("websocket.conn").(*websocket.Conn); ok {
        // 例如:记录日志或做权限校验
        println("RPC called from:", conn.RemoteAddr())
    }

    if addr, ok := args.Context.Value("remote_addr").(string); ok {
        println("Client IP:", addr)
    }

    return nil
}

⚠️ 注意事项与最佳实践

  • 线程安全:context.Context 本身是并发安全的,但注入的值(如 *websocket.Conn)需确保调用方不执行阻塞/写操作;建议仅读取元数据(如 RemoteAddr, Config().Origin),避免在 RPC 中调用 ws.Write()。
  • 性能影响:反射注入仅在每次请求时执行一次,开销极小;若追求极致性能,可预编译字段索引(如用 sync.Once 缓存 FieldByName 结果)。
  • 兼容性:该方案完全兼容标准 jsonrpc 协议,客户端无需任何改动。
  • 替代方案对比:不推荐使用全局 map + goroutine ID 模拟 thread-local —— Go 不提供稳定 goroutine ID,且易引发内存泄漏;context 是 Go 官方推荐的跨 API 边界传递请求范围数据的标准方式。

通过此方法,你既保持了 RPC 层的清晰抽象,又获得了对底层连接的精细控制能力,是构建高可用、可观测 WebSocket-RPC 服务的关键一环。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

184

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

232

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

344

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

210

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

398

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

282

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

196

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

661

2025.06.17

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

2

2026.02.06

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.9万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.3万人学习

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

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