diff --git a/modules/drivers/druid/src/metabase/driver/druid.clj b/modules/drivers/druid/src/metabase/driver/druid.clj
index 9f4dc39a642623d98394666fe4e7eefd233ecd44..906ed8441ecec55df05574460e06797d81d3aff3 100644
--- a/modules/drivers/druid/src/metabase/driver/druid.clj
+++ b/modules/drivers/druid/src/metabase/driver/druid.clj
@@ -83,7 +83,7 @@
         (if-not query-id
           (log/warn interrupted-ex "Client closed connection, no queryId found, can't cancel query")
           (ssh/with-ssh-tunnel [details-with-tunnel details]
-            (log/warnf "Client closed connection, cancelling Druid queryId '%s'" query-id)
+            (log/warnf "Client closed connection, canceling Druid queryId '%s'" query-id)
             (try
               ;; If we can't cancel the query, we don't want to hide the original exception, attempt to cancel, but if
               ;; we can't, we should rethrow the InterruptedException, not an exception from the cancellation
diff --git a/modules/drivers/presto/src/metabase/driver/presto.clj b/modules/drivers/presto/src/metabase/driver/presto.clj
index 349f0add4cb4a0d8125e579a163a27bf83b24282..74e9dbd426a9c1c8001096646ff94c4212ba46c4 100644
--- a/modules/drivers/presto/src/metabase/driver/presto.clj
+++ b/modules/drivers/presto/src/metabase/driver/presto.clj
@@ -140,7 +140,7 @@
                     ;; If we fail to cancel the query, log it but propogate the interrupted exception, instead of
                     ;; covering it up with a failed cancel
                     (catch Exception e
-                      (log/error e (trs "Error cancelling query with ID {0}" id))))
+                      (log/error e (trs "Error canceling query with ID {0}" id))))
                   (log/warn (trs "Client connection closed, no query-id found, can't cancel query")))
                 ;; Propogate the error so that any finalizers can still run
                 (throw e)))))))))
diff --git a/src/metabase/async/api_response.clj b/src/metabase/async/api_response.clj
index 881d192a9ff3f1b743e203247dd59821a2b34db4..506d63511dfb75063381c59378d30e36d00f4b1c 100644
--- a/src/metabase/async/api_response.clj
+++ b/src/metabase/async/api_response.clj
@@ -51,7 +51,7 @@
     (.flush out)
     true
     (catch EofException e
-      (log/debug e (u/format-color 'yellow (trs "connection closed, canceling request")))
+      (log/debug (u/format-color 'yellow (trs "connection closed, canceling request")))
       false)
     (catch Throwable e
       (log/error e (trs "Unexpected error writing keepalive characters"))
diff --git a/src/metabase/async/util.clj b/src/metabase/async/util.clj
index 4a7e671038b344b85cd9a30a3da5e48fce11befd..552e0035543443f90a46ba07c126e40a132375ef 100644
--- a/src/metabase/async/util.clj
+++ b/src/metabase/async/util.clj
@@ -11,8 +11,8 @@
   like.
 
   Returns a channel that will send a single message when such early-closing cancelation occurs. You can listen for
-  this message to implement special cancelation behavior, such as canceling async jobs. This channel automatically
-  closes when either `in-chan` or `out-chan` closes."
+  this message to implement special cancelation/close behavior, such as canceling async jobs. This channel
+  automatically closes when either `in-chan` or `out-chan` closes."
   [in-chan :- ManyToManyChannel, out-chan :- ManyToManyChannel]
   (let [canceled-chan (a/chan 1)]
     ;; fire off a block that will wait for either in-chan to produce a result or out-chan to be closed
diff --git a/src/metabase/driver/sql_jdbc/execute.clj b/src/metabase/driver/sql_jdbc/execute.clj
index 4fa65c7861ca14ff57386297d99abb83ae4d82c6..105f52f4516323a16617698f8a1140da83832f46 100644
--- a/src/metabase/driver/sql_jdbc/execute.clj
+++ b/src/metabase/driver/sql_jdbc/execute.clj
@@ -181,8 +181,8 @@
           ;; we can give up on the query running in the future
           @query-future)
         (catch InterruptedException e
-          (log/warn e (tru "Client closed connection, cancelling query"))
-          ;; This is what does the real work of cancelling the query. We aren't checking the result of
+          (log/warn (tru "Client closed connection, canceling query"))
+          ;; This is what does the real work of canceling the query. We aren't checking the result of
           ;; `query-future` but this will cause an exception to be thrown, saying the query has been cancelled.
           (.cancel stmt)
           (throw e))))))
diff --git a/src/metabase/middleware/auth.clj b/src/metabase/middleware/auth.clj
index e3bcfe8c2edaf0982fde649dcfb411b410335bba..5f63a95f36983a2edd9de25e24dc81d68de99c37 100644
--- a/src/metabase/middleware/auth.clj
+++ b/src/metabase/middleware/auth.clj
@@ -28,7 +28,7 @@
 
 
 (defn enforce-api-key
-  "Middleware that enforces validation of the client via API Key, cancelling the request processing if the check fails.
+  "Middleware that enforces validation of the client via API Key, canceling the request processing if the check fails.
 
   Validation is handled by first checking for the presence of the `:metabase-api-key` on the request.  If the api key
   is available then we validate it by checking it against the configured `:mb-api-key` value set in our global config.
diff --git a/src/metabase/middleware/log.clj b/src/metabase/middleware/log.clj
index 3957da0a386f11944466766d8637be9111894ca6..2bcb32d3739e33326a04317f8bb45125c4f99c50 100644
--- a/src/metabase/middleware/log.clj
+++ b/src/metabase/middleware/log.clj
@@ -1,52 +1,149 @@
 (ns metabase.middleware.log
   "Ring middleware for logging API requests/responses."
-  (:require [clojure.tools.logging :as log]
+  (:require [clojure.core.async :as a]
+            [clojure.string :as str]
+            [clojure.tools.logging :as log]
             [metabase
              [server :as server]
              [util :as u]]
+            [metabase.async.util :as async.u]
             [metabase.middleware.util :as middleware.u]
             [metabase.util
              [date :as du]
              [i18n :refer [trs]]]
             [toucan.db :as db])
-  (:import org.eclipse.jetty.util.thread.QueuedThreadPool))
-
-(def ^:private jetty-stats-coll
-  (juxt :min-threads :max-threads :busy-threads :idle-threads :queue-size))
-
-(defn- jetty-stats []
-  (when-let [jetty-server (server/instance)]
-    (let [^QueuedThreadPool pool (.getThreadPool jetty-server)]
-      {:min-threads  (.getMinThreads pool)
-       :max-threads  (.getMaxThreads pool)
-       :busy-threads (.getBusyThreads pool)
-       :idle-threads (.getIdleThreads pool)
-       :queue-size   (.getQueueSize pool)})))
-
-(defn- log-response [{:keys [uri request-method]} {:keys [status body]} elapsed-time db-call-count]
-  (let [log-error #(log/error %)        ; these are macros so we can't pass by value :sad:
-        log-debug #(log/debug %)
-        log-warn  #(log/warn  %)
-        ;; stats? here is to avoid incurring the cost of collecting the Jetty stats and concatenating the extra
-        ;; strings when they're just going to be ignored. This is automatically handled by the macro , but is bypassed
-        ;; once we wrap it in a function
-        [error? color log-fn stats?] (cond
-                                       (>= status 500) [true  'red   log-error false]
-                                       (=  status 403) [true  'red   log-warn false]
-                                       (>= status 400) [true  'red   log-debug false]
-                                       :else           [false 'green log-debug true])]
-    (log-fn (str (apply u/format-color color
-                        (str "%s %s %d (%s) (%d DB calls)."
-                             (when stats?
-                               " Jetty threads: %s/%s (%s busy, %s idle, %s queued) (%d total active threads)"))
-                        (.toUpperCase (name request-method)) uri status elapsed-time db-call-count
-                        (when stats?
-                          (conj (vec (jetty-stats-coll (jetty-stats)))
-                                (Thread/activeCount))))
-                 ;; only print body on error so we don't pollute our environment by over-logging
-                 (when (and error?
-                            (or (string? body) (coll? body)))
-                   (str "\n" (u/pprint-to-str body)))))))
+  (:import clojure.core.async.impl.channels.ManyToManyChannel
+           org.eclipse.jetty.util.thread.QueuedThreadPool))
+
+;; To simplify passing large amounts of arguments around most functions in this namespace take an "info" map that
+;; looks like
+;;
+;;     {:request ..., :response ..., :start-time ..., :call-count-fn ...}
+;;
+;; This map is created in `log-api-call` at the bottom of this namespace.
+
+;;; +----------------------------------------------------------------------------------------------------------------+
+;;; |                                   Getting & Formatting Request/Response Info                                   |
+;;; +----------------------------------------------------------------------------------------------------------------+
+
+;; These functions take parts of the info map and convert it into formatted strings.
+
+(defn- format-status-info [{:keys [async-status], {:keys [request-method uri]} :request, {:keys [status]} :response}]
+  (str
+   (format "%s %s %d" (str/upper-case (name request-method)) uri status)
+   (when async-status
+     (format " [ASYNC: %s]" async-status))))
+
+(defn- format-performance-info [{:keys [start-time call-count-fn]}]
+  (let [elapsed-time (du/format-nanoseconds (- (System/nanoTime) start-time))
+        db-calls     (call-count-fn)]
+    (format "%s (%d DB calls)" elapsed-time db-calls)))
+
+(defn- format-threads-info [{:keys [include-stats?]}]
+  (str
+   (when-let [^QueuedThreadPool pool (some-> (server/instance) .getThreadPool)]
+     (format "Jetty threads: %s/%s (%s busy, %s idle, %s queued) "
+             (.getMinThreads pool)
+             (.getMaxThreads pool)
+             (.getBusyThreads pool)
+             (.getIdleThreads pool)
+             (.getQueueSize pool)))
+   (format "(%d total active threads)" (Thread/activeCount))))
+
+(defn- format-error-info [{{:keys [body]} :response} {:keys [error?]}]
+  (when (and error?
+             (or (string? body) (coll? body)))
+    (str "\n" (u/pprint-to-str body))))
+
+(defn- format-info [info opts]
+  (str/join " " (filter some? [(format-status-info info)
+                               (format-performance-info info)
+                               (format-threads-info opts)
+                               (format-error-info info opts)])))
+
+
+;;; +----------------------------------------------------------------------------------------------------------------+
+;;; |                                                Logging the Info                                                |
+;;; +----------------------------------------------------------------------------------------------------------------+
+
+;; `log-info` below takes an info map and actually writes the log message, using the format functions from the section
+;; above to create the combined message.
+
+;; `log-options` determines som other formating options, such as the color of the message. The first logger out of the
+;; list below whose `:status-pred` is true will be used to log the API request/response.
+;;
+;; `include-stats?` here is to avoid incurring the cost of collecting the Jetty stats and concatenating the extra
+;; strings when they're just going to be ignored. This is automatically handled by the macro, but is bypassed once we
+;; wrap it in a function
+(def ^:private log-options
+  [{:status-pred    #(>= % 500)
+    :error?         true
+    :color          'red
+    :log-fn         #(log/error %)
+    :include-stats? false}
+   {:status-pred    #(>= % 403)
+    :error?         true
+    :color          'red
+    :log-fn         #(log/warn  %)
+    :include-stats? false}
+   {:status-pred    #(>= % 400)
+    :error?         true
+    :color          'red
+    :log-fn         #(log/debug %)
+    :include-stats? false}
+   {:status-pred    (constantly true)
+    :error?         false
+    :color          'green
+    :log-fn         #(log/debug %)
+    :include-stats? true}])
+
+(defn- log-info [{{:keys [status]} :response, :as info}]
+  (try
+    (let [{:keys [color log-fn], :as opts} (some #(when ((:status-pred %) status)
+                                                    %)
+                                                 log-options)]
+      (log-fn (u/format-color color (format-info info opts))))
+    (catch Throwable e
+      (log/error e (trs "Error logging API request")))))
+
+
+;;; +----------------------------------------------------------------------------------------------------------------+
+;;; |                                                 Async Logging                                                  |
+;;; +----------------------------------------------------------------------------------------------------------------+
+
+;; These functions call `log-info` when appropriate -- right away for non-core.async-channel responses, or after the
+;; channel closes for core.async channels.
+
+(defn- log-core-async-response
+  "For async responses that return a `core.async` channel, wait for the channel to return a response before logging the
+  API request info."
+  [{{in-chan :body, :as response} :response, :as info}]
+  ;; create a new out channel, and pipe response from `in-chan` to `out-chan` using the async util single value pipe.
+  ;; The channel returned by that function can be used to listen for either in or out channels to close
+  (let [out-chan              (a/chan 1)
+        in-or-out-closed-chan (async.u/single-value-pipe in-chan out-chan)]
+    ;; [async] wait for the pipe to close the canceled/finished channel and log the API response
+    (a/go
+      (log-info (assoc info :async-status (if (a/<! in-or-out-closed-chan) "canceled" "completed"))))
+    ;; [sync] return updated response with our new body
+    (assoc response :body out-chan)))
+
+(defn- logged-response
+  "Log an API response. Returns resonse, possibly modified (i.e., core.async channels will be wrapped); this value
+  should be passed to the normal `respond` function."
+  [{{:keys [body], :as response} :response, :as info}]
+  (if (instance? ManyToManyChannel body)
+    (log-core-async-response info)
+    (do (log-info info)
+        response)))
+
+
+;;; +----------------------------------------------------------------------------------------------------------------+
+;;; |                                                   Middleware                                                   |
+;;; +----------------------------------------------------------------------------------------------------------------+
+
+;; Actual middleware. Determines whether request should be logged, and, if so, creates the info dictionary and hands
+;; off to functions above.
 
 (defn- should-log-request? [{:keys [uri], :as request}]
   ;; don't log calls to /health or /util/logs because they clutter up the logs (especially the window in admin) with
@@ -62,12 +159,10 @@
       ;; non-API call or health or logs call, don't log it
       (handler request respond raise)
       ;; API call, log info about it
-      (let [start-time (System/nanoTime)]
-        (db/with-call-counting [call-count]
-          (let [respond (fn [response]
-                          (try
-                            (log-response request response (du/format-nanoseconds (- (System/nanoTime) start-time)) (call-count))
-                            (catch Throwable e
-                              (log/error e (trs "Error logging API request"))))
-                          (respond response))]
-            (handler request respond raise)))))))
+      (db/with-call-counting [call-count-fn]
+        (let [info           {:request       request
+                              :start-time    (System/nanoTime)
+                              :call-count-fn call-count-fn}
+              response->info #(assoc info :response %)
+              respond        (comp respond logged-response response->info)]
+          (handler request respond raise))))))
diff --git a/src/metabase/query_processor.clj b/src/metabase/query_processor.clj
index 4794a3fbead20f13003a7fa24845a68537be8ba6..74b1c16afda6978e2430c9082be795aeeb738872 100644
--- a/src/metabase/query_processor.clj
+++ b/src/metabase/query_processor.clj
@@ -300,7 +300,10 @@
     (throw (ex-info (str (tru "Invalid response from database driver. No :status provided."))
              query-result)))
   (when (= :failed (:status query-result))
-    (log/warn (u/pprint-to-str 'red query-result))
+    (if (= InterruptedException (:class query-result))
+      (log/info (trs "Query canceled"))
+      (log/warn (trs "Query failure")
+                (u/pprint-to-str 'red query-result)))
     (throw (ex-info (str (get query-result :error (tru "General error")))
              query-result))))
 
@@ -341,11 +344,8 @@
       (let [result (process-query query)]
         (assert-query-status-successful result)
         (save-and-return-successful-query! query-execution result))
+      ;; canceled query
       (catch Throwable e
-        (log/warn (u/format-color 'red (trs "Query failure")
-                    (.getMessage e)
-                    "\n"
-                    (u/pprint-to-str (u/filtered-stacktrace e))))
         (save-and-return-failed-query! query-execution e)))))
 
 (s/defn ^:private assoc-query-info [query, options :- mbql.s/Info]