| 
									
										
										
										
											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" | 
					
						
							| 
									
										
										
										
											2020-04-15 15:23:58 +08:00
										 |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2015-01-27 16:19:21 +01:00
										 |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2019-06-24 12:52:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2020-04-22 16:25:36 +08:00
										 |  |  | 	"github.com/ethereum/go-ethereum/crypto" | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2020-04-15 15:23:58 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Additional "special" functions introduced in solidity v0.6.0. | 
					
						
							|  |  |  | 	// It's separated from the original default fallback. Each contract | 
					
						
							|  |  |  | 	// can only define one fallback and receive function. | 
					
						
							|  |  |  | 	Fallback Method // Note it's also used to represent legacy fallback before v0.6.0 | 
					
						
							|  |  |  | 	Receive  Method | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2020-04-20 09:01:04 +02: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) { | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							|  |  |  | 	// 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 { | 
					
						
							| 
									
										
										
										
											2020-04-15 15:23:58 +08:00
										 |  |  | 		Type    string | 
					
						
							|  |  |  | 		Name    string | 
					
						
							|  |  |  | 		Inputs  []Argument | 
					
						
							|  |  |  | 		Outputs []Argument | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Status indicator which can be: "pure", "view", | 
					
						
							|  |  |  | 		// "nonpayable" or "payable". | 
					
						
							| 
									
										
										
										
											2020-01-06 12:03:26 +01:00
										 |  |  | 		StateMutability string | 
					
						
							| 
									
										
										
										
											2020-04-15 15:23:58 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Deprecated Status indicators, but removed in v0.6.0. | 
					
						
							|  |  |  | 		Constant bool // True if function is either pure or view | 
					
						
							|  |  |  | 		Payable  bool // True if function is payable | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Event relevant indicator represents the event is | 
					
						
							|  |  |  | 		// declared as anonymous. | 
					
						
							|  |  |  | 		Anonymous bool | 
					
						
							| 
									
										
										
										
											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": | 
					
						
							| 
									
										
										
										
											2020-04-20 09:01:04 +02:00
										 |  |  | 			abi.Constructor = NewMethod("", "", Constructor, field.StateMutability, field.Constant, field.Payable, field.Inputs, nil) | 
					
						
							| 
									
										
										
										
											2020-04-15 15:23:58 +08:00
										 |  |  | 		case "function": | 
					
						
							| 
									
										
										
										
											2020-04-20 09:01:04 +02:00
										 |  |  | 			name := abi.overloadedMethodName(field.Name) | 
					
						
							|  |  |  | 			abi.Methods[name] = NewMethod(name, field.Name, Function, field.StateMutability, field.Constant, field.Payable, field.Inputs, field.Outputs) | 
					
						
							| 
									
										
										
										
											2020-04-15 15:23:58 +08:00
										 |  |  | 		case "fallback": | 
					
						
							|  |  |  | 			// New introduced function type in v0.6.0, check more detail | 
					
						
							|  |  |  | 			// here https://solidity.readthedocs.io/en/v0.6.0/contracts.html#fallback-function | 
					
						
							|  |  |  | 			if abi.HasFallback() { | 
					
						
							|  |  |  | 				return errors.New("only single fallback is allowed") | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2020-04-20 09:01:04 +02:00
										 |  |  | 			abi.Fallback = NewMethod("", "", Fallback, field.StateMutability, field.Constant, field.Payable, nil, nil) | 
					
						
							| 
									
										
										
										
											2020-04-15 15:23:58 +08:00
										 |  |  | 		case "receive": | 
					
						
							|  |  |  | 			// New introduced function type in v0.6.0, check more detail | 
					
						
							|  |  |  | 			// here https://solidity.readthedocs.io/en/v0.6.0/contracts.html#fallback-function | 
					
						
							|  |  |  | 			if abi.HasReceive() { | 
					
						
							|  |  |  | 				return errors.New("only single receive is allowed") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if field.StateMutability != "payable" { | 
					
						
							|  |  |  | 				return errors.New("the statemutability of receive can only be payable") | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2020-04-20 09:01:04 +02:00
										 |  |  | 			abi.Receive = NewMethod("", "", Receive, field.StateMutability, field.Constant, field.Payable, nil, nil) | 
					
						
							| 
									
										
										
										
											2016-01-27 08:38:53 +01:00
										 |  |  | 		case "event": | 
					
						
							| 
									
										
										
										
											2020-04-20 09:01:04 +02:00
										 |  |  | 			name := abi.overloadedEventName(field.Name) | 
					
						
							|  |  |  | 			abi.Events[name] = NewEvent(name, field.Name, field.Anonymous, field.Inputs) | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			return fmt.Errorf("abi: could not recognize type %v of field %v", field.Type, field.Name) | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-20 09:01:04 +02:00
										 |  |  | // overloadedMethodName returns the next available name for a given function. | 
					
						
							|  |  |  | // Needed since solidity allows for function overload. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // e.g. if the abi contains Methods send, send1 | 
					
						
							|  |  |  | // overloadedMethodName would return send2 for input send. | 
					
						
							|  |  |  | func (abi *ABI) overloadedMethodName(rawName string) string { | 
					
						
							|  |  |  | 	name := rawName | 
					
						
							|  |  |  | 	_, ok := abi.Methods[name] | 
					
						
							|  |  |  | 	for idx := 0; ok; idx++ { | 
					
						
							|  |  |  | 		name = fmt.Sprintf("%s%d", rawName, idx) | 
					
						
							|  |  |  | 		_, ok = abi.Methods[name] | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return name | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // overloadedEventName returns the next available name for a given event. | 
					
						
							|  |  |  | // Needed since solidity allows for event overload. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // e.g. if the abi contains events received, received1 | 
					
						
							|  |  |  | // overloadedEventName would return received2 for input received. | 
					
						
							|  |  |  | func (abi *ABI) overloadedEventName(rawName string) string { | 
					
						
							|  |  |  | 	name := rawName | 
					
						
							|  |  |  | 	_, ok := abi.Events[name] | 
					
						
							|  |  |  | 	for idx := 0; ok; idx++ { | 
					
						
							|  |  |  | 		name = fmt.Sprintf("%s%d", rawName, idx) | 
					
						
							|  |  |  | 		_, ok = abi.Events[name] | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return name | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							| 
									
										
										
										
											2020-04-20 09:01:04 +02: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 { | 
					
						
							| 
									
										
										
										
											2020-04-20 09:01:04 +02: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()) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-04-15 15:23:58 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // HasFallback returns an indicator whether a fallback function is included. | 
					
						
							|  |  |  | func (abi *ABI) HasFallback() bool { | 
					
						
							| 
									
										
										
										
											2020-04-20 09:01:04 +02:00
										 |  |  | 	return abi.Fallback.Type == Fallback | 
					
						
							| 
									
										
										
										
											2020-04-15 15:23:58 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // HasReceive returns an indicator whether a receive function is included. | 
					
						
							|  |  |  | func (abi *ABI) HasReceive() bool { | 
					
						
							| 
									
										
										
										
											2020-04-20 09:01:04 +02:00
										 |  |  | 	return abi.Receive.Type == Receive | 
					
						
							| 
									
										
										
										
											2020-04-15 15:23:58 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-04-22 16:25:36 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // revertSelector is a special function selector for revert reason unpacking. | 
					
						
							|  |  |  | var revertSelector = crypto.Keccak256([]byte("Error(string)"))[:4] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // UnpackRevert resolves the abi-encoded revert reason. According to the solidity | 
					
						
							|  |  |  | // spec https://solidity.readthedocs.io/en/latest/control-structures.html#revert, | 
					
						
							|  |  |  | // the provided revert reason is abi-encoded as if it were a call to a function | 
					
						
							|  |  |  | // `Error(string)`. So it's a special tool for it. | 
					
						
							|  |  |  | func UnpackRevert(data []byte) (string, error) { | 
					
						
							|  |  |  | 	if len(data) < 4 { | 
					
						
							|  |  |  | 		return "", errors.New("invalid data for unpacking") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if !bytes.Equal(data[:4], revertSelector) { | 
					
						
							|  |  |  | 		return "", errors.New("invalid data for unpacking") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	var reason string | 
					
						
							|  |  |  | 	typ, _ := NewType("string", "", nil) | 
					
						
							|  |  |  | 	if err := (Arguments{{Type: typ}}).Unpack(&reason, data[4:]); err != nil { | 
					
						
							|  |  |  | 		return "", err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return reason, nil | 
					
						
							|  |  |  | } |