| 
									
										
										
										
											2017-02-07 12:47:34 +02:00
										 |  |  | // Copyright 2017 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 keystore | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"math/big" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-08 20:25:52 +02:00
										 |  |  | 	ethereum "github.com/ethereum/go-ethereum" | 
					
						
							| 
									
										
										
										
											2017-02-07 12:47:34 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/accounts" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/core/types" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // keystoreWallet implements the accounts.Wallet interface for the original | 
					
						
							|  |  |  | // keystore. | 
					
						
							|  |  |  | type keystoreWallet struct { | 
					
						
							|  |  |  | 	account  accounts.Account // Single account contained in this wallet | 
					
						
							|  |  |  | 	keystore *KeyStore        // Keystore where the account originates from | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // URL implements accounts.Wallet, returning the URL of the account within. | 
					
						
							| 
									
										
										
										
											2017-02-08 15:53:02 +02:00
										 |  |  | func (w *keystoreWallet) URL() accounts.URL { | 
					
						
							| 
									
										
										
										
											2017-02-07 12:47:34 +02:00
										 |  |  | 	return w.account.URL | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Status implements accounts.Wallet, always returning "open", since there is no | 
					
						
							|  |  |  | // concept of open/close for plain keystore accounts. | 
					
						
							|  |  |  | func (w *keystoreWallet) Status() string { | 
					
						
							| 
									
										
										
										
											2017-02-08 15:53:02 +02:00
										 |  |  | 	w.keystore.mu.RLock() | 
					
						
							|  |  |  | 	defer w.keystore.mu.RUnlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if _, ok := w.keystore.unlocked[w.account.Address]; ok { | 
					
						
							|  |  |  | 		return "Unlocked" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return "Locked" | 
					
						
							| 
									
										
										
										
											2017-02-07 12:47:34 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Open implements accounts.Wallet, but is a noop for plain wallets since there | 
					
						
							|  |  |  | // is no connection or decryption step necessary to access the list of accounts. | 
					
						
							|  |  |  | func (w *keystoreWallet) Open(passphrase string) error { return nil } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Close implements accounts.Wallet, but is a noop for plain wallets since is no | 
					
						
							|  |  |  | // meaningful open operation. | 
					
						
							|  |  |  | func (w *keystoreWallet) Close() error { return nil } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Accounts implements accounts.Wallet, returning an account list consisting of | 
					
						
							|  |  |  | // a single account that the plain kestore wallet contains. | 
					
						
							|  |  |  | func (w *keystoreWallet) Accounts() []accounts.Account { | 
					
						
							|  |  |  | 	return []accounts.Account{w.account} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Contains implements accounts.Wallet, returning whether a particular account is | 
					
						
							|  |  |  | // or is not wrapped by this wallet instance. | 
					
						
							|  |  |  | func (w *keystoreWallet) Contains(account accounts.Account) bool { | 
					
						
							| 
									
										
										
										
											2017-02-08 15:53:02 +02:00
										 |  |  | 	return account.Address == w.account.Address && (account.URL == (accounts.URL{}) || account.URL == w.account.URL) | 
					
						
							| 
									
										
										
										
											2017-02-07 12:47:34 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Derive implements accounts.Wallet, but is a noop for plain wallets since there | 
					
						
							|  |  |  | // is no notion of hierarchical account derivation for plain keystore accounts. | 
					
						
							| 
									
										
										
										
											2017-02-08 20:25:52 +02:00
										 |  |  | func (w *keystoreWallet) Derive(path accounts.DerivationPath, pin bool) (accounts.Account, error) { | 
					
						
							| 
									
										
										
										
											2017-02-07 12:47:34 +02:00
										 |  |  | 	return accounts.Account{}, accounts.ErrNotSupported | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-08 20:25:52 +02:00
										 |  |  | // SelfDerive implements accounts.Wallet, but is a noop for plain wallets since | 
					
						
							|  |  |  | // there is no notion of hierarchical account derivation for plain keystore accounts. | 
					
						
							|  |  |  | func (w *keystoreWallet) SelfDerive(base accounts.DerivationPath, chain ethereum.ChainStateReader) {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-07 12:47:34 +02:00
										 |  |  | // SignHash implements accounts.Wallet, attempting to sign the given hash with | 
					
						
							|  |  |  | // the given account. If the wallet does not wrap this particular account, an | 
					
						
							|  |  |  | // error is returned to avoid account leakage (even though in theory we may be | 
					
						
							|  |  |  | // able to sign via our shared keystore backend). | 
					
						
							|  |  |  | func (w *keystoreWallet) SignHash(account accounts.Account, hash []byte) ([]byte, error) { | 
					
						
							|  |  |  | 	// Make sure the requested account is contained within | 
					
						
							|  |  |  | 	if account.Address != w.account.Address { | 
					
						
							|  |  |  | 		return nil, accounts.ErrUnknownAccount | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-08 15:53:02 +02:00
										 |  |  | 	if account.URL != (accounts.URL{}) && account.URL != w.account.URL { | 
					
						
							| 
									
										
										
										
											2017-02-07 12:47:34 +02:00
										 |  |  | 		return nil, accounts.ErrUnknownAccount | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Account seems valid, request the keystore to sign | 
					
						
							|  |  |  | 	return w.keystore.SignHash(account, hash) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SignTx implements accounts.Wallet, attempting to sign the given transaction | 
					
						
							|  |  |  | // with the given account. If the wallet does not wrap this particular account, | 
					
						
							|  |  |  | // an error is returned to avoid account leakage (even though in theory we may | 
					
						
							|  |  |  | // be able to sign via our shared keystore backend). | 
					
						
							|  |  |  | func (w *keystoreWallet) SignTx(account accounts.Account, tx *types.Transaction, chainID *big.Int) (*types.Transaction, error) { | 
					
						
							|  |  |  | 	// Make sure the requested account is contained within | 
					
						
							|  |  |  | 	if account.Address != w.account.Address { | 
					
						
							|  |  |  | 		return nil, accounts.ErrUnknownAccount | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-08 15:53:02 +02:00
										 |  |  | 	if account.URL != (accounts.URL{}) && account.URL != w.account.URL { | 
					
						
							| 
									
										
										
										
											2017-02-07 12:47:34 +02:00
										 |  |  | 		return nil, accounts.ErrUnknownAccount | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Account seems valid, request the keystore to sign | 
					
						
							|  |  |  | 	return w.keystore.SignTx(account, tx, chainID) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SignHashWithPassphrase implements accounts.Wallet, attempting to sign the | 
					
						
							|  |  |  | // given hash with the given account using passphrase as extra authentication. | 
					
						
							|  |  |  | func (w *keystoreWallet) SignHashWithPassphrase(account accounts.Account, passphrase string, hash []byte) ([]byte, error) { | 
					
						
							|  |  |  | 	// Make sure the requested account is contained within | 
					
						
							|  |  |  | 	if account.Address != w.account.Address { | 
					
						
							|  |  |  | 		return nil, accounts.ErrUnknownAccount | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-08 15:53:02 +02:00
										 |  |  | 	if account.URL != (accounts.URL{}) && account.URL != w.account.URL { | 
					
						
							| 
									
										
										
										
											2017-02-07 12:47:34 +02:00
										 |  |  | 		return nil, accounts.ErrUnknownAccount | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Account seems valid, request the keystore to sign | 
					
						
							|  |  |  | 	return w.keystore.SignHashWithPassphrase(account, passphrase, hash) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SignTxWithPassphrase implements accounts.Wallet, attempting to sign the given | 
					
						
							|  |  |  | // transaction with the given account using passphrase as extra authentication. | 
					
						
							|  |  |  | func (w *keystoreWallet) SignTxWithPassphrase(account accounts.Account, passphrase string, tx *types.Transaction, chainID *big.Int) (*types.Transaction, error) { | 
					
						
							|  |  |  | 	// Make sure the requested account is contained within | 
					
						
							|  |  |  | 	if account.Address != w.account.Address { | 
					
						
							|  |  |  | 		return nil, accounts.ErrUnknownAccount | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-08 15:53:02 +02:00
										 |  |  | 	if account.URL != (accounts.URL{}) && account.URL != w.account.URL { | 
					
						
							| 
									
										
										
										
											2017-02-07 12:47:34 +02:00
										 |  |  | 		return nil, accounts.ErrUnknownAccount | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Account seems valid, request the keystore to sign | 
					
						
							|  |  |  | 	return w.keystore.SignTxWithPassphrase(account, passphrase, tx, chainID) | 
					
						
							|  |  |  | } |