github.com/yinchengtsinghua/golang-Eos-dpos-Ethereum@v0.0.0-20190121132951-92cc4225ed8e/accounts/abi/abi.go (about) 1 2 //此源码被清华学神尹成大魔王专业翻译分析并修改 3 //尹成QQ77025077 4 //尹成微信18510341407 5 //尹成所在QQ群721929980 6 //尹成邮箱 yinc13@mails.tsinghua.edu.cn 7 //尹成毕业于清华大学,微软区块链领域全球最有价值专家 8 //https://mvp.microsoft.com/zh-cn/PublicProfile/4033620 9 // 10 //此文件是Go以太坊库的一部分。 11 // 12 //Go-Ethereum库是免费软件:您可以重新分发它和/或修改 13 //根据GNU发布的较低通用公共许可证的条款 14 //自由软件基金会,或者许可证的第3版,或者 15 //(由您选择)任何更高版本。 16 // 17 //Go以太坊图书馆的发行目的是希望它会有用, 18 //但没有任何保证;甚至没有 19 //适销性或特定用途的适用性。见 20 //GNU较低的通用公共许可证,了解更多详细信息。 21 // 22 //你应该收到一份GNU较低级别的公共许可证副本 23 //以及Go以太坊图书馆。如果没有,请参见<http://www.gnu.org/licenses/>。 24 25 package abi 26 27 import ( 28 "bytes" 29 "encoding/json" 30 "fmt" 31 "io" 32 ) 33 34 //ABI保存有关合同上下文的信息,并提供 35 //可调用方法。它将允许您键入check函数调用和 36 //相应地打包数据。 37 type ABI struct { 38 Constructor Method 39 Methods map[string]Method 40 Events map[string]Event 41 } 42 43 //JSON返回已分析的ABI接口,如果失败则返回错误。 44 func JSON(reader io.Reader) (ABI, error) { 45 dec := json.NewDecoder(reader) 46 47 var abi ABI 48 if err := dec.Decode(&abi); err != nil { 49 return ABI{}, err 50 } 51 52 return abi, nil 53 } 54 55 //打包给定的方法名以符合ABI。方法调用的数据 56 //将由方法“id”、“args0”、“arg1…”组成。阿尔金方法ID由 57 //其中4个字节和参数都是32个字节。 58 //方法ID是从 59 //方法字符串签名。(签名=baz(uint32,string32)) 60 func (abi ABI) Pack(name string, args ...interface{}) ([]byte, error) { 61 //获取所请求方法的ABI 62 if name == "" { 63 //构造函数 64 arguments, err := abi.Constructor.Inputs.Pack(args...) 65 if err != nil { 66 return nil, err 67 } 68 return arguments, nil 69 70 } 71 method, exist := abi.Methods[name] 72 if !exist { 73 return nil, fmt.Errorf("method '%s' not found", name) 74 } 75 76 arguments, err := method.Inputs.Pack(args...) 77 if err != nil { 78 return nil, err 79 } 80 //如果不是构造函数,也打包方法ID并返回 81 return append(method.Id(), arguments...), nil 82 } 83 84 //根据ABI规范以V为单位解包输出 85 func (abi ABI) Unpack(v interface{}, name string, output []byte) (err error) { 86 if len(output) == 0 { 87 return fmt.Errorf("abi: unmarshalling empty output") 88 } 89 //既然不能命名与契约和事件的冲突, 90 //我们需要决定是调用方法还是调用事件 91 if method, ok := abi.Methods[name]; ok { 92 if len(output)%32 != 0 { 93 return fmt.Errorf("abi: improperly formatted output") 94 } 95 return method.Outputs.Unpack(v, output) 96 } else if event, ok := abi.Events[name]; ok { 97 return event.Inputs.Unpack(v, output) 98 } 99 return fmt.Errorf("abi: could not locate named method or event") 100 } 101 102 //unmashaljson实现json.unmasheler接口 103 func (abi *ABI) UnmarshalJSON(data []byte) error { 104 var fields []struct { 105 Type string 106 Name string 107 Constant bool 108 Anonymous bool 109 Inputs []Argument 110 Outputs []Argument 111 } 112 113 if err := json.Unmarshal(data, &fields); err != nil { 114 return err 115 } 116 117 abi.Methods = make(map[string]Method) 118 abi.Events = make(map[string]Event) 119 for _, field := range fields { 120 switch field.Type { 121 case "constructor": 122 abi.Constructor = Method{ 123 Inputs: field.Inputs, 124 } 125 //空默认为根据ABI规范的功能 126 case "function", "": 127 abi.Methods[field.Name] = Method{ 128 Name: field.Name, 129 Const: field.Constant, 130 Inputs: field.Inputs, 131 Outputs: field.Outputs, 132 } 133 case "event": 134 abi.Events[field.Name] = Event{ 135 Name: field.Name, 136 Anonymous: field.Anonymous, 137 Inputs: field.Inputs, 138 } 139 } 140 } 141 142 return nil 143 } 144 145 //method by id按4字节的ID查找方法 146 //如果未找到,则返回nil 147 func (abi *ABI) MethodById(sigdata []byte) (*Method, error) { 148 for _, method := range abi.Methods { 149 if bytes.Equal(method.Id(), sigdata[:4]) { 150 return &method, nil 151 } 152 } 153 return nil, fmt.Errorf("no method with id: %#x", sigdata[:4]) 154 }