mirror of
https://github.com/scottlamb/moonfire-nvr.git
synced 2025-01-28 15:06:01 -05:00
86816e862a
I didn't go to quite the latest version of everything, in an effort to minimize duplicates in the cargo tree.
373 lines
14 KiB
Rust
373 lines
14 KiB
Rust
// 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<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)]
|
|
mod tests {
|
|
use super::*;
|
|
use crate::compare;
|
|
use crate::testutil;
|
|
use base::err;
|
|
use base::FastHashMap;
|
|
|
|
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<rusqlite::Connection, Error> {
|
|
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 = FastHashMap::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(())
|
|
}
|
|
}
|