feat(web/panettone): Add a system for database migrations
Add a system for writing, running, and tracking database migrations (changes to the database schema) over time, inspired by but significantly simpler than postmodern-passenger-pigeon. Migrations can be generated by running (PANETTONE.MODEL:GENERATE-MIGRATION "name"), and are numerically ordered lisp files that define (at least) a function called UP, which runs the migration. The migrations that have been run against the database are tracked in the `migrations` table, and when the `(PANETTONE.MODEL:MIGRATE)` function is called (as it is on startup), all migrations that have not yet been run are run within a transaction. This includes one migration `1-init-schema.lisp`, which migrates the database (idempotently) to the current state of the schema. Change-Id: Id243a47763abea649784b12f25a6d05c2267381c Reviewed-on: https://cl.tvl.fyi/c/depot/+/11253 Tested-by: BuildkiteCI Reviewed-by: sterni <sternenseemann@systemli.org>
This commit is contained in:
parent
d5f57ac6e6
commit
a1a29f7c0b
5 changed files with 228 additions and 37 deletions
|
@ -1,4 +1,4 @@
|
||||||
{ depot, ... }:
|
{ depot, pkgs, ... }:
|
||||||
|
|
||||||
depot.nix.buildLisp.program {
|
depot.nix.buildLisp.program {
|
||||||
name = "panettone";
|
name = "panettone";
|
||||||
|
@ -9,6 +9,7 @@ depot.nix.buildLisp.program {
|
||||||
cl-ppcre
|
cl-ppcre
|
||||||
cl-smtp
|
cl-smtp
|
||||||
cl-who
|
cl-who
|
||||||
|
str
|
||||||
defclass-std
|
defclass-std
|
||||||
drakma
|
drakma
|
||||||
easy-routes
|
easy-routes
|
||||||
|
@ -23,6 +24,14 @@ depot.nix.buildLisp.program {
|
||||||
srcs = [
|
srcs = [
|
||||||
./panettone.asd
|
./panettone.asd
|
||||||
./src/packages.lisp
|
./src/packages.lisp
|
||||||
|
(pkgs.writeText "build.lisp" ''
|
||||||
|
(defpackage build
|
||||||
|
(:use :cl :alexandria)
|
||||||
|
(:export :*migrations-dir*))
|
||||||
|
(in-package :build)
|
||||||
|
(declaim (optimize (safety 3)))
|
||||||
|
(defvar *migrations-dir* "${./src/migrations}")
|
||||||
|
'')
|
||||||
./src/util.lisp
|
./src/util.lisp
|
||||||
./src/css.lisp
|
./src/css.lisp
|
||||||
./src/email.lisp
|
./src/email.lisp
|
||||||
|
|
23
web/panettone/src/migrations/1-init-schema.lisp
Normal file
23
web/panettone/src/migrations/1-init-schema.lisp
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
"Initialize the database schema from before migrations were added"
|
||||||
|
|
||||||
|
(defun ddl/create-issue-status ()
|
||||||
|
"Issue DDL to create the `issue-status' type, if it doesn't exist"
|
||||||
|
(unless (query (:select (:exists (:select 1
|
||||||
|
:from 'pg_type
|
||||||
|
:where (:= 'typname "issue_status"))))
|
||||||
|
:single)
|
||||||
|
(query (sql-compile
|
||||||
|
`(:create-enum issue-status ,panettone.model:+issue-statuses+)))))
|
||||||
|
|
||||||
|
(defun ddl/create-tables ()
|
||||||
|
"Issue DDL to create all tables, if they don't already exist."
|
||||||
|
(dolist (table '(panettone.model:issue
|
||||||
|
panettone.model:issue-comment
|
||||||
|
panettone.model:issue-event
|
||||||
|
panettone.model:user-settings))
|
||||||
|
(unless (table-exists-p (dao-table-name table))
|
||||||
|
(create-table table))))
|
||||||
|
|
||||||
|
(defun up ()
|
||||||
|
(ddl/create-issue-status)
|
||||||
|
(ddl/create-tables))
|
|
@ -20,6 +20,19 @@ initialised at launch time.")
|
||||||
"Initialises the connection spec used for all Postgres connections."
|
"Initialises the connection spec used for all Postgres connections."
|
||||||
(setq *pg-spec* (make-pg-spec)))
|
(setq *pg-spec* (make-pg-spec)))
|
||||||
|
|
||||||
|
(defun connect-to-db ()
|
||||||
|
"Connect using *PG-SPEC* at the top-level, for use during development"
|
||||||
|
(apply #'connect-toplevel
|
||||||
|
(loop for v in *pg-spec*
|
||||||
|
until (eq v :pooled-p)
|
||||||
|
collect v)))
|
||||||
|
|
||||||
|
(defun pg-spec->url (&optional (spec *pg-spec*))
|
||||||
|
(destructuring-bind (db user password host &key port &allow-other-keys) spec
|
||||||
|
(format nil
|
||||||
|
"postgres://~A:~A@~A:~A/~A"
|
||||||
|
user password host port db)))
|
||||||
|
|
||||||
;;;
|
;;;
|
||||||
;;; Schema
|
;;; Schema
|
||||||
;;;
|
;;;
|
||||||
|
@ -77,15 +90,6 @@ initialised at launch time.")
|
||||||
"Type specifier for the status of an `issue'"
|
"Type specifier for the status of an `issue'"
|
||||||
(cons 'member +issue-statuses+))
|
(cons 'member +issue-statuses+))
|
||||||
|
|
||||||
(defun ddl/create-issue-status ()
|
|
||||||
"Issue DDL to create the `issue-status' type, if it doesn't exist"
|
|
||||||
(unless (query (:select (:exists (:select 1
|
|
||||||
:from 'pg_type
|
|
||||||
:where (:= 'typname "issue_status"))))
|
|
||||||
:single)
|
|
||||||
(query (sql-compile
|
|
||||||
`(:create-enum issue-status ,+issue-statuses+)))))
|
|
||||||
|
|
||||||
(defclass has-created-at ()
|
(defclass has-created-at ()
|
||||||
((created-at :col-type timestamp
|
((created-at :col-type timestamp
|
||||||
:col-default (local-time:now)
|
:col-default (local-time:now)
|
||||||
|
@ -192,23 +196,168 @@ its new value will be formatted using ~A into NEW-VALUE"))
|
||||||
(!dao-def)
|
(!dao-def)
|
||||||
(!foreign 'issues 'issue-id 'id :on-delete :cascade :on-update :cascade))
|
(!foreign 'issues 'issue-id 'id :on-delete :cascade :on-update :cascade))
|
||||||
|
|
||||||
(define-constant +all-tables+
|
(defclass migration ()
|
||||||
'(issue
|
((version
|
||||||
issue-comment
|
:col-type bigint
|
||||||
issue-event
|
:primary-key t
|
||||||
user-settings)
|
:initarg :version
|
||||||
:test #'equal)
|
:accessor version)
|
||||||
|
(name :col-type string :initarg :name :accessor name)
|
||||||
|
(docstring :col-type string :initarg :docstring :accessor docstring)
|
||||||
|
(path :col-type string
|
||||||
|
:type pathname
|
||||||
|
:initarg :path
|
||||||
|
:accessor path
|
||||||
|
:col-export namestring
|
||||||
|
:col-import parse-namestring)
|
||||||
|
(package :type keyword :initarg :package :accessor migration-package))
|
||||||
|
(:metaclass dao-class)
|
||||||
|
(:keys version)
|
||||||
|
(:table-name migrations)
|
||||||
|
(:documentation "Migration scripts that have been run on the database"))
|
||||||
|
(deftable migration (!dao-def))
|
||||||
|
|
||||||
(defun ddl/create-tables ()
|
;;;
|
||||||
"Issue DDL to create all tables, if they don't already exist."
|
;;; Migrations
|
||||||
(dolist (table +all-tables+)
|
;;;
|
||||||
(unless (table-exists-p (dao-table-name table))
|
|
||||||
(create-table table))))
|
|
||||||
|
|
||||||
(defun ddl/init ()
|
(defun ensure-migrations-table ()
|
||||||
"Idempotently initialize the full database schema for Panettone"
|
"Ensure the migrations table exists"
|
||||||
(ddl/create-issue-status)
|
(unless (table-exists-p (dao-table-name 'migration))
|
||||||
(ddl/create-tables))
|
(create-table 'migration)))
|
||||||
|
|
||||||
|
(defvar *migrations-dir*
|
||||||
|
;; Let the nix build override the migrations dir for us
|
||||||
|
(or (when-let ((package (find-package :build)))
|
||||||
|
(let ((sym (find-symbol "*MIGRATIONS-DIR*" package)))
|
||||||
|
(when (boundp sym)
|
||||||
|
(symbol-value sym))))
|
||||||
|
"migrations/")
|
||||||
|
"The directory where migrations are stored")
|
||||||
|
|
||||||
|
(defun load-migration-docstring (migration-path)
|
||||||
|
"If the first form in the file pointed to by `migration-pathname` is
|
||||||
|
a string, return it, otherwise return NIL."
|
||||||
|
|
||||||
|
(handler-case
|
||||||
|
(with-open-file (s migration-path)
|
||||||
|
(when-let ((form (read s)))
|
||||||
|
(when (stringp form) form)))
|
||||||
|
(t () nil)))
|
||||||
|
|
||||||
|
(defun load-migration (path)
|
||||||
|
(let* ((parts (str:split #\- (pathname-name path) :limit 2))
|
||||||
|
(version (parse-integer (car parts)))
|
||||||
|
(name (cadr parts))
|
||||||
|
(docstring (load-migration-docstring path))
|
||||||
|
(package (intern (format nil "MIGRATION-~A" version)
|
||||||
|
:keyword))
|
||||||
|
(migration (make-instance 'migration
|
||||||
|
:version version
|
||||||
|
:name name
|
||||||
|
:docstring docstring
|
||||||
|
:path path
|
||||||
|
:package package)))
|
||||||
|
(uiop/package:ensure-package package
|
||||||
|
:use '(#:common-lisp
|
||||||
|
#:postmodern
|
||||||
|
#:panettone.model))
|
||||||
|
(let ((*package* (find-package package)))
|
||||||
|
(load path))
|
||||||
|
|
||||||
|
migration))
|
||||||
|
|
||||||
|
(defun run-migration (migration)
|
||||||
|
(declare (type migration migration))
|
||||||
|
(with-transaction ()
|
||||||
|
(format t "Running migration ~A (version ~A)"
|
||||||
|
(name migration)
|
||||||
|
(version migration))
|
||||||
|
(query
|
||||||
|
(sql-compile
|
||||||
|
`(:delete-from migrations
|
||||||
|
:where (= version ,(version migration)))))
|
||||||
|
(uiop:symbol-call (migration-package migration) :up)
|
||||||
|
(insert-dao migration)))
|
||||||
|
|
||||||
|
(defun list-migration-files ()
|
||||||
|
(let ((dir (if (char-equal (uiop:last-char *migrations-dir*) #\/)
|
||||||
|
*migrations-dir*
|
||||||
|
(concatenate 'string *migrations-dir* "/"))))
|
||||||
|
(remove-if-not
|
||||||
|
(lambda (pn) (string= "lisp" (pathname-type pn)))
|
||||||
|
(uiop:directory-files dir))))
|
||||||
|
|
||||||
|
(defun load-migrations ()
|
||||||
|
(mapcar #'load-migration (list-migration-files)))
|
||||||
|
|
||||||
|
(defun generate-migration (name &key documentation)
|
||||||
|
"Generate a new database migration with the given NAME, optionally
|
||||||
|
prepopulated with the given DOCUMENTATION.
|
||||||
|
|
||||||
|
Returns the file that the migration is located at, as a `pathname'. Write Lisp
|
||||||
|
code in this migration file to define a function called `up', which will be run
|
||||||
|
in the context of a database transaction and should perform the migration."
|
||||||
|
(let* ((version (get-universal-time))
|
||||||
|
(filename (format nil "~A-~A.lisp"
|
||||||
|
version
|
||||||
|
name))
|
||||||
|
(pathname
|
||||||
|
(merge-pathnames filename *migrations-dir*)))
|
||||||
|
(with-open-file (stream pathname
|
||||||
|
:direction :output
|
||||||
|
:if-does-not-exist :create)
|
||||||
|
(when documentation
|
||||||
|
(format stream "~S~%~%" documentation))
|
||||||
|
|
||||||
|
(format stream "(defun up ()~%)"))
|
||||||
|
pathname))
|
||||||
|
|
||||||
|
(defun migrations-already-run ()
|
||||||
|
"Query the database for a list of migrations that have already been run"
|
||||||
|
(query-dao 'migration (sql-compile '(:select * :from migrations))))
|
||||||
|
|
||||||
|
(define-condition migration-name-mismatch ()
|
||||||
|
((version :type integer :initarg :version)
|
||||||
|
(name-in-database :type string :initarg :name-in-database)
|
||||||
|
(name-in-code :type string :initarg :name-in-code))
|
||||||
|
(:report
|
||||||
|
(lambda (cond stream)
|
||||||
|
(format stream "Migration mismatch: Migration version ~A has name ~S in the database, but we have name ~S"
|
||||||
|
(slot-value cond 'version)
|
||||||
|
(slot-value cond 'name-in-database)
|
||||||
|
(slot-value cond 'name-in-code)))))
|
||||||
|
|
||||||
|
(defun migrate ()
|
||||||
|
"Migrate the database, running all migrations that have not yet been run"
|
||||||
|
(ensure-migrations-table)
|
||||||
|
(let* ((all-migrations (load-migrations))
|
||||||
|
(already-run (migrations-already-run))
|
||||||
|
(num-migrations-run 0))
|
||||||
|
(iter (for migration in all-migrations)
|
||||||
|
(if-let ((existing (find-if (lambda (existing)
|
||||||
|
(= (version existing)
|
||||||
|
(version migration)))
|
||||||
|
already-run)))
|
||||||
|
(progn
|
||||||
|
(unless (string= (name migration)
|
||||||
|
(name existing))
|
||||||
|
(restart-case
|
||||||
|
(error 'migration-name-mismatch
|
||||||
|
:version (version existing)
|
||||||
|
:name-in-database (name existing)
|
||||||
|
:name-in-code (name migration))
|
||||||
|
(skip ()
|
||||||
|
:report "Skip this migration"
|
||||||
|
(next-iteration))
|
||||||
|
(run-and-overwrite ()
|
||||||
|
:report "Run this migration anyway, overwriting the previous migration"
|
||||||
|
(run-migration migration))))
|
||||||
|
(next-iteration))
|
||||||
|
;; otherwise, run the migration
|
||||||
|
(run-migration migration))
|
||||||
|
(incf num-migrations-run))
|
||||||
|
(format nil "Ran ~A migration~:P" num-migrations-run)))
|
||||||
|
|
||||||
;;;
|
;;;
|
||||||
;;; Querying
|
;;; Querying
|
||||||
|
@ -253,11 +402,11 @@ type `ISSUE-NOT-FOUND'."
|
||||||
(select (if (find :num-comments with)
|
(select (if (find :num-comments with)
|
||||||
`(:select issues.* (:as (:count issue-comments.id)
|
`(:select issues.* (:as (:count issue-comments.id)
|
||||||
num-comments)
|
num-comments)
|
||||||
:from issues
|
:from issues
|
||||||
:left-join issue-comments
|
:left-join issue-comments
|
||||||
:on (:= issues.id issue-comments.issue-id)
|
:on (:= issues.id issue-comments.issue-id)
|
||||||
,@condition
|
,@condition
|
||||||
:group-by issues.id)
|
:group-by issues.id)
|
||||||
`(:select * :from issues ,@condition)))
|
`(:select * :from issues ,@condition)))
|
||||||
(query (sql-compile
|
(query (sql-compile
|
||||||
`(:order-by ,select (:desc id)))))
|
`(:order-by ,select (:desc id)))))
|
||||||
|
@ -409,12 +558,22 @@ explicitly subscribing to / unsubscribing from individual issues."
|
||||||
|
|
||||||
|
|
||||||
(comment
|
(comment
|
||||||
(ddl/init)
|
|
||||||
(make-instance 'issue :subject "test")
|
(make-instance 'issue :subject "test")
|
||||||
(create-issue :subject "test"
|
|
||||||
:author-dn "cn=grfn,ou=users,dc=tvl,dc=fyi")
|
(with-connection *pg-spec*
|
||||||
|
(create-issue :subject "test"
|
||||||
|
:author-dn "cn=aspen,ou=users,dc=tvl,dc=fyi"))
|
||||||
|
|
||||||
(issue-commenter-dns 1)
|
(issue-commenter-dns 1)
|
||||||
(issue-subscribers 1)
|
(issue-subscribers 1)
|
||||||
|
|
||||||
|
;; Creating new migrations
|
||||||
|
(setq *migrations-dir* (merge-pathnames "migrations/"))
|
||||||
|
(generate-migration "init-schema"
|
||||||
|
:documentation "Initialize the database schema")
|
||||||
|
|
||||||
|
;; Running migrations
|
||||||
|
(with-connection *pg-spec*
|
||||||
|
(migrate))
|
||||||
)
|
)
|
||||||
|
|
|
@ -36,16 +36,16 @@
|
||||||
(:import-from :alexandria :if-let :when-let :define-constant)
|
(:import-from :alexandria :if-let :when-let :define-constant)
|
||||||
(:export
|
(:export
|
||||||
:prepare-db-connections
|
:prepare-db-connections
|
||||||
:ddl/init
|
:migrate
|
||||||
:*pg-spec*
|
:*pg-spec*
|
||||||
|
|
||||||
:user-settings
|
:user-settings
|
||||||
:user-dn :enable-email-notifications-p :settings-for-user
|
:user-dn :enable-email-notifications-p :settings-for-user
|
||||||
:update-user-settings :enable-email-notifications
|
:update-user-settings :enable-email-notifications
|
||||||
|
|
||||||
:issue :issue-comment :issue-event
|
:issue :issue-comment :issue-event :migration
|
||||||
:id :subject :body :author-dn :issue-id :status :created-at :acting-user-dn
|
:id :subject :body :author-dn :issue-id :status :created-at :acting-user-dn
|
||||||
:field :previous-value :new-value
|
:field :previous-value :new-value :+issue-statuses+
|
||||||
|
|
||||||
:get-issue :issue-exists-p :list-issues :create-issue :set-issue-status
|
:get-issue :issue-exists-p :list-issues :create-issue :set-issue-status
|
||||||
:update-issue :delete-issue :issue-not-found :not-found-id
|
:update-issue :delete-issue :issue-not-found :not-found-id
|
||||||
|
|
|
@ -606,7 +606,7 @@ given subject an body (in a thread, to avoid blocking)"
|
||||||
(defun migrate-db ()
|
(defun migrate-db ()
|
||||||
"Migrate the database to the latest version of the schema"
|
"Migrate the database to the latest version of the schema"
|
||||||
(pomo:with-connection *pg-spec*
|
(pomo:with-connection *pg-spec*
|
||||||
(model:ddl/init)))
|
(model:migrate)))
|
||||||
|
|
||||||
(defun start-panettone (&key port session-secret)
|
(defun start-panettone (&key port session-secret)
|
||||||
(authn:initialise-oauth2)
|
(authn:initialise-oauth2)
|
||||||
|
|
Loading…
Reference in a new issue