use crate::db::{
models::{
credit_card::{InternalCreditCard, UpdatableCreditCardFields},
Metadata,
},
schema::{CREDIT_CARD_COMMON_COLS, CREDIT_CARD_COMMON_VALS},
};
use crate::error::*;
use rusqlite::{Connection, Transaction};
use sync_guid::Guid;
use types::Timestamp;
pub(crate) fn add_credit_card(
conn: &Connection,
new_credit_card_fields: UpdatableCreditCardFields,
) -> Result<InternalCreditCard> {
let now = Timestamp::now();
let credit_card = InternalCreditCard {
guid: Guid::random(),
cc_name: new_credit_card_fields.cc_name,
cc_number_enc: new_credit_card_fields.cc_number_enc,
cc_number_last_4: new_credit_card_fields.cc_number_last_4,
cc_exp_month: new_credit_card_fields.cc_exp_month,
cc_exp_year: new_credit_card_fields.cc_exp_year,
cc_type: new_credit_card_fields.cc_type,
metadata: Metadata {
time_created: now,
time_last_modified: now,
..Default::default()
},
};
let tx = conn.unchecked_transaction()?;
add_internal_credit_card(&tx, &credit_card)?;
tx.commit()?;
Ok(credit_card)
}
pub(crate) fn add_internal_credit_card(
tx: &Transaction<'_>,
card: &InternalCreditCard,
) -> Result<()> {
tx.execute(
&format!(
"INSERT INTO credit_cards_data (
{common_cols},
sync_change_counter
) VALUES (
{common_vals},
:sync_change_counter
)",
common_cols = CREDIT_CARD_COMMON_COLS,
common_vals = CREDIT_CARD_COMMON_VALS,
),
rusqlite::named_params! {
":guid": card.guid,
":cc_name": card.cc_name,
":cc_number_enc": card.cc_number_enc,
":cc_number_last_4": card.cc_number_last_4,
":cc_exp_month": card.cc_exp_month,
":cc_exp_year": card.cc_exp_year,
":cc_type": card.cc_type,
":time_created": card.metadata.time_created,
":time_last_used": card.metadata.time_last_used,
":time_last_modified": card.metadata.time_last_modified,
":times_used": card.metadata.times_used,
":sync_change_counter": card.metadata.sync_change_counter,
},
)?;
Ok(())
}
pub(crate) fn get_credit_card(conn: &Connection, guid: &Guid) -> Result<InternalCreditCard> {
let sql = format!(
"SELECT
{common_cols},
sync_change_counter
FROM credit_cards_data
WHERE guid = :guid",
common_cols = CREDIT_CARD_COMMON_COLS
);
conn.query_row(&sql, [guid], InternalCreditCard::from_row)
.map_err(|e| match e {
rusqlite::Error::QueryReturnedNoRows => Error::NoSuchRecord(guid.to_string()),
e => e.into(),
})
}
pub(crate) fn get_all_credit_cards(conn: &Connection) -> Result<Vec<InternalCreditCard>> {
let sql = format!(
"SELECT
{common_cols},
sync_change_counter
FROM credit_cards_data",
common_cols = CREDIT_CARD_COMMON_COLS
);
let mut stmt = conn.prepare(&sql)?;
let credit_cards = stmt
.query_map([], InternalCreditCard::from_row)?
.collect::<std::result::Result<Vec<InternalCreditCard>, _>>()?;
Ok(credit_cards)
}
pub fn update_credit_card(
conn: &Connection,
guid: &Guid,
credit_card: &UpdatableCreditCardFields,
) -> Result<()> {
let tx = conn.unchecked_transaction()?;
tx.execute(
"UPDATE credit_cards_data
SET cc_name = :cc_name,
cc_number_enc = :cc_number_enc,
cc_number_last_4 = :cc_number_last_4,
cc_exp_month = :cc_exp_month,
cc_exp_year = :cc_exp_year,
cc_type = :cc_type,
time_last_modified = :time_last_modified,
sync_change_counter = sync_change_counter + 1
WHERE guid = :guid",
rusqlite::named_params! {
":cc_name": credit_card.cc_name,
":cc_number_enc": credit_card.cc_number_enc,
":cc_number_last_4": credit_card.cc_number_last_4,
":cc_exp_month": credit_card.cc_exp_month,
":cc_exp_year": credit_card.cc_exp_year,
":cc_type": credit_card.cc_type,
":time_last_modified": Timestamp::now(),
":guid": guid,
},
)?;
tx.commit()?;
Ok(())
}
pub(crate) fn update_internal_credit_card(
tx: &Transaction<'_>,
card: &InternalCreditCard,
flag_as_changed: bool,
) -> Result<()> {
let change_counter_increment = flag_as_changed as u32; tx.execute(
"UPDATE credit_cards_data
SET cc_name = :cc_name,
cc_number_enc = :cc_number_enc,
cc_number_last_4 = :cc_number_last_4,
cc_exp_month = :cc_exp_month,
cc_exp_year = :cc_exp_year,
cc_type = :cc_type,
time_created = :time_created,
time_last_used = :time_last_used,
time_last_modified = :time_last_modified,
times_used = :times_used,
sync_change_counter = sync_change_counter + :change_incr
WHERE guid = :guid",
rusqlite::named_params! {
":cc_name": card.cc_name,
":cc_number_enc": card.cc_number_enc,
":cc_number_last_4": card.cc_number_last_4,
":cc_exp_month": card.cc_exp_month,
":cc_exp_year": card.cc_exp_year,
":cc_type": card.cc_type,
":time_created": card.metadata.time_created,
":time_last_used": card.metadata.time_last_used,
":time_last_modified": card.metadata.time_last_modified,
":times_used": card.metadata.times_used,
":change_incr": change_counter_increment,
":guid": card.guid,
},
)?;
Ok(())
}
pub fn delete_credit_card(conn: &Connection, guid: &Guid) -> Result<bool> {
let tx = conn.unchecked_transaction()?;
let exists = tx.execute(
"DELETE FROM credit_cards_data
WHERE guid = :guid",
rusqlite::named_params! {
":guid": guid.as_str(),
},
)? != 0;
tx.commit()?;
Ok(exists)
}
pub fn scrub_encrypted_credit_card_data(conn: &Connection) -> Result<()> {
let tx = conn.unchecked_transaction()?;
tx.execute("UPDATE credit_cards_data SET cc_number_enc = ''", [])?;
tx.commit()?;
Ok(())
}
pub fn touch(conn: &Connection, guid: &Guid) -> Result<()> {
let tx = conn.unchecked_transaction()?;
let now_ms = Timestamp::now();
tx.execute(
"UPDATE credit_cards_data
SET time_last_used = :time_last_used,
times_used = times_used + 1,
sync_change_counter = sync_change_counter + 1
WHERE guid = :guid",
rusqlite::named_params! {
":time_last_used": now_ms,
":guid": guid.as_str(),
},
)?;
tx.commit()?;
Ok(())
}
#[cfg(test)]
pub(crate) mod tests {
use super::*;
use crate::db::test::new_mem_db;
use crate::encryption::EncryptorDecryptor;
use sync15::bso::IncomingBso;
pub fn get_all(
conn: &Connection,
table_name: String,
) -> rusqlite::Result<Vec<String>, rusqlite::Error> {
let mut stmt = conn.prepare(&format!(
"SELECT guid FROM {table_name}",
table_name = table_name
))?;
let rows = stmt.query_map([], |row| row.get(0))?;
let mut guids = Vec::new();
for guid_result in rows {
guids.push(guid_result?);
}
Ok(guids)
}
pub fn insert_tombstone_record(
conn: &Connection,
guid: String,
) -> rusqlite::Result<usize, rusqlite::Error> {
conn.execute(
"INSERT INTO credit_cards_tombstones (
guid,
time_deleted
) VALUES (
:guid,
:time_deleted
)",
rusqlite::named_params! {
":guid": guid,
":time_deleted": Timestamp::now(),
},
)
}
pub(crate) fn test_insert_mirror_record(conn: &Connection, bso: IncomingBso) {
conn.execute(
"INSERT INTO credit_cards_mirror (guid, payload)
VALUES (:guid, :payload)",
rusqlite::named_params! {
":guid": &bso.envelope.id,
":payload": &bso.payload,
},
)
.expect("should insert");
}
#[test]
fn test_credit_card_create_and_read() -> Result<()> {
let db = new_mem_db();
let saved_credit_card = add_credit_card(
&db,
UpdatableCreditCardFields {
cc_name: "jane doe".to_string(),
cc_number_enc: "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX".to_string(),
cc_number_last_4: "1234".to_string(),
cc_exp_month: 3,
cc_exp_year: 2022,
cc_type: "visa".to_string(),
},
)?;
assert_ne!(Guid::default(), saved_credit_card.guid);
assert_ne!(0, saved_credit_card.metadata.time_created.as_millis());
assert_ne!(0, saved_credit_card.metadata.time_last_modified.as_millis());
assert_eq!(0, saved_credit_card.metadata.sync_change_counter);
let retrieved_credit_card = get_credit_card(&db, &saved_credit_card.guid)?;
assert_eq!(saved_credit_card.guid, retrieved_credit_card.guid);
assert_eq!(saved_credit_card.cc_name, retrieved_credit_card.cc_name);
assert_eq!(
saved_credit_card.cc_number_enc,
retrieved_credit_card.cc_number_enc
);
assert_eq!(
saved_credit_card.cc_number_last_4,
retrieved_credit_card.cc_number_last_4
);
assert_eq!(
saved_credit_card.cc_exp_month,
retrieved_credit_card.cc_exp_month
);
assert_eq!(
saved_credit_card.cc_exp_year,
retrieved_credit_card.cc_exp_year
);
assert_eq!(saved_credit_card.cc_type, retrieved_credit_card.cc_type);
let delete_result = delete_credit_card(&db, &saved_credit_card.guid);
assert!(delete_result.is_ok());
assert!(delete_result?);
assert!(get_credit_card(&db, &saved_credit_card.guid).is_err());
Ok(())
}
#[test]
fn test_credit_card_missing_guid() {
let db = new_mem_db();
let guid = Guid::random();
let result = get_credit_card(&db, &guid);
assert_eq!(
result.unwrap_err().to_string(),
Error::NoSuchRecord(guid.to_string()).to_string()
);
}
#[test]
fn test_credit_card_read_all() -> Result<()> {
let db = new_mem_db();
let saved_credit_card = add_credit_card(
&db,
UpdatableCreditCardFields {
cc_name: "jane doe".to_string(),
cc_number_enc: "YYYYYYYYYYYYYYYYYYYYYYYYYYYYY".to_string(),
cc_number_last_4: "4321".to_string(),
cc_exp_month: 3,
cc_exp_year: 2022,
cc_type: "visa".to_string(),
},
)?;
let saved_credit_card2 = add_credit_card(
&db,
UpdatableCreditCardFields {
cc_name: "john deer".to_string(),
cc_number_enc: "ZZZZZZZZZZZZZZZZZZZZZZZZZZZZZ".to_string(),
cc_number_last_4: "6543".to_string(),
cc_exp_month: 10,
cc_exp_year: 2025,
cc_type: "mastercard".to_string(),
},
)?;
let saved_credit_card3 = add_credit_card(
&db,
UpdatableCreditCardFields {
cc_name: "abraham lincoln".to_string(),
cc_number_enc: "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA".to_string(),
cc_number_last_4: "9876".to_string(),
cc_exp_month: 1,
cc_exp_year: 2024,
cc_type: "amex".to_string(),
},
)?;
let delete_result = delete_credit_card(&db, &saved_credit_card3.guid);
assert!(delete_result.is_ok());
assert!(delete_result?);
let retrieved_credit_cards = get_all_credit_cards(&db)?;
assert!(!retrieved_credit_cards.is_empty());
let expected_number_of_credit_cards = 2;
assert_eq!(
expected_number_of_credit_cards,
retrieved_credit_cards.len()
);
let retrieved_credit_card_guids = [
retrieved_credit_cards[0].guid.as_str(),
retrieved_credit_cards[1].guid.as_str(),
];
assert!(retrieved_credit_card_guids.contains(&saved_credit_card.guid.as_str()));
assert!(retrieved_credit_card_guids.contains(&saved_credit_card2.guid.as_str()));
Ok(())
}
#[test]
fn test_credit_card_update() -> Result<()> {
let db = new_mem_db();
let saved_credit_card = add_credit_card(
&db,
UpdatableCreditCardFields {
cc_name: "john deer".to_string(),
cc_number_enc: "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA".to_string(),
cc_number_last_4: "4321".to_string(),
cc_exp_month: 10,
cc_exp_year: 2025,
cc_type: "mastercard".to_string(),
},
)?;
let expected_cc_name = "john doe".to_string();
let update_result = update_credit_card(
&db,
&saved_credit_card.guid,
&UpdatableCreditCardFields {
cc_name: expected_cc_name.clone(),
cc_number_enc: "BBBBBBBBBBBBBBBBBBBBBBBBBBBBBB".to_string(),
cc_number_last_4: "1234".to_string(),
cc_type: "mastercard".to_string(),
cc_exp_month: 10,
cc_exp_year: 2025,
},
);
assert!(update_result.is_ok());
let updated_credit_card = get_credit_card(&db, &saved_credit_card.guid)?;
assert_eq!(saved_credit_card.guid, updated_credit_card.guid);
assert_eq!(expected_cc_name, updated_credit_card.cc_name);
assert_eq!(1, updated_credit_card.metadata.sync_change_counter);
Ok(())
}
#[test]
fn test_credit_card_update_internal_credit_card() -> Result<()> {
let mut db = new_mem_db();
let tx = db.transaction()?;
let guid = Guid::random();
add_internal_credit_card(
&tx,
&InternalCreditCard {
guid: guid.clone(),
cc_name: "john deer".to_string(),
cc_number_enc: "BBBBBBBBBBBBBBBBBBBBBBBBBBBBBB".to_string(),
cc_number_last_4: "1234".to_string(),
cc_exp_month: 10,
cc_exp_year: 2025,
cc_type: "mastercard".to_string(),
..Default::default()
},
)?;
let expected_cc_exp_month = 11;
update_internal_credit_card(
&tx,
&InternalCreditCard {
guid: guid.clone(),
cc_name: "john deer".to_string(),
cc_number_enc: "BBBBBBBBBBBBBBBBBBBBBBBBBBBBBB".to_string(),
cc_number_last_4: "1234".to_string(),
cc_exp_month: expected_cc_exp_month,
cc_exp_year: 2025,
cc_type: "mastercard".to_string(),
..Default::default()
},
false,
)?;
let record_exists: bool = tx.query_row(
"SELECT EXISTS (
SELECT 1
FROM credit_cards_data
WHERE guid = :guid
AND cc_exp_month = :cc_exp_month
AND sync_change_counter = 0
)",
[&guid.to_string(), &expected_cc_exp_month.to_string()],
|row| row.get(0),
)?;
assert!(record_exists);
Ok(())
}
#[test]
fn test_credit_card_delete() -> Result<()> {
let db = new_mem_db();
let encdec = EncryptorDecryptor::new_with_random_key().unwrap();
let saved_credit_card = add_credit_card(
&db,
UpdatableCreditCardFields {
cc_name: "john deer".to_string(),
cc_number_enc: encdec.encrypt("1234567812345678", "cc_number")?,
cc_number_last_4: "5678".to_string(),
cc_exp_month: 10,
cc_exp_year: 2025,
cc_type: "mastercard".to_string(),
},
)?;
let delete_result = delete_credit_card(&db, &saved_credit_card.guid);
assert!(delete_result.is_ok());
assert!(delete_result?);
let saved_credit_card2 = add_credit_card(
&db,
UpdatableCreditCardFields {
cc_name: "john doe".to_string(),
cc_number_enc: encdec.encrypt("1234123412341234", "cc_number")?,
cc_number_last_4: "1234".to_string(),
cc_exp_month: 5,
cc_exp_year: 2024,
cc_type: "visa".to_string(),
},
)?;
let cc2_guid = saved_credit_card2.guid.clone();
let payload = saved_credit_card2.into_test_incoming_bso(&encdec, Default::default());
test_insert_mirror_record(&db, payload);
let delete_result2 = delete_credit_card(&db, &cc2_guid);
assert!(delete_result2.is_ok());
assert!(delete_result2?);
let tombstone_exists: bool = db.query_row(
"SELECT EXISTS (
SELECT 1
FROM credit_cards_tombstones
WHERE guid = :guid
)",
[&cc2_guid],
|row| row.get(0),
)?;
assert!(tombstone_exists);
db.execute(
"DELETE FROM credit_cards_tombstones
WHERE guid = :guid",
rusqlite::named_params! {
":guid": cc2_guid,
},
)?;
Ok(())
}
#[test]
fn test_scrub_encrypted_credit_card_data() -> Result<()> {
let db = new_mem_db();
let encdec = EncryptorDecryptor::new_with_random_key().unwrap();
let mut saved_credit_cards = Vec::with_capacity(10);
for _ in 0..5 {
saved_credit_cards.push(add_credit_card(
&db,
UpdatableCreditCardFields {
cc_name: "john deer".to_string(),
cc_number_enc: encdec.encrypt("1234567812345678", "cc_number")?,
cc_number_last_4: "5678".to_string(),
cc_exp_month: 10,
cc_exp_year: 2025,
cc_type: "mastercard".to_string(),
},
)?);
}
scrub_encrypted_credit_card_data(&db)?;
for saved_credit_card in saved_credit_cards.into_iter() {
let retrieved_credit_card = get_credit_card(&db, &saved_credit_card.guid)?;
assert_eq!(retrieved_credit_card.cc_number_enc, "");
}
Ok(())
}
#[test]
fn test_credit_card_trigger_on_create() -> Result<()> {
let db = new_mem_db();
let tx = db.unchecked_transaction()?;
let guid = Guid::random();
insert_tombstone_record(&db, guid.to_string())?;
let credit_card = InternalCreditCard {
guid,
cc_name: "john deer".to_string(),
cc_number_enc: "WWWWWWWWWWWWWWWWWWWWWWWWWWWWWWW".to_string(),
cc_number_last_4: "6543".to_string(),
cc_exp_month: 10,
cc_exp_year: 2025,
cc_type: "mastercard".to_string(),
..Default::default()
};
let add_credit_card_result = add_internal_credit_card(&tx, &credit_card);
assert!(add_credit_card_result.is_err());
let expected_error_message = "guid exists in `credit_cards_tombstones`";
assert!(add_credit_card_result
.unwrap_err()
.to_string()
.contains(expected_error_message));
Ok(())
}
#[test]
fn test_credit_card_trigger_on_delete() -> Result<()> {
let db = new_mem_db();
let tx = db.unchecked_transaction()?;
let guid = Guid::random();
let credit_card = InternalCreditCard {
guid,
cc_name: "jane doe".to_string(),
cc_number_enc: "WWWWWWWWWWWWWWWWWWWWWWWWWWWWWWW".to_string(),
cc_number_last_4: "6543".to_string(),
cc_exp_month: 3,
cc_exp_year: 2022,
cc_type: "visa".to_string(),
..Default::default()
};
add_internal_credit_card(&tx, &credit_card)?;
let tombstone_result = insert_tombstone_record(&db, credit_card.guid.to_string());
let expected_error_message = "guid exists in `credit_cards_data`";
assert!(tombstone_result
.unwrap_err()
.to_string()
.contains(expected_error_message));
Ok(())
}
#[test]
fn test_credit_card_touch() -> Result<()> {
let db = new_mem_db();
let saved_credit_card = add_credit_card(
&db,
UpdatableCreditCardFields {
cc_name: "john doe".to_string(),
cc_number_enc: "WWWWWWWWWWWWWWWWWWWWWWWWWWWWWWW".to_string(),
cc_number_last_4: "6543".to_string(),
cc_exp_month: 5,
cc_exp_year: 2024,
cc_type: "visa".to_string(),
},
)?;
assert_eq!(saved_credit_card.metadata.sync_change_counter, 0);
assert_eq!(saved_credit_card.metadata.times_used, 0);
touch(&db, &saved_credit_card.guid)?;
let touched_credit_card = get_credit_card(&db, &saved_credit_card.guid)?;
assert_eq!(touched_credit_card.metadata.sync_change_counter, 1);
assert_eq!(touched_credit_card.metadata.times_used, 1);
Ok(())
}
}