improve test of upgrade from v0 on up; fix bugs

Now the test actually has a recording and garbage with matching files.
This caught a few problems in the upgrade procedure:

* it didn't work with foreign keys enabled because the new recording
  table was set up after the new camera table, and the old recording
  table was destroyed after the old camera table. And now I enable
  foreign keys all the time. Reorder the procedure to fix.

* the pathname manipulation in the v2 to v3 procedure was incorrect
  since my introduction of nix because I gave it a &[u8] with the
  trailing nul, where I should have used CStr::from_bytes_with_nul.

* it wasn't removing garbage files. It'd be most natural to do this
  in the v2 to v3 upgrade (with the rename) but I historically removed
  the table when upgrading to v2. I can't redefine the schema now, so
  do it unnaturally.

  I'm considering also renaming all uuid-like files on upgrade to v4/v5
  to clean up this mess automatically for installations that have
  already done this upgrade.
This commit is contained in:
Scott Lamb 2019-07-21 22:40:01 -07:00
parent 01d20960ef
commit 433be217ac
5 changed files with 115 additions and 60 deletions

View File

@ -48,6 +48,14 @@ static INIT: sync::Once = sync::ONCE_INIT;
pub const TEST_CAMERA_ID: i32 = 1; pub const TEST_CAMERA_ID: i32 = 1;
pub const TEST_STREAM_ID: i32 = 1; pub const TEST_STREAM_ID: i32 = 1;
pub const TEST_VIDEO_SAMPLE_ENTRY_DATA: &[u8] =
b"\x00\x00\x00\x7D\x61\x76\x63\x31\x00\x00\x00\x00\x00\x00\x00\x01\x00\x00\x00\x00\x00\x00\x00\
\x00\x00\x00\x00\x00\x00\x00\x00\x00\x07\x80\x04\x38\x00\x48\x00\x00\x00\x48\x00\x00\x00\x00\
\x00\x00\x00\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\
\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x18\xFF\xFF\x00\x00\x00\x27\x61\x76\
\x63\x43\x01\x4D\x00\x2A\xFF\xE1\x00\x10\x67\x4D\x00\x2A\x95\xA8\x1E\x00\x89\xF9\x66\xE0\x20\
\x20\x20\x40\x01\x00\x04\x68\xEE\x3C\x80";
/// Performs global initialization for tests. /// Performs global initialization for tests.
/// * set up logging. (Note the output can be confusing unless `RUST_TEST_THREADS=1` is set in /// * set up logging. (Note the output can be confusing unless `RUST_TEST_THREADS=1` is set in
/// the program's environment prior to running.) /// the program's environment prior to running.)

View File

@ -35,7 +35,11 @@
use crate::db; use crate::db;
use failure::{Error, bail}; use failure::{Error, bail};
use log::info; use log::info;
use std::ffi::CStr;
use std::io::Write;
use nix::NixPath;
use rusqlite::params; use rusqlite::params;
use uuid::Uuid;
mod v0_to_v1; mod v0_to_v1;
mod v1_to_v2; mod v1_to_v2;
@ -82,7 +86,7 @@ fn upgrade(args: &Args, target_ver: i32, conn: &mut rusqlite::Connection) -> Res
bail!("Database is at negative version {}!", old_ver); bail!("Database is at negative version {}!", old_ver);
} }
info!("Upgrading database from version {} to version {}...", old_ver, target_ver); info!("Upgrading database from version {} to version {}...", old_ver, target_ver);
set_journal_mode(&conn, args.flag_preset_journal).unwrap(); set_journal_mode(&conn, args.flag_preset_journal)?;
for ver in old_ver .. target_ver { for ver in old_ver .. target_ver {
info!("...from version {} to version {}", ver, ver + 1); info!("...from version {} to version {}", ver, ver + 1);
let tx = conn.transaction()?; let tx = conn.transaction()?;
@ -113,22 +117,46 @@ pub fn run(args: &Args, conn: &mut rusqlite::Connection) -> Result<(), Error> {
// WAL is the preferred journal mode for normal operation; it reduces the number of syncs // WAL is the preferred journal mode for normal operation; it reduces the number of syncs
// without compromising safety. // without compromising safety.
set_journal_mode(&conn, "wal").unwrap(); set_journal_mode(&conn, "wal")?;
if !args.flag_no_vacuum { if !args.flag_no_vacuum {
info!("...vacuuming database after upgrade."); info!("...vacuuming database after upgrade.");
conn.execute_batch(r#" conn.execute_batch(r#"
pragma page_size = 16384; pragma page_size = 16384;
vacuum; vacuum;
"#).unwrap(); "#)?;
} }
info!("...done."); info!("...done.");
Ok(()) Ok(())
} }
/// A uuid-based path, as used in version 0 and version 1 schemas.
struct UuidPath([u8; 37]);
impl UuidPath {
pub(crate) fn from(uuid: Uuid) -> Self {
let mut buf = [0u8; 37];
write!(&mut buf[..36], "{}", uuid.to_hyphenated_ref())
.expect("can't format uuid to pathname buf");
UuidPath(buf)
}
}
impl NixPath for UuidPath {
fn len(&self) -> usize { 36 }
fn with_nix_path<T, F>(&self, f: F) -> Result<T, nix::Error>
where F: FnOnce(&CStr) -> T {
let p = CStr::from_bytes_with_nul(&self.0[..]).expect("no interior nuls");
Ok(f(p))
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use crate::compare; use crate::compare;
use crate::testutil;
use failure::{ResultExt, format_err};
use super::*; use super::*;
fn new_conn() -> Result<rusqlite::Connection, Error> { fn new_conn() -> Result<rusqlite::Connection, Error> {
@ -152,25 +180,59 @@ mod tests {
/// Doesn't (yet) compare any actual data. /// Doesn't (yet) compare any actual data.
#[test] #[test]
fn upgrade_and_compare() -> Result<(), Error> { fn upgrade_and_compare() -> Result<(), Error> {
let tmpdir = tempdir::TempDir::new("moonfire-nvr-test").unwrap(); testutil::init();
let path = tmpdir.path().to_str().unwrap().to_owned(); let tmpdir = tempdir::TempDir::new("moonfire-nvr-test")?;
let path = tmpdir.path().to_str().ok_or_else(|| format_err!("invalid UTF-8"))?.to_owned();
let mut upgraded = new_conn()?; let mut upgraded = new_conn()?;
upgraded.execute_batch(include_str!("v0.sql"))?; upgraded.execute_batch(include_str!("v0.sql"))?;
upgraded.execute_batch(r#"
insert into camera (id, uuid, short_name, description, host, username, password,
main_rtsp_path, sub_rtsp_path, retain_bytes)
values (1, zeroblob(16), 'test camera', 'desc', 'host', 'user', 'pass',
'main', 'sub', 42);
"#)?;
upgraded.execute(r#"
insert into video_sample_entry (id, sha1, width, height, data)
values (1, X'3BA3EDE1BD93B7BCB7AB5BD099C047701451B822',
1920, 1080, ?);
"#, params![testutil::TEST_VIDEO_SAMPLE_ENTRY_DATA])?;
upgraded.execute_batch(r#"
insert into recording (id, camera_id, sample_file_bytes, start_time_90k, duration_90k,
local_time_delta_90k, video_samples, video_sync_samples,
video_sample_entry_id, sample_file_uuid, sample_file_sha1,
video_index)
values (1, 1, 42, 140063580000000, 90000, 0, 1, 1, 1,
X'E69D45E8CBA64DC1BA2ECB1585983A10', zeroblob(20), X'00');
insert into reserved_sample_files values (X'51EF700C933E4197AAE4EE8161E94221', 0),
(X'E69D45E8CBA64DC1BA2ECB1585983A10', 1);
"#)?;
let rec1 = tmpdir.path().join("e69d45e8-cba6-4dc1-ba2e-cb1585983a10");
let garbage = tmpdir.path().join("51ef700c-933e-4197-aae4-ee8161e94221");
std::fs::File::create(&rec1)?;
std::fs::File::create(&garbage)?;
for (ver, fresh_sql) in &[(1, Some(include_str!("v1.sql"))), for (ver, fresh_sql) in &[(1, Some(include_str!("v1.sql"))),
(2, None), // transitional; don't compare schemas. (2, None), // transitional; don't compare schemas.
(3, Some(include_str!("v3.sql"))), (3, Some(include_str!("v3.sql"))),
(4, None), // transitional; don't compare schemas. (4, None), // transitional; don't compare schemas.
(4, Some(include_str!("../schema.sql")))] { (5, Some(include_str!("../schema.sql")))] {
upgrade(&Args { upgrade(&Args {
flag_sample_file_dir: Some(&path), flag_sample_file_dir: Some(&path),
flag_preset_journal: "delete", flag_preset_journal: "delete",
flag_no_vacuum: false, flag_no_vacuum: false,
}, *ver, &mut upgraded)?; }, *ver, &mut upgraded).context(format!("upgrading to version {}", ver))?;
if let Some(f) = fresh_sql { if let Some(f) = fresh_sql {
compare(&upgraded, *ver, f)?; compare(&upgraded, *ver, f)?;
} }
} }
// Check that recording files get renamed.
assert!(!rec1.exists());
assert!(tmpdir.path().join("0000000100000001").exists());
// Check that garbage files get cleaned up.
assert!(!garbage.exists());
Ok(()) Ok(())
} }
} }

View File

@ -34,7 +34,7 @@ use crate::db;
use crate::recording; use crate::recording;
use failure::Error; use failure::Error;
use log::warn; use log::warn;
use rusqlite::types::ToSql; use rusqlite::params;
use std::collections::HashMap; use std::collections::HashMap;
pub fn run(_args: &super::Args, tx: &rusqlite::Transaction) -> Result<(), Error> { pub fn run(_args: &super::Args, tx: &rusqlite::Transaction) -> Result<(), Error> {
@ -88,12 +88,27 @@ pub fn run(_args: &super::Args, tx: &rusqlite::Transaction) -> Result<(), Error>
sample_file_sha1 blob not null check (length(sample_file_sha1) = 20), sample_file_sha1 blob not null check (length(sample_file_sha1) = 20),
video_index blob not null check (length(video_index) > 0) video_index blob not null check (length(video_index) > 0)
); );
insert into camera
select
id,
uuid,
short_name,
description,
host,
username,
password,
main_rtsp_path,
sub_rtsp_path,
retain_bytes,
1 as next_recording_id
from
old_camera;
"#)?; "#)?;
let camera_state = fill_recording(tx).unwrap(); let camera_state = fill_recording(tx)?;
fill_camera(tx, camera_state).unwrap(); update_camera(tx, camera_state)?;
tx.execute_batch(r#" tx.execute_batch(r#"
drop table old_camera;
drop table old_recording; drop table old_recording;
drop table old_camera;
"#)?; "#)?;
Ok(()) Ok(())
} }
@ -142,7 +157,7 @@ fn fill_recording(tx: &rusqlite::Transaction) -> Result<HashMap<i32, CameraState
insert into recording_playback values (:composite_id, :sample_file_uuid, :sample_file_sha1, insert into recording_playback values (:composite_id, :sample_file_uuid, :sample_file_sha1,
:video_index) :video_index)
"#)?; "#)?;
let mut rows = select.query(&[] as &[&dyn ToSql])?; let mut rows = select.query(params![])?;
let mut camera_state: HashMap<i32, CameraState> = HashMap::new(); let mut camera_state: HashMap<i32, CameraState> = HashMap::new();
while let Some(row) = rows.next()? { while let Some(row) = rows.next()? {
let camera_id: i32 = row.get(0)?; let camera_id: i32 = row.get(0)?;
@ -203,44 +218,15 @@ fn fill_recording(tx: &rusqlite::Transaction) -> Result<HashMap<i32, CameraState
Ok(camera_state) Ok(camera_state)
} }
fn fill_camera(tx: &rusqlite::Transaction, camera_state: HashMap<i32, CameraState>) fn update_camera(tx: &rusqlite::Transaction, camera_state: HashMap<i32, CameraState>)
-> Result<(), Error> { -> Result<(), Error> {
let mut select = tx.prepare(r#" let mut stmt = tx.prepare(r#"
select update camera set next_recording_id = :next_recording_id where id = :id
id, uuid, short_name, description, host, username, password, main_rtsp_path,
sub_rtsp_path, retain_bytes
from
old_camera
"#)?; "#)?;
let mut insert = tx.prepare(r#" for (ref id, ref state) in &camera_state {
insert into camera values (:id, :uuid, :short_name, :description, :host, :username, :password, stmt.execute_named(&[
:main_rtsp_path, :sub_rtsp_path, :retain_bytes, :next_recording_id)
"#)?;
let mut rows = select.query(&[] as &[&dyn ToSql])?;
while let Some(row) = rows.next()? {
let id: i32 = row.get(0)?;
let uuid: Vec<u8> = row.get(1)?;
let short_name: String = row.get(2)?;
let description: String = row.get(3)?;
let host: String = row.get(4)?;
let username: String = row.get(5)?;
let password: String = row.get(6)?;
let main_rtsp_path: String = row.get(7)?;
let sub_rtsp_path: String = row.get(8)?;
let retain_bytes: i64 = row.get(9)?;
insert.execute_named(&[
(":id", &id), (":id", &id),
(":uuid", &uuid), (":next_recording_id", &state.next_recording_id),
(":short_name", &short_name),
(":description", &description),
(":host", &host),
(":username", &username),
(":password", &password),
(":main_rtsp_path", &main_rtsp_path),
(":sub_rtsp_path", &sub_rtsp_path),
(":retain_bytes", &retain_bytes),
(":next_recording_id",
&camera_state.get(&id).map(|s| s.next_recording_id).unwrap_or(1)),
])?; ])?;
} }
Ok(()) Ok(())

View File

@ -346,7 +346,16 @@ fn verify_dir_contents(sample_file_path: &str, dir: &mut nix::dir::Dir,
let mut rows = stmt.query(&[] as &[&dyn ToSql])?; let mut rows = stmt.query(&[] as &[&dyn ToSql])?;
while let Some(row) = rows.next()? { while let Some(row) = rows.next()? {
let uuid: crate::db::FromSqlUuid = row.get(0)?; let uuid: crate::db::FromSqlUuid = row.get(0)?;
files.remove(&uuid.0); if files.remove(&uuid.0) {
// Also remove the garbage file. For historical reasons (version 2 was originally
// defined as not having a garbage table so still is), do this here rather than with
// the other path manipulations in v2_to_v3.rs. There's no harm anyway in deleting
// a garbage file so if the upgrade transation fails this is still a valid and complete
// version 1 database.
let p = super::UuidPath::from(uuid.0);
nix::unistd::unlinkat(Some(dir.as_raw_fd()), &p,
nix::unistd::UnlinkatFlags::NoRemoveDir)?;
}
} }
if !files.is_empty() { if !files.is_empty() {

View File

@ -38,10 +38,8 @@ use failure::Error;
use crate::schema; use crate::schema;
use protobuf::prelude::MessageField; use protobuf::prelude::MessageField;
use rusqlite::types::ToSql; use rusqlite::types::ToSql;
use std::io::Write;
use std::os::unix::io::AsRawFd; use std::os::unix::io::AsRawFd;
use std::sync::Arc; use std::sync::Arc;
use uuid::Uuid;
/// Opens the sample file dir. /// Opens the sample file dir.
/// ///
@ -88,9 +86,9 @@ pub fn run(_args: &super::Args, tx: &rusqlite::Transaction) -> Result<(), Error>
while let Some(row) = rows.next()? { while let Some(row) = rows.next()? {
let id = db::CompositeId(row.get(0)?); let id = db::CompositeId(row.get(0)?);
let sample_file_uuid: FromSqlUuid = row.get(1)?; let sample_file_uuid: FromSqlUuid = row.get(1)?;
let from_path = get_uuid_pathname(sample_file_uuid.0); let from_path = super::UuidPath::from(sample_file_uuid.0);
let to_path = crate::dir::CompositeIdPath::from(id); let to_path = crate::dir::CompositeIdPath::from(id);
if let Err(e) = nix::fcntl::renameat(Some(d.fd.as_raw_fd()), &from_path[..], if let Err(e) = nix::fcntl::renameat(Some(d.fd.as_raw_fd()), &from_path,
Some(d.fd.as_raw_fd()), &to_path) { Some(d.fd.as_raw_fd()), &to_path) {
if e == nix::Error::Sys(nix::errno::Errno::ENOENT) { if e == nix::Error::Sys(nix::errno::Errno::ENOENT) {
continue; // assume it was already moved. continue; // assume it was already moved.
@ -119,11 +117,3 @@ pub fn run(_args: &super::Args, tx: &rusqlite::Transaction) -> Result<(), Error>
"#)?; "#)?;
Ok(()) Ok(())
} }
/// Gets a pathname for a sample file suitable for passing to open or unlink.
fn get_uuid_pathname(uuid: Uuid) -> [u8; 37] {
let mut buf = [0u8; 37];
write!(&mut buf[..36], "{}", uuid.to_hyphenated_ref())
.expect("can't format uuid to pathname buf");
buf
}