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

Commit 5c92f2b

Browse files
authored
Merge pull request #15 from shopsmart/pre-1.0
Misc cleanup; one bug fix
2 parents 0f68f6b + 986f3ed commit 5c92f2b

File tree

7 files changed

+117
-60
lines changed

7 files changed

+117
-60
lines changed

README.md

+61-49
Original file line numberDiff line numberDiff line change
@@ -1,58 +1,23 @@
1-
# clj-foundation pre-1.0
1+
# clj-foundation
22

3-
## Definition of Done for 1.0 release
4-
5-
* On APIs - While 1.0 will be a major release with some inevitable API breakage, we will strive:
6-
* To maintain API compatiblity with prior releases to the greatest extent possible
7-
* To remove duplicated functionality (which may contradict the API compatibility principle)
8-
* To maintain API compatibility with code copied from internal Brad's Deals projects to the greatest extent possible.
9-
* Clear naming convention for namespaces that are provisional (non-frozen, WIP) API.
10-
11-
* Known tech debt that may influence APIs
12-
* Consider migrating db.clj's local configuration mechanism to config.clj (and then make an instance of the generic implementation inside db.clj)
13-
14-
* Target Clojure 1.9
15-
* All functions will have type information supplied via Clojure 1.9 specs.
16-
* Functions currently using plumatic/schema to specify type constraints will be migrated to Specs
17-
18-
* On testing
19-
* All functions will be unit tested using a style designed to illustrate behavior under failure modes as well as happy path scenarios.
20-
* We will collectively agree on a style of testing that seems to hit the sweet spot between overspecification and underspecification.
21-
* (Details are not set in stone but definitely up for negotiation)
22-
* When defects are detected, we will first reproduce the defect's root cause using a failing test case that will prevent the defect from reoccurring without the test notifying us.
23-
* Generative testing with Specs
24-
* Integrate kbitz (suggest Clojure idioms), some Clojure linting library; if it's easy to make these available to clients of clj-foundation, do so.
25-
26-
* Debug library
27-
* dbg, ppdbg macros
28-
* trace?
29-
30-
* Documentation
31-
* Machine generate as much as possible?
32-
33-
* Make clj_infrastructure as a separate library
34-
* Move db.clj there?
35-
* Abstract db.clj API using monads at the foundation layer and implement for relational, nosql, etc. in infrastructure?
36-
* http://rea.tech/the-worst-thing-in-our-scala-code-futures/ ?
37-
* Abstractions for creating/consuming lazy sequences?
38-
* Abstractions for aggregating a sequence of database results into a map[concatinated-key value(s)]
39-
40-
## Provisional ideas for 1.0
41-
42-
* Figure out how to isolate namespaces and their dependencies at the classpath level, including runtime reloading and evolution semantics after the fashion of OSGi at the REPL or in a running application.
43-
* Update and include clojure.osgi for clean deployment into OSGi frameworks?
44-
* Deploy to Clojars?
45-
46-
47-
# Why clj-foundation?
3+
## Why clj-foundation?
484

495
clj-foundation supplies namespaces making additional simple things easy and hard things possible in Clojure that are intended for use across all Clojure projects at Brad's Deals.
506

517
* Enhances the core language in resonable, useful, and conservative ways.
528
* Enables programming using a monadic style without requiring explicit monad types.
539
* Describes, specifies, and illustrates best practices at Brad's Deals for working in Clojure.
5410
* The only dependencies are Clojure, Potemkin, and Schema in order to minimize adoption friction.
55-
* *(This is up for renegotiation for 1.0)*
11+
12+
## Other Clojure foundational libraries that compliment this one
13+
14+
* Hara: http://docs.caudate.me/hara/index.html
15+
16+
## High-quality domain-specific libraries complimenting this
17+
18+
* Cassandra as a mutable, versioned map: https://github.com/MyPost/cassius
19+
* Git from the Clojure REPL: https://github.com/zcaudate/gita
20+
5621

5722
# Features
5823

@@ -167,8 +132,8 @@ where "version" currently is "[![Release](http://jitpack.io/v/com.github.shopsma
167132
<repositories>
168133
<repository>
169134
<id>jitpack.io</id>
170-
<name>Jitpack repo</name>
171-
<url>https://jitpack.io</url>
135+
<name>Jitpack repo</name>
136+
<url>https://jitpack.io</url>
172137
</repository>
173138
</repositories>
174139
```
@@ -187,6 +152,53 @@ $ lein jar
187152
$ lein with-profile test test
188153
```
189154

155+
156+
# Definition of Done for 1.0 release
157+
158+
* On APIs - While 1.0 will be a major release with some inevitable API breakage, we will strive:
159+
* To maintain API compatiblity with prior releases to the greatest extent possible
160+
* To remove duplicated functionality (which may contradict the API compatibility principle)
161+
* To maintain API compatibility with code copied from internal Brad's Deals projects to the greatest extent possible.
162+
* Clear naming convention for namespaces that are provisional (non-frozen, WIP) API.
163+
164+
* Known tech debt that may influence APIs
165+
* Consider migrating db.clj's local configuration mechanism to config.clj (and then make an instance of the generic implementation inside db.clj)
166+
167+
* Target Clojure 1.9
168+
* All functions will have type information supplied via Clojure 1.9 specs.
169+
* Functions currently using plumatic/schema to specify type constraints will be migrated to Specs
170+
171+
* On testing
172+
* All functions will be unit tested using a style designed to illustrate behavior under failure modes as well as happy path scenarios.
173+
* We will collectively agree on a style of testing that seems to hit the sweet spot between overspecification and underspecification.
174+
* (Details are not set in stone but definitely up for negotiation)
175+
* When defects are detected, we will first reproduce the defect's root cause using a failing test case that will prevent the defect from reoccurring without the test notifying us.
176+
* Generative testing with Specs
177+
* Integrate kbitz (suggest Clojure idioms), some Clojure linting library; if it's easy to make these available to clients of clj-foundation, do so.
178+
179+
* Debug library
180+
* dbg, ppdbg macros
181+
* trace?
182+
183+
* Documentation
184+
* Machine generate as much as possible?
185+
186+
* Make clj_infrastructure as a separate library
187+
* Move db.clj there?
188+
* Abstract db.clj API using monads at the foundation layer and implement for relational, nosql, etc. in infrastructure?
189+
* http://rea.tech/the-worst-thing-in-our-scala-code-futures/ ?
190+
* Abstractions for creating/consuming lazy sequences?
191+
* Abstractions for aggregating a sequence of database results into a map[concatinated-key value(s)]
192+
193+
## Provisional ideas for 1.0
194+
195+
* Figure out how to isolate namespaces and their dependencies at the classpath level, including runtime reloading and evolution semantics after the fashion of OSGi at the REPL or in a running application.
196+
* Update and include clojure.osgi for clean deployment into OSGi frameworks?
197+
* Deploy to Clojars?
198+
199+
200+
201+
190202
## License
191203

192204
Copyright © 2015, 2016 by ShopSmart, LLC. Licensed under the Eclipse Public License v1.0.

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

+6-6
Original file line numberDiff line numberDiff line change
@@ -167,20 +167,20 @@
167167
Ex. some_name -> :some-name"
168168
([name :- s/Str
169169
naming-exceptions :- #{s/Str}]
170-
(if-let [name-exception (naming-exceptions name)]
171-
name-exception
172-
(keywordize name)))
170+
(if-let [name-exception (naming-exceptions name)]
171+
name-exception
172+
(keywordize name)))
173173
([name :- s/Str]
174-
(keywordize name)))
174+
(keywordize name)))
175175

176176

177177
(defn string-list->keywords
178178
"Convert a list of strings to a list of keywords"
179179
([list]
180-
(map (fn [string]
180+
(map (fn [string]
181181
(string->keyword (name string))) list))
182182
([list naming-exceptions]
183-
(map (fn [string]
183+
(map (fn [string]
184184
(string->keyword (name string) naming-exceptions)) list)))
185185

186186

src/clj_foundation/errors.clj

+1-1
Original file line numberDiff line numberDiff line change
@@ -115,7 +115,7 @@
115115
(throw (IllegalStateException. error-message))
116116
(if-not completed
117117
(do
118-
(Thread/sleep 500)
118+
(Thread/sleep 100)
119119
(recur (predicate)))
120120
completed))))))
121121

src/clj_foundation/io.clj

+5-2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
[clj-foundation.errors :as err]
44
[clj-foundation.templates :as template]
55
[clj-foundation.patterns :as p]
6+
[clj-foundation.data :as data]
67
[clojure.java.io :as io]
78
[clojure.edn :as edn])
89

@@ -142,7 +143,9 @@
142143

143144
(try
144145
(s/validate ClojureFileInputs result)
145-
result
146+
(if (string? result)
147+
(data/replace-nil (io/resource result) result)
148+
result)
146149
(catch Throwable e
147150
(throw (IllegalArgumentException.
148151
(str "Expected result to be one of " (s/explain ClojureFileInputs)
@@ -177,7 +180,7 @@
177180
(= argc 1) (read-file {:resource (first resource-spec)})
178181
(= argc 2) (read-file [(first resource-spec) (second resource-spec)])
179182
:else (throw (IllegalArgumentException.
180-
(str "resource-as-string: Illegal arg list: " resource-spec) )))))
183+
(str "resource-as-string: Illegal arg list: " resource-spec))))))
181184

182185

183186
(defn- parse-extended-file-location [input subs]

test/clj_foundation/errors_test.clj

+39-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,8 @@
66
[clj-foundation.errors :refer :all]
77
[clj-foundation.data :refer [any?]]
88
[clj-foundation.patterns :as p]
9-
[clj-foundation.millis :as millis]))
9+
[clj-foundation.millis :as millis])
10+
(:import [java.util Date]))
1011

1112

1213
(common/register-fixtures)
@@ -71,6 +72,43 @@
7172
(is (= 5 (count (seq<- e5)))))))
7273

7374

75+
(deftest expect-within-test
76+
(testing "Immediate success returns immediately"
77+
(let [before (.getTime (Date.))
78+
success (expect-within (millis/<-seconds 2)
79+
(constantly true)
80+
"Always succeeds immediately")
81+
after (.getTime (Date.))
82+
time (- after before)]
83+
84+
(is success)
85+
(is (< time (millis/<-seconds 1/4)))))
86+
87+
(testing "Eventual success succeeds"
88+
(let [tries (atom 0)
89+
before (.getTime (Date.))
90+
success (expect-within (millis/<-seconds 5)
91+
(fn [] (swap! tries inc) (< 3 @tries))
92+
"Always succeeds eventually")
93+
after (.getTime (Date.))
94+
time (- after before)]
95+
96+
(is success)
97+
(is (< time (millis/<-seconds 4)))))
98+
99+
(testing "Taking too much time throws IllegalStateException"
100+
(let [timeout (millis/<-seconds 1)
101+
before (.getTime (Date.))
102+
result (try* (expect-within timeout
103+
(constantly false)
104+
"Never succeeds"))
105+
after (.getTime (Date.))
106+
time (- after before)]
107+
108+
(is (instance? IllegalStateException result))
109+
(is (< timeout time)))))
110+
111+
74112
(deftest retry?-test
75113
(testing "Retry up to :max-retries times"
76114
(let [r0 (new-default-job "timeout-test"

test/clj_foundation/io_test.clj

+4
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,10 @@
1515

1616

1717
(deftest normalize-file-input-test
18+
(testing "If a string input references a resource, a resource URL is returned."
19+
(is (instance? URL (normalize-file-input "_test-config.edn")))
20+
(is (= "/some/file/path" (normalize-file-input "/some/file/path"))))
21+
1822
(testing "Return the default value as a resource when the envar is not defined"
1923
(is (instance? URL (normalize-file-input ["CONFIG-PRODDD" "_test-config.edn"]))))
2024

0 commit comments

Comments
 (0)