2019-03-31 23:54:34 +00:00
|
|
|
;; copyright (c) 2018-2019 Sean Corfield, all rights reserved
|
|
|
|
|
|
|
|
|
|
(ns next.jdbc.connection
|
2019-04-21 23:13:52 +00:00
|
|
|
"Standard implementations of `get-datasource` and `get-connection`."
|
2019-03-31 23:54:34 +00:00
|
|
|
(:require [next.jdbc.protocols :as p])
|
|
|
|
|
(:import (java.sql Connection DriverManager)
|
|
|
|
|
(javax.sql DataSource)
|
|
|
|
|
(java.util Properties)))
|
|
|
|
|
|
|
|
|
|
(set! *warn-on-reflection* true)
|
|
|
|
|
|
|
|
|
|
(def ^:private classnames
|
2019-04-21 23:13:52 +00:00
|
|
|
"Map of subprotocols to classnames. `:dbtype` specifies one of these keys.
|
2019-03-31 23:54:34 +00:00
|
|
|
|
2019-04-21 23:13:52 +00:00
|
|
|
The subprotocols map below provides aliases for `:dbtype`.
|
2019-03-31 23:54:34 +00:00
|
|
|
|
|
|
|
|
Most databases have just a single class name for their driver but we
|
|
|
|
|
support a sequence of class names to try in order to allow for drivers
|
|
|
|
|
that change their names over time (e.g., MySQL)."
|
|
|
|
|
{"derby" "org.apache.derby.jdbc.EmbeddedDriver"
|
|
|
|
|
"h2" "org.h2.Driver"
|
|
|
|
|
"h2:mem" "org.h2.Driver"
|
|
|
|
|
"hsqldb" "org.hsqldb.jdbcDriver"
|
|
|
|
|
"jtds:sqlserver" "net.sourceforge.jtds.jdbc.Driver"
|
|
|
|
|
"mysql" ["com.mysql.cj.jdbc.Driver"
|
|
|
|
|
"com.mysql.jdbc.Driver"]
|
|
|
|
|
"oracle:oci" "oracle.jdbc.OracleDriver"
|
|
|
|
|
"oracle:thin" "oracle.jdbc.OracleDriver"
|
|
|
|
|
"postgresql" "org.postgresql.Driver"
|
|
|
|
|
"pgsql" "com.impossibl.postgres.jdbc.PGDriver"
|
|
|
|
|
"redshift" "com.amazon.redshift.jdbc.Driver"
|
|
|
|
|
"sqlite" "org.sqlite.JDBC"
|
2019-07-11 02:32:24 +00:00
|
|
|
"sqlserver" "com.microsoft.sqlserver.jdbc.SQLServerDriver"
|
|
|
|
|
"timesten:client" "com.timesten.jdbc.TimesTenClientDriver"
|
|
|
|
|
"timesten:direct" "com.timesten.jdbc.TimesTenDriver"})
|
2019-03-31 23:54:34 +00:00
|
|
|
|
|
|
|
|
(def ^:private aliases
|
2019-04-21 23:13:52 +00:00
|
|
|
"Map of schemes to subprotocols. Used to provide aliases for `:dbtype`."
|
2019-03-31 23:54:34 +00:00
|
|
|
{"hsql" "hsqldb"
|
|
|
|
|
"jtds" "jtds:sqlserver"
|
|
|
|
|
"mssql" "sqlserver"
|
|
|
|
|
"oracle" "oracle:thin"
|
|
|
|
|
"oracle:sid" "oracle:thin"
|
|
|
|
|
"postgres" "postgresql"})
|
|
|
|
|
|
|
|
|
|
(def ^:private host-prefixes
|
|
|
|
|
"Map of subprotocols to non-standard host-prefixes.
|
2019-04-21 23:13:52 +00:00
|
|
|
Anything not listed is assumed to use `//`."
|
2019-03-31 23:54:34 +00:00
|
|
|
{"oracle:oci" "@"
|
|
|
|
|
"oracle:thin" "@"})
|
|
|
|
|
|
|
|
|
|
(def ^:private ports
|
|
|
|
|
"Map of subprotocols to ports."
|
|
|
|
|
{"jtds:sqlserver" 1433
|
|
|
|
|
"mysql" 3306
|
|
|
|
|
"oracle:oci" 1521
|
|
|
|
|
"oracle:sid" 1521
|
|
|
|
|
"oracle:thin" 1521
|
|
|
|
|
"postgresql" 5432
|
|
|
|
|
"sqlserver" 1433})
|
|
|
|
|
|
|
|
|
|
(def ^:private dbname-separators
|
2019-04-21 23:13:52 +00:00
|
|
|
"Map of schemes to separators. The default is `/` but a couple are different."
|
2019-03-31 23:54:34 +00:00
|
|
|
{"mssql" ";DATABASENAME="
|
|
|
|
|
"sqlserver" ";DATABASENAME="
|
2019-07-11 02:32:24 +00:00
|
|
|
"oracle:sid" ":"
|
2019-07-11 02:47:19 +00:00
|
|
|
"timesten:client" ":dsn="
|
2019-07-11 02:32:24 +00:00
|
|
|
"timesten:direct" ":dsn="})
|
2019-03-31 23:54:34 +00:00
|
|
|
|
2019-07-10 18:54:13 +00:00
|
|
|
(def dbtypes
|
|
|
|
|
"A map of all known database types (including aliases) to the class name(s)
|
|
|
|
|
and port that `next.jdbc` supports out of the box. Just for completeness,
|
|
|
|
|
this also includes the prefixes used in the JDBC string for the `:host`
|
|
|
|
|
and `:dbname` (which are `//` and `/` respectively for nearly all types).
|
|
|
|
|
|
|
|
|
|
For known database types, you can use `:dbtype` (and omit `:classname`).
|
|
|
|
|
|
|
|
|
|
If you want to use a database that is not in this list, you can specify
|
|
|
|
|
a new `:dbtype` along with the class name of the JDBC driver in `:classname`.
|
|
|
|
|
You will also need to specify `:port`. For example:
|
|
|
|
|
|
|
|
|
|
`{:dbtype \"acme\" :classname \"com.acme.JdbcDriver\" ...}`
|
|
|
|
|
|
|
|
|
|
JDBC drivers are not provided by `next.jdbc` -- you need to specify the
|
|
|
|
|
driver(s) you need as additional dependencies in your project. For
|
|
|
|
|
example:
|
|
|
|
|
|
|
|
|
|
`[com.acme/jdbc \"1.2.3\"] ; lein/boot`
|
|
|
|
|
or:
|
|
|
|
|
`{com.acme/jdbc {:mvn/version \"1.2.3\"}} ; CLI/deps.edn`
|
|
|
|
|
|
|
|
|
|
Note: the `:classname` value can be a string or a vector of strings. If
|
|
|
|
|
a vector of strings is provided, an attempt will be made to load each
|
|
|
|
|
named class in order, until one succeeds. This allows for a given `:dbtype`
|
|
|
|
|
to be used with different versions of a JDBC driver, if the class name
|
|
|
|
|
has changed over time (such as with MySQL)."
|
|
|
|
|
(let [dbs (merge (zipmap (keys classnames) (keys classnames)) aliases)]
|
|
|
|
|
(reduce-kv (fn [m k v]
|
|
|
|
|
(assoc m
|
|
|
|
|
k
|
|
|
|
|
(cond-> {:classname (classnames v)
|
|
|
|
|
:host-prefix (host-prefixes v "//")
|
|
|
|
|
:dbname-separator (dbname-separators v "/")}
|
|
|
|
|
(ports v)
|
|
|
|
|
(assoc :port (ports v)))))
|
|
|
|
|
{}
|
|
|
|
|
dbs)))
|
|
|
|
|
|
2019-03-31 23:54:34 +00:00
|
|
|
(defn- ^Properties as-properties
|
2019-04-21 23:13:52 +00:00
|
|
|
"Convert any seq of pairs to a `java.util.Properties` instance."
|
2019-03-31 23:54:34 +00:00
|
|
|
[m]
|
|
|
|
|
(let [p (Properties.)]
|
|
|
|
|
(doseq [[k v] m]
|
|
|
|
|
(.setProperty p (name k) (str v)))
|
|
|
|
|
p))
|
|
|
|
|
|
|
|
|
|
(defn- get-driver-connection
|
2019-05-08 22:11:37 +00:00
|
|
|
"Common logic for loading the designated JDBC driver class and
|
|
|
|
|
obtaining the appropriate `Connection` object."
|
2019-03-31 23:54:34 +00:00
|
|
|
[url etc]
|
|
|
|
|
(DriverManager/getConnection url (as-properties etc)))
|
|
|
|
|
|
2019-05-08 22:11:37 +00:00
|
|
|
(def ^:private driver-cache
|
|
|
|
|
"An optimization for repeated calls to get-datasource, or for get-connection
|
|
|
|
|
called on a db-spec hash map, so that we only try to load the classes once."
|
|
|
|
|
(atom {}))
|
|
|
|
|
|
2019-03-31 23:54:34 +00:00
|
|
|
(defn- spec->url+etc
|
2019-04-01 06:17:12 +00:00
|
|
|
"Given a database spec, return a JDBC URL and a map of any additional options."
|
2019-03-31 23:54:34 +00:00
|
|
|
[{:keys [dbtype dbname host port classname] :as db-spec}]
|
|
|
|
|
(let [;; allow aliases for dbtype
|
|
|
|
|
subprotocol (aliases dbtype dbtype)
|
|
|
|
|
host (or host "127.0.0.1")
|
|
|
|
|
port (or port (ports subprotocol))
|
|
|
|
|
db-sep (dbname-separators dbtype "/")
|
|
|
|
|
url (cond (= "h2:mem" dbtype)
|
|
|
|
|
(str "jdbc:" subprotocol ":" dbname ";DB_CLOSE_DELAY=-1")
|
2019-04-19 04:35:38 +00:00
|
|
|
(#{"h2"} subprotocol)
|
|
|
|
|
(str "jdbc:" subprotocol ":"
|
|
|
|
|
(if (re-find #"^([A-Za-z]:)?[\./\\]" dbname)
|
|
|
|
|
;; DB name starts with relative or absolute path
|
|
|
|
|
dbname
|
|
|
|
|
;; otherwise make it local
|
|
|
|
|
(str "./" dbname)))
|
|
|
|
|
(#{"derby" "hsqldb" "sqlite"} subprotocol)
|
2019-03-31 23:54:34 +00:00
|
|
|
(str "jdbc:" subprotocol ":" dbname)
|
2019-07-11 03:00:40 +00:00
|
|
|
(#{"timesten:direct"} subprotocol)
|
|
|
|
|
(str "jdbc:" subprotocol db-sep dbname)
|
2019-03-31 23:54:34 +00:00
|
|
|
:else
|
|
|
|
|
(str "jdbc:" subprotocol ":"
|
|
|
|
|
(host-prefixes subprotocol "//")
|
|
|
|
|
host
|
|
|
|
|
(when port (str ":" port))
|
|
|
|
|
db-sep dbname))
|
2019-05-05 06:59:48 +00:00
|
|
|
etc (dissoc db-spec :dbtype :dbname :host :port :classname)]
|
2019-03-31 23:54:34 +00:00
|
|
|
;; verify the datasource is loadable
|
|
|
|
|
(if-let [class-name (or classname (classnames subprotocol))]
|
2019-05-08 22:11:37 +00:00
|
|
|
(swap! driver-cache update class-name
|
|
|
|
|
#(if % %
|
|
|
|
|
(do
|
|
|
|
|
;; force DriverManager to be loaded
|
|
|
|
|
(DriverManager/getLoginTimeout)
|
|
|
|
|
(if (string? class-name)
|
|
|
|
|
(clojure.lang.RT/loadClassForName class-name)
|
|
|
|
|
(loop [[clazz & more] class-name]
|
|
|
|
|
(let [loaded
|
|
|
|
|
(try
|
|
|
|
|
(clojure.lang.RT/loadClassForName clazz)
|
|
|
|
|
(catch Exception e
|
|
|
|
|
e))]
|
|
|
|
|
(if (instance? Throwable loaded)
|
|
|
|
|
(if (seq more)
|
|
|
|
|
(recur more)
|
|
|
|
|
(throw loaded))
|
|
|
|
|
loaded)))))))
|
2019-03-31 23:54:34 +00:00
|
|
|
(throw (ex-info (str "Unknown dbtype: " dbtype) db-spec)))
|
|
|
|
|
[url etc]))
|
2019-05-23 17:05:27 +00:00
|
|
|
|
2019-03-31 23:54:34 +00:00
|
|
|
(defn- string->url+etc
|
2019-04-01 06:17:12 +00:00
|
|
|
"Given a JDBC URL, return it with an empty set of options with no parsing."
|
2019-03-31 23:54:34 +00:00
|
|
|
[s]
|
|
|
|
|
[s {}])
|
|
|
|
|
|
|
|
|
|
(defn- url+etc->datasource
|
2019-04-21 23:13:52 +00:00
|
|
|
"Given a JDBC URL and a map of options, return a `DataSource` that can be
|
2019-04-01 06:17:12 +00:00
|
|
|
used to obtain a new database connection."
|
2019-03-31 23:54:34 +00:00
|
|
|
[[url etc]]
|
|
|
|
|
(reify DataSource
|
|
|
|
|
(getConnection [_]
|
|
|
|
|
(get-driver-connection url etc))
|
|
|
|
|
(getConnection [_ username password]
|
|
|
|
|
(get-driver-connection url
|
|
|
|
|
(assoc etc
|
2019-04-20 23:03:01 +00:00
|
|
|
:user username
|
2019-06-11 23:47:58 +00:00
|
|
|
:password password)))
|
|
|
|
|
(toString [_] url)))
|
2019-03-31 23:54:34 +00:00
|
|
|
|
|
|
|
|
(defn- make-connection
|
2019-04-21 23:13:52 +00:00
|
|
|
"Given a `DataSource` and a map of options, get a connection and update it
|
2019-04-01 06:17:12 +00:00
|
|
|
as specified by the options.
|
|
|
|
|
|
|
|
|
|
The options supported are:
|
2019-04-21 23:13:52 +00:00
|
|
|
* `:auto-commit` -- whether the connection should be set to auto-commit or not;
|
|
|
|
|
without this option, the defaut is `true` -- connections will auto-commit,
|
|
|
|
|
* `:read-only` -- whether the connection should be set to read-only mode."
|
2019-03-31 23:54:34 +00:00
|
|
|
^Connection
|
|
|
|
|
[^DataSource datasource opts]
|
|
|
|
|
(let [^Connection connection (.getConnection datasource)]
|
2019-04-01 06:17:12 +00:00
|
|
|
(when (contains? opts :auto-commit)
|
|
|
|
|
(.setAutoCommit connection (boolean (:auto-commit opts))))
|
|
|
|
|
(when (contains? opts :read-only)
|
|
|
|
|
(.setReadOnly connection (boolean (:read-only opts))))
|
2019-03-31 23:54:34 +00:00
|
|
|
connection))
|
|
|
|
|
|
|
|
|
|
(extend-protocol p/Sourceable
|
|
|
|
|
clojure.lang.Associative
|
|
|
|
|
(get-datasource [this]
|
|
|
|
|
(url+etc->datasource (spec->url+etc this)))
|
|
|
|
|
javax.sql.DataSource
|
|
|
|
|
(get-datasource [this] this)
|
|
|
|
|
String
|
|
|
|
|
(get-datasource [this]
|
|
|
|
|
(url+etc->datasource (string->url+etc this))))
|
|
|
|
|
|
|
|
|
|
(extend-protocol p/Connectable
|
|
|
|
|
javax.sql.DataSource
|
|
|
|
|
(get-connection [this opts] (make-connection this opts))
|
2019-04-18 06:34:31 +00:00
|
|
|
java.sql.PreparedStatement
|
|
|
|
|
;; note: options are ignored and this should not be closed independently
|
|
|
|
|
;; of the PreparedStatement to which it belongs: this done to allow
|
|
|
|
|
;; datafy/nav across a PreparedStatement only...
|
|
|
|
|
(get-connection [this _] (.getConnection this))
|
2019-03-31 23:54:34 +00:00
|
|
|
Object
|
|
|
|
|
(get-connection [this opts] (p/get-connection (p/get-datasource this) opts)))
|