【golang/jsonrpc】go-ethereum中json rpc初步使用(websocket版本)

说在前面

  • 操作系统:win11 wsl2
  • go-ethereum版本:1.15.8

关于json-rpc

server

  • 定义方法

    go 复制代码
    type CalculatorService struct{}
    
    func (s *CalculatorService) Add(a, b int) int {
    	return a + b
    }
    
    func (s *CalculatorService) Div(a, b int) (int, error) {
    	if b == 0 {
    		return 0, errors.New("divide by zero")
    	}
    	return a / b, nil
    }

    这里我们简单定义两个方法

  • 启动服务

    go 复制代码
    func main() {
    	calculator := new(CalculatorService)
    
    	// 创建 RPC 服务器
    	rpcServer := rpc.NewServer()
    
    	// 注册服务
    	if err := rpcServer.RegisterName("calculator", calculator); err != nil {
    		panic(err)
    	}
    
    	// 启动 HTTP 服务器
    	println("WebSocket server listening on :8080")
    
    	httpHandler := rpcServer.WebsocketHandler([]string{"*"})
    	http.Handle("/ws", httpHandler)
    
    	http.ListenAndServe(":8080", nil)
    }

client

  • 调用
    在调用的时候,方法名需要指定为receiver_method,例如我们要调用CalculatorService.Add,那么调用名为calculator_add,注意,这里的calculator为server端注册时的RegisterName("calculator", calculator),而不是CalculatorService

    go 复制代码
    func main() {
    	// 连接 WebSocket 服务端
    	client, err := rpc.DialWebsocket(context.Background(), "ws://192.168.1.6:8080/ws", "")
    	if err != nil {
    		panic(err)
    	}
    	defer client.Close()
    
    	// 调用远程方法
    	var result int
    	err = client.Call(&result, "calculator_add", 3, 5)
    	if err != nil {
    		panic(err)
    	}
    
    	fmt.Println("Result:", result) // 输出: Result: 8
    }

    假设将server端的RegisterName("calculator", calculator)改为RegisterName("calculate", calculator)

    shell 复制代码
    panic: the method calculator_add does not exist/is not available

注意事项

返回值

  • server端的方法返回值只能有三种数量0,1,2

    • 0,即无返回值
    • 1,即自定义返回值或者一个error
    • 2,即自定义返回值+一个error
  • 这样,在client端调用的时候,err = client.Call(&result, "calculator_add", 3, 5)result对应自定义返回值,err对应返回的error

  • 例如,我们调用

    go 复制代码
    err = client.Call(&result, "calculate_div", 3, 0)
    if err != nil {
    	panic(err)
    }

    得到的err即为server端Div方法返回的error

    shell 复制代码
    panic: divide by zero

server端调用client端

  • 支持反向调用

  • 在我们定义server端方法的时候,可以增加一个context参数

    go 复制代码
    type CalculatorService struct{}
    
    func (s *CalculatorService) Add(ctx context.Context, a, b int) int {
    	return a + b
    }
    
    func (s *CalculatorService) Div(ctx context.Context, a, b int) (int, error) {
    	if b == 0 {
    		return 0, errors.New("divide by zero")
    	}
    	return a / b, nil
    }
  • 这样我们就可以获取到client,进而调用其方法,例如,我们在client端添加一个方法

    go 复制代码
    type CalcService struct{}
    
    func (s *CalcService) GetParam(ctx context.Context) int {
    	return 88
    }

    然后在client端注册

    go 复制代码
    func main() {
    	// 连接 WebSocket 服务端
    	client, err := rpc.DialWebsocket(context.Background(), "ws://192.168.1.6:8080/ws", "")
    	if err != nil {
    		panic(err)
    	}
    	defer client.Close()
    
    	calculator := new(CalcService)
    	client.RegisterName("calc", calculator)
    
    	// 调用远程方法
    	var result int
    	err = client.Call(&result, "calculate_add", 3, 0)
    	if err != nil {
    		panic(err)
    	}
    
    	fmt.Println("Result:", result) // 输出: Result: 8
    }

    在server端的Add方法中,我们将b的值改为从client端获取

    shell 复制代码
    func (s *CalcService) Add(ctx context.Context, a, b int) int {
    	client, ok := rpc.ClientFromContext(ctx)
    	if ok {
    		client.Call(&b, "calc_getParam")
    	}
    	return a + b
    }

    这样运行后的结果为:

    shell 复制代码
    Result: 91
  • 通过这种方式,我们可以在server端添加一个Register的方法,在client连接后,client主动调用Register方法,这样我们就可以管理所有client了

相关推荐
火星数据-Tina1 天前
低成本搭建体育数据中台:一套 API 如何同时支撑比分网与 App?
java·前端·websocket
wavemap1 天前
先到先得:免费订阅一年ChatGPT Go会员
开发语言·chatgpt·golang
浮尘笔记1 天前
Go并发编程核心:Mutex和RWMutex的用法
开发语言·后端·golang
百***06011 天前
【Golang】——Gin 框架中的表单处理与数据绑定
microsoft·golang·gin
百***93501 天前
【Golang】——Gin 框架中间件详解:从基础到实战
中间件·golang·gin
Tony Bai1 天前
Go 2025 密码学年度报告:后量子时代的防御与 FIPS 的“纯 Go”革命
开发语言·后端·golang·密码学
Like_wen1 天前
idea/goland 无法创建目标目录
java·golang
利刃大大1 天前
【c++中间件】WebSocket介绍 && WebSocketpp库的使用
c++·websocket·中间件
资深web全栈开发2 天前
Golang Cobra 教程:构建强大的CLI应用
开发语言·后端·golang
JCGKS2 天前
Go| excelize的流式迭代器
后端·golang·excel·excelize·流式读取·文件解析