cljrs_value/collections/
hash_map.rs1use crate::Value;
2use crate::collections::array_map::PersistentArrayMap;
3use rpds::HashTrieMapSync;
4
5#[derive(Debug, Clone)]
11pub struct PersistentHashMap {
12 inner: rpds::HashTrieMapSync<Value, Value>,
13}
14
15impl PersistentHashMap {
16 pub fn empty() -> Self {
17 Self {
18 inner: rpds::HashTrieMapSync::new_sync(),
19 }
20 }
21
22 pub fn new(map: HashTrieMapSync<Value, Value>) -> Self {
23 Self { inner: map }
24 }
25
26 pub fn count(&self) -> usize {
27 self.inner.size()
28 }
29
30 pub fn is_empty(&self) -> bool {
31 self.inner.is_empty()
32 }
33
34 pub fn get(&self, key: &Value) -> Option<&Value> {
36 self.inner.get(key)
37 }
38
39 pub fn contains_key(&self, key: &Value) -> bool {
40 self.inner.contains_key(key)
41 }
42
43 pub fn assoc(&self, key: Value, value: Value) -> Self {
45 Self {
46 inner: self.inner.insert(key, value),
47 }
48 }
49
50 pub fn dissoc(&self, key: &Value) -> Self {
52 Self {
53 inner: self.inner.remove(key),
54 }
55 }
56
57 pub fn iter(&self) -> impl Iterator<Item = (&Value, &Value)> {
59 self.inner.iter()
60 }
61
62 pub fn keys(&self) -> Vec<Value> {
64 self.inner.keys().cloned().collect()
65 }
66
67 pub fn vals(&self) -> Vec<Value> {
69 self.inner.values().cloned().collect()
70 }
71
72 pub fn merge(&self, other: &Self) -> Self {
74 let mut result = self.clone();
75 for (k, v) in other.inner.iter() {
76 result = result.assoc(k.clone(), v.clone());
77 }
78 result
79 }
80
81 pub fn from_pairs<I: IntoIterator<Item = (Value, Value)>>(iter: I) -> Self {
83 let mut m = Self::empty();
84 for (k, v) in iter {
85 m = m.assoc(k, v);
86 }
87 m
88 }
89
90 pub fn from_array_map(am: &PersistentArrayMap) -> Self {
92 Self::from_pairs(am.iter().map(|(k, v)| (k.clone(), v.clone())))
93 }
94
95 pub fn inner(&self) -> &HashTrieMapSync<Value, Value> {
96 &self.inner
97 }
98}
99
100impl PartialEq for PersistentHashMap {
101 fn eq(&self, other: &Self) -> bool {
102 if self.count() != other.count() {
103 return false;
104 }
105 self.inner.iter().all(|(k, v)| other.get(k) == Some(v))
106 }
107}
108
109impl cljrs_gc::Trace for PersistentHashMap {
110 fn trace(&self, visitor: &mut cljrs_gc::MarkVisitor) {
111 for (k, v) in self.inner.iter() {
112 k.trace(visitor);
113 v.trace(visitor);
114 }
115 }
116}
117
118#[cfg(test)]
119mod tests {
120 use super::*;
121 use crate::Value;
122 use crate::collections::array_map::AssocResult;
123 use cljrs_gc::GcPtr;
124
125 fn kw(s: &str) -> Value {
126 Value::Keyword(GcPtr::new(crate::keyword::Keyword::simple(s)))
127 }
128 fn int(n: i64) -> Value {
129 Value::Long(n)
130 }
131
132 #[test]
133 fn test_basic_ops() {
134 let m = PersistentHashMap::empty();
135 let m = m.assoc(kw("a"), int(1));
136 let m = m.assoc(kw("b"), int(2));
137 assert_eq!(m.count(), 2);
138 assert_eq!(m.get(&kw("a")), Some(&int(1)));
139 assert_eq!(m.get(&kw("b")), Some(&int(2)));
140 assert_eq!(m.get(&kw("c")), None);
141 }
142
143 #[test]
144 fn test_update() {
145 let m = PersistentHashMap::empty()
146 .assoc(kw("a"), int(1))
147 .assoc(kw("a"), int(99));
148 assert_eq!(m.count(), 1);
149 assert_eq!(m.get(&kw("a")), Some(&int(99)));
150 }
151
152 #[test]
153 fn test_dissoc() {
154 let m = PersistentHashMap::empty()
155 .assoc(kw("a"), int(1))
156 .assoc(kw("b"), int(2));
157 let m2 = m.dissoc(&kw("a"));
158 assert_eq!(m2.count(), 1);
159 assert_eq!(m2.get(&kw("a")), None);
160 assert_eq!(m2.get(&kw("b")), Some(&int(2)));
161 }
162
163 #[test]
164 fn test_many_entries() {
165 let mut m = PersistentHashMap::empty();
166 for i in 0i64..200 {
167 m = m.assoc(int(i), int(i * 10));
168 }
169 assert_eq!(m.count(), 200);
170 for i in 0i64..200 {
171 assert_eq!(m.get(&int(i)), Some(&int(i * 10)));
172 }
173 }
174
175 #[test]
176 fn test_merge() {
177 let a = PersistentHashMap::empty()
178 .assoc(kw("a"), int(1))
179 .assoc(kw("b"), int(2));
180 let b = PersistentHashMap::empty()
181 .assoc(kw("b"), int(99))
182 .assoc(kw("c"), int(3));
183 let merged = a.merge(&b);
184 assert_eq!(merged.count(), 3);
185 assert_eq!(merged.get(&kw("a")), Some(&int(1)));
186 assert_eq!(merged.get(&kw("b")), Some(&int(99))); assert_eq!(merged.get(&kw("c")), Some(&int(3)));
188 }
189
190 #[test]
191 fn test_equality() {
192 let a = PersistentHashMap::empty()
193 .assoc(kw("a"), int(1))
194 .assoc(kw("b"), int(2));
195 let b = PersistentHashMap::empty()
196 .assoc(kw("b"), int(2))
197 .assoc(kw("a"), int(1));
198 assert_eq!(a, b);
199 }
200
201 #[test]
202 fn test_from_array_map() {
203 let mut am = PersistentArrayMap::empty();
204 for i in 0..3i64 {
205 let AssocResult::Array(next) = am.assoc(int(i), int(i * 2)) else {
206 panic!()
207 };
208 am = next;
209 }
210 let hm = PersistentHashMap::from_array_map(&am);
211 assert_eq!(hm.count(), 3);
212 for i in 0..3i64 {
213 assert_eq!(hm.get(&int(i)), Some(&int(i * 2)));
214 }
215 }
216}