Skip to content

Commit

Permalink
Refactor to work with Duct main
Browse files Browse the repository at this point in the history
Refactor codebase to work with new 'Duct main' design. Update
dependecies and change migrator to work with a single edn file.
  • Loading branch information
weavejester committed Nov 8, 2024
1 parent 2ed75a7 commit bc7cd68
Show file tree
Hide file tree
Showing 8 changed files with 78 additions and 222 deletions.
17 changes: 8 additions & 9 deletions project.clj
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,12 @@
:url "https://github.com/duct-framework/migrator.ragtime"
:license {:name "Eclipse Public License"
:url "http://www.eclipse.org/legal/epl-v10.html"}
:dependencies [[org.clojure/clojure "1.10.1"]
[duct/core "0.8.0"]
[duct/logger "0.3.0"]
[integrant "0.8.0"]
[pandect "0.6.1"]
[ragtime "0.8.0"]]
:dependencies [[org.clojure/clojure "1.11.4"]
[org.duct-framework/logger "0.4.0"]
[dev.weavejester/ragtime "0.10.1"]
[integrant "0.13.1"]
[pandect "1.0.2"]]
:profiles
{:dev {:dependencies [[duct/database.sql "0.1.0"]
[org.clojure/java.jdbc "0.7.11"]
[org.xerial/sqlite-jdbc "3.30.1"]]}})
{:dev {:dependencies [[org.duct-framework/database.sql "0.3.0"]
[com.github.seancorfield/next.jdbc "1.3.955"]
[org.xerial/sqlite-jdbc "3.47.0.0"]]}})
60 changes: 16 additions & 44 deletions src/duct/migrator/ragtime.clj
Original file line number Diff line number Diff line change
@@ -1,45 +1,25 @@
(ns duct.migrator.ragtime
(:require [clojure.java.io :as io]
[duct.core :as duct]
[duct.logger :as logger]
(:require [duct.logger :as logger]
[integrant.core :as ig]
[pandect.algo.sha1 :refer [sha1]]
[ragtime.core :as ragtime]
[ragtime.jdbc :as jdbc]
[ragtime.sql :as sql]
[ragtime.next-jdbc :as jdbc]
[ragtime.reporter :as reporter]
[ragtime.strategy :as strategy]))

(defn logger-reporter [logger]
(fn [_ op id]
(case op
:up (logger/log logger :report ::applying id)
:down (logger/log logger :report ::rolling-back id))))
:up (logger/report logger ::applying {:id id})
:down (logger/report logger ::rolling-back {:id id}))))

(def strategies
{:apply-new strategy/apply-new
:raise-error strategy/raise-error
:rebase strategy/rebase
:ignore-future strategy/ignore-future})

(defprotocol StringSource
(get-string [source]))

(extend-protocol StringSource
String
(get-string [s] s)
java.net.URL
(get-string [s] (slurp s))
duct.core.resource.Resource
(get-string [s] (slurp s)))

(defn- singularize [coll]
(if (= (count coll) 1) (first coll) coll))

(defn- clean-key [base key]
(if (vector? key)
(singularize (remove #{base} key))
key))

(def ^:private colon (.getBytes ":" "US-ASCII"))
(def ^:private comma (.getBytes "," "US-ASCII"))
(def ^:private u= (.getBytes "u=" "US-ASCII"))
Expand All @@ -62,20 +42,24 @@
(defn- add-hash-to-id [migration]
(update migration :id str "#" (subs (hash-migration migration) 0 8)))

(defn- get-database [{{:keys [spec]} :database :as opts}]
(jdbc/sql-database spec (select-keys opts [:migrations-table])))
(defn- load-migrations [{:keys [migrations-file]}]
(->> (sql/load-migrations migrations-file)
(map (comp jdbc/sql-migration add-hash-to-id))))

(defn- get-database [{{:keys [datasource]} :database :as opts}]
(jdbc/sql-database datasource (select-keys opts [:migrations-table])))

(defn- get-strategy [{:keys [strategy] :or {strategy :raise-error}}]
(strategies strategy))

(defn- get-reporter [{:keys [logger]}]
(if logger (logger-reporter logger) reporter/print))

(defn- migrate [index {:keys [migrations] :as opts}]
(let [db (get-database opts)
strat (get-strategy opts)
rep (get-reporter opts)
migs (flatten migrations)]
(defn- migrate [index options]
(let [db (get-database options)
strat (get-strategy options)
rep (get-reporter options)
migs (load-migrations options)]
(ragtime/migrate-all db index migs {:reporter rep, :strategy strat})
(ragtime/into-index index migs)))

Expand All @@ -84,15 +68,3 @@

(defmethod ig/resume-key :duct.migrator/ragtime [_ options _ index]
(migrate index options))

(defmethod ig/init-key ::sql [key {:keys [up down] :as opts}]
(-> (jdbc/sql-migration {:id (:id opts (clean-key ::sql key))
:up (mapv get-string up)
:down (mapv get-string down)})
(add-hash-to-id)))

(defmethod ig/init-key ::resources [_ {:keys [path]}]
(->> (jdbc/load-resources path) (map add-hash-to-id)))

(defmethod ig/init-key ::directory [_ {:keys [path]}]
(->> (jdbc/load-directory path) (map add-hash-to-id)))
1 change: 0 additions & 1 deletion test/duct/migrator/example.txt

This file was deleted.

1 change: 0 additions & 1 deletion test/duct/migrator/migrations/01-create-foo.down.sql

This file was deleted.

1 change: 0 additions & 1 deletion test/duct/migrator/migrations/01-create-foo.up.sql

This file was deleted.

1 change: 0 additions & 1 deletion test/duct/migrator/migrations/02-create-bar.down.sql

This file was deleted.

1 change: 0 additions & 1 deletion test/duct/migrator/migrations/02-create-bar.up.sql

This file was deleted.

218 changes: 54 additions & 164 deletions test/duct/migrator/ragtime_test.clj
Original file line number Diff line number Diff line change
@@ -1,178 +1,68 @@
(ns duct.migrator.ragtime-test
(:require [clojure.java.jdbc :as jdbc]
[clojure.test :refer :all]
[duct.core :as duct]
[duct.database.sql :as sql]
(:require [clojure.test :refer [deftest is testing]]
[duct.logger :as logger]
[duct.migrator.ragtime :as ragtime]
[integrant.core :as ig]
[clojure.java.io :as io]))

(duct/load-hierarchy)
[next.jdbc :as jdbc]))

(defrecord TestLogger [logs]
logger/Logger
(-log [_ level ns-str file line id event data]
(-log [_ _level _ns-str _file _line _id event data]
(swap! logs conj [event data])))

(def logs
(atom []))

(def db-spec
{:connection (jdbc/get-connection {:connection-uri "jdbc:sqlite:"})})

(def config
{:duct.database/sql db-spec

:duct.migrator/ragtime
{:database (ig/ref :duct.database/sql)
:strategy :rebase
:logger (->TestLogger logs)
:migrations [(ig/ref ::create-foo)
(ig/ref ::create-bar)]}

[:duct.migrator.ragtime/sql ::create-foo]
{:up ["CREATE TABLE foo (id int);"]
:down ["DROP TABLE foo;"]}

[:duct.migrator.ragtime/sql ::create-bar]
{:up ["CREATE TABLE bar (id int);"]
:down ["DROP TABLE bar;"]}})

(def config-resources
{:duct.database/sql db-spec

:duct.migrator/ragtime
{:database (ig/ref :duct.database/sql)
:strategy :rebase
:logger (->TestLogger logs)
:migrations (ig/ref :duct.migrator.ragtime/resources)}

:duct.migrator.ragtime/resources
{:path "duct/migrator/migrations"}})

(def config-directory
{:duct.database/sql db-spec

:duct.migrator/ragtime
{:database (ig/ref :duct.database/sql)
:strategy :rebase
:logger (->TestLogger logs)
:migrations (ig/ref :duct.migrator.ragtime/directory)}

:duct.migrator.ragtime/directory
{:path "test/duct/migrator/migrations"}})

(def config-mixed
{:duct.database/sql db-spec
(defmethod ig/init-key ::logger [_ _]
(->TestLogger (atom [])))

(def base-config
{::logger {}
:duct.database/sql {}
:duct.migrator/ragtime
{:database (ig/ref :duct.database/sql)
:strategy :rebase
:logger (->TestLogger logs)
:migrations [(ig/ref :duct.migrator.ragtime/resources)
(ig/ref ::create-baz)]}

:duct.migrator.ragtime/resources
{:path "duct/migrator/migrations"}

[:duct.migrator.ragtime/sql ::create-baz]
{:up ["CREATE TABLE baz (id int);"]
:down ["DROP TABLE baz;"]}})

(defn- find-tables []
(jdbc/query db-spec ["SELECT name FROM sqlite_master WHERE type='table'"]))

(defn- drop-all-tables []
(doseq [t (find-tables)]
(jdbc/execute! db-spec [(str "DROP TABLE " (:name t))])))
{:logger (ig/ref ::logger)
:database (ig/ref :duct.database/sql)
:strategy :rebase
:migrations-file "test/duct/migrator/migrations1.edn"}})

(defn reset-test-state []
(reset! logs [])
(drop-all-tables))
(defn- find-tables [db]
(jdbc/execute! db ["SELECT name FROM sqlite_master WHERE type='table'"]))

(deftest migration-test
(testing "default migrations table name"
(reset-test-state)
(let [system (ig/init config)]
(is (= (find-tables)
[{:name "ragtime_migrations"} {:name "foo"} {:name "bar"}]))
(is (= @logs
[[::ragtime/applying ":duct.migrator.ragtime-test/create-foo#f1480e44"]
[::ragtime/applying ":duct.migrator.ragtime-test/create-bar#6d969ce8"]]))))

(testing "custom migrations table name"
(reset-test-state)
(let [migrations-table "custom_migrations"
system (-> config
(assoc-in [:duct.migrator/ragtime :migrations-table] migrations-table)
ig/init)]
(is (= (find-tables)
[{:name migrations-table} {:name "foo"} {:name "bar"}]))
(is (= @logs
[[::ragtime/applying ":duct.migrator.ragtime-test/create-foo#f1480e44"]
[::ragtime/applying ":duct.migrator.ragtime-test/create-bar#6d969ce8"]]))))

(testing "migrations from resource directory"
(reset-test-state)
(let [system (ig/init config-resources)]
(is (= (find-tables)
[{:name "ragtime_migrations"} {:name "foo"} {:name "bar"}]))
(is (= @logs
[[::ragtime/applying "01-create-foo#f1480e44"]
[::ragtime/applying "02-create-bar#6d969ce8"]]))))

(testing "migrations from filesystem directory"
(reset-test-state)
(let [system (ig/init config-directory)]
(is (= (find-tables)
[{:name "ragtime_migrations"} {:name "foo"} {:name "bar"}]))
(is (= @logs
[[::ragtime/applying "01-create-foo#f1480e44"]
[::ragtime/applying "02-create-bar#6d969ce8"]]))))

(testing "migrations from multiple sources"
(reset-test-state)
(let [system (ig/init config-mixed)]
(is (= (find-tables)
[{:name "ragtime_migrations"} {:name "foo"} {:name "bar"} {:name "baz"}]))
(is (= @logs
[[::ragtime/applying "01-create-foo#f1480e44"]
[::ragtime/applying "02-create-bar#6d969ce8"]
[::ragtime/applying ":duct.migrator.ragtime-test/create-baz#01f7277d"]])))))

(deftest remove-and-resume-test
(reset-test-state)
(let [system (ig/init config)
config' (update-in config [:duct.migrator/ragtime :migrations] pop)
system' (ig/resume config' system)]
(is (= (find-tables)
[{:name "ragtime_migrations"}
{:name "foo"}]))
(is (= @logs
[[::ragtime/applying ":duct.migrator.ragtime-test/create-foo#f1480e44"]
[::ragtime/applying ":duct.migrator.ragtime-test/create-bar#6d969ce8"]
[::ragtime/rolling-back ":duct.migrator.ragtime-test/create-bar#6d969ce8"]]))))

(deftest change-and-resume-test
(reset-test-state)
(let [system (ig/init config)
config' (assoc config
[:duct.migrator.ragtime/sql ::create-bar]
{:up ["CREATE TABLE barbaz (id int)"]
:down ["DROP TABLE barbaz"]})
system' (ig/resume config' system)]
(is (= (find-tables)
[{:name "ragtime_migrations"}
{:name "foo"}
{:name "barbaz"}]))
(is (= @logs
[[::ragtime/applying ":duct.migrator.ragtime-test/create-foo#f1480e44"]
[::ragtime/applying ":duct.migrator.ragtime-test/create-bar#6d969ce8"]
[::ragtime/rolling-back ":duct.migrator.ragtime-test/create-bar#6d969ce8"]
[::ragtime/applying ":duct.migrator.ragtime-test/create-bar#66068fd2"]]))))
(let [tempfile (java.io.File/createTempFile "duct" "db")
jdbc-url (str "jdbc:sqlite:" tempfile)
config (-> base-config
(doto (ig/load-namespaces))
(assoc-in [:duct.database/sql :jdbcUrl] jdbc-url))
system (atom (ig/init config))]

(testing "initial migrations"
(let [logs (-> @system ::logger :logs)
db (-> @system :duct.database/sql :datasource)]
(is (= ["ragtime_migrations" "foo" "bar"]
(map :sqlite_master/name (find-tables db))))
(is (= [[:duct.migrator.ragtime/applying {:id "create-table-foo#52bfa531"}]
[:duct.migrator.ragtime/applying {:id "create-table-bar#3e718b28"}]]
@logs))))

(testing "change migration"
(let [config (assoc-in config [:duct.migrator/ragtime :migrations-file]
"test/duct/migrator/migrations2.edn")
system (swap! system (fn [sys] (ig/suspend! sys) (ig/resume config sys)))
logs (-> system ::logger :logs)
db (-> system :duct.database/sql :datasource)]
(is (= ["ragtime_migrations" "foo" "baz"]
(map :sqlite_master/name (find-tables db))))
(is (= [[:duct.migrator.ragtime/rolling-back {:id "create-table-bar#3e718b28"}]
[:duct.migrator.ragtime/applying {:id "create-table-baz#055a605e"}]]
@logs))))

(testing "remove migration"
(let [config (assoc-in config [:duct.migrator/ragtime :migrations-file]
"test/duct/migrator/migrations3.edn")
system (swap! system (fn [sys] (ig/suspend! sys) (ig/resume config sys)))
logs (-> system ::logger :logs)
db (-> system :duct.database/sql :datasource)]
(is (= ["ragtime_migrations" "foo"]
(map :sqlite_master/name (find-tables db))))
(is (= [[:duct.migrator.ragtime/rolling-back {:id "create-table-baz#055a605e"}]]
@logs))))

(.delete tempfile)))

(deftest string-source-test
(is (= "foo\n" (ragtime/get-string "foo\n")))
(is (= "foo\n" (ragtime/get-string (io/resource "duct/migrator/example.txt"))))
(is (= "foo\n" (ragtime/get-string (duct/resource "duct/migrator/example.txt")))))

0 comments on commit bc7cd68

Please sign in to comment.