From af1593f952fc338f9085422438ba9fd7e14bda8c Mon Sep 17 00:00:00 2001 From: Ryan Senior <ryan@metabase.com> Date: Fri, 9 Nov 2018 11:27:54 -0600 Subject: [PATCH] Fix unspecified PG database connection strings This issue can occur if the `MB_DB_DBNAME` variable has not been specified. The `:dbname` key has a nil value which causes Ring's `form-encode` function to fail. This commit explicitly removes the `:dbname` as it's extra anyway and will also remove map entries that have a nil key or value before encoding it. --- src/metabase/db/spec.clj | 9 ++++++--- test/metabase/db/spec_test.clj | 36 ++++++++++++++++++++++++++++++++++ 2 files changed, 42 insertions(+), 3 deletions(-) create mode 100644 test/metabase/db/spec_test.clj diff --git a/src/metabase/db/spec.clj b/src/metabase/db/spec.clj index 68f9acd5a39..ba09f511f30 100644 --- a/src/metabase/db/spec.clj +++ b/src/metabase/db/spec.clj @@ -3,9 +3,9 @@ Only databases that are supported as application DBs should have functions in this namespace; otherwise, similar functions are only needed by drivers, and belong in those namespaces." (:require [clojure.string :as str] + [medley.core :as m] [ring.util.codec :as codec])) - (defn h2 "Create a database specification for a h2 database. Opts should include a key for :db which is the path to the database file." @@ -18,11 +18,14 @@ (dissoc opts :db))) (defn- remove-required-keys [db-spec & more-keys] - (apply dissoc db-spec (concat [:host :port :db :user :password :additional-options] + (apply dissoc db-spec (concat [:host :port :db :dbname :user :password :additional-options] more-keys))) +(defn- purge-nil-values [m] + (m/remove-kv (fn [k v] (or (nil? k) (nil? v))) m)) + (defn- make-subname [host port db extra-connection-params] - (let [query-params (codec/form-encode extra-connection-params)] + (let [query-params (codec/form-encode (purge-nil-values extra-connection-params))] (str "//" host ":" port "/" db (when-not (str/blank? query-params) (str "?" query-params))))) diff --git a/test/metabase/db/spec_test.clj b/test/metabase/db/spec_test.clj new file mode 100644 index 00000000000..fcbe3584b34 --- /dev/null +++ b/test/metabase/db/spec_test.clj @@ -0,0 +1,36 @@ +(ns metabase.db.spec-test + (:require [expectations :refer :all] + [metabase.db.spec :refer :all])) + +(defn- default-pg-spec [db] + {:classname "org.postgresql.Driver", :subprotocol "postgresql", + :subname (format "//localhost:5432/%s?OpenSourceSubProtocolOverride=true" db)}) + +;; Basic minimal config +(expect + (default-pg-spec "metabase") + (postgres {:host "localhost" + :port 5432 + :db "metabase"})) + +;; Users that don't specify a `:dbname` (and thus no `:db`) will use the user's default, we should allow that +(expect + (assoc (default-pg-spec "") :dbname nil) + (postgres {:host "localhost" + :port 5432 + :dbname nil + :db nil})) + +;; We should be tolerant of other random nil values sneaking through +(expect + (assoc (default-pg-spec "") :dbname nil, :somethingrandom nil) + (postgres {:host "localhost" + :port 5432 + :dbname nil + :db nil + :somethingrandom nil})) + +;; Not specifying any of the values results in defaults +(expect + (default-pg-spec "") + (postgres {})) -- GitLab