Skip to content
Snippets Groups Projects
Commit db2b6571 authored by Arthur Ulfeldt's avatar Arthur Ulfeldt
Browse files

Use custom streaming implementation in jetty

We want to control when messages actually send to the client
beyond the timing available with the default buffer sized based
jetty implementation.
parent b8120886
No related branches found
No related tags found
No related merge requests found
......@@ -76,7 +76,8 @@
[postgresql "9.3-1102.jdbc41"] ; Postgres driver
[io.crate/crate-jdbc "2.1.6"] ; Crate JDBC driver
[prismatic/schema "1.1.5"] ; Data schema declaration and validation library
[ring/ring-jetty-adapter "1.5.1"] ; Ring adapter using Jetty webserver (used to run a Ring server for unit tests)
[ring/ring-core "1.6.0"]
[ring/ring-jetty-adapter "1.6.0"] ; Ring adapter using Jetty webserver (used to run a Ring server for unit tests)
[ring/ring-json "0.4.0"] ; Ring middleware for reading/writing JSON automatically
[stencil "0.5.0"] ; Mustache templates for Clojure
[toucan "1.0.2" ; Model layer, hydration, and DB utilities
......
......@@ -164,7 +164,7 @@
(log/info "Launching Embedded Jetty Webserver with config:\n" (with-out-str (pprint/pprint (m/filter-keys #(not (re-matches #".*password.*" (str %)))
jetty-config))))
;; NOTE: we always start jetty w/ join=false so we can start the server first then do init in the background
(->> (ring-jetty/run-jetty app (assoc jetty-config :join? false))
(->> (ring-jetty/run-jetty #'app (assoc jetty-config :join? false))
(reset! jetty-instance)))))
(defn stop-jetty!
......
......@@ -15,7 +15,8 @@
[ring.util
[io :as ring-io]
[response :as resp]]
[stencil.core :as stencil]))
[stencil.core :as stencil]
[clojure.tools.logging :as log]))
(defn- load-file-at-path [path]
(slurp (or (io/resource path)
......@@ -50,7 +51,7 @@
(GET "*" [] embed))
(defn- some-very-long-handler [_]
(Thread/sleep 30000)
(Thread/sleep 15000)
{:success true})
(defn- some-naughty-handler-that-barfs [_]
......@@ -59,31 +60,49 @@
(def ^:private ^:const streaming-response-keep-alive-interval-ms
"Interval between sending whitespace bytes to keep Heroku from terminating
requests like queries that take a long time to complete."
(* 20 1000)) ; every 20 ms
(* 1 1000))
;;;;;;;;;;;;; begin messyness ::::::::::::::::::::::::::::::::::::::::::::
(require '[ring.core.protocols :as protocols]) ;; this next section goes to it's own namespace soon
(import '[java.util.concurrent LinkedBlockingQueue]
'java.io.OutputStream)
(extend-protocol protocols/StreamableResponseBody
LinkedBlockingQueue
(write-body-to-stream [output-queue _ ^OutputStream output-stream]
(log/error (u/format-color 'blue "starting"))
(with-open [out (io/writer output-stream)]
(.write out "starting")
(loop [chunk (.take output-queue)]
(log/error (u/format-color 'green "got chunk %s" chunk))
(when-not (= chunk ::EOF)
(.write out (str chunk))
(.flush out)
(recur (.take output-queue)))))))
(defn- streaming-response [handler]
(fn [request]
;; TODO - need maximum timeout for requests
;; TODO - error response should have status code != 200 (how ?)
;; TODO - handle exceptions in JSON encoding as well
(-> (fn [^java.io.PipedOutputStream ostream]
(let [response (future (try (handler request)
(catch Throwable e
{:error (.getMessage e)
:stacktrace (u/filtered-stacktrace e)})))
write-response (future (json/generate-stream @response (io/writer ostream))
(println "Done! closing ostream...")
(.close ostream))]
(loop []
(let [output-queue (LinkedBlockingQueue.)
response (future (try (handler request)
(catch Throwable e
{:error (.getMessage e)
:stacktrace (u/filtered-stacktrace e)})))]
(future
(loop []
(Thread/sleep streaming-response-keep-alive-interval-ms)
(when-not (realized? response)
(println "Response not ready, writing one byte & sleeping...")
(.write ostream (byte \ ))
(.flush ostream)
(recur)))))
ring-io/piped-input-stream
resp/response
(resp/content-type "application/json"))))
(.put output-queue " ")
(recur)))
(.put output-queue @response)
(.put output-queue ::EOF))
{:status 200
:body output-queue})))
;;;;;;;;;;;;; end messyness ::::::::::::::::::::::::::::::::::::::::::::
;; Redirect naughty users who try to visit a page other than setup if setup is not yet complete
(defroutes ^{:doc "Top-level ring routes for Metabase."} routes
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment