about:drewcsillag

Apr 26, 2014 - 5 minute read - programming lisp macros

Fun With Lisp Macros

As far as programming languages go, my truly favorite language is a Lisp. Scheme or Common Lisp are fine. I have a few hesitations about Clojure (the way it does recursion is offputting to me, but perhaps a few macros might clean it up), but for what I’ve done with it when working with Riemann, it’s been a pleasure, even if it’s slow going because I don’t do it frequently enough to have the needed bits already in my head.

In one of my previous posts, I wrote about how I’d written the website software for a website I maintain in Common Lisp. There were a few macros I wrote while building it that I thought were potentially useful and illustrative of the kinds of things you can do with macros that make Lisps cool.

In this post (there may be others later), I’ll be talking about a macro I wrote to deal with database queries. When dealing with database queries, making sure that you write/generate the queries properly with respect to escaping and bind variables, and connection caching, etc. can be a real pain, and I’m not a big fan of ORMs as they never really do what I want.

What I wanted was to be able to just have a macro that would just do The Right Thing ™. Ideally, in a simple case it would be something like this:

(doquery "select body from boxes where bid = " scripture-of-the-week-box-id)

Where this would execute the query:

select body from boxes where bid = ?

with the scripture-of-the-week-box-id bound appropriately. Better yet, if I could do something like this:

(doquery "select name from names where id in " name-ids " order by names " (:SQL sort-order))

where if name-ids were (23423, 125235, 34234) and sort-order were "ASC", it would execute

select name from names where id in (?, ?, ?) order by names ASC

with name-ids bound appropriately.

“Can’t you do the same thing in non-lisp languagues?” you might ask. Not really. “Why not?” you ask. Simple: because if you were in a non-lisp language, you couldn’t tell the difference between the SQL string literals, and any string-resulting expressions that might get passed in. With lisp macros, since you have access to the expressions before they’re evaluated, we can look at the expression to know if an argument to it was a string literal or an expression or a (:SQL expr) construct.

In the macro as used above, if anything we have a little of the opposite problem that results in the need of the (:SQL expr) construct above – it makes it so the query generator doesn’t treat the expression as something that needs to be bound, but in my experience, that tends to be the exception rather than the rule.

So how do we go about putting something together that does this?

(defmacro doquery (&body args)
  ;; stuff here
)

For those who’ve not seen Common Lisp before, the &body args bit is essentially the *args parametr in Python, or whatever the “just give me all the args they passed in” declaration in your language of choice.

First we need a function that can determine if the item in args is a literal string, a (:SQL expr) or a value that we’ll need to stuff into bind variables.

(defun literal-or-dovalue (x)
    (if (stringp x) x
        (util:aif (literal-sql x) util:it
            `(,dovalue ,x)))))

The util:if and util:it bits are an anaphoric if construct. Basically, it’s an if expression where util:it evaluates to the value of the conditional. In this case it’s the return value of the literal-sql function. The literal-sql function is defined as follows:

(defun literal-sql (x)
  (and (consp x)
       (eq (car x) :sql) (cadr x)))

This function returns the expression if the argument x is of the form (:SQL expr). So the effect of literal-or-dovalue is to return strings as is, (:SQL expr) expressions as expr and to turn everything else into (dovalue expr). We’ve not defined dovalue yet, so lets do that:

(let ((sql nil)
      (items (list)))

    (defun dovalue (item)
        (setf items (append items (ensure-list item)))
        (generate-placeholder item))
)

For the moment, we’ll ignore the fact that the function as specified will really only work once only due to the outer let. But what this does, is adds the item to the list of items which will be the list of items we pass as bind arguments to the SQL connection, and returns the proper placeholder for the item to embed into SQL.

The ensure-list function has a fairly obvious implementation:

(defun ensure-list (item)
  (if (consp item) item (list item)))

The generate-placeholder generates “?” for an atomic value and “?,?,…?” for list values using a smattering of format magic:

(defun generate-placeholder (item)
  (if (consp item)
      (format nil "~{~A~^,~}" (mapcar (lambda (x) (declare (ignore x)) "?") item))
      "?"))

All that’s left to do then is to compose the SQL string together:

(setf sql (apply #'util:strcat (mapcar #'literal-or-dovalue args)))

and execute it:

(apply #'plain-odbc:exec-query (append (list conn query) items))

In the real implementation, it gets the connection from a database pool rather than having it passed as a global as we’re doing here.

So now, we macroize the whole thing and get this:

(defmacro doquery (&body args)
  (util:with-gensyms (sql items dovalue)
    (flet ((literal-or-dovalue (x) 
             (if (stringp x) x (util:aif (literal-sql x) util:it `(,dovalue ,x)))))
      `(let ((,sql)
             (,items (list))) 
             (flet ((,dovalue (item)
                      (setf ,items (append ,items (ensure-list item)))
                      (generate-placeholder item)))
               (setf ,sql (apply #'util:strcat (list ,@(mapcar #'literal-or-dovalue args))))

               (apply #'plain-odbc:exec-query (append (list conn query) items)))))))

Here, we first use with-gensyms to make some names so we don’t capture anything we don’t want to. Then we flet to inline literal-or-dovalue since it doesn’t need to be seen outside the macro anywhere. Then we, per macro-invocation, set up the let statement we had before to wrap the dovalue function, so we can use it more than once :). Lastly we generate the sql and exec it.