- wrappers to one database engine (cl-sqlite, postmodern, cl-redis,ā¦),
- interfaces to several DB engines (clsql, sxql,ā¦),
- persistent object databases (bknr.datastore (see chap. 21 of āCommon Lisp Recipesā), ubiquitous,ā¦),
- Object Relational Mappers (Mito),
and other DB-related tools (pgloader).
Weāll begin with an overview of Mito. If you must work with an existing DB, you might want to have a look at cl-dbi and clsql. If you donāt need a SQL database and want automatic persistence of Lisp objects, you also have a choice of libraries.
The Mito ORM and SxQL
Mito is in Quicklisp:
(ql:quickload "mito")
Mito will load another system on the fly depending on your databaseās driver. These systems are:
:dbd-sqlite3
:dbd-mysql
:dbd-postgres
You can āquickloadā one of them now, or let Mito (actually cl-dbi) do it when required.
But if you build an executable of your program, and if you plan on using it on a machine where Quicklisp is not installed, you must reference the required additional system into your .asd system definition.
Overview
Mito is āan ORM for Common Lisp with migrations, relationships and PostgreSQL supportā.
- it supports MySQL, PostgreSQL and SQLite3,
- when defining a model, it adds an
id
(serial primary key),created_at
andupdated_at
fields by default like Rubyās ActiveRecord or Django, - handles DB migrations for the supported backends,
- permits DB schema versioning,
- is tested under SBCL and CCL.
As an ORM, it allows to write class definitions, to specify relationships, and provides functions to query the database. For custom queries, it relies on SxQL, an SQL generator that provides the same interface for several backends.
Working with Mito generally involves these steps:
- connecting to the DB
- writing CLOS classes to define models
- running migrations to create or alter tables
- creating objects, saving same in the DB,
and iterating.
Connecting to a DB
Mito provides the function connect-toplevel
to establish a
connection to RDBMs:
(mito:connect-toplevel :mysql
:database-name "myapp"
:username "fukamachi"
:password "c0mon-1isp")
The driver type can be of :mysql
, :sqlite3
and :postgres
.
With sqlite you donāt need the username and password:
(mito:connect-toplevel :sqlite3 :database-name "myapp")
As usual, you need to create the MySQL or PostgreSQL database beforehand. Refer to their documentation.
Connecting sets mito:*connection*
to the new connection and returns it.
Disconnect with disconnect-toplevel
.
You might make good use of a wrapper function:
(defun connect ()
"Connect to the DB."
(mito:connect-toplevel :sqlite3 :database-name "myapp"))
Connecting to an in-memory DB (SQLite)
To connect to a sqlite3
in-memory database, you use the ā:memory:ā
string as the DB name. It has a special meaning for SQLite.
(mito:connect-toplevel :sqlite3 :database-name ":memory:")
This doesnāt create a file on disk, and the DB will be even faster. But youāll loose all the data when you close the connection. It is consequently specially useful for unit-tests, to load data for temporary analytics, etc.
You can read more about in-memory SQLite databases here.
Models
Defining models
In Mito, you can define a class which corresponds to a database table with the deftable
macro:
(mito:deftable user ()
((name :col-type (:varchar 64))
(email :col-type (or (:varchar 128) :null))))
Alternatively, you can specify (:metaclass mito:dao-table-class)
in a regular class definition.
The deftable
macro automatically adds some slots: a primary key named id
if thereās no primary key, and created_at
and updated_at
for recording timestamps. Specifying (:auto-pk nil)
and (:record-timestamps nil)
in the deftable
form will disable these behaviours. A deftable
class will also come with initializers, named after the slot, and accessors, of form <class-name>-<slot-name>
, for each named slot. For example, for the name
slot in the above table definition, the initarg :name
will be added to the constuctor, and the accessor user-name
will be created.
You can inspect the new class:
(mito.class:table-column-slots (find-class 'user))
;=> (#<MITO.DAO.COLUMN:DAO-TABLE-COLUMN-CLASS MITO.DAO.MIXIN::ID>
; #<MITO.DAO.COLUMN:DAO-TABLE-COLUMN-CLASS COMMON-LISP-USER::NAME>
; #<MITO.DAO.COLUMN:DAO-TABLE-COLUMN-CLASS COMMON-LISP-USER::EMAIL>
; #<MITO.DAO.COLUMN:DAO-TABLE-COLUMN-CLASS MITO.DAO.MIXIN::CREATED-AT>
; #<MITO.DAO.COLUMN:DAO-TABLE-COLUMN-CLASS MITO.DAO.MIXIN::UPDATED-AT>)
The class inherits mito:dao-class
implicitly.
(find-class 'user)
;=> #<MITO.DAO.TABLE:DAO-TABLE-CLASS COMMON-LISP-USER::USER>
(c2mop:class-direct-superclasses *)
;=> (#<STANDARD-CLASS MITO.DAO.TABLE:DAO-CLASS>)
This may be useful when you define methods which can be applied for all table classes.
For more information on using the Common Lisp Object System, see the clos page.
Creating the tables
After defining the models, you must create the tables:
(mito:ensure-table-exists 'user)
So a helper function:
(defun ensure-tables ()
(mapcar #'mito:ensure-table-exists '(user foo bar)))
See Mitoās documentation for a couple more ways.
When you alter the model youāll need to run a DB migration, see the next section.
Fields
Fields types
Field types are:
(:varchar <integer>)
, text
,
:serial
, :bigserial
, :integer
, :bigint
, :unsigned
,
:timestamp
, :timestamptz
,
:bytea
,
Optional fields
Use (or <real type> :null)
:
(email :col-type (or (:varchar 128) :null))
Field constraints
:unique-keys
can be used like so:
(mito:deftable user ()
((name :col-type (:varchar 64))
(email :col-type (:varchar 128))
(:unique-keys email))
We already saw :primary-key
.
You can change the table name with :table-name
.
Relationships
You can define a relationship by specifying a foreign class with :col-type
:
(mito:deftable tweet ()
((status :col-type :text)
;; This slot refers to USER class
(user :col-type user))
(table-definition (find-class 'tweet))
;=> (#<SXQL-STATEMENT: CREATE TABLE tweet (
; id BIGSERIAL NOT NULL PRIMARY KEY,
; status TEXT NOT NULL,
; user_id BIGINT NOT NULL,
; created_at TIMESTAMP,
; updated_at TIMESTAMP
; )>)
Now you can create or retrieve a TWEET
by a USER
object, not a USER-ID
.
(defvar *user* (mito:create-dao 'user :name "Eitaro Fukamachi"))
(mito:create-dao 'tweet :user *user*)
(mito:find-dao 'tweet :user *user*)
Mito doesnāt add foreign key constraints for referring tables.
One-to-one
A one-to-one relationship is simply represented with a simple foreign
key on a slot (as :col-type user
in the tweet
class). Besides, we
can add a unicity constraint, as with (:unique-keys email)
.
One-to-many, many-to-one
The tweet example above shows a one-to-many relationship between a user and his tweets: a user can write many tweets, and a tweet belongs to only one user.
The relationship is defined with a foreign key on the āmanyā side
linking back to the āoneā side. Here the tweet
class defines a
user
foreign key, so a tweet can only have one user. You didnāt need
to edit the user
class.
A many-to-one relationship is actually the contrary of a one-to-many. You have to put the foreign key on the appropriate side.
Many-to-many
A many-to-many relationship needs an intermediate table, which will be the āmanyā side for the two tables it is the intermediary of.
And, thanks to the join table, we can store more information about the relationship.
Letās define a book
class:
(mito:deftable book ()
((title :col-type (:varchar 128))
(ean :col-type (or (:varchar 128) :null))))
A user can have many books, and a book (as the title, not the physical copy) is likely to be in many peopleās library. Hereās the intermediate class:
(mito:deftable user-books ()
((user :col-type user)
(book :col-type book)))
Each time we want to add a book to a userās collection (say in
a add-book
function), we create a new user-books
object.
But someone may very well own many copies of one book. This is an information we can store in the join table:
(mito:deftable user-books ()
((user :col-type user)
(book :col-type book)
;; Set the quantity, 1 by default:
(quantity :col-type :integer)))
Inheritance and mixin
A subclass of DAO-CLASS is allowed to be inherited. This may be useful when you need classes which have similar columns:
(mito:deftable user ()
((name :col-type (:varchar 64))
(email :col-type (:varchar 128)))
(:unique-keys email))
(mito:deftable temporary-user (user)
((registered-at :col-type :timestamp)))
(mito:table-definition 'temporary-user)
;=> (#<SXQL-STATEMENT: CREATE TABLE temporary_user (
; id BIGSERIAL NOT NULL PRIMARY KEY,
; name VARCHAR(64) NOT NULL,
; email VARCHAR(128) NOT NULL,
; registered_at TIMESTAMP NOT NULL,
; created_at TIMESTAMP,
; updated_at TIMESTAMP,
; UNIQUE (email)
; )>)
If you need a ātemplateā for tables which arenāt related to any
database tables, you can use DAO-TABLE-MIXIN
in a defclass
form. The has-email
class below will not create a table.
(defclass has-email ()
((email :col-type (:varchar 128)
:initarg :email
:accessor object-email))
(:metaclass mito:dao-table-mixin)
(:unique-keys email))
;=> #<MITO.DAO.MIXIN:DAO-TABLE-MIXIN COMMON-LISP-USER::HAS-EMAIL>
(mito:deftable user (has-email)
((name :col-type (:varchar 64))))
;=> #<MITO.DAO.TABLE:DAO-TABLE-CLASS COMMON-LISP-USER::USER>
(mito:table-definition 'user)
;=> (#<SXQL-STATEMENT: CREATE TABLE user (
; id BIGSERIAL NOT NULL PRIMARY KEY,
; name VARCHAR(64) NOT NULL,
; email VARCHAR(128) NOT NULL,
; created_at TIMESTAMP,
; updated_at TIMESTAMP,
; UNIQUE (email)
; )>)
See more examples of use in mito-auth.
Troubleshooting
āCannot CHANGE-CLASS objects into CLASS metaobjects.ā
If you get the following error message:
Cannot CHANGE-CLASS objects into CLASS metaobjects.
[Condition of type SB-PCL::METAOBJECT-INITIALIZATION-VIOLATION]
See also:
The Art of the Metaobject Protocol, CLASS [:initialization]
it is certainly because you first wrote a class definition and then added the Mito metaclass and tried to evaluate the class definition again.
If this happens, you must remove the class definition from the current package:
(setf (find-class 'foo) nil)
or, with the Slime inspector, click on the class and find the āremoveā button.
More info here.
Migrations
We can run database migrations manually, as shown below, or we can
automatically run migrations after a change to the model
definitions. To enable automatic migrations, set mito:*auto-migration-mode*
to t
.
The first step is to create the tables, if needed:
(ensure-table-exists 'user)
then alter the tables:
(mito:migrate-table 'user)
You can check the SQL generated code with migration-expressions
'class
. For example, we create the user
table:
(ensure-table-exists 'user)
;-> ;; CREATE TABLE IF NOT EXISTS "user" (
; "id" BIGSERIAL NOT NULL PRIMARY KEY,
; "name" VARCHAR(64) NOT NULL,
; "email" VARCHAR(128),
; "created_at" TIMESTAMP,
; "updated_at" TIMESTAMP
; ) () [0 rows] | MITO.DAO:ENSURE-TABLE-EXISTS
There are no changes from the previous user definition:
(mito:migration-expressions 'user)
;=> NIL
Now letās add a unique email
field:
(mito:deftable user ()
((name :col-type (:varchar 64))
(email :col-type (:varchar 128)))
(:unique-keys email))
The migration will run the following code:
(mito:migration-expressions 'user)
;=> (#<SXQL-STATEMENT: ALTER TABLE user ALTER COLUMN email TYPE character varying(128), ALTER COLUMN email SET NOT NULL>
; #<SXQL-STATEMENT: CREATE UNIQUE INDEX unique_user_email ON user (email)>)
so letās apply it:
(mito:migrate-table 'user)
;-> ;; ALTER TABLE "user" ALTER COLUMN "email" TYPE character varying(128), ALTER COLUMN "email" SET NOT NULL () [0 rows] | MITO.MIGRATION.TABLE:MIGRATE-TABLE
; ;; CREATE UNIQUE INDEX "unique_user_email" ON "user" ("email") () [0 rows] | MITO.MIGRATION.TABLE:MIGRATE-TABLE
;-> (#<SXQL-STATEMENT: ALTER TABLE user ALTER COLUMN email TYPE character varying(128), ALTER COLUMN email SET NOT NULL>
; #<SXQL-STATEMENT: CREATE UNIQUE INDEX unique_user_email ON user (email)>)
Queries
Creating objects
We can create user objects with the regular make-instance
:
(defvar me
(make-instance 'user :name "Eitaro Fukamachi" :email "e.arrows@gmail.com"))
;=> USER
To save it in DB, use insert-dao
:
(mito:insert-dao me)
;-> ;; INSERT INTO `user` (`name`, `email`, `created_at`, `updated_at`) VALUES (?, ?, ?, ?) ("Eitaro Fukamachi", "e.arrows@gmail.com", "2016-02-04T19:55:16.365543Z", "2016-02-04T19:55:16.365543Z") [0 rows] | MITO.DAO:INSERT-DAO
;=> #<USER {10053C4453}>
Do the two steps above at once:
(mito:create-dao 'user :name "Eitaro Fukamachi" :email "e.arrows@gmail.com")
You should not export the user
class and create objects outside of
its package (it is good practice anyway to keep all database-related
operations in say a models
package and file). You should instead use
a helper function:
(defun make-user (&key name)
(make-instance 'user :name name))
Updating fields
(setf (slot-value me 'name) "nitro_idiot")
;=> "nitro_idiot"
and save it:
(mito:save-dao me)
Deleting
(mito:delete-dao me)
;-> ;; DELETE FROM `user` WHERE (`id` = ?) (1) [0 rows] | MITO.DAO:DELETE-DAO
;; or:
(mito:delete-by-values 'user :id 1)
;-> ;; DELETE FROM `user` WHERE (`id` = ?) (1) [0 rows] | MITO.DAO:DELETE-DAO
Get the primary key value
(mito:object-id me)
;=> 1
Count
(mito:count-dao 'user)
;=> 1
Find one
(mito:find-dao 'user :id 1)
;-> ;; SELECT * FROM `user` WHERE (`id` = ?) LIMIT 1 (1) [1 row] | MITO.DB:RETRIEVE-BY-SQL
;=> #<USER {10077C6073}>
So hereās a possibility of generic helpers to find an object by a given key:
(defgeneric find-user (key-name key-value)
(:documentation "Retrieves an user from the data base by one of the unique
keys."))
(defmethod find-user ((key-name (eql :id)) (key-value integer))
(mito:find-dao 'user key-value))
(defmethod find-user ((key-name (eql :name)) (key-value string))
(first (mito:select-dao 'user
(sxql:where (:= :name key-value)))))
Find all
Use the macro select-dao
.
Get a list of all users:
(mito:select-dao 'user)
;(#<USER {10077C6073}>)
;#<SXQL-STATEMENT: SELECT * FROM user>
Find by relationship
As seen above:
(mito:find-dao 'tweet :user *user*)
Custom queries
It is with select-dao
that you can write more precise queries by
giving it SxQL statements.
Example:
(select-dao 'tweet
(where (:like :status "%Japan%")))
another:
(select (:id :name :sex)
(from (:as :person :p))
(where (:and (:>= :age 18)
(:< :age 65)))
(order-by (:desc :age)))
You can compose your queries with regular Lisp code:
(defun find-tweets (&key user)
(select-dao 'tweet
(when user
(where (:= :user user)))
(order-by :object-created)))
select-dao
is a macro that expands to the right thingĀ©.
use
SXQL, then write (sxql:where ā¦)
and (sxql:order-by ā¦)
.
You can compose your queries further with the backquote syntax.
Imagine you receive a query
string, maybe composed of
space-separated words, and you want to search for books that have
either one of these words in their title or in their authorās
name. Searching for ābob adventureā would return a book that has
āadventureā in its title and ābobā in its author name, or both in the
title.
For the example sake, an author is a string, not a link to another table:
(mito:deftable book ()
((title :col-type (:varchar 128))
(author :col-type (:varchar 128))
(ean :col-type (or (:varchar 128) :null))))
You want to add a clause that searches on both fields for each word.
(defun find-books (&key query (order :desc))
"Return a list of books.
If a query string is given, search on both the title
and the author fields."
(mito:select-dao 'book
(when (str:non-blank-string-p query)
(sxql:where
`(:and
,@(loop for word in (str:words query)
:collect `(:or (:like :title
,(str:concat "%" word "%"))
(:like :authors
,(str:concat "%" word "%")))))))
(sxql:order-by `(,order :created-at))))
By the way, we are still using a LIKE
statement, but with a non-small dataset youāll want to use your databaseās full text search engine.
Clauses
See the SxQL documentation.
Examples:
(select-dao 'foo
(where (:and (:> :age 20) (:<= :age 65))))
(order-by :age (:desc :id))
(group-by :sex)
(having (:>= (:sum :hoge) 88))
(limit 0 10)
and join
s, etc.
Operators
:not
:is-null, :not-null
:asc, :desc
:distinct
:=, :!=
:<, :>, :<= :>=
:a<, :a>
:as
:in, :not-in
:like
:and, :or
:+, :-, :* :/ :%
:raw
Triggers
Since insert-dao
, update-dao
and delete-dao
are defined as generic
functions, you can define :before
, :after
or :around
methods to those, like regular method combination.
(defmethod mito:insert-dao :before ((object user))
(format t "~&Adding ~S...~%" (user-name object)))
(mito:create-dao 'user :name "Eitaro Fukamachi" :email "e.arrows@gmail.com")
;-> Adding "Eitaro Fukamachi"...
; ;; INSERT INTO "user" ("name", "email", "created_at", "updated_at") VALUES (?, ?, ?, ?) ("Eitaro Fukamachi", "e.arrows@gmail.com", "2016-02-16 21:13:47", "2016-02-16 21:13:47") [0 rows] | MITO.DAO:INSERT-DAO
;=> #<USER {100835FB33}>
Inflation/Deflation
Inflation/Deflation is a function to convert values between Mito and RDBMS.
(mito:deftable user-report ()
((title :col-type (:varchar 100))
(body :col-type :text
:initform "")
(reported-at :col-type :timestamp
:initform (local-time:now)
:inflate #'local-time:universal-to-timestamp
:deflate #'local-time:timestamp-to-universal)))
Eager loading
One of the pains in the neck to use ORMs is the āN+1 queryā problem.
;; BAD EXAMPLE
(use-package '(:mito :sxql))
(defvar *tweets-contain-japan*
(select-dao 'tweet
(where (:like :status "%Japan%"))))
;; Getting names of tweeted users.
(mapcar (lambda (tweet)
(user-name (tweet-user tweet)))
*tweets-contain-japan*)
This example sends a query to retrieve a user like āSELECT * FROM user WHERE id = ?ā at each iteration.
To prevent this performance issue, add includes
to the above query
which only sends a single WHERE IN query instead of N queries:
;; GOOD EXAMPLE with eager loading
(use-package '(:mito :sxql))
(defvar *tweets-contain-japan*
(select-dao 'tweet
(includes 'user)
(where (:like :status "%Japan%"))))
;-> ;; SELECT * FROM `tweet` WHERE (`status` LIKE ?) ("%Japan%") [3 row] | MITO.DB:RETRIEVE-BY-SQL
;-> ;; SELECT * FROM `user` WHERE (`id` IN (?, ?, ?)) (1, 3, 12) [3 row] | MITO.DB:RETRIEVE-BY-SQL
;=> (#<TWEET {1003513EC3}> #<TWEET {1007BABEF3}> #<TWEET {1007BB9D63}>)
;; No additional SQLs will be executed.
(tweet-user (first *))
;=> #<USER {100361E813}>
Iteration with cursor (do-select)
do-select
is a macro to iterate over results from SELECT
one by
one.
On PostgreSQL, it uses CURSOR
, which can reduce memory usage since
it wonāt load all results in memory.
(do-select (dao (select-dao 'user (where (:< "2024-07-01" :created_at))))
;; Can be a more complex condition
(when (equal (user-name dao) "Eitaro")
(return dao)))
;; Same but without using CURSOR
(loop for dao in (select-dao 'user (where (:< "2024-07-01" :created_at)))
when (equal (user-name dao) "Eitaro")
do (return dao))
NOTE: do-select was added on August of 2024. It requires DBI v0.11.1 or above.
Schema versioning
$ ros install mito
$ mito
Usage: mito command [option...]
Commands:
generate-migrations
migrate
Options:
-t, --type DRIVER-TYPE DBI driver type (one of "mysql", "postgres" or "sqlite3")
-d, --database DATABASE-NAME Database name to use
-u, --username USERNAME Username for RDBMS
-p, --password PASSWORD Password for RDBMS
-s, --system SYSTEM ASDF system to load (several -s's allowed)
-D, --directory DIRECTORY Directory path to keep migration SQL files (default: "/Users/nitro_idiot/Programs/lib/mito/db/")
--dry-run List SQL expressions to migrate
Introspection
Mito provides some functions for introspection.
We can access the information of columns with the functions in
(mito.class.column:...)
:
table-column-[class, name, info, not-null-p,...]
primary-key-p
and likewise for tables with (mito.class.table:...)
.
Given we get a list of slots of our class:
(ql:quickload "closer-mop")
(closer-mop:class-direct-slots (find-class 'user))
;; (#<MITO.DAO.COLUMN:DAO-TABLE-COLUMN-CLASS NAME>
;; #<MITO.DAO.COLUMN:DAO-TABLE-COLUMN-CLASS EMAIL>)
(defparameter user-slots *)
We can answer the following questions:
What is the type of this column ?
(mito.class.column:table-column-type (first user-slots))
;; (:VARCHAR 64)
Is this column nullable ?
(mito.class.column:table-column-not-null-p
(first user-slots))
;; T
(mito.class.column:table-column-not-null-p
(second user-slots))
;; NIL
Testing
We donāt want to test DB operations against the production one. We need to create a temporary DB before each test.
The macro below creates a temporary DB with a random name, creates the tables, runs the code and connects back to the original DB connection.
(defpackage my-test.utils
(:use :cl)
(:import-from :my.models
:*db*
:*db-name*
:connect
:ensure-tables-exist
:migrate-all)
(:export :with-empty-db))
(in-package my-test.utils)
(defun random-string (length)
;; thanks 40ants/hacrm.
(let ((chars "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"))
(coerce (loop repeat length
collect (aref chars (random (length chars))))
'string)))
(defmacro with-empty-db (&body body)
"Run `body` with a new temporary DB."
`(let* ((*random-state* (make-random-state t))
(prefix (concatenate 'string
(random-string 8)
"/"))
;; Save our current DB connection.
(connection mito:*connection*))
(uiop:with-temporary-file (:pathname name :prefix prefix)
;; Bind our *db-name* to a new name, so as to create a new DB.
(let* ((*db-name* name))
;; Always re-connect to our real DB even in case of
;; error in body.
(unwind-protect
(progn
;; our functions to connect to the DB, create the tables
;; and run the migrations.
(connect)
(ensure-tables-exist)
(migrate-all)
,@body)
(setf mito:*connection* connection))))))
Use it like this:
(prove:subtest "Creation in a temporary DB."
(with-empty-db
(let ((user (make-user :name "Cookbook")))
(save-user user)
(prove:is (name user)
"Cookbook"
"Test username in a temp DB."))))
;; Creation in a temporary DB
;; CREATE TABLE "user" (
;; id BIGSERIAL NOT NULL PRIMARY KEY,
;; name VARCHAR(64) NOT NULL,
;; email VARCHAR(128) NOT NULL,
;; created_at TIMESTAMP,
;; updated_at TIMESTAMP,
;; UNIQUE (email)
;; ) () [0 rows] | MITO.DB:EXECUTE-SQL
;; ā Test username in a temp DB.
See also
- mito-attachment
- mito-auth
- can a role-based access right control library
- an advanced ādefmodelā macro.
Page source: databases.md