use crate::data::DynamoDbClient;
use crate::db_connectors::dynamodb::{Interaction, InteractionDeleteInfo, DynamoDbKey};
use crate::{encrypt::encrypt_data, Client, EngineError};
use rusoto_dynamodb::*;
use uuid::Uuid;
use std::collections::HashMap;
use crate::db_connectors::dynamodb::utils::*;
pub fn init_interaction(
event: serde_json::Value,
client: &Client,
db: &mut DynamoDbClient,
) -> Result<String, EngineError> {
let id = Uuid::new_v4();
let encrypted_event = encrypt_data(&event)?;
let interaction = Interaction::new(&id, client, &encrypted_event);
let expr_attr_names = [
(String::from("#hashKey"), String::from("hash")),
(String::from("#rangeKey"), String::from("range")),
]
.iter()
.cloned()
.collect();
let expr_attr_values = [
(
String::from(":hashVal"),
AttributeValue {
s: Some(interaction.hash.to_owned()),
..Default::default()
},
),
(
String::from(":rangeVal"),
AttributeValue {
s: Some(interaction.range.to_owned()),
..Default::default()
},
),
]
.iter()
.cloned()
.collect();
let input = PutItemInput {
table_name: get_table_name()?,
item: serde_dynamodb::to_hashmap(&interaction)?,
condition_expression: Some("#hashKey <> :hashVal AND #rangeKey <> :rangeVal".to_owned()),
expression_attribute_names: Some(expr_attr_names),
expression_attribute_values: Some(expr_attr_values),
..Default::default()
};
let future = db.client.put_item(input);
match db.runtime.block_on(future) {
Ok(_) => {},
Err(e) => {
return Err(EngineError::Manager(format!("init_interaction {:?}", e)))
}
};
Ok(id.to_string())
}
pub fn update_interaction(
interaction_id: &str,
success: bool,
client: &Client,
db: &mut DynamoDbClient,
) -> Result<(), EngineError> {
let key = Interaction::get_key(client, interaction_id);
let expr_attr_names = [
(String::from("#hashKey"), String::from("hash")),
(String::from("#rangeKey"), String::from("range")),
(String::from("#successKey"), String::from("success")),
(String::from("#updatedAtKey"), String::from("updated_at")),
]
.iter()
.cloned()
.collect();
let expr_attr_values = [
(
String::from(":hashVal"),
AttributeValue {
s: Some(key.hash.to_owned()),
..Default::default()
},
),
(
String::from(":rangeVal"),
AttributeValue {
s: Some(key.range.to_owned()),
..Default::default()
},
),
(
String::from(":successVal"),
AttributeValue {
bool: Some(success),
..Default::default()
},
),
(
String::from(":updatedAtVal"),
AttributeValue {
s: Some(get_date_time()),
..Default::default()
},
),
]
.iter()
.cloned()
.collect();
let condition_expr = "#hashKey = :hashVal AND #rangeKey = :rangeVal".to_string();
let update_expr = "SET #updatedAtKey = :updatedAtVal, #successKey = :successVal".to_string();
let input = UpdateItemInput {
table_name: get_table_name()?,
key: serde_dynamodb::to_hashmap(&key)?,
condition_expression: Some(condition_expr),
update_expression: Some(update_expr),
expression_attribute_names: Some(expr_attr_names),
expression_attribute_values: Some(expr_attr_values),
..Default::default()
};
let future = db.client.update_item(input);
match db.runtime.block_on(future) {
Ok(_) => (),
Err(e) => {
return Err(EngineError::Manager(format!("update_interaction {:?}", e)))
}
};
Ok(())
}
fn query_interactions(
client: &Client,
db: &mut DynamoDbClient,
limit: i64,
pagination_key: Option<HashMap<String, AttributeValue>>,
projection_expression: Option<String>,
expression_attribute_names: Option<HashMap<String, String>>
) -> Result<QueryOutput, EngineError> {
let hash = Interaction::get_hash(client);
let expr_attr_values = [
(
String::from(":hashVal"),
AttributeValue {
s: Some(hash),
..Default::default()
},
),
(
String::from(":rangePrefix"),
AttributeValue {
s: Some(String::from("interaction#")),
..Default::default()
},
),
]
.iter()
.cloned()
.collect();
let input = QueryInput {
table_name: get_table_name()?,
key_condition_expression: Some(
"#hashKey = :hashVal AND begins_with(#rangeKey, :rangePrefix)".to_owned(),
),
expression_attribute_names,
expression_attribute_values: Some(expr_attr_values),
limit: Some(limit),
exclusive_start_key: pagination_key,
scan_index_forward: Some(false),
projection_expression,
..Default::default()
};
let future = db.client.query(input);
let data = match db.runtime.block_on(future) {
Ok(data) => data,
Err(e) => {
return Err(EngineError::Manager(format!("query_interactions {:?}", e)))
}
};
Ok(data)
}
pub fn delete_user_interactions(client: &Client, db: &mut DynamoDbClient) -> Result<(), EngineError> {
let mut pagination_key = None;
let expr_attr_names: HashMap<String, String> = [
("#hashKey".to_string(), "hash".to_string()),
("#rangeKey".to_string(), "range".to_string()),
("#id".to_string(), "id".to_string()),
]
.iter()
.cloned()
.collect();
loop {
let data = query_interactions(
client,
db,
25,
pagination_key,
Some("#id".to_owned()),
Some(expr_attr_names.clone()),
)?;
let items = match data.items {
None => return Ok(()),
Some(items) if items.len() == 0 => return Ok(()),
Some(items) => items.clone(),
};
let mut write_requests = vec![];
for item in items {
let interaction: InteractionDeleteInfo = serde_dynamodb::from_hashmap(item.to_owned())?;
let key = serde_dynamodb::to_hashmap(&DynamoDbKey {
hash: Interaction::get_hash(client),
range: Interaction::get_range(&interaction.id),
})?;
write_requests.push(WriteRequest {
delete_request: Some(DeleteRequest {key}),
put_request: None,
});
}
let request_items = [(get_table_name()?, write_requests)]
.iter()
.cloned()
.collect();
let input = BatchWriteItemInput {
request_items,
..Default::default()
};
execute_batch_write_query(db, input)?;
pagination_key = data.last_evaluated_key;
if let None = &pagination_key {
return Ok(())
}
}
}