diff --git a/test/metabase/http_client.clj b/test/metabase/http_client.clj
index 9cba88574736bddcb9af15af182bbed85bc05848..697a262a0a7890bb41203a9626f288e71d5e6174 100644
--- a/test/metabase/http_client.clj
+++ b/test/metabase/http_client.clj
@@ -2,7 +2,8 @@
   "HTTP client for making API calls against the Metabase API. For test/REPL purposes."
   (:require [clojure.data.json :as json]
             [clj-http.client :as client]
-            [metabase.util :as u]))
+            [metabase.util :as u])
+  (:import com.metabase.corvus.api.ApiException))
 
 (declare authenticate
          auto-deserialize-dates
@@ -16,7 +17,7 @@
   "http://localhost:3000/api/")
 
 (defn
-  ^{:arglists ([credentials-map? method expected-status-code? url http-body-map? & url-kwargs])}
+  ^{:arglists ([credentials? method expected-status-code? url http-body-map? & url-kwargs])}
   client
   "Perform an API call and return the response (for test purposes).
    The first arg after URL will be passed as a JSON-encoded body if it is a map.
@@ -30,14 +31,15 @@
 
   Args:
 
-   *  CREDENTIALS-MAP       Optional map of `:email` and `:password` of a User whose credentials we should perform the request with.
+   *  CREDENTIALS           Optional map of `:email` and `:password` or `X-METABASE-SESSION` token of a User who we should perform the request as
    *  METHOD                `:get`, `:post`, `:delete`, or `:put`
    *  EXPECTED-STATUS-CODE  When passed, throw an exception if the response has a different status code.
    *  URL                   Base URL of the request, which will be appended to `*url-prefix*`. e.g. `card/1/favorite`
    *  HTTP-BODY-MAP         Optional map to send a the JSON-serialized HTTP body of the request
    *  URL-KWARGS            key-value pairs that will be encoded and added to the URL as GET params"
   [& args]
-  (let [[credentials [method & args]] (u/optional map? args)
+  (let [[credentials [method & args]] (u/optional #(or (map? %)
+                                                       (string? %)) args)
         [expected-status [url & args]] (u/optional integer? args)
         [body [& {:as url-param-kwargs}]] (u/optional map? args)]
     (-client credentials method expected-status url body url-param-kwargs)))
@@ -48,7 +50,8 @@
 (defn- -client [credentials method expected-status url http-body url-param-kwargs]
   ;; Since the params for this function can get a little complicated make sure we validate them
   {:pre [(or (nil? credentials)
-             (map? credentials))
+             (map? credentials)
+             (string? credentials))
          (contains? #{:get :post :put :delete} method)
          (or (nil? expected-status)
              (integer? expected-status))
@@ -60,7 +63,8 @@
 
   (let [request-map {:content-type :json
                      :accept :json
-                     :headers {"X-METABASE-SESSION" (when credentials (authenticate credentials))}
+                     :headers {"X-METABASE-SESSION" (when credentials (if (map? credentials) (authenticate credentials)
+                                                                          credentials))}
                      :body (json/write-str http-body)}
         request-fn (case method
                      :get  client/get
@@ -87,7 +91,7 @@
                                                          clojure.walk/keywordize-keys)
                                                      (catch Exception _ body))]
           (clojure.pprint/pprint body)
-          (throw (Exception. message)))))
+          (throw (ApiException. status message)))))
 
     ;; Deserialize the JSON response or return as-is if that fails
     (try (-> body
@@ -97,7 +101,7 @@
          (catch Exception _
            body))))
 
-(defn- authenticate [{:keys [email password] :as credentials}]
+(defn authenticate [{:keys [email password] :as credentials}]
   {:pre [(string? email)
          (string? password)]}
   (try
diff --git a/test/metabase/test_data.clj b/test/metabase/test_data.clj
index 6d5b5f9471a063f6cfbc2065e0f55bdc289fd19d..5d89201c26dc4c7e45197084d8525c9c3ced780c 100644
--- a/test/metabase/test_data.clj
+++ b/test/metabase/test_data.clj
@@ -8,7 +8,8 @@
                              [org-perm :refer [OrgPerm]]
                              [table :refer [Table]]
                              [user :refer [User]])
-            [metabase.test-data.load :as load]))
+            [metabase.test-data.load :as load])
+  (:import com.metabase.corvus.api.ApiException))
 
 (declare fetch-or-create-user
          tables
@@ -122,17 +123,28 @@
      {:pre [(contains? usernames username)]}
      (:id (fetch-user username)))))
 
-(def user->client
-  "Returns a `metabase.http-client/client` partially bound with the credentials for User with USERNAME.
-   In addition, it forces lazy creation of the User if needed.
-
-    ((user->client) :get 200 \"meta/table\")"
-  (memoize
-   (fn [username]
-     {:pre [(contains? usernames username)]}
-     ;; Force lazy creation of User if need be
-     (user->id username)
-     (partial http/client (user->credentials username)))))
+(let [tokens (atom {})
+      user->token (fn [user]
+                    (or (@tokens user)
+                        (let [token (http/authenticate (user->credentials user))]
+                          (swap! tokens assoc user token)
+                          token)))]
+  (defn user->client
+    "Returns a `metabase.http-client/client` partially bound with the credentials for User with USERNAME.
+     In addition, it forces lazy creation of the User if needed.
+
+       ((user->client) :get 200 \"meta/table\")"
+    [username]
+    ;; Force lazy creation of User if need be
+    (user->id username)
+    (fn call-client [& args]
+      (try
+        (apply http/client (user->token username) args)
+        (catch ApiException e
+          (if-not (= (.getStatusCode e) 401) (throw e)
+                  ;; If we got a 401 unauthenticated clear the tokens cache + recur
+                  (do (reset! tokens {})
+                      (apply call-client args))))))))
 
 (defn user->org-perm
   "Return the `OrgPerm` for User with USERNAME for the Test Org."