What to do in lua if I want to pass a value to foo(), pass it's return to bar(), pass it's return to baz() etc?
Exhibit A (pseudocode):
backwards(code(reading(like(not(do(I))))))
Exhibit B (pseudocode):
local all = foo(...)
local this = bar(all)
local variables = baz(this)
local obscure_a_simple_idea = quux(variables)
In F# this code (source)
let numbers = [0..100]
let evenNumbers = List.filter isEven numbers
let doubledNumbers = List.map double evenNumbers
List.iter printNumber doubledNumbers
can be rewritten like this:
[0..100]
|> List.filter isEven
|> List.map double
|> List.iter printNumber
In clojure this code (source)
(defn calculate []
(reduce + (map #(* % %) (filter odd? (range 10)))))
Can be written like this
(defn calculate* []
(->> (range 10)
(filter odd? ,,,)
(map #(* % %) ,,,)
(reduce + ,,,)))
I am aware that fennel has threading macros. What I would like to know: what can I do in lua? Did I miss a language feature? Is there some well-known ideom I am not aware of? Am I stuck choosing between exhibit A, exhibit B, or migrating to fennel?
Good news: Lua has first-class functions so pretty much all of this is possible in plain Lua! (It may be more verbose and/or inefficient, though.)
Let's start with function composition:
I've flipped
f
andg
here, sof(g(x))
would be equivalent tocompose(g, f)(x)
.Let's write a left fold on a vararg:
Let's also define the identity function:
Now let's take care of Exhibit A:
Since chaining is just composition with another order, it returns a function, so the argument is still on the right. If you're not happy with this, you could write yourself a function which takes the argument and does the call, similar to the threading F# / Clojure have to offer:
Now your Exhibit A becomes
chaincall(I, dont, like, reading, code, backwards)
.Exhibit B is pretty much the same: You can do
chain(foo, bar, baz, quux)(...)
.chaincall
won't work with a vararg as first parameter, though: The vararg would be truncated to its first value.Voilà! Essentially we've implemented Clojure's
->>
now:chaincall(x, ...)
is equivalent to(->> x ...)
.F#'s
|>
is somewhat harder to do, because it requires us to somehow deal with a list of arguments to follow the "threaded" argument per function. One way to do it would be using a "right curry" which appends arguments to a function call:Then you could just do:
similar to F# (assuming suitable definitions of
range
etc.)As is often the case in Lua, it gives you the power to implement this yourself, pushing tradeoffs and design decisions on you.
Even with the simple implementations I have provided, there are discussions to be had:
compose_multiple
using a table of functions, for example. You could consider using the identity function and making a fold which requires a vararg with at least one element.A related library worth mentioning is
luafun
, which provides functions likefilter
andmap
. That library also showcases another frequently chosen way of (emulating) this "chaining": By means of "methods" on "objects" (tables with metatables) which in turn return the objects. For example you could have an "iterator" object. Callingiterator:filter(f)
would return a new iterator, on which you could then callmap
:iterator:filter(f):map(g)
etc.Using Lua's metatables, you could even get some syntactic sugar. Let's repurpose the right shift operator (
f >> g
) for "flipped" function composition:(In older Lua versions, you'd have to use a different operator. Also note that this probably isn't the best idea in "serious" code due to performance and readability concerns; such "abuse" of the debug library is usually frowned upon in "serious" code, and many "safe" environments may remove or restrict the
debug
library altogether. Static analysis tools might also get confused.)