use std::collections::HashMap;
use std::sync::atomic::{AtomicU64, Ordering};
use chrono::Utc;
use super::StatsigUserLoggable;
use crate::evaluation::dynamic_value::DynamicValue;
use crate::hashing::djb2_number;
use crate::{evaluation::dynamic_string::DynamicString, Statsig};
use crate::{log_w, statsig_metadata, StatsigUser};
pub type FullUserKey = (
u64, u64, u64, u64, u64, u64, u64, Vec<u64>, Vec<u64>, Vec<u64>, Vec<u64>, );
const TAG: &str = stringify!(StatsigUserInternal);
const VERSION_CHECK_THROTTLE_MS: u64 = 60_000;
#[derive(Clone)]
pub struct StatsigUserInternal<'statsig, 'user> {
pub user_ref: &'user StatsigUser,
pub statsig_instance: Option<&'statsig Statsig>,
}
static LAST_VERSION_CHECK: AtomicU64 = AtomicU64::new(0);
impl<'statsig, 'user> StatsigUserInternal<'statsig, 'user> {
pub fn new(user: &'user StatsigUser, statsig_instance: Option<&'statsig Statsig>) -> Self {
throttled_version_check(user);
Self {
user_ref: user,
statsig_instance,
}
}
pub fn get_unit_id(&self, id_type: &DynamicString) -> Option<&DynamicValue> {
self.user_ref.get_unit_id(id_type)
}
pub fn get_user_value(&self, field: &Option<DynamicString>) -> Option<&DynamicValue> {
let field = field.as_ref()?;
let lowered_field = &field.lowercased_value;
let str_value = match lowered_field as &str {
"userid" => &self.user_ref.data.user_id,
"email" => &self.user_ref.data.email,
"ip" => &self.user_ref.data.ip,
"country" => &self.user_ref.data.country,
"locale" => &self.user_ref.data.locale,
"appversion" => &self.user_ref.data.app_version,
"useragent" => &self.user_ref.data.user_agent,
_ => &None,
};
if let Some(value) = str_value {
if let Some(str_val) = &value.string_value {
if !str_val.value.is_empty() {
return Some(value);
}
}
}
if let Some(custom) = &self.user_ref.data.custom {
if let Some(found) = custom.get(field.value.as_str()) {
return Some(found);
}
if let Some(lowered_found) = custom.get(lowered_field.as_str()) {
return Some(lowered_found);
}
}
if let Some(instance) = &self.statsig_instance {
if let Some(val) = instance.get_value_from_global_custom_fields(&field.value) {
return Some(val);
}
if let Some(val) = instance.get_value_from_global_custom_fields(&field.lowercased_value)
{
return Some(val);
}
}
if let Some(private_attributes) = &self.user_ref.data.private_attributes {
if let Some(found) = private_attributes.get(field.value.as_str()) {
return Some(found);
}
if let Some(lowered_found) = private_attributes.get(lowered_field.as_str()) {
return Some(lowered_found);
}
}
let str_value_alt = match lowered_field as &str {
"user_id" => &self.user_ref.data.user_id,
"app_version" => &self.user_ref.data.app_version,
"user_agent" => &self.user_ref.data.user_agent,
_ => &None,
};
if str_value_alt.is_some() {
return str_value_alt.as_ref();
}
None
}
pub fn get_value_from_environment(
&self,
field: &Option<DynamicString>,
) -> Option<DynamicValue> {
let field = field.as_ref()?;
if let Some(statsig_environment) = &self.user_ref.data.statsig_environment {
if let Some(result) = statsig_environment.get(field.value.as_str()) {
return Some(result.clone());
}
}
if let Some(result) = self.statsig_instance?.get_from_statsig_env(&field.value) {
return Some(result);
}
self.statsig_instance?
.get_from_statsig_env(&field.lowercased_value)
}
pub fn to_loggable(&self) -> StatsigUserLoggable {
let mut environment = self.user_ref.data.statsig_environment.clone();
let mut global_custom: Option<HashMap<String, DynamicValue>> = None;
if let Some(statsig_instance) = &self.statsig_instance {
if environment.is_none() {
environment = statsig_instance.use_statsig_env(|e| e.cloned());
}
global_custom = statsig_instance.use_global_custom_fields(|gc| gc.cloned());
}
StatsigUserLoggable::new(&self.user_ref.data, environment, global_custom)
}
pub fn get_hashed_private_attributes(&self) -> Option<String> {
let private_attributes = match &self.user_ref.data.private_attributes {
Some(attrs) => attrs,
None => return None,
};
if private_attributes.is_empty() {
return None;
}
let mut val: i64 = 0;
for (key, value) in private_attributes {
let hash_key = match value.string_value {
Some(ref s) => key.to_owned() + ":" + &s.value,
None => key.to_owned() + ":",
};
val += djb2_number(&hash_key);
val &= 0xFFFF_FFFF;
}
Some(val.to_string())
}
}
fn throttled_version_check(user: &StatsigUser) {
let current_version = statsig_metadata::SDK_VERSION;
if user.sdk_version.as_ptr() == current_version.as_ptr() {
return;
}
if user.sdk_version == current_version {
return;
}
let now = Utc::now().timestamp_millis() as u64;
let last = LAST_VERSION_CHECK.load(Ordering::Relaxed);
if now.saturating_sub(last) < VERSION_CHECK_THROTTLE_MS {
return;
}
if LAST_VERSION_CHECK
.compare_exchange(last, now, Ordering::Relaxed, Ordering::Relaxed)
.is_ok()
{
log_w!(
TAG,
"Multiple SDK versions detected. This may cause unexpected behavior. Expected: {}, Got: {}",
statsig_metadata::SDK_VERSION,
user.sdk_version
);
}
}