| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // Copyright 2015 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // This file is part of the go-ethereum library. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-23 18:35:11 +02:00
										 |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // it under the terms of the GNU Lesser General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-27 16:19:21 +01:00
										 |  |  | package abi | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2017-12-21 16:18:37 +01:00
										 |  |  | 	"bytes" | 
					
						
							| 
									
										
										
										
											2015-01-27 16:19:21 +01:00
										 |  |  | 	"encoding/json" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2019-06-24 12:52:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2015-01-27 16:19:21 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // The ABI holds information about a contract's context and available | 
					
						
							|  |  |  | // invokable methods. It will allow you to type check function calls and | 
					
						
							|  |  |  | // packs data accordingly. | 
					
						
							|  |  |  | type ABI struct { | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 	Constructor Method | 
					
						
							|  |  |  | 	Methods     map[string]Method | 
					
						
							|  |  |  | 	Events      map[string]Event | 
					
						
							| 
									
										
										
										
											2015-01-27 16:19:21 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-20 13:45:37 +01:00
										 |  |  | // JSON returns a parsed ABI interface and error if it failed. | 
					
						
							|  |  |  | func JSON(reader io.Reader) (ABI, error) { | 
					
						
							|  |  |  | 	dec := json.NewDecoder(reader) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var abi ABI | 
					
						
							|  |  |  | 	if err := dec.Decode(&abi); err != nil { | 
					
						
							|  |  |  | 		return ABI{}, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return abi, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-27 16:19:21 +01:00
										 |  |  | // Pack the given method name to conform the ABI. Method call's data | 
					
						
							|  |  |  | // will consist of method_id, args0, arg1, ... argN. Method id consists | 
					
						
							|  |  |  | // of 4 bytes and arguments are all 32 bytes. | 
					
						
							|  |  |  | // Method ids are created from the first 4 bytes of the hash of the | 
					
						
							|  |  |  | // methods string signature. (signature = baz(uint32,string32)) | 
					
						
							|  |  |  | func (abi ABI) Pack(name string, args ...interface{}) ([]byte, error) { | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 	// Fetch the ABI of the requested method | 
					
						
							|  |  |  | 	if name == "" { | 
					
						
							| 
									
										
										
										
											2017-12-21 10:26:30 +01:00
										 |  |  | 		// constructor | 
					
						
							|  |  |  | 		arguments, err := abi.Constructor.Inputs.Pack(args...) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-12-21 10:26:30 +01:00
										 |  |  | 		return arguments, nil | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-12-21 10:26:30 +01:00
										 |  |  | 	method, exist := abi.Methods[name] | 
					
						
							|  |  |  | 	if !exist { | 
					
						
							|  |  |  | 		return nil, fmt.Errorf("method '%s' not found", name) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	arguments, err := method.Inputs.Pack(args...) | 
					
						
							| 
									
										
										
										
											2015-01-27 16:19:21 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 	// Pack up the method ID too if not a constructor and return | 
					
						
							| 
									
										
										
										
											2019-08-02 15:20:46 +08:00
										 |  |  | 	return append(method.ID(), arguments...), nil | 
					
						
							| 
									
										
										
										
											2015-01-27 16:19:21 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-16 12:48:33 +02:00
										 |  |  | // Unpack output in v according to the abi specification | 
					
						
							| 
									
										
										
										
											2019-04-01 08:42:59 -05:00
										 |  |  | func (abi ABI) Unpack(v interface{}, name string, data []byte) (err error) { | 
					
						
							|  |  |  | 	if len(data) == 0 { | 
					
						
							| 
									
										
										
										
											2017-12-21 01:59:14 -08:00
										 |  |  | 		return fmt.Errorf("abi: unmarshalling empty output") | 
					
						
							| 
									
										
										
										
											2016-05-11 13:21:25 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-10-17 06:07:08 -05:00
										 |  |  | 	// since there can't be naming collisions with contracts and events, | 
					
						
							|  |  |  | 	// we need to decide whether we're calling a method or an event | 
					
						
							|  |  |  | 	if method, ok := abi.Methods[name]; ok { | 
					
						
							| 
									
										
										
										
											2019-04-01 08:42:59 -05:00
										 |  |  | 		if len(data)%32 != 0 { | 
					
						
							|  |  |  | 			return fmt.Errorf("abi: improperly formatted output: %s - Bytes: [%+v]", string(data), data) | 
					
						
							| 
									
										
										
										
											2017-12-21 01:59:14 -08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-04-01 08:42:59 -05:00
										 |  |  | 		return method.Outputs.Unpack(v, data) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if event, ok := abi.Events[name]; ok { | 
					
						
							|  |  |  | 		return event.Inputs.Unpack(v, data) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return fmt.Errorf("abi: could not locate named method or event") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // UnpackIntoMap unpacks a log into the provided map[string]interface{} | 
					
						
							|  |  |  | func (abi ABI) UnpackIntoMap(v map[string]interface{}, name string, data []byte) (err error) { | 
					
						
							|  |  |  | 	if len(data) == 0 { | 
					
						
							|  |  |  | 		return fmt.Errorf("abi: unmarshalling empty output") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// since there can't be naming collisions with contracts and events, | 
					
						
							|  |  |  | 	// we need to decide whether we're calling a method or an event | 
					
						
							|  |  |  | 	if method, ok := abi.Methods[name]; ok { | 
					
						
							|  |  |  | 		if len(data)%32 != 0 { | 
					
						
							|  |  |  | 			return fmt.Errorf("abi: improperly formatted output") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return method.Outputs.UnpackIntoMap(v, data) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if event, ok := abi.Events[name]; ok { | 
					
						
							|  |  |  | 		return event.Inputs.UnpackIntoMap(v, data) | 
					
						
							| 
									
										
										
										
											2017-10-17 06:07:08 -05:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-12-21 10:26:30 +01:00
										 |  |  | 	return fmt.Errorf("abi: could not locate named method or event") | 
					
						
							| 
									
										
										
										
											2015-11-20 13:45:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-10 02:48:51 +01:00
										 |  |  | // UnmarshalJSON implements json.Unmarshaler interface | 
					
						
							| 
									
										
										
										
											2015-01-27 16:19:21 +01:00
										 |  |  | func (abi *ABI) UnmarshalJSON(data []byte) error { | 
					
						
							| 
									
										
										
										
											2016-01-27 08:38:53 +01:00
										 |  |  | 	var fields []struct { | 
					
						
							| 
									
										
										
										
											2016-12-22 01:51:20 +01:00
										 |  |  | 		Type      string | 
					
						
							|  |  |  | 		Name      string | 
					
						
							|  |  |  | 		Constant  bool | 
					
						
							|  |  |  | 		Anonymous bool | 
					
						
							|  |  |  | 		Inputs    []Argument | 
					
						
							|  |  |  | 		Outputs   []Argument | 
					
						
							| 
									
										
										
										
											2016-01-27 08:38:53 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if err := json.Unmarshal(data, &fields); err != nil { | 
					
						
							| 
									
										
										
										
											2015-01-27 16:19:21 +01:00
										 |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	abi.Methods = make(map[string]Method) | 
					
						
							| 
									
										
										
										
											2016-01-27 08:38:53 +01:00
										 |  |  | 	abi.Events = make(map[string]Event) | 
					
						
							|  |  |  | 	for _, field := range fields { | 
					
						
							|  |  |  | 		switch field.Type { | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 		case "constructor": | 
					
						
							|  |  |  | 			abi.Constructor = Method{ | 
					
						
							|  |  |  | 				Inputs: field.Inputs, | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-01-27 08:38:53 +01:00
										 |  |  | 		// empty defaults to function according to the abi spec | 
					
						
							|  |  |  | 		case "function", "": | 
					
						
							| 
									
										
										
										
											2019-07-01 22:51:21 +08:00
										 |  |  | 			name := field.Name | 
					
						
							|  |  |  | 			_, ok := abi.Methods[name] | 
					
						
							|  |  |  | 			for idx := 0; ok; idx++ { | 
					
						
							|  |  |  | 				name = fmt.Sprintf("%s%d", field.Name, idx) | 
					
						
							|  |  |  | 				_, ok = abi.Methods[name] | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			abi.Methods[name] = Method{ | 
					
						
							|  |  |  | 				Name:    name, | 
					
						
							| 
									
										
										
										
											2019-08-02 15:20:46 +08:00
										 |  |  | 				RawName: field.Name, | 
					
						
							| 
									
										
										
										
											2016-03-16 11:19:25 +02:00
										 |  |  | 				Const:   field.Constant, | 
					
						
							| 
									
										
										
										
											2016-01-27 08:38:53 +01:00
										 |  |  | 				Inputs:  field.Inputs, | 
					
						
							|  |  |  | 				Outputs: field.Outputs, | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		case "event": | 
					
						
							| 
									
										
										
										
											2019-07-01 22:51:21 +08:00
										 |  |  | 			name := field.Name | 
					
						
							|  |  |  | 			_, ok := abi.Events[name] | 
					
						
							|  |  |  | 			for idx := 0; ok; idx++ { | 
					
						
							|  |  |  | 				name = fmt.Sprintf("%s%d", field.Name, idx) | 
					
						
							|  |  |  | 				_, ok = abi.Events[name] | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			abi.Events[name] = Event{ | 
					
						
							|  |  |  | 				Name:      name, | 
					
						
							| 
									
										
										
										
											2019-08-02 15:20:46 +08:00
										 |  |  | 				RawName:   field.Name, | 
					
						
							| 
									
										
										
										
											2016-12-22 01:51:20 +01:00
										 |  |  | 				Anonymous: field.Anonymous, | 
					
						
							|  |  |  | 				Inputs:    field.Inputs, | 
					
						
							| 
									
										
										
										
											2016-01-27 08:38:53 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-01-27 16:19:21 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-12-21 16:18:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | // MethodById looks up a method by the 4-byte id | 
					
						
							|  |  |  | // returns nil if none found | 
					
						
							| 
									
										
										
										
											2017-12-30 16:07:12 +01:00
										 |  |  | func (abi *ABI) MethodById(sigdata []byte) (*Method, error) { | 
					
						
							| 
									
										
										
										
											2018-10-01 14:17:37 +02:00
										 |  |  | 	if len(sigdata) < 4 { | 
					
						
							| 
									
										
										
										
											2019-02-19 00:34:55 -08:00
										 |  |  | 		return nil, fmt.Errorf("data too short (%d bytes) for abi method lookup", len(sigdata)) | 
					
						
							| 
									
										
										
										
											2018-10-01 14:17:37 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-12-21 16:18:37 +01:00
										 |  |  | 	for _, method := range abi.Methods { | 
					
						
							| 
									
										
										
										
											2019-08-02 15:20:46 +08:00
										 |  |  | 		if bytes.Equal(method.ID(), sigdata[:4]) { | 
					
						
							| 
									
										
										
										
											2017-12-28 11:17:45 +01:00
										 |  |  | 			return &method, nil | 
					
						
							| 
									
										
										
										
											2017-12-21 16:18:37 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-01-13 15:12:52 +01:00
										 |  |  | 	return nil, fmt.Errorf("no method with id: %#x", sigdata[:4]) | 
					
						
							| 
									
										
										
										
											2017-12-21 16:18:37 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-06-24 12:52:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | // EventByID looks an event up by its topic hash in the | 
					
						
							|  |  |  | // ABI and returns nil if none found. | 
					
						
							|  |  |  | func (abi *ABI) EventByID(topic common.Hash) (*Event, error) { | 
					
						
							|  |  |  | 	for _, event := range abi.Events { | 
					
						
							| 
									
										
										
										
											2019-08-02 15:20:46 +08:00
										 |  |  | 		if bytes.Equal(event.ID().Bytes(), topic.Bytes()) { | 
					
						
							| 
									
										
										
										
											2019-06-24 12:52:50 +02:00
										 |  |  | 			return &event, nil | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil, fmt.Errorf("no event with id: %#x", topic.Hex()) | 
					
						
							|  |  |  | } |