modula’s many-multi mega-miscellaneous meta-modulated macro-mutated utilities
A collection of various utility functions and macros I use in my libraries.
This is mostly for deduplication of code so I don’t have to keep several copies of these functions across different projects. I don’t really recommend depending on this library in your code since I may rename, rearrange, or remove things without warning. That being said, if you have any issues or need help with the library, please don’t hesitate to let me know.
All exported symbols are documented via their docstrings, which are of course accessible via Lisp’s standard documentation
and describe
functions.
Quickly generate lists containing repeated items or numeric series.
;; four 4s:
(a 1 4!4 2) ;=> (1 4 4 4 4 2)
;; 10 random numbers:
(a 'hello (random 10)!10 'goodbye) ;=> (HELLO 2 5 3 9 5 0 9 1 9 6 GOODBYE)
;; series from -2 to 2:
(a 'foo -2..2 'bar) ;=> (FOO -2 -1 0 1 2 BAR)
A sweeter way to write lambda
. Underscore (_
) is treated as the argument of the lambda. Multiple arguments can be specified with a number after the underscore (_0
, _1
, etc).
(fn (random 10))
;; is the same as
(lambda () (random 10))
;; and
(fn (+ 2 _))
;; is the same as
(lambda (_) (+ 2 _))
;; and
(fn (- _1 _0))
;; is the same as
(lambda (_0 _1) (- _1 _0))
Another sweet way to write lambda
. Notation for specializing parameters without currying, as described in SRFI 26.
(cut '/ 1 <>) ;=> (lambda (x) (/ 1 x))
(cut <> 1 2) ;=> (lambda (func) (funcall func 1 2))
(cut '+ <> <>) ;=> (lambda (x y) (+ x y))
Get a list of the keys of hash tables, plists, and anything else that defines a method for it. For example, cl-patterns defines a method for its event
class.
(keys '(:foo 1 :bar 2)) ;=> (:FOO :BAR)
(keys (alexandria:plist-hash-table (list :this 1337 :that 69))) ;=> (:THIS :THAT)
Concatenate its non-nil inputs together into a string, similar to the elisp function of the same name.
(defun say-hello (name &optional ask)
(concat "Hello, " name "!" (when ask " How are you doing today?")))
(say-hello "Allison" t) ;=> "Hello, Allison! How are you doing today?"
(say-hello "Gordon" nil) ;=> "Hello, Gordon!"
Parse a string as a boolean by looking for common true/false inputs like y/n, 1/0, on/off, true/false, enable/disable, etc. If the input is not a known boolean-like string, defaults to the specified default value
(parse-boolean "1") ;=> t
(parse-boolean "0") ;=> nil
(parse-boolean "y") ;=> t
(parse-boolean "N") ;=> nil
(parse-boolean "blah" t) ;=> t ;; "blah" can't be interpreted as a boolean, so it defaults to the provided value of t.
A few functions to generate more human-readable strings from numeric values.
;; turn improper fractions into proper ones:
(friendly-ratio-string 13/4) ;=> "3 1/4"
(friendly-ratio-string 99/13) ;=> "7 8/13"
;; turn a number of seconds into the typical time notation:
(friendly-duration-string 67) ;=> "1:07"
;; 3600 seconds is one hour:
(friendly-duration-string 3600) ;=> "1:00:00"
Wrap a number within a range like cl:mod
but taking into account a lower bound as well.
Round, floor, or ceiling to the nearest multiple of a given number with round-by
, floor-by
, and ceiling-by
.
(round-by 0.39 0.2) ;=> 0.4
(round-by 97 25) ;=> 100
(floor-by 0.39 0.2) ;=> 0.2
(floor-by 97 25) ;=> 75
(ceiling-by 0.22 0.2) ;=> 0.4
(ceiling-by 27 25) ;=> 50
Get the most X item in a list, where X can be any comparison function. Similar to the standard reduce
function, except that the key
argument is only used for comparison, and the actual item from the list is still returned.
;; get the item with the smallest car:
(most '< '((2 :bar) (3 :baz) (1 :foo)) :key 'car) ;=> (1 :FOO)
;; compare this to `reduce', which returns the result of calling KEY on the item, instead of returning the item itself:
(reduce 'min '((2 :bar) (3 :baz) (1 :foo)) :key 'car) ;=> 1
Like alexandria:flatten
but only flattens one layer.
(flatten-1 '(1 2 (3 4 (5 6 7) 8) 9 10)) ; => (1 2 3 4 (5 6 7) 8 9 10)
(alexandria:flatten '(1 2 (3 4 (5 6 7) 8) 9 10)) ; => (1 2 3 4 5 6 7 8 9 10)
Like the standard subseq
, but the START and END parameters can be negative to represent indexing from the end of the list.
(subseq* (list 0 1 2 3 4 5) -3) ;=> (3 4 5)
(subseq* (list 0 1 2 3 4 5) -3 -1) ;=> (3 4)
Like string-left-trim
but for lists instead of strings.
Flip a virtual coin, returning t
on heads and nil
on tails.
(loop :repeat 10 :collect (random-coin)) ; => (T T NIL T T NIL NIL NIL T T)
;; a probability of heads can also be specified:
(loop :repeat 10 :collect (random-coin 0.9)) ; => (T T NIL NIL T T T T T T)
Get a random number within a specified range.
(loop :repeat 5 :collect (random-range -10 10)) ; => (-10 8 -1 -1 -9)
;; also supports floats:
(loop :repeat 5 :collect (random-range -10.0 10.0)) ; => (-5.949123 7.214485 -0.6976509 2.68153 9.699009)
Open a URL using the operating system’s default opener.
Functionality for mapping numbers from one range to another.
Looping functionality is in the “loopy” subsystem; run (ql:quickload :mutility/loopy)
to load it.
Like the standard mapcar
and dolist
, but includes the current index into the list.
Your standard “while” loop that repeats its body as long as its test condition is true. Additionally, it will return the last non-nil value it processed in the body or the test.
Like while
, but the body is run before the test condition is checked; i.e. the body is always run at least once.
The opposite of while
; runs its body as long as its test condition is false.
Efficiently append to a list, which is then returned.
(accumulating (dotimes (n 5) (accumulate (random 10)))) ;=> (0 2 3 4 1)
mutility/loopy
is a small collection of various looping constructs likedolist*
,while
,do-while
, etc.mutility/files
is a set of file-related functionality.mutility/generic-cl
defines a few extensions to the generic-cl library.mutility/test-helpers
includes a few functions that are mostly useful for test suites.mutility/tests
is the FiveAM-based test suite for the library.
All source files are in the src/
directory.
- package.lisp - the package definition file.
- mutility.lisp - mutility’s “standard” functionality.
- sugar.lisp - syntax shorteners and sweeteners.
- ringbuffer.lisp - ringbuffer implementation.
- ranges.lisp - define and translate between different types of ranges.
- test-helpers.lisp - a few introspection functions to make testing easier.
- loopy.lisp - various looping primitives.
- scrapyard.lisp - failed experiments, old versions, and other code refuse.
Mutility also includes a few extensions for other systems in src/extensions/
:
- generic-cl-extensions.lisp - extensions to the generic-cl library. FIX
- cl-org-mode-extensions.lisp - extensions to the cl-org-mode library. FIX
- swank-extensions.lisp - extensions to swank. FIX
The test suite is located in t/
. To run the tests:
(asdf:test-system :mutility)
Ideas, and things that need to be done.
- Come up with a better name for the
a
macro. - Write more tests for everything.
- Test docstring examples with the docstring-parsing function once it’s written.
- Write a test to check for symbol clashes against various other libraries:
alexandria
,serapeum
,cl-patterns
,thundersnow
, etc.