// This file is part of Moonfire NVR, a security camera network video recorder. // Copyright (C) 2021 The Moonfire NVR Authors; see AUTHORS and LICENSE.txt. // SPDX-License-Identifier: GPL-v3.0-or-later WITH GPL-3.0-linking-exception use base::FastHashMap; /// Upgrades a version 6 schema to a version 7 schema. use base::{err, Error}; use rusqlite::{named_params, params}; use std::{convert::TryFrom, path::PathBuf}; use tracing::debug; use url::Url; use uuid::Uuid; use crate::{ json::{ CameraConfig, GlobalConfig, SampleFileDirConfig, SignalConfig, SignalTypeConfig, UserConfig, }, SqlUuid, }; fn copy_meta(tx: &rusqlite::Transaction) -> Result<(), Error> { let mut stmt = tx.prepare("select uuid, max_signal_changes from old_meta")?; let mut insert = tx.prepare("insert into meta (uuid, config) values (:uuid, :config)")?; let mut rows = stmt.query(params![])?; while let Some(row) = rows.next()? { let uuid: SqlUuid = row.get(0)?; let max_signal_changes: Option = row.get(1)?; let config = GlobalConfig { max_signal_changes: max_signal_changes .map(|s| { u32::try_from(s).map_err(|e| { err!( OutOfRange, msg("max_signal_changes out of range"), source(e) ) }) }) .transpose()?, ..Default::default() }; insert.execute(named_params! { ":uuid": uuid, ":config": &config, })?; } Ok(()) } fn copy_sample_file_dir(tx: &rusqlite::Transaction) -> Result<(), Error> { let mut stmt = tx.prepare("select id, uuid, path, last_complete_open_id from old_sample_file_dir")?; let mut insert = tx.prepare( r#" insert into sample_file_dir (id, uuid, config, last_complete_open_id) values (:id, :uuid, :config, :last_complete_open_id) "#, )?; let mut rows = stmt.query(params![])?; while let Some(row) = rows.next()? { let id: i32 = row.get(0)?; let path: String = row.get(2)?; let uuid: SqlUuid = row.get(1)?; let config = SampleFileDirConfig { path: PathBuf::from(path), ..Default::default() }; let last_complete_open_id: Option = row.get(3)?; insert.execute(named_params! { ":id": id, ":uuid": uuid, ":config": &config, ":last_complete_open_id": &last_complete_open_id, })?; } Ok(()) } fn copy_users(tx: &rusqlite::Transaction) -> Result<(), Error> { let mut stmt = tx.prepare( r#" select id, username, flags, password_hash, password_id, password_failure_count, unix_uid, permissions from old_user "#, )?; let mut insert = tx.prepare( r#" insert into user (id, username, config, password_hash, password_id, password_failure_count, permissions) values (:id, :username, :config, :password_hash, :password_id, :password_failure_count, :permissions) "#, )?; let mut rows = stmt.query(params![])?; while let Some(row) = rows.next()? { let id: i32 = row.get(0)?; let username: String = row.get(1)?; let flags: i32 = row.get(2)?; let password_hash: String = row.get(3)?; let password_id: i32 = row.get(4)?; let password_failure_count: i32 = row.get(5)?; let unix_uid: Option = row.get(6)?; let permissions: Vec = row.get(7)?; let config = UserConfig { disabled: (flags & 1) != 0, unix_uid: unix_uid .map(u64::try_from) .transpose() .map_err(|_| err!(OutOfRange, msg("bad unix_uid")))?, ..Default::default() }; insert.execute(named_params! { ":id": id, ":username": username, ":config": config, ":password_hash": password_hash, ":password_id": password_id, ":password_failure_count": password_failure_count, ":permissions": permissions, })?; } Ok(()) } fn copy_signal_types(tx: &rusqlite::Transaction) -> Result<(), Error> { let mut types_ = FastHashMap::default(); let mut stmt = tx.prepare("select type_uuid, value, name from signal_type_enum")?; let mut rows = stmt.query(params![])?; while let Some(row) = rows.next()? { let type_uuid: SqlUuid = row.get(0)?; let value: i32 = row.get(1)?; let name: Option = row.get(2)?; let type_ = types_ .entry(type_uuid.0) .or_insert_with(SignalTypeConfig::default); let value = u8::try_from(value).map_err(|_| err!(OutOfRange, msg("bad signal type value")))?; let value_config = type_.values.entry(value).or_insert_with(Default::default); if let Some(n) = name { value_config.name = n; } } let mut insert = tx.prepare("insert into signal_type (uuid, config) values (?, ?)")?; for (&uuid, config) in &types_ { insert.execute(params![SqlUuid(uuid), config])?; } Ok(()) } struct Signal { uuid: Uuid, type_uuid: Uuid, config: SignalConfig, } fn copy_signals(tx: &rusqlite::Transaction) -> Result<(), Error> { let mut signals = FastHashMap::default(); // Read from signal table. { let mut stmt = tx.prepare("select id, source_uuid, type_uuid, short_name from old_signal")?; let mut rows = stmt.query(params![])?; while let Some(row) = rows.next()? { let id: i32 = row.get(0)?; let id = u32::try_from(id).map_err(|e| err!(OutOfRange, msg("bad signal id"), source(e)))?; let source_uuid: SqlUuid = row.get(1)?; let type_uuid: SqlUuid = row.get(2)?; let short_name: String = row.get(3)?; signals.insert( id, Signal { uuid: source_uuid.0, type_uuid: type_uuid.0, config: SignalConfig { short_name, ..Default::default() }, }, ); } } // Read from the signal_camera table. { let mut stmt = tx.prepare("select signal_id, camera_id, type from signal_camera")?; let mut rows = stmt.query(params![])?; while let Some(row) = rows.next()? { let signal_id: i32 = row.get(0)?; let signal_id = u32::try_from(signal_id) .map_err(|e| err!(OutOfRange, msg("bad signal_id"), source(e)))?; let camera_id: i32 = row.get(1)?; let type_: i32 = row.get(2)?; let signal = signals.get_mut(&signal_id).unwrap(); signal.config.camera_associations.insert( camera_id, match type_ { 0 => "direct", _ => "indirect", } .to_owned(), ); } } let mut insert = tx.prepare( r#" insert into signal (id, uuid, type_uuid, config) values (:id, :uuid, :type_uuid, :config) "#, )?; for (&id, signal) in &signals { insert.execute(named_params! { ":id": id, ":uuid": SqlUuid(signal.uuid), ":type_uuid": SqlUuid(signal.type_uuid), ":config": &signal.config, })?; } Ok(()) } fn copy_cameras(tx: &rusqlite::Transaction) -> Result<(), Error> { let mut insert = tx.prepare( r#" insert into camera (id, short_name, uuid, config) values (:id, :short_name, :uuid, :config) "#, )?; let mut stmt = tx.prepare( r#" select id, uuid, short_name, description, onvif_host, username, password from old_camera "#, )?; let mut rows = stmt.query(params![])?; while let Some(row) = rows.next()? { let id: i32 = row.get(0)?; let uuid: SqlUuid = row.get(1)?; let uuid_bytes = &uuid.0.as_bytes()[..]; let short_name: String = row.get(2)?; let mut description: Option = row.get(3)?; let onvif_host: Option = row.get(4)?; let mut username: Option = row.get(5)?; let mut password: Option = row.get(6)?; let config = CameraConfig { description: description.take().unwrap_or_default(), onvif_base_url: onvif_host // Older releases set the onvif host to the empty string instead // of using a SQL NULL, so convert empty to None here. // https://github.com/scottlamb/moonfire-nvr/issues/182 .filter(|h| !h.is_empty()) .map(|h| Url::parse(&format!("http://{h}/"))) .transpose() .map_err(|e| { err!( InvalidArgument, msg("bad onvif_host for camera id {id}"), source(e) ) })?, username: username.take().unwrap_or_default(), password: password.take().unwrap_or_default(), ..Default::default() }; insert.execute(named_params! { ":id": id, ":uuid": uuid_bytes, ":short_name": short_name, ":config": config, })?; } Ok(()) } fn copy_streams(tx: &rusqlite::Transaction) -> Result<(), Error> { let mut insert = tx.prepare( r#" insert into stream (id, camera_id, sample_file_dir_id, type, config, cum_recordings, cum_media_duration_90k, cum_runs) values (:id, :camera_id, :sample_file_dir_id, :type, :config, :cum_recordings, :cum_media_duration_90k, :cum_runs) "#, )?; let mut stmt = tx.prepare( r#" select id, camera_id, sample_file_dir_id, type, record, rtsp_url, retain_bytes, flush_if_sec, cum_recordings, cum_media_duration_90k, cum_runs from old_stream "#, )?; let mut rows = stmt.query(params![])?; while let Some(row) = rows.next()? { let id: i32 = row.get(0)?; let camera_id: i32 = row.get(1)?; let sample_file_dir_id: i32 = row.get(2)?; let type_: String = row.get(3)?; let record: bool = row.get(4)?; let rtsp_url: String = row.get(5)?; let retain_bytes: i64 = row.get(6)?; let flush_if_sec: u32 = row.get(7)?; let cum_recordings: i64 = row.get(8)?; let cum_media_duration_90k: i64 = row.get(9)?; let cum_runs: i64 = row.get(10)?; let config = crate::json::StreamConfig { mode: (if record { crate::json::STREAM_MODE_RECORD } else { "" }) .to_owned(), url: Some(Url::parse(&rtsp_url).map_err(|e| { err!( InvalidArgument, msg("bad rtsp_url for stream id {id}"), source(e) ) })?), retain_bytes, flush_if_sec, ..Default::default() }; insert.execute(named_params! { ":id": id, ":camera_id": camera_id, ":sample_file_dir_id": sample_file_dir_id, ":type": type_, ":config": config, ":cum_recordings": cum_recordings, ":cum_media_duration_90k": cum_media_duration_90k, ":cum_runs": cum_runs, })?; } Ok(()) } pub fn run(_args: &super::Args, tx: &rusqlite::Transaction) -> Result<(), Error> { debug!("pre batch"); tx.execute_batch( r#" alter table open add boot_uuid check (length(boot_uuid) = 16); alter table user rename to old_user; alter table user_session rename to old_user_session; alter table camera rename to old_camera; alter table stream rename to old_stream; alter table signal rename to old_signal; alter table sample_file_dir rename to old_sample_file_dir; alter table meta rename to old_meta; create table meta ( uuid blob not null check (length(uuid) = 16), config text ); create table sample_file_dir ( id integer primary key, uuid blob unique not null check (length(uuid) = 16), config text, last_complete_open_id integer references open (id) ); create table camera ( id integer primary key, uuid blob unique not null check (length(uuid) = 16), short_name text not null, config text not null ); create table stream ( id integer primary key, camera_id integer not null references camera (id), sample_file_dir_id integer references sample_file_dir (id), type text not null check (type in ('main', 'sub', 'ext')), config text not null, cum_recordings integer not null check (cum_recordings >= 0), cum_media_duration_90k integer not null check (cum_media_duration_90k >= 0), cum_runs integer not null check (cum_runs >= 0), unique (camera_id, type) ); create table signal ( id integer primary key, uuid blob unique not null check (length(uuid) = 16), type_uuid blob not null references signal_type (uuid) check (length(type_uuid) = 16), config text ); create table signal_type ( uuid blob primary key check (length(uuid) = 16), config text ) without rowid; create table user ( id integer primary key, username unique not null, config text, password_hash text, password_id integer not null default 0, password_failure_count integer not null default 0, permissions blob not null default X'' ); create table user_session ( session_id_hash blob primary key not null, user_id integer references user (id) not null, seed blob not null, flags integer not null, domain text, description text, creation_password_id integer, creation_time_sec integer not null, creation_user_agent text, creation_peer_addr blob, revocation_time_sec integer, revocation_user_agent text, revocation_peer_addr blob, revocation_reason integer, revocation_reason_detail text, last_use_time_sec integer, last_use_user_agent text, last_use_peer_addr blob, use_count not null default 0, permissions blob not null default X'' ) without rowid; drop index user_session_uid; create index user_session_uid on user_session (user_id); "#, )?; debug!("copying meta"); copy_meta(tx)?; debug!("copying sample_file_dir"); copy_sample_file_dir(tx)?; debug!("copying camera"); copy_cameras(tx)?; debug!("copying signal_type"); copy_signal_types(tx)?; debug!("copying signal"); copy_signals(tx)?; debug!("copying stream"); copy_streams(tx)?; debug!("copying user"); copy_users(tx)?; debug!("post batch"); tx.execute_batch( r#" insert into user_session select * from old_user_session; alter table garbage rename to old_garbage; create table garbage ( sample_file_dir_id integer not null references sample_file_dir (id), composite_id integer not null, primary key (sample_file_dir_id, composite_id) ) without rowid; insert into garbage select * from old_garbage; drop table old_garbage; drop index recording_cover; alter table recording rename to old_recording; create table recording ( composite_id integer primary key, open_id integer not null, stream_id integer not null references stream (id), run_offset integer not null, flags integer not null, sample_file_bytes integer not null check (sample_file_bytes > 0), start_time_90k integer not null check (start_time_90k > 0), prev_media_duration_90k integer not null check (prev_media_duration_90k >= 0), prev_runs integer not null check (prev_runs >= 0), wall_duration_90k integer not null check (wall_duration_90k >= 0 and wall_duration_90k < 5*60*90000), media_duration_delta_90k integer not null, video_samples integer not null check (video_samples > 0), video_sync_samples integer not null check (video_sync_samples > 0), video_sample_entry_id integer references video_sample_entry (id), end_reason text check (composite_id >> 32 = stream_id) ); create index recording_cover on recording ( stream_id, start_time_90k, open_id, wall_duration_90k, media_duration_delta_90k, video_samples, video_sync_samples, video_sample_entry_id, sample_file_bytes, run_offset, flags ); insert into recording select *, null from old_recording; alter table recording_integrity rename to old_recording_integrity; create table recording_integrity ( composite_id integer primary key references recording (composite_id), local_time_delta_90k integer, local_time_since_open_90k integer, wall_time_delta_90k integer, sample_file_blake3 blob check (length(sample_file_blake3) <= 32) ); insert into recording_integrity select * from old_recording_integrity; alter table recording_playback rename to old_recording_playback; create table recording_playback ( composite_id integer primary key references recording (composite_id), video_index blob not null check (length(video_index) > 0) ); insert into recording_playback select * from old_recording_playback; drop table signal_camera; drop table old_signal; drop table old_recording_playback; drop table old_recording_integrity; drop table old_recording; drop table old_stream; drop table old_camera; drop table old_sample_file_dir; drop table old_meta; drop table old_user_session; drop table old_user; drop table signal_type_enum; "#, )?; Ok(()) }