use crate::{atomics::AtomicU64, cow::Cow, IntoLabels, KeyHasher, Label, SharedString};
use std::{
borrow::Borrow,
cmp, fmt,
hash::{Hash, Hasher},
slice::Iter,
sync::atomic::{AtomicBool, Ordering},
};
const NO_LABELS: [Label; 0] = [];
#[derive(Clone, Debug, Eq, Hash, PartialEq, PartialOrd, Ord)]
pub struct KeyName(SharedString);
impl KeyName {
pub const fn from_const_str(name: &'static str) -> Self {
KeyName(SharedString::const_str(name))
}
pub fn as_str(&self) -> &str {
&self.0
}
}
impl<T> From<T> for KeyName
where
T: Into<SharedString>,
{
fn from(name: T) -> Self {
KeyName(name.into())
}
}
impl Borrow<str> for KeyName {
fn borrow(&self) -> &str {
self.0.borrow()
}
}
#[derive(Debug)]
pub struct Key {
name: KeyName,
labels: Cow<'static, [Label]>,
hashed: AtomicBool,
hash: AtomicU64,
}
impl Key {
pub fn from_name<N>(name: N) -> Self
where
N: Into<KeyName>,
{
let name = name.into();
let labels = Cow::from_owned(Vec::new());
Self::builder(name, labels)
}
pub fn from_parts<N, L>(name: N, labels: L) -> Self
where
N: Into<KeyName>,
L: IntoLabels,
{
let name = name.into();
let labels = Cow::from_owned(labels.into_labels());
Self::builder(name, labels)
}
pub fn from_static_labels<N>(name: N, labels: &'static [Label]) -> Self
where
N: Into<KeyName>,
{
Self {
name: name.into(),
labels: Cow::const_slice(labels),
hashed: AtomicBool::new(false),
hash: AtomicU64::new(0),
}
}
pub const fn from_static_name(name: &'static str) -> Self {
Self::from_static_parts(name, &NO_LABELS)
}
pub const fn from_static_parts(name: &'static str, labels: &'static [Label]) -> Self {
Self {
name: KeyName::from_const_str(name),
labels: Cow::const_slice(labels),
hashed: AtomicBool::new(false),
hash: AtomicU64::new(0),
}
}
fn builder(name: KeyName, labels: Cow<'static, [Label]>) -> Self {
let hash = generate_key_hash(&name, &labels);
Self { name, labels, hashed: AtomicBool::new(true), hash: AtomicU64::new(hash) }
}
pub fn name(&self) -> &str {
self.name.0.as_ref()
}
pub fn labels(&self) -> Iter<Label> {
self.labels.iter()
}
pub fn into_parts(self) -> (KeyName, Vec<Label>) {
(self.name, self.labels.into_owned())
}
pub fn with_extra_labels(&self, extra_labels: Vec<Label>) -> Self {
if extra_labels.is_empty() {
return self.clone();
}
let name = self.name.clone();
let mut labels = self.labels.clone().into_owned();
labels.extend(extra_labels);
Self::builder(name, labels.into())
}
pub fn get_hash(&self) -> u64 {
if self.hashed.load(Ordering::Acquire) {
self.hash.load(Ordering::Acquire)
} else {
let hash = generate_key_hash(&self.name, &self.labels);
self.hash.store(hash, Ordering::Release);
self.hashed.store(true, Ordering::Release);
hash
}
}
}
fn generate_key_hash(name: &KeyName, labels: &Cow<'static, [Label]>) -> u64 {
let mut hasher = KeyHasher::default();
key_hasher_impl(&mut hasher, name, labels);
hasher.finish()
}
fn key_hasher_impl<H: Hasher>(state: &mut H, name: &KeyName, labels: &Cow<'static, [Label]>) {
name.0.hash(state);
labels.hash(state);
}
impl Clone for Key {
fn clone(&self) -> Self {
Self {
name: self.name.clone(),
labels: self.labels.clone(),
hashed: AtomicBool::new(self.hashed.load(Ordering::Acquire)),
hash: AtomicU64::new(self.hash.load(Ordering::Acquire)),
}
}
}
impl PartialEq for Key {
fn eq(&self, other: &Self) -> bool {
self.name == other.name && self.labels == other.labels
}
}
impl Eq for Key {}
impl PartialOrd for Key {
fn partial_cmp(&self, other: &Self) -> Option<cmp::Ordering> {
Some(self.cmp(other))
}
}
impl Ord for Key {
fn cmp(&self, other: &Self) -> cmp::Ordering {
(&self.name, &self.labels).cmp(&(&other.name, &other.labels))
}
}
impl Hash for Key {
fn hash<H: Hasher>(&self, state: &mut H) {
key_hasher_impl(state, &self.name, &self.labels);
}
}
impl fmt::Display for Key {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
if self.labels.is_empty() {
write!(f, "Key({})", self.name.as_str())
} else {
write!(f, "Key({}, [", self.name.as_str())?;
let mut first = true;
for label in self.labels.as_ref() {
if first {
write!(f, "{} = {}", label.0, label.1)?;
first = false;
} else {
write!(f, ", {} = {}", label.0, label.1)?;
}
}
write!(f, "])")
}
}
}
impl<T> From<T> for Key
where
T: Into<KeyName>,
{
fn from(name: T) -> Self {
Self::from_name(name)
}
}
impl<N, L> From<(N, L)> for Key
where
N: Into<KeyName>,
L: IntoLabels,
{
fn from(parts: (N, L)) -> Self {
Self::from_parts(parts.0, parts.1)
}
}
#[cfg(test)]
mod tests {
use super::Key;
use crate::{KeyName, Label};
use std::{collections::HashMap, ops::Deref, sync::Arc};
static BORROWED_NAME: &'static str = "name";
static FOOBAR_NAME: &'static str = "foobar";
static BORROWED_BASIC: Key = Key::from_static_name(&BORROWED_NAME);
static LABELS: [Label; 1] = [Label::from_static_parts("key", "value")];
static BORROWED_LABELS: Key = Key::from_static_parts(&BORROWED_NAME, &LABELS);
#[test]
fn test_key_ord_and_partialord() {
let keys_expected: Vec<Key> = vec![
Key::from_name("aaaa").into(),
Key::from_name("bbbb").into(),
Key::from_name("cccc").into(),
];
let keys_unsorted: Vec<Key> = vec![
Key::from_name("bbbb").into(),
Key::from_name("cccc").into(),
Key::from_name("aaaa").into(),
];
let keys = {
let mut keys = keys_unsorted.clone();
keys.sort();
keys
};
assert_eq!(keys, keys_expected);
let keys = {
let mut keys = keys_unsorted.clone();
keys.sort_by(|a, b| a.partial_cmp(b).unwrap());
keys
};
assert_eq!(keys, keys_expected);
}
#[test]
fn test_key_eq_and_hash() {
let mut keys = HashMap::new();
let owned_basic: Key = Key::from_name("name").into();
assert_eq!(&owned_basic, &BORROWED_BASIC);
let previous = keys.insert(owned_basic, 42);
assert!(previous.is_none());
let previous = keys.get(&BORROWED_BASIC);
assert_eq!(previous, Some(&42));
let labels = LABELS.to_vec();
let owned_labels = Key::from_parts(&BORROWED_NAME[..], labels);
assert_eq!(&owned_labels, &BORROWED_LABELS);
let previous = keys.insert(owned_labels, 43);
assert!(previous.is_none());
let previous = keys.get(&BORROWED_LABELS);
assert_eq!(previous, Some(&43));
let basic: Key = "constant_key".into();
let cloned_basic = basic.clone();
assert_eq!(basic, cloned_basic);
}
#[test]
fn test_key_data_proper_display() {
let key1 = Key::from_name("foobar");
let result1 = key1.to_string();
assert_eq!(result1, "Key(foobar)");
let key2 = Key::from_parts(&FOOBAR_NAME[..], vec![Label::new("system", "http")]);
let result2 = key2.to_string();
assert_eq!(result2, "Key(foobar, [system = http])");
let key3 = Key::from_parts(
&FOOBAR_NAME[..],
vec![Label::new("system", "http"), Label::new("user", "joe")],
);
let result3 = key3.to_string();
assert_eq!(result3, "Key(foobar, [system = http, user = joe])");
let key4 = Key::from_parts(
&FOOBAR_NAME[..],
vec![
Label::new("black", "black"),
Label::new("lives", "lives"),
Label::new("matter", "matter"),
],
);
let result4 = key4.to_string();
assert_eq!(result4, "Key(foobar, [black = black, lives = lives, matter = matter])");
}
#[test]
fn test_key_name_equality() {
static KEY_NAME: &'static str = "key_name";
let borrowed_const = KeyName::from_const_str(KEY_NAME);
let borrowed_nonconst = KeyName::from(KEY_NAME);
let owned = KeyName::from(KEY_NAME.to_owned());
let shared_arc = Arc::from(KEY_NAME);
let shared = KeyName::from(Arc::clone(&shared_arc));
assert_eq!(borrowed_const, borrowed_nonconst);
assert_eq!(borrowed_const.as_str(), borrowed_nonconst.as_str());
assert_eq!(borrowed_const, owned);
assert_eq!(borrowed_const.as_str(), owned.as_str());
assert_eq!(borrowed_const, shared);
assert_eq!(borrowed_const.as_str(), shared.as_str());
}
#[test]
fn test_shared_key_name_drop_logic() {
let shared_arc = Arc::from("foo");
let shared = KeyName::from(Arc::clone(&shared_arc));
assert_eq!(shared_arc.deref(), shared.as_str());
assert_eq!(Arc::strong_count(&shared_arc), 2);
drop(shared);
assert_eq!(Arc::strong_count(&shared_arc), 1);
let shared_weak = Arc::downgrade(&shared_arc);
assert_eq!(Arc::strong_count(&shared_arc), 1);
let shared = KeyName::from(Arc::clone(&shared_arc));
assert_eq!(shared_arc.deref(), shared.as_str());
assert_eq!(Arc::strong_count(&shared_arc), 2);
drop(shared_arc);
assert_eq!(shared_weak.strong_count(), 1);
drop(shared);
assert_eq!(shared_weak.strong_count(), 0);
}
}