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

相关推荐
慕容静漪5 小时前
如何本地安装Python Flask并结合内网穿透实现远程开发
开发语言·后端·golang
ErizJ5 小时前
Golang|锁相关
开发语言·后端·golang
亿坊电商7 小时前
PHP + Go 如何协同打造高并发微服务?
微服务·golang·php
终身学习基地8 小时前
第二篇:go包管理
开发语言·后端·golang
游戏开发爱好者88 小时前
Flutter 学习之旅 之 flutter 使用 shared_preferences 实现简单的数据本地化保存封装
websocket·网络协议·tcp/ip·http·网络安全·https·udp
Go高并发架构_王工10 小时前
基于 GoFrame 框架的电子邮件发送实践:优势、特色与经验分享
网络·经验分享·golang
Chandler2410 小时前
Go:接口
开发语言·后端·golang
ErizJ10 小时前
Golang|Channel 相关用法理解
开发语言·后端·golang
automan0210 小时前
golang 在windows 系统的交叉编译
开发语言·后端·golang