|
1 | 1 | # Introduction to clj-foundation
|
2 | 2 |
|
3 |
| -TODO: write [great documentation](http://jacobian.org/writing/what-to-write/) |
| 3 | +Clojure has a great standard library. Even so, some things could be easier than they are. |
| 4 | + |
| 5 | +## Why clj-foundation? |
| 6 | + |
| 7 | +clj-foundation supplies namespaces making additional simple things easy and hard things possible in Clojure for functions needed across all Clojure projects at Brad's Deals. |
| 8 | + |
| 9 | +The following describe our core goals/values for clj-foundation: |
| 10 | + |
| 11 | +* Enhances the core language in resonable, useful, and conservative ways. |
| 12 | +* Not a framework. Rather, a conservative set of generally-useful functions that may be used together or separately. |
| 13 | +* Makes it easy to create new transducers (or pipelines of transducers) from any regular function. |
| 14 | +* Where applicable, enables programming using a monadic style without requiring explicit monad types. |
| 15 | + * Clojure already polymorphically supports mapcat over most core data types. The missing thing is an implementation of the monadic zero as a separate and distinct thing from nil. |
| 16 | +* Describes, specifies, and illustrates best practices at Brad's Deals for working in Clojure. |
| 17 | +* The only dependencies are Clojure.*, Potemkin, io.aviso/pretty (for better stack traces), and Schema in order to minimize adoption friction. In the near future we will migrate from Schema to Specs. |
| 18 | + |
| 19 | + |
| 20 | +## Other Clojure foundational libraries that compliment this one |
| 21 | + |
| 22 | +* Immutant: http://immutant.org/ |
| 23 | +* Hara: http://docs.caudate.me/hara/index.html |
| 24 | + |
| 25 | + |
| 26 | +## High-quality domain-specific libraries complimenting this |
| 27 | + |
| 28 | +* Cassandra as a mutable, versioned map: https://github.com/MyPost/cassius |
| 29 | +* Git from the Clojure REPL: https://github.com/zcaudate/gita |
| 30 | + |
| 31 | + |
| 32 | +## Curated directories of Clojure libraries |
| 33 | + |
| 34 | +* http://clojurewerkz.org |
| 35 | +* http://blog.takipi.com/the-top-100-clojure-libraries-in-2016-after-analyzing-30000-dependencies/ |
| 36 | +* https://github.com/razum2um/awesome-clojure |
| 37 | +* http://www.clojure-toolbox.com/ |
| 38 | + |
| 39 | + |
| 40 | +# Features |
| 41 | + |
| 42 | +The folowing is a small sample of clj-foundation's features: |
| 43 | + |
| 44 | +## Math |
| 45 | + |
| 46 | +* A Mixed Number type |
| 47 | +* Time conversions to/from milliseconds |
| 48 | +* (str (millis/->dmhs elapsed-time)) => e.g.: "5 days 2 hours 3 minutes 2 seconds" |
| 49 | +* A Nothing type that behaves like the identity value for Maps, Seqs, and Strings and is distinct from nil. |
| 50 | +* Functions converting identity values of various types/operations to and from nil. |
| 51 | + |
| 52 | + |
| 53 | +## Ever had to declare a map containing the result of let variable bindings? Now you can do that in one step with let-map. |
| 54 | + |
| 55 | +```clojure |
| 56 | +(let-map [meaning 42 |
| 57 | + secrets (io/read-file "/etc/passwd")]) |
| 58 | +==> |
| 59 | +{:meaning 42 |
| 60 | + :secrets "nobody:*:-2:-2:Unprivileged User:/var/empty:/usr/bin/false\nroot:*:0:0:System Administrator:/var/root:/bin/sh\n...."} |
| 61 | +``` |
| 62 | + |
| 63 | +## Data/error processing enhancements and timeout handling. |
| 64 | + |
| 65 | +```clojure |
| 66 | +;; Replace a nil value with a default |
| 67 | +(replace-nil (:first-name customer) "John") |
| 68 | + |
| 69 | +;; Returns value if non-nil else throws IllegalArgumentException naming the nil value |
| 70 | +(not-nil (:first-name customer) "First name") |
| 71 | + |
| 72 | +;; Expect a condition specified by a predicate to become true within timeout-millis. |
| 73 | +;; Throws IllegalStateException on failure with the specified message. |
| 74 | +(expect-within |
| 75 | + (millis/<-seconds 5) |
| 76 | + (fn [] (= (meaning-of-life) 42)) |
| 77 | + "Couldn't compute the meaning of life.") |
| 78 | + |
| 79 | +;; Retry x times, with millis puase interval a specified function that is failure-prone |
| 80 | +(retry 5 (millis/<-seconds 10) some-unreliable-io-operation) |
| 81 | +``` |
| 82 | + |
| 83 | +## A string interpolation language that is intentionally dumb about the syntax of its hosting language. |
| 84 | + |
| 85 | +```clojure |
| 86 | +;; Given: |
| 87 | +(def content "Say hello to ${NAME}") |
| 88 | + |
| 89 | +;; Then: |
| 90 | +(templates/subst<- content :NAME "Jack") |
| 91 | +==> |
| 92 | +"Say hello to Jack" |
| 93 | + |
| 94 | +(subst<- content :ZIP "46989") |
| 95 | +==> |
| 96 | +ExceptionInfo Not found: '${NAME}' clojure.core/ex-info |
| 97 | + |
| 98 | +(interpolation-vars content) |
| 99 | +==> |
| 100 | +(:NAME) |
| 101 | +``` |
| 102 | + |
| 103 | +* Interpolation variables are resolved from the following places with the following precedence: |
| 104 | + * Java system variables (e.g.: java -DNAME='Jack' com.foo.bar.YourMainProgram) |
| 105 | + * Operating system environment variables |
| 106 | + * Any key/value variables specified in code. |
| 107 | + |
| 108 | +## Extensions to Clojure's file input functions |
| 109 | + |
| 110 | +```clojure |
| 111 | +;; Return a Jar resource as a string |
| 112 | +(resource-as-string "config.txt") |
| 113 | + |
| 114 | +;; Return a Jar resource as a string, but allow the ALT_CONFIG_LOCATION Java system or O/S environment |
| 115 | +;; variable to redirect reading to an external configuration file instead. |
| 116 | +(resource-as-string "ALT_CONFIG_LOCATION" "config.txt") |
| 117 | + |
| 118 | +;; Return a Jar resource as a string, interpolating any template variables inside the Jar resource |
| 119 | +;; using the supplied interpolation variables and the override rules defined by the template language. |
| 120 | +(read-template "config.properties" :NAME "Jack") |
| 121 | + |
| 122 | +;; Return a Jar resource as a string, possibly overridden by the environment variable ALT_CONFIG_LOCATION |
| 123 | +;; per the template engine's precedence rules. Any template variables in the file will also be substuted |
| 124 | +;; using the supplied key/value pairs and/or matching environment variables per the template engine's |
| 125 | +;; precedence rules. |
| 126 | +(read-template "ALT_CONFIG_LOCATION" "config.properties" :NAME "Jack") |
| 127 | +``` |
| 128 | + |
| 129 | +## And more... |
| 130 | + |
| 131 | +* Easier serialization / deserialization in various formats. Currently supports binary and EDN formats. Optionally can use the template language to support substitution variables inside EDN files. |
| 132 | +* A let-map form similar to "let" that returns its names and values as a Map--for those times the values of subsequent keys in a Map depend on the values of prior ones. |
| 133 | +* Simple named implementations of design patterns familiar to Java developers, particularaly Singleton. |
| 134 | + |
| 135 | +# Deployment/usage |
| 136 | + |
| 137 | +## Leiningen coordinates |
| 138 | + |
| 139 | +```clojure |
| 140 | +:user {:repositories [["jitpack" "https://jitpack.io"]] |
| 141 | + |
| 142 | + :dependencies [[com.github.shopsmart/clj-foundation "version"]]} |
| 143 | +``` |
| 144 | + |
| 145 | +where "version" currently is "[](https://jitpack.io/#shopsmart/clj-foundation)". |
| 146 | + |
| 147 | +## Maven coordinates |
| 148 | + |
| 149 | +```xml |
| 150 | +<repositories> |
| 151 | + <repository> |
| 152 | + <id>jitpack.io</id> |
| 153 | + <name>Jitpack repo</name> |
| 154 | + <url>https://jitpack.io</url> |
| 155 | + </repository> |
| 156 | +</repositories> |
| 157 | +``` |
| 158 | + |
| 159 | +* GroupId: com.github.shopsmart |
| 160 | +* ArtifactId: clj-foundation |
| 161 | +* Version: [](https://jitpack.io/#shopsmart/clj-foundation) |
| 162 | + |
| 163 | +## Manual build/test |
| 164 | + |
| 165 | +```bash |
| 166 | +# Build |
| 167 | +$ lein jar |
| 168 | + |
| 169 | +# Test |
| 170 | +$ lein with-profile test test |
| 171 | +``` |
0 commit comments