2017-01-25 18:51:41 +01:00
|
|
|
// Copyright 2015-2017 Parity Technologies (UK) Ltd.
|
2016-05-24 22:38:11 +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-07-06 12:05:23 +02:00
|
|
|
//! Tests for migrations.
|
|
|
|
//! A random temp directory is created. A database is created within it, and migrations
|
|
|
|
//! are performed in temp sub-directories.
|
2016-05-24 22:38:11 +02:00
|
|
|
|
2016-07-06 12:05:23 +02:00
|
|
|
use common::*;
|
2016-09-26 18:24:58 +02:00
|
|
|
use migration::{Batch, Config, Error, SimpleMigration, Migration, Manager};
|
2016-07-28 23:46:24 +02:00
|
|
|
use kvdb::Database;
|
2016-07-06 12:05:23 +02:00
|
|
|
|
|
|
|
use devtools::RandomTempPath;
|
|
|
|
use std::path::PathBuf;
|
|
|
|
|
|
|
|
fn db_path(path: &Path) -> PathBuf {
|
|
|
|
let mut p = path.to_owned();
|
|
|
|
p.push("db");
|
|
|
|
p
|
|
|
|
}
|
|
|
|
|
|
|
|
// initialize a database at the given directory with the given values.
|
|
|
|
fn make_db(path: &Path, pairs: BTreeMap<Vec<u8>, Vec<u8>>) {
|
|
|
|
let db = Database::open_default(path.to_str().unwrap()).expect("failed to open temp database");
|
|
|
|
{
|
2016-08-25 16:43:56 +02:00
|
|
|
let mut transaction = db.transaction();
|
2016-07-06 12:05:23 +02:00
|
|
|
for (k, v) in pairs {
|
2016-08-18 09:43:56 +02:00
|
|
|
transaction.put(None, &k, &v);
|
2016-07-06 12:05:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
db.write(transaction).expect("failed to write db transaction");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// helper for verifying a migrated database.
|
|
|
|
fn verify_migration(path: &Path, pairs: BTreeMap<Vec<u8>, Vec<u8>>) {
|
|
|
|
let db = Database::open_default(path.to_str().unwrap()).unwrap();
|
|
|
|
|
|
|
|
for (k, v) in pairs {
|
2016-07-28 23:46:24 +02:00
|
|
|
let x = db.get(None, &k).unwrap().unwrap();
|
2016-07-06 12:05:23 +02:00
|
|
|
|
|
|
|
assert_eq!(&x[..], &v[..]);
|
2016-05-24 22:38:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Migration0;
|
|
|
|
|
2016-07-06 12:05:23 +02:00
|
|
|
impl SimpleMigration for Migration0 {
|
2016-07-28 23:46:24 +02:00
|
|
|
fn columns(&self) -> Option<u32> { None }
|
|
|
|
|
|
|
|
fn version(&self) -> u32 { 1 }
|
2016-05-24 22:38:11 +02:00
|
|
|
|
2016-09-26 18:24:58 +02:00
|
|
|
fn simple_migrate(&mut self, mut key: Vec<u8>, mut value: Vec<u8>) -> Option<(Vec<u8>, Vec<u8>)> {
|
2016-05-24 22:38:11 +02:00
|
|
|
key.push(0x11);
|
|
|
|
value.push(0x22);
|
2016-09-26 18:24:58 +02:00
|
|
|
|
2016-05-24 22:38:11 +02:00
|
|
|
Some((key, value))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Migration1;
|
|
|
|
|
2016-07-06 12:05:23 +02:00
|
|
|
impl SimpleMigration for Migration1 {
|
2016-07-28 23:46:24 +02:00
|
|
|
fn columns(&self) -> Option<u32> { None }
|
|
|
|
|
|
|
|
fn version(&self) -> u32 { 2 }
|
2016-05-24 22:38:11 +02:00
|
|
|
|
2016-07-11 09:46:33 +02:00
|
|
|
fn simple_migrate(&mut self, key: Vec<u8>, _value: Vec<u8>) -> Option<(Vec<u8>, Vec<u8>)> {
|
2016-05-24 22:38:11 +02:00
|
|
|
Some((key, vec![]))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-26 18:24:58 +02:00
|
|
|
struct AddsColumn;
|
|
|
|
|
|
|
|
impl Migration for AddsColumn {
|
|
|
|
fn pre_columns(&self) -> Option<u32> { None }
|
|
|
|
|
|
|
|
fn columns(&self) -> Option<u32> { Some(1) }
|
|
|
|
|
|
|
|
fn version(&self) -> u32 { 1 }
|
|
|
|
|
2016-10-01 14:33:19 +02:00
|
|
|
fn migrate(&mut self, source: Arc<Database>, config: &Config, dest: &mut Database, col: Option<u32>) -> Result<(), Error> {
|
2016-09-26 18:24:58 +02:00
|
|
|
let mut batch = Batch::new(config, col);
|
|
|
|
|
|
|
|
for (key, value) in source.iter(col) {
|
2016-12-27 12:53:56 +01:00
|
|
|
batch.insert(key.to_vec(), value.to_vec(), dest)?;
|
2016-09-26 18:24:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if col == Some(1) {
|
2016-12-27 12:53:56 +01:00
|
|
|
batch.insert(vec![1, 2, 3], vec![4, 5, 6], dest)?;
|
2016-09-26 18:24:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
batch.commit(dest)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-24 22:38:11 +02:00
|
|
|
#[test]
|
|
|
|
fn one_simple_migration() {
|
2016-07-06 12:05:23 +02:00
|
|
|
let dir = RandomTempPath::create_dir();
|
|
|
|
let db_path = db_path(dir.as_path());
|
2016-05-24 22:38:11 +02:00
|
|
|
let mut manager = Manager::new(Config::default());
|
2016-07-06 12:05:23 +02:00
|
|
|
make_db(&db_path, map![vec![] => vec![], vec![1] => vec![1]]);
|
|
|
|
let expected = map![vec![0x11] => vec![0x22], vec![1, 0x11] => vec![1, 0x22]];
|
2016-05-24 22:38:11 +02:00
|
|
|
|
|
|
|
manager.add_migration(Migration0).unwrap();
|
2016-07-06 12:05:23 +02:00
|
|
|
let end_path = manager.execute(&db_path, 0).unwrap();
|
|
|
|
|
|
|
|
verify_migration(&end_path, expected);
|
2016-05-24 22:38:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2016-05-26 18:24:51 +02:00
|
|
|
#[should_panic]
|
2016-05-24 22:38:11 +02:00
|
|
|
fn no_migration_needed() {
|
2016-07-06 12:05:23 +02:00
|
|
|
let dir = RandomTempPath::create_dir();
|
|
|
|
let db_path = db_path(dir.as_path());
|
2016-05-24 22:38:11 +02:00
|
|
|
let mut manager = Manager::new(Config::default());
|
2016-07-06 12:05:23 +02:00
|
|
|
make_db(&db_path, map![vec![] => vec![], vec![1] => vec![1]]);
|
2016-05-24 22:38:11 +02:00
|
|
|
|
|
|
|
manager.add_migration(Migration0).unwrap();
|
2016-07-06 12:05:23 +02:00
|
|
|
manager.execute(&db_path, 1).unwrap();
|
2016-05-24 22:38:11 +02:00
|
|
|
}
|
|
|
|
|
2016-07-28 12:05:41 +02:00
|
|
|
#[test]
|
|
|
|
#[should_panic]
|
|
|
|
fn wrong_adding_order() {
|
|
|
|
let dir = RandomTempPath::create_dir();
|
|
|
|
let db_path = db_path(dir.as_path());
|
|
|
|
let mut manager = Manager::new(Config::default());
|
|
|
|
make_db(&db_path, map![vec![] => vec![], vec![1] => vec![1]]);
|
|
|
|
|
|
|
|
manager.add_migration(Migration1).unwrap();
|
|
|
|
manager.add_migration(Migration0).unwrap();
|
|
|
|
}
|
|
|
|
|
2016-05-24 22:38:11 +02:00
|
|
|
#[test]
|
|
|
|
fn multiple_migrations() {
|
2016-07-06 12:05:23 +02:00
|
|
|
let dir = RandomTempPath::create_dir();
|
|
|
|
let db_path = db_path(dir.as_path());
|
2016-05-24 22:38:11 +02:00
|
|
|
let mut manager = Manager::new(Config::default());
|
2016-07-06 12:05:23 +02:00
|
|
|
make_db(&db_path, map![vec![] => vec![], vec![1] => vec![1]]);
|
|
|
|
let expected = map![vec![0x11] => vec![], vec![1, 0x11] => vec![]];
|
2016-05-24 22:38:11 +02:00
|
|
|
|
|
|
|
manager.add_migration(Migration0).unwrap();
|
|
|
|
manager.add_migration(Migration1).unwrap();
|
2016-07-06 12:05:23 +02:00
|
|
|
let end_path = manager.execute(&db_path, 0).unwrap();
|
|
|
|
|
|
|
|
verify_migration(&end_path, expected);
|
2016-05-24 22:38:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn second_migration() {
|
2016-07-06 12:05:23 +02:00
|
|
|
let dir = RandomTempPath::create_dir();
|
|
|
|
let db_path = db_path(dir.as_path());
|
2016-05-24 22:38:11 +02:00
|
|
|
let mut manager = Manager::new(Config::default());
|
2016-07-06 12:05:23 +02:00
|
|
|
make_db(&db_path, map![vec![] => vec![], vec![1] => vec![1]]);
|
|
|
|
let expected = map![vec![] => vec![], vec![1] => vec![]];
|
2016-05-24 22:38:11 +02:00
|
|
|
|
|
|
|
manager.add_migration(Migration0).unwrap();
|
|
|
|
manager.add_migration(Migration1).unwrap();
|
2016-07-06 12:05:23 +02:00
|
|
|
let end_path = manager.execute(&db_path, 1).unwrap();
|
|
|
|
|
|
|
|
verify_migration(&end_path, expected);
|
2016-05-24 22:38:11 +02:00
|
|
|
}
|
2016-05-26 18:24:51 +02:00
|
|
|
|
2016-07-28 12:05:41 +02:00
|
|
|
#[test]
|
|
|
|
fn first_and_noop_migration() {
|
|
|
|
let dir = RandomTempPath::create_dir();
|
|
|
|
let db_path = db_path(dir.as_path());
|
|
|
|
let mut manager = Manager::new(Config::default());
|
|
|
|
make_db(&db_path, map![vec![] => vec![], vec![1] => vec![1]]);
|
|
|
|
let expected = map![vec![0x11] => vec![0x22], vec![1, 0x11] => vec![1, 0x22]];
|
|
|
|
|
|
|
|
manager.add_migration(Migration0).unwrap();
|
|
|
|
let end_path = manager.execute(&db_path, 0).unwrap();
|
|
|
|
|
|
|
|
verify_migration(&end_path, expected);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn noop_and_second_migration() {
|
|
|
|
let dir = RandomTempPath::create_dir();
|
|
|
|
let db_path = db_path(dir.as_path());
|
|
|
|
let mut manager = Manager::new(Config::default());
|
|
|
|
make_db(&db_path, map![vec![] => vec![], vec![1] => vec![1]]);
|
|
|
|
let expected = map![vec![] => vec![], vec![1] => vec![]];
|
|
|
|
|
|
|
|
manager.add_migration(Migration1).unwrap();
|
|
|
|
let end_path = manager.execute(&db_path, 0).unwrap();
|
|
|
|
|
|
|
|
verify_migration(&end_path, expected);
|
|
|
|
}
|
|
|
|
|
2016-05-26 18:24:51 +02:00
|
|
|
#[test]
|
|
|
|
fn is_migration_needed() {
|
|
|
|
let mut manager = Manager::new(Config::default());
|
|
|
|
manager.add_migration(Migration0).unwrap();
|
|
|
|
manager.add_migration(Migration1).unwrap();
|
|
|
|
|
|
|
|
assert!(manager.is_needed(0));
|
|
|
|
assert!(manager.is_needed(1));
|
|
|
|
assert!(!manager.is_needed(2));
|
|
|
|
}
|
2016-09-26 18:24:58 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn pre_columns() {
|
|
|
|
let mut manager = Manager::new(Config::default());
|
|
|
|
manager.add_migration(AddsColumn).unwrap();
|
|
|
|
|
|
|
|
let dir = RandomTempPath::create_dir();
|
|
|
|
let db_path = db_path(dir.as_path());
|
|
|
|
|
|
|
|
// this shouldn't fail to open the database even though it's one column
|
|
|
|
// short of the one before it.
|
|
|
|
manager.execute(&db_path, 0).unwrap();
|
|
|
|
}
|
2017-02-26 19:22:51 +01:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn change_columns() {
|
|
|
|
use kvdb::DatabaseConfig;
|
|
|
|
|
|
|
|
let mut manager = Manager::new(Config::default());
|
|
|
|
manager.add_migration(::migration::ChangeColumns {
|
|
|
|
pre_columns: None,
|
|
|
|
post_columns: Some(4),
|
|
|
|
version: 1,
|
|
|
|
}).unwrap();
|
|
|
|
|
|
|
|
let dir = RandomTempPath::create_dir();
|
|
|
|
let db_path = db_path(dir.as_path());
|
|
|
|
|
|
|
|
let new_path = manager.execute(&db_path, 0).unwrap();
|
|
|
|
|
2017-02-26 19:30:54 +01:00
|
|
|
assert_eq!(db_path, new_path, "Changing columns is an in-place migration.");
|
|
|
|
|
2017-02-26 19:22:51 +01:00
|
|
|
let config = DatabaseConfig::with_columns(Some(4));
|
|
|
|
let db = Database::open(&config, new_path.to_str().unwrap()).unwrap();
|
|
|
|
assert_eq!(db.num_columns(), 4);
|
|
|
|
}
|