Compare commits
63 commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e3b80e6128 | ||
|
|
150a4aa605 | ||
|
|
2a30c5cffd | ||
|
|
0c385c36a6 | ||
|
|
af827c0200 | ||
|
|
0ff09dca21 | ||
|
|
fd3c61b1e6 | ||
|
|
a3fba32605 | ||
|
|
f39f6b3d24 | ||
|
|
99075cd408 | ||
|
|
14b4c3435f | ||
|
|
a40abc285e | ||
|
|
16c1faeaf6 | ||
|
|
d1115c8c42 | ||
|
|
60503dac03 | ||
|
|
a829074b99 | ||
|
|
532d77f372 | ||
|
|
e650137f07 | ||
|
|
9fdc24eab7 | ||
|
|
1642cc04ca | ||
|
|
9d1c35b86c | ||
|
|
ed8ba5402a | ||
|
|
403d156331 | ||
|
|
78894f8ba4 | ||
|
|
0d7b3b60ed | ||
|
|
2214e06cfa | ||
|
|
0d7d58fece | ||
|
|
c2b64dbd54 | ||
|
|
24085c459f | ||
|
|
1062f572d5 | ||
|
|
220957a69f | ||
|
|
c89744b05f | ||
|
|
75dab5d843 | ||
|
|
22e96dcb84 | ||
|
|
9ed335dc8d | ||
|
|
e6d1abf3ec | ||
|
|
946d160409 | ||
|
|
1294e35b98 | ||
|
|
fa6f9040d7 | ||
|
|
3de04761cc | ||
|
|
08c08e7f7b | ||
|
|
56bd2356ac | ||
|
|
c6f86b042e | ||
|
|
a592b7ebae | ||
|
|
d71a3b6423 | ||
|
|
295fd5ddf5 | ||
|
|
f9bfb1248a | ||
|
|
d5f284871b | ||
|
|
2dc72d9254 | ||
|
|
b981357d47 | ||
|
|
801e6c923b | ||
|
|
151b7d8d0b | ||
|
|
87f4224c22 | ||
|
|
4db63f6122 | ||
|
|
0282df7629 | ||
|
|
80748a9593 | ||
|
|
9196b34249 | ||
|
|
f43e5490ac | ||
|
|
cad6462c53 | ||
|
|
05cfe1f3fa | ||
|
|
564c43bc79 | ||
|
|
5f0c93642a | ||
|
|
0c50cf28b5 |
53 changed files with 2088 additions and 998 deletions
8
.clj-kondo/imports/com.xtdb/xtdb-api/config.edn
Normal file
8
.clj-kondo/imports/com.xtdb/xtdb-api/config.edn
Normal file
|
|
@ -0,0 +1,8 @@
|
|||
{:linters {:xtql/redundant-pipeline {:level :warning}
|
||||
:xtql/redundant-unify {:level :warning}
|
||||
:xtql/unrecognized-operation {:level :error}
|
||||
:xtql/unrecognized-parameter {:level :warning}
|
||||
:xtql/missing-parameter {:level :error}
|
||||
:xtql/type-mismatch {:level :error}
|
||||
:xtql/invalid-arity {:level :error}}
|
||||
:hooks {:analyze-call {xtdb.api/q hooks.xtql/q}}}
|
||||
567
.clj-kondo/imports/com.xtdb/xtdb-api/hooks/xtql.clj
Normal file
567
.clj-kondo/imports/com.xtdb/xtdb-api/hooks/xtql.clj
Normal file
|
|
@ -0,0 +1,567 @@
|
|||
(ns ^:no-doc hooks.xtql
|
||||
(:require [clj-kondo.hooks-api :as api]))
|
||||
|
||||
(def source-op?
|
||||
#{'from 'rel 'unify})
|
||||
|
||||
(def tail-op?
|
||||
#{'aggregate
|
||||
'limit 'offset
|
||||
'where
|
||||
'order-by
|
||||
'with 'without 'return
|
||||
'unnest})
|
||||
|
||||
(def unify-clause?
|
||||
#{'from 'rel
|
||||
'join 'left-join
|
||||
'unnest
|
||||
'where
|
||||
'with})
|
||||
|
||||
(defn node-map? [node]
|
||||
(contains? #{:map :namespaced-map}
|
||||
(:tag node)))
|
||||
|
||||
(defn node-namespaced-map? [node]
|
||||
(= :namespaced-map (:tag node)))
|
||||
|
||||
(defn map-children [node]
|
||||
(->> (if (node-namespaced-map? node)
|
||||
(-> node :children first)
|
||||
node)
|
||||
:children
|
||||
(partition-all 2)))
|
||||
|
||||
(defn node-vector? [node]
|
||||
(= :vector (:tag node)))
|
||||
|
||||
(defn node-list? [node]
|
||||
(= :list (:tag node)))
|
||||
|
||||
(defn node-symbol? [node]
|
||||
(symbol? (:value node)))
|
||||
|
||||
(defn node-symbol [node]
|
||||
(:value node))
|
||||
|
||||
(defn node-keyword? [node]
|
||||
(keyword? (:k node)))
|
||||
|
||||
(defn node-keyword [node]
|
||||
(:k node))
|
||||
|
||||
(defn node-quote? [node]
|
||||
(= :quote (:tag node)))
|
||||
|
||||
(defn node-op [node]
|
||||
(-> node :children first))
|
||||
|
||||
(declare lint-query)
|
||||
|
||||
(defmulti lint-unify-clause #(-> % node-op node-symbol))
|
||||
(defmulti lint-source-op #(-> % node-op node-symbol))
|
||||
(defmulti lint-tail-op #(-> % node-op node-symbol))
|
||||
|
||||
(defn lint-not-arg-symbol [node]
|
||||
(when (= \$ (-> node node-symbol str first))
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "unexpected parameter in binding"
|
||||
:type :xtql/unrecognized-parameter))))
|
||||
|
||||
(defn lint-bind [node]
|
||||
(cond
|
||||
(node-symbol? node)
|
||||
;; TODO: Make own type, should really be a warning
|
||||
(lint-not-arg-symbol node)
|
||||
|
||||
(node-map? node)
|
||||
(doseq [[k _v] (map-children node)]
|
||||
(when-not (node-keyword? k)
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "all keys in binding maps must be keywords"
|
||||
:type :xtql/type-mismatch))))
|
||||
|
||||
:else
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "expected a symbol or map"
|
||||
:type :xtql/type-mismatch))))
|
||||
|
||||
;; TODO: Lint more unify clauses
|
||||
(defmethod lint-unify-clause :default [node]
|
||||
(when-not (unify-clause? (-> node node-op node-symbol))
|
||||
(api/reg-finding!
|
||||
(assoc (some-> node :children first meta)
|
||||
:message "unrecognized unify clause"
|
||||
:type :xtql/unrecognized-operation))))
|
||||
|
||||
(defmethod lint-unify-clause 'from [node]
|
||||
(lint-source-op node))
|
||||
|
||||
(defmethod lint-unify-clause 'rel [node]
|
||||
(lint-source-op node))
|
||||
|
||||
(defmethod lint-unify-clause 'with [node]
|
||||
(let [opts (-> node :children rest)]
|
||||
(when-not (>= (count opts) 1)
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "expected at least one argument"
|
||||
:type :xtql/invalid-arity)))
|
||||
(doseq [opt opts]
|
||||
(if (node-map? opt)
|
||||
(let [ks (->> opt
|
||||
map-children
|
||||
(map first)
|
||||
(remove node-symbol?))]
|
||||
(doseq [k ks]
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "expected all keys to be symbols in a unify"
|
||||
:type :xtql/type-mismatch))))
|
||||
(api/reg-finding!
|
||||
(assoc (meta opt)
|
||||
:message "opts must be a map"
|
||||
:type :xtql/type-mismatch))))))
|
||||
|
||||
(defn lint-join-clause [node]
|
||||
(let [args (-> node :children rest)]
|
||||
(if-not (= (count args) 2)
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "expected at exactly two arguments"
|
||||
:type :xtql/invalid-arity))
|
||||
(let [[query opts] args]
|
||||
(lint-query query)
|
||||
(cond
|
||||
(node-vector? opts)
|
||||
(->> opts :children (run! lint-bind))
|
||||
(node-map? opts)
|
||||
(let [kvs (map-children opts)
|
||||
ks (->> kvs
|
||||
(map first)
|
||||
(map node-keyword)
|
||||
(remove nil?)
|
||||
(into #{}))]
|
||||
(when-not (contains? ks :bind)
|
||||
(api/reg-finding!
|
||||
(assoc (meta opts)
|
||||
:message "Missing :bind parameter"
|
||||
:type :xtql/missing-parameter)))
|
||||
(doseq [[k v] kvs]
|
||||
(when-not (node-keyword? k)
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "All keys in 'opts' must be keywords"
|
||||
:type :xtql/type-mismatch)))
|
||||
(case (node-keyword k)
|
||||
:bind (if (node-vector? v)
|
||||
(->> v :children (run! lint-bind))
|
||||
(api/reg-finding!
|
||||
(assoc (meta v)
|
||||
:message "expected :bind value to be a vector"
|
||||
:type :xtql/type-mismatch)))
|
||||
:args (if (node-vector? v)
|
||||
;; TODO: Make args specific
|
||||
(->> v :children (run! lint-bind))
|
||||
(api/reg-finding!
|
||||
(assoc (meta v)
|
||||
:message "expected :args value to be a vector"
|
||||
:type :xtql/type-mismatch)))
|
||||
; else
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "unrecognized parameter"
|
||||
:type :xtql/unrecognized-parameter)))))
|
||||
:else
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "opts must be a map or vector"
|
||||
:type :xtql/type-mismatch)))))))
|
||||
|
||||
(defmethod lint-unify-clause 'join [node]
|
||||
(lint-join-clause node))
|
||||
|
||||
(defmethod lint-unify-clause 'inner-join [node]
|
||||
(lint-join-clause node))
|
||||
|
||||
(defmethod lint-unify-clause 'unnest [node]
|
||||
(let [opts (-> node :children rest)]
|
||||
(when-not (= 1 (count opts))
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "expected at exactly one argument"
|
||||
:type :xtql/invalid-arity)))
|
||||
(let [opt (first opts)]
|
||||
(if (node-map? opt)
|
||||
(doseq [[k _v] (map-children opt)]
|
||||
(when-not (node-symbol? k)
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "expected all columns to be symbols"
|
||||
:type :xtql/type-mismatch))))
|
||||
(api/reg-finding!
|
||||
(assoc (meta opt)
|
||||
:message "expected opt to be a map"
|
||||
:type :xtql/type-mismatch))))))
|
||||
|
||||
|
||||
(defmethod lint-source-op :default [node]
|
||||
(let [op (-> node node-op node-symbol)]
|
||||
(if (tail-op? op)
|
||||
(api/reg-finding!
|
||||
(assoc (some-> node :children first meta)
|
||||
:message "tail op in source position"
|
||||
:type :xtql/unrecognized-operation))
|
||||
(when-not (source-op? op)
|
||||
(api/reg-finding!
|
||||
(assoc (some-> node :children first meta)
|
||||
:message "unrecognized source operation"
|
||||
:type :xtql/unrecognized-operation))))))
|
||||
|
||||
(defmethod lint-source-op 'from [node]
|
||||
(let [[_ table opts] (some-> node :children)]
|
||||
(when-not (node-keyword? table)
|
||||
(api/reg-finding!
|
||||
(assoc (meta table)
|
||||
:message "expected 'table' to be a keyword"
|
||||
:type :xtql/type-mismatch)))
|
||||
(case (:tag opts)
|
||||
:vector (->> opts :children (run! lint-bind))
|
||||
:map
|
||||
(let [kvs (map-children opts)
|
||||
ks (->> kvs
|
||||
(map first)
|
||||
(map node-keyword)
|
||||
(remove nil?)
|
||||
(into #{}))]
|
||||
(when-not (contains? ks :bind)
|
||||
(api/reg-finding!
|
||||
(assoc (meta opts)
|
||||
:message "Missing :bind parameter"
|
||||
:type :xtql/missing-parameter)))
|
||||
(doseq [[k v] kvs]
|
||||
(when-not (node-keyword? k)
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "All keys in 'opts' must be keywords"
|
||||
:type :xtql/type-mismatch)))
|
||||
(case (node-keyword k)
|
||||
:bind (if (node-vector? v)
|
||||
(->> v :children (run! lint-bind))
|
||||
(api/reg-finding!
|
||||
(assoc (meta opts)
|
||||
:message "expected :bind value to be a vector"
|
||||
:type :xtql/type-mismatch)))
|
||||
;; TODO
|
||||
:for-valid-time nil
|
||||
;; TODO
|
||||
:for-system-time nil
|
||||
; else
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "unrecognized parameter"
|
||||
:type :xtql/unrecognized-parameter)))))
|
||||
(api/reg-finding!
|
||||
(assoc (meta opts)
|
||||
:message "expected 'opts' to be either a map or vector"
|
||||
:type :xtql/type-mismatch)))))
|
||||
|
||||
(defmethod lint-source-op 'unify [node]
|
||||
(let [[_ & clauses] (some-> node :children)]
|
||||
(doseq [bad-op (remove node-list? clauses)]
|
||||
(api/reg-finding!
|
||||
(assoc (meta bad-op)
|
||||
:message "all operations in a unify must be lists"
|
||||
:type :xtql/type-mismatch)))
|
||||
(when (= (count clauses) 1)
|
||||
(let [clause (first clauses)
|
||||
clause-op (-> clause node-op node-symbol)
|
||||
unify-node (some-> node :children first)]
|
||||
(case clause-op
|
||||
from (api/reg-finding!
|
||||
(assoc (meta unify-node)
|
||||
:message "redundant unify"
|
||||
:type :xtql/redundant-unify))
|
||||
rel (api/reg-finding!
|
||||
(assoc (meta unify-node)
|
||||
:message "redundant unify"
|
||||
:type :xtql/redundant-unify))
|
||||
;; TODO: Cover other operators
|
||||
nil)))
|
||||
(->> clauses
|
||||
(filter node-list?)
|
||||
(run! lint-unify-clause))))
|
||||
|
||||
(defmethod lint-source-op 'rel [node]
|
||||
(let [[_ _expr binds] (some-> node :children)]
|
||||
(if (node-vector? binds)
|
||||
(->> binds :children (run! lint-bind))
|
||||
(api/reg-finding!
|
||||
(assoc (meta binds)
|
||||
:message "expected rel binding to be a vector"
|
||||
:type :xtql/type-mismatch)))))
|
||||
|
||||
;; TODO: Lint more tail ops
|
||||
(defmethod lint-tail-op :default [node]
|
||||
(let [op (-> node node-op node-symbol)]
|
||||
(if (source-op? op)
|
||||
(api/reg-finding!
|
||||
(assoc (some-> node :children first meta)
|
||||
:message "source op in tail position"
|
||||
:type :xtql/unrecognized-operation))
|
||||
(when-not (tail-op? op)
|
||||
(api/reg-finding!
|
||||
(assoc (some-> node :children first meta)
|
||||
:message "unrecognized tail operation"
|
||||
:type :xtql/unrecognized-operation))))))
|
||||
|
||||
(defn lint-keyword [node name]
|
||||
(when-not (node-keyword? node)
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message (str "expected '" name "' to be a keyword")
|
||||
:type :xtql/type-mismatch))))
|
||||
|
||||
(defn lint-enum [node name values]
|
||||
;; TODO: Expand to more than just keywords?
|
||||
;; Maybe a `node-value` function?
|
||||
(when-not (contains? values (node-keyword node))
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message (str "expected '" name "' to be one of " values)
|
||||
;; TODO: change to different type?
|
||||
:type :xtql/type-mismatch))))
|
||||
|
||||
(defmethod lint-tail-op 'limit [node]
|
||||
(let [opts (-> node :children rest)]
|
||||
(when-not (= 1 (count opts))
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "expected exactly one argument"
|
||||
:type :xtql/invalid-arity)))
|
||||
(when-let [opt (first opts)]
|
||||
(when-not (some-> opt :value int?)
|
||||
(api/reg-finding!
|
||||
(assoc (meta opt)
|
||||
:message "expected limit to be an integer"
|
||||
:type :xtql/type-mismatch))))))
|
||||
|
||||
(defmethod lint-tail-op 'offset [node]
|
||||
(let [opts (-> node :children rest)]
|
||||
(when-not (= 1 (count opts))
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "expected exactly one argument"
|
||||
:type :xtql/invalid-arity)))
|
||||
(when-let [opt (first opts)]
|
||||
(when-not (some-> opt :value int?)
|
||||
(api/reg-finding!
|
||||
(assoc (meta opt)
|
||||
:message "expected offset to be an integer"
|
||||
:type :xtql/type-mismatch))))))
|
||||
|
||||
(defmethod lint-tail-op 'with [node]
|
||||
(let [opts (-> node :children rest)]
|
||||
(when-not (>= (count opts) 1)
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "expected at least one argument"
|
||||
:type :xtql/invalid-arity)))
|
||||
(doseq [opt opts]
|
||||
(cond
|
||||
(node-symbol? opt)
|
||||
(lint-not-arg-symbol opt)
|
||||
(node-map? opt)
|
||||
(let [ks (->> opt
|
||||
map-children
|
||||
(map first)
|
||||
(remove node-keyword?))]
|
||||
(doseq [k ks]
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "expected all keys to be keywords"
|
||||
:type :xtql/type-mismatch))))
|
||||
:else
|
||||
(api/reg-finding!
|
||||
(assoc (meta opt)
|
||||
:message "opts must be a symbol or map"
|
||||
:type :xtql/type-mismatch))))))
|
||||
|
||||
(defmethod lint-tail-op 'return [node]
|
||||
(let [opts (-> node :children rest)]
|
||||
(when-not (>= (count opts) 1)
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "expected at least one argument"
|
||||
:type :xtql/invalid-arity)))
|
||||
(doseq [opt opts]
|
||||
(cond
|
||||
(node-symbol? opt)
|
||||
(lint-not-arg-symbol opt)
|
||||
(node-map? opt)
|
||||
(let [ks (->> opt
|
||||
map-children
|
||||
(map first)
|
||||
(remove node-keyword?))]
|
||||
(doseq [k ks]
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "expected all keys to be keywords"
|
||||
:type :xtql/type-mismatch))))
|
||||
:else
|
||||
(api/reg-finding!
|
||||
(assoc (meta opt)
|
||||
:message "opts must be a symbol or map"
|
||||
:type :xtql/type-mismatch))))))
|
||||
|
||||
(defmethod lint-tail-op 'order-by [node]
|
||||
(doseq [opt (-> node :children rest)]
|
||||
(cond
|
||||
(node-symbol? opt)
|
||||
(lint-not-arg-symbol opt)
|
||||
|
||||
(node-map? opt)
|
||||
(let [kvs (map-children opt)
|
||||
ks (->> kvs
|
||||
(map first)
|
||||
(map node-keyword)
|
||||
(remove nil?)
|
||||
(into #{}))]
|
||||
(when-not (contains? ks :val)
|
||||
(api/reg-finding!
|
||||
(assoc (meta opt)
|
||||
:message "Missing :val parameter"
|
||||
:type :xtql/missing-parameter)))
|
||||
(doseq [[k v] kvs]
|
||||
(when-not (node-keyword? k)
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "All keys in 'opts' must be keywords"
|
||||
:type :xtql/type-mismatch)))
|
||||
(case (node-keyword k)
|
||||
:val
|
||||
(cond
|
||||
(node-symbol? v)
|
||||
(lint-not-arg-symbol v)
|
||||
(node-keyword? v)
|
||||
(api/reg-finding!
|
||||
(assoc (meta v)
|
||||
:message "expected :val value to be a symbol or an expression"
|
||||
:type :xtql/type-mismatch)))
|
||||
; else do nothing
|
||||
:dir
|
||||
(if (node-keyword? v)
|
||||
(lint-enum v :dir #{:asc :desc})
|
||||
(lint-keyword v ":dir value"))
|
||||
:nulls
|
||||
(if (node-keyword? v)
|
||||
(lint-enum v :nulls #{:first :last})
|
||||
(lint-keyword v ":nulls value"))
|
||||
; else
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "unrecognized parameter"
|
||||
:type :xtql/unrecognized-parameter)))))
|
||||
|
||||
:else
|
||||
(api/reg-finding!
|
||||
(assoc (meta opt)
|
||||
:message "opts must be a symbol or map"
|
||||
:type :xtql/type-mismatch)))))
|
||||
|
||||
(defmethod lint-tail-op 'without [node]
|
||||
(let [columns (-> node :children rest)]
|
||||
(when-not (>= (count columns) 1)
|
||||
;; TODO: Should be a warning really
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "expected at least one column"
|
||||
:type :xtql/invalid-arity)))
|
||||
(doseq [column columns]
|
||||
(when-not (node-keyword? column)
|
||||
(api/reg-finding!
|
||||
(assoc (meta column)
|
||||
:message "expected column to be a keyword"
|
||||
:type :xtql/type-mismatch))))))
|
||||
|
||||
(defmethod lint-tail-op 'aggregate [node]
|
||||
(let [opts (-> node :children rest)]
|
||||
(when-not (>= (count opts) 1)
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "expected at least one argument"
|
||||
:type :xtql/invalid-arity)))
|
||||
(doseq [opt opts]
|
||||
(cond
|
||||
(node-symbol? opt)
|
||||
(lint-not-arg-symbol opt)
|
||||
(node-map? opt)
|
||||
(doseq [[k _v] (map-children opt)]
|
||||
(when-not (node-keyword? k)
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "expected all keys to be keywords"
|
||||
:type :xtql/type-mismatch))))
|
||||
|
||||
:else
|
||||
(api/reg-finding!
|
||||
(assoc (meta opt)
|
||||
:message "expected opts to be a symbol or map"
|
||||
:type :xtql/type-mismatch))))))
|
||||
|
||||
(defmethod lint-tail-op 'unnest [node]
|
||||
(let [opts (-> node :children rest)]
|
||||
(when-not (= 1 (count opts))
|
||||
(api/reg-finding!
|
||||
(assoc (meta node)
|
||||
:message "expected at exactly one argument"
|
||||
:type :xtql/invalid-arity)))
|
||||
(let [opt (first opts)]
|
||||
(if (node-map? opt)
|
||||
(doseq [[k _v] (map-children opt)]
|
||||
(when-not (node-keyword? k)
|
||||
(api/reg-finding!
|
||||
(assoc (meta k)
|
||||
:message "expected all columns to be keywords"
|
||||
:type :xtql/type-mismatch))))
|
||||
(api/reg-finding!
|
||||
(assoc (meta opt)
|
||||
:message "expected opt to be a map"
|
||||
:type :xtql/type-mismatch))))))
|
||||
|
||||
(defn lint-pipeline [node]
|
||||
(let [[_ & ops] (some-> node :children)]
|
||||
(doseq [bad-op (remove node-list? ops)]
|
||||
(api/reg-finding!
|
||||
(assoc (meta bad-op)
|
||||
:message "all operations in a pipeline must be lists"
|
||||
:type :xtql/type-mismatch)))
|
||||
(when (= 1 (count ops))
|
||||
(api/reg-finding!
|
||||
(assoc (-> node :children first meta)
|
||||
:message "redundant pipeline"
|
||||
:type :xtql/redundant-pipeline)))
|
||||
(let [first-op (first ops)]
|
||||
(when (node-list? first-op)
|
||||
(lint-source-op (first ops))))
|
||||
(->> ops
|
||||
(drop 1)
|
||||
(filter node-list?)
|
||||
(run! lint-tail-op))))
|
||||
|
||||
(defn lint-query [node]
|
||||
(if (= '-> (node-symbol (-> node :children first)))
|
||||
(lint-pipeline node)
|
||||
(lint-source-op node)))
|
||||
|
||||
;; TODO: Lint other functions that take queries
|
||||
|
||||
(defn q [{:keys [node]}]
|
||||
(let [[_ _node quoted-query] (some-> node :children)]
|
||||
(when (node-quote? quoted-query)
|
||||
(let [query (-> quoted-query :children first)]
|
||||
(lint-query query)))))
|
||||
6
.clj-kondo/imports/cond_plus/cond_plus/config.edn
Normal file
6
.clj-kondo/imports/cond_plus/cond_plus/config.edn
Normal file
|
|
@ -0,0 +1,6 @@
|
|||
{:linters {:cond-plus/empty-else {:level :error}
|
||||
:cond-plus/missing-fn {:level :error}
|
||||
:cond-plus/non-final-else {:level :error}
|
||||
:cond-plus/sequence {:level :error}
|
||||
:unresolved-symbol {:exclude [(cond-plus.core/cond+ [=> else])]}}
|
||||
:hooks {:analyze-call {cond-plus.core/cond+ hooks.cond-plus-hook/cond+}}}
|
||||
|
|
@ -0,0 +1,65 @@
|
|||
(ns hooks.cond-plus-hook
|
||||
(:require [clj-kondo.hooks-api :as api]))
|
||||
|
||||
(defn analyze-clauses [clauses]
|
||||
(reduce
|
||||
(fn [found-else? clause]
|
||||
;; non-sequence clause
|
||||
(if (not (or (api/list-node? clause)
|
||||
(api/vector-node? clause)))
|
||||
(let [{:keys [row col]} (meta clause)]
|
||||
(api/reg-finding!
|
||||
{:message "must be sequence"
|
||||
:type :cond-plus/sequence
|
||||
:row row
|
||||
:col col})
|
||||
found-else?)
|
||||
(let [[sym arrow fn-expr] (api/sexpr clause)]
|
||||
(cond
|
||||
;; non-final else
|
||||
found-else?
|
||||
(do (api/reg-finding!
|
||||
(merge
|
||||
{:message ":else must be in final position"
|
||||
:type :cond-plus/non-final-else}
|
||||
found-else?))
|
||||
(reduced nil))
|
||||
;; check fn-exprs
|
||||
(and (or (= :> arrow)
|
||||
(= '=> arrow))
|
||||
(nil? fn-expr))
|
||||
(let [{:keys [row col]} (meta clause)]
|
||||
(api/reg-finding!
|
||||
{:message "fn-expr must have third position symbol"
|
||||
:type :cond-plus/missing-fn
|
||||
:row row
|
||||
:col col})
|
||||
found-else?)
|
||||
;; else handling
|
||||
(or (= :else sym)
|
||||
(= 'else sym))
|
||||
(if found-else?
|
||||
(let [{:keys [row col]} (meta clause)]
|
||||
(api/reg-finding!
|
||||
{:message "only one :else clause allowed"
|
||||
:type :cond-plus/empty-else
|
||||
:row row
|
||||
:col col})
|
||||
;; early exit cuz not worth analyzing the rest
|
||||
(reduced nil))
|
||||
(do (when-not arrow
|
||||
(let [{:keys [row col]} (meta clause)]
|
||||
(api/reg-finding!
|
||||
{:message ":else must have a body"
|
||||
:type :cond-plus/empty-else
|
||||
:row row
|
||||
:col col})))
|
||||
;; Store row and col from existing else as we don't throw until
|
||||
;; we've seen a following clause
|
||||
(select-keys (meta clause) [:row :col])))))))
|
||||
nil
|
||||
clauses))
|
||||
|
||||
(defn cond+ [{:keys [node]}]
|
||||
(analyze-clauses (rest (:children node)))
|
||||
node)
|
||||
23
.clj-kondo/imports/io.github.noahtheduke/lazytest/config.edn
Normal file
23
.clj-kondo/imports/io.github.noahtheduke/lazytest/config.edn
Normal file
|
|
@ -0,0 +1,23 @@
|
|||
{:lint-as {lazytest.core/given clojure.core/let
|
||||
lazytest.core/around clojure.core/fn
|
||||
lazytest.core/defdescribe clojure.core/def
|
||||
;; clojure.test interface
|
||||
lazytest.experimental.interfaces.clojure-test/deftest clojure.test/deftest
|
||||
lazytest.experimental.interfaces.clojure-test/testing clojure.test/testing
|
||||
lazytest.experimental.interfaces.clojure-test/is clojure.test/is
|
||||
lazytest.experimental.interfaces.clojure-test/are clojure.test/are
|
||||
;; xunit interface
|
||||
lazytest.experimental.interfaces.xunit/defsuite clojure.core/def
|
||||
;; Expectations v2
|
||||
lazytest.extensions.expectations/defexpect clojure.core/def
|
||||
lazytest.extensions.expectations/from-each clojure.core/for
|
||||
lazytest.extensions.expectations/=? clojure.core/=
|
||||
}
|
||||
:hooks {:analyze-call {;; Expectations v2
|
||||
lazytest.extensions.expectations/more-> hooks.lazytest.expectations/more->
|
||||
lazytest.extensions.expectations/more-of hooks.lazytest.expectations/more-of
|
||||
}}
|
||||
:linters {:clojure-lsp/unused-public-var
|
||||
{:exclude-when-defined-by #{lazytest.core/defdescribe
|
||||
lazytest.experimental.interfaces.xunit/defsuite
|
||||
lazytest.experimental.interfaces.clojure-test/deftest}}}}
|
||||
|
|
@ -0,0 +1,31 @@
|
|||
;; Copied from https://github.com/clojure-expectations/clojure-test/blob/b90ed5b24924238b3b16b0bbaaee4c3b05a1268a
|
||||
|
||||
(ns hooks.lazytest.expectations
|
||||
(:require [clj-kondo.hooks-api :as api]))
|
||||
|
||||
(defn more-> [{:keys [node]}]
|
||||
(let [tail (rest (:children node))
|
||||
rewritten
|
||||
(api/list-node
|
||||
(list*
|
||||
(api/token-node 'cond->)
|
||||
(api/token-node 'nil)
|
||||
tail))]
|
||||
{:node rewritten}))
|
||||
|
||||
(defn more-of [{:keys [node]}]
|
||||
(let [bindings (fnext (:children node))
|
||||
pairs (partition 2 (nnext (:children node)))
|
||||
rewritten
|
||||
(api/list-node
|
||||
(list*
|
||||
(api/token-node 'fn)
|
||||
(api/vector-node (vector bindings))
|
||||
(map (fn [[e a]]
|
||||
(api/list-node
|
||||
(list
|
||||
(api/token-node 'lazytest.core/expect)
|
||||
e
|
||||
a)))
|
||||
pairs)))]
|
||||
{:node rewritten}))
|
||||
4
.clj-kondo/imports/nubank/matcher-combinators/config.edn
Normal file
4
.clj-kondo/imports/nubank/matcher-combinators/config.edn
Normal file
|
|
@ -0,0 +1,4 @@
|
|||
{:linters
|
||||
{:unresolved-symbol
|
||||
{:exclude [(cljs.test/is [match? thrown-match?])
|
||||
(clojure.test/is [match? thrown-match?])]}}}
|
||||
5
.clj-kondo/imports/rewrite-clj/rewrite-clj/config.edn
Normal file
5
.clj-kondo/imports/rewrite-clj/rewrite-clj/config.edn
Normal file
|
|
@ -0,0 +1,5 @@
|
|||
{:lint-as
|
||||
{rewrite-clj.zip/subedit-> clojure.core/->
|
||||
rewrite-clj.zip/subedit->> clojure.core/->>
|
||||
rewrite-clj.zip/edit-> clojure.core/->
|
||||
rewrite-clj.zip/edit->> clojure.core/->>}}
|
||||
1
.clj-kondo/inline-configs/xtdb.api.clj/config.edn
Normal file
1
.clj-kondo/inline-configs/xtdb.api.clj/config.edn
Normal file
|
|
@ -0,0 +1 @@
|
|||
{:config-in-call {xtdb.api/template {:ignore [:unresolved-symbol :unresolved-namespace]}}}
|
||||
7
.github/workflows/test-and-release.yml
vendored
7
.github/workflows/test-and-release.yml
vendored
|
|
@ -15,11 +15,11 @@ jobs:
|
|||
- uses: actions/setup-java@v4
|
||||
with:
|
||||
distribution: 'temurin'
|
||||
java-version: '11'
|
||||
java-version: '21'
|
||||
- name: Setup Clojure
|
||||
uses: DeLaGuardo/setup-clojure@master
|
||||
with:
|
||||
cli: '1.12.0.1479'
|
||||
cli: '1.12.0.1530'
|
||||
- name: Cache All The Things
|
||||
uses: actions/cache@v4
|
||||
with:
|
||||
|
|
@ -34,7 +34,7 @@ jobs:
|
|||
env:
|
||||
MYSQL_ROOT_PASSWORD: testing
|
||||
- name: Run MariaDB Tests
|
||||
run: clojure -X:test
|
||||
run: clojure -M:test:runner
|
||||
env:
|
||||
MYSQL_ROOT_PASSWORD: testing
|
||||
NEXT_JDBC_TEST_MYSQL: yes
|
||||
|
|
@ -44,6 +44,7 @@ jobs:
|
|||
env:
|
||||
MYSQL_ROOT_PASSWORD: testing
|
||||
NEXT_JDBC_TEST_MYSQL: yes
|
||||
NEXT_JDBC_TEST_XTDB: yes
|
||||
NEXT_JDBC_TEST_MSSQL: yes
|
||||
MSSQL_SA_PASSWORD: Str0ngP4ssw0rd
|
||||
- name: Deploy Release
|
||||
|
|
|
|||
12
.github/workflows/test-and-snapshot.yml
vendored
12
.github/workflows/test-and-snapshot.yml
vendored
|
|
@ -13,11 +13,11 @@ jobs:
|
|||
- uses: actions/setup-java@v4
|
||||
with:
|
||||
distribution: 'temurin'
|
||||
java-version: '11'
|
||||
java-version: '21'
|
||||
- name: Setup Clojure
|
||||
uses: DeLaGuardo/setup-clojure@master
|
||||
with:
|
||||
cli: '1.12.0.1479'
|
||||
cli: '1.12.0.1530'
|
||||
- name: Cache All The Things
|
||||
uses: actions/cache@v4
|
||||
with:
|
||||
|
|
@ -32,7 +32,7 @@ jobs:
|
|||
env:
|
||||
MYSQL_ROOT_PASSWORD: testing
|
||||
- name: Run MariaDB Tests
|
||||
run: clojure -X:test
|
||||
run: clojure -M:test:runner
|
||||
env:
|
||||
MYSQL_ROOT_PASSWORD: testing
|
||||
NEXT_JDBC_TEST_MYSQL: yes
|
||||
|
|
@ -42,6 +42,7 @@ jobs:
|
|||
env:
|
||||
MYSQL_ROOT_PASSWORD: testing
|
||||
NEXT_JDBC_TEST_MYSQL: yes
|
||||
NEXT_JDBC_TEST_XTDB: yes
|
||||
NEXT_JDBC_TEST_MSSQL: yes
|
||||
MSSQL_SA_PASSWORD: Str0ngP4ssw0rd
|
||||
- name: Deploy Snapshot
|
||||
|
|
@ -64,15 +65,14 @@ jobs:
|
|||
- name: Setup Clojure
|
||||
uses: DeLaGuardo/setup-clojure@master
|
||||
with:
|
||||
cli: '1.12.0.1479'
|
||||
cli: '1.12.0.1530'
|
||||
- name: Cache All The Things
|
||||
uses: actions/cache@v4
|
||||
with:
|
||||
path: |
|
||||
~/.m2/repository
|
||||
~/.gitlibs
|
||||
~/.clojure
|
||||
~/.cpcache
|
||||
key: ${{ runner.os }}-${{ hashFiles('**/deps.edn') }}
|
||||
- name: Run Tests
|
||||
run: clojure -T:build test
|
||||
run: clojure -T:build:jdk${{ matrix.java }} test
|
||||
|
|
|
|||
7
.github/workflows/test.yml
vendored
7
.github/workflows/test.yml
vendored
|
|
@ -17,7 +17,7 @@ jobs:
|
|||
- name: Setup Clojure
|
||||
uses: DeLaGuardo/setup-clojure@master
|
||||
with:
|
||||
cli: '1.12.0.1479'
|
||||
cli: '1.12.0.1530'
|
||||
- name: Cache All The Things
|
||||
uses: actions/cache@v4
|
||||
with:
|
||||
|
|
@ -32,15 +32,16 @@ jobs:
|
|||
env:
|
||||
MYSQL_ROOT_PASSWORD: testing
|
||||
- name: Run MariaDB Tests
|
||||
run: clojure -X:test
|
||||
run: clojure -M:test:runner
|
||||
env:
|
||||
MYSQL_ROOT_PASSWORD: testing
|
||||
NEXT_JDBC_TEST_MYSQL: yes
|
||||
NEXT_JDBC_TEST_MARIADB: yes
|
||||
- name: Run All Tests
|
||||
run: clojure -X:test
|
||||
run: clojure -M:test:runner:jdk${{ matrix.java }}
|
||||
env:
|
||||
MYSQL_ROOT_PASSWORD: testing
|
||||
NEXT_JDBC_TEST_MYSQL: yes
|
||||
NEXT_JDBC_TEST_XTDB: yes
|
||||
NEXT_JDBC_TEST_MSSQL: yes
|
||||
MSSQL_SA_PASSWORD: Str0ngP4ssw0rd
|
||||
|
|
|
|||
9
.gitignore
vendored
9
.gitignore
vendored
|
|
@ -1,11 +1,8 @@
|
|||
*.class
|
||||
*.jar
|
||||
*.swp
|
||||
*~
|
||||
.calva/output-window/
|
||||
.calva/repl.calva-repl
|
||||
.classpath
|
||||
.clj-kondo/.cache
|
||||
.clj-kondo/.lock
|
||||
.cpcache
|
||||
.eastwood
|
||||
.factorypath
|
||||
|
|
@ -23,6 +20,10 @@
|
|||
.settings
|
||||
.socket-repl-port
|
||||
.sw*
|
||||
*.class
|
||||
*.jar
|
||||
*.swp
|
||||
*~
|
||||
/checkouts
|
||||
/classes
|
||||
/clojure_test_*
|
||||
|
|
|
|||
|
|
@ -1,3 +0,0 @@
|
|||
FROM gitpod/workspace-full
|
||||
|
||||
RUN brew install clojure/tools/clojure@1.12.0.1479
|
||||
23
.gitpod.yml
23
.gitpod.yml
|
|
@ -1,23 +0,0 @@
|
|||
image:
|
||||
file: .gitpod.dockerfile
|
||||
|
||||
vscode:
|
||||
extensions:
|
||||
- betterthantomorrow.calva
|
||||
- mauricioszabo.clover
|
||||
|
||||
tasks:
|
||||
- name: Prepare deps/clover
|
||||
init: |
|
||||
clojure -A:test -P
|
||||
echo 50505 > .socket-repl-port
|
||||
mkdir ~/.config/clover
|
||||
cp .clover/config.cljs ~/.config/clover/
|
||||
- name: Start REPL
|
||||
command: clojure -J-Dclojure.server.repl="{:address \"0.0.0.0\" :port 50505 :accept clojure.core.server/repl}" -A:test
|
||||
- name: See Changes
|
||||
command: code CHANGELOG.md
|
||||
|
||||
github:
|
||||
prebuilds:
|
||||
develop: true
|
||||
6
.joker
6
.joker
|
|
@ -1,6 +0,0 @@
|
|||
{:known-macros [next.jdbc/with-transaction]
|
||||
:ignored-unused-namespaces [next.jdbc.connection
|
||||
next.jdbc.date-time
|
||||
next.jdbc.prepare
|
||||
next.jdbc.result-set
|
||||
next.jdbc.transaction]}
|
||||
21
CHANGELOG.md
21
CHANGELOG.md
|
|
@ -2,6 +2,27 @@
|
|||
|
||||
Only accretive/fixative changes will be made from now on.
|
||||
|
||||
* 1.3.next in progress
|
||||
* Fix handling of `false` in `clob-column-reader` [#299](https://github.com/seancorfield/next-jdbc/issues/299) via PR [#300](https://github.com/seancorfield/next-jdbc/pull/300) from [@GAumala](https://github.com/GAumala)
|
||||
* Switch tests to LazyTest via PR [#297](https://github.com/seancorfield/next-jdbc/pull/297).
|
||||
* Update dev/test/build deps.
|
||||
|
||||
* 1.3.1002 -- 2025-03-06
|
||||
* Address [#296](https://github.com/seancorfield/next-jdbc/issues/296) by adding an explicit check (and `throw`) for `sql-params` in `next.jdbc` functions.
|
||||
* Address [#295](https://github.com/seancorfield/next-jdbc/issues/295) by providing a way to tell `next.jdbc` that certain options should be passed "as-is" in the `Properties` object when creating a `Connection` -- `:next.jdbc/as-is-properties` accepts a sequence (or set) of keywords, identifying properties that should not be converted to strings.
|
||||
* Fix [#181](https://github.com/seancorfield/next-jdbc/issues/181) (again!) by adding `Wrapped` protocol as a way for `DefaultOptions` and `SQLLogging` to consistently expose the underlying connectable, even when nested.
|
||||
|
||||
* 1.3.994 -- 2025-01-28
|
||||
* Fix [#293](https://github.com/seancorfield/next-jdbc/issues/293) by no longer `locking` on the `Connection` retrieved from a `DataSource`.
|
||||
* Fix documentation examples of `execute-batch!` via PR [#292](https://github.com/seancorfield/next-jdbc/pull/292) from [@devurandom](https://github.com/devurandom).
|
||||
* Update `java.data` to 1.3.113.
|
||||
* Beef up bit/boolean tests and enable them for XTDB.
|
||||
|
||||
* 1.3.981 -- 2024-12-13
|
||||
* Address [#291](https://github.com/seancorfield/next-jdbc/issues/291) by adding an XTDB section to **Tips & Tricks**.
|
||||
* Added XTDB as a supported database for testing via PR [#290](https://github.com/seancorfield/next-jdbc/pull/290). _Note: not all features are tested against XTDB due to several fundamental differences in architecture, mostly around primary key/generated keys and lack of DDL operations (since XTDB is schemaless)._
|
||||
* Update dev/test dependencies.
|
||||
|
||||
* 1.3.967 -- 2024-12-02
|
||||
* Address [#288](https://github.com/seancorfield/next-jdbc/issues/288) by adding speculative support for `:dbtype "xtdb"`.
|
||||
* Fix [#287](https://github.com/seancorfield/next-jdbc/issues/287) by merging user-supplied options over `:return-keys true`.
|
||||
|
|
|
|||
11
README.md
11
README.md
|
|
@ -8,10 +8,11 @@ The next generation of `clojure.java.jdbc`: a new low-level Clojure wrapper for
|
|||
|
||||
The latest versions on Clojars and on cljdoc:
|
||||
|
||||
[](https://clojars.org/com.github.seancorfield/next.jdbc)
|
||||
[](https://cljdoc.org/d/com.github.seancorfield/next.jdbc/CURRENT)
|
||||
[](https://clojars.org/com.github.seancorfield/next.jdbc)
|
||||
[](https://cljdoc.org/d/com.github.seancorfield/next.jdbc/CURRENT)
|
||||
[](https://clojurians.slack.com/app_redirect?channel=sql)
|
||||
[](http://clojurians.net)
|
||||
[](https://clojurians.zulipchat.com/#narrow/channel/152063-sql)
|
||||
|
||||
The documentation on [cljdoc.org](https://cljdoc.org/d/com.github.seancorfield/next.jdbc/CURRENT) is for the current version of `next.jdbc`:
|
||||
|
||||
|
|
@ -20,11 +21,11 @@ The documentation on [cljdoc.org](https://cljdoc.org/d/com.github.seancorfield/n
|
|||
* [Migrating from `clojure.java.jdbc`](https://cljdoc.org/d/com.github.seancorfield/next.jdbc/CURRENT/doc/migration-from-clojure-java-jdbc)
|
||||
* Feedback via [issues](https://github.com/seancorfield/next-jdbc/issues) or in the [`#sql` channel on the Clojurians Slack](https://clojurians.slack.com/messages/C1Q164V29/) or the [`#sql` stream on the Clojurians Zulip](https://clojurians.zulipchat.com/#narrow/stream/152063-sql).
|
||||
|
||||
The documentation on GitHub is for **develop** since the 1.3.967 release -- [see the CHANGELOG](https://github.com/seancorfield/next-jdbc/blob/develop/CHANGELOG.md) and then read the [corresponding updated documentation](https://github.com/seancorfield/next-jdbc/tree/develop/doc) on GitHub if you want. Older versions of `next.jdbc` were published under the `seancorfield` group ID and you can find [older seancorfield/next.jdbc documentation on cljdoc.org](https://cljdoc.org/versions/seancorfield/next.jdbc).
|
||||
The documentation on GitHub is for **develop** since the 1.3.1002 release -- [see the CHANGELOG](https://github.com/seancorfield/next-jdbc/blob/develop/CHANGELOG.md) and then read the [corresponding updated documentation](https://github.com/seancorfield/next-jdbc/tree/develop/doc) on GitHub if you want. Older versions of `next.jdbc` were published under the `seancorfield` group ID and you can find [older seancorfield/next.jdbc documentation on cljdoc.org](https://cljdoc.org/versions/seancorfield/next.jdbc).
|
||||
|
||||
This project follows the version scheme MAJOR.MINOR.COMMITS where MAJOR and MINOR provide some relative indication of the size of the change, but do not follow semantic versioning. In general, all changes endeavor to be non-breaking (by moving to new names rather than by breaking existing names). COMMITS is an ever-increasing counter of commits since the beginning of this repository.
|
||||
|
||||
> Note: every commit to the **develop** branch runs CI (GitHub Actions) and successful runs push a MAJOR.MINOR.999-SNAPSHOT build to Clojars so the very latest version of `next.jdbc` is always available either via that [snapshot on Clojars](https://clojars.org/com.github.seancorfield/next.jdbc) or via a git dependency on the latest SHA.
|
||||
> Note: every commit to the **develop** branch runs CI (GitHub Actions) and successful runs push a MAJOR.MINOR.9999-SNAPSHOT build to Clojars so the very latest version of `next.jdbc` is always available either via that [snapshot on Clojars](https://clojars.org/com.github.seancorfield/next.jdbc) or via a git dependency on the latest SHA.
|
||||
|
||||
## Motivation
|
||||
|
||||
|
|
@ -50,7 +51,7 @@ From a `DataSource`, either you or `next.jdbc` can create a `java.sql.Connection
|
|||
|
||||
The primary SQL execution API in `next.jdbc` is:
|
||||
* `plan` -- yields an `IReduceInit` that, when reduced with an initial value, executes the SQL statement and then reduces over the `ResultSet` with as little overhead as possible.
|
||||
* `execute!` -- executes the SQL statement and produces a vector of realized hash maps, that use qualified keywords for the column names, of the form `:<table>/<column>`. If you join across multiple tables, the qualified keywords will reflect the originating tables for each of the columns. If the SQL produces named values that do not come from an associated table, a simple, unqualified keyword will be used. The realized hash maps returned by `execute!` are `Datafiable` and thus `Navigable` (see Clojure 1.10's `datafy` and `nav` functions, and tools like [Portal](https://github.com/djblue/portal), [Reveal](https://github.com/vlaaad/reveal), and Cognitect's REBL). Alternatively, you can specify `{:builder-fn rs/as-arrays}` and produce a vector with column names followed by vectors of row values. `rs/as-maps` is the default for `:builder-fn` but there are also `rs/as-unqualified-maps` and `rs/as-unqualified-arrays` if you want unqualified `:<column>` column names (and there are also lower-case variants of all of these).
|
||||
* `execute!` -- executes the SQL statement and produces a vector of realized hash maps, that use qualified keywords for the column names, of the form `:<table>/<column>`. If you join across multiple tables, the qualified keywords will reflect the originating tables for each of the columns. If the SQL produces named values that do not come from an associated table, a simple, unqualified keyword will be used. The realized hash maps returned by `execute!` are `Datafiable` and thus `Navigable` (see Clojure 1.10's `datafy` and `nav` functions, and tools like [Portal](https://github.com/djblue/portal), [Reveal](https://github.com/vlaaad/reveal), and Nubank's Morse -- formerly Cognitect's REBL). Alternatively, you can specify `{:builder-fn rs/as-arrays}` and produce a vector with column names followed by vectors of row values. `rs/as-maps` is the default for `:builder-fn` but there are also `rs/as-unqualified-maps` and `rs/as-unqualified-arrays` if you want unqualified `:<column>` column names (and there are also lower-case variants of all of these).
|
||||
* `execute-one!` -- executes the SQL or DDL statement and produces a single realized hash map. The realized hash map returned by `execute-one!` is `Datafiable` and thus `Navigable`.
|
||||
|
||||
In addition, there are API functions to create `PreparedStatement`s (`prepare`) from `Connection`s, which can be passed to `plan`, `execute!`, or `execute-one!`, and to run code inside a transaction (the `transact` function and the `with-transaction` macro).
|
||||
|
|
|
|||
14
build.clj
14
build.clj
|
|
@ -5,29 +5,33 @@
|
|||
clojure -T:build deploy
|
||||
|
||||
Run tests via:
|
||||
clojure -X:test
|
||||
clojure -M:test:runner
|
||||
|
||||
For more information, run:
|
||||
|
||||
clojure -A:deps -T:build help/doc"
|
||||
(:refer-clojure :exclude [test])
|
||||
(:require [clojure.tools.build.api :as b]
|
||||
[deps-deploy.deps-deploy :as dd]))
|
||||
[deps-deploy.deps-deploy :as dd]
|
||||
[clojure.string :as str]))
|
||||
|
||||
(def lib 'com.github.seancorfield/next.jdbc)
|
||||
(defn- the-version [patch] (format "1.3.%s" patch))
|
||||
(def version (the-version (b/git-count-revs nil)))
|
||||
(def snapshot (the-version "999-SNAPSHOT"))
|
||||
(def snapshot (the-version "9999-SNAPSHOT"))
|
||||
(def class-dir "target/classes")
|
||||
|
||||
(defn test "Run all the tests." [opts]
|
||||
(doseq [alias [:1.10 :1.11 :1.12]]
|
||||
(println "\nRunning tests for Clojure" (name alias))
|
||||
(let [basis (b/create-basis {:aliases [:test alias]})
|
||||
(let [basis (b/create-basis
|
||||
{:aliases (cond-> [:test alias]
|
||||
(str/starts-with? (System/getProperty "java.version") "21")
|
||||
(conj :jdk21))})
|
||||
cmds (b/java-command
|
||||
{:basis basis
|
||||
:main 'clojure.main
|
||||
:main-args ["-m" "cognitect.test-runner"]})
|
||||
:main-args ["-m" "lazytest.main"]})
|
||||
{:keys [exit]} (b/process cmds)]
|
||||
(when-not (zero? exit) (throw (ex-info "Tests failed" {})))))
|
||||
opts)
|
||||
|
|
|
|||
55
deps.edn
55
deps.edn
|
|
@ -1,12 +1,13 @@
|
|||
{:mvn/repos {"sonatype" {:url "https://oss.sonatype.org/content/repositories/snapshots/"}}
|
||||
{:mvn/repos {"sonatype" {:url "https://oss.sonatype.org/content/repositories/snapshots/"}
|
||||
"ossrh-snapshots" {:url "https://s01.oss.sonatype.org/content/repositories/snapshots"}}
|
||||
:paths ["src" "resources"]
|
||||
:deps {org.clojure/clojure {:mvn/version "1.10.3"}
|
||||
org.clojure/java.data {:mvn/version "1.2.107"}
|
||||
org.clojure/java.data {:mvn/version "1.3.113"}
|
||||
|
||||
camel-snake-kebab/camel-snake-kebab {:mvn/version "0.4.3"}}
|
||||
:aliases
|
||||
{;; for help: clojure -A:deps -T:build help/doc
|
||||
:build {:deps {io.github.clojure/tools.build {:mvn/version "0.10.5"}
|
||||
:build {:deps {io.github.clojure/tools.build {:mvn/version "0.10.7"}
|
||||
slipset/deps-deploy {:mvn/version "0.2.2"}}
|
||||
:ns-default build}
|
||||
|
||||
|
|
@ -16,36 +17,42 @@
|
|||
:1.12 {:override-deps {org.clojure/clojure {:mvn/version "1.12.0"}}}
|
||||
|
||||
;; running tests/checks of various kinds:
|
||||
:test {:extra-paths ["test"] ; can also run clojure -X:test
|
||||
:test {:extra-paths ["test"]
|
||||
:extra-deps {org.clojure/test.check {:mvn/version "1.1.1"}
|
||||
io.github.cognitect-labs/test-runner
|
||||
{:git/tag "v0.5.1" :git/sha "dfb30dd"}
|
||||
io.github.noahtheduke/lazytest {:mvn/version "1.6.1"}
|
||||
;; connection pooling
|
||||
com.zaxxer/HikariCP {:mvn/version "6.0.0"}
|
||||
com.zaxxer/HikariCP {:mvn/version "6.3.0"}
|
||||
com.mchange/c3p0 {:mvn/version "0.10.1"}
|
||||
;; JDBC drivers
|
||||
;; 10.16.x is JDK17+
|
||||
org.apache.derby/derby {:mvn/version "10.15.2.0"}
|
||||
org.apache.derby/derbyshared {:mvn/version "10.15.2.0"}
|
||||
org.hsqldb/hsqldb {:mvn/version "2.7.3"}
|
||||
org.hsqldb/hsqldb {:mvn/version "2.7.4"}
|
||||
com.h2database/h2 {:mvn/version "2.3.232"}
|
||||
net.sourceforge.jtds/jtds {:mvn/version "1.3.1"}
|
||||
org.mariadb.jdbc/mariadb-java-client {:mvn/version "3.4.1"}
|
||||
com.mysql/mysql-connector-j {:mvn/version "9.0.0"}
|
||||
org.mariadb.jdbc/mariadb-java-client {:mvn/version "3.5.2"}
|
||||
com.mysql/mysql-connector-j {:mvn/version "9.2.0"}
|
||||
;; 42.7.4 changes update count (to -1) for stored procs:
|
||||
org.postgresql/postgresql {:mvn/version "42.7.4"}
|
||||
io.zonky.test/embedded-postgres {:mvn/version "2.0.7"}
|
||||
io.zonky.test.postgres/embedded-postgres-binaries-darwin-amd64 {:mvn/version "17.0.0"}
|
||||
io.zonky.test.postgres/embedded-postgres-binaries-linux-amd64 {:mvn/version "17.0.0"}
|
||||
io.zonky.test.postgres/embedded-postgres-binaries-windows-amd64 {:mvn/version "17.0.0"}
|
||||
org.xerial/sqlite-jdbc {:mvn/version "3.46.1.3"}
|
||||
com.microsoft.sqlserver/mssql-jdbc {:mvn/version "12.8.1.jre11"}
|
||||
org.postgresql/postgresql {:mvn/version "42.7.5"}
|
||||
io.zonky.test/embedded-postgres {:mvn/version "2.1.0"}
|
||||
io.zonky.test.postgres/embedded-postgres-binaries-darwin-amd64 {:mvn/version "17.4.0"}
|
||||
io.zonky.test.postgres/embedded-postgres-binaries-linux-amd64 {:mvn/version "17.4.0"}
|
||||
io.zonky.test.postgres/embedded-postgres-binaries-windows-amd64 {:mvn/version "17.4.0"}
|
||||
org.xerial/sqlite-jdbc {:mvn/version "3.49.1.0"}
|
||||
com.microsoft.sqlserver/mssql-jdbc {:mvn/version "12.10.0.jre11"}
|
||||
;; use log4j2 to reduce log noise during testing:
|
||||
org.apache.logging.log4j/log4j-api {:mvn/version "2.24.0"}
|
||||
org.apache.logging.log4j/log4j-api {:mvn/version "2.24.3"}
|
||||
;; bridge everything into log4j:
|
||||
org.apache.logging.log4j/log4j-1.2-api {:mvn/version "2.24.0"}
|
||||
org.apache.logging.log4j/log4j-jcl {:mvn/version "2.24.0"}
|
||||
org.apache.logging.log4j/log4j-jul {:mvn/version "2.24.0"}
|
||||
org.apache.logging.log4j/log4j-slf4j-impl {:mvn/version "2.24.0"}}
|
||||
:jvm-opts ["-Dlog4j2.configurationFile=log4j2-info.properties"]
|
||||
:exec-fn cognitect.test-runner.api/test}}}
|
||||
org.apache.logging.log4j/log4j-1.2-api {:mvn/version "2.24.3"}
|
||||
org.apache.logging.log4j/log4j-jcl {:mvn/version "2.24.3"}
|
||||
org.apache.logging.log4j/log4j-jul {:mvn/version "2.24.3"}
|
||||
org.apache.logging.log4j/log4j-slf4j-impl {:mvn/version "2.24.3"}
|
||||
org.apache.logging.log4j/log4j-slf4j2-impl {:mvn/version "2.24.3"}}
|
||||
:jvm-opts ["-Dlog4j2.configurationFile=log4j2-info.properties"]}
|
||||
:runner {:main-opts ["-m" "lazytest.main"]}
|
||||
:jdk11 {}
|
||||
:jdk17 {}
|
||||
:jdk21 {:extra-deps {;; only need the XTDB JDBC module:
|
||||
com.xtdb/xtdb-jdbc {:mvn/version "2.0.0-beta7"}}}
|
||||
:jdk24 {:jvm-opts [;; for SQLite on JDK 24 locally
|
||||
"--enable-native-access=ALL-UNNAMED"]}}}
|
||||
|
|
|
|||
|
|
@ -33,6 +33,9 @@ Any path that calls `get-connection` will accept the following options:
|
|||
|
||||
If you need additional options set on a connection, you can either use Java interop to set them directly, or provide them as part of the "db spec" hash map passed to `get-datasource` (although then they will apply to _all_ connections obtained from that datasource).
|
||||
|
||||
Additional options passed are set as `java.util.Properties` and, by default, are coerced to strings.
|
||||
If you are working with a driver that requires a non-string value for a property (such as the Snowflake driver), you can provide a `:next.jdbc/as-is-properties` option containing a sequence of options that should be added as-is, rather than coerced to strings.
|
||||
|
||||
> Note: If `plan`, `execute!`, or `execute-one!` are passed a `DataSource`, a "db spec" hash map, or a JDBC URL string, they will call `get-connection`, so they will accept the above options in those cases.
|
||||
|
||||
## Generating SQL
|
||||
|
|
|
|||
|
|
@ -117,8 +117,8 @@ will use `execute-batch!` under the hood, instead of `execute!`, as follows:
|
|||
{:batch true})
|
||||
;; equivalent to
|
||||
(jdbc/execute-batch! ds
|
||||
["INSERT INTO address (name,email) VALUES (?,?)"
|
||||
["Stella" "stella@artois.beer"]
|
||||
"INSERT INTO address (name,email) VALUES (?,?)"
|
||||
[["Stella" "stella@artois.beer"]
|
||||
["Waldo" "waldo@lagunitas.beer"]
|
||||
["Aunt Sally" "sour@lagunitas.beer"]]
|
||||
{:return-keys true :return-generated-keys true})
|
||||
|
|
@ -131,8 +131,8 @@ will use `execute-batch!` under the hood, instead of `execute!`, as follows:
|
|||
{:batch true})
|
||||
;; equivalent to
|
||||
(jdbc/execute-batch! ds
|
||||
["INSERT INTO address (name,email) VALUES (?,?)"
|
||||
["Stella" "stella@artois.beer"]
|
||||
"INSERT INTO address (name,email) VALUES (?,?)"
|
||||
[["Stella" "stella@artois.beer"]
|
||||
["Waldo" "waldo@lagunitas.beer"]
|
||||
["Aunt Sally" "sour@lagunitas.beer"]]
|
||||
{:return-keys true :return-generated-keys true})
|
||||
|
|
|
|||
|
|
@ -11,21 +11,21 @@ It is designed to work with Clojure 1.10 or later, supports `datafy`/`nav`, and
|
|||
You can add `next.jdbc` to your project with either:
|
||||
|
||||
```clojure
|
||||
com.github.seancorfield/next.jdbc {:mvn/version "1.3.967"}
|
||||
com.github.seancorfield/next.jdbc {:mvn/version "1.3.1002"}
|
||||
```
|
||||
for `deps.edn` or:
|
||||
|
||||
```clojure
|
||||
[com.github.seancorfield/next.jdbc "1.3.967"]
|
||||
[com.github.seancorfield/next.jdbc "1.3.1002"]
|
||||
```
|
||||
for `project.clj` or `build.boot`.
|
||||
|
||||
**In addition, you will need to add dependencies for the JDBC drivers you wish to use for whatever databases you are using. For example:**
|
||||
|
||||
* MySQL: `com.mysql/mysql-connector-j {:mvn/version "8.1.0"}` ([search for latest version](https://search.maven.org/artifact/com.mysql/mysql-connector-j))
|
||||
* PostgreSQL: `org.postgresql/postgresql {:mvn/version "42.6.0"}` ([search for latest version](https://search.maven.org/artifact/org.postgresql/postgresql))
|
||||
* Microsoft SQL Server: `com.microsoft.sqlserver/mssql-jdbc {:mvn/version "12.4.1.jre11"}` ([search for latest version](https://search.maven.org/artifact/com.microsoft.sqlserver/mssql-jdbc))
|
||||
* Sqlite: `org.xerial/sqlite-jdbc {:mvn/version "3.43.0.0"}` ([search for latest version](https://search.maven.org/artifact/org.xerial/sqlite-jdbc))
|
||||
* MySQL: `com.mysql/mysql-connector-j {:mvn/version "9.1.0"}` ([search for latest version](https://search.maven.org/artifact/com.mysql/mysql-connector-j))
|
||||
* PostgreSQL: `org.postgresql/postgresql {:mvn/version "42.7.4"}` ([search for latest version](https://search.maven.org/artifact/org.postgresql/postgresql))
|
||||
* Microsoft SQL Server: `com.microsoft.sqlserver/mssql-jdbc {:mvn/version "12.8.1.jre11"}` ([search for latest version](https://search.maven.org/artifact/com.microsoft.sqlserver/mssql-jdbc))
|
||||
* Sqlite: `org.xerial/sqlite-jdbc {:mvn/version "3.47.1.0"}` ([search for latest version](https://search.maven.org/artifact/org.xerial/sqlite-jdbc))
|
||||
|
||||
> Note: these are the versions that `next.jdbc` is tested against but there may be more recent versions and those should generally work too -- click the "search for latest version" link to see all available versions of those drivers on Maven Central. You can see the full list of drivers and versions that `next.jdbc` is tested against in [the project's `deps.edn` file](https://github.com/seancorfield/next-jdbc/blob/develop/deps.edn#L10-L27), but many other JDBC drivers for other databases should also work (e.g., Oracle, Red Shift).
|
||||
|
||||
|
|
@ -38,8 +38,8 @@ For the examples in this documentation, we will use a local H2 database on disk,
|
|||
```clojure
|
||||
;; deps.edn
|
||||
{:deps {org.clojure/clojure {:mvn/version "1.12.0"}
|
||||
com.github.seancorfield/next.jdbc {:mvn/version "1.3.967"}
|
||||
com.h2database/h2 {:mvn/version "2.2.224"}}}
|
||||
com.github.seancorfield/next.jdbc {:mvn/version "1.3.1002"}
|
||||
com.h2database/h2 {:mvn/version "2.3.232"}}}
|
||||
```
|
||||
|
||||
### Create & Populate a Database
|
||||
|
|
@ -487,9 +487,9 @@ Not all databases support using a `PreparedStatement` for every type of SQL oper
|
|||
First, you need to add the connection pooling library as a dependency, e.g.,
|
||||
|
||||
```clojure
|
||||
com.zaxxer/HikariCP {:mvn/version "5.0.1"}
|
||||
com.zaxxer/HikariCP {:mvn/version "6.2.1"}
|
||||
;; or:
|
||||
com.mchange/c3p0 {:mvn/version "0.9.5.5"}
|
||||
com.mchange/c3p0 {:mvn/version "0.10.1"}
|
||||
```
|
||||
|
||||
_Check those libraries' documentation for the latest version to use!_
|
||||
|
|
|
|||
|
|
@ -55,7 +55,7 @@ be very database-specific. Some database drivers **don't** use the hierarchy
|
|||
above -- notably PostgreSQL, which has a generic `PSQLException` type
|
||||
with its own subclasses and semantics. See [PostgreSQL JDBC issue #963](https://github.com/pgjdbc/pgjdbc/issues/963)
|
||||
for a discussion of the difficulty in adopting the standard JDBC hierarchy
|
||||
(dating back five years).
|
||||
(dating back to 2017!).
|
||||
|
||||
The `java.sql.SQLException` class provides `.getErrorCode()` and
|
||||
`.getSQLState()` methods but the values returned by those are
|
||||
|
|
@ -237,6 +237,7 @@ common approach, there is also a non-JDBC Clojure/Java driver for PostgreSQL cal
|
|||
quite a bit faster than using JDBC.
|
||||
|
||||
When you use `:return-keys true` with `execute!` or `execute-one!` (or you use `insert!`), PostgreSQL returns the entire inserted row (unlike nearly every other database that just returns any generated keys!).
|
||||
_[It seems to achieve this by the equivalent of automatically appending `RETURNING *` to your SQL, if necessary.]_
|
||||
|
||||
The default result set builder for `next.jdbc` is `as-qualified-maps` which
|
||||
uses the `.getTableName()` method on `ResultSetMetaData` to qualify the
|
||||
|
|
@ -574,3 +575,26 @@ If you are using `plan`, you'll most likely be accessing columns by just the lab
|
|||
|
||||
See also [`datafy`, `nav`, and `:schema` > **SQLite**](/doc/datafy-nav-and-schema.md#sqlite)
|
||||
for additional caveats on the `next.jdbc.datafy` namespace when using SQLite.
|
||||
|
||||
## XTDB
|
||||
|
||||
XTDB is a bitemporal, schemaless, document-oriented database that presents
|
||||
itself as a PostgreSQL-compatible database, in terms of JDBC. It has a number
|
||||
of SQL extensions, and some differences from common JDBC behavior. See
|
||||
its documentation for details:
|
||||
* [SQL Overview](https://docs.xtdb.com/quickstart/sql-overview.html)
|
||||
* [SQL Queries](https://docs.xtdb.com/reference/main/sql/queries.html)
|
||||
* [SQL Transactions/DML](https://docs.xtdb.com/reference/main/sql/txs.html)
|
||||
|
||||
`next.jdbc` officially supports XTDB as of 1.3.981 but there are some caveats:
|
||||
* You can use `:dbtype "xtdb"` to identify XTDB as the database type.
|
||||
* You must specify `:dbname "xtdb"` in the db-spec hash map or JDBC URL.
|
||||
* XTDB does not support `.getTableName()` so you always get unqualified column names in result sets.
|
||||
* The primary key on all tables is `_id` and it must be specified in all `INSERT` operations (no auto-generated keys).
|
||||
* That means that `next.jdbc.sql/get-by-id` requires the 5-argument call, so that you can specify the `pk-name` as `:_id` and provide an options map.
|
||||
* If you want to use `next.jdbc`'s built-in `datafy` / `nav` functionality, you need to explicitly specify `:schema-opts {:pk "_id"}` to override the default assumption of `id` as the primary key.
|
||||
* DML operations (`INSERT`, `UPDATE`, and `DELETE`) are essentially asynchronous in XTDB and therefore can not return an accurate `next.jdbc/update-count` (so it is always 0).
|
||||
* `INSERT` operations do not return the inserted row (like PostgreSQL does) nor even the provided `_id` primary key.
|
||||
* That means that the `next.jdbc.defer` namespace functions do not work well with XTDB.
|
||||
* `next.jdbc.sql/insert-multi!` returns an empty vector for XTDB (since `INSERT` operations do not return keys or update counts).
|
||||
* The `next.jdbc.result-set/*-kebab-maps` functions (and associated `next.jdbc/*-kebab-opts` option maps) cause leading `_` to be stripped from column names and cannot be used with XTDB (this is inherent in the underlying library that `next.jdbc` relies on -- you can of course write your own custom result set builder function to handle this).
|
||||
|
|
|
|||
|
|
@ -14,3 +14,8 @@ services:
|
|||
MSSQL_SA_PASSWORD: Str0ngP4ssw0rd
|
||||
ports:
|
||||
- "1433:1433"
|
||||
xtdb:
|
||||
image: ghcr.io/xtdb/xtdb:latest
|
||||
# pull_policy: always
|
||||
ports:
|
||||
- "5432:5432"
|
||||
|
|
|
|||
|
|
@ -5,19 +5,30 @@
|
|||
(defn- run-tests [env v]
|
||||
(when v (println "\nTesting Clojure" v))
|
||||
(let [{:keys [exit]}
|
||||
(p/shell {:extra-env env} "clojure" (str "-X"
|
||||
(when v (str ":" v))
|
||||
":test"))]
|
||||
(p/shell {:extra-env env}
|
||||
"clojure"
|
||||
(str "-M"
|
||||
(when v (str ":" v))
|
||||
":test:runner"
|
||||
;; jdk21+ adds xtdb:
|
||||
(when (System/getenv "NEXT_JDBC_TEST_XTDB")
|
||||
":jdk21")
|
||||
;; to suppress native access warnings on JDK24:
|
||||
":jdk24")
|
||||
"--output" "dots")]
|
||||
(when-not (zero? exit)
|
||||
(System/exit exit))))
|
||||
|
||||
(let [maria? (some #(= "maria" %) *command-line-args*)
|
||||
xtdb? (some #(= "xtdb" %) *command-line-args*)
|
||||
all? (some #(= "all" %) *command-line-args*)
|
||||
env
|
||||
(cond-> {"NEXT_JDBC_TEST_MSSQL" "yes"
|
||||
"NEXT_JDBC_TEST_MYSQL" "yes"
|
||||
"MSSQL_SA_PASSWORD" "Str0ngP4ssw0rd"}
|
||||
maria?
|
||||
(assoc "NEXT_JDBC_TEST_MARIADB" "yes"))]
|
||||
(assoc "NEXT_JDBC_TEST_MARIADB" "yes")
|
||||
xtdb?
|
||||
(assoc "NEXT_JDBC_TEST_XTDB" "yes"))]
|
||||
(doseq [v (if all? ["1.10" "1.11" "1.12"] [nil])]
|
||||
(run-tests env v)))
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2018-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2018-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc
|
||||
"The public API of the next generation java.jdbc library.
|
||||
|
|
@ -177,6 +177,14 @@
|
|||
[spec user password opts]
|
||||
(p/get-connection spec (assoc opts :user user :password password))))
|
||||
|
||||
(defn- ensure-sql-params [sql-params]
|
||||
(when-not (or (nil? sql-params)
|
||||
(and (seqable? sql-params)
|
||||
(or (empty? sql-params)
|
||||
(string? (first sql-params)))))
|
||||
(throw (ex-info "sql-params should be a vector containing a SQL string and any parameters"
|
||||
{:sql-params sql-params}))))
|
||||
|
||||
(defn prepare
|
||||
"Given a connection to a database, and a vector containing SQL and any
|
||||
parameters it needs, return a new `PreparedStatement`.
|
||||
|
|
@ -191,11 +199,13 @@
|
|||
See the list of options above (in the namespace docstring) for what can
|
||||
be passed to prepare."
|
||||
(^java.sql.PreparedStatement
|
||||
[connection sql-params]
|
||||
(p/prepare connection sql-params {}))
|
||||
[connection sql-params]
|
||||
(ensure-sql-params sql-params)
|
||||
(p/prepare connection sql-params {}))
|
||||
(^java.sql.PreparedStatement
|
||||
[connection sql-params opts]
|
||||
(p/prepare connection sql-params opts)))
|
||||
[connection sql-params opts]
|
||||
(ensure-sql-params sql-params)
|
||||
(p/prepare connection sql-params opts)))
|
||||
|
||||
(defn plan
|
||||
"General SQL execution function (for working with result sets).
|
||||
|
|
@ -228,16 +238,18 @@
|
|||
(or they can be different, depending on how you want the row to be built,
|
||||
and how you want any subsequent lazy navigation to be handled)."
|
||||
(^clojure.lang.IReduceInit
|
||||
[stmt]
|
||||
(p/-execute stmt [] {}))
|
||||
[stmt]
|
||||
(p/-execute stmt [] {}))
|
||||
(^clojure.lang.IReduceInit
|
||||
[connectable sql-params]
|
||||
(p/-execute connectable sql-params
|
||||
{:next.jdbc/sql-params sql-params}))
|
||||
[connectable sql-params]
|
||||
(ensure-sql-params sql-params)
|
||||
(p/-execute connectable sql-params
|
||||
{:next.jdbc/sql-params sql-params}))
|
||||
(^clojure.lang.IReduceInit
|
||||
[connectable sql-params opts]
|
||||
(p/-execute connectable sql-params
|
||||
(assoc opts :next.jdbc/sql-params sql-params))))
|
||||
[connectable sql-params opts]
|
||||
(ensure-sql-params sql-params)
|
||||
(p/-execute connectable sql-params
|
||||
(assoc opts :next.jdbc/sql-params sql-params))))
|
||||
|
||||
(defn execute!
|
||||
"General SQL execution function.
|
||||
|
|
@ -252,9 +264,11 @@
|
|||
([stmt]
|
||||
(p/-execute-all stmt [] {}))
|
||||
([connectable sql-params]
|
||||
(ensure-sql-params sql-params)
|
||||
(p/-execute-all connectable sql-params
|
||||
{:next.jdbc/sql-params sql-params}))
|
||||
([connectable sql-params opts]
|
||||
(ensure-sql-params sql-params)
|
||||
(p/-execute-all connectable sql-params
|
||||
(assoc opts :next.jdbc/sql-params sql-params))))
|
||||
|
||||
|
|
@ -271,9 +285,11 @@
|
|||
([stmt]
|
||||
(p/-execute-one stmt [] {}))
|
||||
([connectable sql-params]
|
||||
(ensure-sql-params sql-params)
|
||||
(p/-execute-one connectable sql-params
|
||||
{:next.jdbc/sql-params sql-params}))
|
||||
([connectable sql-params opts]
|
||||
(ensure-sql-params sql-params)
|
||||
(p/-execute-one connectable sql-params
|
||||
(assoc opts :next.jdbc/sql-params sql-params))))
|
||||
|
||||
|
|
@ -336,9 +352,9 @@
|
|||
result))))
|
||||
params)))
|
||||
([connectable sql param-groups opts]
|
||||
(if (or (instance? java.sql.Connection connectable)
|
||||
(and (satisfies? p/Connectable connectable)
|
||||
(instance? java.sql.Connection (:connectable connectable))))
|
||||
(when-not (string? sql)
|
||||
(throw (IllegalArgumentException. "execute-batch! requires a SQL string")))
|
||||
(if (instance? java.sql.Connection (p/unwrap connectable))
|
||||
(with-open [ps (prepare connectable [sql] opts)]
|
||||
(execute-batch! ps param-groups opts))
|
||||
(with-open [con (get-connection connectable)]
|
||||
|
|
@ -365,15 +381,12 @@
|
|||
executes the body, and automatically closes it for you."
|
||||
[[sym connectable] & body]
|
||||
(let [con-sym (vary-meta sym assoc :tag 'java.sql.Connection)]
|
||||
`(let [con-obj# ~connectable]
|
||||
(cond (instance? java.sql.Connection con-obj#)
|
||||
((^{:once true} fn* [~con-sym] ~@body) con-obj#)
|
||||
(and (satisfies? p/Connectable con-obj#)
|
||||
(instance? java.sql.Connection (:connectable con-obj#)))
|
||||
((^{:once true} fn* [~con-sym] ~@body) (:connectable con-obj#))
|
||||
:else
|
||||
(with-open [con# (get-connection con-obj#)]
|
||||
((^{:once true} fn* [~con-sym] ~@body) con#))))))
|
||||
`(let [con-obj# ~connectable
|
||||
bare-con# (p/unwrap con-obj#)]
|
||||
(if (instance? java.sql.Connection bare-con#)
|
||||
((^{:once true} fn* [~con-sym] ~@body) bare-con#)
|
||||
(with-open [con# (get-connection con-obj#)]
|
||||
((^{:once true} fn* [~con-sym] ~@body) con#))))))
|
||||
|
||||
(defmacro on-connection+options
|
||||
"Given a connectable object, assumed to be wrapped with options, gets
|
||||
|
|
@ -403,15 +416,11 @@
|
|||
with `on-connection`."
|
||||
[[sym connectable] & body]
|
||||
`(let [con-obj# ~connectable]
|
||||
(cond (instance? java.sql.Connection con-obj#)
|
||||
((^{:once true} fn* [~sym] ~@body) con-obj#)
|
||||
(and (satisfies? p/Connectable con-obj#)
|
||||
(instance? java.sql.Connection (:connectable con-obj#)))
|
||||
((^{:once true} fn* [~sym] ~@body) con-obj#)
|
||||
:else
|
||||
(with-open [con# (get-connection con-obj#)]
|
||||
((^{:once true} fn* [~sym] ~@body)
|
||||
(with-options con# (:options con-obj# {})))))))
|
||||
(if (instance? java.sql.Connection (p/unwrap con-obj#))
|
||||
((^{:once true} fn* [~sym] ~@body) con-obj#)
|
||||
(with-open [con# (get-connection con-obj#)]
|
||||
((^{:once true} fn* [~sym] ~@body)
|
||||
(with-options con# (:options con-obj# {})))))))
|
||||
|
||||
(defn transact
|
||||
"Given a transactable object and a function (taking a `Connection`),
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2018-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2018-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.connection
|
||||
"Standard implementations of `get-datasource` and `get-connection`.
|
||||
|
|
@ -374,9 +374,12 @@
|
|||
(defn- as-properties
|
||||
"Convert any seq of pairs to a `java.util.Properties` instance."
|
||||
^Properties [m]
|
||||
(let [p (Properties.)]
|
||||
(doseq [[k v] m]
|
||||
(.setProperty p (name k) (str v)))
|
||||
(let [p (Properties.)
|
||||
as-is (set (:next.jdbc/as-is-properties m))]
|
||||
(doseq [[k v] (dissoc m :next.jdbc/as-is-properties)]
|
||||
(if (contains? as-is k)
|
||||
(.put p (name k) v)
|
||||
(.setProperty p (name k) (str v))))
|
||||
p))
|
||||
|
||||
(defn uri->db-spec
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2020-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2020-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns ^:no-doc next.jdbc.default-options
|
||||
"Implementation of default options logic."
|
||||
|
|
@ -8,6 +8,10 @@
|
|||
|
||||
(defrecord DefaultOptions [connectable options])
|
||||
|
||||
(extend-protocol p/Wrapped
|
||||
DefaultOptions
|
||||
(unwrap [this] (p/unwrap (:connectable this))))
|
||||
|
||||
(extend-protocol p/Sourceable
|
||||
DefaultOptions
|
||||
(get-datasource [this]
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2018-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2018-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.protocols
|
||||
"This is the extensible core of the next generation java.jdbc library.
|
||||
|
|
@ -63,3 +63,15 @@
|
|||
:extend-via-metadata true
|
||||
(-transact [this body-fn opts]
|
||||
"Run the `body-fn` inside a transaction."))
|
||||
|
||||
(defprotocol Wrapped
|
||||
"Protocol for (un)wrapping a `next.jdbc` connectable.
|
||||
|
||||
Implementations are provided for `Object` (identity) and `DefaultOptions`
|
||||
and SQLLogging."
|
||||
(unwrap [this]
|
||||
"Unwrap the connectable to get the underlying connectable."))
|
||||
|
||||
(extend-protocol Wrapped
|
||||
Object
|
||||
(unwrap [this] this))
|
||||
|
|
|
|||
|
|
@ -278,6 +278,10 @@
|
|||
:qualifier-fn ->kebab-case
|
||||
:label-fn ->kebab-case)))
|
||||
|
||||
(comment
|
||||
(->kebab-case "_id") ;;=> "id"!!
|
||||
)
|
||||
|
||||
(defn as-unqualified-kebab-maps
|
||||
"Given a `ResultSet` and options, return a `RowBuilder` / `ResultSetBuilder`
|
||||
that produces bare vectors of hash map rows, with simple, kebab-case keys."
|
||||
|
|
@ -319,7 +323,7 @@
|
|||
"An example column-reader that still uses `.getObject` but expands CLOB
|
||||
columns into strings."
|
||||
[^ResultSet rs ^ResultSetMetaData _ ^Integer i]
|
||||
(when-let [value (.getObject rs i)]
|
||||
(let [value (.getObject rs i)]
|
||||
(cond-> value
|
||||
(instance? Clob value)
|
||||
(clob->string))))
|
||||
|
|
|
|||
|
|
@ -110,7 +110,7 @@
|
|||
:opts (s/? ::opts-map)))
|
||||
|
||||
(s/fdef jdbc/prepare
|
||||
:args (s/cat :connection ::connection
|
||||
:args (s/cat :connection ::proto-connectable
|
||||
:sql-params ::sql-params
|
||||
:opts (s/? ::opts-map)))
|
||||
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2021-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2021-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns ^:no-doc next.jdbc.sql-logging
|
||||
"Implementation of sql-logging logic."
|
||||
|
|
@ -8,6 +8,10 @@
|
|||
|
||||
(defrecord SQLLogging [connectable sql-logger result-logger options])
|
||||
|
||||
(extend-protocol p/Wrapped
|
||||
SQLLogging
|
||||
(unwrap [this] (p/unwrap (:connectable this))))
|
||||
|
||||
(extend-protocol p/Sourceable
|
||||
SQLLogging
|
||||
(get-datasource [this]
|
||||
|
|
|
|||
|
|
@ -145,7 +145,11 @@
|
|||
javax.sql.DataSource
|
||||
(-transact [this body-fn opts]
|
||||
(with-open [con (p/get-connection this opts)]
|
||||
(p/-transact con body-fn opts)))
|
||||
;; this connection is assumed unique so we do not need the active-tx check:
|
||||
(let [raw (raw-connection con)]
|
||||
;; we don't lock either, per #293:
|
||||
(binding [*active-tx* (conj *active-tx* raw)]
|
||||
(transact* con body-fn opts)))))
|
||||
Object
|
||||
(-transact [this body-fn opts]
|
||||
(p/-transact (p/get-datasource this) body-fn opts)))
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2019-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2019-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.connection-string-test
|
||||
"Tests for the main hash map spec to JDBC URL logic and the get-datasource
|
||||
|
|
@ -7,15 +7,17 @@
|
|||
At some point, the datasource/connection tests should probably be extended
|
||||
to accept EDN specs from an external source (environment variables?)."
|
||||
(:require [clojure.string :as str]
|
||||
[clojure.test :refer [deftest is testing use-fixtures]]
|
||||
[lazytest.core :refer [around set-ns-context!]]
|
||||
[lazytest.experimental.interfaces.clojure-test :refer [deftest is testing]]
|
||||
[next.jdbc.connection :as c]
|
||||
[next.jdbc.protocols :as p]
|
||||
[next.jdbc.specs :as specs]
|
||||
[next.jdbc.test-fixtures :refer [with-test-db db]]))
|
||||
[next.jdbc.test-fixtures :refer [db with-test-db]])
|
||||
(:import [java.util Properties]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
||||
(use-fixtures :once with-test-db)
|
||||
(set-ns-context! [(around [f] (with-test-db f))])
|
||||
|
||||
(specs/instrument)
|
||||
|
||||
|
|
@ -39,3 +41,16 @@
|
|||
(when (and user password)
|
||||
(with-open [con (p/get-connection ds {})]
|
||||
(is (instance? java.sql.Connection con)))))))
|
||||
|
||||
(deftest property-tests
|
||||
(is (string? (.getProperty ^Properties (#'c/as-properties {:foo [42]}) "foo")))
|
||||
(is (string? (.get ^Properties (#'c/as-properties {:foo [42]}) "foo")))
|
||||
(is (vector? (.get ^Properties (#'c/as-properties
|
||||
{:foo [42]
|
||||
:next.jdbc/as-is-properties [:foo]})
|
||||
"foo")))
|
||||
;; because .getProperty drops non-string values!
|
||||
(is (nil? (.getProperty ^Properties (#'c/as-properties
|
||||
{:foo [42]
|
||||
:next.jdbc/as-is-properties [:foo]})
|
||||
"foo"))))
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2019-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2019-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.connection-test
|
||||
"Tests for the main hash map spec to JDBC URL logic and the get-datasource
|
||||
|
|
@ -7,7 +7,7 @@
|
|||
At some point, the datasource/connection tests should probably be extended
|
||||
to accept EDN specs from an external source (environment variables?)."
|
||||
(:require [clojure.string :as str]
|
||||
[clojure.test :refer [deftest is testing]]
|
||||
[lazytest.experimental.interfaces.clojure-test :refer [deftest is testing]]
|
||||
[next.jdbc.connection :as c]
|
||||
[next.jdbc.protocols :as p])
|
||||
(:import (com.zaxxer.hikari HikariDataSource)
|
||||
|
|
|
|||
|
|
@ -1,21 +1,22 @@
|
|||
;; copyright (c) 2020-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2020-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.datafy-test
|
||||
"Tests for the datafy extensions over JDBC types."
|
||||
(:require [clojure.datafy :as d]
|
||||
[clojure.set :as set]
|
||||
[clojure.test :refer [deftest is testing use-fixtures]]
|
||||
[lazytest.core :refer [around set-ns-context!]]
|
||||
[lazytest.experimental.interfaces.clojure-test :refer [deftest is testing]]
|
||||
[next.jdbc :as jdbc]
|
||||
[next.jdbc.datafy]
|
||||
[next.jdbc.result-set :as rs]
|
||||
[next.jdbc.specs :as specs]
|
||||
[next.jdbc.test-fixtures
|
||||
:refer [with-test-db db ds
|
||||
derby? jtds? mysql? postgres? sqlite?]]))
|
||||
:refer [db derby? ds jtds? mysql? postgres? sqlite? with-test-db
|
||||
xtdb?]]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
||||
(use-fixtures :once with-test-db)
|
||||
(set-ns-context! [(around [f] (with-test-db f))])
|
||||
|
||||
(specs/instrument)
|
||||
|
||||
|
|
@ -83,6 +84,26 @@
|
|||
:rowIdLifetime/exception))
|
||||
(postgres?) (-> (disj :rowIdLifetime)
|
||||
(conj :rowIdLifetime/exception))
|
||||
(xtdb?) (-> (disj :clientInfoProperties
|
||||
:defaultTransactionIsolation
|
||||
:maxCatalogNameLength
|
||||
:maxColumnNameLength
|
||||
:maxCursorNameLength
|
||||
:maxProcedureNameLength
|
||||
:maxSchemaNameLength
|
||||
:maxTableNameLength
|
||||
:maxUserNameLength
|
||||
:rowIdLifetime)
|
||||
(conj :clientInfoProperties/exception
|
||||
:defaultTransactionIsolation/exception
|
||||
:maxCatalogNameLength/exception
|
||||
:maxColumnNameLength/exception
|
||||
:maxCursorNameLength/exception
|
||||
:maxProcedureNameLength/exception
|
||||
:maxSchemaNameLength/exception
|
||||
:maxTableNameLength/exception
|
||||
:maxUserNameLength/exception
|
||||
:rowIdLifetime/exception))
|
||||
(sqlite?) (-> (disj :clientInfoProperties :rowIdLifetime)
|
||||
(conj :clientInfoProperties/exception
|
||||
:rowIdLifetime/exception)))
|
||||
|
|
@ -97,7 +118,8 @@
|
|||
(let [data (d/datafy (.getMetaData con))]
|
||||
(doseq [k (cond-> #{:catalogs :clientInfoProperties :schemas :tableTypes :typeInfo}
|
||||
(jtds?) (disj :clientInfoProperties)
|
||||
(sqlite?) (disj :clientInfoProperties))]
|
||||
(sqlite?) (disj :clientInfoProperties)
|
||||
(xtdb?) (disj :clientInfoProperties))]
|
||||
(let [rs (d/nav data k nil)]
|
||||
(is (vector? rs))
|
||||
(is (every? map? rs))))))))
|
||||
|
|
@ -122,4 +144,5 @@
|
|||
(.execute ps)
|
||||
(.getResultSet ps)
|
||||
(.close ps)
|
||||
(.close con))
|
||||
(.close con)
|
||||
)
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2019-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2019-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.date-time-test
|
||||
"Date/time parameter auto-conversion tests.
|
||||
|
|
@ -6,44 +6,45 @@
|
|||
These tests contain no assertions. Without requiring `next.jdbc.date-time`
|
||||
several of the `insert` operations would throw exceptions for some databases
|
||||
so the test here just checks those operations 'succeed'."
|
||||
(:require [clojure.test :refer [deftest is testing use-fixtures]]
|
||||
(:require [lazytest.core :refer [around set-ns-context!]]
|
||||
[lazytest.experimental.interfaces.clojure-test :refer [deftest]]
|
||||
[next.jdbc :as jdbc]
|
||||
[next.jdbc.date-time] ; to extend SettableParameter to date/time
|
||||
[next.jdbc.test-fixtures :refer [with-test-db db ds
|
||||
mssql?]]
|
||||
[next.jdbc.specs :as specs])
|
||||
(:import (java.sql ResultSet)))
|
||||
[next.jdbc.test-fixtures :refer [with-test-db ds
|
||||
mssql? xtdb?]]
|
||||
[next.jdbc.specs :as specs]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
||||
(use-fixtures :once with-test-db)
|
||||
(set-ns-context! [(around [f] (with-test-db f))])
|
||||
|
||||
(specs/instrument)
|
||||
|
||||
(deftest issue-73
|
||||
(try
|
||||
(jdbc/execute-one! (ds) ["drop table fruit_time"])
|
||||
(catch Throwable _))
|
||||
(jdbc/execute-one! (ds) [(str "create table fruit_time (id int not null, deadline "
|
||||
(if (mssql?) "datetime" "timestamp")
|
||||
" not null)")])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 1 (java.util.Date.)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 2 (java.time.Instant/now)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 3 (java.time.LocalDate/now)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 4 (java.time.LocalDateTime/now)])
|
||||
(try
|
||||
(jdbc/execute-one! (ds) ["drop table fruit_time"])
|
||||
(catch Throwable _))
|
||||
(jdbc/execute-one! (ds) ["create table fruit_time (id int not null, deadline time not null)"])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 1 (java.util.Date.)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 2 (java.time.Instant/now)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 3 (java.time.LocalDate/now)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 4 (java.time.LocalDateTime/now)])
|
||||
(try
|
||||
(jdbc/execute-one! (ds) ["drop table fruit_time"])
|
||||
(catch Throwable _))
|
||||
(jdbc/execute-one! (ds) ["create table fruit_time (id int not null, deadline date not null)"])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 1 (java.util.Date.)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 2 (java.time.Instant/now)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 3 (java.time.LocalDate/now)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 4 (java.time.LocalDateTime/now)]))
|
||||
(when-not (xtdb?)
|
||||
(try
|
||||
(jdbc/execute-one! (ds) ["drop table fruit_time"])
|
||||
(catch Throwable _))
|
||||
(jdbc/execute-one! (ds) [(str "create table fruit_time (id int not null, deadline "
|
||||
(if (mssql?) "datetime" "timestamp")
|
||||
" not null)")])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 1 (java.util.Date.)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 2 (java.time.Instant/now)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 3 (java.time.LocalDate/now)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 4 (java.time.LocalDateTime/now)])
|
||||
(try
|
||||
(jdbc/execute-one! (ds) ["drop table fruit_time"])
|
||||
(catch Throwable _))
|
||||
(jdbc/execute-one! (ds) ["create table fruit_time (id int not null, deadline time not null)"])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 1 (java.util.Date.)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 2 (java.time.Instant/now)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 3 (java.time.LocalDate/now)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 4 (java.time.LocalDateTime/now)])
|
||||
(try
|
||||
(jdbc/execute-one! (ds) ["drop table fruit_time"])
|
||||
(catch Throwable _))
|
||||
(jdbc/execute-one! (ds) ["create table fruit_time (id int not null, deadline date not null)"])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 1 (java.util.Date.)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 2 (java.time.Instant/now)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 3 (java.time.LocalDate/now)])
|
||||
(jdbc/execute-one! (ds) ["insert into fruit_time (id, deadline) values (?,?)" 4 (java.time.LocalDateTime/now)])))
|
||||
|
|
|
|||
|
|
@ -1,9 +1,8 @@
|
|||
;; copyright (c) 2020-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2020-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.default-options-test
|
||||
"Stub test namespace for default options. Nothing can really be tested
|
||||
at this level tho'..."
|
||||
(:require [clojure.test :refer [deftest is testing]]
|
||||
[next.jdbc.default-options :refer :all]))
|
||||
(:require [next.jdbc.default-options]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2024-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.defer-test
|
||||
"The idea behind the next.jdbc.defer namespace is to provide a
|
||||
|
|
@ -11,43 +11,45 @@
|
|||
describes a series of SQL operations to be performed, that
|
||||
are held in a dynamic var, and that can be executed at a
|
||||
later time, in a transaction."
|
||||
(:require [clojure.test :refer [deftest is testing use-fixtures]]
|
||||
(:require [lazytest.core :refer [around set-ns-context!]]
|
||||
[lazytest.experimental.interfaces.clojure-test :refer [deftest is testing]]
|
||||
[next.jdbc :as jdbc]
|
||||
[next.jdbc.defer :as sut]
|
||||
[next.jdbc.test-fixtures
|
||||
:refer [ds with-test-db]]))
|
||||
:refer [ds with-test-db xtdb?]]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
||||
(use-fixtures :once with-test-db)
|
||||
(set-ns-context! [(around [f] (with-test-db f))])
|
||||
|
||||
(deftest basic-test
|
||||
(testing "data structures"
|
||||
(is (= [{:sql-p ["INSERT INTO foo (name) VALUES (?)" "Sean"]
|
||||
:key-fn :GENERATED_KEY
|
||||
:key :id
|
||||
:opts {:key-fn :GENERATED_KEY :key :id}}]
|
||||
@(sut/defer-ops
|
||||
#(sut/insert! :foo {:name "Sean"} {:key-fn :GENERATED_KEY :key :id})))))
|
||||
(testing "execution"
|
||||
(let [effects (sut/with-deferred (ds)
|
||||
(sut/insert! :fruit {:name "Mango"} {:key :test}))]
|
||||
(is (= {:test 1} @effects))
|
||||
(is (= 1 (count (jdbc/execute! (ds)
|
||||
["select * from fruit where name = ?"
|
||||
"Mango"])))))
|
||||
(let [effects (sut/with-deferred (ds)
|
||||
(sut/insert! :fruit {:name "Dragonfruit"} {:key :test})
|
||||
(sut/update! :fruit {:cost 123} {:name "Dragonfruit"})
|
||||
(sut/delete! :fruit {:name "Dragonfruit"}))]
|
||||
(is (= {:test 1} @effects))
|
||||
(is (= 0 (count (jdbc/execute! (ds)
|
||||
["select * from fruit where name = ?"
|
||||
"Dragonfruit"])))))
|
||||
(let [effects (sut/with-deferred (ds)
|
||||
(sut/insert! :fruit {:name "Grapefruit" :bad_column 0} {:key :test}))]
|
||||
(is (= :failed (try @effects
|
||||
(catch Exception _ :failed))))
|
||||
(is (= 0 (count (jdbc/execute! (ds)
|
||||
["select * from fruit where name = ?"
|
||||
"Grapefruit"])))))))
|
||||
(when-not (xtdb?)
|
||||
(testing "data structures"
|
||||
(is (= [{:sql-p ["INSERT INTO foo (name) VALUES (?)" "Sean"]
|
||||
:key-fn :GENERATED_KEY
|
||||
:key :id
|
||||
:opts {:key-fn :GENERATED_KEY :key :id}}]
|
||||
@(sut/defer-ops
|
||||
#(sut/insert! :foo {:name "Sean"} {:key-fn :GENERATED_KEY :key :id})))))
|
||||
(testing "execution"
|
||||
(let [effects (sut/with-deferred (ds)
|
||||
(sut/insert! :fruit {:name "Mango"} {:key :test}))]
|
||||
(is (= {:test 1} @effects))
|
||||
(is (= 1 (count (jdbc/execute! (ds)
|
||||
["select * from fruit where name = ?"
|
||||
"Mango"])))))
|
||||
(let [effects (sut/with-deferred (ds)
|
||||
(sut/insert! :fruit {:name "Dragonfruit"} {:key :test})
|
||||
(sut/update! :fruit {:cost 123} {:name "Dragonfruit"})
|
||||
(sut/delete! :fruit {:name "Dragonfruit"}))]
|
||||
(is (= {:test 1} @effects))
|
||||
(is (= 0 (count (jdbc/execute! (ds)
|
||||
["select * from fruit where name = ?"
|
||||
"Dragonfruit"])))))
|
||||
(let [effects (sut/with-deferred (ds)
|
||||
(sut/insert! :fruit {:name "Grapefruit" :bad_column 0} {:key :test}))]
|
||||
(is (= :failed (try @effects
|
||||
(catch Exception _ :failed))))
|
||||
(is (= 0 (count (jdbc/execute! (ds)
|
||||
["select * from fruit where name = ?"
|
||||
"Grapefruit"]))))))))
|
||||
|
|
|
|||
|
|
@ -1,23 +1,25 @@
|
|||
;; copyright (c) 2019-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2019-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.optional-test
|
||||
"Test namespace for the optional builder functions."
|
||||
(:require [clojure.string :as str]
|
||||
[clojure.test :refer [deftest is testing use-fixtures]]
|
||||
[lazytest.core :refer [around set-ns-context!]]
|
||||
[lazytest.experimental.interfaces.clojure-test :refer [deftest is testing]]
|
||||
[next.jdbc.optional :as opt]
|
||||
[next.jdbc.protocols :as p]
|
||||
[next.jdbc.test-fixtures :refer [with-test-db ds column
|
||||
default-options]])
|
||||
(:import (java.sql ResultSet ResultSetMetaData)))
|
||||
[next.jdbc.test-fixtures :refer [col-kw column default-options ds index
|
||||
with-test-db]])
|
||||
(:import
|
||||
(java.sql ResultSet ResultSetMetaData)))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
||||
(use-fixtures :once with-test-db)
|
||||
(set-ns-context! [(around [f] (with-test-db f))])
|
||||
|
||||
(deftest test-map-row-builder
|
||||
(testing "default row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 1]
|
||||
[(str "select * from fruit where " (index) " = ?") 1]
|
||||
(assoc (default-options)
|
||||
:builder-fn opt/as-maps))]
|
||||
(is (map? row))
|
||||
|
|
@ -26,7 +28,7 @@
|
|||
(is (= "Apple" ((column :FRUIT/NAME) row)))))
|
||||
(testing "unqualified row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 2]
|
||||
[(str "select * from fruit where " (index) " = ?") 2]
|
||||
{:builder-fn opt/as-unqualified-maps})]
|
||||
(is (map? row))
|
||||
(is (not (contains? row (column :COST))))
|
||||
|
|
@ -34,23 +36,23 @@
|
|||
(is (= "Banana" ((column :NAME) row)))))
|
||||
(testing "lower-case row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 3]
|
||||
[(str "select * from fruit where " (index) " = ?") 3]
|
||||
(assoc (default-options)
|
||||
:builder-fn opt/as-lower-maps))]
|
||||
(is (map? row))
|
||||
(is (not (contains? row :fruit/appearance)))
|
||||
(is (= 3 (:fruit/id row)))
|
||||
(is (= "Peach" (:fruit/name row)))))
|
||||
(is (not (contains? row (col-kw :fruit/appearance))))
|
||||
(is (= 3 ((col-kw :fruit/id) row)))
|
||||
(is (= "Peach" ((col-kw :fruit/name) row)))))
|
||||
(testing "unqualified lower-case row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 4]
|
||||
[(str "select * from fruit where " (index) " = ?") 4]
|
||||
{:builder-fn opt/as-unqualified-lower-maps})]
|
||||
(is (map? row))
|
||||
(is (= 4 (:id row)))
|
||||
(is (= "Orange" (:name row)))))
|
||||
(is (= 4 ((col-kw :id) row)))
|
||||
(is (= "Orange" ((col-kw :name) row)))))
|
||||
(testing "custom row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 3]
|
||||
[(str "select * from fruit where " (index) " = ?") 3]
|
||||
(assoc (default-options)
|
||||
:builder-fn opt/as-modified-maps
|
||||
:label-fn str/lower-case
|
||||
|
|
@ -61,13 +63,13 @@
|
|||
(is (= "Peach" ((column :FRUIT/name) row))))))
|
||||
|
||||
(defn- default-column-reader
|
||||
[^ResultSet rs ^ResultSetMetaData rsmeta ^Integer i]
|
||||
[^ResultSet rs ^ResultSetMetaData _ ^Integer i]
|
||||
(.getObject rs i))
|
||||
|
||||
(deftest test-map-row-adapter
|
||||
(testing "default row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 1]
|
||||
[(str "select * from fruit where " (index) " = ?") 1]
|
||||
(assoc (default-options)
|
||||
:builder-fn (opt/as-maps-adapter
|
||||
opt/as-maps
|
||||
|
|
@ -78,7 +80,7 @@
|
|||
(is (= "Apple" ((column :FRUIT/NAME) row)))))
|
||||
(testing "unqualified row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 2]
|
||||
[(str "select * from fruit where " (index) " = ?") 2]
|
||||
{:builder-fn (opt/as-maps-adapter
|
||||
opt/as-unqualified-maps
|
||||
default-column-reader)})]
|
||||
|
|
@ -88,27 +90,27 @@
|
|||
(is (= "Banana" ((column :NAME) row)))))
|
||||
(testing "lower-case row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 3]
|
||||
[(str "select * from fruit where " (index) " = ?") 3]
|
||||
(assoc (default-options)
|
||||
:builder-fn (opt/as-maps-adapter
|
||||
opt/as-lower-maps
|
||||
default-column-reader)))]
|
||||
(is (map? row))
|
||||
(is (not (contains? row :fruit/appearance)))
|
||||
(is (= 3 (:fruit/id row)))
|
||||
(is (= "Peach" (:fruit/name row)))))
|
||||
(is (not (contains? row (col-kw :fruit/appearance))))
|
||||
(is (= 3 ((col-kw :fruit/id) row)))
|
||||
(is (= "Peach" ((col-kw :fruit/name) row)))))
|
||||
(testing "unqualified lower-case row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 4]
|
||||
[(str "select * from fruit where " (index) " = ?") 4]
|
||||
{:builder-fn (opt/as-maps-adapter
|
||||
opt/as-unqualified-lower-maps
|
||||
default-column-reader)})]
|
||||
(is (map? row))
|
||||
(is (= 4 (:id row)))
|
||||
(is (= "Orange" (:name row)))))
|
||||
(is (= 4 ((col-kw :id) row)))
|
||||
(is (= "Orange" ((col-kw :name) row)))))
|
||||
(testing "custom row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 3]
|
||||
[(str "select * from fruit where " (index) " = ?") 3]
|
||||
(assoc (default-options)
|
||||
:builder-fn (opt/as-maps-adapter
|
||||
opt/as-modified-maps
|
||||
|
|
|
|||
|
|
@ -1,72 +1,75 @@
|
|||
;; copyright (c) 2020-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2020-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.plan-test
|
||||
"Tests for the plan helpers."
|
||||
(:require [clojure.test :refer [deftest is use-fixtures]]
|
||||
(:require [lazytest.core :refer [around]]
|
||||
[lazytest.experimental.interfaces.clojure-test :refer [deftest is]]
|
||||
[next.jdbc.plan :as plan]
|
||||
[next.jdbc.specs :as specs]
|
||||
[next.jdbc.test-fixtures
|
||||
:refer [with-test-db ds]]
|
||||
:refer [with-test-db ds col-kw index]]
|
||||
[clojure.string :as str]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
||||
;; around each test because of the folding tests using 1,000 rows
|
||||
(use-fixtures :each with-test-db)
|
||||
|
||||
(specs/instrument)
|
||||
|
||||
(deftest select-one!-tests
|
||||
(is (= {:id 1}
|
||||
(plan/select-one! (ds) [:id] ["select * from fruit order by id"])))
|
||||
{:context [(around [f] (with-test-db f))]}
|
||||
(is (= {(col-kw :id) 1}
|
||||
(plan/select-one! (ds) [(col-kw :id)] [(str "select * from fruit order by " (index))])))
|
||||
(is (= 1
|
||||
(plan/select-one! (ds) :id ["select * from fruit order by id"])))
|
||||
(plan/select-one! (ds) (col-kw :id) [(str "select * from fruit order by " (index))])))
|
||||
(is (= "Banana"
|
||||
(plan/select-one! (ds) :name ["select * from fruit where id = ?" 2])))
|
||||
(plan/select-one! (ds) :name [(str "select * from fruit where " (index) " = ?") 2])))
|
||||
(is (= [1 "Apple"]
|
||||
(plan/select-one! (ds) (juxt :id :name)
|
||||
["select * from fruit order by id"])))
|
||||
(is (= {:id 1 :name "Apple"}
|
||||
(plan/select-one! (ds) #(select-keys % [:id :name])
|
||||
["select * from fruit order by id"]))))
|
||||
(plan/select-one! (ds) (juxt (col-kw :id) :name)
|
||||
[(str "select * from fruit order by " (index))])))
|
||||
(is (= {(col-kw :id) 1 :name "Apple"}
|
||||
(plan/select-one! (ds) #(select-keys % [(col-kw :id) :name])
|
||||
[(str "select * from fruit order by " (index))]))))
|
||||
|
||||
(deftest select-vector-tests
|
||||
(is (= [{:id 1} {:id 2} {:id 3} {:id 4}]
|
||||
(plan/select! (ds) [:id] ["select * from fruit order by id"])))
|
||||
{:context [(around [f] (with-test-db f))]}
|
||||
(is (= [{(col-kw :id) 1} {(col-kw :id) 2} {(col-kw :id) 3} {(col-kw :id) 4}]
|
||||
(plan/select! (ds) [(col-kw :id)] [(str "select * from fruit order by " (index))])))
|
||||
(is (= [1 2 3 4]
|
||||
(plan/select! (ds) :id ["select * from fruit order by id"])))
|
||||
(plan/select! (ds) (col-kw :id) [(str "select * from fruit order by " (index))])))
|
||||
(is (= ["Banana"]
|
||||
(plan/select! (ds) :name ["select * from fruit where id = ?" 2])))
|
||||
(plan/select! (ds) :name [(str "select * from fruit where " (index) " = ?") 2])))
|
||||
(is (= [[2 "Banana"]]
|
||||
(plan/select! (ds) (juxt :id :name)
|
||||
["select * from fruit where id = ?" 2])))
|
||||
(is (= [{:id 2 :name "Banana"}]
|
||||
(plan/select! (ds) [:id :name]
|
||||
["select * from fruit where id = ?" 2]))))
|
||||
(plan/select! (ds) (juxt (col-kw :id) :name)
|
||||
[(str "select * from fruit where " (index) " = ?") 2])))
|
||||
(is (= [{(col-kw :id) 2 :name "Banana"}]
|
||||
(plan/select! (ds) [(col-kw :id) :name]
|
||||
[(str "select * from fruit where " (index) " = ?") 2]))))
|
||||
|
||||
(deftest select-set-tests
|
||||
(is (= #{{:id 1} {:id 2} {:id 3} {:id 4}}
|
||||
(plan/select! (ds) [:id] ["select * from fruit order by id"]
|
||||
{:context [(around [f] (with-test-db f))]}
|
||||
(is (= #{{(col-kw :id) 1} {(col-kw :id) 2} {(col-kw :id) 3} {(col-kw :id) 4}}
|
||||
(plan/select! (ds) [(col-kw :id)] [(str "select * from fruit order by " (index))]
|
||||
{:into #{}})))
|
||||
(is (= #{1 2 3 4}
|
||||
(plan/select! (ds) :id ["select * from fruit order by id"]
|
||||
(plan/select! (ds) (col-kw :id) [(str "select * from fruit order by " (index))]
|
||||
{:into #{}}))))
|
||||
|
||||
(deftest select-map-tests
|
||||
{:context [(around [f] (with-test-db f))]}
|
||||
(is (= {1 "Apple", 2 "Banana", 3 "Peach", 4 "Orange"}
|
||||
(plan/select! (ds) (juxt :id :name) ["select * from fruit order by id"]
|
||||
(plan/select! (ds) (juxt (col-kw :id) :name) [(str "select * from fruit order by " (index))]
|
||||
{:into {}}))))
|
||||
|
||||
(deftest select-issue-227
|
||||
{:context [(around [f] (with-test-db f))]}
|
||||
(is (= ["Apple"]
|
||||
(plan/select! (ds) :name ["select * from fruit where id = ?" 1]
|
||||
(plan/select! (ds) :name [(str "select * from fruit where " (index) " = ?") 1]
|
||||
{:column-fn #(str/replace % "-" "_")})))
|
||||
(is (= ["Apple"]
|
||||
(plan/select! (ds) :foo/name ["select * from fruit where id = ?" 1]
|
||||
(plan/select! (ds) :foo/name [(str "select * from fruit where " (index) " = ?") 1]
|
||||
{:column-fn #(str/replace % "-" "_")})))
|
||||
(is (= ["Apple"]
|
||||
(plan/select! (ds) #(get % "name") ["select * from fruit where id = ?" 1]
|
||||
(plan/select! (ds) #(get % "name") [(str "select * from fruit where " (index) " = ?") 1]
|
||||
{:column-fn #(str/replace % "-" "_")})))
|
||||
(is (= [["Apple"]]
|
||||
(plan/select! (ds) (juxt :name) ["select * from fruit where id = ?" 1]
|
||||
(plan/select! (ds) (juxt :name) [(str "select * from fruit where " (index) " = ?") 1]
|
||||
{:column-fn #(str/replace % "-" "_")}))))
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2019-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2019-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.prepare-test
|
||||
"Stub test namespace for PreparedStatement creation etc.
|
||||
|
|
@ -8,75 +8,40 @@
|
|||
|
||||
The tests for the deprecated version of `execute-batch!` are here
|
||||
as a guard against regressions."
|
||||
(:require [clojure.test :refer [deftest is testing use-fixtures]]
|
||||
(:require [lazytest.core :refer [around set-ns-context!]]
|
||||
[lazytest.experimental.interfaces.clojure-test :refer [deftest is testing]]
|
||||
[next.jdbc :as jdbc]
|
||||
[next.jdbc.test-fixtures
|
||||
:refer [with-test-db ds jtds? mssql? sqlite?]]
|
||||
:refer [with-test-db ds jtds? mssql? sqlite? xtdb?]]
|
||||
[next.jdbc.prepare :as prep]
|
||||
[next.jdbc.specs :as specs]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
||||
(use-fixtures :once with-test-db)
|
||||
(set-ns-context! [(around [f] (with-test-db f))])
|
||||
|
||||
(specs/instrument)
|
||||
|
||||
(deftest execute-batch-tests
|
||||
(testing "simple batch insert"
|
||||
(is (= [1 1 1 1 1 1 1 1 1 13]
|
||||
(jdbc/with-transaction [t (ds) {:rollback-only true}]
|
||||
(with-open [ps (jdbc/prepare t ["
|
||||
(when-not (xtdb?)
|
||||
(testing "simple batch insert"
|
||||
(is (= [1 1 1 1 1 1 1 1 1 13]
|
||||
(jdbc/with-transaction [t (ds) {:rollback-only true}]
|
||||
(with-open [ps (jdbc/prepare t ["
|
||||
INSERT INTO fruit (name, appearance) VALUES (?,?)
|
||||
"])]
|
||||
(let [result (prep/execute-batch! ps [["fruit1" "one"]
|
||||
["fruit2" "two"]
|
||||
["fruit3" "three"]
|
||||
["fruit4" "four"]
|
||||
["fruit5" "five"]
|
||||
["fruit6" "six"]
|
||||
["fruit7" "seven"]
|
||||
["fruit8" "eight"]
|
||||
["fruit9" "nine"]])]
|
||||
(conj result (count (jdbc/execute! t ["select * from fruit"]))))))))
|
||||
(is (= 4 (count (jdbc/execute! (ds) ["select * from fruit"])))))
|
||||
(testing "small batch insert"
|
||||
(is (= [1 1 1 1 1 1 1 1 1 13]
|
||||
(jdbc/with-transaction [t (ds) {:rollback-only true}]
|
||||
(with-open [ps (jdbc/prepare t ["
|
||||
INSERT INTO fruit (name, appearance) VALUES (?,?)
|
||||
"])]
|
||||
(let [result (prep/execute-batch! ps [["fruit1" "one"]
|
||||
["fruit2" "two"]
|
||||
["fruit3" "three"]
|
||||
["fruit4" "four"]
|
||||
["fruit5" "five"]
|
||||
["fruit6" "six"]
|
||||
["fruit7" "seven"]
|
||||
["fruit8" "eight"]
|
||||
["fruit9" "nine"]]
|
||||
{:batch-size 3})]
|
||||
(conj result (count (jdbc/execute! t ["select * from fruit"]))))))))
|
||||
(is (= 4 (count (jdbc/execute! (ds) ["select * from fruit"])))))
|
||||
(testing "big batch insert"
|
||||
(is (= [1 1 1 1 1 1 1 1 1 13]
|
||||
(jdbc/with-transaction [t (ds) {:rollback-only true}]
|
||||
(with-open [ps (jdbc/prepare t ["
|
||||
INSERT INTO fruit (name, appearance) VALUES (?,?)
|
||||
"])]
|
||||
(let [result (prep/execute-batch! ps [["fruit1" "one"]
|
||||
["fruit2" "two"]
|
||||
["fruit3" "three"]
|
||||
["fruit4" "four"]
|
||||
["fruit5" "five"]
|
||||
["fruit6" "six"]
|
||||
["fruit7" "seven"]
|
||||
["fruit8" "eight"]
|
||||
["fruit9" "nine"]]
|
||||
{:batch-size 8})]
|
||||
(conj result (count (jdbc/execute! t ["select * from fruit"]))))))))
|
||||
(is (= 4 (count (jdbc/execute! (ds) ["select * from fruit"])))))
|
||||
(testing "large batch insert"
|
||||
(when-not (or (jtds?) (sqlite?))
|
||||
(let [result (prep/execute-batch! ps [["fruit1" "one"]
|
||||
["fruit2" "two"]
|
||||
["fruit3" "three"]
|
||||
["fruit4" "four"]
|
||||
["fruit5" "five"]
|
||||
["fruit6" "six"]
|
||||
["fruit7" "seven"]
|
||||
["fruit8" "eight"]
|
||||
["fruit9" "nine"]])]
|
||||
(conj result (count (jdbc/execute! t ["select * from fruit"]))))))))
|
||||
(is (= 4 (count (jdbc/execute! (ds) ["select * from fruit"])))))
|
||||
(testing "small batch insert"
|
||||
(is (= [1 1 1 1 1 1 1 1 1 13]
|
||||
(jdbc/with-transaction [t (ds) {:rollback-only true}]
|
||||
(with-open [ps (jdbc/prepare t ["
|
||||
|
|
@ -91,33 +56,70 @@ INSERT INTO fruit (name, appearance) VALUES (?,?)
|
|||
["fruit7" "seven"]
|
||||
["fruit8" "eight"]
|
||||
["fruit9" "nine"]]
|
||||
{:batch-size 4
|
||||
:large true})]
|
||||
{:batch-size 3})]
|
||||
(conj result (count (jdbc/execute! t ["select * from fruit"]))))))))
|
||||
(is (= 4 (count (jdbc/execute! (ds) ["select * from fruit"]))))))
|
||||
(testing "return generated keys"
|
||||
(when-not (or (mssql?) (sqlite?))
|
||||
(let [results
|
||||
(jdbc/with-transaction [t (ds) {:rollback-only true}]
|
||||
(with-open [ps (jdbc/prepare t ["
|
||||
(is (= 4 (count (jdbc/execute! (ds) ["select * from fruit"])))))
|
||||
(testing "big batch insert"
|
||||
(is (= [1 1 1 1 1 1 1 1 1 13]
|
||||
(jdbc/with-transaction [t (ds) {:rollback-only true}]
|
||||
(with-open [ps (jdbc/prepare t ["
|
||||
INSERT INTO fruit (name, appearance) VALUES (?,?)
|
||||
"])]
|
||||
(let [result (prep/execute-batch! ps [["fruit1" "one"]
|
||||
["fruit2" "two"]
|
||||
["fruit3" "three"]
|
||||
["fruit4" "four"]
|
||||
["fruit5" "five"]
|
||||
["fruit6" "six"]
|
||||
["fruit7" "seven"]
|
||||
["fruit8" "eight"]
|
||||
["fruit9" "nine"]]
|
||||
{:batch-size 8})]
|
||||
(conj result (count (jdbc/execute! t ["select * from fruit"]))))))))
|
||||
(is (= 4 (count (jdbc/execute! (ds) ["select * from fruit"])))))
|
||||
(testing "large batch insert"
|
||||
(when-not (or (jtds?) (sqlite?))
|
||||
(is (= [1 1 1 1 1 1 1 1 1 13]
|
||||
(jdbc/with-transaction [t (ds) {:rollback-only true}]
|
||||
(with-open [ps (jdbc/prepare t ["
|
||||
INSERT INTO fruit (name, appearance) VALUES (?,?)
|
||||
"])]
|
||||
(let [result (prep/execute-batch! ps [["fruit1" "one"]
|
||||
["fruit2" "two"]
|
||||
["fruit3" "three"]
|
||||
["fruit4" "four"]
|
||||
["fruit5" "five"]
|
||||
["fruit6" "six"]
|
||||
["fruit7" "seven"]
|
||||
["fruit8" "eight"]
|
||||
["fruit9" "nine"]]
|
||||
{:batch-size 4
|
||||
:large true})]
|
||||
(conj result (count (jdbc/execute! t ["select * from fruit"]))))))))
|
||||
(is (= 4 (count (jdbc/execute! (ds) ["select * from fruit"]))))))
|
||||
(testing "return generated keys"
|
||||
(when-not (or (mssql?) (sqlite?))
|
||||
(let [results
|
||||
(jdbc/with-transaction [t (ds) {:rollback-only true}]
|
||||
(with-open [ps (jdbc/prepare t ["
|
||||
INSERT INTO fruit (name, appearance) VALUES (?,?)
|
||||
"]
|
||||
{:return-keys true})]
|
||||
(let [result (prep/execute-batch! ps [["fruit1" "one"]
|
||||
["fruit2" "two"]
|
||||
["fruit3" "three"]
|
||||
["fruit4" "four"]
|
||||
["fruit5" "five"]
|
||||
["fruit6" "six"]
|
||||
["fruit7" "seven"]
|
||||
["fruit8" "eight"]
|
||||
["fruit9" "nine"]]
|
||||
{:batch-size 4
|
||||
:return-generated-keys true})]
|
||||
(conj result (count (jdbc/execute! t ["select * from fruit"]))))))]
|
||||
(is (= 13 (last results)))
|
||||
(is (every? map? (butlast results)))
|
||||
{:return-keys true})]
|
||||
(let [result (prep/execute-batch! ps [["fruit1" "one"]
|
||||
["fruit2" "two"]
|
||||
["fruit3" "three"]
|
||||
["fruit4" "four"]
|
||||
["fruit5" "five"]
|
||||
["fruit6" "six"]
|
||||
["fruit7" "seven"]
|
||||
["fruit8" "eight"]
|
||||
["fruit9" "nine"]]
|
||||
{:batch-size 4
|
||||
:return-generated-keys true})]
|
||||
(conj result (count (jdbc/execute! t ["select * from fruit"]))))))]
|
||||
(is (= 13 (last results)))
|
||||
(is (every? map? (butlast results)))
|
||||
;; Derby and SQLite only return one generated key per batch so there
|
||||
;; are only three keys, plus the overall count here:
|
||||
(is (< 3 (count results))))
|
||||
(is (= 4 (count (jdbc/execute! (ds) ["select * from fruit"])))))))
|
||||
(is (< 3 (count results))))
|
||||
(is (= 4 (count (jdbc/execute! (ds) ["select * from fruit"]))))))))
|
||||
|
|
|
|||
|
|
@ -1,9 +1,8 @@
|
|||
;; copyright (c) 2019-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2019-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.protocols-test
|
||||
"Stub test namespace for low-level protocols. Nothing can really be tested
|
||||
at this level tho'..."
|
||||
(:require [clojure.test :refer [deftest is testing]]
|
||||
[next.jdbc.protocols :refer :all]))
|
||||
(:require [next.jdbc.protocols]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
|
|
|||
|
|
@ -1,34 +1,30 @@
|
|||
;; copyright (c) 2019-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2019-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.quoted-test
|
||||
"Basic tests for quoting strategies. These are also tested indirectly
|
||||
via the next.jdbc.sql tests."
|
||||
(:require [clojure.test :refer [deftest are testing]]
|
||||
(:require [lazytest.core :refer [defdescribe describe it expect]]
|
||||
[next.jdbc.quoted :refer [ansi mysql sql-server oracle postgres
|
||||
schema]]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
||||
(deftest basic-quoting
|
||||
(are [quote-fn quoted] (= (quote-fn "x") quoted)
|
||||
ansi "\"x\""
|
||||
mysql "`x`"
|
||||
sql-server "[x]"
|
||||
oracle "\"x\""
|
||||
postgres "\"x\""))
|
||||
(def ^:private quote-fns [ansi mysql sql-server oracle postgres])
|
||||
|
||||
(deftest schema-quoting
|
||||
(testing "verify non-schema behavior"
|
||||
(are [quote-fn quoted] (= (quote-fn "x.y") quoted)
|
||||
ansi "\"x.y\""
|
||||
mysql "`x.y`"
|
||||
sql-server "[x.y]"
|
||||
oracle "\"x.y\""
|
||||
postgres "\"x.y\""))
|
||||
(testing "verify schema behavior"
|
||||
(are [quote-fn quoted] (= ((schema quote-fn) "x.y") quoted)
|
||||
ansi "\"x\".\"y\""
|
||||
mysql "`x`.`y`"
|
||||
sql-server "[x].[y]"
|
||||
oracle "\"x\".\"y\""
|
||||
postgres "\"x\".\"y\"")))
|
||||
(defdescribe quoted-functionality
|
||||
(describe "base quoting"
|
||||
(it "should correctly quote simple names"
|
||||
(doseq [[f e] (map vector quote-fns
|
||||
["\"x\"" "`x`" "[x]" "\"x\"" "\"x\""])]
|
||||
(expect (= (f "x") e)))))
|
||||
(describe "dotted name quoting"
|
||||
(describe "basic quoting"
|
||||
(it "should quote dotted names 'as-is'"
|
||||
(doseq [[f e] (map vector quote-fns
|
||||
["\"x.y\"" "`x.y`" "[x.y]" "\"x.y\"" "\"x.y\""])]
|
||||
(expect (= (f "x.y") e)))))
|
||||
(describe "schema quoting"
|
||||
(it "should split and quote dotted names with schema"
|
||||
(doseq [[f e] (map vector quote-fns
|
||||
["\"x\".\"y\"" "`x`.`y`" "[x].[y]" "\"x\".\"y\"" "\"x\".\"y\""])]
|
||||
(expect (= ((schema f) "x.y") e)))))))
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2019-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2019-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.result-set-test
|
||||
"Test namespace for the result set functions.
|
||||
|
|
@ -8,18 +8,19 @@
|
|||
(:require [clojure.core.protocols :as core-p]
|
||||
[clojure.datafy :as d]
|
||||
[clojure.string :as str]
|
||||
[clojure.test :refer [deftest is testing use-fixtures]]
|
||||
[lazytest.core :refer [around set-ns-context!]]
|
||||
[lazytest.experimental.interfaces.clojure-test :refer [deftest is testing]]
|
||||
[next.jdbc.protocols :as p]
|
||||
[next.jdbc.result-set :as rs]
|
||||
[next.jdbc.specs :as specs]
|
||||
[next.jdbc.test-fixtures :refer [with-test-db ds column
|
||||
default-options
|
||||
derby? mssql? mysql? postgres?]])
|
||||
[next.jdbc.test-fixtures :refer [with-test-db ds column index col-kw
|
||||
default-options
|
||||
derby? mssql? mysql? postgres? xtdb?]])
|
||||
(:import (java.sql ResultSet ResultSetMetaData)))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
||||
(use-fixtures :once with-test-db)
|
||||
(set-ns-context! [(around [f] (with-test-db f))])
|
||||
|
||||
(specs/instrument)
|
||||
|
||||
|
|
@ -27,7 +28,9 @@
|
|||
(testing "default schema"
|
||||
(let [connectable (ds)
|
||||
test-row (rs/datafiable-row {:TABLE/FRUIT_ID 1} connectable
|
||||
(default-options))
|
||||
(cond-> (default-options)
|
||||
(xtdb?)
|
||||
(assoc :schema-opts {:pk "_id"})))
|
||||
data (d/datafy test-row)
|
||||
v (get data :TABLE/FRUIT_ID)]
|
||||
;; check datafication is sane
|
||||
|
|
@ -40,7 +43,10 @@
|
|||
(let [connectable (ds)
|
||||
test-row (rs/datafiable-row {:foo/bar 2} connectable
|
||||
(assoc (default-options)
|
||||
:schema {:foo/bar :fruit/id}))
|
||||
:schema {:foo/bar
|
||||
(if (xtdb?)
|
||||
:fruit/_id
|
||||
:fruit/id)}))
|
||||
data (d/datafy test-row)
|
||||
v (get data :foo/bar)]
|
||||
;; check datafication is sane
|
||||
|
|
@ -53,7 +59,10 @@
|
|||
(let [connectable (ds)
|
||||
test-row (rs/datafiable-row {:foo/bar 3} connectable
|
||||
(assoc (default-options)
|
||||
:schema {:foo/bar [:fruit/id]}))
|
||||
:schema {:foo/bar
|
||||
[(if (xtdb?)
|
||||
:fruit/_id
|
||||
:fruit/id)]}))
|
||||
data (d/datafy test-row)
|
||||
v (get data :foo/bar)]
|
||||
;; check datafication is sane
|
||||
|
|
@ -67,7 +76,7 @@
|
|||
(let [connectable (ds)
|
||||
test-row (rs/datafiable-row {:foo/bar 2} connectable
|
||||
(assoc (default-options)
|
||||
:schema {:foo/bar [:fruit :id]}))
|
||||
:schema {:foo/bar [:fruit (col-kw :id)]}))
|
||||
data (d/datafy test-row)
|
||||
v (get data :foo/bar)]
|
||||
;; check datafication is sane
|
||||
|
|
@ -79,7 +88,7 @@
|
|||
(let [connectable (ds)
|
||||
test-row (rs/datafiable-row {:foo/bar 3} connectable
|
||||
(assoc (default-options)
|
||||
:schema {:foo/bar [:fruit :id :many]}))
|
||||
:schema {:foo/bar [:fruit (col-kw :id) :many]}))
|
||||
data (d/datafy test-row)
|
||||
v (get data :foo/bar)]
|
||||
;; check datafication is sane
|
||||
|
|
@ -93,7 +102,7 @@
|
|||
(deftest test-map-row-builder
|
||||
(testing "default row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 1]
|
||||
[(str "select * from fruit where " (index) " = ?") 1]
|
||||
(default-options))]
|
||||
(is (map? row))
|
||||
(is (contains? row (column :FRUIT/GRADE)))
|
||||
|
|
@ -101,7 +110,7 @@
|
|||
(is (= 1 ((column :FRUIT/ID) row)))
|
||||
(is (= "Apple" ((column :FRUIT/NAME) row))))
|
||||
(let [rs (p/-execute-all (ds)
|
||||
["select * from fruit order by id"]
|
||||
[(str "select * from fruit order by " (index))]
|
||||
(default-options))]
|
||||
(is (every? map? rs))
|
||||
(is (= 1 ((column :FRUIT/ID) (first rs))))
|
||||
|
|
@ -110,7 +119,7 @@
|
|||
(is (= "Orange" ((column :FRUIT/NAME) (last rs))))))
|
||||
(testing "unqualified row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 2]
|
||||
[(str "select * from fruit where " (index) " = ?") 2]
|
||||
{:builder-fn rs/as-unqualified-maps})]
|
||||
(is (map? row))
|
||||
(is (contains? row (column :COST)))
|
||||
|
|
@ -119,34 +128,35 @@
|
|||
(is (= "Banana" ((column :NAME) row)))))
|
||||
(testing "lower-case row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 3]
|
||||
[(str "select * from fruit where " (index) " = ?") 3]
|
||||
(assoc (default-options)
|
||||
:builder-fn rs/as-lower-maps))]
|
||||
(is (map? row))
|
||||
(is (contains? row :fruit/appearance))
|
||||
(is (nil? (:fruit/appearance row)))
|
||||
(is (= 3 (:fruit/id row)))
|
||||
(is (= "Peach" (:fruit/name row)))))
|
||||
(is (contains? row (col-kw :fruit/appearance)))
|
||||
(is (nil? ((col-kw :fruit/appearance) row)))
|
||||
(is (= 3 ((col-kw :fruit/id) row)))
|
||||
(is (= "Peach" ((col-kw :fruit/name) row)))))
|
||||
(testing "unqualified lower-case row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 4]
|
||||
[(str "select * from fruit where " (index) " = ?") 4]
|
||||
{:builder-fn rs/as-unqualified-lower-maps})]
|
||||
(is (map? row))
|
||||
(is (= 4 (:id row)))
|
||||
(is (= "Orange" (:name row)))))
|
||||
(is (= 4 ((col-kw :id) row)))
|
||||
(is (= "Orange" ((col-kw :name) row)))))
|
||||
(testing "kebab-case row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select id,name,appearance as looks_like from fruit where id = ?" 3]
|
||||
[(str "select " (index) ",name,appearance as looks_like from fruit where " (index) " = ?") 3]
|
||||
(assoc (default-options)
|
||||
:builder-fn rs/as-kebab-maps))]
|
||||
(is (map? row))
|
||||
(is (contains? row :fruit/looks-like))
|
||||
(is (nil? (:fruit/looks-like row)))
|
||||
(is (= 3 (:fruit/id row)))
|
||||
(is (= "Peach" (:fruit/name row)))))
|
||||
(is (contains? row (col-kw :fruit/looks-like)))
|
||||
(is (nil? ((col-kw :fruit/looks-like) row)))
|
||||
;; kebab-case strips leading _ from _id (XTDB):
|
||||
(is (= 3 ((if (xtdb?) :id :fruit/id) row)))
|
||||
(is (= "Peach" ((col-kw :fruit/name) row)))))
|
||||
(testing "unqualified kebab-case row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select id,name,appearance as looks_like from fruit where id = ?" 4]
|
||||
[(str "select " (index) ",name,appearance as looks_like from fruit where " (index) " = ?") 4]
|
||||
{:builder-fn rs/as-unqualified-kebab-maps})]
|
||||
(is (map? row))
|
||||
(is (contains? row :looks-like))
|
||||
|
|
@ -155,7 +165,7 @@
|
|||
(is (= "Orange" (:name row)))))
|
||||
(testing "custom row builder 1"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select fruit.*, id + 100 as newid from fruit where id = ?" 3]
|
||||
[(str "select fruit.*, " (index) " + 100 as newid from fruit where " (index) " = ?") 3]
|
||||
(assoc (default-options)
|
||||
:builder-fn rs/as-modified-maps
|
||||
:label-fn str/lower-case
|
||||
|
|
@ -168,7 +178,7 @@
|
|||
(is (= "Peach" ((column :FRUIT/name) row)))))
|
||||
(testing "custom row builder 2"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select fruit.*, id + 100 as newid from fruit where id = ?" 3]
|
||||
[(str "select fruit.*, " (index) " + 100 as newid from fruit where " (index) " = ?") 3]
|
||||
(assoc (default-options)
|
||||
:builder-fn rs/as-modified-maps
|
||||
:label-fn str/lower-case
|
||||
|
|
@ -176,12 +186,12 @@
|
|||
(is (map? row))
|
||||
(is (contains? row :vegetable/appearance))
|
||||
(is (nil? (:vegetable/appearance row)))
|
||||
(is (= 3 (:vegetable/id row)))
|
||||
(is (= 3 ((if (xtdb?) :vegetable/_id :vegetable/id) row)))
|
||||
(is (= 103 (:vegetable/newid row))) ; constant qualifier here
|
||||
(is (= "Peach" (:vegetable/name row)))))
|
||||
(testing "adapted row builder"
|
||||
(let [row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 3]
|
||||
[(str "select * from fruit where " (index) " = ?") 3]
|
||||
(assoc
|
||||
(default-options)
|
||||
:builder-fn (rs/as-maps-adapter
|
||||
|
|
@ -207,7 +217,7 @@
|
|||
(fn [^ResultSet rs _ ^Integer i]
|
||||
(.getObject rs i)))
|
||||
row (p/-execute-one (ds)
|
||||
["select * from fruit where id = ?" 3]
|
||||
[(str "select * from fruit where " (index) " = ?") 3]
|
||||
(assoc
|
||||
(default-options)
|
||||
:builder-fn (rs/as-maps-adapter
|
||||
|
|
@ -236,7 +246,7 @@
|
|||
(testing "row-numbers on bare abstraction"
|
||||
(is (= [1 2 3]
|
||||
(into [] (map rs/row-number)
|
||||
(p/-execute (ds) ["select * from fruit where id < ?" 4]
|
||||
(p/-execute (ds) [(str "select * from fruit where " (index) " < ?") 4]
|
||||
;; we do not need a real builder here...
|
||||
(cond-> {:builder-fn (constantly nil)}
|
||||
(derby?)
|
||||
|
|
@ -247,7 +257,7 @@
|
|||
(is (= [1 2 3]
|
||||
(into [] (comp (map #(rs/datafiable-row % (ds) {}))
|
||||
(map rs/row-number))
|
||||
(p/-execute (ds) ["select * from fruit where id < ?" 4]
|
||||
(p/-execute (ds) [(str "select * from fruit where " (index) " < ?") 4]
|
||||
;; ...but datafiable-row requires a real builder
|
||||
(cond-> {:builder-fn rs/as-arrays}
|
||||
(derby?)
|
||||
|
|
@ -257,7 +267,7 @@
|
|||
|
||||
(deftest test-column-names
|
||||
(testing "column-names on bare abstraction"
|
||||
(is (= #{"id" "appearance" "grade" "cost" "name"}
|
||||
(is (= #{(index) "appearance" "grade" "cost" "name"}
|
||||
(reduce (fn [_ row]
|
||||
(-> row
|
||||
(->> (rs/column-names)
|
||||
|
|
@ -265,11 +275,11 @@
|
|||
(set)
|
||||
(reduced))))
|
||||
nil
|
||||
(p/-execute (ds) ["select * from fruit where id < ?" 4]
|
||||
(p/-execute (ds) [(str "select * from fruit where " (index) " < ?") 4]
|
||||
;; column-names require a real builder
|
||||
{:builder-fn rs/as-arrays})))))
|
||||
(testing "column-names on realized row"
|
||||
(is (= #{"id" "appearance" "grade" "cost" "name"}
|
||||
(is (= #{(index) "appearance" "grade" "cost" "name"}
|
||||
(reduce (fn [_ row]
|
||||
(-> row
|
||||
(rs/datafiable-row (ds) {})
|
||||
|
|
@ -278,7 +288,7 @@
|
|||
(set)
|
||||
(reduced))))
|
||||
nil
|
||||
(p/-execute (ds) ["select * from fruit where id < ?" 4]
|
||||
(p/-execute (ds) [(str "select * from fruit where " (index) " < ?") 4]
|
||||
{:builder-fn rs/as-arrays}))))))
|
||||
|
||||
(deftest test-over-partition-all
|
||||
|
|
@ -299,31 +309,31 @@
|
|||
(testing "no row builder is used"
|
||||
(is (= [true]
|
||||
(into [] (map map?) ; it looks like a real map now
|
||||
(p/-execute (ds) ["select * from fruit where id = ?" 1]
|
||||
(p/-execute (ds) [(str "select * from fruit where " (index) " = ?") 1]
|
||||
{:builder-fn (constantly nil)}))))
|
||||
(is (= ["Apple"]
|
||||
(into [] (map :name) ; keyword selection works
|
||||
(p/-execute (ds) ["select * from fruit where id = ?" 1]
|
||||
(p/-execute (ds) [(str "select * from fruit where " (index) " = ?") 1]
|
||||
{:builder-fn (constantly nil)}))))
|
||||
(is (= [[2 [:name "Banana"]]]
|
||||
(into [] (map (juxt #(get % "id") ; get by string key works
|
||||
(into [] (map (juxt #(get % (index)) ; get by string key works
|
||||
#(find % :name))) ; get MapEntry works
|
||||
(p/-execute (ds) ["select * from fruit where id = ?" 2]
|
||||
(p/-execute (ds) [(str "select * from fruit where " (index) " = ?") 2]
|
||||
{:builder-fn (constantly nil)}))))
|
||||
(is (= [{:id 3 :name "Peach"}]
|
||||
(into [] (map #(select-keys % [:id :name])) ; select-keys works
|
||||
(p/-execute (ds) ["select * from fruit where id = ?" 3]
|
||||
(is (= [{(col-kw :id) 3 :name "Peach"}]
|
||||
(into [] (map #(select-keys % [(col-kw :id) :name])) ; select-keys works
|
||||
(p/-execute (ds) [(str "select * from fruit where " (index) " = ?") 3]
|
||||
{:builder-fn (constantly nil)}))))
|
||||
(is (= [[:orange 4]]
|
||||
(into [] (map #(vector (if (contains? % :name) ; contains works
|
||||
(keyword (str/lower-case (:name %)))
|
||||
:unnamed)
|
||||
(get % :id 0))) ; get with not-found works
|
||||
(p/-execute (ds) ["select * from fruit where id = ?" 4]
|
||||
(get % (col-kw :id) 0))) ; get with not-found works
|
||||
(p/-execute (ds) [(str "select * from fruit where " (index) " = ?") 4]
|
||||
{:builder-fn (constantly nil)}))))
|
||||
(is (= [{}]
|
||||
(into [] (map empty) ; return empty map without building
|
||||
(p/-execute (ds) ["select * from fruit where id = ?" 1]
|
||||
(p/-execute (ds) [(str "select * from fruit where " (index) " = ?") 1]
|
||||
{:builder-fn (constantly nil)})))))
|
||||
(testing "count does not build a map"
|
||||
(let [count-builder (fn [_1 _2]
|
||||
|
|
@ -331,7 +341,7 @@
|
|||
(column-count [_] 13)))]
|
||||
(is (= [13]
|
||||
(into [] (map count) ; count relies on columns, not row fields
|
||||
(p/-execute (ds) ["select * from fruit where id = ?" 1]
|
||||
(p/-execute (ds) [(str "select * from fruit where " (index) " = ?") 1]
|
||||
{:builder-fn count-builder}))))))
|
||||
(testing "assoc, dissoc, cons, seq, and = build maps"
|
||||
(is (map? (reduce (fn [_ row] (reduced (assoc row :x 1)))
|
||||
|
|
@ -417,7 +427,7 @@
|
|||
(defn fruit-builder [^ResultSet rs ^ResultSetMetaData rsmeta]
|
||||
(reify
|
||||
rs/RowBuilder
|
||||
(->row [_] (->Fruit (.getObject rs "id")
|
||||
(->row [_] (->Fruit (.getObject rs ^String (index))
|
||||
(.getObject rs "name")
|
||||
(.getObject rs "appearance")
|
||||
(.getObject rs "cost")
|
||||
|
|
@ -434,7 +444,7 @@
|
|||
(valAt [this k] (get this k nil))
|
||||
(valAt [this k not-found]
|
||||
(case k
|
||||
:cols [:id :name :appearance :cost :grade]
|
||||
:cols [(col-kw :id) :name :appearance :cost :grade]
|
||||
:rsmeta rsmeta
|
||||
not-found))))
|
||||
|
||||
|
|
@ -467,7 +477,7 @@
|
|||
metadata))))
|
||||
|
||||
(deftest clob-reading
|
||||
(when-not (or (mssql?) (mysql?) (postgres?)) ; no clob in these
|
||||
(when-not (or (mssql?) (mysql?) (postgres?) (xtdb?)) ; no clob in these
|
||||
(with-open [con (p/get-connection (ds) {})]
|
||||
(try
|
||||
(p/-execute-one con ["DROP TABLE CLOBBER"] {})
|
||||
|
|
@ -497,10 +507,10 @@ CREATE TABLE CLOBBER (
|
|||
(testing "get n on bare abstraction over arrays"
|
||||
(is (= [1 2 3]
|
||||
(into [] (map #(get % 0))
|
||||
(p/-execute (ds) ["select id from fruit where id < ?" 4]
|
||||
(p/-execute (ds) [(str "select " (index) " from fruit where " (index) " < ? order by " (index)) 4]
|
||||
{:builder-fn rs/as-arrays})))))
|
||||
(testing "nth on bare abstraction over arrays"
|
||||
(is (= [1 2 3]
|
||||
(into [] (map #(nth % 0))
|
||||
(p/-execute (ds) ["select id from fruit where id < ?" 4]
|
||||
(p/-execute (ds) [(str "select " (index) " from fruit where " (index) " < ? order by " (index)) 4]
|
||||
{:builder-fn rs/as-arrays}))))))
|
||||
|
|
|
|||
|
|
@ -1,11 +1,10 @@
|
|||
;; copyright (c) 2019-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2019-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.specs-test
|
||||
"Stub test namespace for the specs.
|
||||
|
||||
The specs are used (and 'tested') as part of the tests for the
|
||||
next.jdbc and next.jdbc.sql namespaces."
|
||||
(:require [clojure.test :refer [deftest is testing]]
|
||||
[next.jdbc.specs :refer :all]))
|
||||
(:require [next.jdbc.specs]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
|
|
|||
|
|
@ -1,8 +1,8 @@
|
|||
;; copyright (c) 2019-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2019-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.sql.builder-test
|
||||
"Tests for the SQL string building functions in next.jdbc.sql.builder."
|
||||
(:require [clojure.test :refer [deftest is testing]]
|
||||
(:require [lazytest.experimental.interfaces.clojure-test :refer [deftest is testing thrown?]]
|
||||
[next.jdbc.quoted :refer [mysql sql-server]]
|
||||
[next.jdbc.sql.builder :as builder]))
|
||||
|
||||
|
|
|
|||
|
|
@ -1,25 +1,26 @@
|
|||
;; copyright (c) 2019-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2019-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.sql-test
|
||||
"Tests for the syntactic sugar SQL functions."
|
||||
(:require [clojure.test :refer [deftest is testing use-fixtures]]
|
||||
(:require [lazytest.core :refer [around set-ns-context!]]
|
||||
[lazytest.experimental.interfaces.clojure-test :refer [deftest is testing thrown?]]
|
||||
[next.jdbc :as jdbc]
|
||||
[next.jdbc.specs :as specs]
|
||||
[next.jdbc.sql :as sql]
|
||||
[next.jdbc.test-fixtures
|
||||
:refer [with-test-db ds column default-options
|
||||
derby? jtds? maria? mssql? mysql? postgres? sqlite?]]
|
||||
:refer [col-kw column default-options derby? ds index jtds?
|
||||
maria? mssql? mysql? postgres? sqlite? with-test-db xtdb?]]
|
||||
[next.jdbc.types :refer [as-other as-real as-varchar]]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
||||
(use-fixtures :once with-test-db)
|
||||
(set-ns-context! [(around [f] (with-test-db f))])
|
||||
|
||||
(specs/instrument)
|
||||
|
||||
(deftest test-query
|
||||
(let [ds-opts (jdbc/with-options (ds) (default-options))
|
||||
rs (sql/query ds-opts ["select * from fruit order by id"])]
|
||||
rs (sql/query ds-opts [(str "select * from fruit order by " (index))])]
|
||||
(is (= 4 (count rs)))
|
||||
(is (every? map? rs))
|
||||
(is (every? meta rs))
|
||||
|
|
@ -34,10 +35,10 @@
|
|||
(if (or (mysql?) (sqlite?))
|
||||
{:limit 2 :offset 1}
|
||||
{:offset 1 :fetch 2})
|
||||
:columns [:ID
|
||||
:columns [(col-kw :ID)
|
||||
["CASE WHEN grade > 91 THEN 'ok ' ELSE 'bad' END"
|
||||
:QUALITY]]
|
||||
:order-by [:id]))]
|
||||
:order-by [(col-kw :id)]))]
|
||||
(is (= 2 (count rs)))
|
||||
(is (every? map? rs))
|
||||
(is (every? meta rs))
|
||||
|
|
@ -67,17 +68,18 @@
|
|||
(is (= 1 count-v)))
|
||||
(let [count-v (sql/aggregate-by-keys ds-opts :fruit "count(*)" :all)]
|
||||
(is (= 4 count-v)))
|
||||
(let [max-id (sql/aggregate-by-keys ds-opts :fruit "max(id)" :all)]
|
||||
(let [max-id (sql/aggregate-by-keys ds-opts :fruit (str "max(" (index) ")") :all)]
|
||||
(is (= 4 max-id)))
|
||||
(let [min-name (sql/aggregate-by-keys ds-opts :fruit "min(name)" :all)]
|
||||
(is (= "Apple" min-name)))
|
||||
(when-not (xtdb?) ; XTDB does not support min/max on strings?
|
||||
(let [min-name (sql/aggregate-by-keys ds-opts :fruit "min(name)" :all)]
|
||||
(is (= "Apple" min-name))))
|
||||
(is (thrown? IllegalArgumentException
|
||||
(sql/aggregate-by-keys ds-opts :fruit "count(*)" :all {:columns []})))))
|
||||
|
||||
(deftest test-get-by-id
|
||||
(let [ds-opts (jdbc/with-options (ds) (default-options))]
|
||||
(is (nil? (sql/get-by-id ds-opts :fruit -1)))
|
||||
(let [row (sql/get-by-id ds-opts :fruit 3)]
|
||||
(is (nil? (sql/get-by-id ds-opts :fruit -1 (col-kw :id) {})))
|
||||
(let [row (sql/get-by-id ds-opts :fruit 3 (col-kw :id) {})]
|
||||
(is (map? row))
|
||||
(is (= "Peach" ((column :FRUIT/NAME) row))))
|
||||
(let [row (sql/get-by-id ds-opts :fruit "juicy" :appearance {})]
|
||||
|
|
@ -88,23 +90,28 @@
|
|||
(is (map? row))
|
||||
(is (= 2 ((column :FRUIT/ID) row))))))
|
||||
|
||||
(defn- update-count [n]
|
||||
(if (xtdb?)
|
||||
{:next.jdbc/update-count 0}
|
||||
{:next.jdbc/update-count n}))
|
||||
|
||||
(deftest test-update!
|
||||
(let [ds-opts (jdbc/with-options (ds) (default-options))]
|
||||
(try
|
||||
(is (= {:next.jdbc/update-count 1}
|
||||
(sql/update! ds-opts :fruit {:appearance "brown"} {:id 2})))
|
||||
(is (= (update-count 1)
|
||||
(sql/update! ds-opts :fruit {:appearance "brown"} {(col-kw :id) 2})))
|
||||
(is (= "brown" ((column :FRUIT/APPEARANCE)
|
||||
(sql/get-by-id ds-opts :fruit 2))))
|
||||
(sql/get-by-id ds-opts :fruit 2 (col-kw :id) {}))))
|
||||
(finally
|
||||
(sql/update! ds-opts :fruit {:appearance "yellow"} {:id 2})))
|
||||
(sql/update! ds-opts :fruit {:appearance "yellow"} {(col-kw :id) 2})))
|
||||
(try
|
||||
(is (= {:next.jdbc/update-count 1}
|
||||
(is (= (update-count 1)
|
||||
(sql/update! ds-opts :fruit {:appearance "green"}
|
||||
["name = ?" "Banana"])))
|
||||
(is (= "green" ((column :FRUIT/APPEARANCE)
|
||||
(sql/get-by-id ds-opts :fruit 2))))
|
||||
(sql/get-by-id ds-opts :fruit 2 (col-kw :id) {}))))
|
||||
(finally
|
||||
(sql/update! ds-opts :fruit {:appearance "yellow"} {:id 2})))))
|
||||
(sql/update! ds-opts :fruit {:appearance "yellow"} {(col-kw :id) 2})))))
|
||||
|
||||
(deftest test-insert-delete
|
||||
(let [new-key (cond (derby?) :1
|
||||
|
|
@ -113,18 +120,24 @@
|
|||
(mssql?) :GENERATED_KEYS
|
||||
(mysql?) :GENERATED_KEY
|
||||
(postgres?) :fruit/id
|
||||
;; XTDB does not return the generated key so we fix it
|
||||
;; to be the one we insert here, and then fake it in all
|
||||
;; the other tests.
|
||||
(xtdb?) (constantly 5)
|
||||
:else :FRUIT/ID)]
|
||||
(testing "single insert/delete"
|
||||
(is (== 5 (new-key (sql/insert! (ds) :fruit
|
||||
{:name (as-varchar "Kiwi")
|
||||
:appearance "green & fuzzy"
|
||||
:cost 100 :grade (as-real 99.9)}
|
||||
(cond-> {:name (as-varchar "Kiwi")
|
||||
:appearance "green & fuzzy"
|
||||
:cost 100 :grade (as-real 99.9)}
|
||||
(xtdb?)
|
||||
(assoc :_id 5))
|
||||
{:suffix
|
||||
(when (sqlite?)
|
||||
"RETURNING *")}))))
|
||||
(is (= 5 (count (sql/query (ds) ["select * from fruit"]))))
|
||||
(is (= {:next.jdbc/update-count 1}
|
||||
(sql/delete! (ds) :fruit {:id 5})))
|
||||
(is (= (update-count 1)
|
||||
(sql/delete! (ds) :fruit {(col-kw :id) 5})))
|
||||
(is (= 4 (count (sql/query (ds) ["select * from fruit"])))))
|
||||
(testing "multiple insert/delete"
|
||||
(is (= (cond (derby?)
|
||||
|
|
@ -133,23 +146,28 @@
|
|||
[8M]
|
||||
(maria?)
|
||||
[6]
|
||||
(xtdb?)
|
||||
[]
|
||||
:else
|
||||
[6 7 8])
|
||||
(mapv new-key
|
||||
(sql/insert-multi! (ds) :fruit
|
||||
[:name :appearance :cost :grade]
|
||||
[["Kiwi" "green & fuzzy" 100 99.9]
|
||||
["Grape" "black" 10 50]
|
||||
["Lemon" "yellow" 20 9.9]]
|
||||
(cond->> [:name :appearance :cost :grade]
|
||||
(xtdb?) (cons :_id))
|
||||
(cond->> [["Kiwi" "green & fuzzy" 100 99.9]
|
||||
["Grape" "black" 10 50]
|
||||
["Lemon" "yellow" 20 9.9]]
|
||||
(xtdb?)
|
||||
(map cons [6 7 8]))
|
||||
{:suffix
|
||||
(when (sqlite?)
|
||||
"RETURNING *")}))))
|
||||
(is (= 7 (count (sql/query (ds) ["select * from fruit"]))))
|
||||
(is (= {:next.jdbc/update-count 1}
|
||||
(sql/delete! (ds) :fruit {:id 6})))
|
||||
(is (= (update-count 1)
|
||||
(sql/delete! (ds) :fruit {(col-kw :id) 6})))
|
||||
(is (= 6 (count (sql/query (ds) ["select * from fruit"]))))
|
||||
(is (= {:next.jdbc/update-count 2}
|
||||
(sql/delete! (ds) :fruit ["id > ?" 4])))
|
||||
(is (= (update-count 2)
|
||||
(sql/delete! (ds) :fruit [(str (index) " > ?") 4])))
|
||||
(is (= 4 (count (sql/query (ds) ["select * from fruit"])))))
|
||||
(testing "multiple insert/delete with sequential cols/rows" ; per #43
|
||||
(is (= (cond (derby?)
|
||||
|
|
@ -158,23 +176,28 @@
|
|||
[11M]
|
||||
(maria?)
|
||||
[9]
|
||||
(xtdb?)
|
||||
[]
|
||||
:else
|
||||
[9 10 11])
|
||||
(mapv new-key
|
||||
(sql/insert-multi! (ds) :fruit
|
||||
'(:name :appearance :cost :grade)
|
||||
'(("Kiwi" "green & fuzzy" 100 99.9)
|
||||
("Grape" "black" 10 50)
|
||||
("Lemon" "yellow" 20 9.9))
|
||||
(cond->> '(:name :appearance :cost :grade)
|
||||
(xtdb?) (cons :_id))
|
||||
(cond->> '(("Kiwi" "green & fuzzy" 100 99.9)
|
||||
("Grape" "black" 10 50)
|
||||
("Lemon" "yellow" 20 9.9))
|
||||
(xtdb?)
|
||||
(map cons [9 10 11]))
|
||||
{:suffix
|
||||
(when (sqlite?)
|
||||
"RETURNING *")}))))
|
||||
(is (= 7 (count (sql/query (ds) ["select * from fruit"]))))
|
||||
(is (= {:next.jdbc/update-count 1}
|
||||
(sql/delete! (ds) :fruit {:id 9})))
|
||||
(is (= (update-count 1)
|
||||
(sql/delete! (ds) :fruit {(col-kw :id) 9})))
|
||||
(is (= 6 (count (sql/query (ds) ["select * from fruit"]))))
|
||||
(is (= {:next.jdbc/update-count 2}
|
||||
(sql/delete! (ds) :fruit ["id > ?" 4])))
|
||||
(is (= (update-count 2)
|
||||
(sql/delete! (ds) :fruit [(str (index) " > ?") 4])))
|
||||
(is (= 4 (count (sql/query (ds) ["select * from fruit"])))))
|
||||
(testing "multiple insert/delete with maps"
|
||||
(is (= (cond (derby?)
|
||||
|
|
@ -183,31 +206,35 @@
|
|||
[14M]
|
||||
(maria?)
|
||||
[12]
|
||||
(xtdb?)
|
||||
[]
|
||||
:else
|
||||
[12 13 14])
|
||||
(mapv new-key
|
||||
(sql/insert-multi! (ds) :fruit
|
||||
[{:name "Kiwi"
|
||||
:appearance "green & fuzzy"
|
||||
:cost 100
|
||||
:grade 99.9}
|
||||
{:name "Grape"
|
||||
:appearance "black"
|
||||
:cost 10
|
||||
:grade 50}
|
||||
{:name "Lemon"
|
||||
:appearance "yellow"
|
||||
:cost 20
|
||||
:grade 9.9}]
|
||||
(cond->> [{:name "Kiwi"
|
||||
:appearance "green & fuzzy"
|
||||
:cost 100
|
||||
:grade 99.9}
|
||||
{:name "Grape"
|
||||
:appearance "black"
|
||||
:cost 10
|
||||
:grade 50}
|
||||
{:name "Lemon"
|
||||
:appearance "yellow"
|
||||
:cost 20
|
||||
:grade 9.9}]
|
||||
(xtdb?)
|
||||
(map #(assoc %2 :_id %1) [12 13 14]))
|
||||
{:suffix
|
||||
(when (sqlite?)
|
||||
"RETURNING *")}))))
|
||||
(is (= 7 (count (sql/query (ds) ["select * from fruit"]))))
|
||||
(is (= {:next.jdbc/update-count 1}
|
||||
(sql/delete! (ds) :fruit {:id 12})))
|
||||
(is (= (update-count 1)
|
||||
(sql/delete! (ds) :fruit {(col-kw :id) 12})))
|
||||
(is (= 6 (count (sql/query (ds) ["select * from fruit"]))))
|
||||
(is (= {:next.jdbc/update-count 2}
|
||||
(sql/delete! (ds) :fruit ["id > ?" 10])))
|
||||
(is (= (update-count 2)
|
||||
(sql/delete! (ds) :fruit [(str (index) " > ?") 10])))
|
||||
(is (= 4 (count (sql/query (ds) ["select * from fruit"])))))
|
||||
(testing "empty insert-multi!" ; per #44 and #264
|
||||
(is (= [] (sql/insert-multi! (ds) :fruit
|
||||
|
|
@ -250,12 +277,12 @@
|
|||
(deftest no-empty-order-by
|
||||
(is (thrown? clojure.lang.ExceptionInfo
|
||||
(sql/find-by-keys (ds) :fruit
|
||||
{:name "Apple"}
|
||||
{:order-by []}))))
|
||||
{:name "Apple"}
|
||||
{:order-by []}))))
|
||||
|
||||
(deftest array-in
|
||||
(when (postgres?)
|
||||
(let [data (sql/find-by-keys (ds) :fruit ["id = any(?)" (int-array [1 2 3 4])])]
|
||||
(let [data (sql/find-by-keys (ds) :fruit [(str (index) " = any(?)") (int-array [1 2 3 4])])]
|
||||
(is (= 4 (count data))))))
|
||||
|
||||
(deftest enum-pg
|
||||
|
|
|
|||
|
|
@ -64,16 +64,27 @@
|
|||
(def ^:private test-jtds
|
||||
(when (System/getenv "NEXT_JDBC_TEST_MSSQL") test-jtds-map))
|
||||
|
||||
(def ^:private test-xtdb-map {:dbtype "xtdb" :dbname "xtdb"})
|
||||
|
||||
(def ^:private test-xtdb
|
||||
(when (and (System/getenv "NEXT_JDBC_TEST_XTDB")
|
||||
;; only if we're on jdk21+
|
||||
(str/starts-with? (System/getProperty "java.version") "2"))
|
||||
test-xtdb-map))
|
||||
|
||||
(def ^:private test-db-specs
|
||||
(cond-> [test-derby test-h2-mem test-h2 test-hsql test-sqlite]
|
||||
test-postgres (conj test-postgres)
|
||||
test-mysql (conj test-mysql)
|
||||
test-mssql (conj test-mssql test-jtds)))
|
||||
test-mssql (conj test-mssql test-jtds)
|
||||
test-xtdb (conj test-xtdb)))
|
||||
|
||||
(def ^:private test-db-spec (atom nil))
|
||||
|
||||
(defn derby? [] (= "derby" (:dbtype @test-db-spec)))
|
||||
|
||||
(defn h2? [] (str/starts-with? (:dbtype @test-db-spec) "h2"))
|
||||
|
||||
(defn hsqldb? [] (= "hsqldb" (:dbtype @test-db-spec)))
|
||||
|
||||
(defn jtds? [] (= "jtds" (:dbtype @test-db-spec)))
|
||||
|
|
@ -86,19 +97,34 @@
|
|||
|
||||
(defn postgres? [] (= "embedded-postgres" (:dbtype @test-db-spec)))
|
||||
|
||||
(defn xtdb? [] (= "xtdb" (:dbtype @test-db-spec)))
|
||||
|
||||
(defn sqlite? [] (= "sqlite" (:dbtype @test-db-spec)))
|
||||
|
||||
(defn stored-proc? [] (not (#{"derby" "h2" "h2:mem" "sqlite"} (:dbtype @test-db-spec))))
|
||||
(defn stored-proc? [] (not (#{"derby" "h2" "h2:mem" "sqlite" "xtdb"}
|
||||
(:dbtype @test-db-spec))))
|
||||
|
||||
(defn column [k]
|
||||
(let [n (namespace k)]
|
||||
(keyword (when n (cond (postgres?) (str/lower-case n)
|
||||
(mssql?) (str/lower-case n)
|
||||
(mysql?) (str/lower-case n)
|
||||
(xtdb?) nil
|
||||
:else n))
|
||||
(cond (postgres?) (str/lower-case (name k))
|
||||
(xtdb?) (let [c (str/lower-case (name k))]
|
||||
(if (= "id" c) "_id" c))
|
||||
:else (name k)))))
|
||||
|
||||
(defn index []
|
||||
(if (xtdb?) "_id" "id"))
|
||||
|
||||
(defn col-kw [k]
|
||||
(if (xtdb?)
|
||||
(let [n (name k)]
|
||||
(if (= "id" (str/lower-case n)) :_id (keyword n)))
|
||||
k))
|
||||
|
||||
(defn default-options []
|
||||
(if (mssql?) ; so that we get table names back from queries
|
||||
{:result-type :scroll-insensitive :concurrency :read-only}
|
||||
|
|
@ -156,29 +182,54 @@
|
|||
:else
|
||||
"AUTO_INCREMENT PRIMARY KEY")]
|
||||
(with-open [con (jdbc/get-connection (ds))]
|
||||
(when (stored-proc?)
|
||||
(try
|
||||
(jdbc/execute-one! con ["DROP PROCEDURE FRUITP"])
|
||||
(catch Throwable _)))
|
||||
(try
|
||||
(do-commands con [(str "DROP TABLE " fruit)])
|
||||
(catch Exception _))
|
||||
(try
|
||||
(do-commands con [(str "DROP TABLE " btest)])
|
||||
(catch Exception _))
|
||||
(when (postgres?)
|
||||
(try
|
||||
(do-commands con ["DROP TABLE LANG_TEST"])
|
||||
(catch Exception _))
|
||||
(try
|
||||
(do-commands con ["DROP TYPE LANGUAGE"])
|
||||
(catch Exception _))
|
||||
(do-commands con ["CREATE TYPE LANGUAGE AS ENUM('en','fr','de')"])
|
||||
(do-commands con ["
|
||||
(if (xtdb?) ; no DDL for creation
|
||||
(do
|
||||
(try
|
||||
(do-commands con ["ERASE FROM fruit WHERE true"])
|
||||
(catch Throwable _))
|
||||
(try
|
||||
(do-commands con ["ERASE FROM btest WHERE true"])
|
||||
(catch Throwable _))
|
||||
(sql/insert-multi! con :fruit
|
||||
[:_id :name :appearance :cost]
|
||||
[[1 "Apple" "red" 59]]
|
||||
{:return-keys false})
|
||||
(sql/insert-multi! con :fruit
|
||||
[:_id :name :appearance :grade]
|
||||
[[2 "Banana" "yellow" 92.2]]
|
||||
{:return-keys false})
|
||||
(sql/insert-multi! con :fruit
|
||||
[:_id :name :cost :grade]
|
||||
[[3 "Peach" 139 90.0]]
|
||||
{:return-keys false})
|
||||
(sql/insert-multi! con :fruit
|
||||
[:_id :name :appearance :cost :grade]
|
||||
[[4 "Orange" "juicy" 89 88.6]]
|
||||
{:return-keys false}))
|
||||
(do
|
||||
(when (stored-proc?)
|
||||
(try
|
||||
(jdbc/execute-one! con ["DROP PROCEDURE FRUITP"])
|
||||
(catch Throwable _)))
|
||||
(try
|
||||
(do-commands con [(str "DROP TABLE " fruit)])
|
||||
(catch Exception _))
|
||||
(try
|
||||
(do-commands con [(str "DROP TABLE " btest)])
|
||||
(catch Exception _))
|
||||
(when (postgres?)
|
||||
(try
|
||||
(do-commands con ["DROP TABLE LANG_TEST"])
|
||||
(catch Exception _))
|
||||
(try
|
||||
(do-commands con ["DROP TYPE LANGUAGE"])
|
||||
(catch Exception _))
|
||||
(do-commands con ["CREATE TYPE LANGUAGE AS ENUM('en','fr','de')"])
|
||||
(do-commands con ["
|
||||
CREATE TABLE LANG_TEST (
|
||||
LANG LANGUAGE NOT NULL
|
||||
)"]))
|
||||
(do-commands con [(str "
|
||||
(do-commands con [(str "
|
||||
CREATE TABLE " fruit " (
|
||||
ID INTEGER " auto-inc-pk ",
|
||||
NAME VARCHAR(32),
|
||||
|
|
@ -186,28 +237,28 @@ CREATE TABLE " fruit " (
|
|||
COST INT DEFAULT NULL,
|
||||
GRADE REAL DEFAULT NULL
|
||||
)")])
|
||||
(let [created (atom false)]
|
||||
(let [created (atom false)]
|
||||
;; MS SQL Server does not support bool/boolean:
|
||||
(doseq [btype ["BOOL" "BOOLEAN" "BIT"]]
|
||||
(doseq [btype ["BOOL" "BOOLEAN" "BIT"]]
|
||||
;; Derby does not support bit:
|
||||
(doseq [bitty ["BIT" "SMALLINT"]]
|
||||
(try
|
||||
(when-not @created
|
||||
(do-commands con [(str "
|
||||
(doseq [bitty ["BIT" "SMALLINT"]]
|
||||
(try
|
||||
(when-not @created
|
||||
(do-commands con [(str "
|
||||
CREATE TABLE " btest " (
|
||||
NAME VARCHAR(32),
|
||||
IS_IT " btype ",
|
||||
TWIDDLE " bitty "
|
||||
)")])
|
||||
(reset! created true))
|
||||
(catch Throwable _))))
|
||||
(when-not @created
|
||||
(println (:dbtype db) "failed btest creation")
|
||||
#_(throw (ex-info (str (:dbtype db) " has no boolean type?") {}))))
|
||||
(when (stored-proc?)
|
||||
(let [[begin end] (if (postgres?) ["$$" "$$"] ["BEGIN" "END"])]
|
||||
(try
|
||||
(do-commands con [(str "
|
||||
(reset! created true))
|
||||
(catch Throwable _))))
|
||||
(when-not @created
|
||||
(println (:dbtype db) "failed btest creation")
|
||||
#_(throw (ex-info (str (:dbtype db) " has no boolean type?") {}))))
|
||||
(when (stored-proc?)
|
||||
(let [[begin end] (if (postgres?) ["$$" "$$"] ["BEGIN" "END"])]
|
||||
(try
|
||||
(do-commands con [(str "
|
||||
CREATE PROCEDURE FRUITP" (cond (hsqldb?) "() READS SQL DATA DYNAMIC RESULT SETS 2 "
|
||||
(mssql?) " AS "
|
||||
(postgres?) "() LANGUAGE SQL AS "
|
||||
|
|
@ -223,15 +274,15 @@ CREATE PROCEDURE FRUITP" (cond (hsqldb?) "() READS SQL DATA DYNAMIC RESULT SETS
|
|||
SELECT * FROM " fruit " WHERE GRADE >= 90.0;")) "
|
||||
" end "
|
||||
")])
|
||||
(catch Throwable t
|
||||
(println 'procedure (:dbtype db) (ex-message t))))))
|
||||
(sql/insert-multi! con :fruit
|
||||
[:name :appearance :cost :grade]
|
||||
[["Apple" "red" 59 nil]
|
||||
["Banana" "yellow" nil 92.2]
|
||||
["Peach" nil 139 90.0]
|
||||
["Orange" "juicy" 89 88.6]]
|
||||
{:return-keys false})
|
||||
(catch Throwable t
|
||||
(println 'procedure (:dbtype db) (ex-message t))))))
|
||||
(sql/insert-multi! con :fruit
|
||||
[:name :appearance :cost :grade]
|
||||
[["Apple" "red" 59 nil]
|
||||
["Banana" "yellow" nil 92.2]
|
||||
["Peach" nil 139 90.0]
|
||||
["Orange" "juicy" 89 88.6]]
|
||||
{:return-keys false})))
|
||||
(t)))))
|
||||
|
||||
(create-clojure-test)
|
||||
|
|
|
|||
|
|
@ -1,17 +1,10 @@
|
|||
;; copyright (c) 2019-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2019-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.transaction-test
|
||||
"Stub test namespace for transaction handling."
|
||||
(:require [clojure.test :refer [deftest is testing use-fixtures]]
|
||||
[next.jdbc :as jdbc]
|
||||
[next.jdbc.specs :as specs]
|
||||
[next.jdbc.test-fixtures :refer [with-test-db db ds column
|
||||
default-options
|
||||
derby? mssql? mysql? postgres?]]
|
||||
[next.jdbc.transaction :as tx]))
|
||||
(:require [next.jdbc.specs :as specs]
|
||||
[next.jdbc.transaction]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
||||
(use-fixtures :once with-test-db)
|
||||
|
||||
(specs/instrument)
|
||||
|
|
|
|||
|
|
@ -1,14 +1,19 @@
|
|||
;; copyright (c) 2020-2024 Sean Corfield, all rights reserved
|
||||
;; copyright (c) 2020-2025 Sean Corfield, all rights reserved
|
||||
|
||||
(ns next.jdbc.types-test
|
||||
"Some tests for the type-assist functions."
|
||||
(:require [clojure.test :refer [deftest is]]
|
||||
(:require [lazytest.core :refer [defdescribe describe it expect]]
|
||||
[next.jdbc.types :refer [as-varchar]]))
|
||||
|
||||
(set! *warn-on-reflection* true)
|
||||
|
||||
(deftest as-varchar-test
|
||||
(defdescribe as-varchar-tests
|
||||
(let [v (as-varchar "Hello")]
|
||||
(is (= "Hello" (v)))
|
||||
(is (contains? (meta v) 'next.jdbc.prepare/set-parameter))
|
||||
(is (fn? (get (meta v) 'next.jdbc.prepare/set-parameter)))))
|
||||
(describe "produces a function"
|
||||
(it "yields the original value when invoked"
|
||||
(expect (fn? v))
|
||||
(expect (= "Hello" (v)))))
|
||||
(describe "carries metadata"
|
||||
(it "has a `set-parameter` function"
|
||||
(expect (contains? (meta v) 'next.jdbc.prepare/set-parameter))
|
||||
(expect (fn? (get (meta v) 'next.jdbc.prepare/set-parameter)))))))
|
||||
|
|
|
|||
File diff suppressed because it is too large
Load diff
Loading…
Reference in a new issue