| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | // Copyright 2016 The go-ethereum Authors | 
					
						
							|  |  |  | // This file is part of the go-ethereum library. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // 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. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							|  |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package bind | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2021-07-27 16:24:27 +02:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2016-03-02 13:57:15 +01:00
										 |  |  | 	"crypto/ecdsa" | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2016-03-21 14:34:49 +02:00
										 |  |  | 	"io" | 
					
						
							|  |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 	"math/big" | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-08 15:19:26 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/accounts" | 
					
						
							| 
									
										
										
										
											2019-07-04 03:54:59 +08:00
										 |  |  | 	"github.com/ethereum/go-ethereum/accounts/external" | 
					
						
							| 
									
										
										
										
											2017-01-24 11:49:20 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/accounts/keystore" | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/core/types" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/crypto" | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/log" | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | // ErrNoChainID is returned whenever the user failed to specify a chain id. | 
					
						
							|  |  |  | var ErrNoChainID = errors.New("no chain id specified") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ErrNotAuthorized is returned when an account is not properly unlocked. | 
					
						
							|  |  |  | var ErrNotAuthorized = errors.New("not authorized to sign this account") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | // NewTransactor is a utility method to easily create a transaction signer from | 
					
						
							| 
									
										
										
										
											2016-03-21 14:34:49 +02:00
										 |  |  | // an encrypted json key stream and the associated passphrase. | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | // | 
					
						
							|  |  |  | // Deprecated: Use NewTransactorWithChainID instead. | 
					
						
							| 
									
										
										
										
											2016-03-21 14:34:49 +02:00
										 |  |  | func NewTransactor(keyin io.Reader, passphrase string) (*TransactOpts, error) { | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 	log.Warn("WARNING: NewTransactor has been deprecated in favour of NewTransactorWithChainID") | 
					
						
							| 
									
										
										
										
											2016-03-21 14:34:49 +02:00
										 |  |  | 	json, err := ioutil.ReadAll(keyin) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-01-24 11:49:20 +02:00
										 |  |  | 	key, err := keystore.DecryptKey(json, passphrase) | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-03-02 13:57:15 +01:00
										 |  |  | 	return NewKeyedTransactor(key.PrivateKey), nil | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-04 03:54:59 +08:00
										 |  |  | // NewKeyStoreTransactor is a utility method to easily create a transaction signer from | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | // an decrypted key from a keystore. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Deprecated: Use NewKeyStoreTransactorWithChainID instead. | 
					
						
							| 
									
										
										
										
											2019-06-12 13:06:37 +02:00
										 |  |  | func NewKeyStoreTransactor(keystore *keystore.KeyStore, account accounts.Account) (*TransactOpts, error) { | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 	log.Warn("WARNING: NewKeyStoreTransactor has been deprecated in favour of NewTransactorWithChainID") | 
					
						
							|  |  |  | 	signer := types.HomesteadSigner{} | 
					
						
							| 
									
										
										
										
											2019-06-08 15:19:26 +02:00
										 |  |  | 	return &TransactOpts{ | 
					
						
							|  |  |  | 		From: account.Address, | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 		Signer: func(address common.Address, tx *types.Transaction) (*types.Transaction, error) { | 
					
						
							| 
									
										
										
										
											2019-06-08 15:19:26 +02:00
										 |  |  | 			if address != account.Address { | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 				return nil, ErrNotAuthorized | 
					
						
							| 
									
										
										
										
											2019-06-08 15:19:26 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			signature, err := keystore.SignHash(account, signer.Hash(tx).Bytes()) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return tx.WithSignature(signer, signature) | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-07-27 16:24:27 +02:00
										 |  |  | 		Context: context.Background(), | 
					
						
							| 
									
										
										
										
											2019-06-08 15:19:26 +02:00
										 |  |  | 	}, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | // NewKeyedTransactor is a utility method to easily create a transaction signer | 
					
						
							| 
									
										
										
										
											2016-03-02 13:57:15 +01:00
										 |  |  | // from a single private key. | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | // | 
					
						
							|  |  |  | // Deprecated: Use NewKeyedTransactorWithChainID instead. | 
					
						
							| 
									
										
										
										
											2016-03-02 13:57:15 +01:00
										 |  |  | func NewKeyedTransactor(key *ecdsa.PrivateKey) *TransactOpts { | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 	log.Warn("WARNING: NewKeyedTransactor has been deprecated in favour of NewKeyedTransactorWithChainID") | 
					
						
							| 
									
										
										
										
											2016-03-02 13:57:15 +01:00
										 |  |  | 	keyAddr := crypto.PubkeyToAddress(key.PublicKey) | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 	signer := types.HomesteadSigner{} | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 	return &TransactOpts{ | 
					
						
							| 
									
										
										
										
											2016-03-02 13:57:15 +01:00
										 |  |  | 		From: keyAddr, | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 		Signer: func(address common.Address, tx *types.Transaction) (*types.Transaction, error) { | 
					
						
							| 
									
										
										
										
											2016-03-02 13:57:15 +01:00
										 |  |  | 			if address != keyAddr { | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 				return nil, ErrNotAuthorized | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-01-05 12:35:23 +02:00
										 |  |  | 			signature, err := crypto.Sign(signer.Hash(tx).Bytes(), key) | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-11-02 13:44:13 +01:00
										 |  |  | 			return tx.WithSignature(signer, signature) | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-07-27 16:24:27 +02:00
										 |  |  | 		Context: context.Background(), | 
					
						
							| 
									
										
										
										
											2016-03-17 19:27:37 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-07-04 03:54:59 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | // NewTransactorWithChainID is a utility method to easily create a transaction signer from | 
					
						
							|  |  |  | // an encrypted json key stream and the associated passphrase. | 
					
						
							|  |  |  | func NewTransactorWithChainID(keyin io.Reader, passphrase string, chainID *big.Int) (*TransactOpts, error) { | 
					
						
							|  |  |  | 	json, err := ioutil.ReadAll(keyin) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	key, err := keystore.DecryptKey(json, passphrase) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NewKeyedTransactorWithChainID(key.PrivateKey, chainID) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NewKeyStoreTransactorWithChainID is a utility method to easily create a transaction signer from | 
					
						
							|  |  |  | // an decrypted key from a keystore. | 
					
						
							|  |  |  | func NewKeyStoreTransactorWithChainID(keystore *keystore.KeyStore, account accounts.Account, chainID *big.Int) (*TransactOpts, error) { | 
					
						
							|  |  |  | 	if chainID == nil { | 
					
						
							|  |  |  | 		return nil, ErrNoChainID | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-02-25 07:26:57 -07:00
										 |  |  | 	signer := types.LatestSignerForChainID(chainID) | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 	return &TransactOpts{ | 
					
						
							|  |  |  | 		From: account.Address, | 
					
						
							|  |  |  | 		Signer: func(address common.Address, tx *types.Transaction) (*types.Transaction, error) { | 
					
						
							|  |  |  | 			if address != account.Address { | 
					
						
							|  |  |  | 				return nil, ErrNotAuthorized | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			signature, err := keystore.SignHash(account, signer.Hash(tx).Bytes()) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return tx.WithSignature(signer, signature) | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-07-27 16:24:27 +02:00
										 |  |  | 		Context: context.Background(), | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 	}, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NewKeyedTransactorWithChainID is a utility method to easily create a transaction signer | 
					
						
							|  |  |  | // from a single private key. | 
					
						
							|  |  |  | func NewKeyedTransactorWithChainID(key *ecdsa.PrivateKey, chainID *big.Int) (*TransactOpts, error) { | 
					
						
							|  |  |  | 	keyAddr := crypto.PubkeyToAddress(key.PublicKey) | 
					
						
							|  |  |  | 	if chainID == nil { | 
					
						
							|  |  |  | 		return nil, ErrNoChainID | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-02-25 07:26:57 -07:00
										 |  |  | 	signer := types.LatestSignerForChainID(chainID) | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 	return &TransactOpts{ | 
					
						
							|  |  |  | 		From: keyAddr, | 
					
						
							|  |  |  | 		Signer: func(address common.Address, tx *types.Transaction) (*types.Transaction, error) { | 
					
						
							|  |  |  | 			if address != keyAddr { | 
					
						
							|  |  |  | 				return nil, ErrNotAuthorized | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			signature, err := crypto.Sign(signer.Hash(tx).Bytes(), key) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return tx.WithSignature(signer, signature) | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-07-27 16:24:27 +02:00
										 |  |  | 		Context: context.Background(), | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 	}, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-04 03:54:59 +08:00
										 |  |  | // NewClefTransactor is a utility method to easily create a transaction signer | 
					
						
							|  |  |  | // with a clef backend. | 
					
						
							|  |  |  | func NewClefTransactor(clef *external.ExternalSigner, account accounts.Account) *TransactOpts { | 
					
						
							|  |  |  | 	return &TransactOpts{ | 
					
						
							|  |  |  | 		From: account.Address, | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 		Signer: func(address common.Address, transaction *types.Transaction) (*types.Transaction, error) { | 
					
						
							| 
									
										
										
										
											2019-07-04 03:54:59 +08:00
										 |  |  | 			if address != account.Address { | 
					
						
							| 
									
										
										
										
											2020-12-08 14:44:56 +01:00
										 |  |  | 				return nil, ErrNotAuthorized | 
					
						
							| 
									
										
										
										
											2019-07-04 03:54:59 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			return clef.SignTx(account, transaction, nil) // Clef enforces its own chain id | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-07-27 16:24:27 +02:00
										 |  |  | 		Context: context.Background(), | 
					
						
							| 
									
										
										
										
											2019-07-04 03:54:59 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } |