From 4f5167931da9a79d4805f1b8e648162980eff708 Mon Sep 17 00:00:00 2001 From: Cam Saul <1455846+camsaul@users.noreply.github.com> Date: Thu, 2 Feb 2023 19:33:34 -0800 Subject: [PATCH] `define-` helper macros should generate unique vars with named based on model (#109) * `define-` helper macros should generate unique vars with named based on model * Sort namespaces * Revert MariaDB driver bump * Switch protocol back to mysql:// * Sort namespaces --- deps.edn | 6 ++++- src/toucan2/jdbc/read.clj | 2 +- src/toucan2/query_execution_backend/jdbc.clj | 5 ++-- .../jdbc/mysql_mariadb.clj | 16 ++++++------ src/toucan2/tools/before_insert.clj | 6 ++--- src/toucan2/tools/before_update.clj | 6 ++--- src/toucan2/tools/default_fields.clj | 6 ++--- src/toucan2/tools/transformed.clj | 6 ++--- test/toucan2/test.clj | 2 +- test/toucan2/tools/after_insert_test.clj | 25 +++++++++++++++++++ test/toucan2/tools/after_select_test.clj | 24 ++++++++++++++++++ test/toucan2/tools/after_update_test.clj | 25 +++++++++++++++++++ test/toucan2/tools/before_delete_test.clj | 25 +++++++++++++++++++ test/toucan2/tools/before_insert_test.clj | 24 ++++++++++++++++++ test/toucan2/tools/before_select_test.clj | 24 ++++++++++++++++++ test/toucan2/tools/before_update_test.clj | 23 +++++++++++++++++ test/toucan2/tools/default_fields_test.clj | 23 +++++++++++++++++ test/toucan2/tools/transformed_test.clj | 22 ++++++++++++++++ toucan1/test/toucan/db_test.clj | 7 +++--- 19 files changed, 247 insertions(+), 30 deletions(-) diff --git a/deps.edn b/deps.edn index a4c7f72..820d436 100644 --- a/deps.edn +++ b/deps.edn @@ -23,7 +23,11 @@ org.clojure/java.classpath {:mvn/version "1.0.0"} org.clojure/math.combinatorics {:mvn/version "0.1.6"} org.clojure/tools.namespace {:mvn/version "1.3.0"} - org.mariadb.jdbc/mariadb-java-client {:mvn/version "3.1.2"} + ;; Don't upgrade to 3.x yet. It only returns the first generated key when inserting multiple rows. So keep using v2 + ;; until we decide how to work around it or they fix it. See + ;; + ;; https://mariadb.com/kb/en/about-mariadb-connector-j/#generated-keys-limitation + org.mariadb.jdbc/mariadb-java-client {:mvn/version "2.7.8"} org.postgresql/postgresql {:mvn/version "42.5.2"} pjstadig/humane-test-output {:mvn/version "0.11.0"}} diff --git a/src/toucan2/jdbc/read.clj b/src/toucan2/jdbc/read.clj index ff13ad2..f453664 100644 --- a/src/toucan2/jdbc/read.clj +++ b/src/toucan2/jdbc/read.clj @@ -46,7 +46,7 @@ (log/debugf :results "Column %s %s is of JDBC type %s, native type %s" i - (str (.getTableName rsmeta i) \. (.getColumnLabel rsmeta i)) + (str (some->> (.getTableName rsmeta i) not-empty (str \.)) (.getColumnLabel rsmeta i)) (symbol "java.sql.Types" (type-name col-type)) (.getColumnTypeName rsmeta i)) [(protocols/dispatch-value conn) (protocols/dispatch-value model) col-type]))) diff --git a/src/toucan2/query_execution_backend/jdbc.clj b/src/toucan2/query_execution_backend/jdbc.clj index 6f1b7c7..8a71f7b 100644 --- a/src/toucan2/query_execution_backend/jdbc.clj +++ b/src/toucan2/query_execution_backend/jdbc.clj @@ -79,6 +79,7 @@ (when (class-exists? "org.postgresql.jdbc.PgConnection") (require 'toucan2.query-execution-backend.jdbc.postgres)) -(when (or (class-exists? "com.mysql.cj.MysqlConnection") - (class-exists? "org.mariadb.jdbc.MariaDbConnection")) +(when (some class-exists? ["org.mariadb.jdbc.Connection" + "org.mariadb.jdbc.MariaDbConnection" + "com.mysql.cj.MysqlConnection"]) (require 'toucan2.query-execution-backend.jdbc.mysql-mariadb)) diff --git a/src/toucan2/query_execution_backend/jdbc/mysql_mariadb.clj b/src/toucan2/query_execution_backend/jdbc/mysql_mariadb.clj index 175c90f..8d0c57d 100644 --- a/src/toucan2/query_execution_backend/jdbc/mysql_mariadb.clj +++ b/src/toucan2/query_execution_backend/jdbc/mysql_mariadb.clj @@ -16,15 +16,13 @@ ;;; TODO -- need the MySQL class here too. -(when-let [mariadb-connection-class (try - (Class/forName "org.mariadb.jdbc.MariaDbConnection") - (catch Throwable _))] - (derive mariadb-connection-class ::connection)) - -(when-let [mysql-connection-class (try - (Class/forName "com.mysql.cj.MysqlConnection") - (catch Throwable _))] - (derive mysql-connection-class ::connection)) +(doseq [^String connection-class-name ["org.mariadb.jdbc.Connection" + "org.mariadb.jdbc.MariaDbConnection" + "com.mysql.cj.MysqlConnection"]] + (when-let [connection-class (try + (Class/forName connection-class-name) + (catch Throwable _))] + (derive connection-class ::connection))) (m/defmethod jdbc.read/read-column-thunk [#_conn ::connection #_model :default diff --git a/src/toucan2/tools/before_insert.clj b/src/toucan2/tools/before_insert.clj index 25d54e6..8f4dd9a 100644 --- a/src/toucan2/tools/before_insert.clj +++ b/src/toucan2/tools/before_insert.clj @@ -57,9 +57,9 @@ (defmacro define-before-insert {:style/indent :defn} [model [instance-binding] & body] - `(let [model# ~model] - (u/maybe-derive model# ::before-insert) - (m/defmethod before-insert model# + `(do + (u/maybe-derive ~model ::before-insert) + (m/defmethod before-insert ~model [~'&model ~instance-binding] (cond->> (do ~@body) ~'next-method (~'next-method ~'&model))))) diff --git a/src/toucan2/tools/before_update.clj b/src/toucan2/tools/before_update.clj index 2b73311..c73da33 100644 --- a/src/toucan2/tools/before_update.clj +++ b/src/toucan2/tools/before_update.clj @@ -120,9 +120,9 @@ new-args-maps))))) (defmacro define-before-update [model [instance-binding] & body] - `(let [model# ~model] - (u/maybe-derive model# ::before-update) - (m/defmethod before-update model# + `(do + (u/maybe-derive ~model ::before-update) + (m/defmethod before-update ~model [~'&model ~instance-binding] (cond->> (do ~@body) ~'next-method diff --git a/src/toucan2/tools/default_fields.clj b/src/toucan2/tools/default_fields.clj index a0ddad7..e389728 100644 --- a/src/toucan2/tools/default_fields.clj +++ b/src/toucan2/tools/default_fields.clj @@ -113,9 +113,9 @@ [:toucan.result-type/instances :toucan2.tools.after/model]) (defmacro define-default-fields {:style/indent :defn} [model & body] - `(let [model# ~model] - (u/maybe-derive model# ::default-fields) - (m/defmethod default-fields model# [~'&model] ~@body))) + `(do + (u/maybe-derive ~model ::default-fields) + (m/defmethod default-fields ~model [~'&model] ~@body))) (s/fdef define-default-fields :args (s/cat :model some? diff --git a/src/toucan2/tools/transformed.clj b/src/toucan2/tools/transformed.clj index 1a675fe..6caf840 100644 --- a/src/toucan2/tools/transformed.clj +++ b/src/toucan2/tools/transformed.clj @@ -414,9 +414,9 @@ ```" {:style/indent 1} [model column->direction->fn] - `(let [model# ~model] - (u/maybe-derive model# ::transformed.model) - (m/defmethod transforms model# + `(do + (u/maybe-derive ~model ::transformed.model) + (m/defmethod transforms ~model [~'model] ~column->direction->fn))) diff --git a/test/toucan2/test.clj b/test/toucan2/test.clj index 2d907cd..b93d089 100644 --- a/test/toucan2/test.clj +++ b/test/toucan2/test.clj @@ -184,7 +184,7 @@ (defmethod default-test-db-url :mariadb [_db-type] - "jdbc:mariadb://localhost:3306/metabase_test?user=root") + "jdbc:mysql://localhost:3306/metabase_test?user=root") (defmethod default-test-db-url :h2 [_db-type] diff --git a/test/toucan2/tools/after_insert_test.clj b/test/toucan2/tools/after_insert_test.clj index 49f77db..48d7d88 100644 --- a/test/toucan2/tools/after_insert_test.clj +++ b/test/toucan2/tools/after_insert_test.clj @@ -1,6 +1,8 @@ (ns toucan2.tools.after-insert-test (:require + [clojure.string :as str] [clojure.test :refer :all] + [clojure.walk :as walk] [toucan2.insert :as insert] [toucan2.instance :as instance] [toucan2.protocols :as protocols] @@ -253,3 +255,26 @@ :created-at (LocalDateTime/parse "2017-01-01T00:00") :updated-at (LocalDateTime/parse "2017-01-01T00:00")}] @*venues-awaiting-moderation*))))))))) + +(deftest ^:parallel macroexpansion-test + (testing "define-after-insert should define vars with different names based on the model." + (letfn [(generated-name* [form] + (cond + (sequential? form) + (some generated-name* form) + + (and (symbol? form) + (str/starts-with? (name form) "each-row-fn-primary-method")) + form)) + (generated-name [form] + (let [expanded (walk/macroexpand-all form)] + (or (generated-name* expanded) + ['no-match expanded])))] + (is (= 'each-row-fn-primary-method-toucan-query-type-insert-*-model-1 + (generated-name `(after-insert/define-after-insert :model-1 + [~'venue] + ~'venue)))) + (is (= 'each-row-fn-primary-method-toucan-query-type-insert-*-model-2 + (generated-name `(after-insert/define-after-insert :model-2 + [~'venue] + ~'venue))))))) diff --git a/test/toucan2/tools/after_select_test.clj b/test/toucan2/tools/after_select_test.clj index d325c7d..b234c26 100644 --- a/test/toucan2/tools/after_select_test.clj +++ b/test/toucan2/tools/after_select_test.clj @@ -2,6 +2,7 @@ (:require [clojure.string :as str] [clojure.test :refer :all] + [clojure.walk :as walk] [toucan2.insert :as insert] [toucan2.instance :as instance] [toucan2.protocols :as protocols] @@ -144,3 +145,26 @@ (testing `protocols/changes (is (= nil (protocols/changes person))))))) + +(deftest ^:parallel macroexpansion-test + (testing "define-after-select should define vars with different names based on the model." + (letfn [(generated-name* [form] + (cond + (sequential? form) + (some generated-name* form) + + (and (symbol? form) + (str/starts-with? (name form) "after-select")) + form)) + (generated-name [form] + (let [expanded (walk/macroexpand-all form)] + (or (generated-name* expanded) + ['no-match expanded])))] + (is (= 'after-select-primary-method-model-1 + (generated-name `(after-select/define-after-select :model-1 + [~'venue] + ~'venue)))) + (is (= 'after-select-primary-method-model-2 + (generated-name `(after-select/define-after-select :model-2 + [~'venue] + ~'venue))))))) diff --git a/test/toucan2/tools/after_update_test.clj b/test/toucan2/tools/after_update_test.clj index 0cb593f..60ef9b5 100644 --- a/test/toucan2/tools/after_update_test.clj +++ b/test/toucan2/tools/after_update_test.clj @@ -1,6 +1,8 @@ (ns toucan2.tools.after-update-test (:require + [clojure.string :as str] [clojure.test :refer :all] + [clojure.walk :as walk] [toucan2.instance :as instance] [toucan2.select :as select] [toucan2.test :as test] @@ -150,3 +152,26 @@ :updated-at (LocalDateTime/parse "2017-01-01T00:00")})] (select/select [model :id :name :updated-at] {:order-by [[:id :asc]]}))))))))) + +(deftest ^:parallel macroexpansion-test + (testing "define-after-update should define vars with different names based on the model." + (letfn [(generated-name* [form] + (cond + (sequential? form) + (some generated-name* form) + + (and (symbol? form) + (str/starts-with? (name form) "each-row-fn-primary-method-toucan-query-type-update")) + form)) + (generated-name [form] + (let [expanded (walk/macroexpand-all form)] + (or (generated-name* expanded) + ['no-match expanded])))] + (is (= 'each-row-fn-primary-method-toucan-query-type-update-*-model-1 + (generated-name `(after-update/define-after-update :model-1 + [~'venue] + ~'venue)))) + (is (= 'each-row-fn-primary-method-toucan-query-type-update-*-model-2 + (generated-name `(after-update/define-after-update :model-2 + [~'venue] + ~'venue))))))) diff --git a/test/toucan2/tools/before_delete_test.clj b/test/toucan2/tools/before_delete_test.clj index 9b2cd94..53b52bd 100644 --- a/test/toucan2/tools/before_delete_test.clj +++ b/test/toucan2/tools/before_delete_test.clj @@ -1,6 +1,8 @@ (ns toucan2.tools.before-delete-test (:require + [clojure.string :as str] [clojure.test :refer :all] + [clojure.walk :as walk] [toucan2.delete :as delete] [toucan2.execute :as execute] [toucan2.instance :as instance] @@ -174,3 +176,26 @@ @*deleted-venues*)) (is (= nil (select/select-fn-set :id ::test/venues))))))) + +(deftest ^:parallel macroexpansion-test + (testing "define-before-delete should define vars with different names based on the model." + (letfn [(generated-name* [form] + (cond + (sequential? form) + (some generated-name* form) + + (and (symbol? form) + (str/starts-with? (name form) "before-delete")) + form)) + (generated-name [form] + (let [expanded (walk/macroexpand-all form)] + (or (generated-name* expanded) + ['no-match expanded])))] + (is (= 'before-delete-primary-method-model-1 + (generated-name `(before-delete/define-before-delete :model-1 + [~'venue] + ~'venue)))) + (is (= 'before-delete-primary-method-model-2 + (generated-name `(before-delete/define-before-delete :model-2 + [~'venue] + ~'venue))))))) diff --git a/test/toucan2/tools/before_insert_test.clj b/test/toucan2/tools/before_insert_test.clj index cf72cbf..ffa8cc5 100644 --- a/test/toucan2/tools/before_insert_test.clj +++ b/test/toucan2/tools/before_insert_test.clj @@ -3,6 +3,7 @@ [clojure.edn :as edn] [clojure.string :as str] [clojure.test :refer :all] + [clojure.walk :as walk] [toucan2.insert :as insert] [toucan2.instance :as instance] [toucan2.select :as select] @@ -196,3 +197,26 @@ :h2 (OffsetDateTime/parse "2022-12-31T17:26:00-08:00") (:postgres :mariadb) (OffsetDateTime/parse "2023-01-01T01:26Z"))} (select/select-one ::people.default-values 5))))) + +(deftest ^:parallel macroexpansion-test + (testing "define-before-insert should define vars with different names based on the model." + (letfn [(generated-name* [form] + (cond + (sequential? form) + (some generated-name* form) + + (and (symbol? form) + (str/starts-with? (name form) "before-insert-primary-method")) + form)) + (generated-name [form] + (let [expanded (walk/macroexpand-all form)] + (or (generated-name* expanded) + ['no-match expanded])))] + (is (= 'before-insert-primary-method-model-1 + (generated-name `(before-insert/define-before-insert :model-1 + [~'venue] + ~'venue)))) + (is (= 'before-insert-primary-method-model-2 + (generated-name `(before-insert/define-before-insert :model-2 + [~'venue] + ~'venue))))))) diff --git a/test/toucan2/tools/before_select_test.clj b/test/toucan2/tools/before_select_test.clj index 7176ba5..14ecd23 100644 --- a/test/toucan2/tools/before_select_test.clj +++ b/test/toucan2/tools/before_select_test.clj @@ -1,6 +1,8 @@ (ns toucan2.tools.before-select-test (:require + [clojure.string :as str] [clojure.test :refer :all] + [clojure.walk :as walk] [methodical.core :as m] [toucan2.instance :as instance] [toucan2.select :as select] @@ -62,3 +64,25 @@ (select/select-one [::people.increment-id :id :name] :id 1))) (is (= [{:id 1}] @*select-calls*))))) + +(deftest ^:parallel macroexpansion-test + (testing "define-before-select should define vars with different names based on the model." + (letfn [(generated-name* [form] + (cond + (sequential? form) + (some generated-name* form) + + (and (symbol? form) + (str/starts-with? (name form) "before-select")) form)) + (generated-name [form] + (let [expanded (walk/macroexpand-all form)] + (or (generated-name* expanded) + ['no-match expanded])))] + (is (= 'before-select-primary-method-model-1 + (generated-name `(before-select/define-before-select :model-1 + [~'venue] + ~'venue)))) + (is (= 'before-select-primary-method-model-2 + (generated-name `(before-select/define-before-select :model-2 + [~'venue] + ~'venue))))))) diff --git a/test/toucan2/tools/before_update_test.clj b/test/toucan2/tools/before_update_test.clj index 2a3e6f3..f442fb8 100644 --- a/test/toucan2/tools/before_update_test.clj +++ b/test/toucan2/tools/before_update_test.clj @@ -2,6 +2,7 @@ (:require [clojure.string :as str] [clojure.test :refer :all] + [clojure.walk :as walk] [methodical.core :as m] [toucan2.execute :as execute] [toucan2.instance :as instance] @@ -444,3 +445,25 @@ :name "CAM 2.0" :created-at (OffsetDateTime/parse "2019-01-11T23:56Z")} (select/select-one ::test/people 2))))))) + +(deftest ^:parallel macroexpansion-test + (testing "define-before-update should define vars with different names based on the model." + (letfn [(generated-name* [form] + (cond + (sequential? form) + (some generated-name* form) + + (and (symbol? form) + (str/starts-with? (name form) "before-update")) form)) + (generated-name [form] + (let [expanded (walk/macroexpand-all form)] + (or (generated-name* expanded) + ['no-match expanded])))] + (is (= 'before-update-primary-method-model-1 + (generated-name `(before-update/define-before-update :model-1 + [~'venue] + ~'venue)))) + (is (= 'before-update-primary-method-model-2 + (generated-name `(before-update/define-before-update :model-2 + [~'venue] + ~'venue))))))) diff --git a/test/toucan2/tools/default_fields_test.clj b/test/toucan2/tools/default_fields_test.clj index 783c09d..8ca64a2 100644 --- a/test/toucan2/tools/default_fields_test.clj +++ b/test/toucan2/tools/default_fields_test.clj @@ -2,6 +2,7 @@ (:require [clojure.string :as str] [clojure.test :refer :all] + [clojure.walk :as walk] [toucan2.insert :as insert] [toucan2.instance :as instance] [toucan2.pipeline :as pipeline] @@ -185,3 +186,25 @@ (select/select-one ::venues.default-fields query)) {select [:id :name :updated-at], :where [:= :id 1]} ::named-query.select-venues-override-default-fields))))) + +(deftest ^:parallel macroexpansion-test + (testing "define-default-fields should define vars with different names based on the model." + (letfn [(generated-name* [form] + (cond + (sequential? form) + (some generated-name* form) + + (and (symbol? form) + (str/starts-with? (name form) "default-fields")) form)) + (generated-name [form] + (let [expanded (walk/macroexpand-all form)] + (or (generated-name* expanded) + ['no-match expanded])))] + (is (= 'default-fields-primary-method-model-1 + (generated-name `(default-fields/define-default-fields :model-1 + [~'venue] + ~'venue)))) + (is (= 'default-fields-primary-method-model-2 + (generated-name `(default-fields/define-default-fields :model-2 + [~'venue] + ~'venue))))))) diff --git a/test/toucan2/tools/transformed_test.clj b/test/toucan2/tools/transformed_test.clj index 2b928d8..ba15862 100644 --- a/test/toucan2/tools/transformed_test.clj +++ b/test/toucan2/tools/transformed_test.clj @@ -1,6 +1,8 @@ (ns toucan2.tools.transformed-test (:require + [clojure.string :as str] [clojure.test :refer :all] + [clojure.walk :as walk] [methodical.core :as m] [toucan2.delete :as delete] [toucan2.insert :as insert] @@ -521,3 +523,23 @@ Throwable #"Invalid deftransforms map" (transformed/transforms ::invalid)))) + +(deftest ^:parallel macroexpansion-test + (testing "deftransforms should define vars with different names based on the model." + (letfn [(generated-name* [form] + (cond + (sequential? form) + (some generated-name* form) + + (and (symbol? form) + (str/starts-with? (name form) "transforms-primary-method")) form)) + (generated-name [form] + (let [expanded (walk/macroexpand-all form)] + (or (generated-name* expanded) + ['no-match expanded])))] + (is (= 'transforms-primary-method-model-1 + (generated-name `(transformed/deftransforms :model-1 + {:x {:in ~'inc}})))) + (is (= 'transforms-primary-method-model-2 + (generated-name `(transformed/deftransforms :model-2 + {:x {:in ~'inc}}))))))) diff --git a/toucan1/test/toucan/db_test.clj b/toucan1/test/toucan/db_test.clj index b7f97e0..bb4d010 100644 --- a/toucan1/test/toucan/db_test.clj +++ b/toucan1/test/toucan/db_test.clj @@ -2,7 +2,6 @@ (:require [clojure.string :as str] [clojure.test :refer :all] - [honey.sql :as hsql] [methodical.core :as m] [toucan.db :as t1.db] [toucan.models :as t1.models] @@ -496,14 +495,14 @@ (testing "it must not fail when using SQL function calls." (test/with-discarded-table-changes User (is (= [4 5] - (t1.db/simple-insert-many! User [{:first-name "Grass" :last-name (hsql/call :upper "Hopper")} + (t1.db/simple-insert-many! User [{:first-name "Grass" :last-name [:upper "Hopper"]} {:first-name "Ko" :last-name "Libri"}])))))) (deftest ^:synchronized insert-many!-test (testing "It must return the inserted ids, it must not fail when using SQL function calls." (test/with-discarded-table-changes User (is (= [4 5] - (t1.db/insert-many! User [{:first-name "Grass" :last-name (hsql/call :upper "Hopper")} + (t1.db/insert-many! User [{:first-name "Grass" :last-name [:upper "Hopper"]} {:first-name "Ko" :last-name "Libri"}]))))) (testing "It must call pre-insert hooks" (test/with-discarded-table-changes Category @@ -523,7 +522,7 @@ (testing "The returned data must match what's been inserted in the table" (test/with-discarded-table-changes User (is (= {:id 4, :first-name "Grass", :last-name "HOPPER"} - (t1.db/insert! User {:first-name "Grass" :last-name (hsql/call :upper "Hopper")})))))) + (t1.db/insert! User {:first-name "Grass" :last-name [:upper "Hopper"]})))))) (deftest ^:parallel select-one-test (is (= {:id 1, :first-name "Cam", :last-name "Saul"}