2016-04-10 15:12:20 +02:00
|
|
|
// 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/>.
|
|
|
|
|
|
|
|
//! Parity upgrade logic
|
|
|
|
|
|
|
|
use semver::Version;
|
|
|
|
use std::collections::*;
|
2016-04-14 03:02:16 +02:00
|
|
|
use std::fs::{File, create_dir_all};
|
2016-04-10 15:42:33 +02:00
|
|
|
use std::env;
|
|
|
|
use std::io::{Read, Write};
|
2016-04-10 15:12:20 +02:00
|
|
|
|
2016-04-12 13:54:34 +02:00
|
|
|
#[cfg_attr(feature="dev", allow(enum_variant_names))]
|
2016-04-10 16:15:40 +02:00
|
|
|
#[derive(Debug)]
|
2016-04-10 15:42:33 +02:00
|
|
|
pub enum Error {
|
2016-04-14 03:02:16 +02:00
|
|
|
CannotCreateConfigPath,
|
|
|
|
CannotWriteVersionFile,
|
2016-04-10 15:42:33 +02:00
|
|
|
CannotUpdateVersionFile,
|
|
|
|
}
|
2016-04-10 15:12:20 +02:00
|
|
|
|
|
|
|
const CURRENT_VERSION: &'static str = env!("CARGO_PKG_VERSION");
|
|
|
|
|
|
|
|
#[derive(Hash, PartialEq, Eq)]
|
|
|
|
struct UpgradeKey {
|
|
|
|
pub old_version: Version,
|
|
|
|
pub new_version: Version,
|
|
|
|
}
|
|
|
|
|
|
|
|
type UpgradeList = HashMap<UpgradeKey, fn() -> Result<(), Error>>;
|
|
|
|
|
|
|
|
impl UpgradeKey {
|
|
|
|
// given the following config exist
|
|
|
|
// ver.lock 1.1 (`previous_version`)
|
|
|
|
//
|
|
|
|
// current_version 1.4 (`current_version`)
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//upgrades (set of `UpgradeKey`)
|
|
|
|
// 1.0 -> 1.1 (u1)
|
|
|
|
// 1.1 -> 1.2 (u2)
|
|
|
|
// 1.2 -> 1.3 (u3)
|
|
|
|
// 1.3 -> 1.4 (u4)
|
|
|
|
// 1.4 -> 1.5 (u5)
|
|
|
|
//
|
|
|
|
// then the following upgrades should be applied:
|
|
|
|
// u2, u3, u4
|
|
|
|
fn is_applicable(&self, previous_version: &Version, current_version: &Version) -> bool {
|
|
|
|
self.old_version >= *previous_version && self.new_version <= *current_version
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// dummy upgrade (remove when the first one is in)
|
|
|
|
fn dummy_upgrade() -> Result<(), Error> {
|
2016-04-10 16:15:40 +02:00
|
|
|
println!("Adding ver.lock");
|
2016-04-10 15:12:20 +02:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2016-04-14 03:02:16 +02:00
|
|
|
fn push_upgrades(upgrades: &mut UpgradeList)
|
2016-04-10 15:12:20 +02:00
|
|
|
{
|
|
|
|
// dummy upgrade (remove when the first one is in)
|
|
|
|
upgrades.insert(
|
2016-04-10 16:15:40 +02:00
|
|
|
UpgradeKey { old_version: Version::parse("0.9.0").unwrap(), new_version: Version::parse("1.0.0").unwrap() },
|
2016-04-10 15:12:20 +02:00
|
|
|
dummy_upgrade);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn upgrade_from_version(previous_version: &Version) -> Result<usize, Error> {
|
|
|
|
let mut upgrades = HashMap::new();
|
2016-04-14 03:02:16 +02:00
|
|
|
push_upgrades(&mut upgrades);
|
2016-04-10 15:12:20 +02:00
|
|
|
|
|
|
|
let current_version = Version::parse(CURRENT_VERSION).unwrap();
|
|
|
|
|
|
|
|
let mut count = 0;
|
|
|
|
for upgrade_key in upgrades.keys() {
|
|
|
|
if upgrade_key.is_applicable(previous_version, ¤t_version) {
|
|
|
|
let upgrade_script = upgrades[upgrade_key];
|
|
|
|
try!(upgrade_script());
|
|
|
|
count = count + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(count)
|
|
|
|
}
|
2016-04-10 15:42:33 +02:00
|
|
|
|
2016-04-15 01:49:42 +02:00
|
|
|
fn with_locked_version<F>(db_path: Option<&str>, script: F) -> Result<usize, Error>
|
2016-04-10 15:42:33 +02:00
|
|
|
where F: Fn(&Version) -> Result<usize, Error>
|
|
|
|
{
|
2016-04-15 01:49:42 +02:00
|
|
|
let mut path = db_path.map_or({
|
|
|
|
let mut path = env::home_dir().expect("Applications should have a home dir");
|
|
|
|
path.push(".parity");
|
|
|
|
path
|
|
|
|
}, |s| ::std::path::PathBuf::from(s));
|
2016-04-14 03:02:16 +02:00
|
|
|
try!(create_dir_all(&path).map_err(|_| Error::CannotCreateConfigPath));
|
2016-04-10 15:42:33 +02:00
|
|
|
path.push("ver.lock");
|
|
|
|
|
2016-04-12 05:19:15 +02:00
|
|
|
let version =
|
|
|
|
File::open(&path).ok().and_then(|ref mut file|
|
|
|
|
{
|
2016-04-10 15:42:33 +02:00
|
|
|
let mut version_string = String::new();
|
2016-04-12 05:19:15 +02:00
|
|
|
file.read_to_string(&mut version_string)
|
|
|
|
.ok()
|
|
|
|
.and_then(|_| Version::parse(&version_string).ok())
|
|
|
|
})
|
|
|
|
.unwrap_or_else(|| Version::parse("0.9.0").unwrap());
|
2016-04-10 15:42:33 +02:00
|
|
|
|
2016-04-14 03:02:16 +02:00
|
|
|
let mut lock = try!(File::create(&path).map_err(|_| Error::CannotWriteVersionFile));
|
2016-04-12 13:54:34 +02:00
|
|
|
let result = script(&version);
|
2016-04-10 15:42:33 +02:00
|
|
|
|
2016-04-12 13:54:34 +02:00
|
|
|
let written_version = Version::parse(CURRENT_VERSION).unwrap();
|
|
|
|
try!(lock.write_all(written_version.to_string().as_bytes()).map_err(|_| Error::CannotUpdateVersionFile));
|
|
|
|
result
|
2016-04-10 15:42:33 +02:00
|
|
|
}
|
|
|
|
|
2016-04-15 01:49:42 +02:00
|
|
|
pub fn upgrade(db_path: Option<&str>) -> Result<usize, Error> {
|
|
|
|
with_locked_version(db_path, |ver| {
|
2016-04-10 15:42:33 +02:00
|
|
|
upgrade_from_version(ver)
|
|
|
|
})
|
|
|
|
}
|