2017-01-25 18:51:41 +01:00
|
|
|
// Copyright 2015-2017 Parity Technologies (UK) Ltd.
|
2016-06-20 10:06:49 +02:00
|
|
|
// 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/>.
|
|
|
|
|
2016-08-03 17:58:22 +02:00
|
|
|
use std::collections::HashSet;
|
2017-05-19 17:14:47 +02:00
|
|
|
use std::path::Path;
|
|
|
|
use std::fs;
|
|
|
|
|
2016-06-20 00:10:34 +02:00
|
|
|
use ethkey::Address;
|
2017-05-19 17:14:47 +02:00
|
|
|
use dir::{paths, KeyDirectory, RootDiskDirectory, DiskKeyFileManager, KeyFileManager};
|
2016-06-20 00:10:34 +02:00
|
|
|
use Error;
|
|
|
|
|
2017-05-19 17:14:47 +02:00
|
|
|
/// Import an account from a file.
|
|
|
|
pub fn import_account(path: &Path, dst: &KeyDirectory) -> Result<Address, Error> {
|
|
|
|
let key_manager = DiskKeyFileManager;
|
|
|
|
let existing_accounts = dst.load()?.into_iter().map(|a| a.address).collect::<HashSet<_>>();
|
|
|
|
let filename = path.file_name().and_then(|n| n.to_str()).map(|f| f.to_owned());
|
|
|
|
let account = fs::File::open(&path)
|
|
|
|
.map_err(Into::into)
|
|
|
|
.and_then(|file| key_manager.read(filename, file))?;
|
|
|
|
|
|
|
|
let address = account.address.clone();
|
|
|
|
if !existing_accounts.contains(&address) {
|
|
|
|
dst.insert(account)?;
|
|
|
|
}
|
|
|
|
Ok(address)
|
|
|
|
}
|
|
|
|
|
2017-03-23 13:23:03 +01:00
|
|
|
/// Import all accounts from one directory to the other.
|
2016-06-20 00:10:34 +02:00
|
|
|
pub fn import_accounts(src: &KeyDirectory, dst: &KeyDirectory) -> Result<Vec<Address>, Error> {
|
2016-12-27 12:53:56 +01:00
|
|
|
let accounts = src.load()?;
|
|
|
|
let existing_accounts = dst.load()?.into_iter().map(|a| a.address).collect::<HashSet<_>>();
|
2016-08-03 17:58:22 +02:00
|
|
|
|
|
|
|
accounts.into_iter()
|
|
|
|
.filter(|a| !existing_accounts.contains(&a.address))
|
|
|
|
.map(|a| {
|
|
|
|
let address = a.address.clone();
|
2016-12-27 12:53:56 +01:00
|
|
|
dst.insert(a)?;
|
2016-08-03 17:58:22 +02:00
|
|
|
Ok(address)
|
|
|
|
}).collect()
|
2016-06-20 00:10:34 +02:00
|
|
|
}
|
2016-08-11 18:31:28 +02:00
|
|
|
|
|
|
|
/// Provide a `HashSet` of all accounts available for import from the Geth keystore.
|
|
|
|
pub fn read_geth_accounts(testnet: bool) -> Vec<Address> {
|
2017-03-23 13:23:03 +01:00
|
|
|
RootDiskDirectory::at(paths::geth(testnet))
|
2016-08-11 18:31:28 +02:00
|
|
|
.load()
|
|
|
|
.map(|d| d.into_iter().map(|a| a.address).collect())
|
|
|
|
.unwrap_or_else(|_| Vec::new())
|
|
|
|
}
|
|
|
|
|
2017-03-23 13:23:03 +01:00
|
|
|
/// Import specific `desired` accounts from the Geth keystore into `dst`.
|
2016-08-11 18:31:28 +02:00
|
|
|
pub fn import_geth_accounts(dst: &KeyDirectory, desired: HashSet<Address>, testnet: bool) -> Result<Vec<Address>, Error> {
|
2017-03-23 13:23:03 +01:00
|
|
|
let src = RootDiskDirectory::at(paths::geth(testnet));
|
2016-12-27 12:53:56 +01:00
|
|
|
let accounts = src.load()?;
|
|
|
|
let existing_accounts = dst.load()?.into_iter().map(|a| a.address).collect::<HashSet<_>>();
|
2016-08-11 18:31:28 +02:00
|
|
|
|
|
|
|
accounts.into_iter()
|
|
|
|
.filter(|a| !existing_accounts.contains(&a.address))
|
|
|
|
.filter(|a| desired.contains(&a.address))
|
|
|
|
.map(|a| {
|
|
|
|
let address = a.address.clone();
|
2016-12-27 12:53:56 +01:00
|
|
|
dst.insert(a)?;
|
2016-08-11 18:31:28 +02:00
|
|
|
Ok(address)
|
|
|
|
}).collect()
|
|
|
|
}
|