// This file is part of Moonfire NVR, a security camera network video recorder. // Copyright (C) 2020 The Moonfire NVR Authors; see AUTHORS and LICENSE.txt. // SPDX-License-Identifier: GPL-v3.0-or-later WITH GPL-3.0-linking-exception. //! Upgrades the database schema. //! //! See `guide/schema.md` for more information. use crate::db::{self, EXPECTED_SCHEMA_VERSION}; use base::{bail, Error}; use nix::NixPath; use rusqlite::params; use std::ffi::CStr; use std::io::Write; use tracing::info; use uuid::Uuid; mod v0_to_v1; mod v1_to_v2; mod v2_to_v3; mod v3_to_v4; mod v4_to_v5; mod v5_to_v6; mod v6_to_v7; #[derive(Debug)] pub struct Args<'a> { pub sample_file_dir: Option<&'a std::path::Path>, pub preset_journal: &'a str, pub no_vacuum: bool, } fn set_journal_mode(conn: &rusqlite::Connection, requested: &str) -> Result<(), Error> { assert!(!requested.contains(';')); // quick check for accidental sql injection. let actual = conn.query_row( &format!("pragma journal_mode = {requested}"), params![], |row| row.get::<_, String>(0), )?; info!( "...database now in journal_mode {} (requested {}).", actual, requested ); Ok(()) } fn upgrade( args: &Args, target_schema_ver: i32, sw_version: &str, conn: &mut rusqlite::Connection, ) -> Result<(), Error> { let upgraders = [ v0_to_v1::run, v1_to_v2::run, v2_to_v3::run, v3_to_v4::run, v4_to_v5::run, v5_to_v6::run, v6_to_v7::run, ]; { assert_eq!(upgraders.len(), db::EXPECTED_SCHEMA_VERSION as usize); let old_schema_ver = conn.query_row("select max(id) from version", params![], |row| row.get(0))?; if old_schema_ver > EXPECTED_SCHEMA_VERSION { bail!( FailedPrecondition, msg("database is at version {old_schema_ver}, \ later than expected {EXPECTED_SCHEMA_VERSION}"), ); } else if old_schema_ver < 0 { bail!( FailedPrecondition, msg("Database is at negative version {old_schema_ver}!") ); } info!( "Upgrading database from schema version {} to schema version {}...", old_schema_ver, target_schema_ver ); for ver in old_schema_ver..target_schema_ver { info!( "...from schema version {} to schema version {}", ver, ver + 1 ); let tx = conn.transaction()?; upgraders[ver as usize](args, &tx)?; tx.execute( r#" insert into version (id, unix_time, notes) values (?, cast(strftime('%s', 'now') as int32), ?) "#, params![ver + 1, format!("Upgraded using moonfire-nvr {sw_version}")], )?; tx.commit()?; } } Ok(()) } pub fn run(args: &Args, sw_version: &str, conn: &mut rusqlite::Connection) -> Result<(), Error> { db::check_sqlite_version()?; db::set_integrity_pragmas(conn)?; set_journal_mode(conn, args.preset_journal)?; upgrade(args, EXPECTED_SCHEMA_VERSION, sw_version, conn)?; // As in "moonfire-nvr init": try for page_size=16384 and wal for the reasons explained there. // // Do the vacuum prior to switching back to WAL for two reasons: // * page_size only takes effect on a vacuum in non-WAL mode. // https://www.sqlite.org/pragma.html#pragma_page_size // * vacuum is a huge transaction, and on old versions of SQLite3, that's best done in // non-WAL mode. https://www.sqlite.org/wal.html if !args.no_vacuum { info!("...vacuuming database after upgrade."); conn.execute_batch( r#" pragma page_size = 16384; vacuum; "#, )?; } set_journal_mode(conn, "wal")?; info!("...done."); 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.as_hyphenated()) .expect("can't format uuid to pathname buf"); UuidPath(buf) } } impl NixPath for UuidPath { fn is_empty(&self) -> bool { false } fn len(&self) -> usize { 36 } fn with_nix_path(&self, f: F) -> Result where F: FnOnce(&CStr) -> T, { let p = CStr::from_bytes_with_nul(&self.0[..]).expect("no interior nuls"); Ok(f(p)) } } #[cfg(test)] mod tests { use super::*; use crate::compare; use crate::testutil; use base::err; use fnv::FnvHashMap; const BAD_ANAMORPHIC_VIDEO_SAMPLE_ENTRY: &[u8] = b"\x00\x00\x00\x84\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\x01\x40\x00\xf0\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\x2e\ \x61\x76\x63\x43\x01\x4d\x40\x1e\xff\xe1\x00\x17\x67\x4d\x40\x1e\ \x9a\x66\x0a\x0f\xff\x35\x01\x01\x01\x40\x00\x00\xfa\x00\x00\x03\ \x01\xf4\x01\x01\x00\x04\x68\xee\x3c\x80"; const GOOD_ANAMORPHIC_VIDEO_SAMPLE_ENTRY: &[u8] = b"\x00\x00\x00\x9f\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\ \x02\xc0\x01\xe0\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\x49\x61\x76\x63\x43\x01\x64\ \x00\x16\xff\xe1\x00\x31\x67\x64\x00\x16\xac\x1b\x1a\x80\xb0\x3d\ \xff\xff\x00\x28\x00\x21\x6e\x0c\x0c\x0c\x80\x00\x01\xf4\x00\x00\ \x27\x10\x74\x30\x07\xd0\x00\x07\xa1\x25\xde\x5c\x68\x60\x0f\xa0\ \x00\x0f\x42\x4b\xbc\xb8\x50\x01\x00\x05\x68\xee\x38\x30\x00"; fn new_conn() -> Result { let conn = rusqlite::Connection::open_in_memory()?; conn.execute("pragma foreign_keys = on", params![])?; conn.execute("pragma fullfsync = on", params![])?; conn.execute("pragma synchronous = 2", params![])?; Ok(conn) } fn compare(c: &rusqlite::Connection, ver: i32, fresh_sql: &str) -> Result<(), Error> { let fresh = new_conn()?; fresh.execute_batch(fresh_sql)?; if let Some(diffs) = compare::get_diffs( &format!("upgraded to version {ver}"), c, &format!("fresh version {ver}"), &fresh, )? { panic!("Version {ver}: differences found:\n{diffs}"); } Ok(()) } /// Upgrades and compares schemas. /// Doesn't (yet) compare any actual data. #[test] fn upgrade_and_compare() -> Result<(), Error> { testutil::init(); let tmpdir = tempfile::Builder::new() .prefix("moonfire-nvr-test") .tempdir()?; //let path = tmpdir.path().to_str().ok_or_else(|| err!("invalid UTF-8"))?.to_owned(); let mut upgraded = new_conn()?; 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'0000000000000000000000000000000000000000', 1920, 1080, ?); "#, params![testutil::TEST_VIDEO_SAMPLE_ENTRY_DATA], )?; upgraded.execute( r#" insert into video_sample_entry (id, sha1, width, height, data) values (2, X'0000000000000000000000000000000000000001', 320, 240, ?); "#, params![BAD_ANAMORPHIC_VIDEO_SAMPLE_ENTRY], )?; upgraded.execute( r#" insert into video_sample_entry (id, sha1, width, height, data) values (3, X'0000000000000000000000000000000000000002', 704, 480, ?); "#, params![GOOD_ANAMORPHIC_VIDEO_SAMPLE_ENTRY], )?; upgraded.execute( r#" insert into video_sample_entry (id, sha1, width, height, data) values (4, X'0000000000000000000000000000000000000003', 704, 480, ?); "#, params![GOOD_ANAMORPHIC_VIDEO_SAMPLE_ENTRY], )?; 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'), (2, 1, 42, 140063580090000, 90000, 0, 1, 1, 2, X'94DE8484FF874A5295D488C8038A0312', zeroblob(20), X'00'), (3, 1, 42, 140063580180000, 90000, 0, 1, 1, 3, X'C94D4D0B533746059CD40B29039E641E', 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 rec2 = tmpdir.path().join("94de8484-ff87-4a52-95d4-88c8038a0312"); let rec3 = tmpdir.path().join("c94d4d0b-5337-4605-9cd4-0b29039e641e"); let garbage = tmpdir.path().join("51ef700c-933e-4197-aae4-ee8161e94221"); std::fs::File::create(&rec1)?; std::fs::File::create(&rec2)?; std::fs::File::create(&rec3)?; std::fs::File::create(&garbage)?; for (ver, fresh_sql) in &[ (1, Some(include_str!("v1.sql"))), (2, None), // transitional; don't compare schemas. (3, Some(include_str!("v3.sql"))), (4, None), // transitional; don't compare schemas. (5, Some(include_str!("v5.sql"))), (6, Some(include_str!("v6.sql"))), (7, Some(include_str!("../schema.sql"))), ] { upgrade( &Args { sample_file_dir: Some(tmpdir.path()), preset_journal: "delete", no_vacuum: false, }, *ver, "test", &mut upgraded, ) .map_err(|e| err!(e, msg("upgrade to schema version {ver} failed")))?; if let Some(f) = fresh_sql { compare(&upgraded, *ver, f)?; } if *ver == 3 { // Check that the garbage files is cleaned up properly, but also add it back // to simulate a bug prior to 433be217. The v5 upgrade should take care of // anything left over. assert!(!garbage.exists()); std::fs::File::create(&garbage)?; } else if *ver == 4 { // First version that supports signals. Add them and ensure they don't break // subsequent upgrades. upgraded.execute_batch( r#" insert into signal (id, source_uuid, type_uuid, short_name) values (1, x'1B3889C0A59F400DA24C94EBEB19CC3A', x'EE66270FD9C648198B339720D4CBCA6B', 'a'), (2, x'A4A73D9A53424EBCB9F6366F1E5617FA', x'EE66270FD9C648198B339720D4CBCA6B', 'b'); insert into signal_type_enum (type_uuid, value, name, motion, color) values (x'EE66270FD9C648198B339720D4CBCA6B', 1, 'still', 0, 'black'), (x'EE66270FD9C648198B339720D4CBCA6B', 2, 'moving', 1, 'red'); insert into signal_camera (signal_id, camera_id, type) values (1, 1, 0), (2, 1, 1); "#, )?; } else if *ver == 6 { // Check that the pasp was set properly. let mut stmt = upgraded.prepare( r#" select id, pasp_h_spacing, pasp_v_spacing from video_sample_entry "#, )?; let mut rows = stmt.query(params![])?; let mut pasp_by_id = FnvHashMap::default(); while let Some(row) = rows.next()? { let id: i32 = row.get(0)?; let pasp_h_spacing: i32 = row.get(1)?; let pasp_v_spacing: i32 = row.get(2)?; pasp_by_id.insert(id, (pasp_h_spacing, pasp_v_spacing)); } assert_eq!(pasp_by_id.get(&1), Some(&(1, 1))); assert_eq!(pasp_by_id.get(&2), Some(&(4, 3))); assert_eq!(pasp_by_id.get(&3), Some(&(40, 33))); // No recording references this video_sample_entry, so it gets dropped on upgrade. assert_eq!(pasp_by_id.get(&4), None); } } // 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(()) } }