Merge pull request #3540 from ethcore/delete-accounts
Real deleting accounts
This commit is contained in:
		
						commit
						46e5a84bc5
					
				| @ -276,14 +276,20 @@ impl AccountProvider { | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/// Returns `true` if the password for `account` is `password`. `false` if not.
 | 	/// Returns `true` if the password for `account` is `password`. `false` if not.
 | ||||||
| 	pub fn test_password(&self, account: &Address, password: String) -> Result<bool, Error> { | 	pub fn test_password(&self, account: &Address, password: &str) -> Result<bool, Error> { | ||||||
| 		match self.sstore.sign(account, &password, &Default::default()) { | 		match self.sstore.sign(account, password, &Default::default()) { | ||||||
| 			Ok(_) => Ok(true), | 			Ok(_) => Ok(true), | ||||||
| 			Err(SSError::InvalidPassword) => Ok(false), | 			Err(SSError::InvalidPassword) => Ok(false), | ||||||
| 			Err(e) => Err(Error::SStore(e)), | 			Err(e) => Err(Error::SStore(e)), | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	/// Permanently removes an account.
 | ||||||
|  | 	pub fn kill_account(&self, account: &Address, password: &str) -> Result<(), Error> { | ||||||
|  | 		try!(self.sstore.remove_account(account, &password)); | ||||||
|  | 		Ok(()) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	/// Changes the password of `account` from `password` to `new_password`. Fails if incorrect `password` given.
 | 	/// Changes the password of `account` from `password` to `new_password`. Fails if incorrect `password` given.
 | ||||||
| 	pub fn change_password(&self, account: &Address, password: String, new_password: String) -> Result<(), Error> { | 	pub fn change_password(&self, account: &Address, password: String, new_password: String) -> Result<(), Error> { | ||||||
| 		self.sstore.change_password(account, &password, &new_password).map_err(Error::SStore) | 		self.sstore.change_password(account, &password, &new_password).map_err(Error::SStore) | ||||||
|  | |||||||
| @ -123,6 +123,11 @@ export default class Parity { | |||||||
|       .then((accounts) => (accounts || []).map(outAddress)); |       .then((accounts) => (accounts || []).map(outAddress)); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   killAccount (account, password) { | ||||||
|  |     return this._transport | ||||||
|  |       .execute('parity_killAccount', inAddress(account), password); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   listGethAccounts () { |   listGethAccounts () { | ||||||
|     return this._transport |     return this._transport | ||||||
|       .execute('parity_listGethAccounts') |       .execute('parity_listGethAccounts') | ||||||
|  | |||||||
| @ -59,6 +59,7 @@ export default class Personal { | |||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       switch (data.method) { |       switch (data.method) { | ||||||
|  |         case 'parity_killAccount': | ||||||
|         case 'parity_importGethAccounts': |         case 'parity_importGethAccounts': | ||||||
|         case 'personal_newAccount': |         case 'personal_newAccount': | ||||||
|         case 'parity_newAccountFromPhrase': |         case 'parity_newAccountFromPhrase': | ||||||
|  | |||||||
| @ -238,6 +238,24 @@ export default { | |||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
| 
 | 
 | ||||||
|  |   killAccount: { | ||||||
|  |     desc: 'Deletes an account', | ||||||
|  |     params: [ | ||||||
|  |       { | ||||||
|  |         type: Address, | ||||||
|  |         desc: 'The account to remove' | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         type: String, | ||||||
|  |         desc: 'Account password' | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     returns: { | ||||||
|  |       type: Boolean, | ||||||
|  |       desc: 'true on success' | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|   listGethAccounts: { |   listGethAccounts: { | ||||||
|     desc: 'Returns a list of the accounts available from Geth', |     desc: 'Returns a list of the accounts available from Geth', | ||||||
|     params: [], |     params: [], | ||||||
|  | |||||||
							
								
								
									
										54
									
								
								js/src/modals/DeleteAccount/deleteAccount.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								js/src/modals/DeleteAccount/deleteAccount.css
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,54 @@ | |||||||
|  | /* Copyright 2015, 2016 Ethcore (UK) Ltd. | ||||||
|  | /* This file is part of Parity. | ||||||
|  | /* | ||||||
|  | /* Parity is free software: you can redistribute it and/or modify | ||||||
|  | /* it under the terms of the GNU General Public License as published by | ||||||
|  | /* the Free Software Foundation, either version 3 of the License, or | ||||||
|  | /* (at your option) any later version. | ||||||
|  | /* | ||||||
|  | /* Parity 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 General Public License for more details. | ||||||
|  | /* | ||||||
|  | /* You should have received a copy of the GNU General Public License | ||||||
|  | /* along with Parity.  If not, see <http://www.gnu.org/licenses/>. | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | .body { | ||||||
|  |   .hero { | ||||||
|  |     padding-bottom: 1em; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   .info { | ||||||
|  |     display: inline-block; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   .icon { | ||||||
|  |     display: inline-block; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   .nameinfo { | ||||||
|  |     display: inline-block; | ||||||
|  |     text-align: left; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   .header { | ||||||
|  |     text-transform: uppercase; | ||||||
|  |     font-size: 1.25em; | ||||||
|  |     padding-bottom: 0.25em; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   .address { | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   .description { | ||||||
|  |     padding-top: 1em; | ||||||
|  |     font-size: 0.75em; | ||||||
|  |     color: #aaa; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   .password { | ||||||
|  |     padding: 1em 5em; | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										125
									
								
								js/src/modals/DeleteAccount/deleteAccount.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								js/src/modals/DeleteAccount/deleteAccount.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,125 @@ | |||||||
|  | // Copyright 2015, 2016 Ethcore (UK) Ltd.
 | ||||||
|  | // This file is part of Parity.
 | ||||||
|  | 
 | ||||||
|  | // Parity is free software: you can redistribute it and/or modify
 | ||||||
|  | // it under the terms of the GNU General Public License as published by
 | ||||||
|  | // the Free Software Foundation, either version 3 of the License, or
 | ||||||
|  | // (at your option) any later version.
 | ||||||
|  | 
 | ||||||
|  | // Parity 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 General Public License for more details.
 | ||||||
|  | 
 | ||||||
|  | // You should have received a copy of the GNU General Public License
 | ||||||
|  | // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  | 
 | ||||||
|  | import React, { Component, PropTypes } from 'react'; | ||||||
|  | import { connect } from 'react-redux'; | ||||||
|  | import { bindActionCreators } from 'redux'; | ||||||
|  | 
 | ||||||
|  | import { ConfirmDialog, IdentityIcon, IdentityName, Input } from '../../ui'; | ||||||
|  | import { newError } from '../../redux/actions'; | ||||||
|  | 
 | ||||||
|  | import styles from './deleteAccount.css'; | ||||||
|  | 
 | ||||||
|  | class DeleteAccount extends Component { | ||||||
|  |   static contextTypes = { | ||||||
|  |     api: PropTypes.object.isRequired, | ||||||
|  |     router: PropTypes.object | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   static propTypes = { | ||||||
|  |     account: PropTypes.object.isRequired, | ||||||
|  |     onClose: PropTypes.func.isRequired, | ||||||
|  |     newError: PropTypes.func.isRequired | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   state = { | ||||||
|  |     password: '' | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   render () { | ||||||
|  |     const { account } = this.props; | ||||||
|  |     const { password } = this.state; | ||||||
|  | 
 | ||||||
|  |     return ( | ||||||
|  |       <ConfirmDialog | ||||||
|  |         className={ styles.body } | ||||||
|  |         title='confirm removal' | ||||||
|  |         visible | ||||||
|  |         onDeny={ this.closeDeleteDialog } | ||||||
|  |         onConfirm={ this.onDeleteConfirmed }> | ||||||
|  |         <div className={ styles.hero }> | ||||||
|  |           Are you sure you want to permanently delete the following account? | ||||||
|  |         </div> | ||||||
|  |         <div className={ styles.info }> | ||||||
|  |           <IdentityIcon | ||||||
|  |             className={ styles.icon } | ||||||
|  |             address={ account.address } /> | ||||||
|  |           <div className={ styles.nameinfo }> | ||||||
|  |             <div className={ styles.header }> | ||||||
|  |               <IdentityName address={ account.address } unknown /> | ||||||
|  |             </div> | ||||||
|  |             <div className={ styles.address }> | ||||||
|  |               { account.address } | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         <div className={ styles.description }> | ||||||
|  |           { account.meta.description } | ||||||
|  |         </div> | ||||||
|  |         <div className={ styles.password }> | ||||||
|  |           <Input | ||||||
|  |             label='account password' | ||||||
|  |             hint='provide the account password to confirm the account deletion' | ||||||
|  |             type='password' | ||||||
|  |             value={ password } | ||||||
|  |             onChange={ this.onChangePassword } /> | ||||||
|  |         </div> | ||||||
|  |       </ConfirmDialog> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   onChangePassword = (event, password) => { | ||||||
|  |     this.setState({ password }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   onDeleteConfirmed = () => { | ||||||
|  |     const { api, router } = this.context; | ||||||
|  |     const { account, newError } = this.props; | ||||||
|  |     const { password } = this.state; | ||||||
|  | 
 | ||||||
|  |     api.parity | ||||||
|  |       .killAccount(account.address, password) | ||||||
|  |       .then((result) => { | ||||||
|  |         if (result === true) { | ||||||
|  |           router.push('/accounts'); | ||||||
|  |           this.closeDeleteDialog(); | ||||||
|  |         } else { | ||||||
|  |           newError(new Error('Deletion failed.')); | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |       .catch((error) => { | ||||||
|  |         console.error('onDeleteConfirmed', error); | ||||||
|  |         newError(new Error(`Deletion failed: ${error.message}`)); | ||||||
|  |       }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   closeDeleteDialog = () => { | ||||||
|  |     this.props.onClose(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function mapStateToProps (state) { | ||||||
|  |   return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function mapDispatchToProps (dispatch) { | ||||||
|  |   return bindActionCreators({ newError }, dispatch); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default connect( | ||||||
|  |   mapStateToProps, | ||||||
|  |   mapDispatchToProps | ||||||
|  | )(DeleteAccount); | ||||||
							
								
								
									
										17
									
								
								js/src/modals/DeleteAccount/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								js/src/modals/DeleteAccount/index.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,17 @@ | |||||||
|  | // Copyright 2015, 2016 Ethcore (UK) Ltd.
 | ||||||
|  | // This file is part of Parity.
 | ||||||
|  | 
 | ||||||
|  | // Parity is free software: you can redistribute it and/or modify
 | ||||||
|  | // it under the terms of the GNU General Public License as published by
 | ||||||
|  | // the Free Software Foundation, either version 3 of the License, or
 | ||||||
|  | // (at your option) any later version.
 | ||||||
|  | 
 | ||||||
|  | // Parity 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 General Public License for more details.
 | ||||||
|  | 
 | ||||||
|  | // You should have received a copy of the GNU General Public License
 | ||||||
|  | // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  | 
 | ||||||
|  | export default from './deleteAccount'; | ||||||
| @ -17,6 +17,7 @@ | |||||||
| import AddAddress from './AddAddress'; | import AddAddress from './AddAddress'; | ||||||
| import AddContract from './AddContract'; | import AddContract from './AddContract'; | ||||||
| import CreateAccount from './CreateAccount'; | import CreateAccount from './CreateAccount'; | ||||||
|  | import DeleteAccount from './DeleteAccount'; | ||||||
| import DeployContract from './DeployContract'; | import DeployContract from './DeployContract'; | ||||||
| import EditMeta from './EditMeta'; | import EditMeta from './EditMeta'; | ||||||
| import ExecuteContract from './ExecuteContract'; | import ExecuteContract from './ExecuteContract'; | ||||||
| @ -32,6 +33,7 @@ export { | |||||||
|   AddAddress, |   AddAddress, | ||||||
|   AddContract, |   AddContract, | ||||||
|   CreateAccount, |   CreateAccount, | ||||||
|  |   DeleteAccount, | ||||||
|   DeployContract, |   DeployContract, | ||||||
|   EditMeta, |   EditMeta, | ||||||
|   ExecuteContract, |   ExecuteContract, | ||||||
|  | |||||||
| @ -17,12 +17,13 @@ | |||||||
| import React, { Component, PropTypes } from 'react'; | import React, { Component, PropTypes } from 'react'; | ||||||
| import { connect } from 'react-redux'; | import { connect } from 'react-redux'; | ||||||
| import { bindActionCreators } from 'redux'; | import { bindActionCreators } from 'redux'; | ||||||
|  | import ActionDelete from 'material-ui/svg-icons/action/delete'; | ||||||
| import ContentCreate from 'material-ui/svg-icons/content/create'; | import ContentCreate from 'material-ui/svg-icons/content/create'; | ||||||
| import ContentSend from 'material-ui/svg-icons/content/send'; | import ContentSend from 'material-ui/svg-icons/content/send'; | ||||||
| import LockIcon from 'material-ui/svg-icons/action/lock'; | import LockIcon from 'material-ui/svg-icons/action/lock'; | ||||||
| import VerifyIcon from 'material-ui/svg-icons/action/verified-user'; | import VerifyIcon from 'material-ui/svg-icons/action/verified-user'; | ||||||
| 
 | 
 | ||||||
| import { EditMeta, Shapeshift, SMSVerification, Transfer, PasswordManager } from '../../modals'; | import { EditMeta, DeleteAccount, Shapeshift, SMSVerification, Transfer, PasswordManager } from '../../modals'; | ||||||
| import { Actionbar, Button, Page } from '../../ui'; | import { Actionbar, Button, Page } from '../../ui'; | ||||||
| 
 | 
 | ||||||
| import shapeshiftBtn from '../../../assets/images/shapeshift-btn.png'; | import shapeshiftBtn from '../../../assets/images/shapeshift-btn.png'; | ||||||
| @ -50,6 +51,7 @@ class Account extends Component { | |||||||
|   propName = null |   propName = null | ||||||
| 
 | 
 | ||||||
|   state = { |   state = { | ||||||
|  |     showDeleteDialog: false, | ||||||
|     showEditDialog: false, |     showEditDialog: false, | ||||||
|     showFundDialog: false, |     showFundDialog: false, | ||||||
|     showVerificationDialog: false, |     showVerificationDialog: false, | ||||||
| @ -62,8 +64,8 @@ class Account extends Component { | |||||||
|     const { api } = this.context; |     const { api } = this.context; | ||||||
|     const { address } = this.props.params; |     const { address } = this.props.params; | ||||||
| 
 | 
 | ||||||
|     const store = new VerificationStore(api, address); |     const verificationStore = new VerificationStore(api, address); | ||||||
|     this.setState({ verificationStore: store }); |     this.setState({ verificationStore }); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   render () { |   render () { | ||||||
| @ -79,6 +81,7 @@ class Account extends Component { | |||||||
| 
 | 
 | ||||||
|     return ( |     return ( | ||||||
|       <div className={ styles.account }> |       <div className={ styles.account }> | ||||||
|  |         { this.renderDeleteDialog(account) } | ||||||
|         { this.renderEditDialog(account) } |         { this.renderEditDialog(account) } | ||||||
|         { this.renderFundDialog() } |         { this.renderFundDialog() } | ||||||
|         { this.renderVerificationDialog() } |         { this.renderVerificationDialog() } | ||||||
| @ -131,7 +134,12 @@ class Account extends Component { | |||||||
|         key='passwordManager' |         key='passwordManager' | ||||||
|         icon={ <LockIcon /> } |         icon={ <LockIcon /> } | ||||||
|         label='password' |         label='password' | ||||||
|         onClick={ this.onPasswordClick } /> |         onClick={ this.onPasswordClick } />, | ||||||
|  |       <Button | ||||||
|  |         key='delete' | ||||||
|  |         icon={ <ActionDelete /> } | ||||||
|  |         label='delete account' | ||||||
|  |         onClick={ this.onDeleteClick } /> | ||||||
|     ]; |     ]; | ||||||
| 
 | 
 | ||||||
|     return ( |     return ( | ||||||
| @ -141,6 +149,20 @@ class Account extends Component { | |||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   renderDeleteDialog (account) { | ||||||
|  |     const { showDeleteDialog } = this.state; | ||||||
|  | 
 | ||||||
|  |     if (!showDeleteDialog) { | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return ( | ||||||
|  |       <DeleteAccount | ||||||
|  |         account={ account } | ||||||
|  |         onClose={ this.onDeleteClose } /> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   renderEditDialog (account) { |   renderEditDialog (account) { | ||||||
|     const { showEditDialog } = this.state; |     const { showEditDialog } = this.state; | ||||||
| 
 | 
 | ||||||
| @ -228,6 +250,14 @@ class Account extends Component { | |||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   onDeleteClick = () => { | ||||||
|  |     this.setState({ showDeleteDialog: true }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   onDeleteClose = () => { | ||||||
|  |     this.setState({ showDeleteDialog: false }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   onEditClick = () => { |   onEditClick = () => { | ||||||
|     this.setState({ |     this.setState({ | ||||||
|       showEditDialog: !this.state.showEditDialog |       showEditDialog: !this.state.showEditDialog | ||||||
|  | |||||||
| @ -104,7 +104,7 @@ impl<C: 'static> ParityAccounts for ParityAccountsClient<C> where C: MiningBlock | |||||||
| 		let account: Address = account.into(); | 		let account: Address = account.into(); | ||||||
| 
 | 
 | ||||||
| 		take_weak!(self.accounts) | 		take_weak!(self.accounts) | ||||||
| 			.test_password(&account, password) | 			.test_password(&account, &password) | ||||||
| 			.map_err(|e| errors::account("Could not fetch account info.", e)) | 			.map_err(|e| errors::account("Could not fetch account info.", e)) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -117,6 +117,15 @@ impl<C: 'static> ParityAccounts for ParityAccountsClient<C> where C: MiningBlock | |||||||
| 			.map_err(|e| errors::account("Could not fetch account info.", e)) | 			.map_err(|e| errors::account("Could not fetch account info.", e)) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	fn kill_account(&self, account: RpcH160, password: String) -> Result<bool, Error> { | ||||||
|  | 		try!(self.active()); | ||||||
|  | 		let account: Address = account.into(); | ||||||
|  | 		take_weak!(self.accounts) | ||||||
|  | 			.kill_account(&account, &password) | ||||||
|  | 			.map(|_| true) | ||||||
|  | 			.map_err(|e| errors::account("Could not delete account.", e)) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	fn set_account_name(&self, addr: RpcH160, name: String) -> Result<bool, Error> { | 	fn set_account_name(&self, addr: RpcH160, name: String) -> Result<bool, Error> { | ||||||
| 		try!(self.active()); | 		try!(self.active()); | ||||||
| 		let store = take_weak!(self.accounts); | 		let store = take_weak!(self.accounts); | ||||||
|  | |||||||
| @ -116,3 +116,25 @@ fn should_be_able_to_set_meta() { | |||||||
| 	assert_eq!(res, Some(response)); | 	assert_eq!(res, Some(response)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | #[test] | ||||||
|  | fn should_be_able_to_kill_account() { | ||||||
|  | 	let tester = setup(); | ||||||
|  | 	tester.accounts.new_account("password").unwrap(); | ||||||
|  | 	let accounts = tester.accounts.accounts().unwrap(); | ||||||
|  | 	assert_eq!(accounts.len(), 1); | ||||||
|  | 	let address = accounts[0]; | ||||||
|  | 
 | ||||||
|  | 	let request = format!(r#"{{"jsonrpc": "2.0", "method": "parity_killAccount", "params": ["0xf00baba2f00baba2f00baba2f00baba2f00baba2"], "id": 1}}"#); | ||||||
|  | 	let response = r#"{"jsonrpc":"2.0","error":{"code":-32602,"message":"Invalid params","data":null},"id":1}"#; | ||||||
|  | 	let res = tester.io.handle_request_sync(&request); | ||||||
|  | 	assert_eq!(res, Some(response.into())); | ||||||
|  | 
 | ||||||
|  | 	let request = format!(r#"{{"jsonrpc": "2.0", "method": "parity_killAccount", "params": ["0x{}", "password"], "id": 1}}"#, address.hex()); | ||||||
|  | 	let response = r#"{"jsonrpc":"2.0","result":true,"id":1}"#; | ||||||
|  | 	let res = tester.io.handle_request_sync(&request); | ||||||
|  | 	assert_eq!(res, Some(response.into())); | ||||||
|  | 
 | ||||||
|  | 	let accounts = tester.accounts.accounts().unwrap(); | ||||||
|  | 	assert_eq!(accounts.len(), 0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | |||||||
| @ -53,6 +53,11 @@ build_rpc_trait! { | |||||||
| 		#[rpc(name = "parity_changePassword")] | 		#[rpc(name = "parity_changePassword")] | ||||||
| 		fn change_password(&self, H160, String, String) -> Result<bool, Error>; | 		fn change_password(&self, H160, String, String) -> Result<bool, Error>; | ||||||
| 
 | 
 | ||||||
|  | 		/// Permanently deletes an account.
 | ||||||
|  | 		/// Arguments: `account`, `password`.
 | ||||||
|  | 		#[rpc(name = "parity_killAccount")] | ||||||
|  | 		fn kill_account(&self, H160, String) -> Result<bool, Error>; | ||||||
|  | 
 | ||||||
| 		/// Set an account's name.
 | 		/// Set an account's name.
 | ||||||
| 		#[rpc(name = "parity_setAccountName")] | 		#[rpc(name = "parity_setAccountName")] | ||||||
| 		fn set_account_name(&self, H160, String) -> Result<bool, Error>; | 		fn set_account_name(&self, H160, String) -> Result<bool, Error>; | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user