In Common Lisp, how to define a “meta-macro” which takes as argument a list of macros (and other arguments) and composes these macros to produce the desired code.
The problem is equivalent to writing a “higher-order macro” which defines a macro out of a variable list of other macros.
The concrete situation prompting the question is for me an experiment with CLSQL, where I want to re-express the employee class from the CLSQL-testsuite
(clsql:def-view-class employee ()
((employee-id
:db-kind :key
:db-constraints (:not-null)
:type integer)
(first-name
:accessor employee-first-name
:type (string 30)
:initarg :first-name)
(last-name
:accessor employee-last-name
:type (string 30)
:initarg :last-name)
(email
:accessor employee-email
:type (string 100)
:initarg :email)
(company-id
:type integer
:initarg :company-id)
(company
:accessor employee-company
:db-kind :join
:db-info (:join-class company
:home-key companyid
:foreign-key companyid
:set nil))
(manager-id
:type integer
:nulls-ok t
:initarg :manager-id)
(manager
:accessor employee-manager
:db-kind :join
:db-info (:join-class employee
:home-key managerid
:foreign-key emplid
:set nil))))
as
(def-view-class-with-traits employee ()
(trait-mapsto-company trait-mapsto-manager)
((employee-id
:db-kind :key
:db-constraints (:not-null)
:type integer)
(first-name
:accessor employee-first-name
:type (string 30)
:initarg :first-name)
(last-name
:accessor employee-last-name
:type (string 30)
:initarg :last-name)
(email
:accessor employee-email
:type (string 100)
:initarg :email)))
Having this technique at hand would favour consistency and terseness when defining complex database schemas.
I defined the two traits I need as
(defmacro trait-mapsto-company (class super slots &rest cl-options)
(declare (ignore super slots cl-options))
(let ((company-accessor-name
(intern (concatenate 'string (symbol-name class) "-COMPANY"))))
`((company-id
:type integer
:initarg :company-id)
(company
:accessor ,company-accessor-name
:db-kind :join
:db-info (:join-class company
:home-key companyid
:foreign-key companyid
:set nil)))))
(defmacro trait-mapsto-manager (class super slots &rest cl-options)
(declare (ignore super slots cl-options))
(let ((manager-accessor-name
(intern (concatenate 'string (symbol-name class) "-MANAGER"))))
`((manager-id
:type integer
:initarg :manager-id)
(manager
:accessor ,manager-accessor-name
:db-kind :join
:db-info (:join-class manager
:home-key managerid
:foreign-key emplid
:set nil)))))
However my attempt to write the def-view-class-with-traits
is foiled.
(defmacro def-view-class-with-traits (class super traits slots &rest cl-options)
(let ((actual-slots
(reduce (lambda (trait ax) (append (apply trait class super slots cl-options) ax))
traits
:initial-value slots)))
`(clsql:def-view-class ,class ,super ,actual-slots ,@cl-options)))
In the lambda used for reducing, the trait
stands for a macro, and my use of apply does not make any sense to the Lisp – which is right! – but hopefully convey my intent to other programmers.
How to let def-view-class-with-traits
process the list of macros traits
in the appropriate way?