Skip to content
This repository was archived by the owner on Apr 16, 2021. It is now read-only.

Commit 9402162

Browse files
authored
Merge pull request #16 from shopsmart/pre-1.0
Exhaustively validate retry-with-timeout parameter values
2 parents 5c92f2b + 612e2bd commit 9402162

File tree

4 files changed

+103
-39
lines changed

4 files changed

+103
-39
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -9,3 +9,4 @@ pom.xml.asc
99
/.nrepl-port
1010
.hgignore
1111
.hg/
12+
.nightlight.edn

project.clj

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
(defproject com.github.shopsmart/clj-foundation "0.9.16"
1+
(defproject com.github.shopsmart/clj-foundation "0.9.17"
22
:description "Common patterns enabling simpler to be easier and harder to be possibler."
33
:url "https://github.com/shopsmart/clj-foundation"
44

src/clj_foundation/errors.clj

+4-2
Original file line numberDiff line numberDiff line change
@@ -167,7 +167,7 @@
167167
(fn [f & args] (apply retry tries pause-millis f args)))
168168

169169

170-
;; Retry calling fn with a specified timeout; abort early if abort? returns truthy --
170+
;; Retry calling fn with a specified timeout; abort early if abort?-fn returns truthy
171171

172172

173173
(def TIMEOUT-ERROR
@@ -232,7 +232,7 @@
232232
abort?-fn :- (=> s/Bool [[Throwable]])])
233233

234234

235-
(s/defn retry-with-timeout :- s/Any
235+
(s/defn ^:always-validate retry-with-timeout :- s/Any
236236
"Retry (apply f args) up to tries times with pause-millis time in between invocation and a
237237
timeout value of timeout-millis. On failure, abort?-fn is called with a vector containing the
238238
unwrapped exception stack.
@@ -254,6 +254,8 @@
254254
f :- (=> s/Any [s/Any])
255255
& args :- [s/Any]]
256256

257+
(s/validate RetrySettings settings)
258+
257259
(let [tries (:tries settings)
258260
timeout-millis (:timeout-millis settings)
259261
pause-millis (:pause-millis settings)

test/clj_foundation/errors_test.clj

+97-36
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,8 @@
77
[clj-foundation.data :refer [any?]]
88
[clj-foundation.patterns :as p]
99
[clj-foundation.millis :as millis])
10-
(:import [java.util Date]))
10+
(:import [java.util Date]
11+
[clojure.lang ExceptionInfo]))
1112

1213

1314
(common/register-fixtures)
@@ -157,41 +158,101 @@
157158

158159

159160
(deftest retry-with-timeout-test
160-
(testing "Success -> Results!!!"
161-
(is (= "Results!!!"
162-
(retry-with-timeout
163-
"Success"
164-
(->RetrySettings 1 (millis/<-seconds 1) (millis/<-seconds 5) (constantly false))
165-
(constantly "Results!!!")))))
166-
167-
(testing "Taking too much time fails!"
168-
(is (instance? RuntimeException
169-
(try*
170-
(retry-with-timeout
171-
"Sloooow"
172-
(->RetrySettings 3 (millis/<-seconds 1) 50 (constantly false))
173-
#(Thread/sleep (millis/<-seconds 2)))))))
174-
175-
(testing "Fatal errors abort retrying"
176-
(is (instance? RuntimeException
177-
(try*
178-
(retry-with-timeout
179-
"Ooops..."
180-
(->RetrySettings 3 (millis/<-seconds 1) (millis/<-seconds 5) (constantly true))
181-
#(throw (RuntimeException.)))))))
182-
183-
(testing "If at first you don't succeed, try, try again..."
184-
(let [attempts (atom 0)
185-
job-fn (fn []
186-
(swap! attempts inc)
187-
(when (< @attempts 2) (throw (RuntimeException. "Not this time")))
188-
"Finally--success!")]
189-
(is (= "Finally--success!"
161+
(testing "nil arguments throw ExceptionInfo"
162+
(testing "nil in main argument list"
163+
(is (thrown? ExceptionInfo
164+
(retry-with-timeout
165+
nil
166+
(->RetrySettings 1 (millis/<-seconds 1) (millis/<-seconds 5) (constantly false))
167+
(constantly "nil not allowed!"))))
168+
169+
(is (thrown? ExceptionInfo
170+
(retry-with-timeout
171+
"nil throws!"
172+
nil
173+
(constantly "nil not allowed!"))))
174+
175+
(is (thrown? ExceptionInfo
176+
(retry-with-timeout
177+
"nil throws!"
178+
(->RetrySettings 1 (millis/<-seconds 1) (millis/<-seconds 5) (constantly false))
179+
nil))))
180+
181+
(testing "nil in RetrySettings"
182+
(is (thrown? ExceptionInfo
183+
(retry-with-timeout
184+
"nil throws!"
185+
(->RetrySettings nil (millis/<-seconds 1) (millis/<-seconds 5) (constantly false))
186+
(constantly "nil not allowed!"))))
187+
188+
(is (thrown? ExceptionInfo
189+
(retry-with-timeout
190+
"nil throws!"
191+
(->RetrySettings 1 nil (millis/<-seconds 5) (constantly false))
192+
(constantly "nil not allowed!"))))
193+
194+
(is (thrown? ExceptionInfo
195+
(retry-with-timeout
196+
"nil throws!"
197+
(->RetrySettings 1 (millis/<-seconds 1) nil (constantly false))
198+
(constantly "nil not allowed!"))))
199+
200+
(is (thrown? ExceptionInfo
201+
(retry-with-timeout
202+
"nil throws!"
203+
(->RetrySettings 1 (millis/<-seconds 1) (millis/<-seconds 5) nil)
204+
(constantly "nil not allowed!"))))))
205+
206+
(testing "Happy paths: "
207+
(testing "Success -> Results!!!"
208+
(is (= "Results!!!"
190209
(retry-with-timeout
191-
"Persistance pays off"
192-
(->RetrySettings 3 (millis/<-seconds 1) (millis/<-seconds 5) (constantly false))
193-
job-fn)))
194-
(is (= 2 @attempts)))))
210+
"Success"
211+
(->RetrySettings 1 (millis/<-seconds 1) (millis/<-seconds 5) (constantly false))
212+
(constantly "Results!!!")))))
213+
214+
(testing "If at first you don't succeed, try, try again..."
215+
(let [attempts (atom 0)
216+
job-fn (fn []
217+
(swap! attempts inc)
218+
(when (< @attempts 2) (throw (RuntimeException. "Not this time")))
219+
"Finally--success!")]
220+
(is (= "Finally--success!"
221+
(retry-with-timeout
222+
"Persistance pays off"
223+
(->RetrySettings 3 (millis/<-seconds 1) (millis/<-seconds 5) (constantly false))
224+
job-fn)))
225+
(is (= 2 @attempts)))))
226+
227+
(testing "Sad paths: "
228+
(testing "Taking too much time fails when abort?-fn is (constantly false)!"
229+
(is (thrown? RuntimeException
230+
(retry-with-timeout
231+
"Sloooow"
232+
(->RetrySettings 3 (millis/<-seconds 1) 50 (constantly false))
233+
#(Thread/sleep (millis/<-seconds 2))))))
234+
235+
(testing "Taking too much time fails--even with retries! (Note: timeouts are not exceptions so abort?-fn is not relevant)"
236+
(let [total-tries (atom 0)]
237+
(is (thrown? RuntimeException
238+
(retry-with-timeout
239+
"Sloooow"
240+
(->RetrySettings 3 (millis/<-seconds 1) 50 (constantly true))
241+
(fn []
242+
(swap! total-tries inc)
243+
(Thread/sleep (millis/<-seconds 2))))))
244+
(is (= 4 @total-tries))))
245+
246+
(testing "Fatal errors abort retrying"
247+
(let [total-tries (atom 0)]
248+
(is (thrown? RuntimeException
249+
(retry-with-timeout
250+
"Ooops..."
251+
(->RetrySettings 3 (millis/<-seconds 1) (millis/<-seconds 5) (constantly true))
252+
(fn []
253+
(swap! total-tries inc)
254+
(throw (RuntimeException.))))))
255+
(is (= 1 @total-tries))))))
195256

196257

197258
;; (dis)allowed values ------------------------------------------------------------------
@@ -220,4 +281,4 @@
220281

221282

222283

223-
(run-tests)
284+
;(run-tests)

0 commit comments

Comments
 (0)