2020-09-22 14:53:52 +02:00
|
|
|
// Copyright 2015-2020 Parity Technologies (UK) Ltd.
|
|
|
|
// This file is part of OpenEthereum.
|
2016-07-25 16:09:47 +02:00
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
// OpenEthereum is free software: you can redistribute it and/or modify
|
2016-07-25 16:09:47 +02:00
|
|
|
// 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.
|
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
// OpenEthereum is distributed in the hope that it will be useful,
|
2016-07-25 16:09:47 +02:00
|
|
|
// 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
|
2020-09-22 14:53:52 +02:00
|
|
|
// along with OpenEthereum. If not, see <http://www.gnu.org/licenses/>.
|
2016-07-25 16:09:47 +02:00
|
|
|
|
2017-12-26 07:55:39 +01:00
|
|
|
#![warn(missing_docs)]
|
|
|
|
//! Dir utilities for platform-specific operations
|
2020-09-22 14:53:52 +02:00
|
|
|
///
|
|
|
|
/// Base ($BASE) Paths where we store our data and cache corresponds to:
|
|
|
|
/// Windows:
|
|
|
|
/// UserData: FOLDERID_RoamingAppData: %APPDATA% (%USERPROFILE%\AppData\Roaming)
|
|
|
|
/// UserCache: FOLDERID_LocalAppData: %LOCALAPPDATA% (%USERPROFILE%\AppData\Local)
|
|
|
|
/// MacOS:
|
|
|
|
/// UserData: /Users/Alice/Library/Application Support/
|
|
|
|
/// UserCache: /Users/Alice/Library/Caches/
|
|
|
|
/// Unix:
|
|
|
|
/// UserData is: $HOME/.local/share/
|
|
|
|
/// UserCache is: $HOME/.cache/
|
|
|
|
///
|
|
|
|
/// On this UserData base path we are adding additional application folders:
|
|
|
|
/// If older parity folders are present we will use them as default for backward compatibility:
|
|
|
|
/// Windows: $BASE/Parity/Ethereum/
|
|
|
|
/// Unix/MacOS: $BASE/io.parity.ethereum/
|
|
|
|
///
|
|
|
|
/// For OpenEthereum paths we are using:
|
|
|
|
/// Wndows/MacOS: $BASE/OpenEthereum/
|
|
|
|
/// Unix: $BASE/openethereum/
|
|
|
|
///
|
2017-12-24 09:34:43 +01:00
|
|
|
extern crate app_dirs;
|
2018-01-10 13:35:18 +01:00
|
|
|
extern crate ethereum_types;
|
2018-08-21 15:30:24 +02:00
|
|
|
extern crate home;
|
2020-08-05 06:08:03 +02:00
|
|
|
extern crate journaldb;
|
2017-12-24 09:34:43 +01:00
|
|
|
|
|
|
|
pub mod helpers;
|
2020-09-22 14:53:52 +02:00
|
|
|
use app_dirs::{data_root, get_app_root, AppDataType, AppInfo};
|
2020-08-05 06:08:03 +02:00
|
|
|
use ethereum_types::{H256, H64};
|
2017-07-10 12:57:40 +02:00
|
|
|
use helpers::{replace_home, replace_home_and_local};
|
2020-08-05 06:08:03 +02:00
|
|
|
use journaldb::Algorithm;
|
|
|
|
use std::{
|
|
|
|
fs,
|
|
|
|
path::{Path, PathBuf},
|
|
|
|
};
|
2017-12-27 15:17:39 +01:00
|
|
|
// re-export platform-specific functions
|
|
|
|
use platform::*;
|
2016-07-25 16:09:47 +02:00
|
|
|
|
2018-08-21 15:30:24 +02:00
|
|
|
pub use home::home_dir;
|
|
|
|
|
2018-07-11 16:17:35 +02:00
|
|
|
/// Platform-specific chains path for standard client - Windows only
|
2020-08-05 06:08:03 +02:00
|
|
|
#[cfg(target_os = "windows")]
|
|
|
|
pub const CHAINS_PATH: &str = "$LOCAL/chains";
|
2018-07-11 16:17:35 +02:00
|
|
|
/// Platform-specific chains path for standard client
|
2020-08-05 06:08:03 +02:00
|
|
|
#[cfg(not(target_os = "windows"))]
|
|
|
|
pub const CHAINS_PATH: &str = "$BASE/chains";
|
2017-01-05 14:12:54 +01:00
|
|
|
|
2017-12-26 07:55:39 +01:00
|
|
|
/// Platform-specific cache path - Windows only
|
2020-08-05 06:08:03 +02:00
|
|
|
#[cfg(target_os = "windows")]
|
|
|
|
pub const CACHE_PATH: &str = "$LOCAL/cache";
|
2017-12-26 07:55:39 +01:00
|
|
|
/// Platform-specific cache path
|
2020-08-05 06:08:03 +02:00
|
|
|
#[cfg(not(target_os = "windows"))]
|
|
|
|
pub const CACHE_PATH: &str = "$BASE/cache";
|
2017-07-10 12:57:40 +02:00
|
|
|
|
2016-07-25 16:09:47 +02:00
|
|
|
// this const is irrelevent cause we do have migrations now,
|
|
|
|
// but we still use it for backwards compatibility
|
2018-07-09 16:48:33 +02:00
|
|
|
const LEGACY_CLIENT_DB_VER_STR: &str = "5.3";
|
2016-07-25 16:09:47 +02:00
|
|
|
|
|
|
|
#[derive(Debug, PartialEq)]
|
2017-12-26 07:55:39 +01:00
|
|
|
/// Parity local data directories
|
2016-07-25 16:09:47 +02:00
|
|
|
pub struct Directories {
|
2020-08-05 06:08:03 +02:00
|
|
|
/// Base dir
|
|
|
|
pub base: String,
|
|
|
|
/// Database dir
|
|
|
|
pub db: String,
|
|
|
|
/// Cache dir
|
|
|
|
pub cache: String,
|
|
|
|
/// Dir to store keys
|
|
|
|
pub keys: String,
|
|
|
|
/// Signer dir
|
|
|
|
pub signer: String,
|
|
|
|
/// Secrets dir
|
|
|
|
pub secretstore: String,
|
2016-07-25 16:09:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Directories {
|
2020-08-05 06:08:03 +02:00
|
|
|
fn default() -> Self {
|
|
|
|
let data_dir = default_data_path();
|
|
|
|
let local_dir = default_local_path();
|
|
|
|
Directories {
|
|
|
|
base: replace_home(&data_dir, "$BASE"),
|
|
|
|
db: replace_home_and_local(&data_dir, &local_dir, CHAINS_PATH),
|
|
|
|
cache: replace_home_and_local(&data_dir, &local_dir, CACHE_PATH),
|
|
|
|
keys: replace_home(&data_dir, "$BASE/keys"),
|
|
|
|
signer: replace_home(&data_dir, "$BASE/signer"),
|
|
|
|
secretstore: replace_home(&data_dir, "$BASE/secretstore"),
|
|
|
|
}
|
|
|
|
}
|
2016-07-25 16:09:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Directories {
|
2020-08-05 06:08:03 +02:00
|
|
|
/// Create local directories
|
|
|
|
pub fn create_dirs(
|
|
|
|
&self,
|
|
|
|
signer_enabled: bool,
|
|
|
|
secretstore_enabled: bool,
|
|
|
|
) -> Result<(), String> {
|
|
|
|
fs::create_dir_all(&self.base).map_err(|e| e.to_string())?;
|
|
|
|
fs::create_dir_all(&self.db).map_err(|e| e.to_string())?;
|
|
|
|
fs::create_dir_all(&self.cache).map_err(|e| e.to_string())?;
|
|
|
|
fs::create_dir_all(&self.keys).map_err(|e| e.to_string())?;
|
|
|
|
if signer_enabled {
|
|
|
|
fs::create_dir_all(&self.signer).map_err(|e| e.to_string())?;
|
|
|
|
}
|
|
|
|
if secretstore_enabled {
|
|
|
|
fs::create_dir_all(&self.secretstore).map_err(|e| e.to_string())?;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Database paths.
|
|
|
|
pub fn database(
|
|
|
|
&self,
|
|
|
|
genesis_hash: H256,
|
|
|
|
fork_name: Option<String>,
|
|
|
|
spec_name: String,
|
|
|
|
) -> DatabaseDirectories {
|
|
|
|
DatabaseDirectories {
|
|
|
|
path: self.db.clone(),
|
|
|
|
legacy_path: self.base.clone(),
|
|
|
|
genesis_hash,
|
|
|
|
fork_name,
|
|
|
|
spec_name,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the ipc sockets path
|
|
|
|
pub fn ipc_path(&self) -> PathBuf {
|
|
|
|
let mut dir = Path::new(&self.base).to_path_buf();
|
|
|
|
dir.push("ipc");
|
|
|
|
dir
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Legacy keys path
|
|
|
|
// TODO: remove in 1.7
|
|
|
|
pub fn legacy_keys_path(&self, testnet: bool) -> PathBuf {
|
|
|
|
let mut dir = Path::new(&self.base).to_path_buf();
|
|
|
|
if testnet {
|
|
|
|
dir.push("testnet_keys");
|
|
|
|
} else {
|
|
|
|
dir.push("keys");
|
|
|
|
}
|
|
|
|
dir
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the keys path
|
|
|
|
pub fn keys_path(&self, data_dir: &str) -> PathBuf {
|
|
|
|
let mut dir = PathBuf::from(&self.keys);
|
|
|
|
dir.push(data_dir);
|
|
|
|
dir
|
|
|
|
}
|
2016-09-26 19:21:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, PartialEq)]
|
2017-12-26 07:55:39 +01:00
|
|
|
/// Database directories for the given fork.
|
2016-09-26 19:21:25 +02:00
|
|
|
pub struct DatabaseDirectories {
|
2020-08-05 06:08:03 +02:00
|
|
|
/// Base path
|
|
|
|
pub path: String,
|
|
|
|
/// Legacy path
|
|
|
|
pub legacy_path: String,
|
|
|
|
/// Genesis hash
|
|
|
|
pub genesis_hash: H256,
|
|
|
|
/// Name of current fork
|
|
|
|
pub fork_name: Option<String>,
|
|
|
|
/// Name of current spec
|
|
|
|
pub spec_name: String,
|
2016-09-26 19:21:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl DatabaseDirectories {
|
2020-08-05 06:08:03 +02:00
|
|
|
/// Base DB directory for the given fork.
|
|
|
|
// TODO: remove in 1.7
|
|
|
|
pub fn legacy_fork_path(&self) -> PathBuf {
|
|
|
|
Path::new(&self.legacy_path).join(format!(
|
|
|
|
"{:x}{}",
|
|
|
|
H64::from(self.genesis_hash),
|
|
|
|
self.fork_name
|
|
|
|
.as_ref()
|
|
|
|
.map(|f| format!("-{}", f))
|
|
|
|
.unwrap_or_default()
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Spec root directory for the given fork.
|
|
|
|
pub fn spec_root_path(&self) -> PathBuf {
|
|
|
|
Path::new(&self.path).join(&self.spec_name)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Generic client path
|
|
|
|
pub fn client_path(&self, pruning: Algorithm) -> PathBuf {
|
|
|
|
self.db_root_path()
|
|
|
|
.join(pruning.as_internal_name_str())
|
|
|
|
.join("db")
|
|
|
|
}
|
|
|
|
|
|
|
|
/// DB root path, named after genesis hash
|
|
|
|
pub fn db_root_path(&self) -> PathBuf {
|
|
|
|
self.spec_root_path()
|
|
|
|
.join("db")
|
|
|
|
.join(format!("{:x}", H64::from(self.genesis_hash)))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// DB path
|
|
|
|
pub fn db_path(&self, pruning: Algorithm) -> PathBuf {
|
|
|
|
self.db_root_path().join(pruning.as_internal_name_str())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the root path for database
|
|
|
|
// TODO: remove in 1.7
|
|
|
|
pub fn legacy_version_path(&self, pruning: Algorithm) -> PathBuf {
|
|
|
|
self.legacy_fork_path().join(format!(
|
|
|
|
"v{}-sec-{}",
|
|
|
|
LEGACY_CLIENT_DB_VER_STR,
|
|
|
|
pruning.as_internal_name_str()
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get user defaults path, legacy way
|
|
|
|
// TODO: remove in 1.7
|
|
|
|
pub fn legacy_user_defaults_path(&self) -> PathBuf {
|
|
|
|
self.legacy_fork_path().join("user_defaults")
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get snapshot path, legacy way
|
|
|
|
// TODO: remove in 1.7
|
|
|
|
pub fn legacy_snapshot_path(&self) -> PathBuf {
|
|
|
|
self.legacy_fork_path().join("snapshot")
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get user defaults path, legacy way
|
|
|
|
// TODO: remove in 1.7
|
|
|
|
pub fn legacy_network_path(&self) -> PathBuf {
|
|
|
|
self.legacy_fork_path().join("network")
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get user defauls path
|
|
|
|
pub fn user_defaults_path(&self) -> PathBuf {
|
|
|
|
self.spec_root_path().join("user_defaults")
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the path for the snapshot directory given the genesis hash and fork name.
|
|
|
|
pub fn snapshot_path(&self) -> PathBuf {
|
|
|
|
self.db_root_path().join("snapshot")
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the path for the network directory.
|
|
|
|
pub fn network_path(&self) -> PathBuf {
|
|
|
|
self.spec_root_path().join("network")
|
|
|
|
}
|
2016-07-25 16:09:47 +02:00
|
|
|
}
|
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
fn default_path(t: AppDataType) -> Option<PathBuf> {
|
2020-08-05 06:08:03 +02:00
|
|
|
let app_info = AppInfo {
|
2020-09-22 14:53:52 +02:00
|
|
|
name: PARITY_PRODUCT,
|
|
|
|
author: PARITY_AUTHOR,
|
2020-08-05 06:08:03 +02:00
|
|
|
};
|
2020-09-22 14:53:52 +02:00
|
|
|
let old_root = get_app_root(t, &app_info).ok()?;
|
|
|
|
if old_root.exists() {
|
|
|
|
return Some(old_root);
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut root = data_root(t).ok()?;
|
|
|
|
root.push(if LOWERCASE {
|
|
|
|
"openethereum"
|
|
|
|
} else {
|
|
|
|
"OpenEthereum"
|
|
|
|
});
|
|
|
|
Some(root)
|
2016-12-13 23:38:29 +01:00
|
|
|
}
|
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
fn fallback_path() -> PathBuf {
|
|
|
|
let mut p = PathBuf::new();
|
|
|
|
p.push("$HOME");
|
|
|
|
p.push(".openethereum");
|
|
|
|
p
|
2017-01-05 14:12:54 +01:00
|
|
|
}
|
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
/// Default data path
|
|
|
|
pub fn default_data_pathbuf() -> PathBuf {
|
|
|
|
default_path(AppDataType::UserData).unwrap_or_else(fallback_path)
|
2017-12-24 09:34:43 +01:00
|
|
|
}
|
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
/// Default data path
|
|
|
|
pub fn default_data_path() -> String {
|
|
|
|
default_data_pathbuf().to_string_lossy().into_owned()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Default local path
|
|
|
|
pub fn default_local_path() -> String {
|
|
|
|
default_path(AppDataType::UserCache)
|
|
|
|
.unwrap_or_else(fallback_path)
|
|
|
|
.to_string_lossy()
|
|
|
|
.into_owned()
|
2017-12-24 09:34:43 +01:00
|
|
|
}
|
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
/// these variables are used only for backward compatibility .
|
|
|
|
/// In case that there is folder from older parity version we will use it as default path,
|
|
|
|
/// in case there is not we will create openethereum folder.Algorithm
|
2017-12-27 15:17:39 +01:00
|
|
|
#[cfg(target_os = "macos")]
|
2017-12-26 07:25:19 +01:00
|
|
|
mod platform {
|
2020-09-22 14:53:52 +02:00
|
|
|
pub const LOWERCASE: bool = false;
|
|
|
|
pub const PARITY_AUTHOR: &str = "Parity";
|
|
|
|
pub const PARITY_PRODUCT: &str = "io.parity.ethereum";
|
2017-12-27 15:17:39 +01:00
|
|
|
}
|
|
|
|
#[cfg(windows)]
|
|
|
|
mod platform {
|
2020-09-22 14:53:52 +02:00
|
|
|
pub const LOWERCASE: bool = false;
|
|
|
|
pub const PARITY_AUTHOR: &str = "Parity";
|
|
|
|
pub const PARITY_PRODUCT: &str = "Ethereum";
|
2017-12-27 15:17:39 +01:00
|
|
|
}
|
|
|
|
#[cfg(not(any(target_os = "macos", windows)))]
|
|
|
|
mod platform {
|
2020-09-22 14:53:52 +02:00
|
|
|
pub const LOWERCASE: bool = true;
|
|
|
|
pub const PARITY_AUTHOR: &str = "parity";
|
|
|
|
pub const PARITY_PRODUCT: &str = "io.parity.ethereum";
|
2017-12-24 09:34:43 +01:00
|
|
|
}
|
2017-12-26 07:25:19 +01:00
|
|
|
|
2016-07-25 16:09:47 +02:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2020-08-05 06:08:03 +02:00
|
|
|
use super::Directories;
|
|
|
|
use helpers::{replace_home, replace_home_and_local};
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_default_directories() {
|
|
|
|
let data_dir = super::default_data_path();
|
|
|
|
let local_dir = super::default_local_path();
|
|
|
|
let expected = Directories {
|
|
|
|
base: replace_home(&data_dir, "$BASE"),
|
|
|
|
db: replace_home_and_local(
|
|
|
|
&data_dir,
|
|
|
|
&local_dir,
|
|
|
|
if cfg!(target_os = "windows") {
|
|
|
|
"$LOCAL/chains"
|
|
|
|
} else {
|
|
|
|
"$BASE/chains"
|
|
|
|
},
|
|
|
|
),
|
|
|
|
cache: replace_home_and_local(
|
|
|
|
&data_dir,
|
|
|
|
&local_dir,
|
|
|
|
if cfg!(target_os = "windows") {
|
|
|
|
"$LOCAL/cache"
|
|
|
|
} else {
|
|
|
|
"$BASE/cache"
|
|
|
|
},
|
|
|
|
),
|
|
|
|
keys: replace_home(&data_dir, "$BASE/keys"),
|
|
|
|
signer: replace_home(&data_dir, "$BASE/signer"),
|
|
|
|
secretstore: replace_home(&data_dir, "$BASE/secretstore"),
|
|
|
|
};
|
|
|
|
assert_eq!(expected, Directories::default());
|
|
|
|
}
|
2016-07-25 16:09:47 +02:00
|
|
|
}
|