2019-04-18 21:15:15 +00:00
|
|
|
;; copyright (c) 2019 Sean Corfield, all rights reserved
|
|
|
|
|
|
|
|
|
|
(ns next.jdbc.result-set-test
|
2019-05-26 02:16:30 +00:00
|
|
|
"Test namespace for the result set functions.
|
2019-04-19 04:51:58 +00:00
|
|
|
|
2019-04-20 04:59:42 +00:00
|
|
|
What's left to be tested:
|
|
|
|
|
* ReadableColumn protocol extension point"
|
2019-04-20 04:53:55 +00:00
|
|
|
(:require [clojure.core.protocols :as core-p]
|
|
|
|
|
[clojure.datafy :as d]
|
2019-04-19 06:03:09 +00:00
|
|
|
[clojure.string :as str]
|
2019-04-19 05:43:19 +00:00
|
|
|
[clojure.test :refer [deftest is testing use-fixtures]]
|
2019-04-19 06:03:09 +00:00
|
|
|
[next.jdbc.protocols :as p]
|
|
|
|
|
[next.jdbc.result-set :as rs]
|
2019-04-20 04:53:55 +00:00
|
|
|
[next.jdbc.test-fixtures :refer [with-test-db ds]])
|
|
|
|
|
(:import (java.sql ResultSet ResultSetMetaData)))
|
2019-04-19 05:43:19 +00:00
|
|
|
|
2019-05-29 16:04:21 +00:00
|
|
|
(set! *warn-on-reflection* true)
|
|
|
|
|
|
2019-04-19 05:43:19 +00:00
|
|
|
(use-fixtures :once with-test-db)
|
|
|
|
|
|
|
|
|
|
(deftest test-datafy-nav
|
|
|
|
|
(testing "default schema"
|
|
|
|
|
(let [connectable (ds)
|
2019-04-22 00:10:29 +00:00
|
|
|
test-row (rs/datafiable-row {:TABLE/FRUIT_ID 1} connectable {})
|
2019-04-19 05:43:19 +00:00
|
|
|
data (d/datafy test-row)
|
2019-04-22 00:10:29 +00:00
|
|
|
v (get data :TABLE/FRUIT_ID)]
|
2019-04-19 05:43:19 +00:00
|
|
|
;; check datafication is sane
|
2019-04-19 06:03:09 +00:00
|
|
|
(is (= 1 v))
|
2019-04-19 05:43:19 +00:00
|
|
|
(let [object (d/nav data :table/fruit_id v)]
|
|
|
|
|
;; check nav produces a single map with the expected key/value data
|
|
|
|
|
;; and remember H2 is all UPPERCASE!
|
2019-04-19 06:03:09 +00:00
|
|
|
(is (= 1 (:FRUIT/ID object)))
|
|
|
|
|
(is (= "Apple" (:FRUIT/NAME object))))))
|
2019-04-19 05:43:19 +00:00
|
|
|
(testing "custom schema :one"
|
|
|
|
|
(let [connectable (ds)
|
|
|
|
|
test-row (rs/datafiable-row {:foo/bar 2} connectable
|
|
|
|
|
{:schema {:foo/bar [:fruit :id]}})
|
|
|
|
|
data (d/datafy test-row)
|
|
|
|
|
v (get data :foo/bar)]
|
|
|
|
|
;; check datafication is sane
|
|
|
|
|
(is (= 2 v))
|
|
|
|
|
(let [object (d/nav data :foo/bar v)]
|
|
|
|
|
;; check nav produces a single map with the expected key/value data
|
|
|
|
|
;; and remember H2 is all UPPERCASE!
|
|
|
|
|
(is (= 2 (:FRUIT/ID object)))
|
|
|
|
|
(is (= "Banana" (:FRUIT/NAME object))))))
|
|
|
|
|
(testing "custom schema :many"
|
|
|
|
|
(let [connectable (ds)
|
2019-04-19 06:03:09 +00:00
|
|
|
test-row (rs/datafiable-row {:foo/bar 3} connectable
|
2019-04-19 05:43:19 +00:00
|
|
|
{:schema {:foo/bar [:fruit :id :many]}})
|
|
|
|
|
data (d/datafy test-row)
|
|
|
|
|
v (get data :foo/bar)]
|
|
|
|
|
;; check datafication is sane
|
2019-04-19 06:03:09 +00:00
|
|
|
(is (= 3 v))
|
2019-04-19 05:43:19 +00:00
|
|
|
(let [object (d/nav data :foo/bar v)]
|
|
|
|
|
;; check nav produces a result set with the expected key/value data
|
|
|
|
|
;; and remember H2 is all UPPERCASE!
|
|
|
|
|
(is (vector? object))
|
2019-04-19 06:03:09 +00:00
|
|
|
(is (= 3 (:FRUIT/ID (first object))))
|
|
|
|
|
(is (= "Peach" (:FRUIT/NAME (first object))))))))
|
|
|
|
|
|
|
|
|
|
(deftest test-map-row-builder
|
|
|
|
|
(testing "default row builder"
|
|
|
|
|
(let [row (p/-execute-one (ds)
|
|
|
|
|
["select * from fruit where id = ?" 1]
|
|
|
|
|
{})]
|
|
|
|
|
(is (map? row))
|
2019-05-26 02:16:30 +00:00
|
|
|
(is (contains? row :FRUIT/GRADE))
|
|
|
|
|
(is (nil? (:FRUIT/GRADE row)))
|
2019-04-19 06:03:09 +00:00
|
|
|
(is (= 1 (:FRUIT/ID row)))
|
2019-04-20 04:59:42 +00:00
|
|
|
(is (= "Apple" (:FRUIT/NAME row))))
|
|
|
|
|
(let [rs (p/-execute-all (ds)
|
|
|
|
|
["select * from fruit order by id"]
|
|
|
|
|
{})]
|
|
|
|
|
(is (every? map? rs))
|
|
|
|
|
(is (= 1 (:FRUIT/ID (first rs))))
|
|
|
|
|
(is (= "Apple" (:FRUIT/NAME (first rs))))
|
|
|
|
|
(is (= 4 (:FRUIT/ID (last rs))))
|
|
|
|
|
(is (= "Orange" (:FRUIT/NAME (last rs))))))
|
2019-04-19 06:03:09 +00:00
|
|
|
(testing "unqualified row builder"
|
|
|
|
|
(let [row (p/-execute-one (ds)
|
|
|
|
|
["select * from fruit where id = ?" 2]
|
2019-04-24 21:22:35 +00:00
|
|
|
{:builder-fn rs/as-unqualified-maps})]
|
2019-04-19 06:03:09 +00:00
|
|
|
(is (map? row))
|
2019-05-26 02:16:30 +00:00
|
|
|
(is (contains? row :COST))
|
|
|
|
|
(is (nil? (:COST row)))
|
2019-04-19 06:03:09 +00:00
|
|
|
(is (= 2 (:ID row)))
|
|
|
|
|
(is (= "Banana" (:NAME row)))))
|
|
|
|
|
(testing "lower-case row builder"
|
|
|
|
|
(let [row (p/-execute-one (ds)
|
|
|
|
|
["select * from fruit where id = ?" 3]
|
2019-04-24 21:22:35 +00:00
|
|
|
{:builder-fn rs/as-lower-maps})]
|
2019-04-19 06:03:09 +00:00
|
|
|
(is (map? row))
|
2019-05-26 02:16:30 +00:00
|
|
|
(is (contains? row :fruit/appearance))
|
|
|
|
|
(is (nil? (:fruit/appearance row)))
|
2019-04-21 06:42:22 +00:00
|
|
|
(is (= 3 (:fruit/id row)))
|
|
|
|
|
(is (= "Peach" (:fruit/name row)))))
|
2019-06-05 01:01:19 +00:00
|
|
|
(testing "unqualified lower-case row builder"
|
2019-04-21 06:42:22 +00:00
|
|
|
(let [row (p/-execute-one (ds)
|
|
|
|
|
["select * from fruit where id = ?" 4]
|
2019-04-24 21:22:35 +00:00
|
|
|
{:builder-fn rs/as-unqualified-lower-maps})]
|
2019-04-21 06:42:22 +00:00
|
|
|
(is (map? row))
|
|
|
|
|
(is (= 4 (:id row)))
|
2019-06-05 01:01:19 +00:00
|
|
|
(is (= "Orange" (:name row)))))
|
|
|
|
|
(testing "custom row builder"
|
|
|
|
|
(let [row (p/-execute-one (ds)
|
|
|
|
|
["select * from fruit where id = ?" 3]
|
|
|
|
|
{:builder-fn rs/as-modified-maps
|
|
|
|
|
:label-fn str/lower-case
|
|
|
|
|
:qualifier-fn identity})]
|
|
|
|
|
(is (map? row))
|
|
|
|
|
(is (contains? row :FRUIT/appearance))
|
|
|
|
|
(is (nil? (:FRUIT/appearance row)))
|
|
|
|
|
(is (= 3 (:FRUIT/id row)))
|
|
|
|
|
(is (= "Peach" (:FRUIT/name row))))))
|
2019-04-20 04:53:55 +00:00
|
|
|
|
|
|
|
|
(deftest test-mapify
|
|
|
|
|
(testing "no row builder is used"
|
|
|
|
|
(is (= [false]
|
|
|
|
|
(into [] (map map?) ; it is not a real map
|
|
|
|
|
(p/-execute (ds) ["select * from fruit where id = ?" 1]
|
2019-04-24 21:22:35 +00:00
|
|
|
{:builder-fn (constantly nil)}))))
|
2019-04-20 04:53:55 +00:00
|
|
|
(is (= ["Apple"]
|
|
|
|
|
(into [] (map :name) ; but keyword selection works
|
|
|
|
|
(p/-execute (ds) ["select * from fruit where id = ?" 1]
|
2019-04-24 21:22:35 +00:00
|
|
|
{:builder-fn (constantly nil)}))))
|
2019-04-20 04:53:55 +00:00
|
|
|
(is (= [[2 [:name "Banana"]]]
|
|
|
|
|
(into [] (map (juxt #(get % "id") ; get by string key works
|
|
|
|
|
#(find % :name))) ; get MapEntry works
|
|
|
|
|
(p/-execute (ds) ["select * from fruit where id = ?" 2]
|
2019-04-24 21:22:35 +00:00
|
|
|
{:builder-fn (constantly nil)}))))
|
2019-04-20 04:53:55 +00:00
|
|
|
(is (= [{:id 3 :name "Peach"}]
|
|
|
|
|
(into [] (map #(select-keys % [:id :name])) ; select-keys works
|
|
|
|
|
(p/-execute (ds) ["select * from fruit where id = ?" 3]
|
2019-04-24 21:22:35 +00:00
|
|
|
{:builder-fn (constantly nil)}))))
|
2019-04-20 04:53:55 +00:00
|
|
|
(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]
|
2019-04-24 21:22:35 +00:00
|
|
|
{:builder-fn (constantly nil)})))))
|
2019-04-20 04:53:55 +00:00
|
|
|
(testing "assoc and seq build maps"
|
|
|
|
|
(is (map? (reduce (fn [_ row] (reduced (assoc row :x 1)))
|
|
|
|
|
nil
|
|
|
|
|
(p/-execute (ds) ["select * from fruit"] {}))))
|
|
|
|
|
(is (seq? (reduce (fn [_ row] (reduced (seq row)))
|
|
|
|
|
nil
|
|
|
|
|
(p/-execute (ds) ["select * from fruit"] {}))))
|
|
|
|
|
(is (every? map-entry? (reduce (fn [_ row] (reduced (seq row)))
|
|
|
|
|
nil
|
|
|
|
|
(p/-execute (ds) ["select * from fruit"] {})))))
|
|
|
|
|
(testing "datafiable-row builds map; with metadata"
|
|
|
|
|
(is (map? (reduce (fn [_ row] (reduced (rs/datafiable-row row (ds) {})))
|
|
|
|
|
nil
|
|
|
|
|
(p/-execute (ds) ["select * from fruit"] {}))))
|
|
|
|
|
(is (contains? (meta (reduce (fn [_ row] (reduced (rs/datafiable-row row (ds) {})))
|
|
|
|
|
nil
|
|
|
|
|
(p/-execute (ds) ["select * from fruit"] {})))
|
|
|
|
|
`core-p/datafy))))
|
2019-04-20 05:50:29 +00:00
|
|
|
|
|
|
|
|
;; test that we can create a record-based result set builder:
|
|
|
|
|
|
|
|
|
|
(defrecord Fruit [id name appearance cost grade])
|
|
|
|
|
|
|
|
|
|
(defn fruit-builder [^ResultSet rs opts]
|
|
|
|
|
(reify
|
|
|
|
|
rs/RowBuilder
|
|
|
|
|
(->row [_] (->Fruit (.getObject rs "id")
|
|
|
|
|
(.getObject rs "name")
|
|
|
|
|
(.getObject rs "appearance")
|
|
|
|
|
(.getObject rs "cost")
|
|
|
|
|
(.getObject rs "grade")))
|
|
|
|
|
(with-column [_ row i] row)
|
|
|
|
|
(column-count [_] 0) ; no need to iterate over columns
|
|
|
|
|
(row! [_ row] row)
|
|
|
|
|
rs/ResultSetBuilder
|
|
|
|
|
(->rs [_] (transient []))
|
|
|
|
|
(with-row [_ rs row] (conj! rs row))
|
|
|
|
|
(rs! [_ rs] (persistent! rs))))
|
|
|
|
|
|
|
|
|
|
(deftest custom-map-builder
|
|
|
|
|
(let [row (p/-execute-one (ds)
|
|
|
|
|
["select * from fruit where appearance = ?" "red"]
|
2019-04-24 21:22:35 +00:00
|
|
|
{:builder-fn fruit-builder})]
|
2019-04-20 05:50:29 +00:00
|
|
|
(is (instance? Fruit row))
|
|
|
|
|
(is (= 1 (:id row))))
|
|
|
|
|
(let [rs (p/-execute-all (ds)
|
|
|
|
|
["select * from fruit where appearance = ?" "red"]
|
2019-04-24 21:22:35 +00:00
|
|
|
{:builder-fn fruit-builder})]
|
2019-04-20 05:50:29 +00:00
|
|
|
(is (every? #(instance? Fruit %) rs))
|
|
|
|
|
(is (= 1 (count rs)))
|
|
|
|
|
(is (= 1 (:id (first rs))))))
|