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

Commit 2616c13

Browse files
authored
Merge pull request #17 from shopsmart/pre-1.0
Fix infinite retry loop+more tests
2 parents 9402162 + ec9efda commit 2616c13

File tree

3 files changed

+90
-38
lines changed

3 files changed

+90
-38
lines changed

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.17"
1+
(defproject com.github.shopsmart/clj-foundation "0.9.18"
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

+8-8
Original file line numberDiff line numberDiff line change
@@ -206,12 +206,10 @@
206206
(let [job-abort? (:abort?-fn job)
207207
result-exception (exception<- failure-value)]
208208
(cond
209-
(timeout? failure-value) (if (< (:retries job) (:max-retries job))
210-
:RETRY-TIMEOUT
211-
:ABORT-MAX-RETRIES)
212-
213-
(job-abort? (seq<- result-exception)) :ABORT-FATAL-ERROR
214-
:else :RETRY-FAILURE)))
209+
(<= (:max-retries job) (:retries job)) :ABORT-MAX-RETRIES
210+
(timeout? failure-value) :RETRY-TIMEOUT
211+
(job-abort? (seq<- result-exception)) :ABORT-FATAL-ERROR
212+
:else :RETRY-FAILURE)))
215213

216214

217215
(defn new-default-job
@@ -267,8 +265,10 @@
267265
(case (retry? j result)
268266
:ABORT-MAX-RETRIES (throw (RuntimeException. (str "MAX-RETRIES(" tries ")[" job-name "]: " (.getMessage result)) result))
269267
:ABORT-FATAL-ERROR (throw (RuntimeException. (str "FATAL[" job-name "]: " (.getMessage result)) result))
270-
:RETRY-FAILURE (log/error result (str "RETRY[" job-name "]; " (type result) ": " (.getMessage result)))
271-
:RETRY-TIMEOUT (log/error (RuntimeException. "Timeout.") (str "RETRY[" job-name "]: Took longer than " timeout-millis " ms."))
268+
:RETRY-FAILURE (do (log/error result (str "RETRY[" job-name "]; " (type result) ": " (.getMessage result)))
269+
(Thread/sleep pause-millis))
270+
:RETRY-TIMEOUT (do (log/error (RuntimeException. "Timeout.") (str "RETRY[" job-name "]: Took longer than " timeout-millis " ms."))
271+
(Thread/sleep pause-millis))
272272
:else (throw (IllegalStateException. "Program Error! We should never get here.")))
273273
(recur (update-in j [:retries] inc)))
274274
result)))))

test/clj_foundation/errors_test.clj

+81-29
Original file line numberDiff line numberDiff line change
@@ -106,24 +106,41 @@
106106
after (.getTime (Date.))
107107
time (- after before)]
108108

109-
(is (instance? IllegalStateException result))
110-
(is (< timeout time)))))
109+
(is (instance? IllegalStateException result))
110+
(is (< timeout time)))))
111111

112112

113113
(deftest retry?-test
114-
(testing "Retry up to :max-retries times"
115-
(let [r0 (new-default-job "timeout-test"
116-
3
117-
(millis/<-seconds 0.25)
118-
(constantly false))
119-
r1 (update-in r0 [:retries] inc)
120-
r2 (update-in r1 [:retries] inc)
121-
r3 (update-in r2 [:retries] inc)]
114+
(testing "Retry up to :max-retries times: "
115+
(testing "too many timeouts"
116+
(let [r0 (new-default-job "timeout-test"
117+
3
118+
(millis/<-seconds 0.25)
119+
(constantly false))
120+
r1 (update-in r0 [:retries] inc)
121+
r2 (update-in r1 [:retries] inc)
122+
r3 (update-in r2 [:retries] inc)]
123+
124+
(is (= :RETRY-TIMEOUT (retry? r0 TIMEOUT-ERROR)))
125+
(is (= :RETRY-TIMEOUT (retry? r1 TIMEOUT-ERROR)))
126+
(is (= :RETRY-TIMEOUT (retry? r2 TIMEOUT-ERROR)))
127+
(is (= :ABORT-MAX-RETRIES (retry? r3 TIMEOUT-ERROR))))
128+
129+
(testing "too many errors"
130+
(let [r0 (new-default-job "Too man errors test"
131+
3
132+
(millis/<-seconds 0.25)
133+
(constantly false))
134+
r1 (update-in r0 [:retries] inc)
135+
r2 (update-in r1 [:retries] inc)
136+
r3 (update-in r2 [:retries] inc)
137+
e (Exception. "Something bad happened!")]
138+
139+
(is (= :RETRY-FAILURE (retry? r0 e)))
140+
(is (= :RETRY-FAILURE (retry? r1 e)))
141+
(is (= :RETRY-FAILURE (retry? r2 e)))
142+
(is (= :ABORT-MAX-RETRIES (retry? r3 e)))))))
122143

123-
(is (= :RETRY-TIMEOUT (retry? r0 TIMEOUT-ERROR)))
124-
(is (= :RETRY-TIMEOUT (retry? r1 TIMEOUT-ERROR)))
125-
(is (= :RETRY-TIMEOUT (retry? r2 TIMEOUT-ERROR)))
126-
(is (= :ABORT-MAX-RETRIES (retry? r3 TIMEOUT-ERROR)))))
127144

128145
(testing "Abort on fatal errors"
129146
(let [r0 (new-default-job "abort-test"
@@ -211,18 +228,42 @@
211228
(->RetrySettings 1 (millis/<-seconds 1) (millis/<-seconds 5) (constantly false))
212229
(constantly "Results!!!")))))
213230

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)))))
231+
(testing "If at first you don't succeed, try, try again...: "
232+
(testing "With failures."
233+
(let [attempts (atom 0)
234+
start (System/currentTimeMillis)
235+
timeout-time (millis/<-seconds 1)
236+
pause-time (millis/<-seconds 5)
237+
expected-elapsed-time pause-time
238+
job-fn (fn []
239+
(swap! attempts inc)
240+
(when (< @attempts 2) (throw (RuntimeException. "Not this time")))
241+
"Finally--success!")]
242+
(is (= "Finally--success!"
243+
(retry-with-timeout
244+
"Persistance pays off"
245+
(->RetrySettings 3 timeout-time pause-time (constantly false))
246+
job-fn)))
247+
(is (= 2 @attempts))
248+
(is (<= expected-elapsed-time (- (System/currentTimeMillis) start)))))
249+
250+
(testing "with timeouts."
251+
(let [attempts (atom 0)
252+
start (System/currentTimeMillis)
253+
timeout-time (millis/<-seconds 1)
254+
pause-time (millis/<-seconds 5)
255+
expected-elapsed-time (+ timeout-time pause-time)
256+
job-fn (fn []
257+
(swap! attempts inc)
258+
(when (< @attempts 2) (Thread/sleep (+ timeout-time 500)))
259+
"Finally--success!")]
260+
(is (= "Finally--success!"
261+
(retry-with-timeout
262+
"Persistance pays off"
263+
(->RetrySettings 3 timeout-time pause-time (constantly false))
264+
job-fn)))
265+
(is (= 2 @attempts))
266+
(is (<= expected-elapsed-time (- (System/currentTimeMillis) start)))))))
226267

227268
(testing "Sad paths: "
228269
(testing "Taking too much time fails when abort?-fn is (constantly false)!"
@@ -243,15 +284,26 @@
243284
(Thread/sleep (millis/<-seconds 2))))))
244285
(is (= 4 @total-tries))))
245286

287+
(testing "Too many retries fails"
288+
(let [total-tries (atom 0)]
289+
(is (thrown? Exception
290+
(retry-with-timeout
291+
"Boom"
292+
(->RetrySettings 3 (millis/<-seconds 1) 50 (constantly false))
293+
(fn []
294+
(swap! total-tries inc)
295+
(throw (Exception.))))))
296+
(is (= 4 @total-tries))))
297+
246298
(testing "Fatal errors abort retrying"
247299
(let [total-tries (atom 0)]
248-
(is (thrown? RuntimeException
300+
(is (thrown? Exception
249301
(retry-with-timeout
250302
"Ooops..."
251303
(->RetrySettings 3 (millis/<-seconds 1) (millis/<-seconds 5) (constantly true))
252304
(fn []
253305
(swap! total-tries inc)
254-
(throw (RuntimeException.))))))
306+
(throw (Exception.))))))
255307
(is (= 1 @total-tries))))))
256308

257309

@@ -281,4 +333,4 @@
281333

282334

283335

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

0 commit comments

Comments
 (0)