clj-lib.core

Useful fns, of all purposes, that might be considered language extensions.

assoc-nx

(assoc-nx m k v)
Associates `k` to `v` in `m` only if `k` is not present in `m`.

avg

(avg & vs)
Returns the average of the numbers `vs`; a la clj min/max

between

(between v min max)
Returns true if `v` is between `min` and `max`, inclusive of both.

catch-and-return-throwable

macro

(catch-and-return-throwable & body)
Tries `f`. If `f` throws a `Throwable`, it is caught and returned.

clamp

(clamp nmin nmax)(clamp n nmin nmax)
Clamps number `n` between `min` and `max` inclusive of each. When called with
only two args, produces a transducer.

deep-merge

(deep-merge & maps)
Like merge, but recursive.

deep-merge-with

(deep-merge-with f & maps)
Like `merge-with`, but recursive so it works more than 1-level deep.

defdata

macro

(defdata name path)
Macro for compiling some data at build time from an edn file, to avoid
runtime lookups to the file.

disj-in

(disj-in m ks elem)
Roughly equivalent to `(update-in m ks disj elem)`, except that empty sets
are dissociated.

dissoc-eq

(dissoc-eq m k expected)
Dissociates `k` from `m` only if `(get m k)` equals `expected`.

escape-quotes

(escape-quotes s)
Double-escapes double-quotes in `s`.

get-and-reset!

(get-and-reset! atom newval)
Sets the value of `atom` to `newval` without regard for the current value.
Returns the previous value of `atom`.

get-and-swap!

(get-and-swap! atom f)(get-and-swap! atom f x)(get-and-swap! atom f x y)(get-and-swap! atom f x y & args)
Atomically swaps the value of atom to be:
`(apply f current-value-of-atom args)`. Note that f may be called multiple
times, and thus should be free of side effects.
Returns the last `current-value-of-atom` before swap.

map-kv

(map-kv f m)
Maps `f` on the vals of `m`, producing a new map with the same keys.

maybe?

(maybe? pred? val)
Returns true if either `val` is nil or `(pred? val)` returns truthful.

Useful for situations where `nil` is acceptable, but when non-nil `val` is
expected to meet some criterion.

Examples:

    (maybe? map? m)
    (maybe? string? s)

min-max

(min-max coll)
Returns [(min coll) (max coll)] while iterating coll only once

not-neg?

As named. Equivalent to `(or (pos? v) (zero? v))`.

quote-string

(quote-string s)
Returns `s` surrounded by double-quotes.

removev

(removev v i)
Returns vector `v` with index `i` removed.  `v` is nil, returns nil. `i` must
be a 0-indexed integer between 0 and the length of `v`.

retry

macro

(retry count pred? & body)
Tries to execute `body` up to the number of times specified or until `pred?` returns
truthful given the result of executing `body`.

retry*

(retry* n pred? f)
Tries to execute `f` up to the number of times specified or until `pred?` returns
truthful given the result of executing `f`. If `f` throws, the thrown value is caught
and passed to `pred?`.

This can be used to retry until a non-Exception response, like so:
```
(retry* 5 (complement #(instance? Throwable %)) some-fn)
```

Or, used to retry until a non-nil response, like so:
```
(retry* 5 (complement nil?) some-fn)
```

When `n` retries are reached, the result of the final `f` will be returned or thrown
if it is throwable, regardless of `pred?`.

round

(round precision)(round precision n)
Rounds a number to the specified precision. This is a modified version of the
round fn used here: http://clojure-doc.org/articles/language/functions.html
Compared to the reference, the argument order is reversed so it is more easily
used in partially applied form as a map fn. When called with only a precision,
returns a transducer.

safe

macro

(safe & body)
Suppresses any exceptions that are thrown, returning nil instead.

str=

(str= & args)
Evaulates equality by converting all args to strings and then testing with
string equality. Useful when comparing stringy values that may not all be
strings, such as `1`, `"1"`, and `:1`.

thread

macro

(thread & body)
Executes `body` in a new thread. Like `future`, except it returns the thread
instead of the result of `body`.

throw-if-throwable

(throw-if-throwable v)
Implementation detail.

If `v` is a Throwable, throws `v`. Otherwise, returns `v`.