Skip to content

Commit

Permalink
submap sub map should maintain properties of parent map redplanetla…
Browse files Browse the repository at this point in the history
…bs#235

Adding new SubMap protocol to provide a different implementation of select-keys for TreeMap

Adding tests to ensure order preserved for a large map, as well as one
with a custom comparator
  • Loading branch information
jeff303 committed Sep 18, 2020
1 parent edd7429 commit a8151c6
Show file tree
Hide file tree
Showing 3 changed files with 45 additions and 4 deletions.
4 changes: 2 additions & 2 deletions src/clj/com/rpl/specter.cljc
Original file line number Diff line number Diff line change
Expand Up @@ -905,10 +905,10 @@
submap
[m-keys]
(select* [this structure next-fn]
(next-fn (select-keys structure m-keys)))
(next-fn (n/select-submap structure m-keys)))

(transform* [this structure next-fn]
(let [submap (select-keys structure m-keys)
(let [submap (n/select-submap structure m-keys)
newmap (next-fn submap)]
(merge (reduce dissoc structure m-keys)
newmap))))
Expand Down
18 changes: 18 additions & 0 deletions src/clj/com/rpl/specter/navs.cljc
Original file line number Diff line number Diff line change
Expand Up @@ -691,3 +691,21 @@
((:end-fn end-fn) structure start)
(end-fn structure)
))

(defn- maintain-sorted-submap [structure m-keys]
(into (empty structure) (select-keys structure m-keys)))

(defprotocol SubMap
(select-submap [structure m-keys]))

(extend-protocol SubMap
nil
(select-submap [_ _] nil)

#?(:clj clojure.lang.PersistentTreeMap :cljs cljs.core/PersistentTreeMap)
(select-submap [structure m-keys]
(maintain-sorted-submap structure m-keys))

#?(:clj Object :cljs default)
(select-submap [structure m-keys]
(select-keys structure m-keys)))
27 changes: 25 additions & 2 deletions test/com/rpl/specter/core_test.cljc
Original file line number Diff line number Diff line change
Expand Up @@ -611,7 +611,8 @@
(= (transform (s/subset s3) identity combined) combined)
(= (setval (s/subset s3) #{} combined) (set/difference combined s2))
(= (setval (s/subset s3) s4 combined) (-> combined (set/difference s2) (set/union s4)))))))

#?(:clj
(defstruct test-struct :x :y :z))

(deftest submap-test
(is (= [{:foo 1}]
Expand All @@ -623,7 +624,29 @@
(is (= {:a {:new 1}
:c {:new 1
:old 1}}
(setval [s/ALL s/LAST (s/submap [])] {:new 1} {:a nil, :c {:old 1}}))))
(setval [s/ALL s/LAST (s/submap [])] {:new 1} {:a nil, :c {:old 1}})))
#?(:clj
; make sure struct-map works
(is (= [{:z 3, :y 2}]
(select (s/submap [:z :y]) (into (struct-map test-struct) {:x 1 :y 2 :z 3})))))
; ensure order is preserved for larger sorted map (i.e. TreeMap)
(let [range50 (range 50)]
(is (= (take-nth 2 range50)
(select [(s/submap (shuffle (take-nth 2 range50))) s/MAP-KEYS] (apply sorted-map (interleave range50 range50))))))

; ensure order is preserved for TreeMap with custom comparator
(let [all-keys ["a" "ab" "abc" "abcd" "abcde" "abcdef" "abcdefg"]
cmp (fn [key1 key2] (> (count key1) (count key2)))
input (apply sorted-map-by cmp (interleave all-keys (range (count all-keys))))
result (select (s/submap (shuffle (take-nth 2 all-keys))) input)
result-submap (first result)]
; the comparator sorts by decreasing length of string key (which is opposite of alpha order)
; selecting the desired keys in random order should still result in a TreeMap adhering to the comparator
(is (= (map first result-submap) (take-nth 2 (reverse all-keys))))
#?(:clj
; check the actual comparator is maintained (doesn't work in cljs)
(is (identical? (.comparator result-submap) cmp)))
))

(deftest nil->val-test
(is (= {:a #{:b}}
Expand Down

0 comments on commit a8151c6

Please sign in to comment.