// Copyright 2018 Mozilla // // Licensed under the Apache License, Version 2.0 (the "License"); you may not use // this file except in compliance with the License. You may obtain a copy of the // License at http://www.apache.org/licenses/LICENSE-2.0 // Unless required by applicable law or agreed to in writing, software distributed // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR // CONDITIONS OF ANY KIND, either express or implied. See the License for the // specific language governing permissions and limitations under the License. use std::convert::TryInto; use uuid::Uuid; use core_traits::Entid; use public_traits::errors::Result; use tolstoy_traits::errors::TolstoyError; use crate::types::LocalGlobalTxMapping; // Exposes a tx<->uuid mapping interface. pub struct TxMapper {} impl TxMapper { pub fn set_lg_mappings( db_tx: &mut rusqlite::Transaction<'_>, mappings: Vec>, ) -> Result<()> { let mut stmt = db_tx.prepare_cached("INSERT OR REPLACE INTO tolstoy_tu (tx, uuid) VALUES (?, ?)")?; for mapping in mappings.iter() { let uuid_bytes = mapping.remote.as_bytes().to_vec(); stmt.execute(rusqlite::params![&mapping.local, &uuid_bytes])?; } Ok(()) } pub fn set_lg_mapping( db_tx: &mut rusqlite::Transaction<'_>, mapping: LocalGlobalTxMapping<'_>, ) -> Result<()> { TxMapper::set_lg_mappings(db_tx, vec![mapping]) } // TODO for when we're downloading, right? pub fn get_or_set_uuid_for_tx( db_tx: &mut rusqlite::Transaction<'_>, tx: Entid, ) -> Result { match TxMapper::get(db_tx, tx)? { Some(uuid) => Ok(uuid), None => { let uuid = Uuid::new_v4(); let uuid_bytes = uuid.as_bytes().to_vec(); db_tx.execute( "INSERT INTO tolstoy_tu (tx, uuid) VALUES (?, ?)", rusqlite::params![&tx, &uuid_bytes], )?; Ok(uuid) } } } pub fn get_tx_for_uuid( db_tx: &rusqlite::Transaction<'_>, uuid: &Uuid, ) -> Result> { let mut stmt = db_tx.prepare_cached("SELECT tx FROM tolstoy_tu WHERE uuid = ?")?; let uuid_bytes = uuid.as_bytes().to_vec(); let results = stmt.query_map(&[&uuid_bytes], |r| r.get(0))?; let mut txs = vec![]; txs.extend(results); if txs.is_empty() { return Ok(None); } else if txs.len() > 1 { bail!(TolstoyError::TxIncorrectlyMapped(txs.len())); } Ok(Some(txs.remove(0)?)) } pub fn get(db_tx: &rusqlite::Transaction<'_>, tx: Entid) -> Result> { let mut stmt = db_tx.prepare_cached("SELECT uuid FROM tolstoy_tu WHERE tx = ?")?; let results = stmt.query_and_then(&[&tx], |r| -> Result { let bytes: Vec = r.get(0).unwrap(); Ok(Uuid::from_bytes(bytes.as_slice().try_into().unwrap())) })?; let mut uuids = vec![]; uuids.extend(results); if uuids.is_empty() { return Ok(None); } else if uuids.len() > 1 { bail!(TolstoyError::TxIncorrectlyMapped(uuids.len())); } Ok(Some(uuids.remove(0)?)) } } #[cfg(test)] pub mod tests { use super::*; use crate::schema; #[test] fn test_getters() { let mut conn = schema::tests::setup_conn_bare(); let mut tx = schema::tests::setup_tx(&mut conn); assert_eq!(None, TxMapper::get(&tx, 1).expect("success")); let set_uuid = TxMapper::get_or_set_uuid_for_tx(&mut tx, 1).expect("success"); assert_eq!(Some(set_uuid), TxMapper::get(&tx, 1).expect("success")); } #[test] fn test_bulk_setter() { let mut conn = schema::tests::setup_conn_bare(); let mut tx = schema::tests::setup_tx(&mut conn); TxMapper::set_lg_mappings(&mut tx, vec![]).expect("empty map success"); let uuid1 = Uuid::new_v4(); let uuid2 = Uuid::new_v4(); TxMapper::set_lg_mappings(&mut tx, vec![(1, &uuid1).into(), (2, &uuid2).into()]) .expect("map success"); assert_eq!(Some(uuid1), TxMapper::get(&tx, 1).expect("success")); assert_eq!(Some(uuid2), TxMapper::get(&tx, 2).expect("success")); // Now let's replace one of the mappings. let new_uuid2 = Uuid::new_v4(); TxMapper::set_lg_mappings(&mut tx, vec![(1, &uuid1).into(), (2, &new_uuid2).into()]) .expect("map success"); assert_eq!(Some(uuid1), TxMapper::get(&tx, 1).expect("success")); assert_eq!(Some(new_uuid2), TxMapper::get(&tx, 2).expect("success")); } }