【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了

相关推荐
Yeats_Liao2 小时前
Go Web 编程快速入门 10 - 数据库集成与ORM:连接池、查询优化与事务管理
前端·数据库·后端·golang
Tony Bai7 小时前
【Go模块构建与依赖管理】01 前世今生:从 GOPATH 的“混乱”到 Go Modules 的“秩序”
开发语言·后端·golang
gopyer7 小时前
Go语言2D游戏开发入门004:零基础打造射击游戏《太空大战》3
golang·go·游戏开发
Dobby_058 小时前
【Go】C++转Go:数据结构练习(一)排序算法
数据结构·golang
百锦再9 小时前
Go与Python在AI大模型开发中的深度对比分析
java·开发语言·人工智能·python·学习·golang·maven
钟离墨笺11 小时前
Go语言-->Goroutine 详细解释
开发语言·后端·golang
Yeats_Liao11 小时前
Go Web 编程快速入门 11 - WebSocket实时通信:实时消息推送和双向通信
前端·后端·websocket·golang
yzq-384112 小时前
Websocket两台服务器之间的通信
spring boot·websocket·网络协议
草明14 小时前
当 Go 的 channel 被 close 后读写操作会怎么样?
开发语言·后端·golang