From 53a57b6627826fda38173c63903bfd6198a9186d Mon Sep 17 00:00:00 2001 From: davidnolen Date: Mon, 22 Jan 2024 13:26:06 -0500 Subject: [PATCH 1/4] * add hash-long * add hash-double * reported test case --- src/main/cljs/cljs/core.cljs | 14 +++++++++++++- src/test/cljs/cljs/hashing_test.cljs | 5 +++++ 2 files changed, 18 insertions(+), 1 deletion(-) diff --git a/src/main/cljs/cljs/core.cljs b/src/main/cljs/cljs/core.cljs index 4b9ecc5214..75c44220f5 100644 --- a/src/main/cljs/cljs/core.cljs +++ b/src/main/cljs/cljs/core.cljs @@ -960,6 +960,16 @@ h1 (m3-mix-H1 m3-seed k1)] (m3-fmix h1 4)))) +(defn hash-long [high low] + (bit-xor high low)) + +(defn hash-double [f] + (let [arr (doto (js/Float64Array. 1) (aset 0 f)) + buf (.-buffer arr) + low (.getInt32 (js/DataView. buf 0 4)) + high (.getInt32 (js/DataView. buf 4 4))] + (hash-long high low))) + (defn ^number m3-hash-unencoded-chars [in] (let [h1 (loop [i 1 h1 m3-seed] (if (< i (.-length in)) @@ -1021,7 +1031,9 @@ (number? o) (if ^boolean (js/isFinite o) - (js-mod (Math/floor o) 2147483647) + (if-not (.isInteger js/Number o) + (hash-double o) + (js-mod (Math/floor o) 2147483647)) (case o ##Inf 2146435072 diff --git a/src/test/cljs/cljs/hashing_test.cljs b/src/test/cljs/cljs/hashing_test.cljs index e40178341a..d3b6966c11 100644 --- a/src/test/cljs/cljs/hashing_test.cljs +++ b/src/test/cljs/cljs/hashing_test.cljs @@ -93,3 +93,8 @@ (deftest test-cljs-1818 (is (= (hash true) 1231)) (is (= (hash false) 1237))) + +(deftest test-cljs-3410 + (testing "Small floats should not hash the same" + (is (not= (hash-double -0.32553251) (hash-double -0.0000032553251))) + (is (not= (hash -0.32553251) (hash -0.0000032553251))))) From 107eb74c8b507f11b0d27ef096a083def024648e Mon Sep 17 00:00:00 2001 From: davidnolen Date: Mon, 22 Jan 2024 13:37:42 -0500 Subject: [PATCH 2/4] * typos --- src/main/cljs/cljs/core.cljs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/main/cljs/cljs/core.cljs b/src/main/cljs/cljs/core.cljs index 75c44220f5..3dde23d957 100644 --- a/src/main/cljs/cljs/core.cljs +++ b/src/main/cljs/cljs/core.cljs @@ -966,8 +966,8 @@ (defn hash-double [f] (let [arr (doto (js/Float64Array. 1) (aset 0 f)) buf (.-buffer arr) - low (.getInt32 (js/DataView. buf 0 4)) - high (.getInt32 (js/DataView. buf 4 4))] + high (.getInt32 (js/DataView. buf 0 4)) + low (.getInt32 (js/DataView. buf 4 4))] (hash-long high low))) (defn ^number m3-hash-unencoded-chars [in] @@ -1031,7 +1031,7 @@ (number? o) (if ^boolean (js/isFinite o) - (if-not (.isInteger js/Number o) + (if-not ^boolean (.isInteger js/Number o) (hash-double o) (js-mod (Math/floor o) 2147483647)) (case o From e0d5458016f9f1336f0c9b7e4ef53302dfc44215 Mon Sep 17 00:00:00 2001 From: davidnolen Date: Mon, 22 Jan 2024 15:10:02 -0500 Subject: [PATCH 3/4] * switch to isSafeInteger --- src/main/cljs/cljs/core.cljs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/cljs/cljs/core.cljs b/src/main/cljs/cljs/core.cljs index 3dde23d957..0afb3d4752 100644 --- a/src/main/cljs/cljs/core.cljs +++ b/src/main/cljs/cljs/core.cljs @@ -1031,7 +1031,7 @@ (number? o) (if ^boolean (js/isFinite o) - (if-not ^boolean (.isInteger js/Number o) + (if-not ^boolean (.isSafeInteger js/Number o) (hash-double o) (js-mod (Math/floor o) 2147483647)) (case o From bff567f4b234afbc72a8da96362b9dcb8e4a6138 Mon Sep 17 00:00:00 2001 From: davidnolen Date: Mon, 22 Jan 2024 15:12:04 -0500 Subject: [PATCH 4/4] * more test assertions --- src/test/cljs/cljs/hashing_test.cljs | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/src/test/cljs/cljs/hashing_test.cljs b/src/test/cljs/cljs/hashing_test.cljs index d3b6966c11..2a2ffcf18f 100644 --- a/src/test/cljs/cljs/hashing_test.cljs +++ b/src/test/cljs/cljs/hashing_test.cljs @@ -95,6 +95,10 @@ (is (= (hash false) 1237))) (deftest test-cljs-3410 - (testing "Small floats should not hash the same" + (testing "Small doubles should not hash the same" (is (not= (hash-double -0.32553251) (hash-double -0.0000032553251))) - (is (not= (hash -0.32553251) (hash -0.0000032553251))))) + (is (not= (hash -0.32553251) (hash -0.0000032553251)))) + (testing "Same double hashes the same" + (is (= (hash 0.5) (hash 0.5))) + (is (= (hash -0.32553251) (hash -0.32553251))) + (is (= (hash -0.0000032553251) (hash -0.0000032553251)))))