use super::{Entry, Key};
use core::hash::{BuildHasher, Hash};
use hashbrown::hash_map::DefaultHashBuilder;
use hashbrown::raw::RawTable;
pub trait Equivalent<K: ?Sized> {
fn equivalent(&self, key: &K) -> bool;
}
impl<Q: ?Sized + Eq, K: ?Sized> Equivalent<K> for Q
where
K: std::borrow::Borrow<Q>,
{
fn equivalent(&self, key: &K) -> bool {
self == key.borrow()
}
}
fn equivalent_key<'a, Q>(entries: &'a [Entry], k: &'a Q) -> impl 'a + Fn(&Indexes) -> bool
where
Q: ?Sized + Equivalent<Key>,
{
move |indexes| k.equivalent(&entries[indexes.rep].key)
}
fn make_hasher<'a, S>(entries: &'a [Entry], hash_builder: &'a S) -> impl 'a + Fn(&Indexes) -> u64
where
S: BuildHasher,
{
move |indexes| hash_builder.hash_one(&entries[indexes.rep].key)
}
#[derive(Clone, Debug, PartialEq, Eq)]
pub struct Indexes {
rep: usize,
other: Vec<usize>,
}
impl Indexes {
fn new(rep: usize) -> Self {
Self {
rep,
other: Vec::new(),
}
}
pub fn len(&self) -> usize {
1 + self.other.len()
}
pub fn first(&self) -> usize {
self.rep
}
pub fn is_redundant(&self) -> bool {
!self.other.is_empty()
}
pub fn redundant(&self) -> Option<usize> {
self.other.first().cloned()
}
pub fn redundants(&self) -> &[usize] {
&self.other
}
fn insert(&mut self, mut index: usize) {
if index != self.rep {
if index < self.rep {
core::mem::swap(&mut index, &mut self.rep);
}
if let Err(i) = self.other.binary_search(&index) {
self.other.insert(i, index)
}
}
}
fn remove(&mut self, index: usize) -> bool {
if self.rep == index {
if self.other.is_empty() {
false
} else {
self.rep = self.other.remove(0);
true
}
} else {
if let Ok(i) = self.other.binary_search(&index) {
self.other.remove(i);
}
true
}
}
pub fn shift_down(&mut self, index: usize) {
if self.rep > index {
self.rep -= 1
}
for i in &mut self.other {
if *i > index {
*i -= 1
}
}
}
pub fn shift_up(&mut self, index: usize) {
if self.rep >= index {
self.rep += 1
}
for i in &mut self.other {
if *i >= index {
*i += 1
}
}
}
pub fn iter(&self) -> super::Indexes {
super::Indexes::Some {
first: Some(self.rep),
other: self.other.iter(),
}
}
}
impl<'a> IntoIterator for &'a Indexes {
type Item = usize;
type IntoIter = super::Indexes<'a>;
fn into_iter(self) -> Self::IntoIter {
self.iter()
}
}
#[derive(Clone)]
pub struct IndexMap<S = DefaultHashBuilder> {
hash_builder: S,
table: RawTable<Indexes>,
}
impl<S: Default> IndexMap<S> {
fn default() -> Self {
Self {
hash_builder: S::default(),
table: RawTable::default(),
}
}
}
impl<S> IndexMap<S> {
pub fn new() -> Self
where
S: Default,
{
Self::default()
}
pub fn contains_duplicate_keys(&self) -> bool {
unsafe {
for bucket in self.table.iter() {
if bucket.as_ref().is_redundant() {
return true;
}
}
}
false
}
}
impl<S: BuildHasher> IndexMap<S> {
pub fn get<Q: ?Sized>(&self, entries: &[Entry], key: &Q) -> Option<&Indexes>
where
Q: Hash + Equivalent<Key>,
{
let hash = self.hash_builder.hash_one(key);
self.table.get(hash, equivalent_key(entries, key))
}
pub fn insert(&mut self, entries: &[Entry], index: usize) -> bool {
let key = &entries[index].key;
let hash = self.hash_builder.hash_one(key);
match self.table.get_mut(hash, equivalent_key(entries, key)) {
Some(indexes) => {
indexes.insert(index);
false
}
None => {
self.table.insert(
hash,
Indexes::new(index),
make_hasher::<S>(entries, &self.hash_builder),
);
true
}
}
}
pub fn remove(&mut self, entries: &[Entry], index: usize) {
let key = &entries[index].key;
let hash = self.hash_builder.hash_one(key);
if let Some(bucket) = self.table.find(hash, equivalent_key(entries, key)) {
let indexes = unsafe { bucket.as_mut() };
if !indexes.remove(index) {
unsafe { self.table.remove(bucket) };
}
}
}
pub fn shift_down(&mut self, index: usize) {
unsafe {
for bucket in self.table.iter() {
let indexes = bucket.as_mut();
indexes.shift_down(index)
}
}
}
pub fn shift_up(&mut self, index: usize) {
unsafe {
for bucket in self.table.iter() {
let indexes = bucket.as_mut();
indexes.shift_up(index)
}
}
}
pub fn clear(&mut self) {
self.table.clear()
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::Value;
#[test]
fn insert() {
let entries = [
Entry::new("a".into(), Value::Null),
Entry::new("b".into(), Value::Null),
Entry::new("a".into(), Value::Null),
];
let mut indexes: IndexMap = IndexMap::default();
indexes.insert(&entries, 2);
indexes.insert(&entries, 1);
indexes.insert(&entries, 0);
let mut a = indexes.get(&entries, "a").unwrap().iter();
let mut b = indexes.get(&entries, "b").unwrap().iter();
assert_eq!(a.next(), Some(0));
assert_eq!(a.next(), Some(2));
assert_eq!(a.next(), None);
assert_eq!(b.next(), Some(1));
assert_eq!(b.next(), None);
assert_eq!(indexes.get(&entries, "c"), None)
}
#[test]
fn remove1() {
let entries = [
Entry::new("a".into(), Value::Null),
Entry::new("b".into(), Value::Null),
Entry::new("a".into(), Value::Null),
];
let mut indexes: IndexMap = IndexMap::default();
indexes.insert(&entries, 2);
indexes.insert(&entries, 1);
indexes.insert(&entries, 0);
indexes.remove(&entries, 1);
indexes.remove(&entries, 0);
let mut a = indexes.get(&entries, "a").unwrap().iter();
assert_eq!(a.next(), Some(2));
assert_eq!(a.next(), None);
assert_eq!(indexes.get(&entries, "b"), None)
}
#[test]
fn remove2() {
let entries = [
Entry::new("a".into(), Value::Null),
Entry::new("b".into(), Value::Null),
Entry::new("a".into(), Value::Null),
];
let mut indexes: IndexMap = IndexMap::default();
indexes.insert(&entries, 2);
indexes.insert(&entries, 1);
indexes.insert(&entries, 0);
indexes.remove(&entries, 0);
indexes.remove(&entries, 1);
indexes.remove(&entries, 2);
assert_eq!(indexes.get(&entries, "a"), None);
assert_eq!(indexes.get(&entries, "b"), None)
}
}