以太坊的RPC机制

1 go语言的RPC机制

RPC(Remote Procedure Call,远程过程调用)是一种通过网络从远程计算机程序上请求服

务,而不需要了解底层网络细节的应用程序通信协议。RPC协议构建于TCP或UDP,或者是 HTTP

之上,允许开发者直接调用另一台计算机上的程序,而开发者无需额外地为这个调用过程编写网

络通信相关代码,使得开发包括网络分布式程序在内的应用程序更加容易。

go语言有net/rpc包,net/rpc包允许 RPC 客户端程序通过网络或是其他 I/O 连接调用一个远端对象的公开方法

(必须是大写字母开头、可外部调用的)。在 RPC 服务端,可将一个对象注册为可访问的服务,

之后该对象的公开方法就能够以远程的方式提供访问。一个 RPC 服务端可以注册多个不同类型

的对象,但不允许注册同一类型的多个对象。

    一个对象中只有满足如下这些条件的方法,才能被 RPC 服务端设置为可供远程访问:

必须是在对象外部可公开调用的方法(首字母大写);

必须有两个参数,且参数的类型都必须是包外部可以访问的类型或者是Go内建支持的类型;

第二个参数必须是一个指针;

方法必须返回一个error类型的值。

以上4个条件,可以简单地用如下一行代码表示:

func (t *T) MethodName(argType T1, replyType *T2) error

接下来,我们来看一组 RPC 服务端和客户端交互的示例程序。

服务端:

package main;

import (

    "net/rpc"

    "net/http"

    "log"

)

//go对RPC的支持,支持三个级别:TCP、HTTP、JSONRPC

//go的RPC只支持GO开发的服务器与客户端之间的交互,因为采用了gob编码

//注意字段必须是导出

type Params struct {

    Width, Height int;

}

type Rect struct{}

//函数必须是导出的

//必须有两个导出类型参数

//第一个参数是接收参数

//第二个参数是返回给客户端参数,必须是指针类型

//函数还要有一个返回值error

func (r *Rect) Area(p Params, ret *int) error {

    *ret = p.Width * p.Height;

    return nil;

}

func (r *Rect) Perimeter(p Params, ret *int) error {

    *ret = (p.Width + p.Height) * 2;

    return nil;

}

func main() {

    rect := new(Rect);

    //注册一个rect服务

    rpc.Register(rect);

    //把服务处理绑定到http协议上

    rpc.HandleHTTP();

    err := http.ListenAndServe(":8080", nil);

    if err != nil {

        log.Fatal(err);

    }

}

客户端:

package main;

import (

    "net/rpc"

    "log"

    "fmt"

)

type Params struct {

    Width, Height int;

}

func main() {

    //连接远程rpc服务

    rpc, err := rpc.DialHTTP("tcp", "127.0.0.1:8080");

    if err != nil {

        log.Fatal(err);

    }

    ret := 0;

    //调用远程方法

    //注意第三个参数是指针类型

    err2 := rpc.Call("Rect.Area", Params{50, 100}, &ret);

    if err2 != nil {

        log.Fatal(err2);

    }

    fmt.Println(ret);

    err3 := rpc.Call("Rect.Perimeter", Params{50, 100}, &ret);

    if err3 != nil {

        log.Fatal(err3);

    }

    fmt.Println(ret);

}

2 以太坊RPC机制

以太坊启动RPC服务

以太坊客户端可以用下面方式来启动RPC监听:

geth --rpc --rpcaddr 0.0.0.0 --rpcapi db,eth,net,web3,personal --rpcport 8550

这句明命令启动了Http-RPC服务,rpc监听地址是任意ip地址,rcp使用的api接口包括db,eth,net,web,personal等,rpc端口是8550。

以太坊源码中RPC服务启动流程

在以太坊geth的main函数里,有函数

func geth(ctx *cli.Context) error {

node := makeFullNode(ctx)

startNode(ctx, node)

node.Wait()

return nil

}

这是geth的主执行函数,通过startNode()启动geth节点,startNode继续调用node/node.go中的Start()函数中,Start()函数中调用了startRPC()函数:

// startRPC is a helper method to start all the various RPC endpoint during node

// startup. It's not meant to be called at any time afterwards as it makes certain

// assumptions about the state of the node.

func (n *Node) startRPC(services map[reflect.Type]Service) error {

// Gather all the possible APIs to surface

apis := n.apis()

for _, service := range services {

apis = append(apis, service.APIs()...)

}

// Start the various API endpoints, terminating all in case of errors

if err := n.startInProc(apis); err != nil {

return err

}

if err := n.startIPC(apis); err != nil {

n.stopInProc()

return err

}

if err := n.startHTTP(n.httpEndpoint, apis, n.config.HTTPModules, n.config.HTTPCors, n.config.HTTPVirtualHosts); err != nil {

n.stopIPC()

n.stopInProc()

return err

}

if err := n.startWS(n.wsEndpoint, apis, n.config.WSModules, n.config.WSOrigins, n.config.WSExposeAll); err != nil {

n.stopHTTP()

n.stopIPC()

n.stopInProc()

return err

}

// All API endpoints started successfully

n.rpcAPIs = apis

return nil

}

startRPC()收集了node中和services中所有的rpc.API类型的RPC接口,并启动了各种RPC服务形式,包括IPC、HTTP、WS、PROC等各种形式。下面分析启动Http方式的RPC函数startHTTP():

func (n *Node) startHTTP(endpoint string, apis []rpc.API, modules []string, cors []string, vhosts []string) error {

// Short circuit if the HTTP endpoint isn't being exposed

if endpoint == "" {

return nil

}

// Generate the whitelist based on the allowed modules

whitelist := make(map[string]bool)

for _, module := range modules {

whitelist[module] = true

}

// Register all the APIs exposed by the services

handler := rpc.NewServer()

for _, api := range apis {

if whitelist[api.Namespace] || (len(whitelist) == 0 && api.Public) {

if err := handler.RegisterName(api.Namespace, api.Service); err != nil {

return err

}

n.log.Debug("HTTP registered", "service", api.Service, "namespace", api.Namespace)

}

}

// All APIs registered, start the HTTP listener

var (

listener net.Listener

err      error

)

if listener, err = net.Listen("tcp", endpoint); err != nil {

return err

}

go rpc.NewHTTPServer(cors, vhosts, handler).Serve(listener)

n.log.Info("HTTP endpoint opened", "url", fmt.Sprintf("http://%s", endpoint), "cors", strings.Join(cors, ","), "vhosts", strings.Join(vhosts, ","))

// All listeners booted successfully

n.httpEndpoint = endpoint

n.httpListener = listener

n.httpHandler = handler

return nil

}

可以看到以太坊中通过Http方式启动RPC服务的流程跟go中的rpc包启动方式基本一致。先是通过rpc.newServer()创建了Server,然后再通过registerName()注册API服务,然后启动Http监听。不过以太坊中的RPC接口API并不是按照标准RPC接口写的,它的基本形式是:

func(s *CalcService)Add(a, bint)(int, error)

符合以下标准的方法可用于远程访问:

对象必须导出

方法必须导出

方法返回0,1(响应或错误)或2(响应和错误)值

方法参数必须导出或是内置类型

方法返回值必须导出或是内置类型

客户端调用RPC服务

rpc/client.go中拨号函数:

/ The client reconnects automatically if the connection is lost.

func Dial(rawurl string) (*Client, error) {

return DialContext(context.Background(), rawurl)

}

// DialContext creates a new RPC client, just like Dial.

//

// The context is used to cancel or time out the initial connection establishment. It does

// not affect subsequent interactions with the client.

func DialContext(ctx context.Context, rawurl string) (*Client, error) {

  u, err := url.Parse(rawurl)

  if err != nil {

      return nil, err

  }

  switch u.Scheme {

  case "http", "https":

      return DialHTTP(rawurl)

  case "ws", "wss":

      return DialWebsocket(ctx, rawurl, "")

  case "":

      return DialIPC(ctx, rawurl)

  default:

      return nil, fmt.Errorf("no known transport for URL scheme %q", u.Scheme)

  }

}

调用RPC服务的函数:

// Call performs a JSON-RPC call with the given arguments and unmarshals into

// result if no error occurred.

//

// The result must be a pointer so that package json can unmarshal into it. You

// can also pass nil, in which case the result is ignored.

func (c *Client) Call(result interface{}, method string, args ...interface{}) error {

ctx := context.Background()

return c.CallContext(ctx, result, method, args...)

}

// CallContext performs a JSON-RPC call with the given arguments. If the context is

// canceled before the call has successfully returned, CallContext returns immediately.

//

// The result must be a pointer so that package json can unmarshal into it. You

// can also pass nil, in which case the result is ignored.

func (c *Client) CallContext(ctx context.Context, result interface{}, method string, args ...interface{}) error {

  msg, err := c.newMessage(method, args...)

  if err != nil {

      return err

  }

  op := &requestOp{ids: []json.RawMessage{msg.ID}, resp: make(chan *jsonrpcMessage, 1)}

  if c.isHTTP {

      err = c.sendHTTP(ctx, op, msg)

  } else {

      err = c.send(ctx, op, msg)

  }

  if err != nil {

      return err

  }

  // dispatch has accepted the request and will close the channel it when it quits.

  switch resp, err := op.wait(ctx); {

  case err != nil:

      return err

  case resp.Error != nil:

      return resp.Error

  case len(resp.Result) == 0:

      return ErrNoResult

  default:

      return json.Unmarshal(resp.Result, &result)

  }

}

3 web3.js与控制台调用RPC接口

internal/jsre/deps下有web3.js文件,以及internal/web3ext下的web3ext.go文件,封装了可以在console控制台下访问RPC接口的方法和接口。console下面有admin.importChain方法,搜索importChain,可以看到搜索结果,

importChain对应的一个出现在web3ext.go中,

new web3._extend.Method({

    name: 'importChain',

    call: 'admin_importChain',

    params: 1

}),

函数定义在eth/api.go中:

// ImportChain imports a blockchain from a local file.

func (api *PrivateAdminAPI) ImportChain(file string) (bool, error) {

// Make sure the can access the file to import

in, err := os.Open(file)

if err != nil {

return false, err

}

defer in.Close()

    ......

}

4 自定义RPC接口

依照ImportChain接口的方法,在eth/api.go中定义函数:

func (api *PrivateAdminAPI) TestMul(a,b *int) (int, error) {

return (*a)*(*b),nil;

}

然后在web3ext.go中加入声明:

new web3._extend.Method({

name: 'startRPC',

call: 'admin_startRPC',

params: 4,

inputFormatter: [null, null, null, null]

}),

new web3._extend.Method({

name: 'stopRPC',

call: 'admin_stopRPC'

}),

new web3._extend.Method({

name: 'startWS',

call: 'admin_startWS',

params: 4,

inputFormatter: [null, null, null, null]

}),

new web3._extend.Method({

name: 'stopWS',

call: 'admin_stopWS'

}),

new web3._extend.Method({

name: 'testMul',

call: 'admin_testMul',

params: 2

}),

],

重新编译geth,运行,在控制台输入admin:

可以看到出现了testMul接口,调用testMul接口试一下:

参考:

https://blog.csdn.net/liuzhijun301/article/details/80759920

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351

推荐阅读更多精彩内容

  • Lua 5.1 参考手册 by Roberto Ierusalimschy, Luiz Henrique de F...
    苏黎九歌阅读 13,776评论 0 38
  • 活着真好, 每当我从睡梦中醒来, 睁开朦胧的双眼, 温暖的阳光抚摸着我的脸颊。 我开心地笑了, 因为我还活着。 活...
    亮靓_27d5阅读 671评论 73 79
  • 作为一个打着“走遍全天下”幌子的“美食达人”,想了想:生活不仅仅限于眼前的苟且,还有未知的远方。总有离开父母,离开...
    英Joy夏天阅读 351评论 0 0
  • 童年与恐惧 在那个盛藏着黑白记忆的小房子里,外婆贫穷而又坚强的走完了她这本该继续的一生。漫墙...
    襄南才子阅读 247评论 0 0
  • 因为快过中秋节了,今天早晨和父母到姥姥家去看看,也许是因为小时候是姥姥带到上幼儿园的年纪,才回到城里和父母一起生活...
    弈臻阅读 141评论 0 0